Barretenberg
The ZK-SNARK library at the core of Aztec
Loading...
Searching...
No Matches
context.test.cpp
Go to the documentation of this file.
1#include <gmock/gmock.h>
2#include <gtest/gtest.h>
3
4#include <cstdint>
5
15
16namespace bb::avm2::constraining {
17namespace {
18
19using tracegen::TestTraceContainer;
21using C = Column;
23
24TEST(ContextConstrainingTest, EmptyRow)
25{
26 check_relation<context>(testing::empty_trace());
27}
28
29// This test currently does a lot, consider splitting up the various exit call conditions
30TEST(ContextConstrainingTest, ContextSwitchingCallReturn)
31{
32 constexpr uint32_t top_bytecode_id = 0x12345678;
33 constexpr uint32_t nested_bytecode_id = 0x456789ab;
34
35 TestTraceContainer trace(
36 { {
37 { C::execution_next_context_id, 0 },
38 { C::precomputed_first_row, 1 },
39 // Context Stack Rows
40 { C::context_stack_sel, 1 },
41 { C::context_stack_entered_context_id, 2 },
42 { C::context_stack_context_id, 1 },
43 { C::context_stack_parent_id, 0 },
44 { C::context_stack_next_pc, 2 },
45 { C::context_stack_msg_sender, 0 },
46 { C::context_stack_contract_address, 0 },
47 { C::context_stack_bytecode_id, top_bytecode_id },
48 { C::context_stack_is_static, 0 },
49 { C::context_stack_parent_calldata_addr, 0 },
50 { C::context_stack_parent_calldata_size, 0 },
51 { C::context_stack_parent_l2_gas_limit, 2000 },
52 { C::context_stack_parent_da_gas_limit, 4000 },
53 { C::context_stack_parent_l2_gas_used, 500 },
54 { C::context_stack_parent_da_gas_used, 1500 },
55 },
56 // First Row of execution
57 {
58 { C::execution_sel, 1 },
59 { C::execution_pc, 0 },
60 { C::execution_next_pc, 1 },
61 { C::execution_context_id, 1 },
62 { C::execution_next_context_id, 2 },
63 { C::execution_bytecode_id, top_bytecode_id },
64 { C::execution_is_static, 0 }, // Non-static context
65 { C::execution_parent_l2_gas_limit, 2000 },
66 { C::execution_parent_da_gas_limit, 4000 },
67 { C::execution_parent_l2_gas_used, 500 },
68 { C::execution_parent_da_gas_used, 1500 },
69 { C::execution_enqueued_call_start, 1 },
70 },
71 // CALL
72 {
73 { C::execution_sel, 1 },
74 { C::execution_pc, 1 },
75 { C::execution_next_pc, 2 },
76 { C::execution_sel_execute_call, 1 },
77 { C::execution_sel_execute_static_call, 0 }, // Regular CALL, not STATICCALL
78 { C::execution_sel_enter_call, 1 },
79 { C::execution_context_id, 1 },
80 { C::execution_next_context_id, 2 },
81 { C::execution_bytecode_id, top_bytecode_id }, // Same as previous row (propagated)
82 { C::execution_is_static, 0 }, // Still non-static
83 { C::execution_rop_4_, /*cd offset=*/10 },
84 { C::execution_register_2_, /*contract address=*/0xdeadbeef },
85 { C::execution_register_3_, /*cd size=*/1 },
86 { C::execution_parent_l2_gas_limit, 2000 },
87 { C::execution_parent_da_gas_limit, 4000 },
88 { C::execution_parent_l2_gas_used, 500 },
89 { C::execution_parent_da_gas_used, 1500 },
90 },
91 // First Row in new context
92 {
93 { C::execution_sel, 1 },
94 { C::execution_pc, 0 }, // pc=0 because it is after a CALL
95 { C::execution_next_pc, 20 },
96 { C::execution_context_id, 2 }, // Previous row next_context_id
97 { C::execution_next_context_id, 3 }, // Incremented due to previous call
98 { C::execution_parent_id, 1 }, // Previous row context id
99 { C::execution_is_parent_id_inv, 1 },
100 { C::execution_has_parent_ctx, 1 },
101 { C::execution_contract_address, 0xdeadbeef },
102 { C::execution_bytecode_id, nested_bytecode_id }, // New bytecode_id on entering new context
103 { C::execution_is_static, 0 }, // Remains non-static after regular CALL
104 { C::execution_parent_calldata_addr, 10 },
105 { C::execution_parent_calldata_size, 1 },
106 },
107 // Return Row
108 {
109 { C::execution_sel, 1 },
110 { C::execution_pc, 20 },
111 { C::execution_next_pc, 30 },
112 { C::execution_sel_execute_return, 1 },
113 { C::execution_rop_0_, 500 }, // Return data size offset
114 { C::execution_rop_1_, 600 }, // Return data offset
115 { C::execution_register_0_, 200 }, // Return data size
116 { C::execution_sel_exit_call, 1 },
117 { C::execution_nested_exit_call, 1 },
118 { C::execution_nested_return, 1 },
119 { C::execution_context_id, 2 },
120 { C::execution_next_context_id, 3 },
121 { C::execution_parent_id, 1 },
122 { C::execution_is_parent_id_inv, 1 },
123 { C::execution_has_parent_ctx, 1 },
124 { C::execution_contract_address, 0xdeadbeef },
125 { C::execution_bytecode_id, nested_bytecode_id }, // Propagated within same context
126 { C::execution_parent_calldata_addr, 10 },
127 { C::execution_parent_calldata_size, 1 },
128 },
129 {
130 { C::execution_sel, 1 },
131 { C::execution_next_context_id, 3 },
132 { C::execution_context_id, 1 },
133 { C::execution_parent_id, 0 },
134 { C::execution_last_child_id, 2 }, // Previous context id
135 { C::execution_pc, 2 }, // Based on next_pc of CALL step
136 { C::execution_msg_sender, 0 },
137 { C::execution_contract_address, 0 },
138 { C::execution_bytecode_id, top_bytecode_id }, // Restored from context stack
139 { C::execution_is_static, 0 },
140 { C::execution_parent_calldata_addr, 0 },
141 { C::execution_parent_calldata_size, 0 },
142 { C::execution_last_child_returndata_size, 200 }, // Return data size
143 { C::execution_last_child_returndata_addr, 600 }, // Return data offset
144 { C::execution_last_child_success, 1 }, // Success because return was successful
145 { C::execution_parent_l2_gas_limit, 2000 },
146 { C::execution_parent_da_gas_limit, 4000 },
147 { C::execution_parent_l2_gas_used, 500 },
148 { C::execution_parent_da_gas_used, 1500 },
149 },
150 {
151 { C::execution_sel, 0 },
152 } });
153
154 check_relation<context>(trace);
155
156 check_interaction<tracegen::ExecutionTraceBuilder,
160}
161
162TEST(ContextConstrainingTest, ContextSwitchingExceptionalHalt)
163{
164 constexpr uint32_t top_bytecode_id = 0x12345678;
165 constexpr uint32_t nested_bytecode_id = 0x456789ab;
166
167 TestTraceContainer trace(
168 { {
169 { C::execution_next_context_id, 0 },
170 { C::precomputed_first_row, 1 },
171 // Context Stack Rows
172 { C::context_stack_sel, 1 },
173 { C::context_stack_entered_context_id, 2 },
174 { C::context_stack_context_id, 1 },
175 { C::context_stack_parent_id, 0 },
176 { C::context_stack_next_pc, 2 },
177 { C::context_stack_msg_sender, 0 },
178 { C::context_stack_contract_address, 0 },
179 { C::context_stack_bytecode_id, top_bytecode_id },
180 { C::context_stack_is_static, 0 },
181 { C::context_stack_parent_calldata_addr, 0 },
182 { C::context_stack_parent_calldata_size, 0 },
183 { C::context_stack_parent_l2_gas_limit, 2000 },
184 { C::context_stack_parent_da_gas_limit, 4000 },
185 { C::context_stack_parent_l2_gas_used, 500 },
186 { C::context_stack_parent_da_gas_used, 1500 },
187 },
188 // First Row of execution
189 {
190 { C::execution_sel, 1 },
191 { C::execution_pc, 0 },
192 { C::execution_next_pc, 1 },
193 { C::execution_context_id, 1 },
194 { C::execution_next_context_id, 2 },
195 { C::execution_bytecode_id, top_bytecode_id },
196 { C::execution_parent_l2_gas_limit, 2000 },
197 { C::execution_parent_da_gas_limit, 4000 },
198 { C::execution_parent_l2_gas_used, 500 },
199 { C::execution_parent_da_gas_used, 1500 },
200 { C::execution_enqueued_call_start, 1 },
201 },
202 // CALL
203 {
204 { C::execution_sel, 1 },
205 { C::execution_pc, 1 },
206 { C::execution_next_pc, 2 },
207 { C::execution_sel_execute_call, 1 },
208 { C::execution_sel_enter_call, 1 },
209 { C::execution_context_id, 1 },
210 { C::execution_next_context_id, 2 },
211 { C::execution_bytecode_id, top_bytecode_id }, // Same as previous row (propagated)
212 { C::execution_rop_4_, /*cd offset=*/10 },
213 { C::execution_register_2_, /*contract address=*/0xdeadbeef },
214 { C::execution_register_3_, /*cd size=*/1 },
215 { C::execution_parent_l2_gas_limit, 2000 },
216 { C::execution_parent_da_gas_limit, 4000 },
217 { C::execution_parent_l2_gas_used, 500 },
218 { C::execution_parent_da_gas_used, 1500 },
219 },
220 // First Row in new context
221 {
222 { C::execution_sel, 1 },
223 { C::execution_pc, 0 }, // pc=0 because it is after a CALL
224 { C::execution_next_pc, 20 },
225 { C::execution_context_id, 2 }, // Previous row next_context_id
226 { C::execution_next_context_id, 3 }, // Incremented due to previous call
227 { C::execution_parent_id, 1 }, // Previous row context id
228 { C::execution_is_parent_id_inv, 1 },
229 { C::execution_has_parent_ctx, 1 },
230 { C::execution_contract_address, 0xdeadbeef },
231 { C::execution_bytecode_id, nested_bytecode_id }, // New bytecode_id on entering new context
232 { C::execution_parent_calldata_addr, 10 },
233 { C::execution_parent_calldata_size, 1 },
234 },
235 // Exceptional Halt Row
236 {
237 { C::execution_sel, 1 },
238 { C::execution_pc, 20 },
239 { C::execution_next_pc, 30 },
240 // Note that `sel_execute_return` is 0 because this is an exceptional halt.
241 { C::execution_rop_0_, 500 }, // Return data size offset
242 { C::execution_rop_1_, 600 }, // Return data offset
243 { C::execution_register_0_, 200 }, // Return data size
244 { C::execution_sel_exit_call, 1 },
245 { C::execution_nested_exit_call, 1 },
246 { C::execution_sel_error, 1 }, // Exceptional Halt
247 { C::execution_sel_failure, 1 },
248 { C::execution_nested_failure, 1 },
249 { C::execution_context_id, 2 },
250 { C::execution_next_context_id, 3 },
251 { C::execution_parent_id, 1 },
252 { C::execution_is_parent_id_inv, 1 },
253 { C::execution_has_parent_ctx, 1 },
254 { C::execution_contract_address, 0xdeadbeef },
255 { C::execution_bytecode_id, nested_bytecode_id }, // Propagated within same context
256 { C::execution_parent_calldata_addr, 10 },
257 { C::execution_parent_calldata_size, 1 },
258 },
259 {
260 { C::execution_sel, 1 },
261 { C::execution_next_context_id, 3 },
262 { C::execution_context_id, 1 },
263 { C::execution_parent_id, 0 },
264 { C::execution_last_child_id, 2 }, // Previous context id
265 { C::execution_pc, 2 }, // Based on next_pc of CALL step
266 { C::execution_next_pc, 3 },
267 { C::execution_msg_sender, 0 },
268 { C::execution_contract_address, 0 },
269 { C::execution_bytecode_id, top_bytecode_id }, // Restored from context stack
270 { C::execution_is_static, 0 },
271 { C::execution_parent_calldata_addr, 0 },
272 { C::execution_parent_calldata_size, 0 },
273 { C::execution_last_child_returndata_size, 0 }, // Return data size reset
274 { C::execution_last_child_returndata_addr, 0 }, // Return data offset reset
275 { C::execution_parent_l2_gas_limit, 2000 },
276 { C::execution_parent_da_gas_limit, 4000 },
277 { C::execution_parent_l2_gas_used, 500 },
278 { C::execution_parent_da_gas_used, 1500 },
279 },
280 {
281 { C::execution_sel, 0 },
282 } });
283
284 check_relation<context>(trace);
285
286 check_interaction<tracegen::ExecutionTraceBuilder,
290}
291
292TEST(ContextConstrainingTest, GasNextRow)
293{
294 TestTraceContainer trace({ { { C::precomputed_first_row, 1 } },
295 {
296 // First Row of execution
297 { C::execution_sel, 1 },
298 { C::execution_l2_gas_limit, 1000 },
299 { C::execution_da_gas_limit, 2000 },
300 { C::execution_parent_l2_gas_limit, 2000 },
301 { C::execution_parent_da_gas_limit, 4000 },
302 { C::execution_parent_l2_gas_used, 500 },
303 { C::execution_parent_da_gas_used, 1500 },
304 },
305 {
306 // CALL
307 { C::execution_sel, 1 },
308 { C::execution_sel_enter_call, 1 },
309 { C::execution_l2_gas_used, 200 },
310 { C::execution_da_gas_used, 300 },
311 { C::execution_l2_gas_limit, 1000 },
312 { C::execution_da_gas_limit, 2000 },
313 { C::execution_parent_l2_gas_limit, 2000 },
314 { C::execution_parent_da_gas_limit, 4000 },
315 { C::execution_parent_l2_gas_used, 500 },
316 { C::execution_parent_da_gas_used, 1500 },
317 },
318 {
319 // Return
320 { C::execution_sel, 1 },
321 { C::execution_sel_exit_call, 1 },
322 { C::execution_nested_exit_call, 1 },
323 { C::execution_parent_l2_gas_limit, 1000 },
324 { C::execution_parent_da_gas_limit, 2000 },
325 { C::execution_parent_l2_gas_used, 200 },
326 { C::execution_parent_da_gas_used, 300 },
327 },
328 {
329 // After return
330 { C::execution_sel, 1 },
331 { C::execution_l2_gas_limit, 1000 },
332 { C::execution_da_gas_limit, 2000 },
333 { C::execution_parent_l2_gas_limit, 2000 },
334 { C::execution_parent_da_gas_limit, 4000 },
335 },
336 {
337 { C::execution_sel, 0 },
338 } });
339
340 check_relation<context>(trace,
353
354 // Negative test: after return, restore wrong limits
355 trace.set(C::execution_l2_gas_limit, 4, 1001);
357 "L2_GAS_LIMIT_RESTORE_ON_EXIT");
358 trace.set(C::execution_da_gas_limit, 4, 2001);
360 "DA_GAS_LIMIT_RESTORE_ON_EXIT");
361
362 // Negative test: inside a nested call, store wrong parent limit and used
363 trace.set(C::execution_parent_l2_gas_limit, 3, 2001);
365 "PARENT_L2_GAS_LIMIT_STORE_ON_ENTER");
366 trace.set(C::execution_parent_da_gas_limit, 3, 4001);
368 "PARENT_DA_GAS_LIMIT_STORE_ON_ENTER");
369 trace.set(C::execution_parent_l2_gas_used, 3, 201);
371 "PARENT_L2_GAS_USED_STORE_ON_ENTER");
372 trace.set(C::execution_parent_da_gas_used, 3, 301);
374 "PARENT_DA_GAS_USED_STORE_ON_ENTER");
375
376 // Negative test: when no calls have been made, limits, parent limits, and parent used shouldn't change
377 trace.set(C::execution_l2_gas_limit, 2, 1001);
378 EXPECT_THROW_WITH_MESSAGE(check_relation<context>(trace, context::SR_L2_GAS_LIMIT_NEXT_ROW),
379 "L2_GAS_LIMIT_NEXT_ROW");
380 trace.set(C::execution_da_gas_limit, 2, 2001);
381 EXPECT_THROW_WITH_MESSAGE(check_relation<context>(trace, context::SR_DA_GAS_LIMIT_NEXT_ROW),
382 "DA_GAS_LIMIT_NEXT_ROW");
383
384 trace.set(C::execution_parent_l2_gas_limit, 2, 2001);
386 "PARENT_L2_GAS_LIMIT_NEXT_ROW");
387 trace.set(C::execution_parent_da_gas_limit, 2, 4001);
389 "PARENT_DA_GAS_LIMIT_NEXT_ROW");
390
391 trace.set(C::execution_parent_l2_gas_used, 2, 501);
393 "PARENT_L2_GAS_USED_NEXT_ROW");
394 trace.set(C::execution_parent_da_gas_used, 2, 1501);
396 "PARENT_DA_GAS_USED_NEXT_ROW");
397}
398
399TEST(ContextConstrainingTest, GasUsedContinuity)
400{
401 TestTraceContainer trace({ { { C::precomputed_first_row, 1 } },
402 {
403 // First Row of execution
404 { C::execution_sel, 1 },
405 { C::execution_l2_gas_used, 100 },
406 { C::execution_da_gas_used, 200 },
407 },
408 {
409 // CALL
410 { C::execution_sel, 1 },
411 { C::execution_sel_enter_call, 1 },
412 { C::execution_l2_gas_used, 110 },
413 { C::execution_da_gas_used, 200 },
414 { C::execution_prev_l2_gas_used, 100 },
415 { C::execution_prev_da_gas_used, 200 },
416 },
417 {
418 // Return
419 { C::execution_sel, 1 },
420 { C::execution_sel_exit_call, 1 },
421 { C::execution_nested_exit_call, 1 },
422 { C::execution_l2_gas_used, 50 },
423 { C::execution_da_gas_used, 60 },
424 { C::execution_parent_l2_gas_used, 110 },
425 { C::execution_parent_da_gas_used, 200 },
426 { C::execution_prev_l2_gas_used, 0 },
427 { C::execution_prev_da_gas_used, 0 },
428 },
429 {
430 // After return
431 { C::execution_sel, 1 },
432 { C::execution_l2_gas_used, 170 },
433 { C::execution_da_gas_used, 260 },
434 { C::execution_prev_l2_gas_used, 160 }, // 110 + 50
435 { C::execution_prev_da_gas_used, 260 }, // 200 + 60
436 },
437 {
438 { C::execution_sel, 0 },
439 } });
440
441 check_relation<context>(trace,
448
449 // Negative test: after return, ingest a wrong value
450 trace.set(C::execution_prev_l2_gas_used, 4, 110);
451
453 "L2_GAS_USED_INGEST_AFTER_EXIT");
454
455 trace.set(C::execution_prev_da_gas_used, 4, 60);
457 "DA_GAS_USED_INGEST_AFTER_EXIT");
458
459 // Negative test: inside a nested call, start with non-zero gas used
460 trace.set(C::execution_prev_l2_gas_used, 3, 110);
462 "L2_GAS_USED_ZERO_AFTER_CALL");
463
464 trace.set(C::execution_prev_da_gas_used, 3, 200);
466 "DA_GAS_USED_ZERO_AFTER_CALL");
467
468 // Negative test: when no calls are made, prev gas used should be gas used of the previous row
469 trace.set(C::execution_prev_l2_gas_used, 2, 0);
471 "L2_GAS_USED_CONTINUITY");
472
473 trace.set(C::execution_prev_da_gas_used, 2, 0);
475 "DA_GAS_USED_CONTINUITY");
476}
477
478TEST(ContextConstrainingTest, TreeStateContinuity)
479{
480 TestTraceContainer trace({ { { C::precomputed_first_row, 1 } },
481 {
482 // First Row of execution
483 { C::execution_sel, 1 },
484 { C::execution_note_hash_tree_root, 10 },
485 { C::execution_note_hash_tree_size, 9 },
486 { C::execution_num_note_hashes_emitted, 8 },
487 { C::execution_nullifier_tree_root, 7 },
488 { C::execution_nullifier_tree_size, 6 },
489 { C::execution_num_nullifiers_emitted, 5 },
490 { C::execution_public_data_tree_root, 4 },
491 { C::execution_public_data_tree_size, 3 },
492 { C::execution_written_public_data_slots_tree_root, 2 },
493 { C::execution_written_public_data_slots_tree_size, 1 },
494 { C::execution_l1_l2_tree_root, 27 },
495 { C::execution_retrieved_bytecodes_tree_root, 26 },
496 { C::execution_retrieved_bytecodes_tree_size, 25 },
497 },
498 {
499 // Second row of execution
500 { C::execution_sel, 1 },
501 { C::execution_prev_note_hash_tree_root, 10 },
502 { C::execution_prev_note_hash_tree_size, 9 },
503 { C::execution_prev_num_note_hashes_emitted, 8 },
504 { C::execution_prev_nullifier_tree_root, 7 },
505 { C::execution_prev_nullifier_tree_size, 6 },
506 { C::execution_prev_num_nullifiers_emitted, 5 },
507 { C::execution_prev_public_data_tree_root, 4 },
508 { C::execution_prev_public_data_tree_size, 3 },
509 { C::execution_prev_written_public_data_slots_tree_root, 2 },
510 { C::execution_prev_written_public_data_slots_tree_size, 1 },
511 { C::execution_l1_l2_tree_root, 27 },
512 { C::execution_prev_retrieved_bytecodes_tree_root, 26 },
513 { C::execution_prev_retrieved_bytecodes_tree_size, 25 },
514 { C::execution_enqueued_call_end, 1 },
515 { C::execution_sel_exit_call, 1 },
516 },
517 {
518 // Third row of execution
519 { C::execution_sel, 1 },
520 { C::execution_prev_note_hash_tree_root, 100 },
521 { C::execution_prev_note_hash_tree_size, 90 },
522 { C::execution_prev_num_note_hashes_emitted, 80 },
523 { C::execution_prev_nullifier_tree_root, 70 },
524 { C::execution_prev_nullifier_tree_size, 60 },
525 { C::execution_prev_num_nullifiers_emitted, 50 },
526 { C::execution_prev_public_data_tree_root, 40 },
527 { C::execution_prev_public_data_tree_size, 30 },
528 { C::execution_prev_written_public_data_slots_tree_root, 20 },
529 { C::execution_prev_written_public_data_slots_tree_size, 10 },
530 { C::execution_l1_l2_tree_root, 27 },
531 { C::execution_prev_retrieved_bytecodes_tree_root, 260 },
532 { C::execution_prev_retrieved_bytecodes_tree_size, 250 },
533 } });
534
535 check_relation<context>(trace,
549
550 // Negative test: change note hash tree root
551 trace.set(C::execution_prev_note_hash_tree_root, 2, 100);
553 "NOTE_HASH_TREE_ROOT_CONTINUITY");
554
555 // Negative test: change note hash tree size
556 trace.set(C::execution_prev_note_hash_tree_size, 2, 100);
558 "NOTE_HASH_TREE_SIZE_CONTINUITY");
559
560 // Negative test: change num note hashes emitted
561 trace.set(C::execution_prev_num_note_hashes_emitted, 2, 10);
563 "NUM_NOTE_HASHES_EMITTED_CONTINUITY");
564
565 // Negative test: change nullifier tree root
566 trace.set(C::execution_prev_nullifier_tree_root, 2, 100);
568 "NULLIFIER_TREE_ROOT_CONTINUITY");
569
570 // Negative test: change nullifier tree size
571 trace.set(C::execution_prev_nullifier_tree_size, 2, 100);
573 "NULLIFIER_TREE_SIZE_CONTINUITY");
574
575 // Negative test: change num nullifiers emitted
576 trace.set(C::execution_prev_num_nullifiers_emitted, 2, 100);
578 "NUM_NULLIFIERS_EMITTED_CONTINUITY");
579
580 // Negative test: change public data tree root
581 trace.set(C::execution_prev_public_data_tree_root, 2, 100);
583 "PUBLIC_DATA_TREE_ROOT_CONTINUITY");
584
585 // Negative test: change public data tree size
586 trace.set(C::execution_prev_public_data_tree_size, 2, 100);
588 "PUBLIC_DATA_TREE_SIZE_CONTINUITY");
589
590 // Negative test: change written public data slots tree root
591 trace.set(C::execution_prev_written_public_data_slots_tree_root, 2, 100);
594 "WRITTEN_PUBLIC_DATA_SLOTS_TREE_ROOT_CONTINUITY");
595
596 // Negative test: change written public data slots tree size
597 trace.set(C::execution_prev_written_public_data_slots_tree_size, 2, 100);
600 "WRITTEN_PUBLIC_DATA_SLOTS_TREE_SIZE_CONTINUITY");
601
602 // Negative test: change l1 l2 tree root
603 trace.set(C::execution_l1_l2_tree_root, 2, 100);
605 "L1_L2_TREE_ROOT_CONTINUITY");
606
607 // Negative test: change retrieved bytecodes tree root
608 trace.set(C::execution_prev_retrieved_bytecodes_tree_root, 2, 100);
610 "RETRIEVED_BYTECODES_TREE_ROOT_CONTINUITY");
611
612 // Negative test: change retrieved bytecodes tree size
613 trace.set(C::execution_prev_retrieved_bytecodes_tree_size, 2, 100);
615 "RETRIEVED_BYTECODES_TREE_SIZE_CONTINUITY");
616}
617
618TEST(ContextConstrainingTest, SideEffectStateContinuity)
619{
620 TestTraceContainer trace({
621 { { C::precomputed_first_row, 1 } },
622 {
623 // First Row of execution
624 { C::execution_sel, 1 },
625 { C::execution_num_unencrypted_log_fields, 10 },
626 { C::execution_num_l2_to_l1_messages, 11 },
627 },
628 {
629 // Second row of execution
630 { C::execution_sel, 1 },
631 { C::execution_prev_num_unencrypted_log_fields, 10 },
632 { C::execution_prev_num_l2_to_l1_messages, 11 },
633 },
634 });
635
636 check_relation<context>(
638
639 // Negative test: change num unencrypted logs
640 trace.set(C::execution_prev_num_unencrypted_log_fields, 2, 100);
642 "NUM_UNENCRYPTED_LOGS_CONTINUITY");
643
644 // Negative test: change num l2 to l1 messages
645 trace.set(C::execution_prev_num_l2_to_l1_messages, 2, 100);
647 "NUM_L2_TO_L1_MESSAGES_CONTINUITY");
648}
649
650TEST(ContextConstrainingTest, BytecodeIdPropagation)
651{
652 TestTraceContainer trace({ // First row - setup
653 {
654 { C::precomputed_first_row, 1 },
655 { C::execution_sel, 1 },
656 { C::execution_context_id, 1 },
657 { C::execution_next_context_id, 1 },
658 { C::execution_bytecode_id, 42 }, // Initial bytecode_id
659 },
660 // Second row - should propagate bytecode_id
661 {
662 { C::execution_sel, 1 },
663 { C::execution_context_id, 1 },
664 { C::execution_next_context_id, 1 },
665 { C::execution_bytecode_id, 42 }, // Same bytecode_id (propagated)
666 } });
667
668 check_relation<context>(trace);
669 // mutate the bytecode_id and confirm that it is a violation
670 trace.set(C::execution_bytecode_id, 1, 99);
671 EXPECT_THROW_WITH_MESSAGE(check_relation<context>(trace, context::SR_BYTECODE_ID_NEXT_ROW),
672 "BYTECODE_ID_NEXT_ROW"); // Should fail constraint
673}
674
675TEST(ContextConstrainingTest, IsStaticRegularCallFromNonStaticContext)
676{
677 // Non-static context making a regular CALL - should remain non-static
678 TestTraceContainer trace({
679 { { C::precomputed_first_row, 1 } },
680 {
681 { C::execution_sel, 1 },
682 { C::execution_context_id, 1 },
683 { C::execution_next_context_id, 2 },
684 { C::execution_is_static, 0 }, // Non-static context
685 { C::execution_sel_enter_call, 1 },
686 { C::execution_sel_execute_call, 1 }, // Regular CALL
687 { C::execution_sel_execute_static_call, 0 },
688 },
689 {
690 { C::execution_sel, 1 },
691 { C::execution_context_id, 2 },
692 { C::execution_next_context_id, 3 },
693 { C::execution_is_static, 0 }, // Should remain non-static
694 },
695 });
696 check_relation<context>(
698
699 // Negative test: change is_static
700 // regular call from non-static context cannot become static
701 trace.set(C::execution_is_static, 2, 1);
703 "IS_STATIC_IF_STATIC_CALL");
704
705 // reset is_static
706 trace.set(C::execution_is_static, 2, 0);
707}
708
709TEST(ContextConstrainingTest, IsStaticStaticCallFromNonStaticContext)
710{
711 // Non-static context making a STATICCALL - should become static
712 TestTraceContainer trace({
713 { { C::precomputed_first_row, 1 } },
714 {
715 { C::execution_sel, 1 },
716 { C::execution_context_id, 1 },
717 { C::execution_next_context_id, 2 },
718 { C::execution_is_static, 0 }, // Non-static context
719 { C::execution_sel_enter_call, 1 },
720 { C::execution_sel_execute_call, 0 },
721 { C::execution_sel_execute_static_call, 1 }, // STATICCALL
722 },
723 {
724 { C::execution_sel, 1 },
725 { C::execution_context_id, 2 },
726 { C::execution_next_context_id, 3 },
727 { C::execution_is_static, 1 }, // Should become static
728 },
729 });
730 check_relation<context>(
732
733 // Negative test: change is_static
734 // static call from non-static context MUST become static
735 trace.set(C::execution_is_static, 2, 0);
737 "IS_STATIC_IF_STATIC_CALL");
738
739 // reset is_static
740 trace.set(C::execution_is_static, 2, 1);
741}
742
743TEST(ContextConstrainingTest, IsStaticCallFromStaticContext)
744{
745 // Static context making any call - must remain static
746 TestTraceContainer trace({
747 { { C::precomputed_first_row, 1 } },
748 {
749 { C::execution_sel, 1 },
750 { C::execution_context_id, 1 },
751 { C::execution_next_context_id, 2 },
752 { C::execution_is_static, 1 }, // Static context
753 { C::execution_sel_enter_call, 1 },
754 { C::execution_sel_execute_call, 1 }, // Regular CALL
755 { C::execution_sel_execute_static_call, 0 },
756 },
757 {
758 { C::execution_sel, 1 },
759 { C::execution_context_id, 2 },
760 { C::execution_next_context_id, 3 },
761 { C::execution_is_static, 1 }, // Must remain static
762 },
763 });
764 check_relation<context>(
766
767 // Negative test: change is_static
768 // static call from static context MUST remain static
769 trace.set(C::execution_is_static, 2, 0);
771 "IS_STATIC_IF_CALL_FROM_STATIC_CONTEXT");
772
773 // reset is_static
774 trace.set(C::execution_is_static, 2, 1);
775}
776
777TEST(ContextConstrainingTest, IsStaticPropagationWithoutCalls)
778{
779 // is_static propagation without calls
780 TestTraceContainer trace({
781 { { C::precomputed_first_row, 1 } },
782 {
783 { C::execution_sel, 1 },
784 { C::execution_context_id, 1 },
785 { C::execution_next_context_id, 1 },
786 { C::execution_is_static, 1 }, // Static context
787 },
788 {
789 { C::execution_sel, 1 },
790 { C::execution_context_id, 1 },
791 { C::execution_next_context_id, 1 },
792 { C::execution_is_static, 1 }, // Should propagate
793 },
794 });
795 check_relation<context>(trace, context::SR_IS_STATIC_NEXT_ROW);
796
797 // Negative test: change is_static
798 // staticness must propagate without calls
799 trace.set(C::execution_is_static, 2, 0);
800 EXPECT_THROW_WITH_MESSAGE(check_relation<context>(trace, context::SR_IS_STATIC_NEXT_ROW), "IS_STATIC_NEXT_ROW");
801
802 // reset is_static
803 trace.set(C::execution_is_static, 2, 1);
804}
805
806TEST(ContextConstrainingTest, ContextIdPropagation)
807{
808 TestTraceContainer trace({
809 {
810 { C::precomputed_first_row, 1 },
811 },
812 {
813 { C::execution_sel, 1 },
814 { C::execution_enqueued_call_start, 1 },
815 { C::execution_context_id, 1 },
816 { C::execution_next_context_id, 2 },
817 { C::execution_sel_enter_call, 1 },
818 },
819 {
820 { C::execution_sel, 1 },
821 { C::execution_context_id, 2 },
822 { C::execution_next_context_id, 3 },
823 { C::execution_sel_exit_call, 1 },
824 { C::execution_nested_exit_call, 1 },
825 { C::execution_parent_id, 1 },
826 },
827 {
828 { C::execution_sel, 1 },
829 { C::execution_context_id, 1 },
830 { C::execution_next_context_id, 3 },
831 },
832 {
833 { C::execution_sel, 1 },
834 { C::execution_context_id, 1 },
835 { C::execution_next_context_id, 3 },
836 },
837 });
838 check_relation<context>(trace,
844
845 // Negative test: next context id should be context id + 1 on enqueued call start
846 trace.set(C::execution_next_context_id, 1, 3);
848 "ENQUEUED_CALL_START_NEXT_CTX_ID");
849 trace.set(C::execution_next_context_id, 1, 2);
850
851 // Negative test: next context id should increase on external call
852 trace.set(C::execution_next_context_id, 2, 2);
853 EXPECT_THROW_WITH_MESSAGE(check_relation<context>(trace, context::SR_INCR_NEXT_CONTEXT_ID), "INCR_NEXT_CONTEXT_ID");
854 trace.set(C::execution_next_context_id, 2, 3);
855
856 // Negative test: next context id should be propagated
857 trace.set(C::execution_next_context_id, 4, 4);
858 EXPECT_THROW_WITH_MESSAGE(check_relation<context>(trace, context::SR_INCR_NEXT_CONTEXT_ID), "INCR_NEXT_CONTEXT_ID");
859 trace.set(C::execution_next_context_id, 4, 3);
860
861 // Negative test: context id should be propagated
862 trace.set(C::execution_context_id, 4, 2);
863 EXPECT_THROW_WITH_MESSAGE(check_relation<context>(trace, context::SR_CONTEXT_ID_NEXT_ROW), "CONTEXT_ID_NEXT_ROW");
864 trace.set(C::execution_context_id, 4, 1);
865
866 // Negative test: context id should be next context id when entering call
867 trace.set(C::execution_context_id, 2, 1);
868 EXPECT_THROW_WITH_MESSAGE(check_relation<context>(trace, context::SR_CONTEXT_ID_EXT_CALL), "CONTEXT_ID_EXT_CALL");
869 trace.set(C::execution_context_id, 2, 2);
870
871 // Negative test: context id should be restored on exit
872 trace.set(C::execution_context_id, 3, 2);
874 "CONTEXT_ID_NESTED_EXIT");
875 trace.set(C::execution_context_id, 3, 1);
876}
877
878} // namespace
879} // namespace bb::avm2::constraining
#define EXPECT_THROW_WITH_MESSAGE(code, expectedMessageRegex)
Definition assert.hpp:192
static constexpr size_t SR_PARENT_DA_GAS_USED_NEXT_ROW
Definition context.hpp:87
static constexpr size_t SR_INCR_NEXT_CONTEXT_ID
Definition context.hpp:43
static constexpr size_t SR_L1_L2_TREE_ROOT_CONTINUITY
Definition context.hpp:107
static constexpr size_t SR_DA_GAS_USED_INGEST_AFTER_EXIT
Definition context.hpp:96
static constexpr size_t SR_L2_GAS_USED_ZERO_AFTER_CALL
Definition context.hpp:92
static constexpr size_t SR_PARENT_L2_GAS_LIMIT_STORE_ON_ENTER
Definition context.hpp:82
static constexpr size_t SR_PARENT_L2_GAS_USED_STORE_ON_ENTER
Definition context.hpp:86
static constexpr size_t SR_NOTE_HASH_TREE_SIZE_CONTINUITY
Definition context.hpp:98
static constexpr size_t SR_IS_STATIC_IF_CALL_FROM_STATIC_CONTEXT
Definition context.hpp:57
static constexpr size_t SR_RETRIEVED_BYTECODES_TREE_ROOT_CONTINUITY
Definition context.hpp:89
static constexpr size_t SR_WRITTEN_PUBLIC_DATA_SLOTS_TREE_ROOT_CONTINUITY
Definition context.hpp:105
static constexpr size_t SR_L2_GAS_LIMIT_RESTORE_ON_EXIT
Definition context.hpp:78
static constexpr size_t SR_DA_GAS_USED_CONTINUITY
Definition context.hpp:94
static constexpr size_t SR_L2_GAS_USED_INGEST_AFTER_EXIT
Definition context.hpp:93
static constexpr size_t SR_PUBLIC_DATA_TREE_SIZE_CONTINUITY
Definition context.hpp:104
static constexpr size_t SR_PARENT_DA_GAS_USED_STORE_ON_ENTER
Definition context.hpp:88
static constexpr size_t SR_NUM_NOTE_HASHES_EMITTED_CONTINUITY
Definition context.hpp:99
static constexpr size_t SR_PARENT_L2_GAS_LIMIT_NEXT_ROW
Definition context.hpp:81
static constexpr size_t SR_NULLIFIER_TREE_SIZE_CONTINUITY
Definition context.hpp:101
static constexpr size_t SR_CONTEXT_ID_EXT_CALL
Definition context.hpp:45
static constexpr size_t SR_DA_GAS_USED_ZERO_AFTER_CALL
Definition context.hpp:95
static constexpr size_t SR_NUM_L2_TO_L1_MESSAGES_CONTINUITY
Definition context.hpp:109
static constexpr size_t SR_ENQUEUED_CALL_START_NEXT_CTX_ID
Definition context.hpp:42
static constexpr size_t SR_PARENT_DA_GAS_LIMIT_STORE_ON_ENTER
Definition context.hpp:84
static constexpr size_t SR_NOTE_HASH_TREE_ROOT_CONTINUITY
Definition context.hpp:97
static constexpr size_t SR_CONTEXT_ID_NESTED_EXIT
Definition context.hpp:46
static constexpr size_t SR_WRITTEN_PUBLIC_DATA_SLOTS_TREE_SIZE_CONTINUITY
Definition context.hpp:106
static constexpr size_t SR_BYTECODE_ID_NEXT_ROW
Definition context.hpp:53
static constexpr size_t SR_CONTEXT_ID_NEXT_ROW
Definition context.hpp:44
static constexpr size_t SR_RETRIEVED_BYTECODES_TREE_SIZE_CONTINUITY
Definition context.hpp:90
static constexpr size_t SR_PUBLIC_DATA_TREE_ROOT_CONTINUITY
Definition context.hpp:103
static constexpr size_t SR_NUM_NULLIFIERS_EMITTED_CONTINUITY
Definition context.hpp:102
static constexpr size_t SR_L2_GAS_LIMIT_NEXT_ROW
Definition context.hpp:77
static constexpr size_t SR_NUM_UNENCRYPTED_LOGS_CONTINUITY
Definition context.hpp:108
static constexpr size_t SR_DA_GAS_LIMIT_RESTORE_ON_EXIT
Definition context.hpp:80
static constexpr size_t SR_DA_GAS_LIMIT_NEXT_ROW
Definition context.hpp:79
static constexpr size_t SR_PARENT_DA_GAS_LIMIT_NEXT_ROW
Definition context.hpp:83
static constexpr size_t SR_IS_STATIC_IF_STATIC_CALL
Definition context.hpp:56
static constexpr size_t SR_IS_STATIC_NEXT_ROW
Definition context.hpp:55
static constexpr size_t SR_NULLIFIER_TREE_ROOT_CONTINUITY
Definition context.hpp:100
static constexpr size_t SR_PARENT_L2_GAS_USED_NEXT_ROW
Definition context.hpp:85
static constexpr size_t SR_L2_GAS_USED_CONTINUITY
Definition context.hpp:91
void set(Column col, uint32_t row, const FF &value)
TestTraceContainer trace
StrictMock< MockContext > context
void check_interaction(tracegen::TestTraceContainer &trace)
TEST(AvmFixedVKTests, FixedVKCommitments)
Test that the fixed VK commitments agree with the ones computed from precomputed columns.
TestTraceContainer empty_trace()
Definition fixtures.cpp:153
lookup_settings< lookup_context_ctx_stack_rollback_settings_ > lookup_context_ctx_stack_rollback_settings
lookup_settings< lookup_context_ctx_stack_return_settings_ > lookup_context_ctx_stack_return_settings
permutation_settings< perm_context_ctx_stack_call_settings_ > perm_context_ctx_stack_call_settings