1#include <gmock/gmock.h>
2#include <gtest/gtest.h>
37using tracegen::TestTraceContainer;
41using simulation::RangeCheckEvent;
42using tracegen::AluTraceBuilder;
43using tracegen::ExecutionTraceBuilder;
44using tracegen::FieldGreaterThanTraceBuilder;
45using tracegen::GreaterThanTraceBuilder;
46using tracegen::PrecomputedTraceBuilder;
47using tracegen::RangeCheckTraceBuilder;
49constexpr uint8_t NUM_OF_TAGS =
static_cast<uint8_t
>(
MemoryTag::MAX) + 1;
98 for (
const auto c : out) {
99 ThreeOperandTestParams params =
tuple_cat(TEST_VALUES_IN.at(i), std::make_tuple(c));
100 res.push_back(params);
110 for (
const auto c : out) {
111 TwoOperandTestParams params = std::make_tuple(
std::get<0>(TEST_VALUES_IN.at(i)), c);
112 res.push_back(params);
118class AluConstrainingTest :
public ::testing::Test {
127TEST_F(AluConstrainingTest, EmptyRow)
132TEST_F(AluConstrainingTest, NegativeAluWrongOpId)
134 auto trace = TestTraceContainer({
137 { C::alu_sel_op_add, 1 },
158class AluAddConstrainingTest :
public AluConstrainingTest,
159 public ::testing::WithParamInterface<ThreeOperandTestParams> {
161 TestTraceContainer process_basic_add_trace(ThreeOperandTestParams params)
163 auto [
a,
b, c] = params;
164 auto tag =
static_cast<uint8_t
>(
a.get_tag());
165 auto trace = TestTraceContainer({
168 { C::alu_ia_tag,
tag },
170 { C::alu_ib_tag,
tag },
172 { C::alu_ic_tag,
tag },
177 { C::alu_sel_op_add, 1 },
178 { C::alu_sel_is_ff,
tag == 0 ? 1 : 0 },
179 { C::alu_tag_ff_diff_inv,
tag == 0 ? 0 :
FF(
tag).invert() },
181 { C::alu_tag_u128_diff_inv,
185 { C::execution_mem_tag_reg_0_,
tag },
186 { C::execution_mem_tag_reg_1_,
tag },
187 { C::execution_mem_tag_reg_2_,
tag },
188 { C::execution_register_0_,
a },
189 { C::execution_register_1_,
b },
190 { C::execution_register_2_, c },
191 { C::execution_sel_exec_dispatch_alu, 1 },
201 TestTraceContainer process_basic_add_with_tracegen(ThreeOperandTestParams params,
bool error =
false)
203 TestTraceContainer
trace;
204 auto [
a,
b, c] = params;
217 TestTraceContainer process_carry_add_trace(ThreeOperandTestParams params)
219 auto [
a,
b, c] = params;
220 auto mem_tag =
a.get_tag();
223 auto tag =
static_cast<uint8_t
>(mem_tag);
224 auto trace = TestTraceContainer({
228 { C::alu_ia_tag,
tag },
230 { C::alu_ib_tag,
tag },
232 { C::alu_ic_tag,
tag },
237 { C::alu_sel_op_add, 1 },
238 { C::alu_sel_is_ff,
tag == 0 ? 1 : 0 },
239 { C::alu_tag_ff_diff_inv,
tag == 0 ? 0 :
FF(
tag).invert() },
241 { C::alu_tag_u128_diff_inv,
245 { C::execution_mem_tag_reg_0_,
tag },
246 { C::execution_mem_tag_reg_1_,
tag },
247 { C::execution_mem_tag_reg_2_,
tag },
248 { C::execution_register_0_,
a },
249 { C::execution_register_1_,
b },
250 { C::execution_register_2_, c },
251 { C::execution_sel_exec_dispatch_alu, 1 },
261 TestTraceContainer process_carry_add_with_tracegen(ThreeOperandTestParams params)
263 TestTraceContainer
trace;
264 auto [
a,
b, c] = params;
265 auto mem_tag =
a.get_tag();
283TEST_P(AluAddConstrainingTest, AluBasicAdd)
285 auto trace = process_basic_add_trace(GetParam());
286 check_all_interactions<AluTraceBuilder>(trace);
287 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
288 check_relation<alu>(trace);
291TEST_P(AluAddConstrainingTest, AluBasicAddTraceGen)
293 auto trace = process_basic_add_with_tracegen(GetParam());
294 check_all_interactions<AluTraceBuilder>(trace);
295 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
296 check_relation<alu>(trace);
299TEST_P(AluAddConstrainingTest, AluCarryAdd)
301 auto trace = process_carry_add_trace(GetParam());
302 check_all_interactions<AluTraceBuilder>(trace);
303 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
304 check_relation<alu>(trace);
307TEST_P(AluAddConstrainingTest, AluCarryAddTraceGen)
309 auto trace = process_carry_add_with_tracegen(GetParam());
310 check_all_interactions<AluTraceBuilder>(trace);
311 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
312 check_relation<alu>(trace);
315TEST_P(AluAddConstrainingTest, NegativeBasicAdd)
317 auto trace = process_basic_add_trace(GetParam());
318 check_relation<alu>(trace);
319 trace.set(Column::alu_ic, 0,
trace.get(Column::alu_ic, 0) + 1);
323TEST_P(AluAddConstrainingTest, NegativeAluCarryAdd)
325 auto params = GetParam();
326 auto trace = process_carry_add_trace(params);
327 auto correct_max_value =
trace.get(Column::alu_max_value, 0);
329 check_all_interactions<AluTraceBuilder>(trace);
330 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
331 check_relation<alu>(trace);
334 trace.set(Column::alu_cf, 0, 0);
338 trace.set(Column::alu_cf, 0, 1);
339 trace.set(Column::alu_max_value, 0, 0);
343 trace.set(Column::alu_max_value, 0, correct_max_value);
348 trace.set(Column::alu_ic, 0, correct_max_value + 2);
352TEST_P(AluAddConstrainingTest, NegativeAddWrongTagABMismatch)
354 auto params = GetParam();
356 auto trace = process_basic_add_trace(params);
357 trace.set(Column::alu_ib_tag, 0,
tag - 1);
359 trace.set(Column::alu_ab_tags_diff_inv, 0, 1);
360 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
363 trace.set(Column::alu_sel_tag_err, 0, 1);
366 trace.set(Column::alu_sel_err, 0, 1);
368 check_relation<alu>(trace);
370 trace.set(Column::alu_ab_tags_diff_inv, 0, 0);
372 trace.set(Column::alu_ab_tags_diff_inv, 0, 1);
374 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 0);
375 trace.set(Column::alu_sel_tag_err, 0, 0);
376 trace.set(Column::alu_sel_err, 0, 0);
380TEST_P(AluAddConstrainingTest, NegativeAddTraceGenWrongTagABMismatch)
382 auto [
a,
b, c] = GetParam();
383 auto trace = process_basic_add_with_tracegen(
385 check_all_interactions<AluTraceBuilder>(trace);
386 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
387 check_relation<alu>(trace);
390TEST_P(AluAddConstrainingTest, NegativeAddWrongTagCMismatch)
392 auto params = GetParam();
394 auto trace = process_basic_add_trace(params);
395 check_relation<alu>(trace);
396 trace.set(Column::alu_ic_tag, 0,
tag - 1);
414class AluSubConstrainingTest :
public AluConstrainingTest,
415 public ::testing::WithParamInterface<ThreeOperandTestParams> {
417 TestTraceContainer process_sub_trace(ThreeOperandTestParams params)
419 auto [
a,
b, c] = params;
420 auto tag =
static_cast<uint8_t
>(
a.get_tag());
421 auto trace = TestTraceContainer({
423 { C::alu_cf,
a.as_ff() -
b.as_ff() != c.as_ff() ? 1 : 0 },
425 { C::alu_ia_tag,
tag },
427 { C::alu_ib_tag,
tag },
429 { C::alu_ic_tag,
tag },
434 { C::alu_sel_op_sub, 1 },
435 { C::alu_sel_is_ff,
tag == 0 ? 1 : 0 },
437 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
438 { C::alu_tag_u128_diff_inv,
439 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
442 { C::execution_mem_tag_reg_0_,
tag },
443 { C::execution_mem_tag_reg_1_,
tag },
444 { C::execution_mem_tag_reg_2_,
tag },
445 { C::execution_register_0_,
a },
446 { C::execution_register_1_,
b },
447 { C::execution_register_2_, c },
448 { C::execution_sel_exec_dispatch_alu, 1 },
458 TestTraceContainer process_sub_with_tracegen(ThreeOperandTestParams params)
460 TestTraceContainer
trace;
461 auto [
a,
b, c] = params;
465 { .operation = simulation::AluOperation::SUB, .a =
a, .b =
b, .c = c },
477TEST_P(AluSubConstrainingTest, AluSub)
479 auto trace = process_sub_trace(GetParam());
480 check_all_interactions<AluTraceBuilder>(trace);
481 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
482 check_relation<alu>(trace);
485TEST_P(AluSubConstrainingTest, AluSubTraceGen)
487 auto trace = process_sub_with_tracegen(GetParam());
488 check_all_interactions<AluTraceBuilder>(trace);
489 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
490 check_relation<alu>(trace);
493TEST_P(AluSubConstrainingTest, AluSubNegative)
495 auto params = GetParam();
496 auto is_ff =
std::get<0>(params).get_tag() == MemoryTag::FF;
497 auto trace = process_sub_trace(GetParam());
498 check_all_interactions<AluTraceBuilder>(trace);
499 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
500 check_relation<alu>(trace);
502 auto c =
trace.get(Column::alu_ic, 0);
504 trace.set(Column::alu_ic, 0, c + 1);
507 trace.set(Column::alu_ic, 0, c);
508 check_relation<alu>(trace);
512 trace.set(Column::alu_cf, 0,
trace.get(Column::alu_cf, 0) == 1 ? 0 : 1);
520 MemoryValue::from_tag(MemoryTag::U1, 0),
521 MemoryValue::from_tag(MemoryTag::U8, 16),
522 MemoryValue::from_tag(MemoryTag::U16, 64456),
523 MemoryValue::from_tag(MemoryTag::U32, (
uint256_t(1) << 32) - 50),
524 MemoryValue::from_tag(MemoryTag::U64, (
uint256_t(1) << 64) - 50),
525 MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 128) - 50),
526 MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 8),
531class AluMulConstrainingTest :
public AluConstrainingTest,
532 public ::testing::WithParamInterface<ThreeOperandTestParams> {
534 TestTraceContainer process_mul_trace(ThreeOperandTestParams params)
536 auto [
a,
b, c] = params;
537 auto mem_tag =
a.get_tag();
538 auto tag =
static_cast<uint8_t
>(mem_tag);
540 auto is_u128 = mem_tag == MemoryTag::U128;
545 if (mem_tag != MemoryTag::FF && mem_tag != MemoryTag::U128) {
549 auto trace = TestTraceContainer({
551 { C::alu_c_hi, c_hi },
552 { C::alu_constant_64, 64 },
554 { C::alu_ia_tag,
tag },
556 { C::alu_ib_tag,
tag },
558 { C::alu_ic_tag,
tag },
563 { C::alu_sel_decompose_a, is_u128 ? 1 : 0 },
564 { C::alu_sel_is_ff, mem_tag == MemoryTag::FF ? 1 : 0 },
566 { C::alu_sel_is_u128, is_u128 ? 1 : 0 },
567 { C::alu_sel_mul_div_u128, is_u128 ? 1 : 0 },
568 { C::alu_sel_op_mul, 1 },
569 { C::alu_sel_mul_no_err_non_ff, mem_tag == MemoryTag::FF ? 0 : 1 },
570 { C::alu_tag_u128_diff_inv, is_u128 ? 0 :
FF(
tag -
static_cast<uint8_t
>(MemoryTag::U128)).
invert() },
571 { C::execution_mem_tag_reg_0_,
tag },
572 { C::execution_mem_tag_reg_1_,
tag },
573 { C::execution_mem_tag_reg_2_,
tag },
574 { C::execution_register_0_,
a },
575 { C::execution_register_1_,
b },
576 { C::execution_register_2_, c },
577 { C::execution_sel_exec_dispatch_alu, 1 },
588 auto a_decomp = simulation::decompose_128(
a.as<
uint128_t>());
589 auto b_decomp = simulation::decompose_128(
b.as<
uint128_t>());
591 auto hi_operand =
static_cast<uint256_t>(a_decomp.hi) *
static_cast<uint256_t>(b_decomp.hi);
592 c_hi = ((c_int >> 128) - hi_operand) % (
uint256_t(1) << 64);
594 { { { Column::alu_a_lo, a_decomp.lo },
595 { Column::alu_a_lo_bits, 64 },
596 { Column::alu_a_hi, a_decomp.hi },
597 { Column::alu_a_hi_bits, 64 },
598 { Column::alu_b_lo, b_decomp.lo },
599 { Column::alu_b_hi, b_decomp.hi },
600 { Column::alu_c_hi, c_hi },
601 { Column::alu_cf, hi_operand > (
uint256_t(1) << 64) ? 1 : 0 } } });
603 range_check_events.insert(range_check_events.end(),
604 { { .value = a_decomp.lo, .num_bits = 64 },
605 { .value = a_decomp.hi, .num_bits = 64 },
606 { .value = b_decomp.lo, .num_bits = 64 },
607 { .value = b_decomp.hi, .num_bits = 64 } });
610 range_check_events.push_back({ .value =
static_cast<uint128_t>(c_hi), .num_bits = 64 });
616 TestTraceContainer process_mul_with_tracegen(ThreeOperandTestParams params)
618 TestTraceContainer
trace;
619 auto [
a,
b, c] = params;
620 auto mem_tag =
a.get_tag();
624 { .operation = simulation::AluOperation::MUL, .a =
a, .b =
b, .c = c },
630 auto c_hi = mem_tag == MemoryTag::FF ? 0 : (a_int * b_int) >>
get_tag_bits(mem_tag);
631 if (mem_tag == MemoryTag::U128) {
632 auto a_decomp = simulation::decompose_128(
a.as<
uint128_t>());
633 auto b_decomp = simulation::decompose_128(
b.as<
uint128_t>());
635 auto c_hi_full = (a_int * b_int) >> 128;
636 auto hi_operand =
static_cast<uint256_t>(a_decomp.hi) *
static_cast<uint256_t>(b_decomp.hi);
637 c_hi = (c_hi_full - hi_operand) % (
uint256_t(1) << 64);
639 { .value = a_decomp.hi, .num_bits = 64 },
640 { .value = b_decomp.lo, .num_bits = 64 },
641 { .value = b_decomp.hi, .num_bits = 64 },
642 { .value =
static_cast<uint128_t>(c_hi), .num_bits = 64 } },
655TEST_P(AluMulConstrainingTest, AluMul)
657 auto trace = process_mul_trace(GetParam());
658 check_all_interactions<AluTraceBuilder>(trace);
659 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
660 check_relation<alu>(trace);
663TEST_P(AluMulConstrainingTest, AluMulTraceGen)
665 auto trace = process_mul_with_tracegen(GetParam());
666 check_all_interactions<AluTraceBuilder>(trace);
667 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
668 check_relation<alu>(trace);
671TEST_F(AluConstrainingTest, AluMulU128Carry)
674 auto b = MemoryValue::from_tag(MemoryTag::U128,
get_tag_max_value(MemoryTag::U128) - 2);
678 auto tag =
static_cast<uint8_t
>(MemoryTag::U128);
680 auto a_decomp = simulation::decompose_128(
a.as<
uint128_t>());
681 auto b_decomp = simulation::decompose_128(
b.as<
uint128_t>());
685 ((overflow_c_int >> 128) -
static_cast<uint256_t>(a_decomp.hi) *
static_cast<uint256_t>(b_decomp.hi));
686 auto c_hi = hi_operand % (
uint256_t(1) << 64);
687 auto cf = hi_operand >> 64;
688 auto trace = TestTraceContainer({
690 { C::alu_a_hi, a_decomp.hi },
691 { C::alu_a_hi_bits, 64 },
692 { C::alu_a_lo, a_decomp.lo },
693 { C::alu_a_lo_bits, 64 },
694 { C::alu_b_hi, b_decomp.hi },
695 { C::alu_b_lo, b_decomp.lo },
696 { C::alu_c_hi, c_hi },
698 { C::alu_constant_64, 64 },
700 { C::alu_ia_tag,
tag },
702 { C::alu_ib_tag,
tag },
704 { C::alu_ic_tag,
tag },
709 { C::alu_sel_decompose_a, 1 },
710 { C::alu_sel_is_u128, 1 },
711 { C::alu_sel_mul_div_u128, 1 },
712 { C::alu_sel_op_mul, 1 },
713 { C::alu_sel_mul_no_err_non_ff, 1 },
714 { C::alu_tag_u128_diff_inv, 0 },
716 { C::execution_mem_tag_reg_0_,
tag },
717 { C::execution_mem_tag_reg_1_,
tag },
718 { C::execution_mem_tag_reg_2_,
tag },
719 { C::execution_register_0_,
a },
720 { C::execution_register_1_,
b },
721 { C::execution_register_2_, c },
722 { C::execution_sel_exec_dispatch_alu, 1 },
730 { .value = a_decomp.hi, .num_bits = 64 },
731 { .value = b_decomp.lo, .num_bits = 64 },
732 { .value = b_decomp.hi, .num_bits = 64 },
733 { .value =
static_cast<uint128_t>(c_hi), .num_bits = 64 } },
736 check_all_interactions<AluTraceBuilder>(trace);
737 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
738 check_relation<alu>(trace);
741 auto should_fail_overflowed = MemoryValue::from_tag_truncating(MemoryTag::U128,
a.as_ff() *
b.as_ff());
742 trace.set(Column::alu_ic, 0, should_fail_overflowed);
746TEST_P(AluMulConstrainingTest, NegativeAluMul)
748 auto trace = process_mul_trace(GetParam());
749 check_all_interactions<AluTraceBuilder>(trace);
750 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
751 check_relation<alu>(trace);
752 trace.set(Column::alu_ic, 0,
trace.get(Column::alu_ic, 0) + 1);
759 MemoryValue::from_tag(MemoryTag::U1, 0),
760 MemoryValue::from_tag(MemoryTag::U8, 4),
761 MemoryValue::from_tag(MemoryTag::U16, 0),
762 MemoryValue::from_tag(MemoryTag::U32, 0x33333331),
763 MemoryValue::from_tag(MemoryTag::U64, 0x3333333333333331ULL),
764 MemoryValue::from_tag(MemoryTag::U128, (((
uint256_t(1) << 128) - 11) / 5)),
769class AluDivConstrainingTest :
public AluConstrainingTest,
770 public ::testing::WithParamInterface<ThreeOperandTestParams> {
772 TestTraceContainer process_div_trace(ThreeOperandTestParams params)
774 auto [
a,
b, c] = params;
775 auto mem_tag =
a.get_tag();
776 auto tag =
static_cast<uint8_t
>(mem_tag);
777 auto remainder =
a -
b * c;
779 auto div_0_error =
b.as_ff() ==
FF(0);
780 auto is_u128 = mem_tag == MemoryTag::U128;
782 auto trace = TestTraceContainer({
784 { C::alu_b_inv, div_0_error ? 0 :
b.as_ff().invert() },
785 { C::alu_constant_64, 64 },
786 { C::alu_helper1, div_0_error ? 0 : remainder.as_ff() },
788 { C::alu_ia_tag,
tag },
790 { C::alu_ib_tag,
tag },
792 { C::alu_ic_tag,
tag },
797 { C::alu_sel_decompose_a, is_u128 ? 1 : 0 },
798 { C::alu_sel_div_0_err, div_0_error ? 1 : 0 },
799 { C::alu_sel_div_no_err, div_0_error ? 0 : 1 },
800 { C::alu_sel_int_gt, div_0_error ? 0 : 1 },
801 { C::alu_gt_input_a,
b.as_ff() },
802 { C::alu_gt_input_b, div_0_error ? 0 : remainder.as_ff() },
803 { C::alu_gt_result_c, div_0_error ? 0 : 1 },
804 { C::alu_sel_err, div_0_error ? 1 : 0 },
805 { C::alu_sel_is_u128, is_u128 ? 1 : 0 },
806 { C::alu_sel_mul_div_u128, is_u128 ? 1 : 0 },
807 { C::alu_sel_op_div, 1 },
808 { C::alu_tag_ff_diff_inv,
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
809 { C::alu_tag_u128_diff_inv, is_u128 ? 0 :
FF(
tag -
static_cast<uint8_t
>(MemoryTag::U128)).
invert() },
810 { C::execution_mem_tag_reg_0_,
tag },
811 { C::execution_mem_tag_reg_1_,
tag },
812 { C::execution_mem_tag_reg_2_,
tag },
813 { C::execution_register_0_,
a },
814 { C::execution_register_1_,
b },
815 { C::execution_register_2_, c },
816 { C::execution_sel_exec_dispatch_alu, 1 },
817 { C::execution_sel_opcode_error, div_0_error ? 1 : 0 },
825 .
b =
static_cast<uint128_t>(remainder.as_ff()),
830 auto c_decomp = simulation::decompose_128(c.as<
uint128_t>());
831 auto b_decomp = simulation::decompose_128(
b.as<
uint128_t>());
834 { { { Column::alu_a_lo, c_decomp.lo },
835 { Column::alu_a_lo_bits, 64 },
836 { Column::alu_a_hi, c_decomp.hi },
837 { Column::alu_a_hi_bits, 64 },
838 { Column::alu_b_lo, b_decomp.lo },
839 { Column::alu_b_hi, b_decomp.hi } } });
846 { .value = c_decomp.lo, .num_bits = 64 },
847 { .value = c_decomp.hi, .num_bits = 64 },
848 { .value = b_decomp.lo, .num_bits = 64 },
849 { .value = b_decomp.hi, .num_bits = 64 } },
853 { .value = c_decomp.hi, .num_bits = 64 },
854 { .value = b_decomp.lo, .num_bits = 64 },
855 { .value = b_decomp.hi, .num_bits = 64 } },
858 }
else if (!div_0_error) {
867 TestTraceContainer process_div_with_tracegen(ThreeOperandTestParams params)
869 TestTraceContainer
trace;
870 auto [
a,
b, c] = params;
871 bool div_0_error =
b.as_ff() ==
FF(0);
872 auto mem_tag =
a.get_tag();
874 MemoryValue remainder = MemoryValue::from_tag(MemoryTag::FF, 0);
875 if (!div_0_error && mem_tag ==
b.get_tag() && mem_tag != MemoryTag::FF) {
876 remainder =
a -
b * c;
881 { .operation = simulation::AluOperation::DIV, .a =
a, .b =
b, .c = c, .error = div_0_error },
885 if (mem_tag == MemoryTag::U128) {
886 auto c_decomp = simulation::decompose_128(
static_cast<uint128_t>(c.as_ff()));
887 auto b_decomp = simulation::decompose_128(
static_cast<uint128_t>(
b.as_ff()));
894 { .value = c_decomp.lo, .num_bits = 64 },
895 { .value = c_decomp.hi, .num_bits = 64 },
896 { .value = b_decomp.lo, .num_bits = 64 },
897 { .value = b_decomp.hi, .num_bits = 64 } },
901 { .value = c_decomp.hi, .num_bits = 64 },
902 { .value = b_decomp.lo, .num_bits = 64 },
903 { .value = b_decomp.hi, .num_bits = 64 } },
906 }
else if (!div_0_error) {
914 .
b =
static_cast<uint128_t>(remainder.as_ff()),
923TEST_P(AluDivConstrainingTest, AluDiv)
925 auto trace = process_div_trace(GetParam());
926 check_all_interactions<AluTraceBuilder>(trace);
927 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
928 check_relation<alu>(trace);
931TEST_P(AluDivConstrainingTest, AluDivTraceGen)
933 auto trace = process_div_with_tracegen(GetParam());
934 check_all_interactions<AluTraceBuilder>(trace);
935 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
936 check_relation<alu>(trace);
939TEST_F(AluDivConstrainingTest, AluDivByZeroMismatchTagsTraceGen)
941 auto a = MemoryValue::from_tag(MemoryTag::U128, 2);
942 auto b = MemoryValue::from_tag(MemoryTag::U64, 0);
943 auto c = MemoryValue::from_tag(MemoryTag::FF, 0);
945 auto trace = process_div_with_tracegen({
a,
b, c });
946 check_relation<alu>(trace);
947 check_all_interactions<AluTraceBuilder>(trace);
950TEST_F(AluDivConstrainingTest, AluDivByZeroTagFFAndMismatchTagsTraceGen)
952 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
953 auto b = MemoryValue::from_tag(MemoryTag::U32, 0);
954 auto c = MemoryValue::from_tag(MemoryTag::FF, 0);
956 auto trace = process_div_with_tracegen({
a,
b, c });
957 check_relation<alu>(trace);
958 check_all_interactions<AluTraceBuilder>(trace);
961TEST_F(AluDivConstrainingTest, NegativeAluDivUnderflow)
964 auto a = MemoryValue::from_tag(MemoryTag::U32, 2);
965 auto b = MemoryValue::from_tag(MemoryTag::U32, 5);
967 auto trace = process_div_trace({
a,
b, c });
968 check_all_interactions<AluTraceBuilder>(trace);
969 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
970 check_relation<alu>(trace);
975 c = MemoryValue::from_tag(MemoryTag::U32, 2);
976 auto wrong_remainder =
a.as_ff() -
b.as_ff() * c.as_ff();
978 trace.set(Column::alu_ic, 0, c);
979 trace.set(Column::alu_helper1, 0, wrong_remainder);
980 trace.set(Column::alu_gt_input_b, 0, wrong_remainder);
983 check_relation<alu>(trace);
986 (check_interaction<AluTraceBuilder, lookup_alu_range_check_div_remainder_settings>(trace)),
987 "RANGE_CHECK_DIV_REMAINDER");
990 "LOOKUP_ALU_INT_GT");
993TEST_F(AluDivConstrainingTest, NegativeAluDivU128Carry)
996 auto a = MemoryValue::from_tag(MemoryTag::U128, 2);
997 auto b = MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 64) + 2);
1000 auto trace = process_div_trace({
a,
b, c });
1002 check_all_interactions<AluTraceBuilder>(trace);
1003 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1004 check_relation<alu>(trace);
1008 c = MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 64) + 3);
1009 auto wrong_remainder =
a.as_ff() -
FF(
static_cast<uint256_t>(
b.as_ff()) *
static_cast<uint256_t>(c.as_ff()));
1013 trace.set(Column::alu_ic, 0, c);
1014 trace.set(Column::alu_helper1, 0, wrong_remainder);
1019 auto c_decomp = simulation::decompose_128(c.as<
uint128_t>());
1020 trace.set(Column::alu_a_lo, 0, c_decomp.lo);
1021 trace.set(Column::alu_a_hi, 0, c_decomp.hi);
1027TEST_F(AluDivConstrainingTest, NegativeAluDivByZero)
1029 auto a = MemoryValue::from_tag(MemoryTag::U32, 2);
1030 auto b = MemoryValue::from_tag(MemoryTag::U32, 5);
1033 for (
const bool with_tracegen : {
false,
true }) {
1034 auto trace = with_tracegen ? process_div_with_tracegen({
a,
b, c }) : process_div_trace({
a,
b, c });
1035 check_all_interactions<AluTraceBuilder>(trace);
1036 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1037 check_relation<alu>(trace);
1040 trace.set(Column::alu_ib, 0, 0);
1041 trace.set(Column::alu_b_inv, 0, 0);
1045 trace.set(Column::alu_sel_div_0_err, 0, 1);
1046 trace.set(Column::alu_sel_div_no_err, 0, 0);
1049 trace.set(Column::alu_sel_err, 0, 1);
1050 trace.set(Column::alu_sel_int_gt, 0, 0);
1051 trace.set(Column::alu_gt_input_a, 0, 0);
1052 trace.set(Column::alu_gt_input_b, 0, 0);
1053 trace.set(Column::alu_gt_result_c, 0, 0);
1054 check_relation<alu>(trace);
1057 trace.set(Column::alu_sel_op_div, 0, 0);
1058 trace.set(Column::alu_sel_op_mul, 0, 1);
1062 trace.set(Column::alu_sel_op_div, 0, 1);
1063 trace.set(Column::alu_sel_op_mul, 0, 0);
1065 check_relation<alu>(trace);
1068 trace.set(Column::alu_ib, 0,
b);
1074TEST_F(AluDivConstrainingTest, NegativeAluDivFF)
1076 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1077 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1079 auto trace = process_div_with_tracegen({
a,
b, c });
1082 trace.set(Column::alu_sel_tag_err, 0, 1);
1083 trace.set(Column::alu_sel_err, 0, 1);
1084 trace.set(Column::alu_sel_div_no_err, 0, 0);
1085 trace.set(Column::alu_sel_int_gt, 0, 0);
1086 trace.set(Column::alu_gt_input_a, 0, 0);
1087 trace.set(Column::alu_gt_input_b, 0, 0);
1088 trace.set(Column::alu_gt_result_c, 0, 0);
1089 check_relation<alu>(trace);
1090 check_all_interactions<AluTraceBuilder>(trace);
1091 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1094TEST_F(AluDivConstrainingTest, NegativeAluDivByZeroFF)
1097 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1098 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1100 auto trace = process_div_with_tracegen({
a,
b, c });
1101 trace.set(Column::alu_sel_tag_err, 0, 1);
1102 trace.set(Column::alu_sel_err, 0, 1);
1103 trace.set(Column::alu_sel_div_no_err, 0, 0);
1104 trace.set(Column::alu_sel_int_gt, 0, 0);
1105 trace.set(Column::alu_gt_input_a, 0, 0);
1106 trace.set(Column::alu_gt_input_b, 0, 0);
1107 trace.set(Column::alu_gt_result_c, 0, 0);
1108 check_relation<alu>(trace);
1110 trace.set(Column::alu_ib, 0, 0);
1111 trace.set(Column::alu_b_inv, 0, 0);
1113 trace.set(Column::alu_sel_div_0_err, 0, 1);
1114 check_relation<alu>(trace);
1115 check_all_interactions<AluTraceBuilder>(trace);
1116 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1119TEST_F(AluDivConstrainingTest, NegativeAluDivByZeroFFTagMismatch)
1122 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1123 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1125 auto trace = process_div_with_tracegen({
a,
b, c });
1126 trace.set(Column::alu_sel_tag_err, 0, 1);
1127 trace.set(Column::alu_sel_err, 0, 1);
1128 trace.set(Column::alu_sel_div_no_err, 0, 0);
1129 trace.set(Column::alu_sel_int_gt, 0, 0);
1130 trace.set(Column::alu_gt_input_a, 0, 0);
1131 trace.set(Column::alu_gt_input_b, 0, 0);
1132 trace.set(Column::alu_gt_result_c, 0, 0);
1133 check_relation<alu>(trace);
1135 trace.set(Column::alu_ib_tag, 0,
static_cast<uint8_t
>(MemoryTag::U8));
1137 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
1138 trace.set(Column::alu_ab_tags_diff_inv,
1140 (
FF(
static_cast<uint8_t
>(MemoryTag::FF)) -
FF(
static_cast<uint8_t
>(MemoryTag::U8))).invert());
1141 check_relation<alu>(trace);
1143 trace.set(Column::alu_ib, 0, 0);
1144 trace.set(Column::alu_b_inv, 0, 0);
1146 trace.set(Column::alu_sel_div_0_err, 0, 1);
1147 trace.set(Column::alu_sel_div_no_err, 0, 0);
1148 check_relation<alu>(trace);
1149 check_all_interactions<AluTraceBuilder>(trace);
1150 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1158 MemoryValue::from_tag(MemoryTag::FF, 0),
1159 MemoryValue::from_tag(MemoryTag::FF, 4),
1160 MemoryValue::from_tag(MemoryTag::FF,
FF(
"0x1e980ebbc51694827ee20074ac28b250a037a43eb44b38e6aa367c57a05e6d48")),
1161 MemoryValue::from_tag(MemoryTag::FF,
FF(
"0x135b52945a13d9aa49b9b57c33cd568ba9ae5ce9ca4a2d06e7f3fbd4f9999998")),
1162 MemoryValue::from_tag(MemoryTag::FF,
FF(
"0x135b52945a13d9aa49b9b57c33cd568ba9ae5ce9ca4a2d071b272f07f9999998")),
1163 MemoryValue::from_tag(MemoryTag::FF,
FF(
"0x135b52945a13d9aa49b9b57c33cd568bdce1901cfd7d603a1b272f07f9999998")),
1164 MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 2),
1169class AluFDivConstrainingTest :
public AluConstrainingTest,
1170 public ::testing::WithParamInterface<ThreeOperandTestParams> {
1172 TestTraceContainer process_fdiv_trace(ThreeOperandTestParams params)
1174 auto [
a,
b, c] = params;
1175 a = MemoryValue::from_tag(MemoryTag::FF,
a);
1176 b = MemoryValue::from_tag(MemoryTag::FF,
b);
1177 c = MemoryValue::from_tag(MemoryTag::FF, c);
1178 auto div_0_error =
b.as_ff() ==
FF(0);
1180 auto mem_tag =
a.get_tag();
1181 auto tag =
static_cast<uint8_t
>(mem_tag);
1183 auto trace = TestTraceContainer({
1185 { C::alu_b_inv, div_0_error ? 0 :
b.as_ff().invert() },
1187 { C::alu_ia_tag,
tag },
1189 { C::alu_ib_tag,
tag },
1191 { C::alu_ic_tag,
tag },
1196 { C::alu_sel_div_0_err, div_0_error ? 1 : 0 },
1197 { C::alu_sel_err, div_0_error ? 1 : 0 },
1198 { C::alu_sel_is_ff, 1 },
1199 { C::alu_sel_op_fdiv, 1 },
1200 { C::alu_tag_u128_diff_inv,
FF(-
static_cast<uint8_t
>(MemoryTag::U128)).
invert() },
1201 { C::execution_mem_tag_reg_0_,
tag },
1202 { C::execution_mem_tag_reg_1_,
tag },
1203 { C::execution_mem_tag_reg_2_,
tag },
1204 { C::execution_register_0_,
a },
1205 { C::execution_register_1_,
b },
1206 { C::execution_register_2_, c },
1207 { C::execution_sel_exec_dispatch_alu, 1 },
1208 { C::execution_sel_opcode_error, div_0_error ? 1 : 0 },
1219 TestTraceContainer process_fdiv_with_tracegen(ThreeOperandTestParams params,
bool upcast_to_ff =
false)
1221 TestTraceContainer
trace;
1222 auto [
a,
b, c] = params;
1225 a = MemoryValue::from_tag(MemoryTag::FF,
a);
1226 b = MemoryValue::from_tag(MemoryTag::FF,
b);
1227 c = MemoryValue::from_tag(MemoryTag::FF, c);
1230 bool div_0_error =
b.as_ff() ==
FF(0);
1234 { .operation = simulation::AluOperation::FDIV, .a =
a, .b =
b, .c = c, .error = div_0_error },
1247TEST_P(AluFDivConstrainingTest, AluFDiv)
1249 auto trace = process_fdiv_trace(GetParam());
1250 check_all_interactions<AluTraceBuilder>(trace);
1251 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1252 check_relation<alu>(trace);
1255TEST_P(AluFDivConstrainingTest, AluFDivTraceGen)
1257 auto trace = process_fdiv_with_tracegen(GetParam(),
true);
1260 check_all_interactions<AluTraceBuilder>(trace);
1261 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1262 check_relation<alu>(trace);
1265TEST_F(AluFDivConstrainingTest, AluFDivByZeroNonFFTagTraceGen)
1267 auto a = MemoryValue::from_tag(MemoryTag::U8, 2);
1268 auto b = MemoryValue::from_tag(MemoryTag::U8, 0);
1269 auto c = MemoryValue::from_tag(MemoryTag::FF, 0);
1271 auto trace = process_fdiv_with_tracegen({
a,
b, c });
1272 check_relation<alu>(trace);
1273 check_all_interactions<AluTraceBuilder>(trace);
1276TEST_F(AluFDivConstrainingTest, AluFDivByZeroNonFFTagMismatchTraceGen)
1278 auto a = MemoryValue::from_tag(MemoryTag::U8, 2);
1279 auto b = MemoryValue::from_tag(MemoryTag::U16, 0);
1280 auto c = MemoryValue::from_tag(MemoryTag::FF, 0);
1281 auto trace = process_fdiv_with_tracegen({
a,
b, c });
1282 check_relation<alu>(trace);
1283 check_all_interactions<AluTraceBuilder>(trace);
1286TEST_F(AluFDivConstrainingTest, NegativeAluFDivByZero)
1288 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1289 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1291 auto trace = process_fdiv_trace({
a,
b, c });
1292 check_all_interactions<AluTraceBuilder>(trace);
1293 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1294 check_relation<alu>(trace);
1297 trace.set(Column::alu_ib, 0, 0);
1298 trace.set(Column::alu_b_inv, 0, 0);
1302 trace.set(Column::alu_sel_div_0_err, 0, 1);
1305 trace.set(Column::alu_sel_err, 0, 1);
1306 check_relation<alu>(trace);
1309 trace.set(Column::alu_ib, 0,
b);
1314TEST_F(AluFDivConstrainingTest, NegativeAluFDivByZeroNonFFTagMismatch)
1316 auto a = MemoryValue::from_tag(MemoryTag::U8, 4);
1317 auto b = MemoryValue::from_tag(MemoryTag::U8, 2);
1320 auto c = MemoryValue::from_tag(MemoryTag::FF, 2);
1321 auto tag =
static_cast<uint8_t
>(MemoryTag::U8);
1323 auto trace = TestTraceContainer({
1325 { C::alu_b_inv,
b.as_ff().invert() },
1327 { C::alu_ia_tag,
tag },
1329 { C::alu_ib_tag,
tag },
1331 { C::alu_ic_tag,
static_cast<uint8_t
>(MemoryTag::FF) },
1336 { C::alu_sel_op_fdiv, 1 },
1337 { C::alu_sel_tag_err, 1 },
1338 { C::alu_sel_err, 1 },
1339 { C::alu_tag_ff_diff_inv, (
FF(
tag) -
FF(
static_cast<uint8_t
>(MemoryTag::FF))).invert() },
1340 { C::alu_tag_u128_diff_inv, (
FF(
tag) -
FF(
static_cast<uint8_t
>(MemoryTag::U128))).invert() },
1341 { C::execution_mem_tag_reg_0_,
tag },
1342 { C::execution_mem_tag_reg_1_,
tag },
1343 { C::execution_mem_tag_reg_2_,
static_cast<uint8_t
>(MemoryTag::FF) },
1344 { C::execution_register_0_,
a },
1345 { C::execution_register_1_,
b },
1346 { C::execution_register_2_, c },
1347 { C::execution_sel_exec_dispatch_alu, 1 },
1349 { C::execution_sel_opcode_error, 1 },
1356 check_relation<alu>(trace);
1357 check_all_interactions<AluTraceBuilder>(trace);
1358 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1361 trace.set(Column::alu_sel_tag_err, 0, 0);
1362 trace.set(Column::alu_sel_err, 0, 0);
1366 trace.set(Column::alu_tag_ff_diff_inv, 0, 0);
1367 trace.set(Column::alu_sel_is_ff, 0, 1);
1371 trace.set(Column::alu_tag_ff_diff_inv, 0, (
FF(
tag) -
FF(
static_cast<uint8_t
>(MemoryTag::FF))).invert());
1372 trace.set(Column::alu_sel_is_ff, 0, 0);
1373 trace.set(Column::alu_sel_tag_err, 0, 1);
1374 trace.set(Column::alu_sel_err, 0, 1);
1375 check_relation<alu>(trace);
1378 trace.set(Column::alu_ib, 0, 0);
1379 trace.set(Column::alu_b_inv, 0, 0);
1381 trace.set(Column::alu_sel_div_0_err, 0, 1);
1382 check_relation<alu>(trace);
1383 check_all_interactions<AluTraceBuilder>(trace);
1384 trace.set(Column::execution_register_1_, 0, 0);
1385 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1388 trace.set(Column::alu_ib_tag, 0,
static_cast<uint8_t
>(MemoryTag::U16));
1390 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
1391 trace.set(Column::alu_ab_tags_diff_inv, 0, (
FF(
tag) -
FF(
static_cast<uint8_t
>(MemoryTag::U16))).invert());
1392 check_relation<alu>(trace);
1401class AluEQConstrainingTest :
public AluConstrainingTest,
public ::testing::WithParamInterface<ThreeOperandTestParams> {
1403 TestTraceContainer process_eq_with_tracegen(
const ThreeOperandTestParams& params)
1405 TestTraceContainer
trace;
1406 auto [
a,
b, c] = params;
1410 { .operation = simulation::AluOperation::EQ, .a =
a, .b =
b, .c = c },
1422TEST_P(AluEQConstrainingTest, AluEQTraceGen)
1426 process_eq_with_tracegen(ThreeOperandTestParams{ param, param, MemoryValue::from_tag(MemoryTag::U1, 1) });
1427 check_all_interactions<AluTraceBuilder>(trace);
1428 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1429 check_relation<alu>(trace);
1432TEST_P(AluEQConstrainingTest, AluInEQTraceGen)
1434 auto trace = process_eq_with_tracegen(GetParam());
1435 check_all_interactions<AluTraceBuilder>(trace);
1436 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1437 check_relation<alu>(trace);
1440TEST_P(AluEQConstrainingTest, NegativeAluEqResult)
1442 auto params = GetParam();
1443 for (
const bool is_eq : {
false,
true }) {
1444 auto trace = process_eq_with_tracegen(is_eq ? ThreeOperandTestParams{
std::get<0>(params),
1446 MemoryValue::from_tag(MemoryTag::U1, 1) }
1448 check_relation<alu>(trace);
1449 check_all_interactions<AluTraceBuilder>(trace);
1450 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1451 bool c =
trace.get(Column::alu_ic, 0) == 1;
1453 trace.set(Column::alu_ic, 0,
static_cast<uint8_t
>(!c));
1458TEST_P(AluEQConstrainingTest, NegativeAluEqHelper)
1460 auto trace = process_eq_with_tracegen(GetParam());
1461 check_relation<alu>(trace);
1462 check_all_interactions<AluTraceBuilder>(trace);
1463 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1464 auto ab_diff_inv =
trace.get(Column::alu_ab_diff_inv, 0);
1465 trace.set(Column::alu_ab_diff_inv, 0, ab_diff_inv + 1);
1472 MemoryValue::from_tag(MemoryTag::U1, 0), MemoryValue::from_tag(MemoryTag::U1, 0),
1473 MemoryValue::from_tag(MemoryTag::U1, 0), MemoryValue::from_tag(MemoryTag::U1, 1),
1474 MemoryValue::from_tag(MemoryTag::U1, 0), MemoryValue::from_tag(MemoryTag::U1, 0),
1475 MemoryValue::from_tag(MemoryTag::U1, 0),
1480class AluLTConstrainingTest :
public AluConstrainingTest,
public ::testing::WithParamInterface<ThreeOperandTestParams> {
1482 TestTraceContainer process_lt_trace(ThreeOperandTestParams params)
1484 auto [
a,
b, c] = params;
1485 auto mem_tag =
a.get_tag();
1486 auto tag =
static_cast<uint8_t
>(mem_tag);
1487 auto is_ff = mem_tag == MemoryTag::FF;
1489 auto trace = TestTraceContainer({
1492 { C::alu_ia_tag,
tag },
1494 { C::alu_ib_tag,
tag },
1496 { C::alu_ic_tag,
static_cast<uint8_t
>(MemoryTag::U1) },
1497 { C::alu_gt_input_a,
b },
1498 { C::alu_gt_input_b,
a },
1499 { C::alu_gt_result_c, c },
1504 { C::alu_sel_ff_gt,
static_cast<uint8_t
>(is_ff) },
1505 { C::alu_sel_int_gt,
static_cast<uint8_t
>(!is_ff) },
1506 { C::alu_sel_is_ff,
static_cast<uint8_t
>(is_ff) },
1507 { C::alu_sel_op_lt, 1 },
1508 { C::alu_tag_ff_diff_inv, is_ff ? 0 :
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
1509 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
1510 { C::alu_tag_u128_diff_inv,
1511 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
1514 { C::execution_mem_tag_reg_0_,
tag },
1515 { C::execution_mem_tag_reg_1_,
tag },
1516 { C::execution_mem_tag_reg_2_,
static_cast<uint8_t
>(MemoryTag::U1) },
1517 { C::execution_register_0_,
a },
1518 { C::execution_register_1_,
b },
1519 { C::execution_register_2_, c },
1520 { C::execution_sel_exec_dispatch_alu, 1 },
1530 .result = c.as_ff() == 1 } },
1539 TestTraceContainer process_lt_with_tracegen(ThreeOperandTestParams params)
1541 TestTraceContainer
trace;
1542 auto [
a,
b, c] = params;
1543 auto is_ff =
a.get_tag() == MemoryTag::FF;
1547 { .operation = simulation::AluOperation::LT, .a =
a, .b =
b, .c = c },
1556 .result = c.as_ff() == 1 } },
1567TEST_P(AluLTConstrainingTest, AluLT)
1569 auto trace = process_lt_trace(GetParam());
1570 check_all_interactions<AluTraceBuilder>(trace);
1571 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1572 check_relation<alu>(trace);
1575TEST_P(AluLTConstrainingTest, AluLTTraceGen)
1577 auto trace = process_lt_with_tracegen(GetParam());
1578 check_all_interactions<AluTraceBuilder>(trace);
1579 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1580 check_relation<alu>(trace);
1583TEST_P(AluLTConstrainingTest, NegativeAluLT)
1585 auto params = GetParam();
1586 auto trace = process_lt_trace(params);
1587 auto is_ff =
std::get<0>(params).get_tag() == MemoryTag::FF;
1588 check_relation<alu>(trace);
1589 check_all_interactions<AluTraceBuilder>(trace);
1590 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1591 bool c =
trace.get(Column::alu_ic, 0) == 1;
1593 trace.set(Column::alu_ic, 0,
static_cast<uint8_t
>(!c));
1595 trace.set(Column::alu_gt_result_c, 0,
static_cast<uint8_t
>(!c));
1599 "LOOKUP_ALU_FF_GT");
1602 "LOOKUP_ALU_INT_GT");
1610class AluLTEConstrainingTest :
public AluConstrainingTest,
1611 public ::testing::WithParamInterface<ThreeOperandTestParams> {
1613 TestTraceContainer process_lte_trace(ThreeOperandTestParams params,
bool eq =
false)
1615 auto [
a,
b, c] = params;
1616 auto mem_tag =
a.get_tag();
1617 auto tag =
static_cast<uint8_t
>(mem_tag);
1618 auto is_ff = mem_tag == MemoryTag::FF;
1620 c = eq ? MemoryValue::from_tag(MemoryTag::U1, 1) : c;
1622 auto trace = TestTraceContainer({
1625 { C::alu_ia_tag,
tag },
1627 { C::alu_ib_tag,
tag },
1629 { C::alu_ic_tag,
static_cast<uint8_t
>(MemoryTag::U1) },
1630 { C::alu_gt_input_a,
a },
1631 { C::alu_gt_input_b,
b },
1632 { C::alu_gt_result_c, c.as_ff() == 0 ? 1 : 0 },
1637 { C::alu_sel_ff_gt,
static_cast<uint8_t
>(is_ff) },
1638 { C::alu_sel_int_gt,
static_cast<uint8_t
>(!is_ff) },
1639 { C::alu_sel_is_ff,
static_cast<uint8_t
>(is_ff) },
1640 { C::alu_sel_op_lte, 1 },
1641 { C::alu_tag_ff_diff_inv, is_ff ? 0 :
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
1642 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
1643 { C::alu_tag_u128_diff_inv,
1644 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
1647 { C::execution_mem_tag_reg_0_,
tag },
1648 { C::execution_mem_tag_reg_1_,
tag },
1649 { C::execution_mem_tag_reg_2_,
static_cast<uint8_t
>(MemoryTag::U1) },
1650 { C::execution_register_0_,
a },
1651 { C::execution_register_1_,
b },
1652 { C::execution_register_2_, c },
1653 { C::execution_sel_exec_dispatch_alu, 1 },
1663 .result = c.as_ff() == 0 } },
1671 TestTraceContainer process_lte_with_tracegen(ThreeOperandTestParams params,
bool eq =
false)
1673 TestTraceContainer
trace;
1674 auto [
a,
b, c] = params;
1675 auto is_ff =
a.get_tag() == MemoryTag::FF;
1677 c = eq ? MemoryValue::from_tag(MemoryTag::U1, 1) : c;
1681 { .operation = simulation::AluOperation::LTE, .a =
a, .b =
b, .c = c },
1690 .result = c.as_ff() == 0 } },
1701TEST_P(AluLTEConstrainingTest, AluLTE)
1703 auto trace = process_lte_trace(GetParam());
1704 check_all_interactions<AluTraceBuilder>(trace);
1705 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1706 check_relation<alu>(trace);
1709TEST_P(AluLTEConstrainingTest, AluLTEEq)
1711 auto trace = process_lte_trace(GetParam(),
true);
1712 check_all_interactions<AluTraceBuilder>(trace);
1713 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1714 check_relation<alu>(trace);
1717TEST_P(AluLTEConstrainingTest, AluLTETraceGen)
1719 auto trace = process_lte_with_tracegen(GetParam());
1720 check_all_interactions<AluTraceBuilder>(trace);
1721 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1722 check_relation<alu>(trace);
1725TEST_P(AluLTEConstrainingTest, AluLTEEqTraceGen)
1727 auto trace = process_lte_with_tracegen(GetParam(),
true);
1728 check_all_interactions<AluTraceBuilder>(trace);
1729 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1730 check_relation<alu>(trace);
1733TEST_P(AluLTEConstrainingTest, NegativeAluLTEResult)
1735 auto params = GetParam();
1737 for (
const bool is_eq : {
false,
true }) {
1738 auto trace = process_lte_trace(params, is_eq);
1739 auto is_ff =
std::get<0>(params).get_tag() == MemoryTag::FF;
1740 check_relation<alu>(trace);
1741 check_all_interactions<AluTraceBuilder>(trace);
1742 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1743 bool c =
trace.get(Column::alu_ic, 0) == 1;
1745 trace.set(Column::alu_ic, 0,
static_cast<uint8_t
>(!c));
1747 trace.set(Column::alu_gt_result_c, 0,
static_cast<uint8_t
>(c));
1751 "LOOKUP_ALU_FF_GT");
1754 "LOOKUP_ALU_INT_GT");
1759TEST_P(AluLTEConstrainingTest, NegativeAluLTEInput)
1761 auto params = GetParam();
1763 for (
const bool is_eq : {
false,
true }) {
1764 auto trace = process_lte_trace(params, is_eq);
1765 auto is_ff =
std::get<0>(params).get_tag() == MemoryTag::FF;
1766 check_relation<alu>(trace);
1767 check_all_interactions<AluTraceBuilder>(trace);
1768 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1769 bool c =
trace.get(Column::alu_ic, 0) == 1;
1770 auto a =
trace.get(Column::alu_ia, 0);
1771 auto wrong_b = c ?
a - 1 :
a + 1;
1772 trace.set(Column::alu_ib, 0, wrong_b);
1773 trace.set(Column::alu_gt_input_b, 0, wrong_b);
1775 check_relation<alu>(trace);
1781 "LOOKUP_ALU_FF_GT");
1784 "LOOKUP_ALU_INT_GT");
1792 MemoryValue::from_tag(MemoryTag::U1, 0),
1793 MemoryValue::from_tag(MemoryTag::U8, 55),
1794 MemoryValue::from_tag(MemoryTag::U16, 65505),
1795 MemoryValue::from_tag(MemoryTag::U32, 9),
1796 MemoryValue::from_tag(MemoryTag::U64, 9),
1797 MemoryValue::from_tag(MemoryTag::U128, 9),
1798 MemoryValue::from_tag(
static_cast<MemoryTag>(0), 0),
1803class AluNotConstrainingTest :
public AluConstrainingTest,
public ::testing::WithParamInterface<TwoOperandTestParams> {
1805 TestTraceContainer process_not_with_tracegen(
const TwoOperandTestParams& params,
bool error =
false)
1807 TestTraceContainer
trace;
1808 auto [
a,
b] = params;
1809 auto is_ff =
a.get_tag() == MemoryTag::FF;
1813 { .operation = simulation::AluOperation::NOT, .a =
a, .b =
b, .error = error || is_ff },
1825TEST_P(AluNotConstrainingTest, AluNotTraceGen)
1827 auto trace = process_not_with_tracegen(GetParam());
1828 check_all_interactions<AluTraceBuilder>(trace);
1829 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1830 check_relation<alu>(trace);
1833TEST_P(AluNotConstrainingTest, NegativeAluNotTraceGen)
1835 auto params = GetParam();
1836 auto trace = process_not_with_tracegen(params);
1837 check_all_interactions<AluTraceBuilder>(trace);
1838 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1839 check_relation<alu>(trace);
1840 trace.set(Column::alu_ib, 0,
trace.get(Column::alu_ib, 0) + 1);
1842 if (
std::get<0>(params).get_tag() != MemoryTag::FF) {
1848TEST_P(AluNotConstrainingTest, AluNotTraceGenTagError)
1850 auto [
a,
b] = GetParam();
1851 auto trace = process_not_with_tracegen(
1852 TwoOperandTestParams{
a, MemoryValue::from_tag(TAG_ERROR_TEST_VALUES.at(
b.get_tag()),
b.as_ff()) },
true);
1853 check_all_interactions<AluTraceBuilder>(trace);
1854 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1859TEST_F(AluNotConstrainingTest, AluNotTraceGenTagErrorFF)
1861 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1862 auto b = MemoryValue::from_tag(MemoryTag::FF, 253);
1863 auto trace = process_not_with_tracegen(TwoOperandTestParams{
a,
b },
true);
1864 check_all_interactions<AluTraceBuilder>(trace);
1865 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1866 check_relation<alu>(trace);
1872 MemoryValue::from_tag(MemoryTag::U1, 1),
1873 MemoryValue::from_tag(MemoryTag::U8, 0),
1874 MemoryValue::from_tag(MemoryTag::U16, 0),
1875 MemoryValue::from_tag(MemoryTag::U32, 0xfffffec0),
1876 MemoryValue::from_tag(MemoryTag::U64, 0xfffffffffffffec0ULL),
1877 MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 128) - 320),
1882class AluShlConstrainingTest :
public AluConstrainingTest,
1883 public ::testing::WithParamInterface<ThreeOperandTestParams> {
1885 TestTraceContainer process_shl_trace(ThreeOperandTestParams params)
1887 auto [
a,
b, c] = params;
1889 auto mem_tag =
a.get_tag();
1890 auto tag =
static_cast<uint8_t
>(mem_tag);
1892 auto a_num =
static_cast<uint128_t>(
a.as_ff());
1893 auto b_num =
static_cast<uint128_t>(
b.as_ff());
1895 auto overflow = b_num > tag_bits;
1896 uint128_t shift_lo_bits = overflow ? tag_bits : tag_bits - b_num;
1897 uint128_t shift_hi_bits = overflow ? tag_bits : b_num;
1898 auto two_pow_shift_lo_bits =
static_cast<uint128_t>(1) << shift_lo_bits;
1899 auto a_lo = overflow ? b_num - tag_bits : a_num % two_pow_shift_lo_bits;
1900 auto a_hi = a_num >> shift_lo_bits;
1902 auto trace = TestTraceContainer({
1904 { C::alu_a_hi, a_hi },
1905 { C::alu_a_hi_bits, shift_hi_bits },
1906 { C::alu_a_lo, a_lo },
1907 { C::alu_a_lo_bits, shift_lo_bits },
1908 { C::alu_helper1,
static_cast<uint128_t>(1) << b_num },
1910 { C::alu_ia_tag,
tag },
1912 { C::alu_ib_tag,
tag },
1914 { C::alu_ic_tag,
tag },
1915 { C::alu_max_bits, tag_bits },
1919 { C::alu_sel_decompose_a, 1 },
1920 { C::alu_sel_op_shl, 1 },
1921 { C::alu_sel_shift_ops_no_overflow, overflow ? 0 : 1 },
1922 { C::alu_shift_lo_bits, shift_lo_bits },
1923 { C::alu_tag_ff_diff_inv,
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
1924 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
1925 { C::alu_tag_u128_diff_inv,
1926 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
1929 { C::alu_two_pow_shift_lo_bits, two_pow_shift_lo_bits },
1930 { C::execution_mem_tag_reg_0_,
tag },
1931 { C::execution_mem_tag_reg_1_,
tag },
1932 { C::execution_mem_tag_reg_2_,
tag },
1933 { C::execution_register_0_,
a },
1934 { C::execution_register_1_,
b },
1935 { C::execution_register_2_, c },
1936 { C::execution_sel_exec_dispatch_alu, 1 },
1945 range_check_builder.process({ { .value = a_lo, .num_bits =
static_cast<uint8_t
>(shift_lo_bits) },
1946 { .value = a_hi, .num_bits =
static_cast<uint8_t
>(shift_hi_bits) } },
1952 TestTraceContainer process_shl_with_tracegen(ThreeOperandTestParams params,
bool error =
false)
1954 TestTraceContainer
trace;
1955 auto [
a,
b, c] = params;
1956 auto b_num =
static_cast<uint128_t>(
b.as_ff());
1958 auto overflow = b_num > tag_bits;
1959 uint128_t shift_lo_bits = overflow ? tag_bits : tag_bits - b_num;
1960 auto a_lo = overflow ? b_num - tag_bits
1965 { .operation = simulation::AluOperation::SHL, .a =
a, .b =
b, .c = c, .error = error },
1972 range_check_builder.process({ { .value = a_lo, .num_bits =
static_cast<uint8_t
>(shift_lo_bits) },
1973 { .value =
static_cast<uint128_t>(
a.as_ff()) >> shift_lo_bits,
1974 .num_bits =
static_cast<uint8_t
>(overflow ? tag_bits : b_num) } },
1982TEST_P(AluShlConstrainingTest, AluShl)
1984 auto trace = process_shl_trace(GetParam());
1985 check_all_interactions<AluTraceBuilder>(trace);
1986 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1987 check_relation<alu>(trace);
1990TEST_P(AluShlConstrainingTest, AluShlTraceGen)
1992 auto trace = process_shl_with_tracegen(GetParam());
1993 check_all_interactions<AluTraceBuilder>(trace);
1994 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1995 check_relation<alu>(trace);
1998TEST_F(AluShlConstrainingTest, NegativeAluShlFF)
2000 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
2001 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
2002 auto c = MemoryValue::from_tag(MemoryTag::FF, 2 << 5);
2003 auto trace = process_shl_with_tracegen({
a,
b, c },
true);
2004 check_relation<alu>(trace);
2005 check_all_interactions<AluTraceBuilder>(trace);
2008 trace.set(Column::alu_ib, 0, 0);
2009 check_relation<alu>(trace);
2010 check_all_interactions<AluTraceBuilder>(trace);
2013 trace.set(Column::alu_sel_tag_err, 0, 0);
2014 trace.set(Column::alu_sel_err, 0, 0);
2018TEST_F(AluShlConstrainingTest, NegativeAluShlTagMismatchOverflow)
2020 auto a = MemoryValue::from_tag(MemoryTag::U8, 2);
2021 auto b = MemoryValue::from_tag(MemoryTag::U32, 256);
2022 auto c = MemoryValue::from_tag(MemoryTag::U8, 0);
2023 auto trace = process_shl_with_tracegen({
a,
b, c },
true);
2024 check_relation<alu>(trace);
2025 check_all_interactions<AluTraceBuilder>(trace);
2028 trace.set(Column::alu_sel_tag_err, 0, 0);
2029 trace.set(Column::alu_sel_err, 0, 0);
2031 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 0);
2035 trace.set(Column::alu_ab_tags_diff_inv, 0, 0);
2039 trace.set(Column::alu_ab_tags_diff_inv,
2041 (
FF(
static_cast<uint8_t
>(MemoryTag::U8)) -
FF(
static_cast<uint8_t
>(MemoryTag::U32))).invert());
2042 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
2049 MemoryValue::from_tag(MemoryTag::U1, 1),
2050 MemoryValue::from_tag(MemoryTag::U8, 0),
2051 MemoryValue::from_tag(MemoryTag::U16, 0),
2052 MemoryValue::from_tag(MemoryTag::U32, 0x7ffffff),
2053 MemoryValue::from_tag(MemoryTag::U64, 0x7ffffffffffffffULL),
2054 MemoryValue::from_tag(MemoryTag::U128,
2060class AluShrConstrainingTest :
public AluConstrainingTest,
2061 public ::testing::WithParamInterface<ThreeOperandTestParams> {
2063 TestTraceContainer process_shr_trace(ThreeOperandTestParams params)
2065 auto [
a,
b, c] = params;
2067 auto mem_tag =
a.get_tag();
2068 auto tag =
static_cast<uint8_t
>(mem_tag);
2070 auto a_num =
static_cast<uint128_t>(
a.as_ff());
2071 auto b_num =
static_cast<uint128_t>(
b.as_ff());
2073 auto overflow = b_num > tag_bits;
2074 uint128_t shift_lo_bits = overflow ? tag_bits : b_num;
2075 uint128_t shift_hi_bits = overflow ? tag_bits : tag_bits - b_num;
2076 auto two_pow_shift_lo_bits =
static_cast<uint128_t>(1) << shift_lo_bits;
2077 auto a_lo = overflow ? b_num - tag_bits : a_num % two_pow_shift_lo_bits;
2078 auto a_hi = a_num >> shift_lo_bits;
2080 auto trace = TestTraceContainer({
2082 { C::alu_a_hi, a_hi },
2083 { C::alu_a_hi_bits, shift_hi_bits },
2084 { C::alu_a_lo, a_lo },
2085 { C::alu_a_lo_bits, shift_lo_bits },
2087 { C::alu_ia_tag,
tag },
2089 { C::alu_ib_tag,
tag },
2091 { C::alu_ic_tag,
tag },
2092 { C::alu_max_bits, tag_bits },
2096 { C::alu_sel_decompose_a, 1 },
2097 { C::alu_sel_op_shr, 1 },
2098 { C::alu_sel_shift_ops_no_overflow, overflow ? 0 : 1 },
2099 { C::alu_shift_lo_bits, shift_lo_bits },
2100 { C::alu_tag_ff_diff_inv,
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
2101 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
2102 { C::alu_tag_u128_diff_inv,
2103 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
2106 { C::alu_two_pow_shift_lo_bits, two_pow_shift_lo_bits },
2107 { C::execution_mem_tag_reg_0_,
tag },
2108 { C::execution_mem_tag_reg_1_,
tag },
2109 { C::execution_mem_tag_reg_2_,
tag },
2110 { C::execution_register_0_,
a },
2111 { C::execution_register_1_,
b },
2112 { C::execution_register_2_, c },
2113 { C::execution_sel_exec_dispatch_alu, 1 },
2122 range_check_builder.process({ { .value = a_lo, .num_bits =
static_cast<uint8_t
>(shift_lo_bits) },
2123 { .value = a_hi, .num_bits =
static_cast<uint8_t
>(shift_hi_bits) } },
2129 TestTraceContainer process_shr_with_tracegen(ThreeOperandTestParams params,
bool error =
false)
2131 TestTraceContainer
trace;
2132 auto [
a,
b, c] = params;
2133 auto b_num =
static_cast<uint128_t>(
b.as_ff());
2135 auto overflow = b_num > tag_bits;
2136 uint128_t shift_lo_bits = overflow ? tag_bits : b_num;
2137 auto a_lo = overflow ? b_num - tag_bits
2142 { .operation = simulation::AluOperation::SHR, .a =
a, .b =
b, .c = c, .error = error },
2149 range_check_builder.process({ { .value = a_lo, .num_bits =
static_cast<uint8_t
>(shift_lo_bits) },
2150 { .value =
static_cast<uint128_t>(
a.as_ff()) >> shift_lo_bits,
2151 .num_bits =
static_cast<uint8_t
>(overflow ? tag_bits : tag_bits - b_num) } },
2159TEST_P(AluShrConstrainingTest, AluShr)
2161 auto trace = process_shr_trace(GetParam());
2162 check_all_interactions<AluTraceBuilder>(trace);
2163 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
2164 check_relation<alu>(trace);
2167TEST_P(AluShrConstrainingTest, AluShrTraceGen)
2169 auto trace = process_shr_with_tracegen(GetParam());
2170 check_all_interactions<AluTraceBuilder>(trace);
2171 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
2172 check_relation<alu>(trace);
2175TEST_F(AluShrConstrainingTest, NegativeAluShrFF)
2177 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
2178 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
2179 auto c = MemoryValue::from_tag(MemoryTag::FF, 2 << 5);
2180 auto trace = process_shr_with_tracegen({
a,
b, c },
true);
2181 check_relation<alu>(trace);
2182 check_all_interactions<AluTraceBuilder>(trace);
2185 trace.set(Column::alu_ib, 0, 0);
2186 check_relation<alu>(trace);
2187 check_all_interactions<AluTraceBuilder>(trace);
2190 trace.set(Column::alu_sel_tag_err, 0, 0);
2191 trace.set(Column::alu_sel_err, 0, 0);
2195TEST_F(AluShrConstrainingTest, NegativeAluShrTagMismatchOverflow)
2197 auto a = MemoryValue::from_tag(MemoryTag::U16, 2);
2198 auto b = MemoryValue::from_tag(MemoryTag::U64, 123456);
2199 auto c = MemoryValue::from_tag(MemoryTag::U16, 0);
2200 auto trace = process_shr_with_tracegen({
a,
b, c },
true);
2201 check_relation<alu>(trace);
2202 check_all_interactions<AluTraceBuilder>(trace);
2204 trace.set(Column::alu_sel_tag_err, 0, 0);
2205 trace.set(Column::alu_sel_err, 0, 0);
2207 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 0);
2211 trace.set(Column::alu_ab_tags_diff_inv, 0, 0);
2215 trace.set(Column::alu_ab_tags_diff_inv,
2217 (
FF(
static_cast<uint8_t
>(MemoryTag::U16)) -
FF(
static_cast<uint8_t
>(MemoryTag::U64))).invert());
2218 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
2227 { MemoryValue::from_tag(MemoryTag::FF, 1),
2228 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U1)),
2229 MemoryValue::from_tag(MemoryTag::U1, 1) },
2230 { MemoryValue::from_tag(MemoryTag::FF, 42),
2231 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U8)),
2232 MemoryValue::from_tag(MemoryTag::U8, 42) },
2233 { MemoryValue::from_tag(MemoryTag::FF, 12345),
2234 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U16)),
2235 MemoryValue::from_tag(MemoryTag::U16, 12345) },
2236 { MemoryValue::from_tag(MemoryTag::FF, 123456789),
2237 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2238 MemoryValue::from_tag(MemoryTag::U32, 123456789) },
2239 { MemoryValue::from_tag(MemoryTag::FF, 1234567890123456789ULL),
2240 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2241 MemoryValue::from_tag(MemoryTag::U64, 1234567890123456789ULL) },
2242 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(1) << 127) + 23423429816234ULL),
2243 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U128)),
2244 MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 127) + 23423429816234ULL) },
2245 { MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 3),
2246 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::FF)),
2247 MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 3) },
2249 { MemoryValue::from_tag(MemoryTag::FF, 212),
2250 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U1)),
2251 MemoryValue::from_tag(MemoryTag::U1, 0) },
2252 { MemoryValue::from_tag(MemoryTag::FF, 257),
2253 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U8)),
2254 MemoryValue::from_tag(MemoryTag::U8, 1) },
2255 { MemoryValue::from_tag(MemoryTag::FF, 65540),
2256 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U16)),
2257 MemoryValue::from_tag(MemoryTag::U16, 4) },
2258 { MemoryValue::from_tag(MemoryTag::FF, 4294967298ULL),
2259 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2260 MemoryValue::from_tag(MemoryTag::U32, 2) },
2261 { MemoryValue::from_tag(MemoryTag::FF, 18446744073709551615ULL + 4),
2262 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2263 MemoryValue::from_tag(MemoryTag::U64, 3) },
2265 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 129) + 986132),
2266 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U1)),
2267 MemoryValue::from_tag(MemoryTag::U1, 0) },
2268 { MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 128735618772ULL),
2269 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U8)),
2270 MemoryValue::from_tag(MemoryTag::U8, 45) },
2271 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(999) << 128) - 986132ULL),
2272 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U16)),
2273 MemoryValue::from_tag(MemoryTag::U16, 62444) },
2274 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 198) + 986132ULL),
2275 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2276 MemoryValue::from_tag(MemoryTag::U32, 986132ULL) },
2277 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 198) - 986132ULL),
2278 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2279 MemoryValue::from_tag(MemoryTag::U64,
static_cast<uint64_t
>(-986132ULL)) },
2280 { MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 8723),
2281 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U128)),
2285class AluTruncateConstrainingTest :
public AluConstrainingTest,
2286 public ::testing::WithParamInterface<ThreeOperandTestParams> {
2288 TestTraceContainer process_truncate_with_tracegen(
const ThreeOperandTestParams& params, TestTraceContainer& trace)
2290 auto [
a,
b, c] = params;
2294 { .operation = simulation::AluOperation::TRUNCATE, .a =
a, .b =
b, .c = c },
2301 auto is_non_trivial =
trace.get(Column::alu_sel_trunc_non_trivial, 0) == 1;
2303 if (is_non_trivial) {
2304 auto a_decomp = simulation::decompose_256(
static_cast<uint256_t>(
a.as_ff()));
2308 .num_bits =
static_cast<uint8_t
>(128 - bits) } },
2310 auto is_gte_128 =
trace.get(Column::alu_sel_trunc_gte_128, 0) == 1;
2312 auto p_limbs = simulation::decompose_256(
FF::modulus);
2313 simulation::LimbsComparisonWitness p_sub_a_witness = { .lo = p_limbs.lo - a_decomp.lo,
2314 .hi = p_limbs.hi - a_decomp.hi,
2316 field_gt_builder.process({ { .operation = simulation::FieldGreaterOperation::CANONICAL_DECOMPOSITION,
2318 .a_limbs = a_decomp,
2319 .p_sub_a_witness = p_sub_a_witness } },
2327 TestTraceContainer process_set_with_tracegen(
const ThreeOperandTestParams& params)
2329 TestTraceContainer
trace;
2330 auto [
a,
b, _c] = params;
2332 auto c = MemoryValue::from_tag_truncating(
dst_tag,
a);
2335 { Column::execution_sel_exec_dispatch_set, 1 },
2336 { Column::execution_rop_2_,
a },
2337 { Column::execution_rop_1_,
static_cast<uint8_t
>(
dst_tag) },
2339 { Column::execution_register_0_, c.as_ff() },
2340 { Column::execution_mem_tag_reg_0_,
static_cast<uint8_t
>(
dst_tag) },
2343 process_truncate_with_tracegen(params, trace);
2348 TestTraceContainer process_cast_with_tracegen(
const ThreeOperandTestParams& params)
2350 TestTraceContainer
trace;
2351 auto [
a,
b, _c] = params;
2353 auto c = MemoryValue::from_tag_truncating(
dst_tag,
a);
2356 { Column::execution_sel_exec_dispatch_cast, 1 },
2357 { Column::execution_register_0_,
a },
2358 { Column::execution_rop_2_,
static_cast<uint8_t
>(
dst_tag) },
2360 { Column::execution_register_1_, c.as_ff() },
2361 { Column::execution_mem_tag_reg_1_,
static_cast<uint8_t
>(
dst_tag) },
2364 process_truncate_with_tracegen(params, trace);
2370INSTANTIATE_TEST_SUITE_P(AluConstrainingTest, AluTruncateConstrainingTest, ::testing::ValuesIn(TEST_VALUES_TRUNCATE));
2372TEST_P(AluTruncateConstrainingTest, AluSet)
2374 auto trace = process_set_with_tracegen(GetParam());
2375 check_all_interactions<AluTraceBuilder>(trace);
2376 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_set_settings>(trace);
2377 check_relation<alu>(trace);
2380TEST_P(AluTruncateConstrainingTest, AluCast)
2382 auto trace = process_cast_with_tracegen(GetParam());
2383 check_all_interactions<AluTraceBuilder>(trace);
2384 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_set_settings>(trace);
2385 check_relation<alu>(trace);
2388TEST_P(AluTruncateConstrainingTest, NegativeTruncateWrongTrivialCase)
2390 TestTraceContainer
trace;
2391 process_truncate_with_tracegen(GetParam(), trace);
2392 check_relation<alu>(trace);
2393 bool is_trivial =
trace.get(Column::alu_sel_trunc_trivial, 0) == 1;
2394 trace.set(Column::alu_sel_trunc_trivial, 0,
static_cast<uint8_t
>(!is_trivial));
2396 trace.set(Column::alu_sel_trunc_trivial, 0,
static_cast<uint8_t
>(is_trivial));
2397 trace.set(Column::alu_sel_trunc_non_trivial, 0,
static_cast<uint8_t
>(is_trivial));
2401TEST_P(AluTruncateConstrainingTest, NegativeTruncateWrong128BitsCase)
2403 TestTraceContainer
trace;
2404 process_truncate_with_tracegen(GetParam(), trace);
2405 check_relation<alu>(trace);
2406 bool is_lt_128 =
trace.get(Column::alu_sel_trunc_lt_128, 0) == 1;
2407 trace.set(Column::alu_sel_trunc_lt_128, 0,
static_cast<uint8_t
>(!is_lt_128));
2409 trace.set(Column::alu_sel_trunc_lt_128, 0,
static_cast<uint8_t
>(is_lt_128));
2410 check_relation<alu>(trace);
2411 bool is_gte_128 =
trace.get(Column::alu_sel_trunc_gte_128, 0) == 1;
2412 trace.set(Column::alu_sel_trunc_gte_128, 0,
static_cast<uint8_t
>(!is_gte_128));
2416TEST_F(AluTruncateConstrainingTest, NegativeTruncateWrongMid)
2418 TestTraceContainer
trace;
2419 process_truncate_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, 4294967298ULL),
2420 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2421 MemoryValue::from_tag(MemoryTag::U32, 2) },
2423 check_relation<alu>(trace);
2424 trace.set(Column::alu_mid, 0, 1234ULL);
2428TEST_F(AluTruncateConstrainingTest, NegativeTruncateWrongMidBits)
2430 TestTraceContainer
trace;
2431 process_truncate_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 2),
2432 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U1)),
2433 MemoryValue::from_tag(MemoryTag::U1, 1) },
2435 check_relation<alu>(trace);
2436 trace.set(Column::alu_mid_bits, 0, 32);
2440TEST_F(AluTruncateConstrainingTest, NegativeTruncateWrongLo128FromCanonDec)
2442 TestTraceContainer
trace;
2443 process_truncate_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 198) - 986132ULL),
2444 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2445 MemoryValue::from_tag(MemoryTag::U64,
static_cast<uint64_t
>(-986132ULL)) },
2447 check_relation<alu>(trace);
2448 check_all_interactions<AluTraceBuilder>(trace);
2449 trace.set(Column::alu_a_lo, 0, 1234ULL);
2451 (check_interaction<AluTraceBuilder, lookup_alu_large_trunc_canonical_dec_settings>(trace)),
2452 "Failed.*LARGE_TRUNC_CANONICAL_DEC. Could not find tuple in destination.");
2455TEST_F(AluTruncateConstrainingTest, NegativeTruncateWrongMidIntoRangeCheck)
2457 TestTraceContainer
trace;
2458 process_truncate_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 198) - 986132ULL),
2459 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2460 MemoryValue::from_tag(MemoryTag::U64,
static_cast<uint64_t
>(-986132ULL)) },
2462 check_relation<alu>(trace);
2463 check_all_interactions<AluTraceBuilder>(trace);
2464 trace.set(Column::alu_mid, 0, 1234ULL);
2466 "Failed.*RANGE_CHECK_TRUNC_MID. Could not find tuple in destination.");
2469TEST_F(AluTruncateConstrainingTest, NegativeCastWrongDispatching)
2472 process_cast_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, 4294967298ULL),
2473 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2474 MemoryValue::from_tag(MemoryTag::U32, 2) });
2475 check_relation<alu>(trace);
2476 check_all_interactions<AluTraceBuilder>(trace);
2477 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_cast_settings>(trace);
2478 trace.set(Column::execution_register_0_, 0,
trace.get(Column::execution_register_0_, 0) + 1);
2480 (check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_cast_settings>(trace)),
2481 "Failed.*EXECUTION_DISPATCH_TO_CAST. Could not find tuple in destination.");
2484TEST_F(AluTruncateConstrainingTest, NegativeSetWrongDispatching)
2486 auto trace = process_set_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, 4294967298ULL),
2487 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2488 MemoryValue::from_tag(MemoryTag::U32, 2) });
2489 check_relation<alu>(trace);
2490 check_all_interactions<AluTraceBuilder>(trace);
2491 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_set_settings>(trace);
2492 trace.set(Column::execution_rop_2_, 0,
trace.get(Column::execution_rop_2_, 0) + 1);
2494 (check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_set_settings>(trace)),
2495 "Failed.*EXECUTION_DISPATCH_TO_SET. Could not find tuple in destination.");
#define EXPECT_THROW_WITH_MESSAGE(code, expectedMessageRegex)
bb::field< bb::Bn254FrParams > FF
#define AVM_EXEC_OP_ID_ALU_TRUNCATE
#define AVM_EXEC_OP_ID_ALU_LTE
#define AVM_EXEC_OP_ID_ALU_DIV
#define AVM_EXEC_OP_ID_ALU_ADD
#define AVM_EXEC_OP_ID_ALU_SHL
#define AVM_EXEC_OP_ID_ALU_SUB
#define AVM_EXEC_OP_ID_ALU_MUL
#define AVM_EXEC_OP_ID_ALU_FDIV
#define AVM_EXEC_OP_ID_ALU_SHR
#define AVM_EXEC_OP_ID_ALU_LT
static TaggedValue from_tag(ValueTag tag, FF value)
static constexpr size_t SR_DISPATCH_OPERATION
RangeCheckTraceBuilder range_check_builder
PrecomputedTraceBuilder precomputed_builder
FieldGreaterThanTraceBuilder field_gt_builder
GreaterThanTraceBuilder gt_builder
TEST_F(BoomerangIPARecursiveTests, FullRecursiveVerifierMediumRandom)
TEST_P(AvmRecursiveTestsParameterized, TwoLayerAvmRecursion)
A test of the Two Layer AVM recursive verifier.
INSTANTIATE_TEST_SUITE_P(PaddingVariants, AvmRecursiveTestsParameterized, ::testing::Values(false, true), [](const auto &info) { return info.param ? "Padded" :"Unpadded";})
TEST_F(AvmRecursiveTests, TwoLayerAvmRecursionFailsWithWrongPIs)
TestTraceContainer empty_trace()
uint8_t get_tag_bits(ValueTag tag)
uint256_t get_tag_max_value(ValueTag tag)
constexpr decltype(auto) get(::tuplet::tuple< T... > &&t) noexcept
constexpr auto tuple_cat(T &&... ts)
unsigned __int128 uint128_t
static constexpr uint256_t modulus
constexpr field invert() const noexcept