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 } },
64 check_relation<merkle_check>(
67 const uint32_t last_row_idx = 3;
69 trace.
set(C::merkle_check_end, last_row_idx, 0);
72 "COMPUTATION_FINISH_AT_END");
75TEST(MerkleCheckConstrainingTest, EndCannotBeOneOnFirstRow)
78 TestTraceContainer
trace({
80 { { C::precomputed_first_row, 1 }, { C::merkle_check_sel, 0 }, { C::merkle_check_end, 0 } },
84 check_relation<merkle_check>(trace);
87 trace.
set(C::merkle_check_sel, 0, 1);
88 trace.
set(C::merkle_check_end, 0, 1);
93TEST(MerkleCheckConstrainingTest, SelectorOnEnd)
97 TestTraceContainer
trace({
98 { { C::merkle_check_end, 1 }, { C::merkle_check_sel, 1 } },
104 trace.
set(C::merkle_check_sel, 0, 0);
107 "SELECTOR_ON_START_OR_END");
110TEST(MerkleCheckConstrainingTest, SelectorOnStart)
114 TestTraceContainer
trace({
115 { { C::merkle_check_start, 1 }, { C::merkle_check_sel, 1 } },
121 trace.
set(C::merkle_check_sel, 0, 0);
124 "SELECTOR_ON_START_OR_END");
127TEST(MerkleCheckConstrainingTest, PropagateReadRoot)
132 TestTraceContainer
trace({
133 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_read_root, 123 } },
135 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_read_root, 123 } },
137 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_read_root, 456 } },
143 trace.
set(C::merkle_check_read_root, 1, 456);
146 "PROPAGATE_READ_ROOT");
149TEST(MerkleCheckConstrainingTest, PropagateWriteRoot)
154 TestTraceContainer
trace({
155 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_write_root, 123 } },
157 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_write_root, 123 } },
159 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_write_root, 456 } },
165 trace.
set(C::merkle_check_write_root, 1, 456);
168 "PROPAGATE_WRITE_ROOT");
171TEST(MerkleCheckConstrainingTest, PropagateWrite)
176 TestTraceContainer
trace({
177 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_write, 1 } },
179 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_write, 1 } },
181 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_write, 0 } },
187 trace.
set(C::merkle_check_write, 1, 0);
192TEST(MerkleCheckConstrainingTest, PathLenDecrements)
194 TestTraceContainer
trace({
196 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_path_len, 3 } },
197 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 0 }, { C::merkle_check_path_len, 2 } },
198 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_path_len, 1 } },
200 { { C::merkle_check_sel, 1 }, { C::merkle_check_end, 1 }, { C::merkle_check_path_len, 5 } },
206 trace.
set(C::merkle_check_path_len, 1, 1);
209 "PATH_LEN_DECREMENTS");
212TEST(MerkleCheckConstrainingTest, EndWhenPathLenOne)
214 TestTraceContainer
trace({
215 { { C::merkle_check_sel, 1 },
216 { C::merkle_check_path_len, 2 },
217 { C::merkle_check_path_len_min_one_inv,
FF(1).invert() },
218 { C::merkle_check_end, 0 } },
219 { { C::merkle_check_sel, 1 },
220 { C::merkle_check_path_len, 1 },
221 { C::merkle_check_path_len_min_one_inv, 0 },
222 { C::merkle_check_end, 1 } },
228 trace.
set(C::merkle_check_end, 1, 0);
231 "END_IFF_REM_PATH_EMPTY");
234TEST(MerkleCheckConstrainingTest, NextIndexIsHalved)
236 TestTraceContainer
trace({
237 { { C::merkle_check_sel, 1 },
238 { C::merkle_check_end, 0 },
239 { C::merkle_check_index, 6 },
240 { C::merkle_check_index_is_even, 1 } },
241 { { C::merkle_check_sel, 1 },
242 { C::merkle_check_end, 0 },
243 { C::merkle_check_index, 3 },
244 { C::merkle_check_index_is_even, 0 } },
245 { { C::merkle_check_sel, 1 },
246 { C::merkle_check_end, 1 },
247 { C::merkle_check_index, 1 },
248 { C::merkle_check_index_is_even, 0 } },
254 TestTraceContainer trace2({
255 { { C::merkle_check_sel, 1 },
256 { C::merkle_check_end, 0 },
257 { C::merkle_check_index, 7 },
258 { C::merkle_check_index_is_even, 0 } },
259 { { C::merkle_check_sel, 1 },
260 { C::merkle_check_end, 0 },
261 { C::merkle_check_index, 3 },
262 { C::merkle_check_index_is_even, 0 } },
263 { { C::merkle_check_sel, 1 },
264 { C::merkle_check_end, 1 },
265 { C::merkle_check_index, 1 },
266 { C::merkle_check_index_is_even, 0 } },
272 trace2.set(C::merkle_check_index, 1, 4);
275 "NEXT_INDEX_IS_HALVED");
278TEST(MerkleCheckConstrainingTest, AssignReadNodesEven)
281 TestTraceContainer
trace({
283 { C::merkle_check_sel, 1 },
284 { C::merkle_check_index_is_even, 1 },
285 { C::merkle_check_read_node, 123 },
286 { C::merkle_check_sibling, 456 },
287 { C::merkle_check_read_left_node, 123 },
288 { C::merkle_check_read_right_node, 456 },
295 trace.
set(C::merkle_check_read_left_node, 0, 456);
296 trace.
set(C::merkle_check_read_right_node, 0, 123);
302TEST(MerkleCheckConstrainingTest, AssignReadNodesOdd)
305 TestTraceContainer
trace({
307 { C::merkle_check_sel, 1 },
308 { C::merkle_check_index_is_even, 0 },
309 { C::merkle_check_read_node, 123 },
310 { C::merkle_check_sibling, 456 },
311 { C::merkle_check_read_left_node, 456 },
312 { C::merkle_check_read_right_node, 123 },
319 trace.
set(C::merkle_check_read_left_node, 0, 123);
320 trace.
set(C::merkle_check_read_right_node, 0, 456);
326TEST(MerkleCheckConstrainingTest, AssignWriteNodesEven)
329 TestTraceContainer
trace({
331 { C::merkle_check_sel, 1 },
332 { C::merkle_check_write, 1 },
333 { C::merkle_check_index_is_even, 1 },
334 { C::merkle_check_write_node, 123 },
335 { C::merkle_check_sibling, 456 },
336 { C::merkle_check_write_left_node, 123 },
337 { C::merkle_check_write_right_node, 456 },
344 trace.
set(C::merkle_check_write_left_node, 0, 456);
345 trace.
set(C::merkle_check_write_right_node, 0, 123);
352TEST(MerkleCheckConstrainingTest, AssignWriteNodesOdd)
355 TestTraceContainer
trace({
357 { C::merkle_check_sel, 1 },
358 { C::merkle_check_write, 1 },
359 { C::merkle_check_index_is_even, 0 },
360 { C::merkle_check_write_node, 123 },
361 { C::merkle_check_sibling, 456 },
362 { C::merkle_check_write_left_node, 456 },
363 { C::merkle_check_write_right_node, 123 },
370 trace.
set(C::merkle_check_write_left_node, 0, 123);
371 trace.
set(C::merkle_check_write_right_node, 0, 456);
378TEST(MerkleCheckConstrainingTest, ReadOutputHashIsNextRowsNode)
380 FF left_node =
FF(123);
381 FF right_node =
FF(456);
382 FF output_hash = UnconstrainedPoseidon2::hash({ left_node, right_node });
384 TestTraceContainer
trace({
385 { { C::merkle_check_sel, 1 },
386 { C::merkle_check_end, 0 },
387 { C::merkle_check_read_node, left_node },
388 { C::merkle_check_read_right_node, right_node },
389 { C::merkle_check_read_output_hash, output_hash } },
390 { { C::merkle_check_sel, 1 },
391 { C::merkle_check_end, 1 },
392 { C::merkle_check_read_node, output_hash } },
398 trace.
set(C::merkle_check_read_node, 1, output_hash + 1);
401 "OUTPUT_HASH_IS_NEXT_ROWS_READ_NODE");
404TEST(MerkleCheckConstrainingTest, WriteOutputHashIsNextRowsNode)
406 FF left_node =
FF(123);
407 FF right_node =
FF(456);
408 FF output_hash = UnconstrainedPoseidon2::hash({ left_node, right_node });
410 TestTraceContainer
trace({
411 { { C::merkle_check_sel, 1 },
412 { C::merkle_check_end, 0 },
413 { C::merkle_check_write_node, left_node },
414 { C::merkle_check_write_right_node, right_node },
415 { C::merkle_check_write_output_hash, output_hash } },
416 { { C::merkle_check_sel, 1 },
417 { C::merkle_check_end, 1 },
418 { C::merkle_check_write_node, output_hash } },
424 trace.
set(C::merkle_check_write_node, 1, output_hash + 1);
427 "OUTPUT_HASH_IS_NEXT_ROWS_WRITE_NODE");
430TEST(MerkleCheckConstrainingTest, OutputHashIsNotNextRowsCurrentNodeValueForLastRow)
432 FF output_hash =
FF(456);
433 FF next_current_node =
FF(789);
435 TestTraceContainer
trace({
436 { { C::merkle_check_sel, 1 },
437 { C::merkle_check_end, 1 },
438 { C::merkle_check_read_output_hash, output_hash },
439 { C::merkle_check_write_output_hash, output_hash } },
440 { { C::merkle_check_sel, 1 },
441 { C::merkle_check_read_node, next_current_node },
442 { C::merkle_check_write_node, next_current_node } },
449TEST(MerkleCheckConstrainingTest, ReadWithTracegen)
451 TestTraceContainer
trace({
452 { { C::precomputed_first_row, 1 } },
454 MerkleCheckTraceBuilder
builder;
457 FF leaf_value =
FF(123);
458 uint64_t leaf_index = 5;
461 std::vector<FF> sibling_path = {
FF(456),
FF(789),
FF(3333) };
466 MerkleCheckEvent
event = {
467 .leaf_value = leaf_value, .leaf_index = leaf_index, .sibling_path = sibling_path, .root = root
473 check_relation<merkle_check>(trace);
478 trace.
set(C::merkle_check_path_len, last_row, 66);
483TEST(MerkleCheckConstrainingTest, WriteWithTracegen)
485 TestTraceContainer
trace({
486 { { C::precomputed_first_row, 1 } },
488 MerkleCheckTraceBuilder
builder;
491 FF leaf_value =
FF(123);
492 FF new_leaf_value =
FF(456);
493 uint64_t leaf_index = 5;
496 std::vector<FF> sibling_path = {
FF(456),
FF(789),
FF(3333) };
503 MerkleCheckEvent
event = { .leaf_value = leaf_value,
504 .new_leaf_value = new_leaf_value,
505 .leaf_index = leaf_index,
506 .sibling_path = sibling_path,
508 .new_root = new_root };
513 check_relation<merkle_check>(trace);
516class MerkleCheckPoseidon2Test :
public ::testing::Test {
518 MerkleCheckPoseidon2Test() =
default;
527 Poseidon2(execution_id_manager, mock_gt, hash_event_emitter, perm_event_emitter, perm_mem_event_emitter);
530TEST_F(MerkleCheckPoseidon2Test, ReadWithInteractions)
532 EventEmitter<MerkleCheckEvent> merkle_event_emitter;
533 MerkleCheck merkle_check_sim(
poseidon2, merkle_event_emitter);
535 TestTraceContainer
trace({ { { C::precomputed_first_row, 1 } } });
537 MerkleCheckTraceBuilder merkle_check_builder;
540 uint64_t leaf_index = 30;
541 std::vector<FF> sibling_path = { 10, 2, 30, 4, 50, 6 };
543 merkle_check_sim.assert_membership(leaf_value, leaf_index, sibling_path, root);
546 merkle_check_builder.process(merkle_event_emitter.dump_events(), trace);
552 check_relation<merkle_check>(trace);
555 trace.
set(Column::merkle_check_read_output_hash,
static_cast<uint32_t
>(sibling_path.size()), 66);
558 (check_interaction<MerkleCheckTraceBuilder, lookup_merkle_check_merkle_poseidon2_read_settings>(trace)),
559 "Failed.*LOOKUP_MERKLE_CHECK_MERKLE_POSEIDON2.* Could not find tuple in destination");
560 check_interaction<MerkleCheckTraceBuilder, lookup_merkle_check_merkle_poseidon2_write_settings>(trace);
563TEST_F(MerkleCheckPoseidon2Test, WriteWithInteractions)
565 EventEmitter<MerkleCheckEvent> merkle_event_emitter;
566 MerkleCheck merkle_check_sim(
poseidon2, merkle_event_emitter);
568 TestTraceContainer
trace({ { { C::precomputed_first_row, 1 } } });
570 MerkleCheckTraceBuilder merkle_check_builder;
573 FF new_leaf_value = 444;
574 uint64_t leaf_index = 30;
575 std::vector<FF> sibling_path = { 10, 2, 30, 4, 50, 6 };
579 FF new_root = merkle_check_sim.write(leaf_value, new_leaf_value, leaf_index, sibling_path, root);
581 EXPECT_EQ(new_root, expected_new_root);
584 merkle_check_builder.process(merkle_event_emitter.dump_events(), trace);
590 check_relation<merkle_check>(trace);
593 trace.
set(Column::merkle_check_read_output_hash,
static_cast<uint32_t
>(sibling_path.size()), 66);
594 trace.
set(Column::merkle_check_write_output_hash,
static_cast<uint32_t
>(sibling_path.size()), 77);
597 (check_interaction<MerkleCheckTraceBuilder, lookup_merkle_check_merkle_poseidon2_read_settings>(trace)),
598 "Failed.*LOOKUP_MERKLE_CHECK_MERKLE_POSEIDON2.* Could not find tuple in destination");
601 (check_interaction<MerkleCheckTraceBuilder, lookup_merkle_check_merkle_poseidon2_write_settings>(trace)),
602 "Failed.*LOOKUP_MERKLE_CHECK_MERKLE_POSEIDON2_WRITE.* Could not find tuple in "
606TEST_F(MerkleCheckPoseidon2Test, MultipleWithTracegen)
608 TestTraceContainer
trace({
609 { { C::precomputed_first_row, 1 } },
611 MerkleCheckTraceBuilder
builder;
614 uint64_t leaf_index = 30;
615 std::vector<FF> sibling_path = { 10, 2, 30, 4, 50, 6 };
617 MerkleCheckEvent
event = {
618 .leaf_value = leaf_value, .leaf_index = leaf_index, .sibling_path = sibling_path, .root = root
621 FF leaf_value2 = 444;
622 FF new_leaf_value2 = 555;
623 uint64_t leaf_index2 = 40;
624 std::vector<FF> sibling_path2 = { 11, 22, 33, 44, 55, 66 };
627 MerkleCheckEvent event2 = { .leaf_value = leaf_value2,
628 .new_leaf_value = new_leaf_value2,
629 .leaf_index = leaf_index2,
630 .sibling_path = sibling_path2,
632 .new_root = new_root2 };
637 uint32_t after_last_row_index = 1 +
static_cast<uint32_t
>(sibling_path.size() + sibling_path2.size());
638 trace.
set(Column::merkle_check_sel, after_last_row_index, 0);
639 trace.
set(Column::merkle_check_write, after_last_row_index, 0);
640 trace.
set(Column::merkle_check_read_node, after_last_row_index, 0);
641 trace.
set(Column::merkle_check_write_node, after_last_row_index, 0);
642 trace.
set(Column::merkle_check_index, after_last_row_index, 0);
643 trace.
set(Column::merkle_check_path_len, after_last_row_index, 0);
644 trace.
set(Column::merkle_check_path_len_min_one_inv, after_last_row_index, 0);
645 trace.
set(Column::merkle_check_read_root, after_last_row_index, 0);
646 trace.
set(Column::merkle_check_write_root, after_last_row_index, 0);
647 trace.
set(Column::merkle_check_sibling, after_last_row_index, 0);
648 trace.
set(Column::merkle_check_start, after_last_row_index, 0);
649 trace.
set(Column::merkle_check_end, after_last_row_index, 0);
650 trace.
set(Column::merkle_check_index_is_even, after_last_row_index, 0);
651 trace.
set(Column::merkle_check_read_left_node, after_last_row_index, 0);
652 trace.
set(Column::merkle_check_read_right_node, after_last_row_index, 0);
653 trace.
set(Column::merkle_check_write_left_node, after_last_row_index, 0);
654 trace.
set(Column::merkle_check_write_right_node, after_last_row_index, 0);
655 trace.
set(Column::merkle_check_read_output_hash, after_last_row_index, 0);
656 trace.
set(Column::merkle_check_write_output_hash, after_last_row_index, 0);
658 check_relation<merkle_check>(trace);
661TEST_F(MerkleCheckPoseidon2Test, MultipleWithInteractions)
663 EventEmitter<MerkleCheckEvent> merkle_event_emitter;
664 MerkleCheck merkle_check_sim(
poseidon2, merkle_event_emitter);
666 TestTraceContainer
trace({ { { C::precomputed_first_row, 1 } } });
667 MerkleCheckTraceBuilder merkle_check_builder;
671 uint64_t leaf_index = 30;
672 std::vector<FF> sibling_path = { 10, 2, 30, 4, 50, 6 };
675 merkle_check_sim.assert_membership(leaf_value, leaf_index, sibling_path, root);
677 FF leaf_value2 = 444;
678 FF new_leaf_value2 = 555;
679 uint64_t leaf_index2 = 40;
680 std::vector<FF> sibling_path2 = { 11, 22, 33, 44, 55, 66 };
684 FF new_root2 = merkle_check_sim.write(leaf_value2, new_leaf_value2, leaf_index2, sibling_path2, root2);
685 EXPECT_EQ(new_root2, expected_new_root2);
688 merkle_check_builder.process(merkle_event_emitter.dump_events(), trace);
694 check_relation<merkle_check>(trace);
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_NEXT_INDEX_IS_HALVED
static constexpr size_t SR_PROPAGATE_WRITE_ROOT
static constexpr size_t SR_SELECTOR_ON_START_OR_END
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_END_IFF_REM_PATH_EMPTY
static constexpr size_t SR_COMPUTATION_FINISH_AT_END
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)
uint32_t get_num_rows() const
void set(Column col, uint32_t row, const FF &value)
Implements a parallelized batch insertion indexed tree Accepts template argument of the type of store...
ExecutionIdManager execution_id_manager
#define EXPECT_THROW_WITH_MESSAGE(code, expectedMessage)
TEST_F(AvmRecursiveTests, GoblinRecursion)
A test of the Goblinized AVM recursive verifier.
void check_interaction(tracegen::TestTraceContainer &trace)
TEST(TxExecutionConstrainingTest, WriteTreeValue)
FF unconstrained_root_from_path(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