diff --git a/avm-transpiler/src/transpile.rs b/avm-transpiler/src/transpile.rs index b5251154a725..d19221c251cf 100644 --- a/avm-transpiler/src/transpile.rs +++ b/avm-transpiler/src/transpile.rs @@ -413,9 +413,6 @@ fn handle_foreign_call( "avmOpcodeNullifierExists" => handle_nullifier_exists(avm_instrs, destinations, inputs), "avmOpcodeL1ToL2MsgExists" => handle_l1_to_l2_msg_exists(avm_instrs, destinations, inputs), "avmOpcodeSendL2ToL1Msg" => handle_send_l2_to_l1_msg(avm_instrs, destinations, inputs), - "avmOpcodeGetContractInstance" => { - handle_get_contract_instance(avm_instrs, destinations, inputs); - } "avmOpcodeCalldataCopy" => handle_calldata_copy(avm_instrs, destinations, inputs), "avmOpcodeReturn" => handle_return(avm_instrs, destinations, inputs), "avmOpcodeStorageRead" => handle_storage_read(avm_instrs, destinations, inputs), @@ -425,6 +422,10 @@ fn handle_foreign_call( _ if inputs.is_empty() && destinations.len() == 1 => { handle_getter_instruction(avm_instrs, function, destinations, inputs); } + // Get contract instance variations. + _ if function.starts_with("avmOpcodeGetContractInstance") => { + handle_get_contract_instance(avm_instrs, function, destinations, inputs); + } // Anything else. _ => panic!("Transpiler doesn't know how to process ForeignCall function {}", function), } @@ -1274,22 +1275,42 @@ fn handle_storage_write( /// Emit a GETCONTRACTINSTANCE opcode fn handle_get_contract_instance( avm_instrs: &mut Vec, + function: &str, destinations: &Vec, inputs: &Vec, ) { + enum ContractInstanceMember { + DEPLOYER, + CLASS_ID, + INIT_HASH, + } + assert!(inputs.len() == 1); - assert!(destinations.len() == 1); + assert!(destinations.len() == 2); + + let member_idx = match function { + "avmOpcodeGetContractInstanceDeployer" => ContractInstanceMember::DEPLOYER, + "avmOpcodeGetContractInstanceClassId" => ContractInstanceMember::CLASS_ID, + "avmOpcodeGetContractInstanceInitializationHash" => ContractInstanceMember::INIT_HASH, + _ => panic!("Transpiler doesn't know how to process function {:?}", function), + }; let address_offset_maybe = inputs[0]; let address_offset = match address_offset_maybe { - ValueOrArray::MemoryAddress(slot_offset) => slot_offset, + ValueOrArray::MemoryAddress(offset) => offset, _ => panic!("GETCONTRACTINSTANCE address should be a single value"), }; let dest_offset_maybe = destinations[0]; let dest_offset = match dest_offset_maybe { - ValueOrArray::HeapArray(HeapArray { pointer, .. }) => pointer, - _ => panic!("GETCONTRACTINSTANCE destination should be an array"), + ValueOrArray::MemoryAddress(offset) => offset, + _ => panic!("GETCONTRACTINSTANCE dst destination should be a single value"), + }; + + let exists_offset_maybe = destinations[1]; + let exists_offset = match exists_offset_maybe { + ValueOrArray::MemoryAddress(offset) => offset, + _ => panic!("GETCONTRACTINSTANCE exists destination should be a single value"), }; avm_instrs.push(AvmInstruction { @@ -1297,12 +1318,15 @@ fn handle_get_contract_instance( indirect: Some( AddressingModeBuilder::default() .direct_operand(&address_offset) - .indirect_operand(&dest_offset) + .direct_operand(&dest_offset) + .direct_operand(&exists_offset) .build(), ), operands: vec![ - AvmOperand::U32 { value: address_offset.to_usize() as u32 }, - AvmOperand::U32 { value: dest_offset.to_usize() as u32 }, + AvmOperand::U8 { value: member_idx as u8 }, + AvmOperand::U16 { value: address_offset.to_usize() as u16 }, + AvmOperand::U16 { value: dest_offset.to_usize() as u16 }, + AvmOperand::U16 { value: exists_offset.to_usize() as u16 }, ], ..Default::default() }); diff --git a/barretenberg/cpp/pil/avm/main.pil b/barretenberg/cpp/pil/avm/main.pil index 188f8dacca26..ee7a6c1f130b 100644 --- a/barretenberg/cpp/pil/avm/main.pil +++ b/barretenberg/cpp/pil/avm/main.pil @@ -350,9 +350,8 @@ namespace main(256); // op_err * (sel_op_fdiv + sel_op_XXX + ... - 1) == 0 // Note that the above is even a stronger constraint, as it shows // that exactly one sel_op_XXX must be true. - // At this time, we have only division producing an error. #[SUBOP_ERROR_RELEVANT_OP] - op_err * ((sel_op_fdiv + sel_op_div) - 1) = 0; + op_err * ((sel_op_fdiv + sel_op_div + sel_op_get_contract_instance) - 1) = 0; // TODO: constraint that we stop execution at the first error (tag_err or op_err) // An error can only happen at the last sub-operation row. diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/generated/relations/main.hpp b/barretenberg/cpp/src/barretenberg/vm/avm/generated/relations/main.hpp index 8869a0fb8476..91cc7ded6142 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/generated/relations/main.hpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/generated/relations/main.hpp @@ -567,7 +567,9 @@ template class mainImpl { } { using Accumulator = typename std::tuple_element_t<79, ContainerOverSubrelations>; - auto tmp = (new_term.main_op_err * ((new_term.main_sel_op_fdiv + new_term.main_sel_op_div) - FF(1))); + auto tmp = (new_term.main_op_err * (((new_term.main_sel_op_fdiv + new_term.main_sel_op_div) + + new_term.main_sel_op_get_contract_instance) - + FF(1))); tmp *= scaling_factor; std::get<79>(evals) += typename Accumulator::View(tmp); } 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 03cce00a6a84..286051762a1d 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/tests/execution.test.cpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/tests/execution.test.cpp @@ -2055,43 +2055,83 @@ TEST_F(AvmExecutionTests, opCallOpcodes) validate_trace(std::move(trace), public_inputs, calldata, returndata); } -TEST_F(AvmExecutionTests, opGetContractInstanceOpcodes) +TEST_F(AvmExecutionTests, opGetContractInstanceOpcode) { - std::string bytecode_hex = to_hex(OpCode::SET_8) + // opcode SET - "00" // Indirect flag - + to_hex(AvmMemoryTag::U32) + - "00" // val - "00" // dst_offset - + to_hex(OpCode::SET_8) + // opcode SET - "00" // Indirect flag - + to_hex(AvmMemoryTag::U32) + - "01" // val - "01" + - to_hex(OpCode::CALLDATACOPY) + // opcode CALLDATACOPY for addr - "00" // Indirect flag - "0000" // cd_offset - "0001" // copy_size - "0001" // dst_offset, (i.e. where we store the addr) - + to_hex(OpCode::SET_8) + // opcode SET for the indirect dst offset - "00" // Indirect flag - + to_hex(AvmMemoryTag::U32) + - "03" // val i - "02" + // dst_offset 2 - to_hex(OpCode::GETCONTRACTINSTANCE) + // opcode CALL - "02" // Indirect flag - "00000001" // address offset - "00000002" // dst offset - + to_hex(OpCode::RETURN) + // opcode RETURN - "00" // Indirect flag - "0003" // ret offset 3 - "0006"; // ret size 6 + const uint8_t address_byte = 0x42; + const FF address(address_byte); + const FF deployer = 42; + const FF class_id = 66; + const FF init_hash = 99; + const FF exists = 1; + + // Generate Hint for call operation + // Note: opcode does not write 'address' into memory + // We store this random value since it's part of the return - we could return the rest as well but we don't need to. + auto returned_point = grumpkin::g1::affine_element::random_element(); + PublicKeysHint public_keys_hints = { + returned_point, + grumpkin::g1::affine_element::random_element(), + grumpkin::g1::affine_element::random_element(), + grumpkin::g1::affine_element::random_element(), + }; + const auto execution_hints = ExecutionHints().with_contract_instance_hints( + { { address, { address, exists, /*salt=*/2, deployer, class_id, init_hash, public_keys_hints } } }); + + std::string bytecode_hex = to_hex(OpCode::SET_8) + // opcode SET + "00" // Indirect flag + + to_hex(AvmMemoryTag::U8) + to_hex(address_byte) + // val + "01" // dst_offset 0 + + to_hex(OpCode::GETCONTRACTINSTANCE) + // opcode GETCONTRACTINSTANCE + "00" // Indirect flag + + to_hex(static_cast(ContractInstanceMember::DEPLOYER)) + // member enum + "0001" // address offset + "0010" // dst offset + "0011" // exists offset + + to_hex(OpCode::GETCONTRACTINSTANCE) + // opcode GETCONTRACTINSTANCE + "00" // Indirect flag + + to_hex(static_cast(ContractInstanceMember::CLASS_ID)) + // member enum + "0001" // address offset + "0012" // dst offset + "0013" // exists offset + + to_hex(OpCode::GETCONTRACTINSTANCE) + // opcode GETCONTRACTINSTANCE + "00" // Indirect flag + + to_hex(static_cast(ContractInstanceMember::INIT_HASH)) + // member enum + "0001" // address offset + "0014" // dst offset + "0015" // exists offset + + to_hex(OpCode::RETURN) + // opcode RETURN + "00" // Indirect flag + "0010" // ret offset 1 + "0006"; // ret size 6 (dst & exists for all 3) auto bytecode = hex_to_bytes(bytecode_hex); auto instructions = Deserialization::parse(bytecode); - FF address = 10; - std::vector calldata = { address }; - std::vector returndata = {}; + ASSERT_THAT(instructions, SizeIs(5)); + + std::vector const calldata{}; + // alternating member value, exists bool + std::vector const expected_returndata = { + deployer, 1, class_id, 1, init_hash, 1, + }; + + std::vector returndata{}; + auto trace = Execution::gen_trace(instructions, returndata, calldata, public_inputs_vec, execution_hints); + + validate_trace(std::move(trace), public_inputs, calldata, returndata); + + // Validate returndata + EXPECT_EQ(returndata, expected_returndata); +} + +TEST_F(AvmExecutionTests, opGetContractInstanceOpcodeBadEnum) +{ + const uint8_t address_byte = 0x42; + const FF address(address_byte); + const FF deployer = 42; + const FF class_id = 66; + const FF init_hash = 99; + const FF exists = 1; // Generate Hint for call operation // Note: opcode does not write 'address' into memory @@ -2103,14 +2143,31 @@ TEST_F(AvmExecutionTests, opGetContractInstanceOpcodes) grumpkin::g1::affine_element::random_element(), grumpkin::g1::affine_element::random_element(), }; - auto execution_hints = - ExecutionHints().with_contract_instance_hints({ { address, { address, 1, 2, 3, 4, 5, public_keys_hints } } }); + const auto execution_hints = ExecutionHints().with_contract_instance_hints( + { { address, { address, exists, /*salt=*/2, deployer, class_id, init_hash, public_keys_hints } } }); + + std::string bytecode_hex = to_hex(OpCode::SET_8) + // opcode SET + "00" // Indirect flag + + to_hex(AvmMemoryTag::U8) + to_hex(address_byte) + // val + "01" // dst_offset 0 + + to_hex(OpCode::GETCONTRACTINSTANCE) + // opcode GETCONTRACTINSTANCE + "00" // Indirect flag + + to_hex(static_cast(ContractInstanceMember::MAX_MEMBER)) + // member enum + "0001" // address offset + "0010" // dst offset + "0011"; // exists offset + + auto bytecode = hex_to_bytes(bytecode_hex); + auto instructions = Deserialization::parse(bytecode); + + std::vector const calldata{}; + std::vector returndata{}; auto trace = Execution::gen_trace(instructions, returndata, calldata, public_inputs_vec, execution_hints); - EXPECT_EQ(returndata, std::vector({ 1, 2, 3, 4, 5, returned_point.x })); // The first one represents true validate_trace(std::move(trace), public_inputs, calldata, returndata); } + // Negative test detecting an invalid opcode byte. TEST_F(AvmExecutionTests, invalidOpcode) { diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/deserialization.cpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/deserialization.cpp index 88be6bca8dfe..779eb0c21e89 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/deserialization.cpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/deserialization.cpp @@ -136,7 +136,8 @@ const std::unordered_map> OPCODE_WIRE_FORMAT = OperandType::UINT16, /*TODO: leafIndexOffset is not constrained*/ OperandType::UINT16, OperandType::UINT16 } }, - { OpCode::GETCONTRACTINSTANCE, { OperandType::INDIRECT8, OperandType::UINT32, OperandType::UINT32 } }, + { OpCode::GETCONTRACTINSTANCE, + { OperandType::INDIRECT8, OperandType::UINT8, OperandType::UINT16, OperandType::UINT16, OperandType::UINT16 } }, { OpCode::EMITUNENCRYPTEDLOG, { OperandType::INDIRECT8, diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.cpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.cpp index 0c276efe07db..654829472d2c 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.cpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/execution.cpp @@ -621,8 +621,10 @@ std::vector Execution::gen_trace(std::vector const& instructio break; case OpCode::GETCONTRACTINSTANCE: trace_builder.op_get_contract_instance(std::get(inst.operands.at(0)), - std::get(inst.operands.at(1)), - std::get(inst.operands.at(2))); + std::get(inst.operands.at(1)), + std::get(inst.operands.at(2)), + std::get(inst.operands.at(3)), + std::get(inst.operands.at(4))); break; // Accrued Substate diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/opcode.hpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/opcode.hpp index 4aef84ee5bfd..b0c5efd9a84a 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/opcode.hpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/opcode.hpp @@ -128,6 +128,14 @@ enum class EnvironmentVariable { MAX_ENV_VAR }; +enum class ContractInstanceMember { + DEPLOYER, + CLASS_ID, + INIT_HASH, + // sentinel + MAX_MEMBER, +}; + class Bytecode { public: static bool is_valid(uint8_t byte); diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.cpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.cpp index e7133c81dbbc..3659ee2ac218 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.cpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.cpp @@ -1259,51 +1259,70 @@ Row AvmTraceBuilder::create_kernel_lookup_opcode(uint8_t indirect, uint32_t dst_ void AvmTraceBuilder::op_get_env_var(uint8_t indirect, uint8_t env_var, uint32_t dst_offset) { - ASSERT(env_var < static_cast(EnvironmentVariable::MAX_ENV_VAR)); - EnvironmentVariable var = static_cast(env_var); - - switch (var) { - case EnvironmentVariable::ADDRESS: - op_address(indirect, dst_offset); - break; - case EnvironmentVariable::SENDER: - op_sender(indirect, dst_offset); - break; - case EnvironmentVariable::FUNCTIONSELECTOR: - op_function_selector(indirect, dst_offset); - break; - case EnvironmentVariable::TRANSACTIONFEE: - op_transaction_fee(indirect, dst_offset); - break; - case EnvironmentVariable::CHAINID: - op_chain_id(indirect, dst_offset); - break; - case EnvironmentVariable::VERSION: - op_version(indirect, dst_offset); - break; - case EnvironmentVariable::BLOCKNUMBER: - op_block_number(indirect, dst_offset); - break; - case EnvironmentVariable::TIMESTAMP: - op_timestamp(indirect, dst_offset); - break; - case EnvironmentVariable::FEEPERL2GAS: - op_fee_per_l2_gas(indirect, dst_offset); - break; - case EnvironmentVariable::FEEPERDAGAS: - op_fee_per_da_gas(indirect, dst_offset); - break; - case EnvironmentVariable::ISSTATICCALL: - op_is_static_call(indirect, dst_offset); - break; - case EnvironmentVariable::L2GASLEFT: - op_l2gasleft(indirect, dst_offset); - break; - case EnvironmentVariable::DAGASLEFT: - op_dagasleft(indirect, dst_offset); - break; - default: - throw std::runtime_error("Invalid environment variable"); + if (env_var >= static_cast(EnvironmentVariable::MAX_ENV_VAR)) { + // Error, bad enum operand + // TODO(9395): constrain this via range check + auto const clk = static_cast(main_trace.size()) + 1; + const auto row = Row{ + .main_clk = clk, + .main_call_ptr = call_ptr, + .main_internal_return_ptr = internal_return_ptr, + .main_op_err = FF(1), + .main_pc = pc++, + .main_sel_op_address = FF(1), // TODO(9407): what selector should this be? + }; + + // Constrain gas cost + gas_trace_builder.constrain_gas(static_cast(row.main_clk), OpCode::GETENVVAR_16); + + main_trace.push_back(row); + } else { + EnvironmentVariable var = static_cast(env_var); + + switch (var) { + case EnvironmentVariable::ADDRESS: + op_address(indirect, dst_offset); + break; + case EnvironmentVariable::SENDER: + op_sender(indirect, dst_offset); + break; + case EnvironmentVariable::FUNCTIONSELECTOR: + op_function_selector(indirect, dst_offset); + break; + case EnvironmentVariable::TRANSACTIONFEE: + op_transaction_fee(indirect, dst_offset); + break; + case EnvironmentVariable::CHAINID: + op_chain_id(indirect, dst_offset); + break; + case EnvironmentVariable::VERSION: + op_version(indirect, dst_offset); + break; + case EnvironmentVariable::BLOCKNUMBER: + op_block_number(indirect, dst_offset); + break; + case EnvironmentVariable::TIMESTAMP: + op_timestamp(indirect, dst_offset); + break; + case EnvironmentVariable::FEEPERL2GAS: + op_fee_per_l2_gas(indirect, dst_offset); + break; + case EnvironmentVariable::FEEPERDAGAS: + op_fee_per_da_gas(indirect, dst_offset); + break; + case EnvironmentVariable::ISSTATICCALL: + op_is_static_call(indirect, dst_offset); + break; + case EnvironmentVariable::L2GASLEFT: + op_l2gasleft(indirect, dst_offset); + break; + case EnvironmentVariable::DAGASLEFT: + op_dagasleft(indirect, dst_offset); + break; + default: + throw std::runtime_error("Invalid environment variable"); + break; + } } } @@ -2357,48 +2376,100 @@ void AvmTraceBuilder::op_l1_to_l2_msg_exists(uint8_t indirect, debug("l1_to_l2_msg_exists side-effect cnt: ", side_effect_counter); } -void AvmTraceBuilder::op_get_contract_instance(uint8_t indirect, uint32_t address_offset, uint32_t dst_offset) +void AvmTraceBuilder::op_get_contract_instance( + uint8_t indirect, uint8_t member_enum, uint16_t address_offset, uint16_t dst_offset, uint16_t exists_offset) { auto clk = static_cast(main_trace.size()) + 1; + // Constrain gas cost + gas_trace_builder.constrain_gas(clk, OpCode::GETCONTRACTINSTANCE); - auto [resolved_address_offset, resolved_dst_offset] = - Addressing<2>::fromWire(indirect, call_ptr).resolve({ address_offset, dst_offset }, mem_trace_builder); + if (member_enum >= static_cast(ContractInstanceMember::MAX_MEMBER)) { + // Error, bad enum operand + // TODO(9393): constrain this via range check + const auto row = Row{ + .main_clk = clk, + .main_call_ptr = call_ptr, + .main_internal_return_ptr = internal_return_ptr, + .main_op_err = FF(1), + .main_pc = pc++, + .main_sel_op_get_contract_instance = FF(1), + }; + main_trace.push_back(row); - auto read_address = constrained_read_from_memory( - call_ptr, clk, resolved_address_offset, AvmMemoryTag::FF, AvmMemoryTag::FF, IntermRegister::IA); - bool tag_match = read_address.tag_match; + } else { - // Constrain gas cost - gas_trace_builder.constrain_gas(clk, OpCode::GETCONTRACTINSTANCE); + ContractInstanceMember chosen_member = static_cast(member_enum); + + auto [resolved_address_offset, resolved_dst_offset, resolved_exists_offset] = + Addressing<3>::fromWire(indirect, call_ptr) + .resolve({ address_offset, dst_offset, exists_offset }, mem_trace_builder); + + auto read_address = constrained_read_from_memory( + call_ptr, clk, resolved_address_offset, AvmMemoryTag::FF, AvmMemoryTag::FF, IntermRegister::IA); + bool tag_match = read_address.tag_match; + + // Read the contract instance + ContractInstanceHint instance = execution_hints.contract_instance_hints.at(read_address.val); + + FF member_value; + switch (chosen_member) { + case ContractInstanceMember::DEPLOYER: + member_value = instance.deployer_addr; + break; + case ContractInstanceMember::CLASS_ID: + member_value = instance.contract_class_id; + break; + case ContractInstanceMember::INIT_HASH: + member_value = instance.initialisation_hash; + break; + default: + member_value = 0; + break; + } - main_trace.push_back(Row{ - .main_clk = clk, - .main_ia = read_address.val, - .main_ind_addr_a = FF(read_address.indirect_address), - .main_internal_return_ptr = FF(internal_return_ptr), - .main_mem_addr_a = FF(read_address.direct_address), - .main_pc = FF(pc++), - .main_r_in_tag = FF(static_cast(AvmMemoryTag::FF)), - .main_sel_mem_op_a = FF(1), - .main_sel_op_get_contract_instance = FF(1), - .main_sel_resolve_ind_addr_a = FF(static_cast(read_address.is_indirect)), - .main_tag_err = FF(static_cast(!tag_match)), - }); + // TODO:(8603): once instructions can have multiple different tags for writes, write dst as FF and exists as U1 + // auto write_dst = constrained_write_to_memory(call_ptr, clk, resolved_dst_offset, member_value, + // AvmMemoryTag::FF, AvmMemoryTag::FF, IntermRegister::IC); auto write_exists = + // constrained_write_to_memory(call_ptr, clk, resolved_exists_offset, instance.instance_found_in_address, + // AvmMemoryTag::FF, AvmMemoryTag::FF, IntermRegister::ID); - // Read the contract instance - ContractInstanceHint contract_instance = execution_hints.contract_instance_hints.at(read_address.val); - std::vector public_key_fields = contract_instance.public_keys.to_fields(); - // 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, - contract_instance.salt, - contract_instance.deployer_addr, - contract_instance.contract_class_id, - contract_instance.initialisation_hash }; - contract_instance_vec.insert(contract_instance_vec.end(), public_key_fields.begin(), public_key_fields.end()); - write_slice_to_memory(resolved_dst_offset, AvmMemoryTag::FF, contract_instance_vec); - - debug("contract_instance cnt: ", side_effect_counter); - side_effect_counter++; + main_trace.push_back(Row{ + .main_clk = clk, + .main_call_ptr = call_ptr, + .main_ia = read_address.val, + // TODO:(8603): uncomment this and below blocks once instructions can have multiple different tags for + // writes + //.main_ic = write_dst.val, + //.main_id = write_exists.val, + .main_ind_addr_a = FF(read_address.indirect_address), + //.main_ind_addr_c = FF(write_dst.indirect_address), + //.main_ind_addr_d = FF(write_exists.indirect_address), + .main_internal_return_ptr = FF(internal_return_ptr), + .main_mem_addr_a = FF(read_address.direct_address), + //.main_mem_addr_c = FF(write_dst.direct_address), + //.main_mem_addr_d = FF(write_exists.direct_address), + .main_pc = FF(pc++), + .main_r_in_tag = FF(static_cast(AvmMemoryTag::FF)), + .main_sel_mem_op_a = FF(1), + //.main_sel_mem_op_c = FF(1), + //.main_sel_mem_op_d = FF(1), + .main_sel_op_get_contract_instance = FF(1), + .main_sel_resolve_ind_addr_a = FF(static_cast(read_address.is_indirect)), + //.main_sel_resolve_ind_addr_c = FF(static_cast(write_dst.is_indirect)), + //.main_sel_resolve_ind_addr_d = FF(static_cast(write_exists.is_indirect)), + .main_tag_err = FF(static_cast(!tag_match)), + }); + + // TODO:(8603): once instructions can have multiple different tags for writes, remove this and do a constrained + // writes + write_to_memory(resolved_dst_offset, member_value, AvmMemoryTag::FF); + write_to_memory(resolved_exists_offset, instance.instance_found_in_address, AvmMemoryTag::U1); + + // TODO(dbanks12): compute contract address nullifier from instance preimage and perform membership check + + debug("contract_instance cnt: ", side_effect_counter); + side_effect_counter++; + } } /************************************************************************************************** diff --git a/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.hpp b/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.hpp index 9380c36846e3..843d6f8c83f9 100644 --- a/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.hpp +++ b/barretenberg/cpp/src/barretenberg/vm/avm/trace/trace.hpp @@ -112,7 +112,8 @@ class AvmTraceBuilder { uint32_t log_offset, uint32_t leaf_index_offset, uint32_t dest_offset); - void op_get_contract_instance(uint8_t indirect, uint32_t address_offset, uint32_t dst_offset); + void op_get_contract_instance( + uint8_t indirect, uint8_t member_enum, uint16_t address_offset, uint16_t dst_offset, uint16_t exists_offset); // Accrued Substate void op_emit_unencrypted_log(uint8_t indirect, uint32_t log_offset, uint32_t log_size_offset); diff --git a/noir-projects/aztec-nr/aztec/src/context/public_context.nr b/noir-projects/aztec-nr/aztec/src/context/public_context.nr index 8c09a49c8c0e..32513cadf7f8 100644 --- a/noir-projects/aztec-nr/aztec/src/context/public_context.nr +++ b/noir-projects/aztec-nr/aztec/src/context/public_context.nr @@ -210,13 +210,14 @@ unconstrained fn address() -> AztecAddress { unconstrained fn sender() -> AztecAddress { sender_opcode() } +// TODO(9396): Remove. unconstrained fn portal() -> EthAddress { portal_opcode() } -// UNUSED: Remove. -// unconstrained fn function_selector() -> u32 { -// function_selector_opcode() -// } +// TODO(9396): Remove. +unconstrained fn function_selector() -> u32 { + function_selector_opcode() +} unconstrained fn transaction_fee() -> Field { transaction_fee_opcode() } @@ -317,9 +318,9 @@ unconstrained fn sender_opcode() -> AztecAddress {} #[oracle(avmOpcodePortal)] unconstrained fn portal_opcode() -> EthAddress {} -// UNUSED: Remove. -// #[oracle(avmOpcodeFunctionSelector)] -// unconstrained fn function_selector_opcode() -> u32 {} +// TODO(9396): Remove. +#[oracle(avmOpcodeFunctionSelector)] +unconstrained fn function_selector_opcode() -> u32 {} #[oracle(avmOpcodeTransactionFee)] unconstrained fn transaction_fee_opcode() -> Field {} diff --git a/noir-projects/aztec-nr/aztec/src/initializer.nr b/noir-projects/aztec-nr/aztec/src/initializer.nr index ec966af74af2..0f5ddae86684 100644 --- a/noir-projects/aztec-nr/aztec/src/initializer.nr +++ b/noir-projects/aztec-nr/aztec/src/initializer.nr @@ -5,7 +5,9 @@ use dep::protocol_types::{ use crate::{ context::{PrivateContext, PublicContext}, oracle::get_contract_instance::get_contract_instance, - oracle::get_contract_instance::get_contract_instance_avm, + oracle::get_contract_instance::{ + get_contract_instance_deployer_avm, get_contract_instance_initialization_hash_avm, + }, }; pub fn mark_as_initialized_public(context: &mut PublicContext) { @@ -36,11 +38,12 @@ fn compute_unsiloed_contract_initialization_nullifier(address: AztecAddress) -> pub fn assert_initialization_matches_address_preimage_public(context: PublicContext) { let address = context.this_address(); - let instance = get_contract_instance_avm(address).unwrap(); + let deployer = get_contract_instance_deployer_avm(address).unwrap(); + let initialization_hash = get_contract_instance_initialization_hash_avm(address).unwrap(); let expected_init = compute_initialization_hash(context.selector(), context.get_args_hash()); - assert(instance.initialization_hash == expected_init, "Initialization hash does not match"); + assert(initialization_hash == expected_init, "Initialization hash does not match"); assert( - (instance.deployer.is_zero()) | (instance.deployer == context.msg_sender()), + (deployer.is_zero()) | (deployer == context.msg_sender()), "Initializer address is not the contract deployer", ); } diff --git a/noir-projects/aztec-nr/aztec/src/oracle/get_contract_instance.nr b/noir-projects/aztec-nr/aztec/src/oracle/get_contract_instance.nr index 7e8416c50c9e..b5f593a32bb6 100644 --- a/noir-projects/aztec-nr/aztec/src/oracle/get_contract_instance.nr +++ b/noir-projects/aztec-nr/aztec/src/oracle/get_contract_instance.nr @@ -1,31 +1,22 @@ use dep::protocol_types::{ - address::AztecAddress, contract_instance::ContractInstance, constants::CONTRACT_INSTANCE_LENGTH, - utils::reader::Reader, + address::AztecAddress, contract_class_id::ContractClassId, contract_instance::ContractInstance, + constants::CONTRACT_INSTANCE_LENGTH, }; +// NOTE: this is for use in private only #[oracle(getContractInstance)] unconstrained fn get_contract_instance_oracle( _address: AztecAddress, ) -> [Field; CONTRACT_INSTANCE_LENGTH] {} -// Returns a ContractInstance plus a boolean indicating whether the instance was found. -#[oracle(avmOpcodeGetContractInstance)] -unconstrained fn get_contract_instance_oracle_avm( - _address: AztecAddress, -) -> [Field; CONTRACT_INSTANCE_LENGTH + 1] {} - +// NOTE: this is for use in private only unconstrained fn get_contract_instance_internal( address: AztecAddress, ) -> [Field; CONTRACT_INSTANCE_LENGTH] { get_contract_instance_oracle(address) } -pub unconstrained fn get_contract_instance_internal_avm( - address: AztecAddress, -) -> [Field; CONTRACT_INSTANCE_LENGTH + 1] { - get_contract_instance_oracle_avm(address) -} - +// NOTE: this is for use in private only pub fn get_contract_instance(address: AztecAddress) -> ContractInstance { let instance = unsafe { ContractInstance::deserialize(get_contract_instance_internal(address)) }; @@ -36,12 +27,58 @@ pub fn get_contract_instance(address: AztecAddress) -> ContractInstance { instance } -pub fn get_contract_instance_avm(address: AztecAddress) -> Option { - let mut reader = Reader::new(get_contract_instance_internal_avm(address)); - let found = reader.read(); - if found == 0 { +// These oracles each return a ContractInstance member +// plus a boolean indicating whether the instance was found. +#[oracle(avmOpcodeGetContractInstanceDeployer)] +unconstrained fn get_contract_instance_deployer_oracle_avm( + _address: AztecAddress, +) -> (Field, bool) {} +#[oracle(avmOpcodeGetContractInstanceClassId)] +unconstrained fn get_contract_instance_class_id_oracle_avm( + _address: AztecAddress, +) -> (Field, bool) {} +#[oracle(avmOpcodeGetContractInstanceInitializationHash)] +unconstrained fn get_contract_instance_initialization_hash_oracle_avm( + _address: AztecAddress, +) -> (Field, bool) {} + +pub unconstrained fn get_contract_instance_deployer_internal_avm( + address: AztecAddress, +) -> (Field, bool) { + get_contract_instance_deployer_oracle_avm(address) +} +pub unconstrained fn get_contract_instance_class_id_internal_avm( + address: AztecAddress, +) -> (Field, bool) { + get_contract_instance_class_id_oracle_avm(address) +} +pub unconstrained fn get_contract_instance_initialization_hash_internal_avm( + address: AztecAddress, +) -> (Field, bool) { + get_contract_instance_initialization_hash_oracle_avm(address) +} + +pub fn get_contract_instance_deployer_avm(address: AztecAddress) -> Option { + let (member, exists) = get_contract_instance_deployer_internal_avm(address); + if exists { + Option::some(AztecAddress::from_field(member)) + } else { + Option::none() + } +} +pub fn get_contract_instance_class_id_avm(address: AztecAddress) -> Option { + let (member, exists) = get_contract_instance_class_id_internal_avm(address); + if exists { + Option::some(ContractClassId::from_field(member)) + } else { Option::none() + } +} +pub fn get_contract_instance_initialization_hash_avm(address: AztecAddress) -> Option { + let (member, exists) = get_contract_instance_initialization_hash_internal_avm(address); + if exists { + Option::some(member) } else { - Option::some(reader.read_struct(ContractInstance::deserialize)) + Option::none() } } diff --git a/noir-projects/noir-contracts/contracts/avm_test_contract/src/main.nr b/noir-projects/noir-contracts/contracts/avm_test_contract/src/main.nr index ff31737f6abc..7c1e5f1d42dc 100644 --- a/noir-projects/noir-contracts/contracts/avm_test_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/avm_test_contract/src/main.nr @@ -34,13 +34,13 @@ contract AvmTest { use dep::aztec::prelude::Map; use dep::aztec::state_vars::PublicMutable; use dep::aztec::protocol_types::{ - address::{AztecAddress, EthAddress}, point::Point, scalar::Scalar, + abis::function_selector::FunctionSelector, address::{AztecAddress, EthAddress}, + contract_class_id::ContractClassId, contract_instance::ContractInstance, point::Point, + scalar::Scalar, storage::map::derive_storage_slot_in_map, }; use dep::aztec::oracle::get_contract_instance::{ - get_contract_instance_avm, get_contract_instance_internal_avm, - }; - use dep::aztec::protocol_types::{ - abis::function_selector::FunctionSelector, storage::map::derive_storage_slot_in_map, + get_contract_instance_deployer_avm, get_contract_instance_class_id_avm, + get_contract_instance_initialization_hash_avm, }; use dep::compressed_string::CompressedString; use dep::aztec::macros::{storage::storage, functions::{public, private}}; @@ -289,33 +289,46 @@ contract AvmTest { * Contract instance ************************************************************************/ #[public] - fn test_get_contract_instance_raw() { - let fields = get_contract_instance_internal_avm(context.this_address()); - // The values here should match those in `avm_simulator.test.ts>Contract>GETCONTRACTINSTANCE deserializes correctly` - assert(fields.len() == CONTRACT_INSTANCE_LENGTH + 1); - assert(fields[0] == 0x1); - assert(fields[1] == 0x123); - assert(fields[2] == 0x456); - assert(fields[3] == 0x789); - assert(fields[4] == 0x101112); - assert(fields[5] == 0x131415); - assert(fields[6] == 0x161718); - assert(fields[7] == 0x00); - assert(fields[8] == 0x192021); - assert(fields[9] == 0x222324); - assert(fields[10] == 0x00); - assert(fields[11] == 0x252627); - assert(fields[12] == 0x282930); - assert(fields[13] == 0x00); - assert(fields[14] == 0x313233); - assert(fields[15] == 0x343536); - assert(fields[16] == 0x00); - } - - #[public] - fn test_get_contract_instance() { - let ci = get_contract_instance_avm(context.this_address()); - assert(ci.is_some(), "Contract instance not found!"); + fn test_get_contract_instance(address: AztecAddress) { + let deployer = get_contract_instance_deployer_avm(address); + let class_id = get_contract_instance_class_id_avm(address); + let initialization_hash = get_contract_instance_initialization_hash_avm(address); + + assert(deployer.is_some(), "Contract instance not found when getting DEPLOYER!"); + assert(class_id.is_some(), "Contract instance not found when getting CLASS_ID!"); + assert( + initialization_hash.is_some(), + "Contract instance not found when getting INIT_HASH!", + ); + + // The values here should match those in `avm_simulator.test.ts` + assert(deployer.unwrap().eq(AztecAddress::from_field(0x456))); + assert(class_id.unwrap().eq(ContractClassId::from_field(0x789))); + assert(initialization_hash.unwrap() == 0x101112); + } + + #[public] + fn test_get_contract_instance_matches( + address: AztecAddress, + expected_deployer: AztecAddress, + expected_class_id: ContractClassId, + expected_initialization_hash: Field, + ) { + let deployer = get_contract_instance_deployer_avm(address); + let class_id = get_contract_instance_class_id_avm(address); + let initialization_hash = get_contract_instance_initialization_hash_avm(address); + + assert(deployer.is_some(), "Contract instance not found when getting DEPLOYER!"); + assert(class_id.is_some(), "Contract instance not found when getting CLASS_ID!"); + assert( + initialization_hash.is_some(), + "Contract instance not found when getting INIT_HASH!", + ); + + // The values here should match those in `avm_simulator.test.ts` + assert(deployer.unwrap().eq(expected_deployer)); + assert(class_id.unwrap().eq(expected_class_id)); + assert(initialization_hash.unwrap().eq(expected_initialization_hash)); } /************************************************************************ @@ -555,7 +568,7 @@ contract AvmTest { dep::aztec::oracle::debug_log::debug_log("pedersen_hash_with_index"); let _ = pedersen_hash_with_index(args_field); dep::aztec::oracle::debug_log::debug_log("test_get_contract_instance"); - test_get_contract_instance(); + test_get_contract_instance(context.this_address()); dep::aztec::oracle::debug_log::debug_log("get_address"); let _ = get_address(); dep::aztec::oracle::debug_log::debug_log("get_sender"); diff --git a/yarn-project/bb-prover/src/avm_proving.test.ts b/yarn-project/bb-prover/src/avm_proving.test.ts index 50af482b3642..b3705e3a8f18 100644 --- a/yarn-project/bb-prover/src/avm_proving.test.ts +++ b/yarn-project/bb-prover/src/avm_proving.test.ts @@ -66,6 +66,7 @@ const proveAndVerifyAvmTestContract = async ( const environment = initExecutionEnvironment({ functionSelector, calldata, globals }); const worldStateDB = mock(); + // The values here should match those in `avm_simulator.test.ts` const contractInstance = new SerializableContractInstance({ version: 1, salt: new Fr(0x123), diff --git a/yarn-project/end-to-end/src/e2e_avm_simulator.test.ts b/yarn-project/end-to-end/src/e2e_avm_simulator.test.ts index bff43d408406..c594663f8dec 100644 --- a/yarn-project/end-to-end/src/e2e_avm_simulator.test.ts +++ b/yarn-project/end-to-end/src/e2e_avm_simulator.test.ts @@ -95,7 +95,15 @@ describe('e2e_avm_simulator', () => { describe('Contract instance', () => { it('Works', async () => { - const tx = await avmContract.methods.test_get_contract_instance().send().wait(); + const tx = await avmContract.methods + .test_get_contract_instance_matches( + avmContract.address, + avmContract.instance.deployer, + avmContract.instance.contractClassId, + avmContract.instance.initializationHash, + ) + .send() + .wait(); expect(tx.status).toEqual(TxStatus.SUCCESS); }); }); diff --git a/yarn-project/simulator/src/avm/avm_simulator.test.ts b/yarn-project/simulator/src/avm/avm_simulator.test.ts index a4c39654a5a4..ad9d1837c176 100644 --- a/yarn-project/simulator/src/avm/avm_simulator.test.ts +++ b/yarn-project/simulator/src/avm/avm_simulator.test.ts @@ -1,4 +1,4 @@ -import { GasFees, PublicKeys } from '@aztec/circuits.js'; +import { GasFees, PublicKeys, SerializableContractInstance } from '@aztec/circuits.js'; import { Grumpkin } from '@aztec/circuits.js/barretenberg'; import { computeVarArgsHash } from '@aztec/circuits.js/hash'; import { FunctionSelector } from '@aztec/foundation/abi'; @@ -814,10 +814,10 @@ describe('AVM simulator: transpiled Noir contracts', () => { describe('Contract Instance Retrieval', () => { it(`Can getContractInstance`, async () => { - const context = createContext(); + const calldata = [address]; + const context = createContext(calldata); // Contract instance must match noir - const contractInstance = { - address: AztecAddress.random(), + const contractInstance = new SerializableContractInstance({ version: 1 as const, salt: new Fr(0x123), deployer: AztecAddress.fromBigInt(0x456n), @@ -829,16 +829,16 @@ describe('AVM simulator: transpiled Noir contracts', () => { new Point(new Fr(0x252627), new Fr(0x282930), false), new Point(new Fr(0x313233), new Fr(0x343536), false), ), - }; - mockGetContractInstance(worldStateDB, contractInstance); + }); + mockGetContractInstance(worldStateDB, contractInstance.withAddress(address)); - const bytecode = getAvmTestContractBytecode('test_get_contract_instance_raw'); + const bytecode = getAvmTestContractBytecode('test_get_contract_instance'); const results = await new AvmSimulator(context).executeBytecode(bytecode); expect(results.reverted).toBe(false); - expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(1); - expect(trace.traceGetContractInstance).toHaveBeenCalledWith({ exists: true, ...contractInstance }); + expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(3); // called for each enum value + expect(trace.traceGetContractInstance).toHaveBeenCalledWith(address, /*exists=*/ true, contractInstance); }); }); diff --git a/yarn-project/simulator/src/avm/journal/journal.test.ts b/yarn-project/simulator/src/avm/journal/journal.test.ts index e3feaf9ee536..993bc1585df2 100644 --- a/yarn-project/simulator/src/avm/journal/journal.test.ts +++ b/yarn-project/simulator/src/avm/journal/journal.test.ts @@ -1,4 +1,3 @@ -import { randomContractInstanceWithAddress } from '@aztec/circuit-types'; import { SerializableContractInstance } from '@aztec/circuits.js'; import { Fr } from '@aztec/foundation/fields'; @@ -147,11 +146,11 @@ describe('journal', () => { describe('Getting contract instances', () => { it('Should get contract instance', async () => { - const contractInstance = randomContractInstanceWithAddress(/*(base instance) opts=*/ {}, /*address=*/ address); - mockGetContractInstance(worldStateDB, contractInstance); + const contractInstance = SerializableContractInstance.default(); + mockGetContractInstance(worldStateDB, contractInstance.withAddress(address)); await persistableState.getContractInstance(address); expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(1); - expect(trace.traceGetContractInstance).toHaveBeenCalledWith({ exists: true, ...contractInstance }); + expect(trace.traceGetContractInstance).toHaveBeenCalledWith(address, /*exists=*/ true, contractInstance); }); it('Can get undefined contract instance', async () => { const defaultContractInstance = SerializableContractInstance.default().withAddress(address); @@ -159,6 +158,7 @@ describe('journal', () => { expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(1); expect(trace.traceGetContractInstance).toHaveBeenCalledWith({ exists: false, ...defaultContractInstance }); + expect(trace.traceGetContractInstance).toHaveBeenCalledWith(address, /*exists=*/ false, defaultContractInstance); }); }); diff --git a/yarn-project/simulator/src/avm/journal/journal.ts b/yarn-project/simulator/src/avm/journal/journal.ts index 3f6f115f5a0d..ad5045c88b96 100644 --- a/yarn-project/simulator/src/avm/journal/journal.ts +++ b/yarn-project/simulator/src/avm/journal/journal.ts @@ -4,7 +4,6 @@ import { createDebugLogger } from '@aztec/foundation/log'; import { getPublicFunctionDebugName } from '../../common/debug_fn_name.js'; 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'; import { type AvmContractCallResult } from '../avm_contract_call_result.js'; import { type AvmExecutionEnvironment } from '../avm_execution_environment.js'; @@ -212,20 +211,22 @@ export class AvmPersistableStateManager { * @param contractAddress - address of the contract instance to retrieve. * @returns the contract instance with an "exists" flag */ - public async getContractInstance(contractAddress: Fr): Promise { - let exists = true; + public async getContractInstance(contractAddress: Fr): Promise<[boolean, SerializableContractInstance]> { const aztecAddress = AztecAddress.fromField(contractAddress); - let instance = await this.worldStateDB.getContractInstance(aztecAddress); - if (instance === undefined) { - instance = SerializableContractInstance.default().withAddress(aztecAddress); - exists = false; + const instanceWithAddress = await this.worldStateDB.getContractInstance(aztecAddress); + + let exists = false; + let instance = SerializableContractInstance.default(); + if (instanceWithAddress !== undefined) { + exists = true; + instance = new SerializableContractInstance(instanceWithAddress); } + this.log.debug( `Get Contract instance (address=${contractAddress}): exists=${exists}, instance=${JSON.stringify(instance)}`, ); - const tracedInstance = { ...instance, exists }; - this.trace.traceGetContractInstance(tracedInstance); - return Promise.resolve(tracedInstance); + this.trace.traceGetContractInstance(contractAddress, exists, instance); + return Promise.resolve([exists, instance]); } /** diff --git a/yarn-project/simulator/src/avm/opcodes/contract.test.ts b/yarn-project/simulator/src/avm/opcodes/contract.test.ts index 1ba6ff2c2c15..37deecfb0805 100644 --- a/yarn-project/simulator/src/avm/opcodes/contract.test.ts +++ b/yarn-project/simulator/src/avm/opcodes/contract.test.ts @@ -1,19 +1,22 @@ -import { randomContractInstanceWithAddress } from '@aztec/circuit-types'; -import { AztecAddress, PublicKeys, SerializableContractInstance } from '@aztec/circuits.js'; +import { Fr, SerializableContractInstance } from '@aztec/circuits.js'; import { mock } from 'jest-mock-extended'; import { type WorldStateDB } from '../../public/public_db_sources.js'; import { type PublicSideEffectTraceInterface } from '../../public/side_effect_trace_interface.js'; import { type AvmContext } from '../avm_context.js'; -import { Field } from '../avm_memory_types.js'; +import { Field, TypeTag, Uint1 } from '../avm_memory_types.js'; import { initContext, initPersistableStateManager } from '../fixtures/index.js'; import { type AvmPersistableStateManager } from '../journal/journal.js'; import { mockGetContractInstance } from '../test_utils.js'; -import { GetContractInstance } from './contract.js'; +import { ContractInstanceMember, GetContractInstance } from './contract.js'; describe('Contract opcodes', () => { - const address = AztecAddress.random(); + const address = Fr.random(); + const contractInstance = SerializableContractInstance.random(); + const deployer = contractInstance.deployer; + const contractClassId = contractInstance.contractClassId; + const initializationHash = contractInstance.initializationHash; let worldStateDB: WorldStateDB; let trace: PublicSideEffectTraceInterface; @@ -32,61 +35,108 @@ describe('Contract opcodes', () => { const buf = Buffer.from([ GetContractInstance.opcode, // opcode 0x01, // indirect - ...Buffer.from('12345678', 'hex'), // addressOffset - ...Buffer.from('a2345678', 'hex'), // dstOffset + 0x02, // memberEnum (immediate) + ...Buffer.from('1234', 'hex'), // addressOffset + ...Buffer.from('a234', 'hex'), // dstOffset + ...Buffer.from('b234', 'hex'), // existsOffset ]); const inst = new GetContractInstance( /*indirect=*/ 0x01, - /*addressOffset=*/ 0x12345678, - /*dstOffset=*/ 0xa2345678, + /*memberEnum=*/ 0x02, + /*addressOffset=*/ 0x1234, + /*dstOffset=*/ 0xa234, + /*existsOffset=*/ 0xb234, ); expect(GetContractInstance.deserialize(buf)).toEqual(inst); expect(inst.serialize()).toEqual(buf); }); - it('should copy contract instance to memory if found', async () => { - const contractInstance = randomContractInstanceWithAddress(/*(base instance) opts=*/ {}, /*address=*/ address); - mockGetContractInstance(worldStateDB, contractInstance); - - context.machineState.memory.set(0, new Field(address.toField())); - await new GetContractInstance(/*indirect=*/ 0, /*addressOffset=*/ 0, /*dstOffset=*/ 1).execute(context); - - const actual = context.machineState.memory.getSlice(1, 17); - - expect(actual).toEqual([ - new Field(1), // found - new Field(contractInstance.salt), - new Field(contractInstance.deployer), - new Field(contractInstance.contractClassId), - new Field(contractInstance.initializationHash), - ...contractInstance.publicKeys.toFields().map(f => new Field(f)), - ]); - - expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(1); - expect(trace.traceGetContractInstance).toHaveBeenCalledWith({ exists: true, ...contractInstance }); + describe.each([ + [ContractInstanceMember.DEPLOYER, deployer.toField()], + [ContractInstanceMember.CLASS_ID, contractClassId.toField()], + [ContractInstanceMember.INIT_HASH, initializationHash.toField()], + ])('GETCONTRACTINSTANCE member instruction ', (memberEnum: ContractInstanceMember, value: Fr) => { + it(`Should read '${ContractInstanceMember[memberEnum]}' correctly`, async () => { + mockGetContractInstance(worldStateDB, contractInstance.withAddress(address)); + + context.machineState.memory.set(0, new Field(address)); + await new GetContractInstance( + /*indirect=*/ 0, + memberEnum, + /*addressOffset=*/ 0, + /*dstOffset=*/ 1, + /*existsOffset=*/ 2, + ).execute(context); + + // value should be right + expect(context.machineState.memory.getTag(1)).toBe(TypeTag.FIELD); + const actual = context.machineState.memory.get(1); + expect(actual).toEqual(new Field(value)); + + // exists should be true + expect(context.machineState.memory.getTag(2)).toBe(TypeTag.UINT1); + const exists = context.machineState.memory.get(2); + expect(exists).toEqual(new Uint1(1)); + + expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(1); + expect(trace.traceGetContractInstance).toHaveBeenCalledWith(address, /*exists=*/ true, contractInstance); + }); }); - it('should return zeroes if not found', async () => { - const defaultContractInstance = SerializableContractInstance.default().withAddress(address); - context.machineState.memory.set(0, new Field(address.toField())); - - await new GetContractInstance(/*indirect=*/ 0, /*addressOffset=*/ 0, /*dstOffset=*/ 1).execute(context); - - const actual = context.machineState.memory.getSlice(1, 17); - expect(actual).toEqual([ - new Field(0), // found - new Field(0), - new Field(0), - new Field(0), - new Field(0), - ...PublicKeys.default() - .toFields() - .map(f => new Field(f)), - ]); + describe.each([ + [ContractInstanceMember.DEPLOYER], + [ContractInstanceMember.CLASS_ID], + [ContractInstanceMember.INIT_HASH], + ])( + 'GETCONTRACTINSTANCE member instruction works when contract does not exist', + (memberEnum: ContractInstanceMember) => { + it(`'${ContractInstanceMember[memberEnum]}' should be 0 when contract does not exist `, async () => { + const defaultContractInstance = SerializableContractInstance.default(); + + context.machineState.memory.set(0, new Field(address)); + await new GetContractInstance( + /*indirect=*/ 0, + memberEnum, + /*addressOffset=*/ 0, + /*dstOffset=*/ 1, + /*existsOffset=*/ 2, + ).execute(context); + + // value should be 0 + expect(context.machineState.memory.getTag(1)).toBe(TypeTag.FIELD); + const actual = context.machineState.memory.get(1); + expect(actual).toEqual(new Field(0)); + + // exists should be false + expect(context.machineState.memory.getTag(2)).toBe(TypeTag.UINT1); + const exists = context.machineState.memory.get(2); + expect(exists).toEqual(new Uint1(0)); + + expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(1); + expect(trace.traceGetContractInstance).toHaveBeenCalledWith( + address, + /*exists=*/ false, + defaultContractInstance, + ); + }); + }, + ); + + it(`GETCONTRACTINSTANCE reverts for bad enum operand`, async () => { + const invalidEnum = 255; + const instruction = new GetContractInstance( + /*indirect=*/ 0, + /*memberEnum=*/ invalidEnum, + /*addressOffset=*/ 0, + /*dstOffset=*/ 1, + /*existsOffset=*/ 2, + ); + await expect(instruction.execute(context)).rejects.toThrow( + `Invalid GETCONSTRACTINSTANCE member enum ${invalidEnum}`, + ); - expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(1); - expect(trace.traceGetContractInstance).toHaveBeenCalledWith({ exists: false, ...defaultContractInstance }); + expect(trace.traceGetContractInstance).toHaveBeenCalledTimes(0); }); }); }); diff --git a/yarn-project/simulator/src/avm/opcodes/contract.ts b/yarn-project/simulator/src/avm/opcodes/contract.ts index 4ee050e1c9e7..232b18d28daf 100644 --- a/yarn-project/simulator/src/avm/opcodes/contract.ts +++ b/yarn-project/simulator/src/avm/opcodes/contract.ts @@ -1,23 +1,36 @@ -import { Fr } from '@aztec/circuits.js'; - import type { AvmContext } from '../avm_context.js'; -import { Field, TypeTag } from '../avm_memory_types.js'; +import { Field, TypeTag, Uint1 } from '../avm_memory_types.js'; +import { InstructionExecutionError } from '../errors.js'; import { Opcode, OperandType } from '../serialization/instruction_serialization.js'; import { Addressing } from './addressing_mode.js'; import { Instruction } from './instruction.js'; +export enum ContractInstanceMember { + DEPLOYER, + CLASS_ID, + INIT_HASH, +} + export class GetContractInstance extends Instruction { static readonly type: string = 'GETCONTRACTINSTANCE'; static readonly opcode: Opcode = Opcode.GETCONTRACTINSTANCE; // Informs (de)serialization. See Instruction.deserialize. static readonly wireFormat: OperandType[] = [ - OperandType.UINT8, - OperandType.UINT8, - OperandType.UINT32, - OperandType.UINT32, + OperandType.UINT8, // opcode + OperandType.UINT8, // indirect bits + OperandType.UINT8, // member enum (immediate) + OperandType.UINT16, // addressOffset + OperandType.UINT16, // dstOffset + OperandType.UINT16, // existsOfsset ]; - constructor(private indirect: number, private addressOffset: number, private dstOffset: number) { + constructor( + private indirect: number, + private memberEnum: number, + private addressOffset: number, + private dstOffset: number, + private existsOffset: number, + ) { super(); } @@ -25,27 +38,35 @@ export class GetContractInstance extends Instruction { const memory = context.machineState.memory.track(this.type); context.machineState.consumeGas(this.gasCost()); - const operands = [this.addressOffset, this.dstOffset]; + if (!(this.memberEnum in ContractInstanceMember)) { + throw new InstructionExecutionError(`Invalid GETCONSTRACTINSTANCE member enum ${this.memberEnum}`); + } + + const operands = [this.addressOffset, this.dstOffset, this.existsOffset]; const addressing = Addressing.fromWire(this.indirect, operands.length); - const [addressOffset, dstOffset] = addressing.resolve(operands, memory); + const [addressOffset, dstOffset, existsOffset] = addressing.resolve(operands, memory); memory.checkTag(TypeTag.FIELD, addressOffset); const address = memory.get(addressOffset).toFr(); - const instance = await context.persistableState.getContractInstance(address); + const [exists, instance] = await context.persistableState.getContractInstance(address); - const data = [ - new Fr(instance.exists), - instance.salt, - instance.deployer.toField(), - instance.contractClassId, - instance.initializationHash, - // This this okay ? - ...instance.publicKeys.toFields(), - ].map(f => new Field(f)); + let memberValue = new Field(0); + switch (this.memberEnum as ContractInstanceMember) { + case ContractInstanceMember.DEPLOYER: + memberValue = new Field(instance.deployer.toField()); + break; + case ContractInstanceMember.CLASS_ID: + memberValue = new Field(instance.contractClassId.toField()); + break; + case ContractInstanceMember.INIT_HASH: + memberValue = new Field(instance.initializationHash); + break; + } - memory.setSlice(dstOffset, data); + memory.set(existsOffset, new Uint1(exists ? 1 : 0)); + memory.set(dstOffset, exists ? memberValue : new Field(0)); - memory.assert({ reads: 1, writes: 17, addressing }); + memory.assert({ reads: 1, writes: 2, addressing }); context.machineState.incrementPc(); } } diff --git a/yarn-project/simulator/src/avm/opcodes/environment_getters.test.ts b/yarn-project/simulator/src/avm/opcodes/environment_getters.test.ts index 6f30a8615775..8821e52cdf11 100644 --- a/yarn-project/simulator/src/avm/opcodes/environment_getters.test.ts +++ b/yarn-project/simulator/src/avm/opcodes/environment_getters.test.ts @@ -83,4 +83,10 @@ describe('Environment getters', () => { expect(actual).toEqual(value); }); }); + + it(`GETENVVAR reverts for bad enum operand`, async () => { + const invalidEnum = 255; + const instruction = new GetEnvVar(/*indirect=*/ 0, invalidEnum, /*dstOffset=*/ 0); + await expect(instruction.execute(context)).rejects.toThrowError(`Invalid GETENVVAR var enum ${invalidEnum}`); + }); }); diff --git a/yarn-project/simulator/src/avm/opcodes/environment_getters.ts b/yarn-project/simulator/src/avm/opcodes/environment_getters.ts index 5bfb947d7324..4f9205adff79 100644 --- a/yarn-project/simulator/src/avm/opcodes/environment_getters.ts +++ b/yarn-project/simulator/src/avm/opcodes/environment_getters.ts @@ -1,5 +1,6 @@ import type { AvmContext } from '../avm_context.js'; import { Field, Uint32, Uint64 } from '../avm_memory_types.js'; +import { InstructionExecutionError } from '../errors.js'; import { Opcode, OperandType } from '../serialization/instruction_serialization.js'; import { Addressing } from './addressing_mode.js'; import { Instruction } from './instruction.js'; @@ -63,7 +64,7 @@ export class GetEnvVar extends Instruction { OperandType.UINT16, // dstOffset ]; - constructor(private indirect: number, private varEnum: EnvironmentVariable, private dstOffset: number) { + constructor(private indirect: number, private varEnum: number, private dstOffset: number) { super(); } @@ -71,11 +72,15 @@ export class GetEnvVar extends Instruction { const memory = context.machineState.memory.track(this.type); context.machineState.consumeGas(this.gasCost()); + if (!(this.varEnum in EnvironmentVariable)) { + throw new InstructionExecutionError(`Invalid GETENVVAR var enum ${this.varEnum}`); + } + const operands = [this.dstOffset]; const addressing = Addressing.fromWire(this.indirect, operands.length); const [dstOffset] = addressing.resolve(operands, memory); - memory.set(dstOffset, getValue(this.varEnum, context)); + memory.set(dstOffset, getValue(this.varEnum as EnvironmentVariable, context)); memory.assert({ writes: 1, addressing }); context.machineState.incrementPc(); diff --git a/yarn-project/simulator/src/public/dual_side_effect_trace.ts b/yarn-project/simulator/src/public/dual_side_effect_trace.ts index f2c82196605b..22b3a2d273d9 100644 --- a/yarn-project/simulator/src/public/dual_side_effect_trace.ts +++ b/yarn-project/simulator/src/public/dual_side_effect_trace.ts @@ -1,7 +1,7 @@ import { type CombinedConstantData, - type ContractInstanceWithAddress, type Gas, + type SerializableContractInstance, type VMCircuitPublicInputs, } from '@aztec/circuits.js'; import { type Fr } from '@aztec/foundation/fields'; @@ -15,8 +15,6 @@ import { type PublicExecutionResult } from './execution.js'; import { type PublicSideEffectTrace } from './side_effect_trace.js'; import { type PublicSideEffectTraceInterface } from './side_effect_trace_interface.js'; -export type TracedContractInstance = { exists: boolean } & ContractInstanceWithAddress; - export class DualSideEffectTrace implements PublicSideEffectTraceInterface { constructor( public readonly innerCallTrace: PublicSideEffectTrace, @@ -78,9 +76,9 @@ export class DualSideEffectTrace implements PublicSideEffectTraceInterface { this.enqueuedCallTrace.traceUnencryptedLog(contractAddress, log); } - public traceGetContractInstance(instance: TracedContractInstance) { - this.innerCallTrace.traceGetContractInstance(instance); - this.enqueuedCallTrace.traceGetContractInstance(instance); + public traceGetContractInstance(contractAddress: Fr, exists: boolean, instance: SerializableContractInstance) { + this.innerCallTrace.traceGetContractInstance(contractAddress, exists, instance); + this.enqueuedCallTrace.traceGetContractInstance(contractAddress, exists, instance); } /** diff --git a/yarn-project/simulator/src/public/enqueued_call_side_effect_trace.test.ts b/yarn-project/simulator/src/public/enqueued_call_side_effect_trace.test.ts index 38201e377dca..1505fcd9de22 100644 --- a/yarn-project/simulator/src/public/enqueued_call_side_effect_trace.test.ts +++ b/yarn-project/simulator/src/public/enqueued_call_side_effect_trace.test.ts @@ -34,15 +34,9 @@ import { randomBytes, randomInt } from 'crypto'; import { AvmContractCallResult } from '../avm/avm_contract_call_result.js'; import { initExecutionEnvironment } from '../avm/fixtures/index.js'; -import { PublicEnqueuedCallSideEffectTrace, type TracedContractInstance } from './enqueued_call_side_effect_trace.js'; +import { PublicEnqueuedCallSideEffectTrace } from './enqueued_call_side_effect_trace.js'; import { SideEffectLimitReachedError } from './side_effect_errors.js'; -function randomTracedContractInstance(): TracedContractInstance { - const instance = SerializableContractInstance.random(); - const address = AztecAddress.random(); - return { exists: true, ...instance, address }; -} - describe('Enqueued-call Side Effect Trace', () => { const address = Fr.random(); const utxo = Fr.random(); @@ -52,7 +46,7 @@ describe('Enqueued-call Side Effect Trace', () => { const recipient = Fr.random(); const content = Fr.random(); const log = [Fr.random(), Fr.random(), Fr.random()]; - const contractInstance = SerializableContractInstance.default().withAddress(new Fr(42)); + const contractInstance = SerializableContractInstance.default(); const startGasLeft = Gas.fromFields([new Fr(randomInt(10000)), new Fr(randomInt(10000))]); const endGasLeft = Gas.fromFields([new Fr(randomInt(10000)), new Fr(randomInt(10000))]); @@ -234,18 +228,19 @@ describe('Enqueued-call Side Effect Trace', () => { }); it('Should trace get contract instance', () => { - const instance = randomTracedContractInstance(); + const instance = SerializableContractInstance.random(); const { version: _, ...instanceWithoutVersion } = instance; - trace.traceGetContractInstance(instance); + const exists = true; + trace.traceGetContractInstance(address, exists, instance); expect(trace.getCounter()).toBe(startCounterPlus1); //const circuitPublicInputs = toVMCircuitPublicInputs(trace); - // TODO(dbanks12): once this emits nullifier read, check here expect(trace.getAvmCircuitHints().contractInstances.items).toEqual([ { // hint omits "version" and has "exists" as an Fr + address, + exists: new Fr(exists), ...instanceWithoutVersion, - exists: new Fr(instance.exists), }, ]); }); @@ -348,11 +343,11 @@ describe('Enqueued-call Side Effect Trace', () => { for (let i = 0; i < MAX_NULLIFIER_READ_REQUESTS_PER_TX; i++) { trace.traceNullifierCheck(new Fr(i), new Fr(i), new Fr(i), true, true); } - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: true })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ true, contractInstance)).toThrow( SideEffectLimitReachedError, ); // NOTE: also cannot do a existent check once non-existent checks have filled up - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: false })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ false, contractInstance)).toThrow( SideEffectLimitReachedError, ); }); @@ -361,11 +356,11 @@ describe('Enqueued-call Side Effect Trace', () => { for (let i = 0; i < MAX_NULLIFIER_NON_EXISTENT_READ_REQUESTS_PER_TX; i++) { trace.traceNullifierCheck(new Fr(i), new Fr(i), new Fr(i), false, true); } - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: false })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ false, contractInstance)).toThrow( SideEffectLimitReachedError, ); // NOTE: also cannot do a existent check once non-existent checks have filled up - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: true })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ true, contractInstance)).toThrow( SideEffectLimitReachedError, ); }); @@ -417,10 +412,10 @@ describe('Enqueued-call Side Effect Trace', () => { expect(() => trace.traceUnencryptedLog(new Fr(42), [new Fr(42), new Fr(42)])).toThrow( SideEffectLimitReachedError, ); - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: false })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ false, contractInstance)).toThrow( SideEffectLimitReachedError, ); - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: true })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ true, contractInstance)).toThrow( SideEffectLimitReachedError, ); }); @@ -454,9 +449,9 @@ describe('Enqueued-call Side Effect Trace', () => { testCounter++; nestedTrace.traceUnencryptedLog(address, log); testCounter++; - nestedTrace.traceGetContractInstance({ ...contractInstance, exists: true }); + nestedTrace.traceGetContractInstance(address, /*exists=*/ true, contractInstance); testCounter++; - nestedTrace.traceGetContractInstance({ ...contractInstance, exists: false }); + nestedTrace.traceGetContractInstance(address, /*exists=*/ false, contractInstance); testCounter++; trace.traceNestedCall(nestedTrace, avmEnvironment, startGasLeft, endGasLeft, bytecode, callResults); diff --git a/yarn-project/simulator/src/public/enqueued_call_side_effect_trace.ts b/yarn-project/simulator/src/public/enqueued_call_side_effect_trace.ts index 0e7036a37ef4..fbb3458e315f 100644 --- a/yarn-project/simulator/src/public/enqueued_call_side_effect_trace.ts +++ b/yarn-project/simulator/src/public/enqueued_call_side_effect_trace.ts @@ -7,7 +7,6 @@ import { AztecAddress, CallContext, type CombinedConstantData, - type ContractInstanceWithAddress, ContractStorageRead, ContractStorageUpdateRequest, EthAddress, @@ -44,6 +43,7 @@ import { ScopedNoteHash, type ScopedNullifier, ScopedReadRequest, + type SerializableContractInstance, TreeLeafReadRequest, VMCircuitPublicInputs, } from '@aztec/circuits.js'; @@ -58,8 +58,6 @@ import { type AvmExecutionEnvironment } from '../avm/avm_execution_environment.j import { SideEffectLimitReachedError } from './side_effect_errors.js'; import { type PublicSideEffectTraceInterface } from './side_effect_trace_interface.js'; -export type TracedContractInstance = { exists: boolean } & ContractInstanceWithAddress; - /** * A struct containing just the side effects as regular arrays * as opposed to "Tuple" arrays used by circuit public inputs. @@ -307,14 +305,13 @@ export class PublicEnqueuedCallSideEffectTrace implements PublicSideEffectTraceI this.incrementSideEffectCounter(); } - public traceGetContractInstance(instance: TracedContractInstance) { + public traceGetContractInstance(contractAddress: Fr, exists: boolean, instance: SerializableContractInstance) { this.enforceLimitOnNullifierChecks('(contract address nullifier from GETCONTRACTINSTANCE)'); - // TODO(dbanks12): should emit a nullifier read request this.avmCircuitHints.contractInstances.items.push( new AvmContractInstanceHint( - instance.address, - new Fr(instance.exists ? 1 : 0), + contractAddress, + new Fr(exists), instance.salt, instance.deployer, instance.contractClassId, diff --git a/yarn-project/simulator/src/public/side_effect_trace.test.ts b/yarn-project/simulator/src/public/side_effect_trace.test.ts index 096f64d9f225..a3f683aba1a3 100644 --- a/yarn-project/simulator/src/public/side_effect_trace.test.ts +++ b/yarn-project/simulator/src/public/side_effect_trace.test.ts @@ -23,13 +23,7 @@ import { randomBytes, randomInt } from 'crypto'; import { AvmContractCallResult } from '../avm/avm_contract_call_result.js'; import { initExecutionEnvironment } from '../avm/fixtures/index.js'; import { SideEffectLimitReachedError } from './side_effect_errors.js'; -import { PublicSideEffectTrace, type TracedContractInstance } from './side_effect_trace.js'; - -function randomTracedContractInstance(): TracedContractInstance { - const instance = SerializableContractInstance.random(); - const address = AztecAddress.random(); - return { exists: true, ...instance, address }; -} +import { PublicSideEffectTrace } from './side_effect_trace.js'; describe('Side Effect Trace', () => { const address = Fr.random(); @@ -40,7 +34,7 @@ describe('Side Effect Trace', () => { const recipient = Fr.random(); const content = Fr.random(); const log = [Fr.random(), Fr.random(), Fr.random()]; - const contractInstance = SerializableContractInstance.default().withAddress(new Fr(42)); + const contractInstance = SerializableContractInstance.default(); const startGasLeft = Gas.fromFields([new Fr(randomInt(10000)), new Fr(randomInt(10000))]); const endGasLeft = Gas.fromFields([new Fr(randomInt(10000)), new Fr(randomInt(10000))]); @@ -232,18 +226,19 @@ describe('Side Effect Trace', () => { }); it('Should trace get contract instance', () => { - const instance = randomTracedContractInstance(); + const instance = SerializableContractInstance.random(); const { version: _, ...instanceWithoutVersion } = instance; - trace.traceGetContractInstance(instance); + const exists = true; + trace.traceGetContractInstance(address, exists, instance); expect(trace.getCounter()).toBe(startCounterPlus1); const pxResult = toPxResult(trace); - // TODO(dbanks12): once this emits nullifier read, check here expect(pxResult.avmCircuitHints.contractInstances.items).toEqual([ { // hint omits "version" and has "exists" as an Fr + address, + exists: new Fr(exists), ...instanceWithoutVersion, - exists: new Fr(instance.exists), }, ]); }); @@ -346,11 +341,11 @@ describe('Side Effect Trace', () => { for (let i = 0; i < MAX_NULLIFIER_READ_REQUESTS_PER_TX; i++) { trace.traceNullifierCheck(new Fr(i), new Fr(i), new Fr(i), true, true); } - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: true })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ true, contractInstance)).toThrow( SideEffectLimitReachedError, ); // NOTE: also cannot do a existent check once non-existent checks have filled up - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: false })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ false, contractInstance)).toThrow( SideEffectLimitReachedError, ); }); @@ -359,11 +354,11 @@ describe('Side Effect Trace', () => { for (let i = 0; i < MAX_NULLIFIER_NON_EXISTENT_READ_REQUESTS_PER_TX; i++) { trace.traceNullifierCheck(new Fr(i), new Fr(i), new Fr(i), false, true); } - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: false })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ false, contractInstance)).toThrow( SideEffectLimitReachedError, ); // NOTE: also cannot do a existent check once non-existent checks have filled up - expect(() => trace.traceGetContractInstance({ ...contractInstance, exists: true })).toThrow( + expect(() => trace.traceGetContractInstance(address, /*exists=*/ true, contractInstance)).toThrow( SideEffectLimitReachedError, ); }); @@ -396,9 +391,9 @@ describe('Side Effect Trace', () => { testCounter++; nestedTrace.traceUnencryptedLog(address, log); testCounter++; - nestedTrace.traceGetContractInstance({ ...contractInstance, exists: true }); + nestedTrace.traceGetContractInstance(address, /*exists=*/ true, contractInstance); testCounter++; - nestedTrace.traceGetContractInstance({ ...contractInstance, exists: false }); + nestedTrace.traceGetContractInstance(address, /*exists=*/ false, contractInstance); testCounter++; trace.traceNestedCall(nestedTrace, avmEnvironment, startGasLeft, endGasLeft, bytecode, avmCallResults); diff --git a/yarn-project/simulator/src/public/side_effect_trace.ts b/yarn-project/simulator/src/public/side_effect_trace.ts index 1c24a9f9117d..85435adb85e2 100644 --- a/yarn-project/simulator/src/public/side_effect_trace.ts +++ b/yarn-project/simulator/src/public/side_effect_trace.ts @@ -27,6 +27,7 @@ import { Nullifier, type PublicInnerCallRequest, ReadRequest, + type SerializableContractInstance, TreeLeafReadRequest, } from '@aztec/circuits.js'; import { Fr } from '@aztec/foundation/fields'; @@ -216,14 +217,13 @@ export class PublicSideEffectTrace implements PublicSideEffectTraceInterface { this.incrementSideEffectCounter(); } - public traceGetContractInstance(instance: TracedContractInstance) { + public traceGetContractInstance(contractAddress: Fr, exists: boolean, instance: SerializableContractInstance) { this.enforceLimitOnNullifierChecks('(contract address nullifier from GETCONTRACTINSTANCE)'); - // TODO(dbanks12): should emit a nullifier read request this.avmCircuitHints.contractInstances.items.push( new AvmContractInstanceHint( - instance.address, - new Fr(instance.exists ? 1 : 0), + contractAddress, + new Fr(exists), instance.salt, instance.deployer, instance.contractClassId, 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 e3f7b7c2ae2a..c24235dd6868 100644 --- a/yarn-project/simulator/src/public/side_effect_trace_interface.ts +++ b/yarn-project/simulator/src/public/side_effect_trace_interface.ts @@ -1,9 +1,8 @@ -import { type Gas } from '@aztec/circuits.js'; +import { type Gas, type SerializableContractInstance } from '@aztec/circuits.js'; import { type Fr } from '@aztec/foundation/fields'; import { type AvmContractCallResult } from '../avm/avm_contract_call_result.js'; import { type AvmExecutionEnvironment } from '../avm/avm_execution_environment.js'; -import { type TracedContractInstance } from './side_effect_trace.js'; export interface PublicSideEffectTraceInterface { fork(): PublicSideEffectTraceInterface; @@ -18,8 +17,7 @@ export interface PublicSideEffectTraceInterface { traceL1ToL2MessageCheck(contractAddress: Fr, msgHash: Fr, msgLeafIndex: Fr, exists: boolean): void; traceNewL2ToL1Message(contractAddress: Fr, recipient: Fr, content: Fr): void; traceUnencryptedLog(contractAddress: Fr, log: Fr[]): void; - // TODO(dbanks12): odd that getContractInstance is a one-off in that it accepts an entire object instead of components - traceGetContractInstance(instance: TracedContractInstance): void; + traceGetContractInstance(contractAddress: Fr, exists: boolean, instance: SerializableContractInstance): void; traceNestedCall( /** The trace of the nested call. */ nestedCallTrace: PublicSideEffectTraceInterface, diff --git a/yarn-project/txe/src/txe_service/txe_service.ts b/yarn-project/txe/src/txe_service/txe_service.ts index 13cc7637562c..9b67026dfab1 100644 --- a/yarn-project/txe/src/txe_service/txe_service.ts +++ b/yarn-project/txe/src/txe_service/txe_service.ts @@ -616,18 +616,30 @@ export class TXEService { return toForeignCallResult([]); } - async avmOpcodeGetContractInstance(address: ForeignCallSingle) { + async avmOpcodeGetContractInstanceDeployer(address: ForeignCallSingle) { const instance = await this.typedOracle.getContractInstance(fromSingle(address)); return toForeignCallResult([ - toArray([ - // AVM requires an extra boolean indicating the instance was found - new Fr(1), - instance.salt, - instance.deployer, - instance.contractClassId, - instance.initializationHash, - ...instance.publicKeys.toFields(), - ]), + toSingle(instance.deployer), + // AVM requires an extra boolean indicating the instance was found + toSingle(new Fr(1)), + ]); + } + + async avmOpcodeGetContractInstanceClassId(address: ForeignCallSingle) { + const instance = await this.typedOracle.getContractInstance(fromSingle(address)); + return toForeignCallResult([ + toSingle(instance.contractClassId), + // AVM requires an extra boolean indicating the instance was found + toSingle(new Fr(1)), + ]); + } + + async avmOpcodeGetContractInstanceInitializationHash(address: ForeignCallSingle) { + const instance = await this.typedOracle.getContractInstance(fromSingle(address)); + return toForeignCallResult([ + toSingle(instance.initializationHash), + // AVM requires an extra boolean indicating the instance was found + toSingle(new Fr(1)), ]); }