From c5ee3f0bc028bbebf6fa4000358abf31b8b26ceb Mon Sep 17 00:00:00 2001 From: ledwards2225 Date: Fri, 17 May 2024 19:24:20 +0000 Subject: [PATCH 1/7] genericize ecc op wire and databus poly handling --- .../protogalaxy_recursive_verifier.cpp | 26 ++++++-------- .../verifier/ultra_recursive_verifier.cpp | 25 ++++++-------- .../goblin_ultra_flavor.hpp | 10 +++--- .../barretenberg/ultra_honk/oink_prover.cpp | 34 +++++++------------ .../barretenberg/ultra_honk/oink_verifier.cpp | 26 ++++++-------- 5 files changed, 50 insertions(+), 71 deletions(-) diff --git a/barretenberg/cpp/src/barretenberg/stdlib/honk_recursion/verifier/protogalaxy_recursive_verifier.cpp b/barretenberg/cpp/src/barretenberg/stdlib/honk_recursion/verifier/protogalaxy_recursive_verifier.cpp index ad5e41af9f1..20459d10113 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib/honk_recursion/verifier/protogalaxy_recursive_verifier.cpp +++ b/barretenberg/cpp/src/barretenberg/stdlib/honk_recursion/verifier/protogalaxy_recursive_verifier.cpp @@ -35,22 +35,16 @@ void ProtoGalaxyRecursiveVerifier_::receive_and_finalise_inst witness_commitments.w_o = transcript->template receive_from_prover(domain_separator + "_" + labels.w_o); if constexpr (IsGoblinFlavor) { - witness_commitments.ecc_op_wire_1 = - transcript->template receive_from_prover(domain_separator + "_" + labels.ecc_op_wire_1); - witness_commitments.ecc_op_wire_2 = - transcript->template receive_from_prover(domain_separator + "_" + labels.ecc_op_wire_2); - witness_commitments.ecc_op_wire_3 = - transcript->template receive_from_prover(domain_separator + "_" + labels.ecc_op_wire_3); - witness_commitments.ecc_op_wire_4 = - transcript->template receive_from_prover(domain_separator + "_" + labels.ecc_op_wire_4); - witness_commitments.calldata = - transcript->template receive_from_prover(domain_separator + "_" + labels.calldata); - witness_commitments.calldata_read_counts = - transcript->template receive_from_prover(domain_separator + "_" + labels.calldata_read_counts); - witness_commitments.return_data = - transcript->template receive_from_prover(domain_separator + "_" + labels.return_data); - witness_commitments.return_data_read_counts = transcript->template receive_from_prover( - domain_separator + "_" + labels.return_data_read_counts); + // Receive ECC op wire commitments + for (auto [commitment, label] : zip_view(witness_commitments.get_ecc_op_wires(), labels.get_ecc_op_wires())) { + commitment = transcript->template receive_from_prover(domain_separator + "_" + label); + } + + // Receive DataBus related polynomial commitments + for (auto [commitment, label] : + zip_view(witness_commitments.get_databus_entities(), labels.get_databus_entities())) { + commitment = transcript->template receive_from_prover(domain_separator + "_" + label); + } } // Get challenge for sorted list batching and wire four memory records commitment diff --git a/barretenberg/cpp/src/barretenberg/stdlib/honk_recursion/verifier/ultra_recursive_verifier.cpp b/barretenberg/cpp/src/barretenberg/stdlib/honk_recursion/verifier/ultra_recursive_verifier.cpp index 538c7376bac..fc79a7c8ea3 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib/honk_recursion/verifier/ultra_recursive_verifier.cpp +++ b/barretenberg/cpp/src/barretenberg/stdlib/honk_recursion/verifier/ultra_recursive_verifier.cpp @@ -57,20 +57,17 @@ std::array UltraRecursiveVerifier_::ve // If Goblin, get commitments to ECC op wire polynomials and DataBus columns if constexpr (IsGoblinFlavor) { - commitments.ecc_op_wire_1 = - transcript->template receive_from_prover(commitment_labels.ecc_op_wire_1); - commitments.ecc_op_wire_2 = - transcript->template receive_from_prover(commitment_labels.ecc_op_wire_2); - commitments.ecc_op_wire_3 = - transcript->template receive_from_prover(commitment_labels.ecc_op_wire_3); - commitments.ecc_op_wire_4 = - transcript->template receive_from_prover(commitment_labels.ecc_op_wire_4); - commitments.calldata = transcript->template receive_from_prover(commitment_labels.calldata); - commitments.calldata_read_counts = - transcript->template receive_from_prover(commitment_labels.calldata_read_counts); - commitments.return_data = transcript->template receive_from_prover(commitment_labels.return_data); - commitments.return_data_read_counts = - transcript->template receive_from_prover(commitment_labels.return_data_read_counts); + // Receive ECC op wire commitments + for (auto [commitment, label] : + zip_view(commitments.get_ecc_op_wires(), commitment_labels.get_ecc_op_wires())) { + commitment = transcript->template receive_from_prover(label); + } + + // Receive DataBus related polynomial commitments + for (auto [commitment, label] : + zip_view(commitments.get_databus_entities(), commitment_labels.get_databus_entities())) { + commitment = transcript->template receive_from_prover(label); + } } // Get challenge for sorted list batching and wire four memory records diff --git a/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_flavor.hpp b/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_flavor.hpp index a417fbe4a97..7113aad02cf 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_flavor.hpp +++ b/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_flavor.hpp @@ -209,6 +209,12 @@ class GoblinUltraFlavor { { return RefArray{ this->ecc_op_wire_1, this->ecc_op_wire_2, this->ecc_op_wire_3, this->ecc_op_wire_4 }; } + auto get_databus_entities() // Excludes the derived inverse polynomials + { + return RefArray{ + this->calldata, this->calldata_read_counts, this->return_data, this->return_data_read_counts + }; + } }; template class ShiftedEntities { @@ -250,10 +256,6 @@ class GoblinUltraFlavor { auto get_sigmas() { return RefArray{ this->sigma_1, this->sigma_2, this->sigma_3, this->sigma_4 }; }; auto get_ids() { return RefArray{ this->id_1, this->id_2, this->id_3, this->id_4 }; }; auto get_tables() { return RefArray{ this->table_1, this->table_2, this->table_3, this->table_4 }; }; - auto get_ecc_op_wires() - { - return RefArray{ this->ecc_op_wire_1, this->ecc_op_wire_2, this->ecc_op_wire_3, this->ecc_op_wire_4 }; - }; // Gemini-specific getters. auto get_unshifted() { diff --git a/barretenberg/cpp/src/barretenberg/ultra_honk/oink_prover.cpp b/barretenberg/cpp/src/barretenberg/ultra_honk/oink_prover.cpp index b34fd9588d4..acd714bd919 100644 --- a/barretenberg/cpp/src/barretenberg/ultra_honk/oink_prover.cpp +++ b/barretenberg/cpp/src/barretenberg/ultra_honk/oink_prover.cpp @@ -76,30 +76,22 @@ template void OinkProver::execute_wire_commitment } if constexpr (IsGoblinFlavor) { + // Commit to Goblin ECC op wires - witness_commitments.ecc_op_wire_1 = commitment_key->commit(proving_key.polynomials.ecc_op_wire_1); - witness_commitments.ecc_op_wire_2 = commitment_key->commit(proving_key.polynomials.ecc_op_wire_2); - witness_commitments.ecc_op_wire_3 = commitment_key->commit(proving_key.polynomials.ecc_op_wire_3); - witness_commitments.ecc_op_wire_4 = commitment_key->commit(proving_key.polynomials.ecc_op_wire_4); - - auto op_wire_comms = witness_commitments.get_ecc_op_wires(); - auto labels = commitment_labels.get_ecc_op_wires(); - for (size_t idx = 0; idx < Flavor::NUM_WIRES; ++idx) { - transcript->send_to_verifier(domain_separator + labels[idx], op_wire_comms[idx]); + for (auto [commitment, polynomial, label] : zip_view(witness_commitments.get_ecc_op_wires(), + proving_key.polynomials.get_ecc_op_wires(), + commitment_labels.get_ecc_op_wires())) { + commitment = commitment_key->commit(polynomial); + transcript->send_to_verifier(domain_separator + label, commitment); } - // Commit to DataBus columns and corresponding read counts - witness_commitments.calldata = commitment_key->commit(proving_key.polynomials.calldata); - witness_commitments.calldata_read_counts = commitment_key->commit(proving_key.polynomials.calldata_read_counts); - transcript->send_to_verifier(domain_separator + commitment_labels.calldata, witness_commitments.calldata); - transcript->send_to_verifier(domain_separator + commitment_labels.calldata_read_counts, - witness_commitments.calldata_read_counts); - witness_commitments.return_data = commitment_key->commit(proving_key.polynomials.return_data); - witness_commitments.return_data_read_counts = - commitment_key->commit(proving_key.polynomials.return_data_read_counts); - transcript->send_to_verifier(domain_separator + commitment_labels.return_data, witness_commitments.return_data); - transcript->send_to_verifier(domain_separator + commitment_labels.return_data_read_counts, - witness_commitments.return_data_read_counts); + // Commit to DataBus related polynomials + for (auto [commitment, polynomial, label] : zip_view(witness_commitments.get_databus_entities(), + proving_key.polynomials.get_databus_entities(), + commitment_labels.get_databus_entities())) { + commitment = commitment_key->commit(polynomial); + transcript->send_to_verifier(domain_separator + label, commitment); + } } } diff --git a/barretenberg/cpp/src/barretenberg/ultra_honk/oink_verifier.cpp b/barretenberg/cpp/src/barretenberg/ultra_honk/oink_verifier.cpp index 27b91e2a190..55995d81c4a 100644 --- a/barretenberg/cpp/src/barretenberg/ultra_honk/oink_verifier.cpp +++ b/barretenberg/cpp/src/barretenberg/ultra_honk/oink_verifier.cpp @@ -62,22 +62,16 @@ template void OinkVerifier::execute_wire_commitme // If Goblin, get commitments to ECC op wire polynomials and DataBus columns if constexpr (IsGoblinFlavor) { - witness_comms.ecc_op_wire_1 = - transcript->template receive_from_prover(domain_separator + comm_labels.ecc_op_wire_1); - witness_comms.ecc_op_wire_2 = - transcript->template receive_from_prover(domain_separator + comm_labels.ecc_op_wire_2); - witness_comms.ecc_op_wire_3 = - transcript->template receive_from_prover(domain_separator + comm_labels.ecc_op_wire_3); - witness_comms.ecc_op_wire_4 = - transcript->template receive_from_prover(domain_separator + comm_labels.ecc_op_wire_4); - witness_comms.calldata = - transcript->template receive_from_prover(domain_separator + comm_labels.calldata); - witness_comms.calldata_read_counts = - transcript->template receive_from_prover(domain_separator + comm_labels.calldata_read_counts); - witness_comms.return_data = - transcript->template receive_from_prover(domain_separator + comm_labels.return_data); - witness_comms.return_data_read_counts = transcript->template receive_from_prover( - domain_separator + comm_labels.return_data_read_counts); + // Receive ECC op wire commitments + for (auto [commitment, label] : zip_view(witness_comms.get_ecc_op_wires(), comm_labels.get_ecc_op_wires())) { + commitment = transcript->template receive_from_prover(domain_separator + label); + } + + // Receive DataBus related polynomial commitments + for (auto [commitment, label] : + zip_view(witness_comms.get_databus_entities(), comm_labels.get_databus_entities())) { + commitment = transcript->template receive_from_prover(domain_separator + label); + } } } From 41e8aa254c20ef153735795c11584b34b96ba5a6 Mon Sep 17 00:00:00 2001 From: ledwards2225 Date: Fri, 17 May 2024 20:30:20 +0000 Subject: [PATCH 2/7] add read tags but dont use them in relation yet --- .../goblin_ultra_flavor.hpp | 29 +++++++++++++------ .../sumcheck/instance/prover_instance.cpp | 12 +++++--- .../goblin_ultra_transcript.test.cpp | 2 ++ 3 files changed, 30 insertions(+), 13 deletions(-) diff --git a/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_flavor.hpp b/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_flavor.hpp index 7113aad02cf..ad8f831c2b8 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_flavor.hpp +++ b/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_flavor.hpp @@ -40,12 +40,12 @@ class GoblinUltraFlavor { // The number of multivariate polynomials on which a sumcheck prover sumcheck operates (including shifts). We often // need containers of this size to hold related data, so we choose a name more agnostic than `NUM_POLYNOMIALS`. // Note: this number does not include the individual sorted list polynomials. - static constexpr size_t NUM_ALL_ENTITIES = 58; + static constexpr size_t NUM_ALL_ENTITIES = 60; // The number of polynomials precomputed to describe a circuit and to aid a prover in constructing a satisfying // assignment of witnesses. We again choose a neutral name. static constexpr size_t NUM_PRECOMPUTED_ENTITIES = 30; // The total number of witness entities not including shifts. - static constexpr size_t NUM_WITNESS_ENTITIES = 17; + static constexpr size_t NUM_WITNESS_ENTITIES = 19; // Total number of folded polynomials, which is just all polynomials except the shifts static constexpr size_t NUM_FOLDED_ENTITIES = NUM_PRECOMPUTED_ENTITIES + NUM_WITNESS_ENTITIES; @@ -188,10 +188,12 @@ class GoblinUltraFlavor { ecc_op_wire_4, // column 10 calldata, // column 11 calldata_read_counts, // column 12 - calldata_inverses, // column 13 - return_data, // column 14 - return_data_read_counts, // column 15 - return_data_inverses); // column 16 + calldata_read_tags, // column 13 + calldata_inverses, // column 14 + return_data, // column 15 + return_data_read_counts, // column 16 + return_data_read_tags, // column 17 + return_data_inverses); // column 18 }; /** @@ -211,9 +213,8 @@ class GoblinUltraFlavor { } auto get_databus_entities() // Excludes the derived inverse polynomials { - return RefArray{ - this->calldata, this->calldata_read_counts, this->return_data, this->return_data_read_counts - }; + return RefArray{ this->calldata, this->calldata_read_counts, this->calldata_read_tags, + this->return_data, this->return_data_read_counts, this->return_data_read_tags }; } }; @@ -666,9 +667,11 @@ class GoblinUltraFlavor { ecc_op_wire_4 = "ECC_OP_WIRE_4"; calldata = "CALLDATA"; calldata_read_counts = "CALLDATA_READ_COUNTS"; + calldata_read_tags = "CALLDATA_READ_TAGS"; calldata_inverses = "CALLDATA_INVERSES"; return_data = "RETURN_DATA"; return_data_read_counts = "RETURN_DATA_READ_COUNTS"; + return_data_read_tags = "RETURN_DATA_READ_TAGS"; return_data_inverses = "RETURN_DATA_INVERSES"; q_c = "Q_C"; @@ -758,9 +761,11 @@ class GoblinUltraFlavor { this->ecc_op_wire_4 = commitments.ecc_op_wire_4; this->calldata = commitments.calldata; this->calldata_read_counts = commitments.calldata_read_counts; + this->calldata_read_tags = commitments.calldata_read_tags; this->calldata_inverses = commitments.calldata_inverses; this->return_data = commitments.return_data; this->return_data_read_counts = commitments.return_data_read_counts; + this->return_data_read_tags = commitments.return_data_read_tags; this->return_data_inverses = commitments.return_data_inverses; } } @@ -788,9 +793,11 @@ class GoblinUltraFlavor { Commitment ecc_op_wire_4_comm; Commitment calldata_comm; Commitment calldata_read_counts_comm; + Commitment calldata_read_tags_comm; Commitment calldata_inverses_comm; Commitment return_data_comm; Commitment return_data_read_counts_comm; + Commitment return_data_read_tags_comm; Commitment return_data_inverses_comm; Commitment sorted_accum_comm; Commitment w_4_comm; @@ -844,9 +851,11 @@ class GoblinUltraFlavor { ecc_op_wire_4_comm = deserialize_from_buffer(proof_data, num_frs_read); calldata_comm = deserialize_from_buffer(proof_data, num_frs_read); calldata_read_counts_comm = deserialize_from_buffer(proof_data, num_frs_read); + calldata_read_tags_comm = deserialize_from_buffer(proof_data, num_frs_read); calldata_inverses_comm = deserialize_from_buffer(proof_data, num_frs_read); return_data_comm = deserialize_from_buffer(proof_data, num_frs_read); return_data_read_counts_comm = deserialize_from_buffer(proof_data, num_frs_read); + return_data_read_tags_comm = deserialize_from_buffer(proof_data, num_frs_read); return_data_inverses_comm = deserialize_from_buffer(proof_data, num_frs_read); sorted_accum_comm = deserialize_from_buffer(proof_data, num_frs_read); w_4_comm = deserialize_from_buffer(proof_data, num_frs_read); @@ -885,9 +894,11 @@ class GoblinUltraFlavor { serialize_to_buffer(ecc_op_wire_4_comm, proof_data); serialize_to_buffer(calldata_comm, proof_data); serialize_to_buffer(calldata_read_counts_comm, proof_data); + serialize_to_buffer(calldata_read_tags_comm, proof_data); serialize_to_buffer(calldata_inverses_comm, proof_data); serialize_to_buffer(return_data_comm, proof_data); serialize_to_buffer(return_data_read_counts_comm, proof_data); + serialize_to_buffer(return_data_read_tags_comm, proof_data); serialize_to_buffer(return_data_inverses_comm, proof_data); serialize_to_buffer(sorted_accum_comm, proof_data); serialize_to_buffer(w_4_comm, proof_data); diff --git a/barretenberg/cpp/src/barretenberg/sumcheck/instance/prover_instance.cpp b/barretenberg/cpp/src/barretenberg/sumcheck/instance/prover_instance.cpp index 06f08d6b302..24c4f8b8803 100644 --- a/barretenberg/cpp/src/barretenberg/sumcheck/instance/prover_instance.cpp +++ b/barretenberg/cpp/src/barretenberg/sumcheck/instance/prover_instance.cpp @@ -43,20 +43,24 @@ void ProverInstance_::construct_databus_polynomials(Circuit& circuit) { auto& public_calldata = proving_key.polynomials.calldata; auto& calldata_read_counts = proving_key.polynomials.calldata_read_counts; + auto& calldata_read_tags = proving_key.polynomials.calldata_read_tags; auto& public_return_data = proving_key.polynomials.return_data; auto& return_data_read_counts = proving_key.polynomials.return_data_read_counts; + auto& return_data_read_tags = proving_key.polynomials.return_data_read_tags; auto calldata = circuit.get_calldata(); auto return_data = circuit.get_return_data(); // Note: We do not utilize a zero row for databus columns for (size_t idx = 0; idx < calldata.size(); ++idx) { - public_calldata[idx] = circuit.get_variable(calldata[idx]); - calldata_read_counts[idx] = calldata.get_read_count(idx); + public_calldata[idx] = circuit.get_variable(calldata[idx]); // calldata values + calldata_read_counts[idx] = calldata.get_read_count(idx); // read counts + calldata_read_tags[idx] = calldata_read_counts[idx] > 0 ? 1 : 0; // has row been read or not } for (size_t idx = 0; idx < return_data.size(); ++idx) { - public_return_data[idx] = circuit.get_variable(return_data[idx]); - return_data_read_counts[idx] = return_data.get_read_count(idx); + public_return_data[idx] = circuit.get_variable(return_data[idx]); // return data values + return_data_read_counts[idx] = return_data.get_read_count(idx); // read counts + return_data_read_tags[idx] = return_data_read_counts[idx] > 0 ? 1 : 0; // has row been read or not } auto& databus_id = proving_key.polynomials.databus_id; diff --git a/barretenberg/cpp/src/barretenberg/ultra_honk/goblin_ultra_transcript.test.cpp b/barretenberg/cpp/src/barretenberg/ultra_honk/goblin_ultra_transcript.test.cpp index 0bccfa8084f..231392a31c2 100644 --- a/barretenberg/cpp/src/barretenberg/ultra_honk/goblin_ultra_transcript.test.cpp +++ b/barretenberg/cpp/src/barretenberg/ultra_honk/goblin_ultra_transcript.test.cpp @@ -60,8 +60,10 @@ class GoblinUltraTranscriptTests : public ::testing::Test { manifest_expected.add_entry(round, "ECC_OP_WIRE_4", frs_per_G); manifest_expected.add_entry(round, "CALLDATA", frs_per_G); manifest_expected.add_entry(round, "CALLDATA_READ_COUNTS", frs_per_G); + manifest_expected.add_entry(round, "CALLDATA_READ_TAGS", frs_per_G); manifest_expected.add_entry(round, "RETURN_DATA", frs_per_G); manifest_expected.add_entry(round, "RETURN_DATA_READ_COUNTS", frs_per_G); + manifest_expected.add_entry(round, "RETURN_DATA_READ_TAGS", frs_per_G); manifest_expected.add_challenge(round, "eta", "eta_two", "eta_three"); round++; From a72a077c996a3d1d70a0583b7821b937afbdf2b7 Mon Sep 17 00:00:00 2001 From: ledwards2225 Date: Fri, 17 May 2024 20:54:26 +0000 Subject: [PATCH 3/7] use read tags in relation --- .../relations/databus_lookup_relation.hpp | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/barretenberg/cpp/src/barretenberg/relations/databus_lookup_relation.hpp b/barretenberg/cpp/src/barretenberg/relations/databus_lookup_relation.hpp index 3c897ce3909..5d586b7b62e 100644 --- a/barretenberg/cpp/src/barretenberg/relations/databus_lookup_relation.hpp +++ b/barretenberg/cpp/src/barretenberg/relations/databus_lookup_relation.hpp @@ -79,6 +79,7 @@ template class DatabusLookupRelationImpl { static auto& inverses(AllEntities& in) { return in.calldata_inverses; } static auto& inverses(const AllEntities& in) { return in.calldata_inverses; } // const version static auto& read_counts(const AllEntities& in) { return in.calldata_read_counts; } + static auto& read_tags(const AllEntities& in) { return in.calldata_read_tags; } }; // Specialization for return data (bus_idx = 1) @@ -88,6 +89,7 @@ template class DatabusLookupRelationImpl { static auto& inverses(AllEntities& in) { return in.return_data_inverses; } static auto& inverses(const AllEntities& in) { return in.return_data_inverses; } // const version static auto& read_counts(const AllEntities& in) { return in.return_data_read_counts; } + static auto& read_tags(const AllEntities& in) { return in.return_data_read_tags; } }; /** @@ -101,8 +103,8 @@ template class DatabusLookupRelationImpl { template static bool operation_exists_at_row(const AllValues& row) { auto read_selector = get_read_selector(row); - auto read_counts = BusData::read_counts(row); - return (read_selector == 1 || read_counts > 0); + auto read_tag = BusData::read_tags(row); + return (read_selector == 1 || read_tag == 1); } /** @@ -117,10 +119,10 @@ template class DatabusLookupRelationImpl { { using View = typename Accumulator::View; - const auto is_read_gate = get_read_selector(in); - const auto read_counts = View(BusData::read_counts(in)); + const auto is_read_gate = get_read_selector(in); // is this a read gate + const auto read_tag = View(BusData::read_tags(in)); // does row contain data being read - return is_read_gate + read_counts - (is_read_gate * read_counts); + return is_read_gate + read_tag - (is_read_gate * read_tag); } /** From 3c8047607db485e1494149f64916ac5820beadf3 Mon Sep 17 00:00:00 2001 From: ledwards2225 Date: Fri, 17 May 2024 21:17:20 +0000 Subject: [PATCH 4/7] duplicate reads test passes --- .../goblin_ultra_circuit_builder.cpp | 2 - .../barretenberg/ultra_honk/databus.test.cpp | 39 +++++++++++++++++++ 2 files changed, 39 insertions(+), 2 deletions(-) diff --git a/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_circuit_builder.cpp b/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_circuit_builder.cpp index dd12ae51109..8c09af3d5e8 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_circuit_builder.cpp +++ b/barretenberg/cpp/src/barretenberg/stdlib_circuit_builders/goblin_ultra_circuit_builder.cpp @@ -201,8 +201,6 @@ uint32_t GoblinUltraCircuitBuilder_::read_bus_vector(BusId bus_idx, const ui const uint32_t read_idx = static_cast(uint256_t(this->get_variable(read_idx_witness_idx))); ASSERT(read_idx < bus_vector.size()); // Ensure that the read index is valid - // NOTE(https://github.com/AztecProtocol/barretenberg/issues/937): Multiple reads at same index is not supported. - ASSERT(bus_vector.get_read_count(read_idx) < 1); // Create a variable corresponding to the result of the read. Note that we do not in general connect reads from // databus via copy constraints (i.e. we create a unique variable for the result of each read) diff --git a/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp b/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp index 0da9d99d0ae..e6e3d64eecf 100644 --- a/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp +++ b/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp @@ -164,3 +164,42 @@ TEST_F(DataBusTests, CallDataAndReturnData) bool result = construct_and_verify_proof(builder); EXPECT_TRUE(result); } + +/** + * @brief Test proof construction/verification for a circuit with duplicate calldata reads + * + */ +TEST_F(DataBusTests, CallDataDuplicateRead) +{ + // Construct a circuit and add some ecc op gates and arithmetic gates + auto builder = construct_test_builder(); + + // Add some values to calldata + std::vector calldata_values = { 7, 10, 3, 12, 1 }; + for (auto& val : calldata_values) { + builder.add_public_calldata(builder.add_variable(val)); + } + + // Define some read indices with a duplicate + std::vector read_indices = { 1, 4, 1 }; + + // Create some calldata read gates and store the variable indices of the result for later + std::vector result_witness_indices; + for (uint32_t& read_idx : read_indices) { + // Create a variable corresponding to the index at which we want to read into calldata + uint32_t read_idx_witness_idx = builder.add_variable(read_idx); + + auto value_witness_idx = builder.read_calldata(read_idx_witness_idx); + result_witness_indices.emplace_back(value_witness_idx); + } + + // auto expected_read_result = calldata_values[1]; + // auto duplicate_read_result_1 = builder.get_variable(result_witness_indices[0]); + // auto duplicate_read_result_2 = builder.get_variable(result_witness_indices[2]); + // EXPECT_EQ(duplicate_read_result_1, expected_read_result); + // EXPECT_EQ(duplicate_read_result_1, duplicate_read_result_2); + + // Construct and verify Honk proof + bool result = construct_and_verify_proof(builder); + EXPECT_TRUE(result); +} \ No newline at end of file From 179183e8dd758179d05dc35ea9e9bd90ebae4069 Mon Sep 17 00:00:00 2001 From: ledwards2225 Date: Fri, 17 May 2024 22:55:08 +0000 Subject: [PATCH 5/7] duplicate reads stdlib test --- .../primitives/databus/databus.test.cpp | 43 +++++++++++++++++++ 1 file changed, 43 insertions(+) diff --git a/barretenberg/cpp/src/barretenberg/stdlib/primitives/databus/databus.test.cpp b/barretenberg/cpp/src/barretenberg/stdlib/primitives/databus/databus.test.cpp index e8daaa52170..acd6629e881 100644 --- a/barretenberg/cpp/src/barretenberg/stdlib/primitives/databus/databus.test.cpp +++ b/barretenberg/cpp/src/barretenberg/stdlib/primitives/databus/databus.test.cpp @@ -135,3 +135,46 @@ TEST(Databus, BadCopyFailure) // Since the output data is not a copy of the input, the checker should fail EXPECT_FALSE(CircuitChecker::check(builder)); } + +/** + * @brief Check that multiple reads from the same index results in a valid circuit + * + */ +TEST(Databus, DuplicateRead) +{ + Builder builder; + databus_ct databus; + + // Define some arbitrary bus data + std::array raw_calldata_values = { 5, 1, 2 }; + std::array raw_return_data_values = { 25, 6, 3 }; + + // Populate the calldata in the databus + std::vector calldata_values; + for (auto& value : raw_calldata_values) { + calldata_values.emplace_back(witness_ct(&builder, value)); + } + databus.calldata.set_values(calldata_values); + + // Populate the return data in the databus + std::vector return_data_values; + for (auto& value : raw_return_data_values) { + return_data_values.emplace_back(witness_ct(&builder, value)); + } + databus.return_data.set_values(return_data_values); + + // Perform some arbitrary reads from both calldata and return data with some repeated indices + field_ct idx_1(witness_ct(&builder, 1)); + field_ct idx_2(witness_ct(&builder, 2)); + + databus.calldata[idx_1]; + databus.calldata[idx_1]; + databus.calldata[idx_1]; + databus.calldata[idx_2]; + + databus.return_data[idx_2]; + databus.return_data[idx_2]; + databus.return_data[idx_1]; + + EXPECT_TRUE(CircuitChecker::check(builder)); +} \ No newline at end of file From 16551d8dec83b3aa006d025aad9154ae61c789dc Mon Sep 17 00:00:00 2001 From: ledwards2225 Date: Fri, 17 May 2024 22:57:38 +0000 Subject: [PATCH 6/7] cleanup --- .../cpp/src/barretenberg/ultra_honk/databus.test.cpp | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp b/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp index e6e3d64eecf..41cad81c9be 100644 --- a/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp +++ b/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp @@ -193,11 +193,12 @@ TEST_F(DataBusTests, CallDataDuplicateRead) result_witness_indices.emplace_back(value_witness_idx); } - // auto expected_read_result = calldata_values[1]; - // auto duplicate_read_result_1 = builder.get_variable(result_witness_indices[0]); - // auto duplicate_read_result_2 = builder.get_variable(result_witness_indices[2]); - // EXPECT_EQ(duplicate_read_result_1, expected_read_result); - // EXPECT_EQ(duplicate_read_result_1, duplicate_read_result_2); + // Check that the read result is as axpected and that the duplicate reads produce the same result + auto expected_read_result = calldata_values[1]; + auto duplicate_read_result_1 = builder.get_variable(result_witness_indices[0]); + auto duplicate_read_result_2 = builder.get_variable(result_witness_indices[2]); + EXPECT_EQ(duplicate_read_result_1, expected_read_result); + EXPECT_EQ(duplicate_read_result_1, duplicate_read_result_2); // Construct and verify Honk proof bool result = construct_and_verify_proof(builder); From 91eeeef8f219381d8df96eeda9203475ba18d1ef Mon Sep 17 00:00:00 2001 From: codygunton Date: Sun, 14 Jul 2024 14:30:49 +0000 Subject: [PATCH 7/7] Clarify test and fix typo --- .../src/barretenberg/ultra_honk/databus.test.cpp | 13 ++++++++----- 1 file changed, 8 insertions(+), 5 deletions(-) diff --git a/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp b/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp index 5f19f4cf0b1..5f148c3c007 100644 --- a/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp +++ b/barretenberg/cpp/src/barretenberg/ultra_honk/databus.test.cpp @@ -193,12 +193,15 @@ TEST_F(DataBusTests, CallDataDuplicateRead) result_witness_indices.emplace_back(value_witness_idx); } - // Check that the read result is as axpected and that the duplicate reads produce the same result - auto expected_read_result = calldata_values[1]; - auto duplicate_read_result_1 = builder.get_variable(result_witness_indices[0]); + // Check that the read result is as expected and that the duplicate reads produce the same result + auto expected_read_result_at_1 = calldata_values[1]; + auto expected_read_result_at_4 = calldata_values[4]; + auto duplicate_read_result_0 = builder.get_variable(result_witness_indices[0]); + auto duplicate_read_result_1 = builder.get_variable(result_witness_indices[1]); auto duplicate_read_result_2 = builder.get_variable(result_witness_indices[2]); - EXPECT_EQ(duplicate_read_result_1, expected_read_result); - EXPECT_EQ(duplicate_read_result_1, duplicate_read_result_2); + EXPECT_EQ(duplicate_read_result_0, expected_read_result_at_1); + EXPECT_EQ(duplicate_read_result_1, expected_read_result_at_4); + EXPECT_EQ(duplicate_read_result_2, expected_read_result_at_1); // Construct and verify Honk proof bool result = construct_and_verify_proof(builder);