From 2fabc9fe38b25c30b88193edb0cabfc4daead9e2 Mon Sep 17 00:00:00 2001 From: IlyasRidhuan Date: Wed, 25 Sep 2024 14:27:35 +0000 Subject: [PATCH] feat(avm): class id + contract address --- barretenberg/cpp/src/barretenberg/bb/main.cpp | 12 +- .../vm/avm/tests/execution.test.cpp | 103 ++++++++----- .../vm/avm/trace/bytecode_trace.cpp | 4 +- .../vm/avm/trace/bytecode_trace.hpp | 5 +- .../barretenberg/vm/avm/trace/execution.cpp | 58 +++---- .../barretenberg/vm/avm/trace/execution.hpp | 14 +- .../vm/avm/trace/execution_hints.hpp | 67 +++++++- .../src/barretenberg/vm/avm/trace/trace.cpp | 8 +- .../src/barretenberg/vm/avm/trace/trace.hpp | 3 +- .../bb-prover/src/avm_proving.test.ts | 9 +- yarn-project/bb-prover/src/bb/execute.ts | 7 - .../circuits.js/src/structs/avm/avm.ts | 145 ++++++++++++++---- .../circuits.js/src/tests/factories.ts | 20 ++- .../src/avm_integration.test.ts | 19 +-- .../src/orchestrator/orchestrator.ts | 1 - .../simulator/src/avm/avm_simulator.ts | 2 +- .../simulator/src/avm/journal/journal.ts | 37 ++++- .../simulator/src/public/side_effect_trace.ts | 36 ++++- .../src/public/side_effect_trace_interface.ts | 7 +- 19 files changed, 394 insertions(+), 163 deletions(-) diff --git a/barretenberg/cpp/src/barretenberg/bb/main.cpp b/barretenberg/cpp/src/barretenberg/bb/main.cpp index cadf836cc536..b4221357fd34 100644 --- a/barretenberg/cpp/src/barretenberg/bb/main.cpp +++ b/barretenberg/cpp/src/barretenberg/bb/main.cpp @@ -939,18 +939,17 @@ void vk_as_fields(const std::string& vk_path, const std::string& output_path) * @param hints_path Path to the file containing the serialised avm circuit hints * @param output_path Path (directory) to write the output proof and verification keys */ -void avm_prove(const std::filesystem::path& bytecode_path, - const std::filesystem::path& calldata_path, +void avm_prove(const std::filesystem::path& calldata_path, const std::filesystem::path& public_inputs_path, const std::filesystem::path& hints_path, const std::filesystem::path& output_path) { - std::vector const bytecode = read_file(bytecode_path); std::vector const calldata = many_from_buffer(read_file(calldata_path)); std::vector const public_inputs_vec = many_from_buffer(read_file(public_inputs_path)); auto const avm_hints = bb::avm_trace::ExecutionHints::from(read_file(hints_path)); - vinfo("bytecode size: ", bytecode.size()); + // Using [0] is fine now for the top-level call, but we might need to index by address in future + vinfo("bytecode size: ", avm_hints.all_contract_bytecode[0].bytecode.size()); vinfo("calldata size: ", calldata.size()); vinfo("public_inputs size: ", public_inputs_vec.size()); vinfo("hints.storage_value_hints size: ", avm_hints.storage_value_hints.size()); @@ -965,7 +964,7 @@ void avm_prove(const std::filesystem::path& bytecode_path, // Prove execution and return vk auto const [verification_key, proof] = - AVM_TRACK_TIME_V("prove/all", avm_trace::Execution::prove(bytecode, calldata, public_inputs_vec, avm_hints)); + AVM_TRACK_TIME_V("prove/all", avm_trace::Execution::prove(calldata, public_inputs_vec, avm_hints)); std::vector vk_as_fields = verification_key.to_field_elements(); @@ -1520,7 +1519,6 @@ int main(int argc, char* argv[]) write_recursion_inputs_honk(bytecode_path, witness_path, output_path); #ifndef DISABLE_AZTEC_VM } else if (command == "avm_prove") { - std::filesystem::path avm_bytecode_path = get_option(args, "--avm-bytecode", "./target/avm_bytecode.bin"); std::filesystem::path avm_calldata_path = get_option(args, "--avm-calldata", "./target/avm_calldata.bin"); std::filesystem::path avm_public_inputs_path = get_option(args, "--avm-public-inputs", "./target/avm_public_inputs.bin"); @@ -1529,7 +1527,7 @@ int main(int argc, char* argv[]) std::filesystem::path output_path = get_option(args, "-o", "./proofs"); extern std::filesystem::path avm_dump_trace_path; avm_dump_trace_path = get_option(args, "--avm-dump-trace", ""); - avm_prove(avm_bytecode_path, avm_calldata_path, avm_public_inputs_path, avm_hints_path, output_path); + avm_prove(avm_calldata_path, avm_public_inputs_path, avm_hints_path, output_path); } else if (command == "avm_verify") { return avm_verify(proof_path, vk_path) ? 0 : 1; #endif diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/tests/execution.test.cpp b/barretenberg/cpp/src/barretenberg/vm/avm/tests/execution.test.cpp index 0dd0f49d34ba..d778ebc8d5e7 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/tests/execution.test.cpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/tests/execution.test.cpp @@ -35,13 +35,9 @@ class AvmExecutionTests : public ::testing::Test { Execution::set_trace_builder_constructor([](VmPublicInputs public_inputs, ExecutionHints execution_hints, uint32_t side_effect_counter, - std::vector calldata, - const std::vector>& all_contracts_bytecode) { - return AvmTraceBuilder(std::move(public_inputs), - std::move(execution_hints), - side_effect_counter, - std::move(calldata), - all_contracts_bytecode) + std::vector calldata) { + return AvmTraceBuilder( + std::move(public_inputs), std::move(execution_hints), side_effect_counter, std::move(calldata)) .set_full_precomputed_tables(false) .set_range_check_required(false); }); @@ -56,6 +52,7 @@ class AvmExecutionTests : public ::testing::Test { srs::init_crs_factory("../srs_db/ignition"); public_inputs_vec.at(DA_START_GAS_LEFT_PCPI_OFFSET) = DEFAULT_INITIAL_DA_GAS; public_inputs_vec.at(L2_START_GAS_LEFT_PCPI_OFFSET) = DEFAULT_INITIAL_L2_GAS; + public_inputs_vec.at(ADDRESS_SELECTOR) = 0xdeadbeef; public_inputs = Execution::convert_public_inputs(public_inputs_vec); }; @@ -70,7 +67,21 @@ class AvmExecutionTests : public ::testing::Test { std::vector calldata{}; std::vector returndata{}; - return Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + auto execution_hints = ExecutionHints().with_avm_contract_bytecode({ bytecode }); + execution_hints.all_contract_bytecode[0].contract_instance.address = 0xdeadbeef; + + return AvmExecutionTests::gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); + } + + std::vector gen_trace(std::vector bytecode, + std::vector const& calldata, + std::vector const& public_inputs_vec, + std::vector& returndata, + ExecutionHints& execution_hints) const + { + execution_hints.all_contract_bytecode = { bytecode }; + execution_hints.all_contract_bytecode[0].contract_instance.address = 0xdeadbeef; + return Execution::gen_trace(calldata, public_inputs_vec, returndata, execution_hints); } void feed_output(uint32_t output_offset, FF const& value, FF const& side_effect_counter, FF const& metadata) @@ -474,8 +485,8 @@ TEST_F(AvmExecutionTests, jumpAndCalldatacopy) Field(&Instruction::operands, ElementsAre(VariantWith(5))))); std::vector returndata; - auto trace = - Execution::gen_trace(bytecode, std::vector{ 13, 156 }, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, std::vector{ 13, 156 }, public_inputs_vec, returndata, execution_hints); // Expected sequence of PCs during execution std::vector pc_sequence{ @@ -566,10 +577,9 @@ TEST_F(AvmExecutionTests, jumpiAndCalldatacopy) ElementsAre(VariantWith(0), VariantWith(6), VariantWith(10))))); std::vector returndata; - auto trace_jump = - Execution::gen_trace(bytecode, std::vector{ 9873123 }, public_inputs_vec, returndata, ExecutionHints()); - auto trace_no_jump = - Execution::gen_trace(bytecode, std::vector{ 0 }, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace_jump = gen_trace(bytecode, std::vector{ 9873123 }, public_inputs_vec, returndata, execution_hints); + auto trace_no_jump = gen_trace(bytecode, std::vector{ 0 }, public_inputs_vec, returndata, execution_hints); // Expected sequence of PCs during execution with jump std::vector pc_sequence_jump{ 0, 1, 2, 3, 4, 6, 7 }; @@ -832,8 +842,9 @@ TEST_F(AvmExecutionTests, toRadixLeOpcode) // Assign a vector that we will mutate internally in gen_trace to store the return values; std::vector returndata; - auto trace = Execution::gen_trace( - bytecode, std::vector{ FF::modulus - FF(1) }, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = + gen_trace(bytecode, std::vector{ FF::modulus - FF(1) }, public_inputs_vec, returndata, execution_hints); // Find the first row enabling the TORADIXLE selector // Expected output is bitwise decomposition of MODULUS - 1..could hardcode the result but it's a bit long @@ -898,8 +909,9 @@ TEST_F(AvmExecutionTests, toRadixLeOpcodeBitsMode) // Assign a vector that we will mutate internally in gen_trace to store the return values; std::vector returndata; - auto trace = Execution::gen_trace( - bytecode, std::vector{ FF::modulus - FF(1) }, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = + gen_trace(bytecode, std::vector{ FF::modulus - FF(1) }, public_inputs_vec, returndata, execution_hints); // Find the first row enabling the TORADIXLE selector // Expected output is bitwise decomposition of MODULUS - 1..could hardcode the result but it's a bit long @@ -975,7 +987,8 @@ TEST_F(AvmExecutionTests, sha256CompressionOpcode) // 4091010797,3974542186]), std::vector expected_output = { 1862536192, 526086805, 2067405084, 593147560, 726610467, 813867028, 4091010797ULL, 3974542186ULL }; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, expected_output); @@ -1036,7 +1049,8 @@ TEST_F(AvmExecutionTests, poseidon2PermutationOpCode) FF(std::string("0x018555a8eb50cf07f64b019ebaf3af3c925c93e631f3ecd455db07bbb52bbdd3")), FF(std::string("0x0cbea457c91c22c6c31fd89afd2541efc2edf31736b9f721e823b2165c90fd41")) }; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, expected_output); @@ -1109,7 +1123,8 @@ TEST_F(AvmExecutionTests, keccakf1600OpCode) // Assign a vector that we will mutate internally in gen_trace to store the return values; std::vector calldata = std::vector(); std::vector returndata = std::vector(); - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, expected_output); @@ -1166,7 +1181,8 @@ TEST_F(AvmExecutionTests, keccakOpCode) // Assign a vector that we will mutate internally in gen_trace to store the return values; std::vector calldata = std::vector(); std::vector returndata = std::vector(); - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, expected_output); @@ -1228,7 +1244,8 @@ TEST_F(AvmExecutionTests, pedersenHashOpCode) // Assign a vector that we will mutate internally in gen_trace to store the return values; std::vector returndata = std::vector(); std::vector calldata = { FF(1), FF(1) }; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata[0], expected_output); @@ -1295,7 +1312,8 @@ TEST_F(AvmExecutionTests, embeddedCurveAddOpCode) // Assign a vector that we will mutate internally in gen_trace to store the return values; std::vector returndata; std::vector calldata = { a.x, a.y, FF(a_is_inf ? 1 : 0), b.x, b.y, FF(b_is_inf ? 1 : 0) }; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, expected_output); @@ -1382,7 +1400,8 @@ TEST_F(AvmExecutionTests, msmOpCode) // Assign a vector that we will mutate internally in gen_trace to store the return values; std::vector returndata; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, expected_output); @@ -1455,7 +1474,8 @@ TEST_F(AvmExecutionTests, pedersenCommitmentOpcode) // Assign a vector that we will mutate internally in gen_trace to store the return values; std::vector returndata; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, expected_output); @@ -1625,7 +1645,7 @@ TEST_F(AvmExecutionTests, kernelInputOpcodes) std::vector calldata; FF sender = 1; - FF address = 2; + FF address = 0xdeadbeef; // NOTE: address doesn't actually exist in public circuit public inputs, // so storage address is just an alias of address for now FF storage_address = address; @@ -1667,7 +1687,8 @@ TEST_F(AvmExecutionTests, kernelInputOpcodes) public_inputs_vec[FEE_PER_L2_GAS_OFFSET] = feeperl2gas; std::vector returndata; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); // Validate returndata EXPECT_EQ(returndata, expected_returndata); @@ -1840,7 +1861,8 @@ TEST_F(AvmExecutionTests, ExecutorThrowsWithIncorrectNumberOfPublicInputs) auto bytecode = hex_to_bytes(bytecode_hex); auto instructions = Deserialization::parse(bytecode); - EXPECT_THROW_WITH_MESSAGE(Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()), + ExecutionHints execution_hints; + EXPECT_THROW_WITH_MESSAGE(gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints), "Public inputs vector is not of PUBLIC_CIRCUIT_PUBLIC_INPUTS_LENGTH"); } @@ -1883,7 +1905,8 @@ TEST_F(AvmExecutionTests, kernelOutputEmitOpcodes) std::vector calldata = {}; std::vector returndata = {}; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); // CHECK EMIT NOTE HASH // Check output data + side effect counters have been set correctly @@ -1918,7 +1941,7 @@ TEST_F(AvmExecutionTests, kernelOutputEmitOpcodes) auto emit_log_row = std::ranges::find_if(trace.begin(), trace.end(), [](Row r) { return r.main_sel_op_emit_unencrypted_log == 1; }); // Trust me bro for now, this is the truncated sha output - FF expected_hash = FF(std::string("0x006db65fd59fd356f6729140571b5bcd6bb3b83492a16e1bf0a3884442fc3c8a")); + FF expected_hash = FF(std::string("0x003383cbb254941b33c0aaf8476c4b9b532d70a2fb105ee908dd332f7d942df6")); EXPECT_EQ(emit_log_row->main_ia, expected_hash); EXPECT_EQ(emit_log_row->main_side_effect_counter, 2); @@ -1981,7 +2004,7 @@ TEST_F(AvmExecutionTests, kernelOutputStorageLoadOpcodeSimple) // side effect counter 0 = value 42 auto execution_hints = ExecutionHints().with_storage_value_hints({ { 0, 42 } }); - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); // CHECK SLOAD // Check output data + side effect counters have been set correctly @@ -2035,7 +2058,8 @@ TEST_F(AvmExecutionTests, kernelOutputStorageStoreOpcodeSimple) std::vector returndata; - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, ExecutionHints()); + ExecutionHints execution_hints; + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); // CHECK SSTORE auto sstore_row = std::ranges::find_if(trace.begin(), trace.end(), [](Row r) { return r.main_sel_op_sstore == 1; }); EXPECT_EQ(sstore_row->main_ia, 42); // Read value @@ -2098,7 +2122,7 @@ TEST_F(AvmExecutionTests, kernelOutputStorageOpcodes) // side effect counter 0 = value 42 auto execution_hints = ExecutionHints().with_storage_value_hints({ { 0, 42 } }); - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); // CHECK SLOAD // Check output data + side effect counters have been set correctly @@ -2181,7 +2205,7 @@ TEST_F(AvmExecutionTests, kernelOutputHashExistsOpcodes) .with_storage_value_hints({ { 0, 1 }, { 1, 1 }, { 2, 1 } }) .with_note_hash_exists_hints({ { 0, 1 }, { 1, 1 }, { 2, 1 } }); - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); // CHECK NOTEHASHEXISTS auto note_hash_row = @@ -2315,10 +2339,10 @@ TEST_F(AvmExecutionTests, opCallOpcodes) .l2_gas_used = 0, .da_gas_used = 0, .end_side_effect_counter = 0, - .bytecode = {}, + .contract_address = 0, } }); - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, std::vector({ 9, 8, 1 })); // The 1 represents the success validate_trace(std::move(trace), public_inputs, calldata, returndata); @@ -2365,9 +2389,8 @@ TEST_F(AvmExecutionTests, opGetContractInstanceOpcodes) // Generate Hint for call operation // Note: opcode does not write 'address' into memory auto execution_hints = - ExecutionHints().with_contract_instance_hints({ { address, { address, 1, 2, 3, 4, 5, 6 } } }); - - auto trace = Execution::gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); + ExecutionHints().with_contract_instance_hints({ { address, { address, true, 2, 3, 4, 5, 6 } } }); + auto trace = gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints); EXPECT_EQ(returndata, std::vector({ 1, 2, 3, 4, 5, 6 })); // The first one represents true validate_trace(std::move(trace), public_inputs, calldata, returndata); diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/bytecode_trace.cpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/bytecode_trace.cpp index 82041e02871f..9fc743010ded 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/bytecode_trace.cpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/bytecode_trace.cpp @@ -4,7 +4,7 @@ namespace bb::avm_trace { using poseidon2 = crypto::Poseidon2; -AvmBytecodeTraceBuilder::AvmBytecodeTraceBuilder(const std::vector>& all_contracts_bytecode) +AvmBytecodeTraceBuilder::AvmBytecodeTraceBuilder(const std::vector& all_contracts_bytecode) : all_contracts_bytecode(all_contracts_bytecode) {} @@ -31,7 +31,7 @@ void AvmBytecodeTraceBuilder::build_bytecode_columns() // This is the main loop that will generate the bytecode trace for (auto& contract_bytecode : all_contracts_bytecode) { FF running_hash = FF::zero(); - auto packed_bytecode = pack_bytecode(contract_bytecode); + auto packed_bytecode = pack_bytecode(contract_bytecode.bytecode); // This size is already based on the number of fields for (size_t i = 0; i < packed_bytecode.size(); ++i) { bytecode_trace.push_back(BytecodeTraceEntry{ diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/bytecode_trace.hpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/bytecode_trace.hpp index a62784fe59e5..aab3d6d6a7f6 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/bytecode_trace.hpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/bytecode_trace.hpp @@ -22,9 +22,8 @@ class AvmBytecodeTraceBuilder { // Derive the contract address FF contract_address{}; }; - AvmBytecodeTraceBuilder() = default; // These interfaces will change when we start feeding in more inputs and hints - AvmBytecodeTraceBuilder(const std::vector>& all_contracts_bytecode); + AvmBytecodeTraceBuilder(const std::vector& all_contracts_bytecode); size_t size() const { return bytecode_trace.size(); } void reset(); @@ -38,7 +37,7 @@ class AvmBytecodeTraceBuilder { std::vector bytecode_trace; // The first element is the main top-level contract, the rest are external calls - std::vector> all_contracts_bytecode; + std::vector all_contracts_bytecode; // TODO: Come back to this // VmPublicInputs public_inputs; // ExecutionHints hints; diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.cpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.cpp index eefac51d8382..fb47ff6203b7 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.cpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.cpp @@ -146,18 +146,13 @@ void show_trace_info(const auto& trace) } // namespace // Needed for dependency injection in tests. -Execution::TraceBuilderConstructor Execution::trace_builder_constructor = - [](VmPublicInputs public_inputs, - ExecutionHints execution_hints, - uint32_t side_effect_counter, - std::vector calldata, - std::vector> all_contract_bytecode) { - return AvmTraceBuilder(std::move(public_inputs), - std::move(execution_hints), - side_effect_counter, - std::move(calldata), - all_contract_bytecode); - }; +Execution::TraceBuilderConstructor Execution::trace_builder_constructor = [](VmPublicInputs public_inputs, + ExecutionHints execution_hints, + uint32_t side_effect_counter, + std::vector calldata) { + return AvmTraceBuilder( + std::move(public_inputs), std::move(execution_hints), side_effect_counter, std::move(calldata)); +}; /** * @brief Temporary routine to generate default public inputs (gas values) until we get @@ -180,8 +175,7 @@ std::vector Execution::getDefaultPublicInputs() * @throws runtime_error exception when the bytecode is invalid. * @return The verifier key and zk proof of the execution. */ -std::tuple Execution::prove(std::vector const& bytecode, - std::vector const& calldata, +std::tuple Execution::prove(std::vector const& calldata, std::vector const& public_inputs_vec, ExecutionHints const& execution_hints) { @@ -190,8 +184,8 @@ std::tuple Execution::prove(std::vector returndata; - std::vector trace = AVM_TRACK_TIME_V( - "prove/gen_trace", gen_trace(bytecode, calldata, public_inputs_vec, returndata, execution_hints)); + std::vector trace = + AVM_TRACK_TIME_V("prove/gen_trace", gen_trace(calldata, public_inputs_vec, returndata, execution_hints)); if (!avm_dump_trace_path.empty()) { info("Dumping trace as CSV to: " + avm_dump_trace_path.string()); dump_trace_as_csv(trace, avm_dump_trace_path); @@ -419,15 +413,13 @@ bool Execution::verify(AvmFlavor::VerificationKey vk, HonkProof const& proof) * @param public_inputs expressed as a vector of finite field elements. * @return The trace as a vector of Row. */ -std::vector Execution::gen_trace(std::vector const& bytecode, - std::vector const& calldata, +std::vector Execution::gen_trace(std::vector const& calldata, std::vector const& public_inputs_vec, std::vector& returndata, ExecutionHints const& execution_hints) { - std::vector instructions = Deserialization::parse(bytecode); - vinfo("Deserialized " + std::to_string(instructions.size()) + " instructions"); + vinfo("------- GENERATING TRACE -------"); // TODO(https://github.com/AztecProtocol/aztec-packages/issues/6718): construction of the public input columns // should be done in the kernel - this is stubbed and underconstrained @@ -435,15 +427,23 @@ std::vector Execution::gen_trace(std::vector const& bytecode, uint32_t start_side_effect_counter = !public_inputs_vec.empty() ? static_cast(public_inputs_vec[PCPI_START_SIDE_EFFECT_COUNTER_OFFSET]) : 0; - std::vector> all_contract_bytecode; - all_contract_bytecode.reserve(execution_hints.externalcall_hints.size() + 1); - // Start with the main, top-level contract bytecode - all_contract_bytecode.push_back(bytecode); - for (const auto& externalcall_hint : execution_hints.externalcall_hints) { - all_contract_bytecode.emplace_back(externalcall_hint.bytecode); - } - AvmTraceBuilder trace_builder = Execution::trace_builder_constructor( - public_inputs, execution_hints, start_side_effect_counter, calldata, all_contract_bytecode); + + // This address is the top-level contract address + vinfo("Length of all contract bytecode: ", execution_hints.all_contract_bytecode.size()); + + FF contract_address = std::get<0>(public_inputs)[ADDRESS_SELECTOR]; + vinfo("Top level contract address: ", contract_address); + // We use it to extract the bytecode we need to execute + std::vector bytecode = + std::find_if(execution_hints.all_contract_bytecode.begin(), + execution_hints.all_contract_bytecode.end(), + [&](auto& contract) { return contract.contract_instance.address == contract_address; }) + ->bytecode; + + std::vector instructions = Deserialization::parse(bytecode); + vinfo("Deserialized " + std::to_string(instructions.size()) + " instructions"); + AvmTraceBuilder trace_builder = + Execution::trace_builder_constructor(public_inputs, execution_hints, start_side_effect_counter, calldata); // Copied version of pc maintained in trace builder. The value of pc is evolving based // on opcode logic and therefore is not maintained here. However, the next opcode in the execution diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.hpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.hpp index da222b87f672..6b20392decc8 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.hpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.hpp @@ -15,12 +15,10 @@ namespace bb::avm_trace { class Execution { public: static constexpr size_t SRS_SIZE = 1 << 22; - using TraceBuilderConstructor = - std::function calldata, - const std::vector>& all_contract_bytecode)>; + using TraceBuilderConstructor = std::function calldata)>; Execution() = default; @@ -30,8 +28,7 @@ class Execution { // Bytecode is currently the bytecode of the top-level function call // Eventually this will be the bytecode of the dispatch function of top-level contract - static std::vector gen_trace(std::vector const& bytecode, - std::vector const& calldata, + static std::vector gen_trace(std::vector const& calldata, std::vector const& public_inputs, std::vector& returndata, ExecutionHints const& execution_hints); @@ -43,7 +40,6 @@ class Execution { } static std::tuple prove( - std::vector const& bytecode, std::vector const& calldata = {}, std::vector const& public_inputs_vec = getDefaultPublicInputs(), ExecutionHints const& execution_hints = {}); diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution_hints.hpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution_hints.hpp index db01e5e3a8c1..1ea58dde7601 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution_hints.hpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution_hints.hpp @@ -12,7 +12,7 @@ struct ExternalCallHint { uint32_t l2_gas_used; uint32_t da_gas_used; FF end_side_effect_counter; - std::vector bytecode; + FF contract_address; }; // Add support for deserialization of ExternalCallHint. This is implicitly used by serialize::read @@ -25,12 +25,26 @@ inline void read(uint8_t const*& it, ExternalCallHint& hint) read(it, hint.l2_gas_used); read(it, hint.da_gas_used); read(it, hint.end_side_effect_counter); - read(it, hint.bytecode); + read(it, hint.contract_address); +} + +struct ContractClassIdHint { + FF artifact_hash; + FF private_fn_root; + FF public_bytecode_commitment; +}; + +inline void read(uint8_t const*& it, ContractClassIdHint& preimage) +{ + using serialize::read; + read(it, preimage.artifact_hash); + read(it, preimage.private_fn_root); + read(it, preimage.public_bytecode_commitment); } struct ContractInstanceHint { FF address; - FF instance_found_in_address; + bool exists; // Useful for membership checks FF salt; FF deployer_addr; FF contract_class_id; @@ -43,7 +57,7 @@ inline void read(uint8_t const*& it, ContractInstanceHint& hint) { using serialize::read; read(it, hint.address); - read(it, hint.instance_found_in_address); + read(it, hint.exists); read(it, hint.salt); read(it, hint.deployer_addr); read(it, hint.contract_class_id); @@ -51,6 +65,32 @@ inline void read(uint8_t const*& it, ContractInstanceHint& hint) read(it, hint.public_key_hash); } +struct AvmContractBytecode { + std::vector bytecode; + ContractInstanceHint contract_instance; + ContractClassIdHint contract_class_id_preimage; + + AvmContractBytecode() = default; + AvmContractBytecode(std::vector bytecode, + ContractInstanceHint contract_instance, + ContractClassIdHint contract_class_id_preimage) + : bytecode(std::move(bytecode)) + , contract_instance(contract_instance) + , contract_class_id_preimage(contract_class_id_preimage) + {} + AvmContractBytecode(std::vector bytecode) + : bytecode(std::move(bytecode)) + {} +}; + +inline void read(uint8_t const*& it, AvmContractBytecode& bytecode) +{ + using serialize::read; + read(it, bytecode.bytecode); + read(it, bytecode.contract_instance); + read(it, bytecode.contract_class_id_preimage); +} + struct ExecutionHints { std::vector> storage_value_hints; std::vector> note_hash_exists_hints; @@ -58,6 +98,8 @@ struct ExecutionHints { std::vector> l1_to_l2_message_exists_hints; std::vector externalcall_hints; std::map contract_instance_hints; + // We could make this address-indexed + std::vector all_contract_bytecode; ExecutionHints() = default; @@ -92,6 +134,11 @@ struct ExecutionHints { this->contract_instance_hints = std::move(contract_instance_hints); return *this; } + ExecutionHints& with_avm_contract_bytecode(std::vector all_contract_bytecode) + { + this->all_contract_bytecode = std::move(all_contract_bytecode); + return *this; + } static void push_vec_into_map(std::unordered_map& into_map, const std::vector>& from_pair_vec) @@ -144,14 +191,18 @@ struct ExecutionHints { contract_instance_hints[instance.address] = instance; } + std::vector all_contract_bytecode; + read(it, all_contract_bytecode); + if (it != data.data() + data.size()) { - throw_or_abort("Failed to deserialize ExecutionHints: only read" + std::to_string(it - data.data()) + + throw_or_abort("Failed to deserialize ExecutionHints: only read " + std::to_string(it - data.data()) + " bytes out of " + std::to_string(data.size()) + " bytes"); } return { std::move(storage_value_hints), std::move(note_hash_exists_hints), std::move(nullifier_exists_hints), std::move(l1_to_l2_message_exists_hints), - std::move(externalcall_hints), std::move(contract_instance_hints) }; + std::move(externalcall_hints), std::move(contract_instance_hints), + std::move(all_contract_bytecode) }; } private: @@ -160,13 +211,15 @@ struct ExecutionHints { std::vector> nullifier_exists_hints, std::vector> l1_to_l2_message_exists_hints, std::vector externalcall_hints, - std::map contract_instance_hints) + std::map contract_instance_hints, + std::vector all_contract_bytecode) : storage_value_hints(std::move(storage_value_hints)) , note_hash_exists_hints(std::move(note_hash_exists_hints)) , nullifier_exists_hints(std::move(nullifier_exists_hints)) , l1_to_l2_message_exists_hints(std::move(l1_to_l2_message_exists_hints)) , externalcall_hints(std::move(externalcall_hints)) , contract_instance_hints(std::move(contract_instance_hints)) + , all_contract_bytecode(std::move(all_contract_bytecode)) {} }; diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.cpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.cpp index 3765a2f79246..721dfb24a471 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.cpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.cpp @@ -23,6 +23,7 @@ #include "barretenberg/vm/avm/generated/full_row.hpp" #include "barretenberg/vm/avm/trace/bytecode_trace.hpp" #include "barretenberg/vm/avm/trace/common.hpp" +#include "barretenberg/vm/avm/trace/execution.hpp" #include "barretenberg/vm/avm/trace/fixed_bytes.hpp" #include "barretenberg/vm/avm/trace/fixed_gas.hpp" #include "barretenberg/vm/avm/trace/fixed_powers.hpp" @@ -280,14 +281,13 @@ void AvmTraceBuilder::finalise_mem_trace_lookup_counts() AvmTraceBuilder::AvmTraceBuilder(VmPublicInputs public_inputs, ExecutionHints execution_hints_, uint32_t side_effect_counter, - std::vector calldata, - const std::vector>& all_contract_bytecode) + std::vector calldata) // NOTE: we initialise the environment builder here as it requires public inputs : calldata(std::move(calldata)) , side_effect_counter(side_effect_counter) , execution_hints(std::move(execution_hints_)) , kernel_trace_builder(side_effect_counter, public_inputs, execution_hints) - , bytecode_trace_builder(all_contract_bytecode) + , bytecode_trace_builder(execution_hints_.all_contract_bytecode) { // TODO: think about cast gas_trace_builder.set_initial_gas( @@ -2545,7 +2545,7 @@ void AvmTraceBuilder::op_get_contract_instance(uint8_t indirect, uint32_t addres ContractInstanceHint contract_instance = execution_hints.contract_instance_hints.at(read_address.val); // NOTE: we don't write the first entry (the contract instance's address/key) to memory - std::vector contract_instance_vec = { contract_instance.instance_found_in_address, + std::vector contract_instance_vec = { contract_instance.exists ? FF::one() : FF::zero(), contract_instance.salt, contract_instance.deployer_addr, contract_instance.contract_class_id, diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.hpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.hpp index 684d41082411..0a18ec884931 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.hpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.hpp @@ -56,8 +56,7 @@ class AvmTraceBuilder { AvmTraceBuilder(VmPublicInputs public_inputs = {}, ExecutionHints execution_hints = {}, uint32_t side_effect_counter = 0, - std::vector calldata = {}, - const std::vector>& all_contract_bytecode = {}); + std::vector calldata = {}); uint32_t getPc() const { return pc; } diff --git a/yarn-project/bb-prover/src/avm_proving.test.ts b/yarn-project/bb-prover/src/avm_proving.test.ts index 20645d878e5f..d61e0900e7ac 100644 --- a/yarn-project/bb-prover/src/avm_proving.test.ts +++ b/yarn-project/bb-prover/src/avm_proving.test.ts @@ -1,4 +1,5 @@ import { AvmCircuitInputs, AvmVerificationKeyData, FunctionSelector, Gas, GlobalVariables } from '@aztec/circuits.js'; +import { makeContractClassPublic } from '@aztec/circuits.js/testing'; import { Fr } from '@aztec/foundation/fields'; import { createDebugLogger } from '@aztec/foundation/log'; import { AvmSimulator, PublicSideEffectTrace, type WorldStateDB } from '@aztec/simulator'; @@ -68,6 +69,9 @@ const proveAndVerifyAvmTestContract = async ( }).withAddress(environment.address); worldStateDB.getContractInstance.mockResolvedValue(Promise.resolve(contractInstance)); + const contractClass = makeContractClassPublic(); + worldStateDB.getContractClass.mockResolvedValue(Promise.resolve(contractClass)); + const storageValue = new Fr(5); worldStateDB.storageRead.mockResolvedValue(Promise.resolve(storageValue)); @@ -83,14 +87,14 @@ const proveAndVerifyAvmTestContract = async ( const logger = (msg: string, _data?: any) => internalLogger.verbose(msg); // Use a simple contract that emits a side effect - const bytecode = getAvmTestContractBytecode(functionName); + // const bytecode = getAvmTestContractBytecode(functionName); // The paths for the barretenberg binary and the write path are hardcoded for now. const bbPath = path.resolve('../../barretenberg/cpp/build/bin/bb'); const bbWorkingDirectory = await fs.mkdtemp(path.join(tmpdir(), 'bb-')); // First we simulate (though it's not needed in this simple case). const simulator = new AvmSimulator(context); - const avmResult = await simulator.executeBytecode(bytecode); + const avmResult = await simulator.execute(); if (assertionErrString == undefined) { expect(avmResult.reverted).toBe(false); @@ -112,7 +116,6 @@ const proveAndVerifyAvmTestContract = async ( const avmCircuitInputs = new AvmCircuitInputs( functionName, - /*bytecode=*/ simulator.getBytecode()!, // uncompressed bytecode /*calldata=*/ context.environment.calldata, /*publicInputs=*/ getPublicInputs(pxResult), /*avmHints=*/ pxResult.avmCircuitHints, diff --git a/yarn-project/bb-prover/src/bb/execute.ts b/yarn-project/bb-prover/src/bb/execute.ts index 22c4b5e9393f..f55a80033ae5 100644 --- a/yarn-project/bb-prover/src/bb/execute.ts +++ b/yarn-project/bb-prover/src/bb/execute.ts @@ -504,7 +504,6 @@ export async function generateAvmProof( } // Paths for the inputs - const bytecodePath = join(workingDirectory, AVM_BYTECODE_FILENAME); const calldataPath = join(workingDirectory, AVM_CALLDATA_FILENAME); const publicInputsPath = join(workingDirectory, AVM_PUBLIC_INPUTS_FILENAME); const avmHintsPath = join(workingDirectory, AVM_HINTS_FILENAME); @@ -525,10 +524,6 @@ export async function generateAvmProof( try { // Write the inputs to the working directory. - await fs.writeFile(bytecodePath, input.bytecode); - if (!filePresent(bytecodePath)) { - return { status: BB_RESULT.FAILURE, reason: `Could not write bytecode at ${bytecodePath}` }; - } await fs.writeFile( calldataPath, input.calldata.map(fr => fr.toBuffer()), @@ -553,8 +548,6 @@ export async function generateAvmProof( } const args = [ - '--avm-bytecode', - bytecodePath, '--avm-calldata', calldataPath, '--avm-public-inputs', diff --git a/yarn-project/circuits.js/src/structs/avm/avm.ts b/yarn-project/circuits.js/src/structs/avm/avm.ts index 21d7062d0e8d..b9832c0d34f9 100644 --- a/yarn-project/circuits.js/src/structs/avm/avm.ts +++ b/yarn-project/circuits.js/src/structs/avm/avm.ts @@ -1,7 +1,9 @@ +import { AztecAddress } from '@aztec/foundation/aztec-address'; import { Fr } from '@aztec/foundation/fields'; import { BufferReader, serializeToBuffer } from '@aztec/foundation/serialize'; import { type FieldsOf } from '@aztec/foundation/types'; +import { type ContractClassIdPreimage } from '../../contract/contract_class_id.js'; import { Gas } from '../gas.js'; import { PublicCircuitPublicInputs } from '../public_circuit_public_inputs.js'; import { Vector } from '../shared.js'; @@ -81,14 +83,13 @@ export class AvmExternalCallHint { * @param returnData the data returned by the external call. * @param gasUsed gas used by the external call (not including the cost of the CALL opcode itself). * @param endSideEffectCounter value of side effect counter at the end of the external call. - * @param bytecode currently the bytecode of the nested call function, will be changed to the contract bytecode (via the dispatch function) of the nested call */ constructor( public readonly success: Fr, returnData: Fr[], public readonly gasUsed: Gas, public readonly endSideEffectCounter: Fr, - public readonly bytecode: Buffer, + public readonly contractAddress: AztecAddress, ) { this.returnData = new Vector(returnData); } @@ -119,7 +120,7 @@ export class AvmExternalCallHint { this.returnData.items.length == 0 && this.gasUsed.isEmpty() && this.endSideEffectCounter.isZero() && - this.bytecode.length == 0 + this.contractAddress.isZero() ); } @@ -134,7 +135,7 @@ export class AvmExternalCallHint { fields.returnData.items, fields.gasUsed, fields.endSideEffectCounter, - fields.bytecode, + fields.contractAddress, ); } @@ -144,18 +145,7 @@ export class AvmExternalCallHint { * @returns An array of fields. */ static getFields(fields: FieldsOf) { - // Buffers aren't serialised the same way as they are read (lenth prefixed), so we need to do this manually. - const lengthPrefixedBytecode = Buffer.alloc(fields.bytecode.length + 4); - // Add a 4-byte length prefix to the bytecode. - lengthPrefixedBytecode.writeUInt32BE(fields.bytecode.length); - fields.bytecode.copy(lengthPrefixedBytecode, 4); - return [ - fields.success, - fields.returnData, - fields.gasUsed, - fields.endSideEffectCounter, - lengthPrefixedBytecode, - ] as const; + return [fields.success, fields.returnData, fields.gasUsed, fields.endSideEffectCounter, fields.contractAddress]; } /** @@ -168,9 +158,9 @@ export class AvmExternalCallHint { return new AvmExternalCallHint( Fr.fromBuffer(reader), reader.readVector(Fr), - reader.readObject(Gas), + Gas.fromBuffer(reader), Fr.fromBuffer(reader), - reader.readBuffer(), + AztecAddress.fromBuffer(reader), ); } @@ -187,7 +177,7 @@ export class AvmExternalCallHint { export class AvmContractInstanceHint { constructor( public readonly address: Fr, - public readonly exists: Fr, + public readonly exists: boolean, public readonly salt: Fr, public readonly deployer: Fr, public readonly contractClassId: Fr, @@ -217,7 +207,7 @@ export class AvmContractInstanceHint { isEmpty(): boolean { return ( this.address.isZero() && - this.exists.isZero() && + !this.exists && this.salt.isZero() && this.deployer.isZero() && this.contractClassId.isZero() && @@ -261,7 +251,7 @@ export class AvmContractInstanceHint { const reader = BufferReader.asReader(buff); return new AvmContractInstanceHint( Fr.fromBuffer(reader), - Fr.fromBuffer(reader), + reader.readBoolean(), Fr.fromBuffer(reader), Fr.fromBuffer(reader), Fr.fromBuffer(reader), @@ -280,6 +270,99 @@ export class AvmContractInstanceHint { } } +export class AvmContractBytecodeHints { + /* + * @param bytecode currently the bytecode of the nested call function, will be changed to the contract bytecode (via the dispatch function) of the nested call + * @param contractInstance the contract instance of the nested call, used to derive the contract address + * @param contractClassPreimage the contract class preimage of the nested call, used to derive the class id + * */ + constructor( + public readonly bytecode: Buffer, + public contractInstanceHint: AvmContractInstanceHint, + public contractClassHint: ContractClassIdPreimage, + ) {} + /** + * Serializes the inputs to a buffer. + * @returns - The inputs serialized to a buffer. + */ + toBuffer() { + return serializeToBuffer(...AvmContractBytecodeHints.getFields(this)); + } + + /** + * Serializes the inputs to a hex string. + * @returns The instance serialized to a hex string. + */ + toString() { + return this.toBuffer().toString('hex'); + } + + /** + * Is the struct empty? + * @returns whether all members are empty. + */ + isEmpty(): boolean { + return this.bytecode.length == 0; + } + + /** + * Creates a new instance from fields. + * @param fields - Fields to create the instance from. + * @returns A new AvmHint instance. + */ + static from(fields: FieldsOf): AvmContractBytecodeHints { + return new AvmContractBytecodeHints(fields.bytecode, fields.contractInstanceHint, fields.contractClassHint); + } + + /** + * Extracts fields from an instance. + * @param fields - Fields to create the instance from. + * @returns An array of fields. + */ + static getFields(fields: FieldsOf) { + // Buffers aren't serialised the same way as they are read (lenth prefixed), so we need to do this manually. + const lengthPrefixedBytecode = Buffer.alloc(fields.bytecode.length + 4); + // Add a 4-byte length prefix to the bytecode. + lengthPrefixedBytecode.writeUInt32BE(fields.bytecode.length); + fields.bytecode.copy(lengthPrefixedBytecode, 4); + return [ + lengthPrefixedBytecode, + /* Contract Instance - exclude version */ + fields.contractInstanceHint, + /* Contract Class */ + fields.contractClassHint.artifactHash, + fields.contractClassHint.privateFunctionsRoot, + fields.contractClassHint.publicBytecodeCommitment, + ] as const; + } + + /** + * Deserializes from a buffer or reader. + * @param buffer - Buffer or reader to read from. + * @returns The deserialized instance. + */ + static fromBuffer(buff: Buffer | BufferReader): AvmContractBytecodeHints { + const reader = BufferReader.asReader(buff); + const bytecode = reader.readBuffer(); + const contractInstanceHint = AvmContractInstanceHint.fromBuffer(reader); + const contractClassHint = { + artifactHash: Fr.fromBuffer(reader), + privateFunctionsRoot: Fr.fromBuffer(reader), + publicBytecodeCommitment: Fr.fromBuffer(reader), + }; + return new AvmContractBytecodeHints(bytecode, contractInstanceHint, contractClassHint); + } + + /** + * Deserializes from a hex string. + * @param str - Hex string to read from. + * @returns The deserialized instance. + */ + static fromString(str: string): AvmContractBytecodeHints { + return AvmContractBytecodeHints.fromBuffer(Buffer.from(str, 'hex')); + } +} + // TODO(dbanks12): rename AvmCircuitHints export class AvmExecutionHints { public readonly storageValues: Vector; @@ -288,6 +371,7 @@ export class AvmExecutionHints { public readonly l1ToL2MessageExists: Vector; public readonly externalCalls: Vector; public readonly contractInstances: Vector; + public readonly contractBytecodeHints: Vector; constructor( storageValues: AvmKeyValueHint[], @@ -296,6 +380,7 @@ export class AvmExecutionHints { l1ToL2MessageExists: AvmKeyValueHint[], externalCalls: AvmExternalCallHint[], contractInstances: AvmContractInstanceHint[], + contractBytecodeHints: AvmContractBytecodeHints[], ) { this.storageValues = new Vector(storageValues); this.noteHashExists = new Vector(noteHashExists); @@ -303,6 +388,7 @@ export class AvmExecutionHints { this.l1ToL2MessageExists = new Vector(l1ToL2MessageExists); this.externalCalls = new Vector(externalCalls); this.contractInstances = new Vector(contractInstances); + this.contractBytecodeHints = new Vector(contractBytecodeHints); } /** @@ -310,7 +396,7 @@ export class AvmExecutionHints { * @returns an empty instance. */ empty() { - return new AvmExecutionHints([], [], [], [], [], []); + return new AvmExecutionHints([], [], [], [], [], [], []); } /** @@ -340,7 +426,8 @@ export class AvmExecutionHints { this.nullifierExists.items.length == 0 && this.l1ToL2MessageExists.items.length == 0 && this.externalCalls.items.length == 0 && - this.contractInstances.items.length == 0 + this.contractInstances.items.length == 0 && + this.contractBytecodeHints.items.length == 0 ); } @@ -357,6 +444,7 @@ export class AvmExecutionHints { fields.l1ToL2MessageExists.items, fields.externalCalls.items, fields.contractInstances.items, + fields.contractBytecodeHints.items, ); } @@ -373,6 +461,7 @@ export class AvmExecutionHints { fields.l1ToL2MessageExists, fields.externalCalls, fields.contractInstances, + fields.contractBytecodeHints, ] as const; } @@ -390,6 +479,7 @@ export class AvmExecutionHints { reader.readVector(AvmKeyValueHint), reader.readVector(AvmExternalCallHint), reader.readVector(AvmContractInstanceHint), + reader.readVector(AvmContractBytecodeHints), ); } @@ -407,14 +497,13 @@ export class AvmExecutionHints { * @returns The empty instance. */ static empty() { - return new AvmExecutionHints([], [], [], [], [], []); + return new AvmExecutionHints([], [], [], [], [], [], []); } } export class AvmCircuitInputs { constructor( public readonly functionName: string, // only informational - public readonly bytecode: Buffer, public readonly calldata: Fr[], public readonly publicInputs: PublicCircuitPublicInputs, public readonly avmHints: AvmExecutionHints, @@ -429,8 +518,6 @@ export class AvmCircuitInputs { return serializeToBuffer( functionNameBuffer.length, functionNameBuffer, - this.bytecode.length, - this.bytecode, this.calldata.length, this.calldata, this.publicInputs.toBuffer(), @@ -453,7 +540,6 @@ export class AvmCircuitInputs { isEmpty(): boolean { return ( this.functionName.length == 0 && - this.bytecode.length == 0 && this.calldata.length == 0 && this.publicInputs.isEmpty() && this.avmHints.isEmpty() @@ -475,7 +561,7 @@ export class AvmCircuitInputs { * @returns An array of fields. */ static getFields(fields: FieldsOf) { - return [fields.functionName, fields.bytecode, fields.calldata, fields.publicInputs, fields.avmHints] as const; + return [fields.functionName, fields.calldata, fields.publicInputs, fields.avmHints] as const; } /** @@ -487,7 +573,6 @@ export class AvmCircuitInputs { const reader = BufferReader.asReader(buff); return new AvmCircuitInputs( /*functionName=*/ reader.readBuffer().toString(), - /*bytecode=*/ reader.readBuffer(), /*calldata=*/ reader.readVector(Fr), PublicCircuitPublicInputs.fromBuffer(reader), AvmExecutionHints.fromBuffer(reader), diff --git a/yarn-project/circuits.js/src/tests/factories.ts b/yarn-project/circuits.js/src/tests/factories.ts index 5b37aea61402..76f2c941bc9b 100644 --- a/yarn-project/circuits.js/src/tests/factories.ts +++ b/yarn-project/circuits.js/src/tests/factories.ts @@ -147,6 +147,7 @@ import { GasSettings } from '../structs/gas_settings.js'; import { GlobalVariables } from '../structs/global_variables.js'; import { Header } from '../structs/header.js'; import { + AvmContractBytecodeHints, EnqueuedCallData, PublicAccumulatedDataArrayLengths, PublicDataLeafHint, @@ -1400,10 +1401,23 @@ export function makeAvmExternalCallHint(seed = 0): AvmExternalCallHint { makeArray((seed % 100) + 10, i => new Fr(i), seed + 0x1000), new Gas(seed + 0x200, seed), new Fr(seed + 0x300), - makeBytes((seed % 100) + 10, seed + 0x400), + new Fr(seed + 0x400), ); } +export function makeAvmBytecodeHints(seed = 0): AvmContractBytecodeHints { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + const instance = makeAvmContractInstanceHint(seed); + const { artifactHash, privateFunctionsRoot, packedBytecode } = makeContractClassPublic(seed); + const publicBytecodeCommitment = computePublicBytecodeCommitment(packedBytecode); + + return new AvmContractBytecodeHints(packedBytecode, instance, { + artifactHash, + privateFunctionsRoot, + publicBytecodeCommitment, + }); +} + /** * Makes arbitrary AvmContractInstanceHint. * @param seed - The seed to use for generating the state reference. @@ -1412,7 +1426,7 @@ export function makeAvmExternalCallHint(seed = 0): AvmExternalCallHint { export function makeAvmContractInstanceHint(seed = 0): AvmContractInstanceHint { return new AvmContractInstanceHint( new Fr(seed), - new Fr(seed + 0x1), + true /* exists */, new Fr(seed + 0x2), new Fr(seed + 0x3), new Fr(seed + 0x4), @@ -1441,6 +1455,7 @@ export function makeAvmExecutionHints( l1ToL2MessageExists: makeVector(baseLength + 3, makeAvmKeyValueHint, seed + 0x4500), externalCalls: makeVector(baseLength + 4, makeAvmExternalCallHint, seed + 0x4600), contractInstances: makeVector(baseLength + 5, makeAvmContractInstanceHint, seed + 0x4700), + contractBytecodeHints: makeVector(baseLength + 6, makeAvmBytecodeHints, seed + 0x4800), ...overrides, }); } @@ -1453,7 +1468,6 @@ export function makeAvmExecutionHints( export function makeAvmCircuitInputs(seed = 0, overrides: Partial> = {}): AvmCircuitInputs { return AvmCircuitInputs.from({ functionName: `function${seed}`, - bytecode: makeBytes((seed % 100) + 100, seed), calldata: makeArray((seed % 100) + 10, i => new Fr(i), seed + 0x1000), publicInputs: makePublicCircuitPublicInputs(seed + 0x2000), avmHints: makeAvmExecutionHints(seed + 0x3000), diff --git a/yarn-project/ivc-integration/src/avm_integration.test.ts b/yarn-project/ivc-integration/src/avm_integration.test.ts index c2bb09e3675b..062386799a43 100644 --- a/yarn-project/ivc-integration/src/avm_integration.test.ts +++ b/yarn-project/ivc-integration/src/avm_integration.test.ts @@ -12,11 +12,12 @@ import { AVM_VERIFICATION_KEY_LENGTH_IN_FIELDS, PUBLIC_CIRCUIT_PUBLIC_INPUTS_LENGTH, } from '@aztec/circuits.js/constants'; +import { makeContractClassPublic } from '@aztec/circuits.js/testing'; import { Fr } from '@aztec/foundation/fields'; import { createDebugLogger } from '@aztec/foundation/log'; import { BufferReader } from '@aztec/foundation/serialize'; import { type FixedLengthArray } from '@aztec/noir-protocol-circuits-types/types'; -import { AvmSimulator, type PublicContractsDB, PublicSideEffectTrace, type WorldStateDB } from '@aztec/simulator'; +import { AvmSimulator, PublicSideEffectTrace, type WorldStateDB } from '@aztec/simulator'; import { getAvmTestContractBytecode, initContext, @@ -129,12 +130,12 @@ const proveAvmTestContract = async ( calldata: Fr[] = [], assertionErrString?: string, ): Promise => { + const worldStateDB = mock(); const startSideEffectCounter = 0; const functionSelector = FunctionSelector.random(); const globals = GlobalVariables.empty(); const environment = initExecutionEnvironment({ functionSelector, calldata, globals }); - const contractsDb = mock(); const contractInstance = new SerializableContractInstance({ version: 1, salt: new Fr(0x123), @@ -143,28 +144,29 @@ const proveAvmTestContract = async ( initializationHash: new Fr(0x101112), publicKeysHash: new Fr(0x161718), }).withAddress(environment.address); - contractsDb.getContractInstance.mockResolvedValue(await Promise.resolve(contractInstance)); + worldStateDB.getContractInstance.mockResolvedValue(await Promise.resolve(contractInstance)); + + const contractClass = makeContractClassPublic(); + worldStateDB.getContractClass.mockResolvedValue(await Promise.resolve(contractClass)); - const worldStateDB = mock(); const storageValue = new Fr(5); worldStateDB.storageRead.mockResolvedValue(await Promise.resolve(storageValue)); const trace = new PublicSideEffectTrace(startSideEffectCounter); const persistableState = initPersistableStateManager({ worldStateDB, trace }); const context = initContext({ env: environment, persistableState }); - const nestedCallBytecode = getAvmTestContractBytecode('add_args_return'); - jest.spyOn(worldStateDB, 'getBytecode').mockResolvedValue(nestedCallBytecode); + const bytecode = getAvmTestContractBytecode(functionName); + jest.spyOn(worldStateDB, 'getBytecode').mockResolvedValue(bytecode); const startGas = new Gas(context.machineState.gasLeft.daGas, context.machineState.gasLeft.l2Gas); // Use a simple contract that emits a side effect - const bytecode = getAvmTestContractBytecode(functionName); // The paths for the barretenberg binary and the write path are hardcoded for now. const bbPath = path.resolve('../../barretenberg/cpp/build/bin/bb'); const bbWorkingDirectory = await fs.mkdtemp(path.join(tmpdir(), 'bb-')); // First we simulate (though it's not needed in this simple case). const simulator = new AvmSimulator(context); - const avmResult = await simulator.executeBytecode(bytecode); + const avmResult = await simulator.execute(); if (assertionErrString == undefined) { expect(avmResult.reverted).toBe(false); @@ -186,7 +188,6 @@ const proveAvmTestContract = async ( const avmCircuitInputs = new AvmCircuitInputs( functionName, - /*bytecode=*/ simulator.getBytecode()!, // uncompressed bytecode /*calldata=*/ context.environment.calldata, /*publicInputs=*/ getPublicInputs(pxResult), /*avmHints=*/ pxResult.avmCircuitHints, diff --git a/yarn-project/prover-client/src/orchestrator/orchestrator.ts b/yarn-project/prover-client/src/orchestrator/orchestrator.ts index aec2271a83a9..cf3a1b50d59c 100644 --- a/yarn-project/prover-client/src/orchestrator/orchestrator.ts +++ b/yarn-project/prover-client/src/orchestrator/orchestrator.ts @@ -979,7 +979,6 @@ export class ProvingOrchestrator implements BlockProver { async (signal: AbortSignal) => { const inputs: AvmCircuitInputs = new AvmCircuitInputs( publicFunction.vmRequest!.functionName, - publicFunction.vmRequest!.bytecode, publicFunction.vmRequest!.calldata, publicFunction.vmRequest!.kernelRequest.inputs.publicCall.callStackItem.publicInputs, publicFunction.vmRequest!.avmHints, diff --git a/yarn-project/simulator/src/avm/avm_simulator.ts b/yarn-project/simulator/src/avm/avm_simulator.ts index e768d27958d6..7bbdb296feeb 100644 --- a/yarn-project/simulator/src/avm/avm_simulator.ts +++ b/yarn-project/simulator/src/avm/avm_simulator.ts @@ -105,7 +105,7 @@ export class AvmSimulator { const revertReason = revertReasonFromExceptionalHalt(err, this.context); // Note: "exceptional halts" cannot return data, hence [] - const results = new AvmContractCallResult(/*reverted=*/ true, /*output=*/ [], revertReason); + const results = new AvmContractCallResult(/*reverted=*/ true, /*outut=*/ [], revertReason); this.log.debug(`Context execution results: ${results.toString()}`); // Return results for processing by calling context return results; diff --git a/yarn-project/simulator/src/avm/journal/journal.ts b/yarn-project/simulator/src/avm/journal/journal.ts index 843b6ed90aec..0fd6d7e6bbe1 100644 --- a/yarn-project/simulator/src/avm/journal/journal.ts +++ b/yarn-project/simulator/src/avm/journal/journal.ts @@ -1,8 +1,10 @@ -import { AztecAddress, type FunctionSelector, type Gas } from '@aztec/circuits.js'; +import { AztecAddress, type FunctionSelector, type Gas, computePublicBytecodeCommitment } from '@aztec/circuits.js'; import { Fr } from '@aztec/foundation/fields'; import { type DebugLogger, createDebugLogger } from '@aztec/foundation/log'; import { SerializableContractInstance } from '@aztec/types/contracts'; +import assert from 'assert'; + import { type WorldStateDB } from '../../public/public_db_sources.js'; import { type TracedContractInstance } from '../../public/side_effect_trace.js'; import { type PublicSideEffectTraceInterface } from '../../public/side_effect_trace_interface.js'; @@ -236,10 +238,39 @@ export class AvmPersistableStateManager { } /** - * Get a contract's bytecode from the contracts DB + * Get a contract's bytecode from the contracts DB, also trace the contract class and instance */ public async getBytecode(contractAddress: AztecAddress, selector: FunctionSelector): Promise { - return await this.worldStateDB.getBytecode(contractAddress, selector); + let exists = true; + // If the bytecode is not found, we let the executor decide that to do + const bytecode = await this.worldStateDB.getBytecode(contractAddress, selector); + let contractInstance = await this.worldStateDB.getContractInstance(contractAddress); + // If the contract instance is not found, we assume it has not be deployed. We will also be unable to find the + // contract class as we will not have the id. While the class might exist, we hopefully won't need it to generate a proof (tbd). + if (contractInstance === undefined) { + exists = false; + contractInstance = SerializableContractInstance.empty().withAddress(contractAddress); + this.trace.traceExecutionStart( + bytecode ?? Buffer.alloc(1), + { exists, ...contractInstance }, + { + artifactHash: Fr.zero(), + privateFunctionsRoot: Fr.zero(), + publicBytecodeCommitment: Fr.zero(), + }, + ); + return bytecode; + } + const contractClass = await this.worldStateDB.getContractClass(contractInstance.contractClassId); + assert(contractClass, 'Contract class not found in DB'); + const contractClassPreimage = { + artifactHash: contractClass.artifactHash, + privateFunctionsRoot: contractClass.privateFunctionsRoot, + publicBytecodeCommitment: computePublicBytecodeCommitment(contractClass.packedBytecode), + }; + this.trace.traceExecutionStart(bytecode!, { exists, ...contractInstance }, contractClassPreimage); + + return bytecode; } /** diff --git a/yarn-project/simulator/src/public/side_effect_trace.ts b/yarn-project/simulator/src/public/side_effect_trace.ts index f96fac888441..d21f07eb7b91 100644 --- a/yarn-project/simulator/src/public/side_effect_trace.ts +++ b/yarn-project/simulator/src/public/side_effect_trace.ts @@ -1,11 +1,13 @@ import { PublicExecutionRequest, UnencryptedFunctionL2Logs, UnencryptedL2Log } from '@aztec/circuit-types'; import { + AvmContractBytecodeHints, AvmContractInstanceHint, AvmExecutionHints, AvmExternalCallHint, AvmKeyValueHint, AztecAddress, CallContext, + type ContractClassIdPreimage, ContractStorageRead, ContractStorageUpdateRequest, EthAddress, @@ -81,6 +83,36 @@ export class PublicSideEffectTrace implements PublicSideEffectTraceInterface { this.sideEffectCounter++; } + // This tracing function gets called everytime we start simulation/execution. + // This happens both when starting a new top-level trace and the start of every nested trace + // We use this to collect the AvmContractBytecodeHints + public traceExecutionStart( + bytecode: Buffer, + contractInstance: TracedContractInstance, + contractClass: ContractClassIdPreimage, + ) { + // Deduplicate - we might want a map here to make this more efficient + const idx = this.avmCircuitHints.contractBytecodeHints.items.findIndex( + hint => hint.contractInstanceHint.address === contractInstance.address, + ); + // If this is the first time we have seen the contract instance, add it to the hints + if (idx === -1) { + const instance = new AvmContractInstanceHint( + contractInstance.address, + contractInstance.exists, + contractInstance.salt, + contractInstance.deployer, + contractInstance.contractClassId, + contractInstance.initializationHash, + contractInstance.publicKeysHash, + ); + this.avmCircuitHints.contractBytecodeHints.items.push( + new AvmContractBytecodeHints(bytecode, instance, contractClass), + ); + this.logger.verbose(`New contract instance: ${contractInstance.address} added`); + } + } + public tracePublicStorageRead(storageAddress: Fr, slot: Fr, value: Fr, _exists: boolean, _cached: boolean) { // TODO(4805): check if some threshold is reached for max storage reads // (need access to parent length, or trace needs to be initialized with parent's contents) @@ -196,7 +228,7 @@ export class PublicSideEffectTrace implements PublicSideEffectTraceInterface { this.avmCircuitHints.contractInstances.items.push( new AvmContractInstanceHint( instance.address, - new Fr(instance.exists ? 1 : 0), + instance.exists, instance.salt, instance.deployer, instance.contractClassId, @@ -256,7 +288,7 @@ export class PublicSideEffectTrace implements PublicSideEffectTraceInterface { result.returnValues, gasUsed, result.endSideEffectCounter, - bytecode, + nestedEnvironment.address, ), ); } diff --git a/yarn-project/simulator/src/public/side_effect_trace_interface.ts b/yarn-project/simulator/src/public/side_effect_trace_interface.ts index 91326fb021b4..8aa4e3b71284 100644 --- a/yarn-project/simulator/src/public/side_effect_trace_interface.ts +++ b/yarn-project/simulator/src/public/side_effect_trace_interface.ts @@ -1,4 +1,4 @@ -import { type Gas } from '@aztec/circuits.js'; +import { type ContractClassIdPreimage, type Gas } from '@aztec/circuits.js'; import { type Fr } from '@aztec/foundation/fields'; import { type AvmContractCallResult } from '../avm/avm_contract_call_result.js'; @@ -8,6 +8,11 @@ import { type TracedContractInstance } from './side_effect_trace.js'; export interface PublicSideEffectTraceInterface { fork(): PublicSideEffectTraceInterface; getCounter(): number; + traceExecutionStart( + bytecode: Buffer, + contractInstance: TracedContractInstance, + contractClass: ContractClassIdPreimage, + ): void; tracePublicStorageRead(storageAddress: Fr, slot: Fr, value: Fr, exists: boolean, cached: boolean): void; tracePublicStorageWrite(storageAddress: Fr, slot: Fr, value: Fr): void; traceNoteHashCheck(storageAddress: Fr, noteHash: Fr, leafIndex: Fr, exists: boolean): void;