Skip to content

Commit

Permalink
chore(avm-simulator): make sure we support Map storage (#5207)
Browse files Browse the repository at this point in the history
Brillig expects comparison operations to return `u1`. The AVM returns the type of the input operands. I'm fixing this by adding a cast in the transpiler but I think that longer term we should also return u1 (well, u8) in the AVM.
  • Loading branch information
fcarreiro authored Mar 14, 2024
1 parent 8bdb921 commit 08835f9
Show file tree
Hide file tree
Showing 4 changed files with 120 additions and 6 deletions.
2 changes: 1 addition & 1 deletion avm-transpiler/src/opcodes.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
/// All AVM opcodes
/// Keep updated with TS and yellow paper!
#[derive(PartialEq, Copy, Clone)]
#[derive(PartialEq, Copy, Clone, Debug)]
pub enum AvmOpcode {
// Compute
ADD,
Expand Down
17 changes: 17 additions & 0 deletions avm-transpiler/src/transpile.rs
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,10 @@ pub fn brillig_to_avm(brillig: &Brillig) -> Vec<u8> {
},
],
});
// Brillig currently expects comparison instructions to return an u1 (for us, an u8).
if avm_opcode == AvmOpcode::EQ {
avm_instrs.push(generate_cast_instruction(destination.to_usize() as u32, destination.to_usize() as u32, AvmTypeTag::UINT8));
}
}
BrilligOpcode::BinaryIntOp {
destination,
Expand Down Expand Up @@ -114,6 +118,10 @@ pub fn brillig_to_avm(brillig: &Brillig) -> Vec<u8> {
},
],
});
// Brillig currently expects comparison instructions to return an u1 (for us, an u8).
if avm_opcode == AvmOpcode::EQ || avm_opcode == AvmOpcode::LT || avm_opcode == AvmOpcode::LTE {
avm_instrs.push(generate_cast_instruction(destination.to_usize() as u32, destination.to_usize() as u32, AvmTypeTag::UINT8));
}
}
BrilligOpcode::CalldataCopy { destination_address, size, offset } => {
avm_instrs.push(AvmInstruction {
Expand Down Expand Up @@ -1005,6 +1013,15 @@ fn map_brillig_pcs_to_avm_pcs(initial_offset: usize, brillig: &Brillig) -> Vec<u
for i in 0..brillig.bytecode.len() - 1 {
let num_avm_instrs_for_this_brillig_instr = match &brillig.bytecode[i] {
BrilligOpcode::Const { bit_size: 254, .. } => 2,
// Brillig currently expects comparison instructions to return an u1 (for us, an u8).
BrilligOpcode::BinaryIntOp {
op: BinaryIntOp::Equals | BinaryIntOp::LessThan | BinaryIntOp::LessThanEquals,
..
} => 2,
BrilligOpcode::BinaryFieldOp {
op: BinaryFieldOp::Equals,
..
} => 2,
_ => 1,
};
// next Brillig pc will map to an AVM pc offset by the
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,9 +21,11 @@ contract AvmTest {
use crate::Note;

// Libs
use dep::aztec::prelude::Map;
use dep::aztec::state_vars::PublicMutable;
use dep::aztec::protocol_types::{address::{AztecAddress, EthAddress}, constants::L1_TO_L2_MESSAGE_LENGTH};
use dep::aztec::protocol_types::abis::function_selector::FunctionSelector;
use dep::aztec::protocol_types::{hash::pedersen_hash, traits::{ToField}};
use dep::compressed_string::CompressedString;

// avm lib
Expand All @@ -32,6 +34,7 @@ contract AvmTest {
struct Storage {
single: PublicMutable<Field>,
list: PublicMutable<Note>,
map: Map<AztecAddress, PublicMutable<u32>>,
}

#[aztec(public-vm)]
Expand Down Expand Up @@ -61,6 +64,26 @@ contract AvmTest {
note.serialize()
}

#[aztec(public-vm)]
fn setStorageMap(to: AztecAddress, amount: u32) -> pub Field {
storage.map.at(to).write(amount);
// returns storage slot for key
pedersen_hash([storage.map.storage_slot, to.to_field()], 0)
}

#[aztec(public-vm)]
fn addStorageMap(to: AztecAddress, amount: u32) -> pub Field {
let new_balance = storage.map.at(to).read().add(amount);
storage.map.at(to).write(new_balance);
// returns storage slot for key
pedersen_hash([storage.map.storage_slot, to.to_field()], 0)
}

#[aztec(public-vm)]
fn readStorageMap(address: AztecAddress) -> pub u32 {
storage.map.at(address).read()
}

#[aztec(public-vm)]
fn addArgsReturn(argA: Field, argB: Field) -> pub Field {
argA + argB
Expand Down
84 changes: 79 additions & 5 deletions yarn-project/simulator/src/avm/avm_simulator.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -499,7 +499,7 @@ describe('AVM simulator', () => {
});

describe('Storage accesses', () => {
it('Should set a single value in storage', async () => {
it('Should set value in storage (single)', async () => {
const slot = 1n;
const address = AztecAddress.fromField(new Fr(420));
const value = new Fr(88);
Expand All @@ -525,7 +525,7 @@ describe('AVM simulator', () => {
expect(slotTrace).toEqual([value]);
});

it('Should read a single value in storage', async () => {
it('Should read value in storage (single)', async () => {
const slot = 1n;
const value = new Fr(12345);
const address = AztecAddress.fromField(new Fr(420));
Expand All @@ -551,7 +551,7 @@ describe('AVM simulator', () => {
expect(slotTrace).toEqual([value]);
});

it('Should set and read a value from storage', async () => {
it('Should set and read a value from storage (single)', async () => {
const slot = 1n;
const value = new Fr(12345);
const address = AztecAddress.fromField(new Fr(420));
Expand All @@ -578,7 +578,7 @@ describe('AVM simulator', () => {
expect(slotWriteTrace).toEqual([value]);
});

it('Should set multiple values in storage', async () => {
it('Should set a value in storage (list)', async () => {
const slot = 2n;
const sender = AztecAddress.fromField(new Fr(1));
const address = AztecAddress.fromField(new Fr(420));
Expand All @@ -603,7 +603,7 @@ describe('AVM simulator', () => {
expect(storageTrace.get(slot + 1n)).toEqual([calldata[1]]);
});

it('Should read multiple values in storage', async () => {
it('Should read a value in storage (list)', async () => {
const slot = 2n;
const address = AztecAddress.fromField(new Fr(420));
const values = [new Fr(1), new Fr(2)];
Expand All @@ -630,6 +630,80 @@ describe('AVM simulator', () => {
expect(storageTrace.get(slot)).toEqual([values[0]]);
expect(storageTrace.get(slot + 1n)).toEqual([values[1]]);
});

it('Should set a value in storage (map)', async () => {
const address = AztecAddress.fromField(new Fr(420));
const value = new Fr(12345);
const calldata = [address.toField(), value];

const context = initContext({
env: initExecutionEnvironment({ address, calldata, storageAddress: address }),
});
const bytecode = getAvmTestContractBytecode('avm_setStorageMap');
const results = await new AvmSimulator(context).executeBytecode(bytecode);

expect(results.reverted).toBe(false);
// returns the storage slot for modified key
const slotNumber = results.output[0].toBigInt();

const worldState = context.persistableState.flush();
const storageSlot = worldState.currentStorageValue.get(address.toBigInt())!;
expect(storageSlot.get(slotNumber)).toEqual(value);

// Tracing
const storageTrace = worldState.storageWrites.get(address.toBigInt())!;
expect(storageTrace.get(slotNumber)).toEqual([value]);
});

it('Should read-add-set a value in storage (map)', async () => {
const address = AztecAddress.fromField(new Fr(420));
const value = new Fr(12345);
const calldata = [address.toField(), value];

const context = initContext({
env: initExecutionEnvironment({ address, calldata, storageAddress: address }),
});
const bytecode = getAvmTestContractBytecode('avm_addStorageMap');
const results = await new AvmSimulator(context).executeBytecode(bytecode);

expect(results.reverted).toBe(false);
// returns the storage slot for modified key
const slotNumber = results.output[0].toBigInt();

const worldState = context.persistableState.flush();
const storageSlot = worldState.currentStorageValue.get(address.toBigInt())!;
expect(storageSlot.get(slotNumber)).toEqual(value);

// Tracing
const storageReadTrace = worldState.storageReads.get(address.toBigInt())!;
expect(storageReadTrace.get(slotNumber)).toEqual([new Fr(0)]);
const storageWriteTrace = worldState.storageWrites.get(address.toBigInt())!;
expect(storageWriteTrace.get(slotNumber)).toEqual([value]);
});

it('Should read value in storage (map)', async () => {
const value = new Fr(12345);
const address = AztecAddress.fromField(new Fr(420));
const calldata = [address.toField()];

const context = initContext({
env: initExecutionEnvironment({ calldata, address, storageAddress: address }),
});
jest
.spyOn(context.persistableState.hostStorage.publicStateDb, 'storageRead')
.mockReturnValue(Promise.resolve(value));
const bytecode = getAvmTestContractBytecode('avm_readStorageMap');
const results = await new AvmSimulator(context).executeBytecode(bytecode);

// Get contract function artifact
expect(results.reverted).toBe(false);
expect(results.output).toEqual([value]);

// Tracing
const worldState = context.persistableState.flush();
const storageTrace = worldState.storageReads.get(address.toBigInt())!;
expect([...storageTrace.values()]).toEqual([[value]]);
});
});
});
});

0 comments on commit 08835f9

Please sign in to comment.