1#include <gmock/gmock.h>
2#include <gtest/gtest.h>
26using ::testing::NiceMock;
28using simulation::EventEmitter;
29using simulation::MerkleCheck;
30using simulation::MerkleCheckEvent;
31using simulation::MockExecutionIdManager;
32using simulation::MockGreaterThan;
33using simulation::NoopEventEmitter;
34using simulation::Poseidon2;
35using simulation::Poseidon2HashEvent;
36using simulation::Poseidon2PermutationEvent;
37using simulation::Poseidon2PermutationMemoryEvent;
40using tracegen::MerkleCheckTraceBuilder;
41using tracegen::Poseidon2TraceBuilder;
42using tracegen::TestTraceContainer;
49TEST(MerkleCheckConstrainingTest, EmptyRow)
54TEST(MerkleCheckConstrainingTest, ComputationCannotBeStoppedPrematurely)
56 TestTraceContainer
trace({
57 { { C::precomputed_first_row, 1 }, { C::merkle_check_sel, 0 } },
58 { { C::merkle_check_sel, 1 } },
59 { { C::merkle_check_sel, 1 } },
60 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 } },
61 { { C::merkle_check_sel, 0 } },
66 const uint32_t last_row_idx = 3;
68 trace.
set(C::merkle_check_end, last_row_idx, 0);
74TEST(MerkleCheckConstrainingTest, EndCannotBeOneOnFirstRow)
77 TestTraceContainer
trace({
79 { { C::precomputed_first_row, 1 }, { C::merkle_check_sel, 0 }, { C::merkle_check_end, 0 } },
83 check_relation<merkle_check>(trace);
86 trace.
set(C::merkle_check_sel, 0, 1);
87 trace.
set(C::merkle_check_end, 0, 1);
92TEST(MerkleCheckConstrainingTest, SelectorOnEnd)
96 TestTraceContainer
trace({
97 { { C::merkle_check_end, 1 }, { C::merkle_check_sel, 1 } },
103 trace.
set(C::merkle_check_sel, 0, 0);
106 "SEL_ON_START_OR_END");
109TEST(MerkleCheckConstrainingTest, SelectorOnStart)
113 TestTraceContainer
trace({
114 { { C::merkle_check_start, 1 }, { C::merkle_check_sel, 1 } },
120 trace.
set(C::merkle_check_sel, 0, 0);
123 "SEL_ON_START_OR_END");
126TEST(MerkleCheckConstrainingTest, PropagateReadRoot)
131 TestTraceContainer
trace({
132 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_read_root, 123 } },
134 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_read_root, 123 } },
136 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_read_root, 456 } },
142 trace.
set(C::merkle_check_read_root, 1, 456);
145 "PROPAGATE_READ_ROOT");
148TEST(MerkleCheckConstrainingTest, PropagateWriteRoot)
153 TestTraceContainer
trace({
154 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_write_root, 123 } },
156 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_write_root, 123 } },
158 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_write_root, 456 } },
164 trace.
set(C::merkle_check_write_root, 1, 456);
167 "PROPAGATE_WRITE_ROOT");
170TEST(MerkleCheckConstrainingTest, PropagateWrite)
175 TestTraceContainer
trace({
176 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_write, 1 } },
178 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_write, 1 } },
180 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_write, 0 } },
186 trace.
set(C::merkle_check_write, 1, 0);
191TEST(MerkleCheckConstrainingTest, PathLenDecrements)
193 TestTraceContainer
trace({
195 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_path_len, 3 } },
196 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_path_len, 2 } },
197 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_path_len, 1 } },
199 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_path_len, 5 } },
205 trace.
set(C::merkle_check_path_len, 1, 1);
208 "PATH_LEN_DECREMENTS");
211TEST(MerkleCheckConstrainingTest, EndWhenPathLenOne)
213 TestTraceContainer
trace({
214 { { C::merkle_check_sel, 1 },
215 { C::merkle_check_path_len, 2 },
216 { C::merkle_check_path_len_min_one_inv,
FF(1).invert() },
217 { C::merkle_check_end, 0 } },
218 { { C::merkle_check_sel, 1 },
219 { C::merkle_check_path_len, 1 },
220 { C::merkle_check_path_len_min_one_inv, 0 },
221 { C::merkle_check_end, 1 } },
227 trace.
set(C::merkle_check_end, 1, 0);
230 "END_IFF_REM_PATH_EMPTY");
233TEST(MerkleCheckConstrainingTest, NextIndexIsHalved)
235 TestTraceContainer
trace({
236 { { C::merkle_check_sel, 1 },
237 { C::merkle_check_end, 0 },
238 { C::merkle_check_index, 6 },
239 { C::merkle_check_index_is_even, 1 } },
240 { { C::merkle_check_sel, 1 },
241 { C::merkle_check_end, 0 },
242 { C::merkle_check_index, 3 },
243 { C::merkle_check_index_is_even, 0 } },
244 { { C::merkle_check_sel, 1 },
245 { C::merkle_check_end, 1 },
246 { C::merkle_check_index, 1 },
247 { C::merkle_check_index_is_even, 0 } },
253 TestTraceContainer trace2({
254 { { C::merkle_check_sel, 1 },
255 { C::merkle_check_end, 0 },
256 { C::merkle_check_index, 7 },
257 { C::merkle_check_index_is_even, 0 } },
258 { { C::merkle_check_sel, 1 },
259 { C::merkle_check_end, 0 },
260 { C::merkle_check_index, 3 },
261 { C::merkle_check_index_is_even, 0 } },
262 { { C::merkle_check_sel, 1 },
263 { C::merkle_check_end, 1 },
264 { C::merkle_check_index, 1 },
265 { C::merkle_check_index_is_even, 0 } },
271 trace2.set(C::merkle_check_index, 1, 4);
274 "NEXT_INDEX_IS_HALVED");
277TEST(MerkleCheckConstrainingTest, AssignReadNodesEven)
280 TestTraceContainer
trace({
282 { C::merkle_check_sel, 1 },
283 { C::merkle_check_index_is_even, 1 },
284 { C::merkle_check_read_node, 123 },
285 { C::merkle_check_sibling, 456 },
286 { C::merkle_check_read_left_node, 123 },
287 { C::merkle_check_read_right_node, 456 },
294 trace.
set(C::merkle_check_read_left_node, 0, 456);
295 trace.
set(C::merkle_check_read_right_node, 0, 123);
301TEST(MerkleCheckConstrainingTest, AssignReadNodesOdd)
304 TestTraceContainer
trace({
306 { C::merkle_check_sel, 1 },
307 { C::merkle_check_index_is_even, 0 },
308 { C::merkle_check_read_node, 123 },
309 { C::merkle_check_sibling, 456 },
310 { C::merkle_check_read_left_node, 456 },
311 { C::merkle_check_read_right_node, 123 },
318 trace.
set(C::merkle_check_read_left_node, 0, 123);
319 trace.
set(C::merkle_check_read_right_node, 0, 456);
325TEST(MerkleCheckConstrainingTest, AssignWriteNodesEven)
328 TestTraceContainer
trace({
330 { C::merkle_check_sel, 1 },
331 { C::merkle_check_write, 1 },
332 { C::merkle_check_index_is_even, 1 },
333 { C::merkle_check_write_node, 123 },
334 { C::merkle_check_sibling, 456 },
335 { C::merkle_check_write_left_node, 123 },
336 { C::merkle_check_write_right_node, 456 },
343 trace.
set(C::merkle_check_write_left_node, 0, 456);
344 trace.
set(C::merkle_check_write_right_node, 0, 123);
351TEST(MerkleCheckConstrainingTest, AssignWriteNodesOdd)
354 TestTraceContainer
trace({
356 { C::merkle_check_sel, 1 },
357 { C::merkle_check_write, 1 },
358 { C::merkle_check_index_is_even, 0 },
359 { C::merkle_check_write_node, 123 },
360 { C::merkle_check_sibling, 456 },
361 { C::merkle_check_write_left_node, 456 },
362 { C::merkle_check_write_right_node, 123 },
369 trace.
set(C::merkle_check_write_left_node, 0, 123);
370 trace.
set(C::merkle_check_write_right_node, 0, 456);
377TEST(MerkleCheckConstrainingTest, ReadOutputHashIsNextRowsNode)
379 FF left_node =
FF(123);
380 FF right_node =
FF(456);
381 FF output_hash = UnconstrainedPoseidon2::hash({ left_node, right_node });
383 TestTraceContainer
trace({
384 { { C::merkle_check_sel, 1 },
385 { C::merkle_check_end, 0 },
386 { C::merkle_check_read_node, left_node },
387 { C::merkle_check_read_right_node, right_node },
388 { C::merkle_check_read_output_hash, output_hash } },
389 { { C::merkle_check_sel, 1 },
390 { C::merkle_check_end, 1 },
391 { C::merkle_check_read_node, output_hash } },
397 trace.
set(C::merkle_check_read_node, 1, output_hash + 1);
400 "OUTPUT_HASH_IS_NEXT_ROWS_READ_NODE");
403TEST(MerkleCheckConstrainingTest, WriteOutputHashIsNextRowsNode)
405 FF left_node =
FF(123);
406 FF right_node =
FF(456);
407 FF output_hash = UnconstrainedPoseidon2::hash({ left_node, right_node });
409 TestTraceContainer
trace({
410 { { C::merkle_check_sel, 1 },
411 { C::merkle_check_end, 0 },
412 { C::merkle_check_write_node, left_node },
413 { C::merkle_check_write_right_node, right_node },
414 { C::merkle_check_write_output_hash, output_hash } },
415 { { C::merkle_check_sel, 1 },
416 { C::merkle_check_end, 1 },
417 { C::merkle_check_write_node, output_hash } },
423 trace.
set(C::merkle_check_write_node, 1, output_hash + 1);
426 "OUTPUT_HASH_IS_NEXT_ROWS_WRITE_NODE");
429TEST(MerkleCheckConstrainingTest, OutputHashIsNotNextRowsCurrentNodeValueForLastRow)
431 FF output_hash =
FF(456);
432 FF next_current_node =
FF(789);
434 TestTraceContainer
trace({
435 { { C::merkle_check_sel, 1 },
436 { C::merkle_check_end, 1 },
437 { C::merkle_check_read_output_hash, output_hash },
438 { C::merkle_check_write_output_hash, output_hash } },
439 { { C::merkle_check_sel, 1 },
440 { C::merkle_check_read_node, next_current_node },
441 { C::merkle_check_write_node, next_current_node } },
448TEST(MerkleCheckConstrainingTest, ReadWithTracegen)
450 TestTraceContainer
trace({
451 { { C::precomputed_first_row, 1 } },
453 MerkleCheckTraceBuilder
builder;
456 FF leaf_value =
FF(123);
457 uint64_t leaf_index = 5;
460 std::vector<FF> sibling_path = {
FF(456),
FF(789),
FF(3333) };
466 .leaf_value = leaf_value,
467 .leaf_index = leaf_index,
468 .sibling_path = sibling_path,
474 check_relation<merkle_check>(trace);
479 trace.
set(C::merkle_check_path_len, last_row, 66);
484TEST(MerkleCheckConstrainingTest, WriteWithTracegen)
486 TestTraceContainer
trace({
487 { { C::precomputed_first_row, 1 } },
489 MerkleCheckTraceBuilder
builder;
492 FF leaf_value =
FF(123);
493 FF new_leaf_value =
FF(456);
494 uint64_t leaf_index = 5;
497 std::vector<FF> sibling_path = {
FF(456),
FF(789),
FF(3333) };
505 .leaf_value = leaf_value,
506 .new_leaf_value = new_leaf_value,
507 .leaf_index = leaf_index,
508 .sibling_path = sibling_path,
510 .new_root = new_root };
515 check_relation<merkle_check>(trace);
518class MerkleCheckPoseidon2Test :
public ::testing::Test {
520 MerkleCheckPoseidon2Test() =
default;
529 Poseidon2(execution_id_manager, mock_gt, hash_event_emitter, perm_event_emitter, perm_mem_event_emitter);
532TEST_F(MerkleCheckPoseidon2Test, ReadWithInteractions)
534 EventEmitter<MerkleCheckEvent> merkle_event_emitter;
535 MerkleCheck merkle_check_sim(
poseidon2, merkle_event_emitter);
537 TestTraceContainer
trace({ { { C::precomputed_first_row, 1 } } });
539 MerkleCheckTraceBuilder merkle_check_builder;
542 uint64_t leaf_index = 30;
543 std::vector<FF> sibling_path = { 10, 2, 30, 4, 50, 6 };
545 merkle_check_sim.assert_membership(
DOM_SEP__MERKLE_HASH, leaf_value, leaf_index, sibling_path, root);
548 merkle_check_builder.process(merkle_event_emitter.dump_events(), trace);
554 check_relation<merkle_check>(trace);
557 trace.
set(Column::merkle_check_read_output_hash,
static_cast<uint32_t
>(sibling_path.size()), 66);
560 (check_interaction<MerkleCheckTraceBuilder, lookup_merkle_check_merkle_poseidon2_read_settings>(trace)),
561 "Failed.*LOOKUP_MERKLE_CHECK_MERKLE_POSEIDON2.* Could not find tuple in destination");
562 check_interaction<MerkleCheckTraceBuilder, lookup_merkle_check_merkle_poseidon2_write_settings>(trace);
565TEST_F(MerkleCheckPoseidon2Test, WriteWithInteractions)
567 EventEmitter<MerkleCheckEvent> merkle_event_emitter;
568 MerkleCheck merkle_check_sim(
poseidon2, merkle_event_emitter);
570 TestTraceContainer
trace({ { { C::precomputed_first_row, 1 } } });
572 MerkleCheckTraceBuilder merkle_check_builder;
575 FF new_leaf_value = 444;
576 uint64_t leaf_index = 30;
577 std::vector<FF> sibling_path = { 10, 2, 30, 4, 50, 6 };
582 merkle_check_sim.write(
DOM_SEP__MERKLE_HASH, leaf_value, new_leaf_value, leaf_index, sibling_path, root);
584 EXPECT_EQ(new_root, expected_new_root);
587 merkle_check_builder.process(merkle_event_emitter.dump_events(), trace);
593 check_relation<merkle_check>(trace);
596 trace.
set(Column::merkle_check_read_output_hash,
static_cast<uint32_t
>(sibling_path.size()), 66);
597 trace.
set(Column::merkle_check_write_output_hash,
static_cast<uint32_t
>(sibling_path.size()), 77);
600 (check_interaction<MerkleCheckTraceBuilder, lookup_merkle_check_merkle_poseidon2_read_settings>(trace)),
601 "Failed.*LOOKUP_MERKLE_CHECK_MERKLE_POSEIDON2.* Could not find tuple in destination");
604 (check_interaction<MerkleCheckTraceBuilder, lookup_merkle_check_merkle_poseidon2_write_settings>(trace)),
605 "Failed.*LOOKUP_MERKLE_CHECK_MERKLE_POSEIDON2_WRITE.* Could not find tuple in "
609TEST_F(MerkleCheckPoseidon2Test, MultipleWithTracegen)
611 TestTraceContainer
trace({
612 { { C::precomputed_first_row, 1 } },
614 MerkleCheckTraceBuilder
builder;
617 uint64_t leaf_index = 30;
618 std::vector<FF> sibling_path = { 10, 2, 30, 4, 50, 6 };
621 .leaf_value = leaf_value,
622 .leaf_index = leaf_index,
623 .sibling_path = sibling_path,
626 FF leaf_value2 = 444;
627 FF new_leaf_value2 = 555;
628 uint64_t leaf_index2 = 40;
629 std::vector<FF> sibling_path2 = { 11, 22, 33, 44, 55, 66 };
633 .leaf_value = leaf_value2,
634 .new_leaf_value = new_leaf_value2,
635 .leaf_index = leaf_index2,
636 .sibling_path = sibling_path2,
638 .new_root = new_root2 };
643 uint32_t after_last_row_index = 1 +
static_cast<uint32_t
>(sibling_path.size() + sibling_path2.size());
644 trace.
set(Column::merkle_check_sel, after_last_row_index, 0);
645 trace.
set(Column::merkle_check_write, after_last_row_index, 0);
646 trace.
set(Column::merkle_check_read_node, after_last_row_index, 0);
647 trace.
set(Column::merkle_check_write_node, after_last_row_index, 0);
648 trace.
set(Column::merkle_check_index, after_last_row_index, 0);
649 trace.
set(Column::merkle_check_path_len, after_last_row_index, 0);
650 trace.
set(Column::merkle_check_path_len_min_one_inv, after_last_row_index, 0);
651 trace.
set(Column::merkle_check_read_root, after_last_row_index, 0);
652 trace.
set(Column::merkle_check_write_root, after_last_row_index, 0);
653 trace.
set(Column::merkle_check_sibling, after_last_row_index, 0);
654 trace.
set(Column::merkle_check_start, after_last_row_index, 0);
655 trace.
set(Column::merkle_check_end, after_last_row_index, 0);
656 trace.
set(Column::merkle_check_index_is_even, after_last_row_index, 0);
657 trace.
set(Column::merkle_check_read_left_node, after_last_row_index, 0);
658 trace.
set(Column::merkle_check_read_right_node, after_last_row_index, 0);
659 trace.
set(Column::merkle_check_write_left_node, after_last_row_index, 0);
660 trace.
set(Column::merkle_check_write_right_node, after_last_row_index, 0);
661 trace.
set(Column::merkle_check_read_output_hash, after_last_row_index, 0);
662 trace.
set(Column::merkle_check_write_output_hash, after_last_row_index, 0);
664 check_relation<merkle_check>(trace);
667TEST_F(MerkleCheckPoseidon2Test, MultipleWithInteractions)
669 EventEmitter<MerkleCheckEvent> merkle_event_emitter;
670 MerkleCheck merkle_check_sim(
poseidon2, merkle_event_emitter);
672 TestTraceContainer
trace({ { { C::precomputed_first_row, 1 } } });
673 MerkleCheckTraceBuilder merkle_check_builder;
677 uint64_t leaf_index = 30;
678 std::vector<FF> sibling_path = { 10, 2, 30, 4, 50, 6 };
681 merkle_check_sim.assert_membership(
DOM_SEP__MERKLE_HASH, leaf_value, leaf_index, sibling_path, root);
683 FF leaf_value2 = 444;
684 FF new_leaf_value2 = 555;
685 uint64_t leaf_index2 = 40;
686 std::vector<FF> sibling_path2 = { 11, 22, 33, 44, 55, 66 };
688 FF expected_new_root2 =
692 merkle_check_sim.write(
DOM_SEP__MERKLE_HASH, leaf_value2, new_leaf_value2, leaf_index2, sibling_path2, root2);
693 EXPECT_EQ(new_root2, expected_new_root2);
696 merkle_check_builder.process(merkle_event_emitter.dump_events(), trace);
702 check_relation<merkle_check>(trace);
#define EXPECT_THROW_WITH_MESSAGE(code, expectedMessageRegex)
#define DOM_SEP__MERKLE_HASH
StrictMock< MockGreaterThan > mock_gt
EventEmitter< Poseidon2PermutationMemoryEvent > perm_mem_event_emitter
EventEmitter< Poseidon2PermutationEvent > perm_event_emitter
EventEmitter< Poseidon2HashEvent > hash_event_emitter
Poseidon2TraceBuilder poseidon2_builder
static constexpr size_t SR_READ_RIGHT_NODE
static constexpr size_t SR_OUTPUT_HASH_IS_NEXT_ROWS_WRITE_NODE
static constexpr size_t SR_READ_LEFT_NODE
static constexpr size_t SR_PROPAGATE_READ_ROOT
static constexpr size_t SR_PROPAGATE_WRITE
static constexpr size_t SR_WRITE_RIGHT_NODE
static constexpr size_t SR_TRACE_CONTINUITY
static constexpr size_t SR_NEXT_INDEX_IS_HALVED
static constexpr size_t SR_PROPAGATE_WRITE_ROOT
static constexpr size_t SR_PATH_LEN_DECREMENTS
static constexpr size_t SR_WRITE_LEFT_NODE
static constexpr size_t SR_OUTPUT_HASH_IS_NEXT_ROWS_READ_NODE
static constexpr size_t SR_SEL_ON_START_OR_END
static constexpr size_t SR_END_IFF_REM_PATH_EMPTY
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_hash(const simulation::EventEmitterInterface< simulation::Poseidon2HashEvent >::Container &hash_events, TraceContainer &trace)
Processes the hash events for the Poseidon2 hash function. It populates the columns for the poseidon2...
uint32_t get_num_rows() const
void set(Column col, uint32_t row, const FF &value)
Native Poseidon2 hash function implementation.
ExecutionIdManager execution_id_manager
void check_interaction(tracegen::TestTraceContainer &trace)
TEST(AvmFixedVKTests, FixedVKCommitments)
Test that the fixed VK commitments agree with the ones computed from precomputed columns.
TEST_F(AvmRecursiveTests, TwoLayerAvmRecursionFailsWithWrongPIs)
FF unconstrained_root_from_path(uint64_t domain_separator, const FF &leaf_value, const uint64_t leaf_index, std::span< const FF > path)
TestTraceContainer empty_trace()
lookup_settings< lookup_merkle_check_merkle_poseidon2_read_settings_ > lookup_merkle_check_merkle_poseidon2_read_settings
lookup_settings< lookup_merkle_check_merkle_poseidon2_write_settings_ > lookup_merkle_check_merkle_poseidon2_write_settings
simulation::PublicDataTreeReadWriteEvent event