Barretenberg
The ZK-SNARK library at the core of Aztec
Loading...
Searching...
No Matches
gt.test.cpp
Go to the documentation of this file.
1#include <gmock/gmock.h>
2#include <gtest/gtest.h>
3
4#include <cstdint>
5
15
16namespace bb::avm2::constraining {
17namespace {
18
19using tracegen::TestTraceContainer;
21using C = Column;
22using gt = bb::avm2::gt<FF>;
23using tracegen::GreaterThanTraceBuilder;
24using tracegen::RangeCheckTraceBuilder;
25
26const std::vector<std::array<uint128_t, 3>> TEST_VALUES = { { 1, 2, 16 },
27 { 2, 1, 16 },
28 { 2, 2, 16 },
29 { uint128_t{ (uint256_t(1) << 128) - 1 }, 1, 128 },
30 { 1, uint128_t{ (uint256_t(1) << 128) - 1 }, 128 } };
31
32class GreaterThanTest : public ::testing::TestWithParam<std::array<uint128_t, 3>> {};
33
34INSTANTIATE_TEST_SUITE_P(GreaterThanConstrainingTest, GreaterThanTest, ::testing::ValuesIn(TEST_VALUES));
35
36TEST_P(GreaterThanTest, GreaterThan)
37{
38 RangeCheckTraceBuilder range_check_builder;
39 auto [a, b, num_bits] = GetParam();
40 bool res = a > b;
41 uint128_t abs_diff = res ? a - b - 1 : b - a;
43 {
44 .gt_abs_diff = abs_diff,
45 .gt_input_a = a,
46 .gt_input_b = b,
47 .gt_num_bits = num_bits,
48 .gt_res = static_cast<uint8_t>(res),
49 .gt_sel = 1,
50 },
51 });
52 range_check_builder.process({ { .value = abs_diff, .num_bits = static_cast<uint8_t>(num_bits) } }, trace);
53 check_all_interactions<GreaterThanTraceBuilder>(trace);
54 check_relation<gt>(trace);
55}
56
57TEST_P(GreaterThanTest, GreaterThanTraceGen)
58{
59 RangeCheckTraceBuilder range_check_builder;
60 TestTraceContainer trace;
61 GreaterThanTraceBuilder builder;
62 auto [a, b, num_bits] = GetParam();
63 builder.process(
64 {
65 {
66 .a = a,
67 .b = b,
68 .result = a > b,
69 },
70 },
71 trace);
72 range_check_builder.process({ { .value = a > b ? a - b - 1 : b - a, .num_bits = static_cast<uint8_t>(num_bits) } },
73 trace);
74 check_all_interactions<GreaterThanTraceBuilder>(trace);
75 check_relation<gt>(trace);
76}
77
78TEST(GreaterThanConstrainingTest, NegativeGT)
79{
80 RangeCheckTraceBuilder range_check_builder;
81 uint128_t a = 2;
82 uint128_t b = 1;
83 bool res = a > b;
84 uint128_t abs_diff = res ? a - b - 1 : b - a;
86 {
87 .gt_abs_diff = abs_diff,
88 .gt_input_a = a,
89 .gt_input_b = b,
90 .gt_num_bits = 16,
91 .gt_res = static_cast<uint8_t>(res),
92 .gt_sel = 1,
93 },
94 });
95 range_check_builder.process({ { .value = abs_diff, .num_bits = 16 } }, trace);
96 check_all_interactions<GreaterThanTraceBuilder>(trace);
97 check_relation<gt>(trace);
98 auto wrong_b = res ? FF(a) + 1 : FF(a) - 1;
99 trace.set(Column::gt_input_b, 0, wrong_b);
100 // The absolute diff is now wrong:
101 EXPECT_THROW_WITH_MESSAGE(check_relation<gt>(trace), "GT_RESULT");
102 // Correct the diff based on incorrect input:
103 auto new_abs_diff = res ? FF(a) - wrong_b - 1 : wrong_b - FF(a);
104 trace.set(Column::gt_abs_diff, 0, new_abs_diff);
105 // Now, we are range checking the correct value...
106 check_relation<gt>(trace);
107 // ..but the check itself correctly fails (note: new_result_to_range_check doesn't fit in a u128, I'm just adding
108 // an event which will definitely fail):
109 range_check_builder.process({ { .value = static_cast<uint128_t>(new_abs_diff), .num_bits = 128 } }, trace);
110 EXPECT_THROW_WITH_MESSAGE((check_all_interactions<GreaterThanTraceBuilder>(trace)), "LOOKUP_GT_GT_RANGE");
111}
112
113TEST(GreaterThanConstrainingTest, NegativeGTResult)
114{
115 RangeCheckTraceBuilder range_check_builder;
116 uint128_t a = 2;
117 uint128_t b = 1;
118 bool res = a > b;
119 uint128_t abs_diff = res ? a - b - 1 : b - a;
121 {
122 .gt_abs_diff = abs_diff,
123 .gt_input_a = a,
124 .gt_input_b = b,
125 .gt_num_bits = 16,
126 .gt_res = static_cast<uint8_t>(res),
127 .gt_sel = 1,
128 },
129 });
130 range_check_builder.process({ { .value = abs_diff, .num_bits = 16 } }, trace);
131 check_all_interactions<GreaterThanTraceBuilder>(trace);
132 check_relation<gt>(trace);
133 trace.set(Column::gt_res, 0, static_cast<uint8_t>(!res));
134 // The absolute diff is now wrong:
135 EXPECT_THROW_WITH_MESSAGE(check_relation<gt>(trace), "GT_RESULT");
136 // Correct the diff based on incorrect res:
137 auto new_abs_diff = res ? FF(b) - FF(a) : FF(a) - FF(b) - 1;
138 trace.set(Column::gt_abs_diff, 0, new_abs_diff);
139 // Now, we are range checking the correct value...
140 check_relation<gt>(trace);
141 // ..but the check itself correctly fails (note: new_result_to_range_check doesn't fit in a u128, I'm just adding
142 // an event which will definitely fail):
143 range_check_builder.process({ { .value = static_cast<uint128_t>(new_abs_diff), .num_bits = 128 } }, trace);
144 EXPECT_THROW_WITH_MESSAGE((check_all_interactions<GreaterThanTraceBuilder>(trace)), "LOOKUP_GT_GT_RANGE");
145}
146
147} // namespace
148} // namespace bb::avm2::constraining
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)
static TestTraceContainer from_rows(const std::vector< AvmFullRow > &rows)
void set(Column col, uint32_t row, const FF &value)
RangeCheckTraceBuilder range_check_builder
Definition alu.test.cpp:120
AluTraceBuilder builder
Definition alu.test.cpp:123
GreaterThan gt
TestTraceContainer trace
FF a
FF b
#define EXPECT_THROW_WITH_MESSAGE(code, expectedMessage)
Definition macros.hpp:7
AvmFlavorSettings::FF FF
TEST(TxExecutionConstrainingTest, WriteTreeValue)
Definition tx.test.cpp:508
typename Flavor::FF FF
constexpr decltype(auto) get(::tuplet::tuple< T... > &&t) noexcept
Definition tuple.hpp:13
unsigned __int128 uint128_t
Definition serialize.hpp:44