Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Hook up secondary calldata column in dsl #7759

Merged
merged 13 commits into from
Aug 7, 2024
Merged
4 changes: 4 additions & 0 deletions barretenberg/acir_tests/reset_acir_tests.sh
Original file line number Diff line number Diff line change
@@ -1,8 +1,12 @@
# Run from within barretenberg/acir_tests

# clean and rebuild noir then compile the test programs
cd ../../noir/noir-repo
cargo clean
noirup -p .
cd test_programs && ./rebuild.sh

# remove and repopulate the test artifacts in bberg
cd ../../../barretenberg/acir_tests
rm -rf acir_tests
./clone_test_vectors.sh
Original file line number Diff line number Diff line change
Expand Up @@ -190,6 +190,7 @@ void build_constraints(Builder& builder,
}

// Add block constraints
assign_calldata_ids<Builder>(constraint_system.block_constraints);
for (size_t i = 0; i < constraint_system.block_constraints.size(); ++i) {
const auto& constraint = constraint_system.block_constraints.at(i);
create_block_constraints(builder, constraint, has_valid_witness_assignments);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,7 @@ class AcirIntegrationTest : public ::testing::Test {
info("log circuit size = ", prover.instance->proving_key.log_circuit_size);
#endif
auto proof = prover.construct_proof();

// Verify Honk proof
auto verification_key = std::make_shared<VerificationKey>(prover.instance->proving_key);
Verifier verifier{ verification_key };
Expand Down Expand Up @@ -430,7 +431,7 @@ INSTANTIATE_TEST_SUITE_P(AcirTests,
testing::Values("fold_basic", "fold_basic_nested_call"));

/**
*@brief A basic test of a circuit generated in noir that makes use of the databus
* @brief A basic test of a circuit generated in noir that makes use of the databus
*
*/
TEST_F(AcirIntegrationTest, DISABLED_Databus)
Expand All @@ -452,6 +453,59 @@ TEST_F(AcirIntegrationTest, DISABLED_Databus)
EXPECT_TRUE(prove_and_verify_honk<Flavor>(builder));
}

/**
* @brief Test a program that uses two databus calldata columns
* @details In addition to checking that a proof of the resulting circuit verfies, check that the specific structure of
* the calldata/return data interaction in the noir program is reflected in the bberg circuit
*/
TEST_F(AcirIntegrationTest, DISABLED_DatabusTwoCalldata)
{
using Flavor = MegaFlavor;
using Builder = Flavor::CircuitBuilder;

std::string test_name = "databus_two_calldata";
info("Test: ", test_name);
acir_format::AcirProgram acir_program = get_program_data_from_test_file(test_name);

// Construct a bberg circuit from the acir representation
Builder builder = acir_format::create_circuit<Builder>(acir_program.constraints, 0, acir_program.witness);

// Check that the databus columns in the builder have been populated as expected
const auto& calldata = builder.get_calldata();
const auto& secondary_calldata = builder.get_secondary_calldata();
const auto& return_data = builder.get_return_data();

ASSERT(calldata.size() == 4);
ASSERT(secondary_calldata.size() == 3);
ASSERT(return_data.size() == 4);

// Check that return data was computed from the two calldata inputs as expected
ASSERT_EQ(builder.get_variable(calldata[0]) + builder.get_variable(secondary_calldata[0]),
builder.get_variable(return_data[0]));
ASSERT_EQ(builder.get_variable(calldata[1]) + builder.get_variable(secondary_calldata[1]),
builder.get_variable(return_data[1]));
ASSERT_EQ(builder.get_variable(calldata[2]) + builder.get_variable(secondary_calldata[2]),
builder.get_variable(return_data[2]));
ASSERT_EQ(builder.get_variable(calldata[3]), builder.get_variable(return_data[3]));

// Ensure that every index of each bus column was read once as expected
for (size_t idx = 0; idx < calldata.size(); ++idx) {
ASSERT_EQ(calldata.get_read_count(idx), 1);
}
for (size_t idx = 0; idx < secondary_calldata.size(); ++idx) {
ASSERT_EQ(secondary_calldata.get_read_count(idx), 1);
}
for (size_t idx = 0; idx < return_data.size(); ++idx) {
ASSERT_EQ(return_data.get_read_count(idx), 1);
}

// This prints a summary of the types of gates in the circuit
builder.blocks.summarize();

// Construct and verify Honk proof
EXPECT_TRUE(prove_and_verify_honk<Flavor>(builder));
}

/**
* @brief Ensure that adding gates post-facto to a circuit generated from acir still results in a valid circuit
* @details This is a pattern required by e.g. ClientIvc which appends recursive verifiers to acir-generated circuits
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -163,20 +163,34 @@ void process_call_data_operations(Builder& builder,
using databus_ct = stdlib::databus<Builder>;

databus_ct databus;
// Populate the calldata in the databus
databus.calldata.set_values(init);
for (const auto& op : constraint.trace) {
ASSERT(op.access_type == 0);
field_ct value = poly_to_field_ct(op.value, builder);
field_ct index = poly_to_field_ct(op.index, builder);
fr w_value = 0;
if (has_valid_witness_assignments) {
// If witness are assigned, we use the correct value for w
w_value = index.get_value();

// Method for processing operations on a generic databus calldata array
auto process_calldata = [&](auto& calldata_array) {
calldata_array.set_values(init); // Initialize the data in the bus array

for (const auto& op : constraint.trace) {
ASSERT(op.access_type == 0);
field_ct value = poly_to_field_ct(op.value, builder);
field_ct index = poly_to_field_ct(op.index, builder);
fr w_value = 0;
if (has_valid_witness_assignments) {
// If witness are assigned, we use the correct value for w
w_value = index.get_value();
}
field_ct w = field_ct::from_witness(&builder, w_value);
value.assert_equal(calldata_array[w]);
w.assert_equal(index);
}
field_ct w = field_ct::from_witness(&builder, w_value);
value.assert_equal(databus.calldata[w]);
w.assert_equal(index);
};

// Process primary or secondary calldata based on calldata_id
if (constraint.calldata_id == 0) {
process_calldata(databus.calldata);
} else if (constraint.calldata_id == 1) {
process_calldata(databus.secondary_calldata);
} else {
info("Databus only supports two calldata arrays.");
ASSERT(false);
}
}

Expand All @@ -199,4 +213,20 @@ void process_return_data_operations(const BlockConstraint& constraint, std::vect
ASSERT(constraint.trace.size() == 0);
}

// Do nothing for Ultra since it does not support Databus
template <> void assign_calldata_ids<UltraCircuitBuilder>([[maybe_unused]] std::vector<BlockConstraint>& constraints) {}

template <> void assign_calldata_ids<MegaCircuitBuilder>(std::vector<BlockConstraint>& constraints)
{
// Assign unique ID to each calldata block constraint
uint32_t calldata_id = 0;
for (auto& constraint : constraints) {
if (constraint.type == BlockType::CallData) {
constraint.calldata_id = calldata_id++;
}
}
// The backend only supports 2 calldata columns
ASSERT(calldata_id <= 2);
}

} // namespace acir_format
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ struct BlockConstraint {
std::vector<bb::poly_triple> init;
std::vector<MemOp> trace;
BlockType type;
uint32_t calldata_id{ 0 };
};

template <typename Builder>
Expand All @@ -47,6 +48,16 @@ void process_call_data_operations(Builder& builder,
template <typename Builder>
void process_return_data_operations(const BlockConstraint& constraint, std::vector<bb::stdlib::field_t<Builder>>& init);

/**
* @brief Assign a unique ID to each calldata block constraint based on the order in which it was recieved
* TODO(https://github.com/AztecProtocol/barretenberg/issues/1070): this is a workaround to allow calldata inputs to be
* distinguished by the backend since no identifiers are received from noir.
*
* @tparam Builder
* @param constraints
*/
template <typename Builder> void assign_calldata_ids(std::vector<BlockConstraint>& constraints);

template <typename B> inline void read(B& buf, MemOp& mem_op)
{
using serialize::read;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ use noirc_frontend::hir_def::function::FunctionSignature;

use super::FunctionBuilder;

#[derive(Clone)]
#[derive(Clone, Debug, Eq, PartialEq)]
pub(crate) enum DatabusVisibility {
None,
CallData(u32),
Expand All @@ -34,7 +34,7 @@ impl DataBusBuilder {
}
}

/// Generates a vector telling which (ssa) parameters from the given function signature
/// Generates a vector telling which flattened parameters from the given function signature
/// are tagged with databus visibility
pub(crate) fn is_databus(main_signature: &FunctionSignature) -> Vec<DatabusVisibility> {
let mut params_is_databus = Vec::new();
Expand Down Expand Up @@ -157,11 +157,16 @@ impl FunctionBuilder {
/// and a vector telling which ones are call-data
pub(crate) fn call_data_bus(
&mut self,
is_params_databus: Vec<DatabusVisibility>,
flattened_databus_visibilities: Vec<DatabusVisibility>,
) -> Vec<DataBusBuilder> {
//filter parameters of the first block that have call-data visibility
let first_block = self.current_function.entry_block();
let params = self.current_function.dfg[first_block].parameters();

// Reshape the is_params_databus to map to the SSA-level parameters
let is_params_databus =
self.deflatten_databus_visibilities(params, flattened_databus_visibilities);

let mut databus_param: BTreeMap<u32, Vec<ValueId>> = BTreeMap::new();
for (param, databus_attribute) in params.iter().zip(is_params_databus) {
match databus_attribute {
Expand All @@ -186,4 +191,40 @@ impl FunctionBuilder {
}
result
}

/// This function takes the flattened databus visibilities and generates the databus visibility for each ssa parameter
/// asserting that an ssa parameter is not assigned two different databus visibilities
fn deflatten_databus_visibilities(
&self,
ssa_params: &[ValueId],
flattened_params_databus_visibility: Vec<DatabusVisibility>,
) -> Vec<DatabusVisibility> {
// To do so, create a vec the size the flattened arguments where the items are ssa param index they correspond to
let ssa_param_indices: Vec<_> = ssa_params
.iter()
.enumerate()
.flat_map(|(ssa_param_index, ssa_param)| {
let flattened_size =
self.current_function.dfg[*ssa_param].get_type().flattened_size();
std::iter::repeat(ssa_param_index).take(flattened_size)
})
.collect();

let mut is_ssa_params_databus = Vec::with_capacity(ssa_params.len());
assert!(flattened_params_databus_visibility.len() == ssa_param_indices.len());
for (databus_visibility, ssa_index) in
flattened_params_databus_visibility.into_iter().zip(ssa_param_indices)
{
if let Some(previous_databus_visibility) = is_ssa_params_databus.get(ssa_index) {
assert!(
*previous_databus_visibility == databus_visibility,
"inconsistent databus visibility for ssa param"
);
} else {
assert!(ssa_index == is_ssa_params_databus.len());
is_ssa_params_databus.push(databus_visibility);
}
}
is_ssa_params_databus
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "databus_two_calldata"
type = "bin"
authors = [""]

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
x = [0,1,2,3]
y = [0,2,4]
z = [1,3,5,7]
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
fn main(mut x: [u32; 4], y: call_data(0) [u32; 3], z: call_data(1) [u32; 4]) -> return_data [u32; 4] {
let mut result = [0; 4];
for i in 0..3 {
let idx = x[i];
result[idx] = y[idx] + z[idx];
}
result[x[3]] = z[x[3]];

result
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "databus_two_calldata_simple"
type = "bin"
authors = [""]

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
idx = "1"
y = [7, 9]
z = [1,2,3,4]
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
fn main(mut idx: u32, y: call_data(0) [u32; 2], z: call_data(1) [u32; 4]) -> return_data u32 {
let a = y[idx];
let b = z[idx];
a + b
}
Loading