diff --git a/barretenberg/cpp/src/barretenberg/proof_system/circuit_builder/ultra_circuit_builder.cpp b/barretenberg/cpp/src/barretenberg/proof_system/circuit_builder/ultra_circuit_builder.cpp index 21aa754edfe..e65527a4867 100644 --- a/barretenberg/cpp/src/barretenberg/proof_system/circuit_builder/ultra_circuit_builder.cpp +++ b/barretenberg/cpp/src/barretenberg/proof_system/circuit_builder/ultra_circuit_builder.cpp @@ -633,12 +633,12 @@ plookup::BasicTable& UltraCircuitBuilder_::get_table(const ploo template plookup::ReadData UltraCircuitBuilder_::create_gates_from_plookup_accumulators( - const plookup::MultiTableId& id, + const plookup::MultiTableIdOrPtr& id, const plookup::ReadData& read_values, const uint32_t key_a_index, std::optional key_b_index) { - const auto& multi_table = plookup::create_table(id); + const auto& multi_table = plookup::get_table(id); const size_t num_lookups = read_values[plookup::ColumnIdx::C1].size(); plookup::ReadData read_data; for (size_t i = 0; i < num_lookups; ++i) { diff --git a/barretenberg/cpp/src/barretenberg/proof_system/circuit_builder/ultra_circuit_builder.hpp b/barretenberg/cpp/src/barretenberg/proof_system/circuit_builder/ultra_circuit_builder.hpp index 78242840d9c..1abd54ef5de 100644 --- a/barretenberg/cpp/src/barretenberg/proof_system/circuit_builder/ultra_circuit_builder.hpp +++ b/barretenberg/cpp/src/barretenberg/proof_system/circuit_builder/ultra_circuit_builder.hpp @@ -1009,7 +1009,7 @@ class UltraCircuitBuilder_ : public CircuitBuilderBase create_gates_from_plookup_accumulators( - const plookup::MultiTableId& id, + const plookup::MultiTableIdOrPtr& id, const plookup::ReadData& read_values, const uint32_t key_a_index, std::optional key_b_index = std::nullopt); diff --git a/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/plookup_tables.cpp b/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/plookup_tables.cpp index 6a2257bf102..ee05af86e28 100644 --- a/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/plookup_tables.cpp +++ b/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/plookup_tables.cpp @@ -108,13 +108,21 @@ const MultiTable& create_table(const MultiTableId id) return MULTI_TABLES[id]; } -ReadData get_lookup_accumulators(const MultiTableId id, +const MultiTable& get_table(const MultiTableIdOrPtr& id) +{ + if (id.ptr == nullptr) { + return create_table(id.id); + } + return *id.ptr; +} + +ReadData get_lookup_accumulators(const MultiTableIdOrPtr& id, const fr& key_a, const fr& key_b, const bool is_2_to_1_lookup) { // return multi-table, populating global array of all multi-tables if need be - const auto& multi_table = create_table(id); + const auto& multi_table = get_table(id); const size_t num_lookups = multi_table.lookup_ids.size(); ReadData lookup; diff --git a/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/plookup_tables.hpp b/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/plookup_tables.hpp index 492793150d3..be85303fa8d 100644 --- a/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/plookup_tables.hpp +++ b/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/plookup_tables.hpp @@ -19,8 +19,9 @@ namespace bb::plookup { const MultiTable& create_table(MultiTableId id); +const MultiTable& get_table(const MultiTableIdOrPtr& id); -ReadData get_lookup_accumulators(MultiTableId id, +ReadData get_lookup_accumulators(const MultiTableIdOrPtr& id, const bb::fr& key_a, const bb::fr& key_b = 0, bool is_2_to_1_lookup = false); diff --git a/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/types.hpp b/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/types.hpp index 489d9e60071..f4dd1083a6e 100644 --- a/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/types.hpp +++ b/barretenberg/cpp/src/barretenberg/proof_system/plookup_tables/types.hpp @@ -132,17 +132,17 @@ struct MultiTable { std::vector column_1_step_sizes; std::vector column_2_step_sizes; std::vector column_3_step_sizes; - typedef std::array table_out; - typedef std::array table_in; + using table_out = std::array; + using table_in = std::array; std::vector get_table_values; private: void init_step_sizes() { const size_t num_lookups = column_1_coefficients.size(); - column_1_step_sizes.emplace_back(bb::fr(1)); - column_2_step_sizes.emplace_back(bb::fr(1)); - column_3_step_sizes.emplace_back(bb::fr(1)); + column_1_step_sizes.emplace_back(1); + column_2_step_sizes.emplace_back(1); + column_3_step_sizes.emplace_back(1); std::vector coefficient_inverses(column_1_coefficients.begin(), column_1_coefficients.end()); std::copy(column_2_coefficients.begin(), column_2_coefficients.end(), std::back_inserter(coefficient_inverses)); @@ -184,7 +184,7 @@ struct MultiTable { init_step_sizes(); } - MultiTable(){}; + MultiTable() = default; MultiTable(const MultiTable& other) = default; MultiTable(MultiTable&& other) = default; @@ -192,66 +192,22 @@ struct MultiTable { MultiTable& operator=(MultiTable&& other) = default; }; -// struct PlookupLargeKeyTable { -// struct KeyEntry { -// uint256_t key; -// std::array value{ bb::fr(0), bb::fr(0) }; -// bool operator<(const KeyEntry& other) const { return key < other.key; } - -// std::array to_sorted_list_components(const bool use_two_keys) const -// { -// return { -// key[0], -// value[0], -// value[1], -// }; -// } -// }; - -// BasicTableId id; -// size_t table_index; -// size_t size; -// bool use_twin_keys; - -// bb::fr column_1_step_size = bb::fr(0); -// bb::fr column_2_step_size = bb::fr(0); -// bb::fr column_3_step_size = bb::fr(0); -// std::vector column_1; -// std::vector column_3; -// std::vector column_2; -// std::vector lookup_gates; - -// std::array (*get_values_from_key)(const std::array); -// }; - -// struct PlookupFatKeyTable { -// struct KeyEntry { -// bb::fr key; -// std::array values{ 0, 0 }; -// bool operator<(const KeyEntry& other) const -// { -// return (key.from_montgomery_form() < other.key.from_montgomery_form()); -// } - -// std::array to_sorted_list_components() const { return { key, values[0], values[0] }; } -// } - -// BasicTableId id; -// size_t table_index; -// size_t size; -// bool use_twin_keys; - -// bb::fr column_1_step_size = bb::fr(0); -// bb::fr column_2_step_size = bb::fr(0); -// bb::fr column_3_step_size = bb::fr(0); -// std::vector column_1; -// std::vector column_3; -// std::vector column_2; -// std::vector lookup_gates; - -// std::array (*get_values_from_key)(const std::array); - -// } +// Represents either a predefined table from our enum list of supported lookup tables, or a dynamic lookup table defined +// by ACIR +struct MultiTableIdOrPtr { + // Used if we are using a lookup table from our predefined list, otherwise set to NUM_MULTI_TABLES and unused. + MultiTableId id; + // Used if we are using a lookup table from a lookup table defined by e.g. ACIR, otherwise set to nullptr. + MultiTable* ptr; + MultiTableIdOrPtr(MultiTable* ptr) + : id(NUM_MULTI_TABLES) + , ptr(ptr) + {} + MultiTableIdOrPtr(MultiTableId id) + : id(id) + , ptr(nullptr) + {} +}; /** * @brief The structure contains the most basic table serving one function (for, example an xor table) diff --git a/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.cpp b/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.cpp index 103784ba2de..becddf86d14 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.cpp +++ b/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.cpp @@ -14,7 +14,7 @@ using plookup::MultiTableId; using namespace bb; template -plookup::ReadData> plookup_read::get_lookup_accumulators(const MultiTableId id, +plookup::ReadData> plookup_read::get_lookup_accumulators(const plookup::MultiTableIdOrPtr& id, const field_t& key_a_in, const field_t& key_b_in, const bool is_2_to_1_lookup) @@ -64,8 +64,8 @@ plookup::ReadData> plookup_read::get_lookup_accumulato } template -std::pair, field_t> plookup_read::read_pair_from_table(const MultiTableId id, - const field_t& key) +std::pair, field_t> plookup_read::read_pair_from_table( + const plookup::MultiTableIdOrPtr& id, const field_t& key) { const auto lookup = get_lookup_accumulators(id, key); @@ -73,7 +73,7 @@ std::pair, field_t> plookup_read::read_pair_f } template -field_t plookup_read::read_from_2_to_1_table(const MultiTableId id, +field_t plookup_read::read_from_2_to_1_table(const plookup::MultiTableIdOrPtr& id, const field_t& key_a, const field_t& key_b) { @@ -83,7 +83,8 @@ field_t plookup_read::read_from_2_to_1_table(const MultiTableI } template -field_t plookup_read::read_from_1_to_2_table(const MultiTableId id, const field_t& key_a) +field_t plookup_read::read_from_1_to_2_table(const plookup::MultiTableIdOrPtr& id, + const field_t& key_a) { const auto lookup = get_lookup_accumulators(id, key_a); diff --git a/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.hpp b/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.hpp index eb85f164242..36193072132 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.hpp +++ b/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.hpp @@ -12,14 +12,15 @@ template class plookup_read { typedef field_t field_pt; public: - static std::pair read_pair_from_table(const plookup::MultiTableId id, const field_pt& key); + static std::pair read_pair_from_table(const plookup::MultiTableIdOrPtr& id, + const field_pt& key); - static field_pt read_from_2_to_1_table(const plookup::MultiTableId id, + static field_pt read_from_2_to_1_table(const plookup::MultiTableIdOrPtr& id, const field_pt& key_a, const field_pt& key_b); - static field_pt read_from_1_to_2_table(const plookup::MultiTableId id, const field_pt& key_a); + static field_pt read_from_1_to_2_table(const plookup::MultiTableIdOrPtr& id, const field_pt& key_a); - static plookup::ReadData get_lookup_accumulators(const plookup::MultiTableId id, + static plookup::ReadData get_lookup_accumulators(const plookup::MultiTableIdOrPtr& id, const field_pt& key_a, const field_pt& key_b = 0, const bool is_2_to_1_lookup = false); diff --git a/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.test.cpp b/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.test.cpp index 00d2b6846ee..e147fdbae89 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.test.cpp +++ b/barretenberg/cpp/src/barretenberg/stdlib/primitives/plookup/plookup.test.cpp @@ -468,7 +468,7 @@ TEST(stdlib_plookup, blake2s_xor) EXPECT_EQ(result, true); } -TEST(stdlib_plookup, uint32_and) +static void test_uint32_and(const MultiTableIdOrPtr& id) { Builder builder = Builder(); @@ -480,7 +480,7 @@ TEST(stdlib_plookup, uint32_and) field_ct left = witness_ct(&builder, bb::fr(left_value)); field_ct right = witness_ct(&builder, bb::fr(right_value)); - const auto lookup = plookup_read::get_lookup_accumulators(MultiTableId::UINT32_AND, left, right, true); + const auto lookup = plookup_read::get_lookup_accumulators(id, left, right, true); const auto left_slices = numeric::slice_input(left_value, 1 << 6, num_lookups); const auto right_slices = numeric::slice_input(right_value, 1 << 6, num_lookups); std::vector out_expected(num_lookups); @@ -510,6 +510,18 @@ TEST(stdlib_plookup, uint32_and) EXPECT_EQ(result, true); } +// Tests the dynamic multitable interface used by ACIR (the Noir interface to bb) +TEST(stdlib_plookup, dynamic_uint32_and) +{ + MultiTable and_table = bb::plookup::uint_tables::get_uint32_and_table(); + test_uint32_and(&and_table); +} + +TEST(stdlib_plookup, uint32_and) +{ + test_uint32_and(MultiTableId::UINT32_AND); +} + TEST(stdlib_plookup, secp256k1_generator) { using curve = stdlib::secp256k1;