Barretenberg
The ZK-SNARK library at the core of Aztec
Loading...
Searching...
No Matches
get_contract_instance.test.cpp
Go to the documentation of this file.
1#include <gmock/gmock.h>
2#include <gtest/gtest.h>
3
4#include <cstdint>
5
22
23namespace bb::avm2::constraining {
24namespace {
25
26using simulation::EventEmitter;
27using simulation::GetContractInstanceEvent;
28using tracegen::GetContractInstanceTraceBuilder;
29using tracegen::MemoryTraceBuilder;
30using tracegen::PrecomputedTraceBuilder;
31using tracegen::TestTraceContainer;
33using C = Column;
34using get_contract_instance = bb::avm2::get_contract_instance<FF>;
35
36TEST(GetContractInstanceConstrainingTest, EmptyRow)
37{
38 check_relation<get_contract_instance>(testing::empty_trace());
39}
40
41TEST(GetContractInstanceConstrainingTest, WriteInBoundsCheck)
42{
43 // Test constants
44 const FF dst_offset = FF(100); // Use a smaller offset for clear testing
45 const FF dst_offset_diff_max = FF(AVM_HIGHEST_MEM_ADDRESS) - dst_offset; // AVM_HIGHEST_MEM_ADDRESS - 100
46 const FF dst_offset_diff_max_inv = dst_offset_diff_max.invert(); // 1/DST_OFFSET_DIFF_MAX
47 const FF wrong_inv_value = FF(42);
48
49 TestTraceContainer trace({
50 { { C::precomputed_first_row, 1 } },
51 { { C::get_contract_instance_sel, 1 },
52 { C::get_contract_instance_dst_offset, dst_offset },
53 { C::get_contract_instance_is_valid_writes_in_bounds, 1 },
54 { C::get_contract_instance_dst_offset_diff_max_inv, dst_offset_diff_max_inv } },
55 });
56
57 check_relation<get_contract_instance>(trace, get_contract_instance::SR_WRITE_OUT_OF_BOUNDS_CHECK);
58
59 // Negative test: mutate to incorrect dst_offset_diff_max_inv
60 trace.set(C::get_contract_instance_dst_offset_diff_max_inv, 1, wrong_inv_value); // Wrong inv value
62 check_relation<get_contract_instance>(trace, get_contract_instance::SR_WRITE_OUT_OF_BOUNDS_CHECK),
63 "WRITE_OUT_OF_BOUNDS_CHECK");
64 // Reset
65 trace.set(C::get_contract_instance_dst_offset_diff_max_inv, 1, dst_offset_diff_max_inv);
66
67 // Negative test: mutate to incorrect sel_write_in_bounds
68 trace.set(C::get_contract_instance_is_valid_writes_in_bounds, 1, 0); // Out of bounds
70 check_relation<get_contract_instance>(trace, get_contract_instance::SR_WRITE_OUT_OF_BOUNDS_CHECK),
71 "WRITE_OUT_OF_BOUNDS_CHECK");
72 // Reset
73 trace.set(C::get_contract_instance_is_valid_writes_in_bounds, 1, 1);
74}
75
76TEST(GetContractInstanceConstrainingTest, WriteOutOfBoundsCheck)
77{
78 // Test constants
79 const FF dst_offset = FF(AVM_HIGHEST_MEM_ADDRESS); // Boundary case: dst_offset + 1 is out of bounds
80 const FF dst_offset_diff_max_inv = FF(0);
81
82 TestTraceContainer trace({
83 { { C::precomputed_first_row, 1 } },
84 { { C::get_contract_instance_sel, 1 },
85 { C::get_contract_instance_dst_offset, dst_offset },
86 { C::get_contract_instance_is_valid_writes_in_bounds, 0 },
87 { C::get_contract_instance_dst_offset_diff_max_inv, dst_offset_diff_max_inv } },
88 });
89
90 check_relation<get_contract_instance>(trace, get_contract_instance::SR_WRITE_OUT_OF_BOUNDS_CHECK);
91
92 // Negative test: mutate to incorrect sel_write_in_bounds
93 trace.set(C::get_contract_instance_is_valid_writes_in_bounds, 1, 1);
95 check_relation<get_contract_instance>(trace, get_contract_instance::SR_WRITE_OUT_OF_BOUNDS_CHECK),
96 "WRITE_OUT_OF_BOUNDS_CHECK");
97 // Reset
98 trace.set(C::get_contract_instance_is_valid_writes_in_bounds, 1, 0);
99}
100
101TEST(GetContractInstanceConstrainingTest, ErrorAggregationConstraint)
102{
103 // Test error aggregation subrelation
104 TestTraceContainer trace({
105 { { C::precomputed_first_row, 1 } },
106 // No error case
107 { { C::get_contract_instance_sel, 1 },
108 { C::get_contract_instance_sel_error, 0 },
109 { C::get_contract_instance_is_valid_writes_in_bounds, 1 },
110 { C::get_contract_instance_is_valid_member_enum, 1 } },
111 });
112
113 check_relation<get_contract_instance>(trace, get_contract_instance::SR_ERROR_AGGREGATION);
114
115 // Test bounds error
116 trace.set(C::get_contract_instance_sel_error, 1, 1);
117 trace.set(C::get_contract_instance_is_valid_writes_in_bounds, 1, 0); // Out of bounds
118 check_relation<get_contract_instance>(trace, get_contract_instance::SR_ERROR_AGGREGATION);
119
120 // Test enum error
121 trace.set(C::get_contract_instance_sel_error, 1, 1);
122 trace.set(C::get_contract_instance_is_valid_writes_in_bounds, 1, 1); // In bounds
123 trace.set(C::get_contract_instance_is_valid_member_enum, 1, 0); // Invalid enum
124 check_relation<get_contract_instance>(trace, get_contract_instance::SR_ERROR_AGGREGATION);
125
126 // Test both errors
127 trace.set(C::get_contract_instance_sel_error, 1, 1);
128 trace.set(C::get_contract_instance_is_valid_writes_in_bounds, 1, 0); // Out of bounds
129 trace.set(C::get_contract_instance_is_valid_member_enum, 1, 0); // Invalid enum
130 check_relation<get_contract_instance>(trace, get_contract_instance::SR_ERROR_AGGREGATION);
131
132 // Negative test: wrong error value
133 trace.set(C::get_contract_instance_sel_error, 1, 0);
134 EXPECT_THROW_WITH_MESSAGE(check_relation<get_contract_instance>(trace, get_contract_instance::SR_ERROR_AGGREGATION),
135 "ERROR_AGGREGATION");
136}
137
138TEST(GetContractInstanceConstrainingTest, SelectedMemberConstraint)
139{
140 // Test constants
141 const FF deployer_addr = 0x1234;
142 const FF class_id = 0x5678;
143 const FF init_hash = 0x9ABC;
144 const FF wrong_value = 0x1111;
145
146 // Test selected member subrelation
147 TestTraceContainer trace({
148 { { C::precomputed_first_row, 1 } },
149 // DEPLOYER selection
150 { { C::get_contract_instance_sel, 1 },
151 { C::get_contract_instance_selected_member, deployer_addr },
152 { C::get_contract_instance_is_deployer, 1 },
153 { C::get_contract_instance_is_class_id, 0 },
154 { C::get_contract_instance_is_init_hash, 0 },
155 { C::get_contract_instance_retrieved_deployer_addr, deployer_addr },
156 { C::get_contract_instance_retrieved_class_id, class_id },
157 { C::get_contract_instance_retrieved_init_hash, init_hash } },
158 });
159
160 check_relation<get_contract_instance>(trace, get_contract_instance::SR_SELECTED_MEMBER);
161
162 // Test CLASS_ID selection
163 trace.set(C::get_contract_instance_selected_member, 1, class_id);
164 trace.set(C::get_contract_instance_is_deployer, 1, 0);
165 trace.set(C::get_contract_instance_is_class_id, 1, 1);
166 check_relation<get_contract_instance>(trace, get_contract_instance::SR_SELECTED_MEMBER);
167
168 // Test INIT_HASH selection
169 trace.set(C::get_contract_instance_selected_member, 1, init_hash);
170 trace.set(C::get_contract_instance_is_class_id, 1, 0);
171 trace.set(C::get_contract_instance_is_init_hash, 1, 1);
172 check_relation<get_contract_instance>(trace, get_contract_instance::SR_SELECTED_MEMBER);
173
174 // Negative test: wrong selected member
175 trace.set(C::get_contract_instance_selected_member, 1, wrong_value); // Wrong value
176 EXPECT_THROW_WITH_MESSAGE(check_relation<get_contract_instance>(trace, get_contract_instance::SR_SELECTED_MEMBER),
177 "SELECTED_MEMBER");
178}
179
180TEST(GetContractInstanceConstrainingTest, ComplexMultiRowSequence)
181{
182 // Test constants
183 const uint32_t dst_offset_1 = 100;
184 const uint32_t dst_offset_2 = 200;
185 const uint32_t dst_offset_3 = 300;
186 const uint8_t deployer_enum = 0;
187 const uint8_t class_id_enum = 1;
188 const uint8_t invalid_enum = 5;
189 const FF deployer_addr_1 = 0x1234;
190 const FF class_id_1 = 0x5678;
191 const FF init_hash_1 = 0x9ABC;
192 const FF deployer_addr_2 = 0x1111;
193 const FF class_id_2 = 0x2222;
194 const FF init_hash_2 = 0x3333;
195 const FF deployer_addr_3 = 0x4444;
196 const FF class_id_3 = 0x5555;
197 const FF init_hash_3 = 0x6666;
198 const uint32_t member_write_offset_1 = 101;
199 const uint32_t member_write_offset_2 = 201;
200 const uint32_t member_write_offset_3 = 301;
201 const uint8_t u1_tag = static_cast<uint8_t>(ValueTag::U1);
202 const uint8_t ff_tag = static_cast<uint8_t>(ValueTag::FF);
203
204 // Test multiple GetContractInstance operations in sequence
205 TestTraceContainer trace({
206 { { C::precomputed_first_row, 1 } },
207 // Row 1: skippable gadget selector
208 { { C::get_contract_instance_sel, 0 } }, // Must satisfy error constraint
209 // Row 2: Valid DEPLOYER retrieval
210 { { C::get_contract_instance_sel, 1 },
211 { C::get_contract_instance_dst_offset, dst_offset_1 },
212 { C::get_contract_instance_member_enum, deployer_enum }, // DEPLOYER
213 { C::get_contract_instance_is_valid_writes_in_bounds, 1 },
214 { C::get_contract_instance_dst_offset_diff_max_inv, FF(AVM_HIGHEST_MEM_ADDRESS - dst_offset_1).invert() },
215 { C::get_contract_instance_sel_error, 0 },
216 { C::get_contract_instance_is_valid_member_enum, 1 },
217 { C::get_contract_instance_is_deployer, 1 },
218 { C::get_contract_instance_is_class_id, 0 },
219 { C::get_contract_instance_is_init_hash, 0 },
220 { C::get_contract_instance_retrieved_deployer_addr, deployer_addr_1 },
221 { C::get_contract_instance_retrieved_class_id, class_id_1 },
222 { C::get_contract_instance_retrieved_init_hash, init_hash_1 },
223 { C::get_contract_instance_selected_member, deployer_addr_1 },
224 { C::get_contract_instance_member_write_offset, member_write_offset_1 },
225 { C::get_contract_instance_exists_tag, u1_tag },
226 { C::get_contract_instance_member_tag, ff_tag } },
227 // Row 3: Valid CLASS_ID retrieval
228 { { C::get_contract_instance_sel, 1 },
229 { C::get_contract_instance_dst_offset, dst_offset_2 },
230 { C::get_contract_instance_member_enum, class_id_enum }, // CLASS_ID
231 { C::get_contract_instance_is_valid_writes_in_bounds, 1 },
232 { C::get_contract_instance_dst_offset_diff_max_inv, FF(AVM_HIGHEST_MEM_ADDRESS - dst_offset_2).invert() },
233 { C::get_contract_instance_sel_error, 0 },
234 { C::get_contract_instance_is_valid_member_enum, 1 },
235 { C::get_contract_instance_is_deployer, 0 },
236 { C::get_contract_instance_is_class_id, 1 },
237 { C::get_contract_instance_is_init_hash, 0 },
238 { C::get_contract_instance_retrieved_deployer_addr, deployer_addr_2 },
239 { C::get_contract_instance_retrieved_class_id, class_id_2 },
240 { C::get_contract_instance_retrieved_init_hash, init_hash_2 },
241 { C::get_contract_instance_selected_member, class_id_2 },
242 { C::get_contract_instance_member_write_offset, member_write_offset_2 },
243 { C::get_contract_instance_exists_tag, u1_tag },
244 { C::get_contract_instance_member_tag, ff_tag } },
245 // Row 4: Invalid member enum with error
246 { { C::get_contract_instance_sel, 1 },
247 { C::get_contract_instance_dst_offset, dst_offset_3 },
248 { C::get_contract_instance_member_enum, invalid_enum }, // Invalid
249 { C::get_contract_instance_is_valid_writes_in_bounds, 1 },
250 { C::get_contract_instance_dst_offset_diff_max_inv, FF(AVM_HIGHEST_MEM_ADDRESS - dst_offset_3).invert() },
251 { C::get_contract_instance_sel_error, 1 }, // Error due to invalid enum
252 { C::get_contract_instance_is_valid_member_enum, 0 },
253 { C::get_contract_instance_is_deployer, 0 },
254 { C::get_contract_instance_is_class_id, 0 },
255 { C::get_contract_instance_is_init_hash, 0 },
256 { C::get_contract_instance_retrieved_deployer_addr, deployer_addr_3 },
257 { C::get_contract_instance_retrieved_class_id, class_id_3 },
258 { C::get_contract_instance_retrieved_init_hash, init_hash_3 },
259 { C::get_contract_instance_selected_member, 0 }, // No selection due to invalid enum
260 { C::get_contract_instance_member_write_offset, member_write_offset_3 },
261 { C::get_contract_instance_exists_tag, u1_tag },
262 { C::get_contract_instance_member_tag, ff_tag } },
263 });
264
265 check_relation<get_contract_instance>(trace);
266}
267
268// Integration-style tests using tracegen components
269TEST(GetContractInstanceConstrainingTest, IntegrationTracegenValid)
270{
271 // Test constants
272 const uint32_t execution_clk = 42;
273 const FF contract_address = 0x1234;
274 const uint32_t dst_offset = 100;
275 const uint8_t deployer_enum = static_cast<uint8_t>(ContractInstanceMember::DEPLOYER);
276 const uint16_t space_id = 1;
277 const FF nullifier_tree_root = 0x1234;
278 const FF public_data_tree_root = 0x5678;
279 const FF deployer_addr = 0x5678;
280 const FF class_id = 0x9ABC;
281 const FF init_hash = 0xDEF0;
282
283 // Use real tracegen to generate a valid trace
284 EventEmitter<GetContractInstanceEvent> emitter;
285
286 GetContractInstanceEvent event{
287 .execution_clk = execution_clk,
288 .contract_address = contract_address,
289 .dst_offset = dst_offset,
290 .member_enum = deployer_enum,
291 .space_id = space_id,
292 .nullifier_tree_root = nullifier_tree_root,
293 .public_data_tree_root = public_data_tree_root,
294 .instance_exists = true,
295 .retrieved_deployer_addr = deployer_addr,
296 .retrieved_class_id = class_id,
297 .retrieved_init_hash = init_hash,
298 };
299
300 emitter.emit(std::move(event));
301 auto events = emitter.dump_events();
302
303 TestTraceContainer trace;
304 GetContractInstanceTraceBuilder builder;
305 builder.process(events, trace);
306
307 // Add precomputed table entries
308 PrecomputedTraceBuilder precomputed_builder;
310
311 check_relation<get_contract_instance>(trace);
312}
313
314TEST(GetContractInstanceConstrainingTest, IntegrationTracegenInvalidEnum)
315{
316 // Test constants
317 const uint32_t execution_clk = 42;
318 const FF contract_address = 0x1234;
319 const uint32_t dst_offset = 100;
320 const uint8_t invalid_enum = 200;
321 const uint16_t space_id = 1;
322 const FF nullifier_tree_root = 0x1234;
323 const FF public_data_tree_root = 0x5678;
324 const FF deployer_addr = 0x5678;
325 const FF class_id = 0x9ABC;
326 const FF init_hash = 0xDEF0;
327
328 // Test with invalid member enum
329 EventEmitter<GetContractInstanceEvent> emitter;
330
331 GetContractInstanceEvent event{
332 .execution_clk = execution_clk,
333 .contract_address = contract_address,
334 .dst_offset = dst_offset,
335 .member_enum = invalid_enum,
336 .space_id = space_id,
337 .nullifier_tree_root = nullifier_tree_root,
338 .public_data_tree_root = public_data_tree_root,
339 .instance_exists = true,
340 .retrieved_deployer_addr = deployer_addr,
341 .retrieved_class_id = class_id,
342 .retrieved_init_hash = init_hash,
343 };
344
345 emitter.emit(std::move(event));
346 auto events = emitter.dump_events();
347
348 TestTraceContainer trace;
349 GetContractInstanceTraceBuilder builder;
350 builder.process(events, trace);
351
352 // Add precomputed table entries
353 PrecomputedTraceBuilder precomputed_builder;
355
356 check_relation<get_contract_instance>(trace);
357}
358
359TEST(GetContractInstanceConstrainingTest, IntegrationTracegenOutOfBounds)
360{
361 // Test constants
362 const uint32_t execution_clk = 42;
363 const FF contract_address = 0x1234;
364 const uint32_t dst_offset = AVM_HIGHEST_MEM_ADDRESS;
365 const uint8_t class_id_enum = static_cast<uint8_t>(ContractInstanceMember::CLASS_ID);
366 const uint16_t space_id = 1;
367 const FF nullifier_tree_root = 0x1234;
368 const FF public_data_tree_root = 0x5678;
369 const FF deployer_addr = 0x5678;
370 const FF class_id = 0x9ABC;
371 const FF init_hash = 0xDEF0;
372
373 // Test with out-of-bounds destination
374 EventEmitter<GetContractInstanceEvent> emitter;
375
376 GetContractInstanceEvent event{
377 .execution_clk = execution_clk,
378 .contract_address = contract_address,
379 .dst_offset = dst_offset,
380 .member_enum = class_id_enum,
381 .space_id = space_id,
382 .nullifier_tree_root = nullifier_tree_root,
383 .public_data_tree_root = public_data_tree_root,
384 .instance_exists = true,
385 .retrieved_deployer_addr = deployer_addr,
386 .retrieved_class_id = class_id,
387 .retrieved_init_hash = init_hash,
388 };
389
390 emitter.emit(std::move(event));
391 auto events = emitter.dump_events();
392
393 TestTraceContainer trace;
394 GetContractInstanceTraceBuilder builder;
395 builder.process(events, trace);
396
397 // Add precomputed table entries
398 PrecomputedTraceBuilder precomputed_builder;
400
401 check_relation<get_contract_instance>(trace);
402}
403
404// =====================================================================
405// Ghost Row Injection Vulnerability Tests
406// =====================================================================
407// These tests verify that ghost rows (sel=0) cannot fire permutations.
408// The fix: is_valid_member_enum * (1 - sel) = 0 ensures is_valid_member_enum
409// is forced to 0 when sel=0, preventing ghost rows from firing permutations.
410//
411// Test that the fix blocks ghost row injection attacks.
412// Attack pattern:
413// 1. Create legitimate memory WRITE events (destination side)
414// 2. Build memory trace from those events
415// 3. Inject ghost get_contract_instance row with sel=0 but is_valid_member_enum=1
416// 4. The fix should cause the relation check to fail
417TEST(GetContractInstanceConstrainingTest, NegativeGhostRowInjectionBlocked)
418{
419 TestTraceContainer trace;
420 MemoryTraceBuilder memory_trace_builder;
421 PrecomputedTraceBuilder precomputed_trace_builder;
422
423 // Attacker-controlled values
424 uint32_t malicious_clk = 42;
425 uint16_t malicious_space_id = 1;
426 MemoryAddress malicious_dst_offset = 0x100;
427 uint1_t malicious_instance_exists = 1;
428 MemoryTag exists_tag = MemoryTag::U1;
429
430 // Create legitimate memory events
432 {
433 .execution_clk = malicious_clk,
435 .addr = malicious_dst_offset,
436 .value = MemoryValue::from<uint1_t>(malicious_instance_exists),
437 .space_id = malicious_space_id,
438 },
439 };
440
441 // Build memory trace (destination side)
442 precomputed_trace_builder.process_sel_range_8(trace);
443 precomputed_trace_builder.process_sel_range_16(trace);
444 precomputed_trace_builder.process_misc(trace, 1 << 16);
445 precomputed_trace_builder.process_tag_parameters(trace);
446 memory_trace_builder.process(mem_events, trace);
447
448 // Find where the memory row was placed
449 uint32_t memory_row = 0;
450 for (uint32_t row = 0; row < trace.get_num_rows(); row++) {
451 if (trace.get(C::memory_sel, row) == 1) {
452 memory_row = row;
453 break;
454 }
455 }
456
457 // Inject ghost get_contract_instance row
458 // Ghost row: sel = 0, but is_valid_member_enum = 1 (attack attempt)
459 uint32_t ghost_row = 0;
460 trace.set(ghost_row,
461 std::vector<std::pair<Column, FF>>{
462 { C::precomputed_first_row, 1 },
463 { C::precomputed_clk, ghost_row },
464 { C::get_contract_instance_sel, 0 },
465 { C::get_contract_instance_is_valid_member_enum, 1 },
466 { C::get_contract_instance_is_valid_writes_in_bounds, 1 },
467 { C::get_contract_instance_exists_tag, static_cast<uint8_t>(exists_tag) },
468 { C::get_contract_instance_clk, malicious_clk },
469 { C::get_contract_instance_space_id, malicious_space_id },
470 { C::get_contract_instance_dst_offset, malicious_dst_offset },
471 { C::get_contract_instance_instance_exists, static_cast<uint64_t>(malicious_instance_exists) },
472 { C::get_contract_instance_sel_error, 0 },
473 { C::get_contract_instance_is_deployer, 0 },
474 { C::get_contract_instance_is_class_id, 0 },
475 { C::get_contract_instance_is_init_hash, 0 },
476 { C::get_contract_instance_selected_member, 0 },
477 { C::get_contract_instance_member_write_offset, malicious_dst_offset + 1 },
478 { C::get_contract_instance_member_tag, static_cast<uint8_t>(MemoryTag::FF) },
479 });
480
481 trace.set(C::memory_sel_get_contract_instance_exists_write, memory_row, 1);
482
483 // The fix: is_valid_writes_in_bounds * (1 - sel) = 0 should cause the relation check to fail
484 // (in conjunction with WRITES_OUT_OF_BOUNDS * is_valid_member_enum = 0)
485 EXPECT_THROW_WITH_MESSAGE(check_relation<get_contract_instance>(trace), "IS_VALID_WRITES_IN_BOUNDS_REQUIRES_SEL");
486}
487
488// M-1: Verifies tracegen sets member_write_offset = 0 when writes are out of bounds.
489// PIL constraint: member_write_offset = is_valid_writes_in_bounds * (dst_offset + 1)
490TEST(GetContractInstanceConstrainingTest, TracegenMemberWriteOffsetOutOfBounds)
491{
492 EventEmitter<GetContractInstanceEvent> emitter;
493 emitter.emit(GetContractInstanceEvent{
494 .execution_clk = 1,
495 .contract_address = 0x1234,
496 .dst_offset = AVM_HIGHEST_MEM_ADDRESS, // Boundary case: writes out of bounds
497 .member_enum = static_cast<uint8_t>(ContractInstanceMember::DEPLOYER),
498 .space_id = 1,
499 .nullifier_tree_root = 0x5678,
500 .public_data_tree_root = 0x9ABC,
501 .instance_exists = true,
502 .retrieved_deployer_addr = 0xDEAD,
503 .retrieved_class_id = 0xBEEF,
504 .retrieved_init_hash = 0xCAFE,
505 });
506
507 TestTraceContainer trace;
508 GetContractInstanceTraceBuilder().process(emitter.dump_events(), trace);
509 PrecomputedTraceBuilder().process_get_contract_instance_table(trace);
510
511 // Row 1 has the data (row 0 is skippable setup)
512 constexpr uint32_t data_row = 1;
513 EXPECT_EQ(trace.get(C::get_contract_instance_is_valid_writes_in_bounds, data_row), FF(0));
514 EXPECT_EQ(trace.get(C::get_contract_instance_member_write_offset, data_row), FF(0));
515 check_relation<get_contract_instance>(trace);
516
517 // Negative: incorrect value fails the constraint
518 trace.set(C::get_contract_instance_member_write_offset, data_row, FF(AVM_HIGHEST_MEM_ADDRESS) + FF(1));
520 check_relation<get_contract_instance>(trace, get_contract_instance::SR_MEMBER_WRITE_OFFSET),
521 "MEMBER_WRITE_OFFSET");
522}
523
524} // namespace
525} // namespace bb::avm2::constraining
#define EXPECT_THROW_WITH_MESSAGE(code, expectedMessageRegex)
Definition assert.hpp:193
#define AVM_HIGHEST_MEM_ADDRESS
static constexpr size_t SR_ERROR_AGGREGATION
static constexpr size_t SR_SELECTED_MEMBER
static constexpr size_t SR_MEMBER_WRITE_OFFSET
static constexpr size_t SR_WRITE_OUT_OF_BOUNDS_CHECK
void process(const simulation::EventEmitterInterface< simulation::AluEvent >::Container &events, TraceContainer &trace)
Process the ALU events and populate the ALU relevant columns in the trace.
void process_get_contract_instance_table(TraceContainer &trace)
const FF & get(Column col, uint32_t row) const
void set(Column col, uint32_t row, const FF &value)
PrecomputedTraceBuilder precomputed_builder
Definition alu.test.cpp:120
AluTraceBuilder builder
Definition alu.test.cpp:124
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
uint32_t MemoryAddress
AvmFlavorSettings::FF FF
Definition field.hpp:10
constexpr decltype(auto) get(::tuplet::tuple< T... > &&t) noexcept
Definition tuple.hpp:13
simulation::PublicDataTreeReadWriteEvent event
constexpr field invert() const noexcept