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 } } });
842 { .value = c_decomp.hi, .num_bits = 64 },
843 { .value = b_decomp.lo, .num_bits = 64 },
844 { .value = b_decomp.hi, .num_bits = 64 } },
851 TestTraceContainer process_div_with_tracegen(ThreeOperandTestParams params)
853 TestTraceContainer
trace;
854 auto [
a,
b, c] = params;
855 bool div_0_error =
b.as_ff() ==
FF(0);
856 auto mem_tag =
a.get_tag();
858 MemoryValue remainder = MemoryValue::from_tag(MemoryTag::FF, 0);
859 if (!div_0_error && mem_tag ==
b.get_tag() && mem_tag != MemoryTag::FF) {
860 remainder =
a -
b * c;
865 { .operation = simulation::AluOperation::DIV, .a =
a, .b =
b, .c = c, .error = div_0_error },
869 if (mem_tag == MemoryTag::U128) {
870 auto c_decomp = simulation::decompose_128(
static_cast<uint128_t>(c.as_ff()));
871 auto b_decomp = simulation::decompose_128(
static_cast<uint128_t>(
b.as_ff()));
874 { .value = c_decomp.hi, .num_bits = 64 },
875 { .value = b_decomp.lo, .num_bits = 64 },
876 { .value = b_decomp.hi, .num_bits = 64 } },
882 .b =
static_cast<uint128_t>(remainder.as_ff()),
891TEST_P(AluDivConstrainingTest, AluDiv)
893 auto trace = process_div_trace(GetParam());
894 check_all_interactions<AluTraceBuilder>(trace);
895 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
896 check_relation<alu>(trace);
899TEST_P(AluDivConstrainingTest, AluDivTraceGen)
901 auto trace = process_div_with_tracegen(GetParam());
902 check_all_interactions<AluTraceBuilder>(trace);
903 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
904 check_relation<alu>(trace);
907TEST_F(AluDivConstrainingTest, AluDivByZeroMismatchTagsTraceGen)
909 auto a = MemoryValue::from_tag(MemoryTag::U128, 2);
910 auto b = MemoryValue::from_tag(MemoryTag::U64, 0);
911 auto c = MemoryValue::from_tag(MemoryTag::FF, 0);
913 auto trace = process_div_with_tracegen({
a,
b, c });
914 check_relation<alu>(trace);
915 check_all_interactions<AluTraceBuilder>(trace);
918TEST_F(AluDivConstrainingTest, AluDivByZeroTagFFAndMismatchTagsTraceGen)
920 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
921 auto b = MemoryValue::from_tag(MemoryTag::U32, 0);
922 auto c = MemoryValue::from_tag(MemoryTag::FF, 0);
924 auto trace = process_div_with_tracegen({
a,
b, c });
925 check_relation<alu>(trace);
926 check_all_interactions<AluTraceBuilder>(trace);
929TEST_F(AluDivConstrainingTest, NegativeAluDivUnderflow)
932 auto a = MemoryValue::from_tag(MemoryTag::U32, 2);
933 auto b = MemoryValue::from_tag(MemoryTag::U32, 5);
935 auto trace = process_div_trace({
a,
b, c });
936 check_all_interactions<AluTraceBuilder>(trace);
937 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
938 check_relation<alu>(trace);
943 c = MemoryValue::from_tag(MemoryTag::U32, 2);
944 auto wrong_remainder =
a.as_ff() -
b.as_ff() * c.as_ff();
946 trace.set(Column::alu_ic, 0, c);
947 trace.set(Column::alu_helper1, 0, wrong_remainder);
948 trace.set(Column::alu_gt_input_b, 0, wrong_remainder);
951 check_relation<alu>(trace);
956TEST_F(AluDivConstrainingTest, NegativeAluDivU128Carry)
959 auto a = MemoryValue::from_tag(MemoryTag::U128, 2);
960 auto b = MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 64) + 2);
963 auto trace = process_div_trace({
a,
b, c });
965 check_all_interactions<AluTraceBuilder>(trace);
966 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
967 check_relation<alu>(trace);
971 c = MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 64) + 3);
972 auto wrong_remainder =
a.as_ff() -
FF(
static_cast<uint256_t>(
b.as_ff()) *
static_cast<uint256_t>(c.as_ff()));
976 trace.set(Column::alu_ic, 0, c);
977 trace.set(Column::alu_helper1, 0, wrong_remainder);
982 auto c_decomp = simulation::decompose_128(c.as<
uint128_t>());
983 trace.set(Column::alu_a_lo, 0, c_decomp.lo);
984 trace.set(Column::alu_a_hi, 0, c_decomp.hi);
990TEST_F(AluDivConstrainingTest, NegativeAluDivByZero)
992 auto a = MemoryValue::from_tag(MemoryTag::U32, 2);
993 auto b = MemoryValue::from_tag(MemoryTag::U32, 5);
996 for (
const bool with_tracegen : {
false,
true }) {
997 auto trace = with_tracegen ? process_div_with_tracegen({
a,
b, c }) : process_div_trace({
a,
b, c });
998 check_all_interactions<AluTraceBuilder>(trace);
999 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1000 check_relation<alu>(trace);
1003 trace.set(Column::alu_ib, 0, 0);
1004 trace.set(Column::alu_b_inv, 0, 0);
1008 trace.set(Column::alu_sel_div_0_err, 0, 1);
1009 trace.set(Column::alu_sel_div_no_err, 0, 0);
1012 trace.set(Column::alu_sel_err, 0, 1);
1013 trace.set(Column::alu_sel_int_gt, 0, 0);
1014 trace.set(Column::alu_gt_input_a, 0, 0);
1015 trace.set(Column::alu_gt_input_b, 0, 0);
1016 trace.set(Column::alu_gt_result_c, 0, 0);
1017 check_relation<alu>(trace);
1020 trace.set(Column::alu_sel_op_div, 0, 0);
1021 trace.set(Column::alu_sel_op_mul, 0, 1);
1025 trace.set(Column::alu_sel_op_div, 0, 1);
1026 trace.set(Column::alu_sel_op_mul, 0, 0);
1028 check_relation<alu>(trace);
1031 trace.set(Column::alu_ib, 0,
b);
1037TEST_F(AluDivConstrainingTest, NegativeAluDivFF)
1039 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1040 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1042 auto trace = process_div_with_tracegen({
a,
b, c });
1045 trace.set(Column::alu_sel_tag_err, 0, 1);
1046 trace.set(Column::alu_sel_err, 0, 1);
1047 trace.set(Column::alu_sel_div_no_err, 0, 0);
1048 trace.set(Column::alu_sel_int_gt, 0, 0);
1049 trace.set(Column::alu_gt_input_a, 0, 0);
1050 trace.set(Column::alu_gt_input_b, 0, 0);
1051 trace.set(Column::alu_gt_result_c, 0, 0);
1052 check_relation<alu>(trace);
1053 check_all_interactions<AluTraceBuilder>(trace);
1054 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1057TEST_F(AluDivConstrainingTest, NegativeAluDivByZeroFF)
1060 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1061 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1063 auto trace = process_div_with_tracegen({
a,
b, c });
1064 trace.set(Column::alu_sel_tag_err, 0, 1);
1065 trace.set(Column::alu_sel_err, 0, 1);
1066 trace.set(Column::alu_sel_div_no_err, 0, 0);
1067 trace.set(Column::alu_sel_int_gt, 0, 0);
1068 trace.set(Column::alu_gt_input_a, 0, 0);
1069 trace.set(Column::alu_gt_input_b, 0, 0);
1070 trace.set(Column::alu_gt_result_c, 0, 0);
1071 check_relation<alu>(trace);
1073 trace.set(Column::alu_ib, 0, 0);
1074 trace.set(Column::alu_b_inv, 0, 0);
1076 trace.set(Column::alu_sel_div_0_err, 0, 1);
1077 check_relation<alu>(trace);
1078 check_all_interactions<AluTraceBuilder>(trace);
1079 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1082TEST_F(AluDivConstrainingTest, NegativeAluDivByZeroFFTagMismatch)
1085 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1086 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1088 auto trace = process_div_with_tracegen({
a,
b, c });
1089 trace.set(Column::alu_sel_tag_err, 0, 1);
1090 trace.set(Column::alu_sel_err, 0, 1);
1091 trace.set(Column::alu_sel_div_no_err, 0, 0);
1092 trace.set(Column::alu_sel_int_gt, 0, 0);
1093 trace.set(Column::alu_gt_input_a, 0, 0);
1094 trace.set(Column::alu_gt_input_b, 0, 0);
1095 trace.set(Column::alu_gt_result_c, 0, 0);
1096 check_relation<alu>(trace);
1098 trace.set(Column::alu_ib_tag, 0,
static_cast<uint8_t
>(MemoryTag::U8));
1100 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
1101 trace.set(Column::alu_ab_tags_diff_inv,
1103 (
FF(
static_cast<uint8_t
>(MemoryTag::FF)) -
FF(
static_cast<uint8_t
>(MemoryTag::U8))).invert());
1104 check_relation<alu>(trace);
1106 trace.set(Column::alu_ib, 0, 0);
1107 trace.set(Column::alu_b_inv, 0, 0);
1109 trace.set(Column::alu_sel_div_0_err, 0, 1);
1110 trace.set(Column::alu_sel_div_no_err, 0, 0);
1111 check_relation<alu>(trace);
1112 check_all_interactions<AluTraceBuilder>(trace);
1113 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1121 MemoryValue::from_tag(MemoryTag::FF, 0),
1122 MemoryValue::from_tag(MemoryTag::FF, 4),
1123 MemoryValue::from_tag(MemoryTag::FF,
FF(
"0x1e980ebbc51694827ee20074ac28b250a037a43eb44b38e6aa367c57a05e6d48")),
1124 MemoryValue::from_tag(MemoryTag::FF,
FF(
"0x135b52945a13d9aa49b9b57c33cd568ba9ae5ce9ca4a2d06e7f3fbd4f9999998")),
1125 MemoryValue::from_tag(MemoryTag::FF,
FF(
"0x135b52945a13d9aa49b9b57c33cd568ba9ae5ce9ca4a2d071b272f07f9999998")),
1126 MemoryValue::from_tag(MemoryTag::FF,
FF(
"0x135b52945a13d9aa49b9b57c33cd568bdce1901cfd7d603a1b272f07f9999998")),
1127 MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 2),
1132class AluFDivConstrainingTest :
public AluConstrainingTest,
1133 public ::testing::WithParamInterface<ThreeOperandTestParams> {
1135 TestTraceContainer process_fdiv_trace(ThreeOperandTestParams params)
1137 auto [
a,
b, c] = params;
1138 a = MemoryValue::from_tag(MemoryTag::FF,
a);
1139 b = MemoryValue::from_tag(MemoryTag::FF,
b);
1140 c = MemoryValue::from_tag(MemoryTag::FF, c);
1141 auto div_0_error =
b.as_ff() ==
FF(0);
1143 auto mem_tag =
a.get_tag();
1144 auto tag =
static_cast<uint8_t
>(mem_tag);
1146 auto trace = TestTraceContainer({
1148 { C::alu_b_inv, div_0_error ? 0 :
b.as_ff().invert() },
1150 { C::alu_ia_tag,
tag },
1152 { C::alu_ib_tag,
tag },
1154 { C::alu_ic_tag,
tag },
1159 { C::alu_sel_div_0_err, div_0_error ? 1 : 0 },
1160 { C::alu_sel_err, div_0_error ? 1 : 0 },
1161 { C::alu_sel_is_ff, 1 },
1162 { C::alu_sel_op_fdiv, 1 },
1163 { C::alu_tag_u128_diff_inv,
FF(-
static_cast<uint8_t
>(MemoryTag::U128)).
invert() },
1164 { C::execution_mem_tag_reg_0_,
tag },
1165 { C::execution_mem_tag_reg_1_,
tag },
1166 { C::execution_mem_tag_reg_2_,
tag },
1167 { C::execution_register_0_,
a },
1168 { C::execution_register_1_,
b },
1169 { C::execution_register_2_, c },
1170 { C::execution_sel_exec_dispatch_alu, 1 },
1171 { C::execution_sel_opcode_error, div_0_error ? 1 : 0 },
1182 TestTraceContainer process_fdiv_with_tracegen(ThreeOperandTestParams params,
bool upcast_to_ff =
false)
1184 TestTraceContainer
trace;
1185 auto [
a,
b, c] = params;
1188 a = MemoryValue::from_tag(MemoryTag::FF,
a);
1189 b = MemoryValue::from_tag(MemoryTag::FF,
b);
1190 c = MemoryValue::from_tag(MemoryTag::FF, c);
1193 bool div_0_error =
b.as_ff() ==
FF(0);
1197 { .operation = simulation::AluOperation::FDIV, .a =
a, .b =
b, .c = c, .error = div_0_error },
1210TEST_P(AluFDivConstrainingTest, AluFDiv)
1212 auto trace = process_fdiv_trace(GetParam());
1213 check_all_interactions<AluTraceBuilder>(trace);
1214 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1215 check_relation<alu>(trace);
1218TEST_P(AluFDivConstrainingTest, AluFDivTraceGen)
1220 auto trace = process_fdiv_with_tracegen(GetParam(),
true);
1223 check_all_interactions<AluTraceBuilder>(trace);
1224 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1225 check_relation<alu>(trace);
1228TEST_F(AluFDivConstrainingTest, AluFDivByZeroNonFFTagTraceGen)
1230 auto a = MemoryValue::from_tag(MemoryTag::U8, 2);
1231 auto b = MemoryValue::from_tag(MemoryTag::U8, 0);
1232 auto c = MemoryValue::from_tag(MemoryTag::FF, 0);
1234 auto trace = process_fdiv_with_tracegen({
a,
b, c });
1235 check_relation<alu>(trace);
1236 check_all_interactions<AluTraceBuilder>(trace);
1239TEST_F(AluFDivConstrainingTest, AluFDivByZeroNonFFTagMismatchTraceGen)
1241 auto a = MemoryValue::from_tag(MemoryTag::U8, 2);
1242 auto b = MemoryValue::from_tag(MemoryTag::U16, 0);
1243 auto c = MemoryValue::from_tag(MemoryTag::FF, 0);
1244 auto trace = process_fdiv_with_tracegen({
a,
b, c });
1245 check_relation<alu>(trace);
1246 check_all_interactions<AluTraceBuilder>(trace);
1249TEST_F(AluFDivConstrainingTest, NegativeAluFDivByZero)
1251 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1252 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1254 auto trace = process_fdiv_trace({
a,
b, c });
1255 check_all_interactions<AluTraceBuilder>(trace);
1256 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1257 check_relation<alu>(trace);
1260 trace.set(Column::alu_ib, 0, 0);
1261 trace.set(Column::alu_b_inv, 0, 0);
1265 trace.set(Column::alu_sel_div_0_err, 0, 1);
1268 trace.set(Column::alu_sel_err, 0, 1);
1269 check_relation<alu>(trace);
1272 trace.set(Column::alu_ib, 0,
b);
1277TEST_F(AluFDivConstrainingTest, NegativeAluFDivByZeroNonFFTagMismatch)
1279 auto a = MemoryValue::from_tag(MemoryTag::U8, 4);
1280 auto b = MemoryValue::from_tag(MemoryTag::U8, 2);
1283 auto c = MemoryValue::from_tag(MemoryTag::FF, 2);
1284 auto tag =
static_cast<uint8_t
>(MemoryTag::U8);
1286 auto trace = TestTraceContainer({
1288 { C::alu_b_inv,
b.as_ff().invert() },
1290 { C::alu_ia_tag,
tag },
1292 { C::alu_ib_tag,
tag },
1294 { C::alu_ic_tag,
static_cast<uint8_t
>(MemoryTag::FF) },
1299 { C::alu_sel_op_fdiv, 1 },
1300 { C::alu_sel_tag_err, 1 },
1301 { C::alu_sel_err, 1 },
1302 { C::alu_tag_ff_diff_inv, (
FF(
tag) -
FF(
static_cast<uint8_t
>(MemoryTag::FF))).invert() },
1303 { C::alu_tag_u128_diff_inv, (
FF(
tag) -
FF(
static_cast<uint8_t
>(MemoryTag::U128))).invert() },
1304 { C::execution_mem_tag_reg_0_,
tag },
1305 { C::execution_mem_tag_reg_1_,
tag },
1306 { C::execution_mem_tag_reg_2_,
static_cast<uint8_t
>(MemoryTag::FF) },
1307 { C::execution_register_0_,
a },
1308 { C::execution_register_1_,
b },
1309 { C::execution_register_2_, c },
1310 { C::execution_sel_exec_dispatch_alu, 1 },
1312 { C::execution_sel_opcode_error, 1 },
1319 check_relation<alu>(trace);
1320 check_all_interactions<AluTraceBuilder>(trace);
1321 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1324 trace.set(Column::alu_sel_tag_err, 0, 0);
1325 trace.set(Column::alu_sel_err, 0, 0);
1329 trace.set(Column::alu_tag_ff_diff_inv, 0, 0);
1330 trace.set(Column::alu_sel_is_ff, 0, 1);
1334 trace.set(Column::alu_tag_ff_diff_inv, 0, (
FF(
tag) -
FF(
static_cast<uint8_t
>(MemoryTag::FF))).invert());
1335 trace.set(Column::alu_sel_is_ff, 0, 0);
1336 trace.set(Column::alu_sel_tag_err, 0, 1);
1337 trace.set(Column::alu_sel_err, 0, 1);
1338 check_relation<alu>(trace);
1341 trace.set(Column::alu_ib, 0, 0);
1342 trace.set(Column::alu_b_inv, 0, 0);
1344 trace.set(Column::alu_sel_div_0_err, 0, 1);
1345 check_relation<alu>(trace);
1346 check_all_interactions<AluTraceBuilder>(trace);
1347 trace.set(Column::execution_register_1_, 0, 0);
1348 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1351 trace.set(Column::alu_ib_tag, 0,
static_cast<uint8_t
>(MemoryTag::U16));
1353 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
1354 trace.set(Column::alu_ab_tags_diff_inv, 0, (
FF(
tag) -
FF(
static_cast<uint8_t
>(MemoryTag::U16))).invert());
1355 check_relation<alu>(trace);
1364class AluEQConstrainingTest :
public AluConstrainingTest,
public ::testing::WithParamInterface<ThreeOperandTestParams> {
1366 TestTraceContainer process_eq_with_tracegen(
const ThreeOperandTestParams& params)
1368 TestTraceContainer
trace;
1369 auto [
a,
b, c] = params;
1373 { .operation = simulation::AluOperation::EQ, .a =
a, .b =
b, .c = c },
1385TEST_P(AluEQConstrainingTest, AluEQTraceGen)
1389 process_eq_with_tracegen(ThreeOperandTestParams{ param, param, MemoryValue::from_tag(MemoryTag::U1, 1) });
1390 check_all_interactions<AluTraceBuilder>(trace);
1391 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1392 check_relation<alu>(trace);
1395TEST_P(AluEQConstrainingTest, AluInEQTraceGen)
1397 auto trace = process_eq_with_tracegen(GetParam());
1398 check_all_interactions<AluTraceBuilder>(trace);
1399 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1400 check_relation<alu>(trace);
1403TEST_P(AluEQConstrainingTest, NegativeAluEqResult)
1405 auto params = GetParam();
1406 for (
const bool is_eq : {
false,
true }) {
1407 auto trace = process_eq_with_tracegen(is_eq ? ThreeOperandTestParams{
std::get<0>(params),
1409 MemoryValue::from_tag(MemoryTag::U1, 1) }
1411 check_relation<alu>(trace);
1412 check_all_interactions<AluTraceBuilder>(trace);
1413 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1414 bool c =
trace.get(Column::alu_ic, 0) == 1;
1416 trace.set(Column::alu_ic, 0,
static_cast<uint8_t
>(!c));
1421TEST_P(AluEQConstrainingTest, NegativeAluEqHelper)
1423 auto trace = process_eq_with_tracegen(GetParam());
1424 check_relation<alu>(trace);
1425 check_all_interactions<AluTraceBuilder>(trace);
1426 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1427 auto ab_diff_inv =
trace.get(Column::alu_ab_diff_inv, 0);
1428 trace.set(Column::alu_ab_diff_inv, 0, ab_diff_inv + 1);
1435 MemoryValue::from_tag(MemoryTag::U1, 0), MemoryValue::from_tag(MemoryTag::U1, 0),
1436 MemoryValue::from_tag(MemoryTag::U1, 0), MemoryValue::from_tag(MemoryTag::U1, 1),
1437 MemoryValue::from_tag(MemoryTag::U1, 0), MemoryValue::from_tag(MemoryTag::U1, 0),
1438 MemoryValue::from_tag(MemoryTag::U1, 0),
1443class AluLTConstrainingTest :
public AluConstrainingTest,
public ::testing::WithParamInterface<ThreeOperandTestParams> {
1445 TestTraceContainer process_lt_trace(ThreeOperandTestParams params)
1447 auto [
a,
b, c] = params;
1448 auto mem_tag =
a.get_tag();
1449 auto tag =
static_cast<uint8_t
>(mem_tag);
1450 auto is_ff = mem_tag == MemoryTag::FF;
1452 auto trace = TestTraceContainer({
1455 { C::alu_ia_tag,
tag },
1457 { C::alu_ib_tag,
tag },
1459 { C::alu_ic_tag,
static_cast<uint8_t
>(MemoryTag::U1) },
1460 { C::alu_gt_input_a,
b },
1461 { C::alu_gt_input_b,
a },
1462 { C::alu_gt_result_c, c },
1467 { C::alu_sel_ff_gt,
static_cast<uint8_t
>(is_ff) },
1468 { C::alu_sel_int_gt,
static_cast<uint8_t
>(!is_ff) },
1469 { C::alu_sel_is_ff,
static_cast<uint8_t
>(is_ff) },
1470 { C::alu_sel_op_lt, 1 },
1471 { C::alu_tag_ff_diff_inv, is_ff ? 0 :
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
1472 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
1473 { C::alu_tag_u128_diff_inv,
1474 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
1477 { C::execution_mem_tag_reg_0_,
tag },
1478 { C::execution_mem_tag_reg_1_,
tag },
1479 { C::execution_mem_tag_reg_2_,
static_cast<uint8_t
>(MemoryTag::U1) },
1480 { C::execution_register_0_,
a },
1481 { C::execution_register_1_,
b },
1482 { C::execution_register_2_, c },
1483 { C::execution_sel_exec_dispatch_alu, 1 },
1493 .result = c.as_ff() == 1 } },
1502 TestTraceContainer process_lt_with_tracegen(ThreeOperandTestParams params)
1504 TestTraceContainer
trace;
1505 auto [
a,
b, c] = params;
1506 auto is_ff =
a.get_tag() == MemoryTag::FF;
1510 { .operation = simulation::AluOperation::LT, .a =
a, .b =
b, .c = c },
1519 .result = c.as_ff() == 1 } },
1530TEST_P(AluLTConstrainingTest, AluLT)
1532 auto trace = process_lt_trace(GetParam());
1533 check_all_interactions<AluTraceBuilder>(trace);
1534 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1535 check_relation<alu>(trace);
1538TEST_P(AluLTConstrainingTest, AluLTTraceGen)
1540 auto trace = process_lt_with_tracegen(GetParam());
1541 check_all_interactions<AluTraceBuilder>(trace);
1542 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1543 check_relation<alu>(trace);
1546TEST_P(AluLTConstrainingTest, NegativeAluLT)
1548 auto params = GetParam();
1549 auto trace = process_lt_trace(params);
1550 auto is_ff =
std::get<0>(params).get_tag() == MemoryTag::FF;
1551 check_relation<alu>(trace);
1552 check_all_interactions<AluTraceBuilder>(trace);
1553 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1554 bool c =
trace.get(Column::alu_ic, 0) == 1;
1556 trace.set(Column::alu_ic, 0,
static_cast<uint8_t
>(!c));
1558 trace.set(Column::alu_gt_result_c, 0,
static_cast<uint8_t
>(!c));
1562 "LOOKUP_ALU_FF_GT");
1565 "LOOKUP_ALU_INT_GT");
1573class AluLTEConstrainingTest :
public AluConstrainingTest,
1574 public ::testing::WithParamInterface<ThreeOperandTestParams> {
1576 TestTraceContainer process_lte_trace(ThreeOperandTestParams params,
bool eq =
false)
1578 auto [
a,
b, c] = params;
1579 auto mem_tag =
a.get_tag();
1580 auto tag =
static_cast<uint8_t
>(mem_tag);
1581 auto is_ff = mem_tag == MemoryTag::FF;
1583 c = eq ? MemoryValue::from_tag(MemoryTag::U1, 1) : c;
1585 auto trace = TestTraceContainer({
1588 { C::alu_ia_tag,
tag },
1590 { C::alu_ib_tag,
tag },
1592 { C::alu_ic_tag,
static_cast<uint8_t
>(MemoryTag::U1) },
1593 { C::alu_gt_input_a,
a },
1594 { C::alu_gt_input_b,
b },
1595 { C::alu_gt_result_c, c.as_ff() == 0 ? 1 : 0 },
1600 { C::alu_sel_ff_gt,
static_cast<uint8_t
>(is_ff) },
1601 { C::alu_sel_int_gt,
static_cast<uint8_t
>(!is_ff) },
1602 { C::alu_sel_is_ff,
static_cast<uint8_t
>(is_ff) },
1603 { C::alu_sel_op_lte, 1 },
1604 { C::alu_tag_ff_diff_inv, is_ff ? 0 :
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
1605 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
1606 { C::alu_tag_u128_diff_inv,
1607 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
1610 { C::execution_mem_tag_reg_0_,
tag },
1611 { C::execution_mem_tag_reg_1_,
tag },
1612 { C::execution_mem_tag_reg_2_,
static_cast<uint8_t
>(MemoryTag::U1) },
1613 { C::execution_register_0_,
a },
1614 { C::execution_register_1_,
b },
1615 { C::execution_register_2_, c },
1616 { C::execution_sel_exec_dispatch_alu, 1 },
1626 .result = c.as_ff() == 0 } },
1634 TestTraceContainer process_lte_with_tracegen(ThreeOperandTestParams params,
bool eq =
false)
1636 TestTraceContainer
trace;
1637 auto [
a,
b, c] = params;
1638 auto is_ff =
a.get_tag() == MemoryTag::FF;
1640 c = eq ? MemoryValue::from_tag(MemoryTag::U1, 1) : c;
1644 { .operation = simulation::AluOperation::LTE, .a =
a, .b =
b, .c = c },
1653 .result = c.as_ff() == 0 } },
1664TEST_P(AluLTEConstrainingTest, AluLTE)
1666 auto trace = process_lte_trace(GetParam());
1667 check_all_interactions<AluTraceBuilder>(trace);
1668 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1669 check_relation<alu>(trace);
1672TEST_P(AluLTEConstrainingTest, AluLTEEq)
1674 auto trace = process_lte_trace(GetParam(),
true);
1675 check_all_interactions<AluTraceBuilder>(trace);
1676 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1677 check_relation<alu>(trace);
1680TEST_P(AluLTEConstrainingTest, AluLTETraceGen)
1682 auto trace = process_lte_with_tracegen(GetParam());
1683 check_all_interactions<AluTraceBuilder>(trace);
1684 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1685 check_relation<alu>(trace);
1688TEST_P(AluLTEConstrainingTest, AluLTEEqTraceGen)
1690 auto trace = process_lte_with_tracegen(GetParam(),
true);
1691 check_all_interactions<AluTraceBuilder>(trace);
1692 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1693 check_relation<alu>(trace);
1696TEST_P(AluLTEConstrainingTest, NegativeAluLTEResult)
1698 auto params = GetParam();
1700 for (
const bool is_eq : {
false,
true }) {
1701 auto trace = process_lte_trace(params, is_eq);
1702 auto is_ff =
std::get<0>(params).get_tag() == MemoryTag::FF;
1703 check_relation<alu>(trace);
1704 check_all_interactions<AluTraceBuilder>(trace);
1705 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1706 bool c =
trace.get(Column::alu_ic, 0) == 1;
1708 trace.set(Column::alu_ic, 0,
static_cast<uint8_t
>(!c));
1710 trace.set(Column::alu_gt_result_c, 0,
static_cast<uint8_t
>(c));
1714 "LOOKUP_ALU_FF_GT");
1717 "LOOKUP_ALU_INT_GT");
1722TEST_P(AluLTEConstrainingTest, NegativeAluLTEInput)
1724 auto params = GetParam();
1726 for (
const bool is_eq : {
false,
true }) {
1727 auto trace = process_lte_trace(params, is_eq);
1728 auto is_ff =
std::get<0>(params).get_tag() == MemoryTag::FF;
1729 check_relation<alu>(trace);
1730 check_all_interactions<AluTraceBuilder>(trace);
1731 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1732 bool c =
trace.get(Column::alu_ic, 0) == 1;
1733 auto a =
trace.get(Column::alu_ia, 0);
1734 auto wrong_b = c ?
a - 1 :
a + 1;
1735 trace.set(Column::alu_ib, 0, wrong_b);
1736 trace.set(Column::alu_gt_input_b, 0, wrong_b);
1738 check_relation<alu>(trace);
1744 "LOOKUP_ALU_FF_GT");
1747 "LOOKUP_ALU_INT_GT");
1755 MemoryValue::from_tag(MemoryTag::U1, 0),
1756 MemoryValue::from_tag(MemoryTag::U8, 55),
1757 MemoryValue::from_tag(MemoryTag::U16, 65505),
1758 MemoryValue::from_tag(MemoryTag::U32, 9),
1759 MemoryValue::from_tag(MemoryTag::U64, 9),
1760 MemoryValue::from_tag(MemoryTag::U128, 9),
1761 MemoryValue::from_tag(
static_cast<MemoryTag>(0), 0),
1766class AluNotConstrainingTest :
public AluConstrainingTest,
public ::testing::WithParamInterface<TwoOperandTestParams> {
1768 TestTraceContainer process_not_with_tracegen(
const TwoOperandTestParams& params,
bool error =
false)
1770 TestTraceContainer
trace;
1771 auto [
a,
b] = params;
1772 auto is_ff =
a.get_tag() == MemoryTag::FF;
1776 { .operation = simulation::AluOperation::NOT, .a =
a, .b =
b, .error = error || is_ff },
1788TEST_P(AluNotConstrainingTest, AluNotTraceGen)
1790 auto trace = process_not_with_tracegen(GetParam());
1791 check_all_interactions<AluTraceBuilder>(trace);
1792 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1793 check_relation<alu>(trace);
1796TEST_P(AluNotConstrainingTest, NegativeAluNotTraceGen)
1798 auto params = GetParam();
1799 auto trace = process_not_with_tracegen(params);
1800 check_all_interactions<AluTraceBuilder>(trace);
1801 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1802 check_relation<alu>(trace);
1803 trace.set(Column::alu_ib, 0,
trace.get(Column::alu_ib, 0) + 1);
1805 if (
std::get<0>(params).get_tag() != MemoryTag::FF) {
1811TEST_P(AluNotConstrainingTest, AluNotTraceGenTagError)
1813 auto [
a,
b] = GetParam();
1814 auto trace = process_not_with_tracegen(
1815 TwoOperandTestParams{
a, MemoryValue::from_tag(TAG_ERROR_TEST_VALUES.at(
b.get_tag()),
b.as_ff()) },
true);
1816 check_all_interactions<AluTraceBuilder>(trace);
1817 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1822TEST_F(AluNotConstrainingTest, AluNotTraceGenTagErrorFF)
1824 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1825 auto b = MemoryValue::from_tag(MemoryTag::FF, 253);
1826 auto trace = process_not_with_tracegen(TwoOperandTestParams{
a,
b },
true);
1827 check_all_interactions<AluTraceBuilder>(trace);
1828 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1829 check_relation<alu>(trace);
1835 MemoryValue::from_tag(MemoryTag::U1, 1),
1836 MemoryValue::from_tag(MemoryTag::U8, 0),
1837 MemoryValue::from_tag(MemoryTag::U16, 0),
1838 MemoryValue::from_tag(MemoryTag::U32, 0xfffffec0),
1839 MemoryValue::from_tag(MemoryTag::U64, 0xfffffffffffffec0ULL),
1840 MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 128) - 320),
1845class AluShlConstrainingTest :
public AluConstrainingTest,
1846 public ::testing::WithParamInterface<ThreeOperandTestParams> {
1848 TestTraceContainer process_shl_trace(ThreeOperandTestParams params)
1850 auto [
a,
b, c] = params;
1852 auto mem_tag =
a.get_tag();
1853 auto tag =
static_cast<uint8_t
>(mem_tag);
1855 auto a_num =
static_cast<uint128_t>(
a.as_ff());
1856 auto b_num =
static_cast<uint128_t>(
b.as_ff());
1858 auto overflow = b_num > tag_bits;
1859 uint128_t shift_lo_bits = overflow ? tag_bits : tag_bits - b_num;
1860 uint128_t shift_hi_bits = overflow ? tag_bits : b_num;
1861 auto two_pow_shift_lo_bits =
static_cast<uint128_t>(1) << shift_lo_bits;
1862 auto a_lo = overflow ? b_num - tag_bits : a_num % two_pow_shift_lo_bits;
1863 auto a_hi = a_num >> shift_lo_bits;
1865 auto trace = TestTraceContainer({
1867 { C::alu_a_hi, a_hi },
1868 { C::alu_a_hi_bits, shift_hi_bits },
1869 { C::alu_a_lo, a_lo },
1870 { C::alu_a_lo_bits, shift_lo_bits },
1871 { C::alu_helper1,
static_cast<uint128_t>(1) << b_num },
1873 { C::alu_ia_tag,
tag },
1875 { C::alu_ib_tag,
tag },
1877 { C::alu_ic_tag,
tag },
1878 { C::alu_max_bits, tag_bits },
1882 { C::alu_sel_decompose_a, 1 },
1883 { C::alu_sel_op_shl, 1 },
1884 { C::alu_sel_shift_ops_no_overflow, overflow ? 0 : 1 },
1885 { C::alu_shift_lo_bits, shift_lo_bits },
1886 { C::alu_tag_ff_diff_inv,
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
1887 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
1888 { C::alu_tag_u128_diff_inv,
1889 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
1892 { C::alu_two_pow_shift_lo_bits, two_pow_shift_lo_bits },
1893 { C::execution_mem_tag_reg_0_,
tag },
1894 { C::execution_mem_tag_reg_1_,
tag },
1895 { C::execution_mem_tag_reg_2_,
tag },
1896 { C::execution_register_0_,
a },
1897 { C::execution_register_1_,
b },
1898 { C::execution_register_2_, c },
1899 { C::execution_sel_exec_dispatch_alu, 1 },
1908 range_check_builder.process({ { .value = a_lo, .num_bits =
static_cast<uint8_t
>(shift_lo_bits) },
1909 { .value = a_hi, .num_bits =
static_cast<uint8_t
>(shift_hi_bits) } },
1915 TestTraceContainer process_shl_with_tracegen(ThreeOperandTestParams params,
bool error =
false)
1917 TestTraceContainer
trace;
1918 auto [
a,
b, c] = params;
1919 auto b_num =
static_cast<uint128_t>(
b.as_ff());
1921 auto overflow = b_num > tag_bits;
1922 uint128_t shift_lo_bits = overflow ? tag_bits : tag_bits - b_num;
1923 auto a_lo = overflow ? b_num - tag_bits
1928 { .operation = simulation::AluOperation::SHL, .a =
a, .b =
b, .c = c, .error = error },
1935 range_check_builder.process({ { .value = a_lo, .num_bits =
static_cast<uint8_t
>(shift_lo_bits) },
1936 { .value =
static_cast<uint128_t>(
a.as_ff()) >> shift_lo_bits,
1937 .num_bits =
static_cast<uint8_t
>(overflow ? tag_bits : b_num) } },
1945TEST_P(AluShlConstrainingTest, AluShl)
1947 auto trace = process_shl_trace(GetParam());
1948 check_all_interactions<AluTraceBuilder>(trace);
1949 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1950 check_relation<alu>(trace);
1953TEST_P(AluShlConstrainingTest, AluShlTraceGen)
1955 auto trace = process_shl_with_tracegen(GetParam());
1956 check_all_interactions<AluTraceBuilder>(trace);
1957 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
1958 check_relation<alu>(trace);
1961TEST_F(AluShlConstrainingTest, NegativeAluShlFF)
1963 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
1964 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
1965 auto c = MemoryValue::from_tag(MemoryTag::FF, 2 << 5);
1966 auto trace = process_shl_with_tracegen({
a,
b, c },
true);
1967 check_relation<alu>(trace);
1968 check_all_interactions<AluTraceBuilder>(trace);
1971 trace.set(Column::alu_ib, 0, 0);
1972 check_relation<alu>(trace);
1973 check_all_interactions<AluTraceBuilder>(trace);
1976 trace.set(Column::alu_sel_tag_err, 0, 0);
1977 trace.set(Column::alu_sel_err, 0, 0);
1981TEST_F(AluShlConstrainingTest, NegativeAluShlTagMismatchOverflow)
1983 auto a = MemoryValue::from_tag(MemoryTag::U8, 2);
1984 auto b = MemoryValue::from_tag(MemoryTag::U32, 256);
1985 auto c = MemoryValue::from_tag(MemoryTag::U8, 0);
1986 auto trace = process_shl_with_tracegen({
a,
b, c },
true);
1987 check_relation<alu>(trace);
1988 check_all_interactions<AluTraceBuilder>(trace);
1991 trace.set(Column::alu_sel_tag_err, 0, 0);
1992 trace.set(Column::alu_sel_err, 0, 0);
1994 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 0);
1998 trace.set(Column::alu_ab_tags_diff_inv, 0, 0);
2002 trace.set(Column::alu_ab_tags_diff_inv,
2004 (
FF(
static_cast<uint8_t
>(MemoryTag::U8)) -
FF(
static_cast<uint8_t
>(MemoryTag::U32))).invert());
2005 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
2012 MemoryValue::from_tag(MemoryTag::U1, 1),
2013 MemoryValue::from_tag(MemoryTag::U8, 0),
2014 MemoryValue::from_tag(MemoryTag::U16, 0),
2015 MemoryValue::from_tag(MemoryTag::U32, 0x7ffffff),
2016 MemoryValue::from_tag(MemoryTag::U64, 0x7ffffffffffffffULL),
2017 MemoryValue::from_tag(MemoryTag::U128,
2023class AluShrConstrainingTest :
public AluConstrainingTest,
2024 public ::testing::WithParamInterface<ThreeOperandTestParams> {
2026 TestTraceContainer process_shr_trace(ThreeOperandTestParams params)
2028 auto [
a,
b, c] = params;
2030 auto mem_tag =
a.get_tag();
2031 auto tag =
static_cast<uint8_t
>(mem_tag);
2033 auto a_num =
static_cast<uint128_t>(
a.as_ff());
2034 auto b_num =
static_cast<uint128_t>(
b.as_ff());
2036 auto overflow = b_num > tag_bits;
2037 uint128_t shift_lo_bits = overflow ? tag_bits : b_num;
2038 uint128_t shift_hi_bits = overflow ? tag_bits : tag_bits - b_num;
2039 auto two_pow_shift_lo_bits =
static_cast<uint128_t>(1) << shift_lo_bits;
2040 auto a_lo = overflow ? b_num - tag_bits : a_num % two_pow_shift_lo_bits;
2041 auto a_hi = a_num >> shift_lo_bits;
2043 auto trace = TestTraceContainer({
2045 { C::alu_a_hi, a_hi },
2046 { C::alu_a_hi_bits, shift_hi_bits },
2047 { C::alu_a_lo, a_lo },
2048 { C::alu_a_lo_bits, shift_lo_bits },
2050 { C::alu_ia_tag,
tag },
2052 { C::alu_ib_tag,
tag },
2054 { C::alu_ic_tag,
tag },
2055 { C::alu_max_bits, tag_bits },
2059 { C::alu_sel_decompose_a, 1 },
2060 { C::alu_sel_op_shr, 1 },
2061 { C::alu_sel_shift_ops_no_overflow, overflow ? 0 : 1 },
2062 { C::alu_shift_lo_bits, shift_lo_bits },
2063 { C::alu_tag_ff_diff_inv,
FF(
tag -
static_cast<uint8_t
>(MemoryTag::FF)).
invert() },
2064 { C::alu_sel_is_u128,
tag ==
static_cast<uint8_t
>(MemoryTag::U128) ? 1 : 0 },
2065 { C::alu_tag_u128_diff_inv,
2066 tag ==
static_cast<uint8_t
>(MemoryTag::U128)
2069 { C::alu_two_pow_shift_lo_bits, two_pow_shift_lo_bits },
2070 { C::execution_mem_tag_reg_0_,
tag },
2071 { C::execution_mem_tag_reg_1_,
tag },
2072 { C::execution_mem_tag_reg_2_,
tag },
2073 { C::execution_register_0_,
a },
2074 { C::execution_register_1_,
b },
2075 { C::execution_register_2_, c },
2076 { C::execution_sel_exec_dispatch_alu, 1 },
2085 range_check_builder.process({ { .value = a_lo, .num_bits =
static_cast<uint8_t
>(shift_lo_bits) },
2086 { .value = a_hi, .num_bits =
static_cast<uint8_t
>(shift_hi_bits) } },
2092 TestTraceContainer process_shr_with_tracegen(ThreeOperandTestParams params,
bool error =
false)
2094 TestTraceContainer
trace;
2095 auto [
a,
b, c] = params;
2096 auto b_num =
static_cast<uint128_t>(
b.as_ff());
2098 auto overflow = b_num > tag_bits;
2099 uint128_t shift_lo_bits = overflow ? tag_bits : b_num;
2100 auto a_lo = overflow ? b_num - tag_bits
2105 { .operation = simulation::AluOperation::SHR, .a =
a, .b =
b, .c = c, .error = error },
2112 range_check_builder.process({ { .value = a_lo, .num_bits =
static_cast<uint8_t
>(shift_lo_bits) },
2113 { .value =
static_cast<uint128_t>(
a.as_ff()) >> shift_lo_bits,
2114 .num_bits =
static_cast<uint8_t
>(overflow ? tag_bits : tag_bits - b_num) } },
2122TEST_P(AluShrConstrainingTest, AluShr)
2124 auto trace = process_shr_trace(GetParam());
2125 check_all_interactions<AluTraceBuilder>(trace);
2126 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
2127 check_relation<alu>(trace);
2130TEST_P(AluShrConstrainingTest, AluShrTraceGen)
2132 auto trace = process_shr_with_tracegen(GetParam());
2133 check_all_interactions<AluTraceBuilder>(trace);
2134 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_alu_settings>(trace);
2135 check_relation<alu>(trace);
2138TEST_F(AluShrConstrainingTest, NegativeAluShrFF)
2140 auto a = MemoryValue::from_tag(MemoryTag::FF, 2);
2141 auto b = MemoryValue::from_tag(MemoryTag::FF, 5);
2142 auto c = MemoryValue::from_tag(MemoryTag::FF, 2 << 5);
2143 auto trace = process_shr_with_tracegen({
a,
b, c },
true);
2144 check_relation<alu>(trace);
2145 check_all_interactions<AluTraceBuilder>(trace);
2148 trace.set(Column::alu_ib, 0, 0);
2149 check_relation<alu>(trace);
2150 check_all_interactions<AluTraceBuilder>(trace);
2153 trace.set(Column::alu_sel_tag_err, 0, 0);
2154 trace.set(Column::alu_sel_err, 0, 0);
2158TEST_F(AluShrConstrainingTest, NegativeAluShrTagMismatchOverflow)
2160 auto a = MemoryValue::from_tag(MemoryTag::U16, 2);
2161 auto b = MemoryValue::from_tag(MemoryTag::U64, 123456);
2162 auto c = MemoryValue::from_tag(MemoryTag::U16, 0);
2163 auto trace = process_shr_with_tracegen({
a,
b, c },
true);
2164 check_relation<alu>(trace);
2165 check_all_interactions<AluTraceBuilder>(trace);
2167 trace.set(Column::alu_sel_tag_err, 0, 0);
2168 trace.set(Column::alu_sel_err, 0, 0);
2170 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 0);
2174 trace.set(Column::alu_ab_tags_diff_inv, 0, 0);
2178 trace.set(Column::alu_ab_tags_diff_inv,
2180 (
FF(
static_cast<uint8_t
>(MemoryTag::U16)) -
FF(
static_cast<uint8_t
>(MemoryTag::U64))).invert());
2181 trace.set(Column::alu_sel_ab_tag_mismatch, 0, 1);
2190 { MemoryValue::from_tag(MemoryTag::FF, 1),
2191 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U1)),
2192 MemoryValue::from_tag(MemoryTag::U1, 1) },
2193 { MemoryValue::from_tag(MemoryTag::FF, 42),
2194 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U8)),
2195 MemoryValue::from_tag(MemoryTag::U8, 42) },
2196 { MemoryValue::from_tag(MemoryTag::FF, 12345),
2197 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U16)),
2198 MemoryValue::from_tag(MemoryTag::U16, 12345) },
2199 { MemoryValue::from_tag(MemoryTag::FF, 123456789),
2200 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2201 MemoryValue::from_tag(MemoryTag::U32, 123456789) },
2202 { MemoryValue::from_tag(MemoryTag::FF, 1234567890123456789ULL),
2203 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2204 MemoryValue::from_tag(MemoryTag::U64, 1234567890123456789ULL) },
2205 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(1) << 127) + 23423429816234ULL),
2206 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U128)),
2207 MemoryValue::from_tag(MemoryTag::U128, (
uint256_t(1) << 127) + 23423429816234ULL) },
2208 { MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 3),
2209 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::FF)),
2210 MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 3) },
2212 { MemoryValue::from_tag(MemoryTag::FF, 212),
2213 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U1)),
2214 MemoryValue::from_tag(MemoryTag::U1, 0) },
2215 { MemoryValue::from_tag(MemoryTag::FF, 257),
2216 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U8)),
2217 MemoryValue::from_tag(MemoryTag::U8, 1) },
2218 { MemoryValue::from_tag(MemoryTag::FF, 65540),
2219 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U16)),
2220 MemoryValue::from_tag(MemoryTag::U16, 4) },
2221 { MemoryValue::from_tag(MemoryTag::FF, 4294967298ULL),
2222 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2223 MemoryValue::from_tag(MemoryTag::U32, 2) },
2224 { MemoryValue::from_tag(MemoryTag::FF, 18446744073709551615ULL + 4),
2225 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2226 MemoryValue::from_tag(MemoryTag::U64, 3) },
2228 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 129) + 986132),
2229 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U1)),
2230 MemoryValue::from_tag(MemoryTag::U1, 0) },
2231 { MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 128735618772ULL),
2232 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U8)),
2233 MemoryValue::from_tag(MemoryTag::U8, 45) },
2234 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(999) << 128) - 986132ULL),
2235 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U16)),
2236 MemoryValue::from_tag(MemoryTag::U16, 62444) },
2237 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 198) + 986132ULL),
2238 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2239 MemoryValue::from_tag(MemoryTag::U32, 986132ULL) },
2240 { MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 198) - 986132ULL),
2241 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2242 MemoryValue::from_tag(MemoryTag::U64,
static_cast<uint64_t
>(-986132ULL)) },
2243 { MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 8723),
2244 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U128)),
2248class AluTruncateConstrainingTest :
public AluConstrainingTest,
2249 public ::testing::WithParamInterface<ThreeOperandTestParams> {
2251 TestTraceContainer process_truncate_with_tracegen(
const ThreeOperandTestParams& params, TestTraceContainer& trace)
2253 auto [
a,
b, c] = params;
2257 { .operation = simulation::AluOperation::TRUNCATE, .a =
a, .b =
b, .c = c },
2264 auto is_non_trivial =
trace.get(Column::alu_sel_trunc_non_trivial, 0) == 1;
2266 if (is_non_trivial) {
2267 auto a_decomp = simulation::decompose_256(
static_cast<uint256_t>(
a.as_ff()));
2271 .num_bits =
static_cast<uint8_t
>(128 - bits) } },
2273 auto is_gte_128 =
trace.get(Column::alu_sel_trunc_gte_128, 0) == 1;
2275 auto p_limbs = simulation::decompose_256(
FF::modulus);
2276 simulation::LimbsComparisonWitness p_sub_a_witness = { .lo = p_limbs.lo - a_decomp.lo,
2277 .hi = p_limbs.hi - a_decomp.hi,
2279 field_gt_builder.process({ { .operation = simulation::FieldGreaterOperation::CANONICAL_DECOMPOSITION,
2281 .a_limbs = a_decomp,
2282 .p_sub_a_witness = p_sub_a_witness } },
2290 TestTraceContainer process_set_with_tracegen(
const ThreeOperandTestParams& params)
2292 TestTraceContainer
trace;
2293 auto [
a,
b, _c] = params;
2295 auto c = MemoryValue::from_tag_truncating(
dst_tag,
a);
2298 { Column::execution_sel_exec_dispatch_set, 1 },
2299 { Column::execution_rop_2_,
a },
2300 { Column::execution_rop_1_,
static_cast<uint8_t
>(
dst_tag) },
2302 { Column::execution_register_0_, c.as_ff() },
2303 { Column::execution_mem_tag_reg_0_,
static_cast<uint8_t
>(
dst_tag) },
2306 process_truncate_with_tracegen(params, trace);
2311 TestTraceContainer process_cast_with_tracegen(
const ThreeOperandTestParams& params)
2313 TestTraceContainer
trace;
2314 auto [
a,
b, _c] = params;
2316 auto c = MemoryValue::from_tag_truncating(
dst_tag,
a);
2319 { Column::execution_sel_exec_dispatch_cast, 1 },
2320 { Column::execution_register_0_,
a },
2321 { Column::execution_rop_2_,
static_cast<uint8_t
>(
dst_tag) },
2323 { Column::execution_register_1_, c.as_ff() },
2324 { Column::execution_mem_tag_reg_1_,
static_cast<uint8_t
>(
dst_tag) },
2327 process_truncate_with_tracegen(params, trace);
2333INSTANTIATE_TEST_SUITE_P(AluConstrainingTest, AluTruncateConstrainingTest, ::testing::ValuesIn(TEST_VALUES_TRUNCATE));
2335TEST_P(AluTruncateConstrainingTest, AluSet)
2337 auto trace = process_set_with_tracegen(GetParam());
2338 check_all_interactions<AluTraceBuilder>(trace);
2339 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_set_settings>(trace);
2340 check_relation<alu>(trace);
2343TEST_P(AluTruncateConstrainingTest, AluCast)
2345 auto trace = process_cast_with_tracegen(GetParam());
2346 check_all_interactions<AluTraceBuilder>(trace);
2347 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_set_settings>(trace);
2348 check_relation<alu>(trace);
2351TEST_P(AluTruncateConstrainingTest, NegativeTruncateWrongTrivialCase)
2353 TestTraceContainer
trace;
2354 process_truncate_with_tracegen(GetParam(), trace);
2355 check_relation<alu>(trace);
2356 bool is_trivial =
trace.get(Column::alu_sel_trunc_trivial, 0) == 1;
2357 trace.set(Column::alu_sel_trunc_trivial, 0,
static_cast<uint8_t
>(!is_trivial));
2359 trace.set(Column::alu_sel_trunc_trivial, 0,
static_cast<uint8_t
>(is_trivial));
2360 trace.set(Column::alu_sel_trunc_non_trivial, 0,
static_cast<uint8_t
>(is_trivial));
2364TEST_P(AluTruncateConstrainingTest, NegativeTruncateWrong128BitsCase)
2366 TestTraceContainer
trace;
2367 process_truncate_with_tracegen(GetParam(), trace);
2368 check_relation<alu>(trace);
2369 bool is_lt_128 =
trace.get(Column::alu_sel_trunc_lt_128, 0) == 1;
2370 trace.set(Column::alu_sel_trunc_lt_128, 0,
static_cast<uint8_t
>(!is_lt_128));
2372 trace.set(Column::alu_sel_trunc_lt_128, 0,
static_cast<uint8_t
>(is_lt_128));
2373 check_relation<alu>(trace);
2374 bool is_gte_128 =
trace.get(Column::alu_sel_trunc_gte_128, 0) == 1;
2375 trace.set(Column::alu_sel_trunc_gte_128, 0,
static_cast<uint8_t
>(!is_gte_128));
2379TEST_F(AluTruncateConstrainingTest, NegativeTruncateWrongMid)
2381 TestTraceContainer
trace;
2382 process_truncate_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, 4294967298ULL),
2383 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2384 MemoryValue::from_tag(MemoryTag::U32, 2) },
2386 check_relation<alu>(trace);
2387 trace.set(Column::alu_mid, 0, 1234ULL);
2391TEST_F(AluTruncateConstrainingTest, NegativeTruncateWrongMidBits)
2393 TestTraceContainer
trace;
2394 process_truncate_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF,
FF::modulus - 2),
2395 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U1)),
2396 MemoryValue::from_tag(MemoryTag::U1, 1) },
2398 check_relation<alu>(trace);
2399 trace.set(Column::alu_mid_bits, 0, 32);
2403TEST_F(AluTruncateConstrainingTest, NegativeTruncateWrongLo128FromCanonDec)
2405 TestTraceContainer
trace;
2406 process_truncate_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 198) - 986132ULL),
2407 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2408 MemoryValue::from_tag(MemoryTag::U64,
static_cast<uint64_t
>(-986132ULL)) },
2410 check_relation<alu>(trace);
2411 check_all_interactions<AluTraceBuilder>(trace);
2412 trace.set(Column::alu_a_lo, 0, 1234ULL);
2414 (check_interaction<AluTraceBuilder, lookup_alu_large_trunc_canonical_dec_settings>(trace)),
2415 "Failed.*LARGE_TRUNC_CANONICAL_DEC. Could not find tuple in destination.");
2418TEST_F(AluTruncateConstrainingTest, NegativeTruncateWrongMidIntoRangeCheck)
2420 TestTraceContainer
trace;
2421 process_truncate_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, (
uint256_t(134534) << 198) - 986132ULL),
2422 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U64)),
2423 MemoryValue::from_tag(MemoryTag::U64,
static_cast<uint64_t
>(-986132ULL)) },
2425 check_relation<alu>(trace);
2426 check_all_interactions<AluTraceBuilder>(trace);
2427 trace.set(Column::alu_mid, 0, 1234ULL);
2429 "Failed.*RANGE_CHECK_TRUNC_MID. Could not find tuple in destination.");
2432TEST_F(AluTruncateConstrainingTest, NegativeCastWrongDispatching)
2435 process_cast_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, 4294967298ULL),
2436 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2437 MemoryValue::from_tag(MemoryTag::U32, 2) });
2438 check_relation<alu>(trace);
2439 check_all_interactions<AluTraceBuilder>(trace);
2440 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_cast_settings>(trace);
2441 trace.set(Column::execution_register_0_, 0,
trace.get(Column::execution_register_0_, 0) + 1);
2443 (check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_cast_settings>(trace)),
2444 "Failed.*EXECUTION_DISPATCH_TO_CAST. Could not find tuple in destination.");
2447TEST_F(AluTruncateConstrainingTest, NegativeSetWrongDispatching)
2449 auto trace = process_set_with_tracegen({ MemoryValue::from_tag(MemoryTag::FF, 4294967298ULL),
2450 MemoryValue::from_tag(MemoryTag::FF,
static_cast<uint8_t
>(MemoryTag::U32)),
2451 MemoryValue::from_tag(MemoryTag::U32, 2) });
2452 check_relation<alu>(trace);
2453 check_all_interactions<AluTraceBuilder>(trace);
2454 check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_set_settings>(trace);
2455 trace.set(Column::execution_rop_2_, 0,
trace.get(Column::execution_rop_2_, 0) + 1);
2457 (check_interaction<ExecutionTraceBuilder, lookup_execution_dispatch_to_set_settings>(trace)),
2458 "Failed.*EXECUTION_DISPATCH_TO_SET. Could not find tuple in destination.");
INSTANTIATE_TEST_SUITE_P(AcirTests, AcirIntegrationSingleTest, testing::Values("a_1327_concrete_in_generic", "a_1_mul", "a_2_div", "a_3_add", "a_4_sub", "a_5_over", "a_6", "a_6_array", "a_7", "a_7_function", "aes128_encrypt", "arithmetic_binary_operations", "array_dynamic", "array_dynamic_blackbox_input", "array_dynamic_main_output", "array_dynamic_nested_blackbox_input", "array_eq", "array_if_cond_simple", "array_len", "array_neq", "array_sort", "array_to_slice", "array_to_slice_constant_length", "assert", "assert_statement", "assign_ex", "bigint", "bit_and", "bit_not", "bit_shifts_comptime", "bit_shifts_runtime", "blake3", "bool_not", "bool_or", "break_and_continue", "brillig_acir_as_brillig", "brillig_array_eq", "brillig_array_to_slice", "brillig_arrays", "brillig_assert", "brillig_bit_shifts_runtime", "brillig_blake2s", "brillig_blake3", "brillig_calls", "brillig_calls_array", "brillig_calls_conditionals", "brillig_conditional", "brillig_cow", "brillig_cow_assign", "brillig_cow_regression", "brillig_ecdsa_secp256k1", "brillig_ecdsa_secp256r1", "brillig_embedded_curve", "brillig_fns_as_values", "brillig_hash_to_field", "brillig_identity_function", "brillig_keccak", "brillig_loop", "brillig_nested_arrays", "brillig_not", "brillig_oracle", "brillig_pedersen", "brillig_recursion", "brillig_references", "brillig_schnorr", "brillig_sha256", "brillig_signed_cmp", "brillig_signed_div", "brillig_slices", "brillig_to_be_bytes", "brillig_to_bits", "brillig_to_bytes_integration", "brillig_to_le_bytes", "brillig_top_level", "brillig_uninitialized_arrays", "brillig_wrapping", "cast_bool", "closures_mut_ref", "conditional_1", "conditional_2", "conditional_regression_421", "conditional_regression_547", "conditional_regression_661", "conditional_regression_short_circuit", "conditional_regression_underflow", "custom_entry", "databus", "debug_logs", "diamond_deps_0", "double_verify_nested_proof", "double_verify_proof", "ecdsa_secp256k1", "ecdsa_secp256r1", "ecdsa_secp256r1_3x", "eddsa", "embedded_curve_ops", "field_attribute", "generics", "global_consts", "hash_to_field", "hashmap", "higher_order_functions", "if_else_chain", "import", "inline_never_basic", "integer_array_indexing", "keccak256", "main_bool_arg", "main_return", "merkle_insert", "missing_closure_env", "modules", "modules_more", "modulus", "nested_array_dynamic", "nested_array_dynamic_simple", "nested_array_in_slice", "nested_arrays_from_brillig", "no_predicates_basic", "no_predicates_brillig", "no_predicates_numeric_generic_poseidon", "operator_overloading", "pedersen_check", "pedersen_commitment", "pedersen_hash", "poseidon_bn254_hash", "poseidonsponge_x5_254", "pred_eq", "prelude", "references", "regression", "regression_2660", "regression_3051", "regression_3394", "regression_3607", "regression_3889", "regression_4088", "regression_4124", "regression_4202", "regression_4449", "regression_4709", "regression_5045", "regression_capacity_tracker", "regression_mem_op_predicate", "regression_method_cannot_be_found", "regression_struct_array_conditional", "schnorr", "sha256", "sha2_byte", "side_effects_constrain_array", "signed_arithmetic", "signed_comparison", "signed_division", "simple_2d_array", "simple_add_and_ret_arr", "simple_array_param", "simple_bitwise", "simple_comparison", "simple_mut", "simple_not", "simple_print", "simple_program_addition", "simple_radix", "simple_shield", "simple_shift_left_right", "slice_coercion", "slice_dynamic_index", "slice_loop", "slices", "strings", "struct", "struct_array_inputs", "struct_fields_ordering", "struct_inputs", "submodules", "to_be_bytes", "to_bytes_consistent", "to_bytes_integration", "to_le_bytes", "trait_as_return_type", "trait_impl_base_type", "traits_in_crates_1", "traits_in_crates_2", "tuple_inputs", "tuples", "type_aliases", "u128", "u16_support", "unconstrained_empty", "unit_value", "unsafe_range_constraint", "witness_compression", "xor"))
TEST_P(AcirIntegrationSingleTest, DISABLED_ProveAndVerifyProgram)
TEST_F(ChonkAPITests, DISABLED_ProveAndVerifyFileBasedFlow)
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
#define EXPECT_THROW_WITH_MESSAGE(code, expectedMessage)
TEST_F(AvmRecursiveTests, GoblinRecursion)
A test of the Goblinized AVM recursive verifier.
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