Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

feat(avm): storage #4673

Merged
merged 50 commits into from
Mar 6, 2024
Merged
Show file tree
Hide file tree
Changes from 46 commits
Commits
Show all changes
50 commits
Select commit Hold shift + click to select a range
d630b44
git subrepo pull --branch=master --force noir
sirasistant Feb 14, 2024
518ec41
wip: restricting bit sizes
sirasistant Feb 15, 2024
636c316
fix: more fixes for restricted bit sizes
sirasistant Feb 15, 2024
f92dcdd
more u128
sirasistant Feb 16, 2024
c7986a3
get them all compiling
Maddiaa0 Feb 21, 2024
ab832ff
hack build
Maddiaa0 Feb 21, 2024
5c5156f
fix: noir stdlib error message
Maddiaa0 Feb 21, 2024
aecdd3a
Merge branch 'master' into md/02-21-get_them_all_compiling
Maddiaa0 Feb 21, 2024
1002bb9
fix: update gas rebate contract
Maddiaa0 Feb 21, 2024
ec6ce1d
Merge branch 'master' into md/02-21-get_them_all_compiling
Maddiaa0 Feb 26, 2024
38181cd
fix: merge
Maddiaa0 Feb 26, 2024
9eeb153
exp: blacklist return field
Maddiaa0 Feb 26, 2024
ea0d0c6
fix: underflow error messages
Maddiaa0 Feb 26, 2024
9d4fe7e
fix
Maddiaa0 Feb 26, 2024
e9db646
fix: update lending simulator check
Maddiaa0 Feb 27, 2024
7af5de5
fmt
Maddiaa0 Feb 27, 2024
621f593
fix
Maddiaa0 Feb 27, 2024
f2fe7fc
fix
Maddiaa0 Feb 27, 2024
e92208f
Merge branch 'master' into md/02-21-get_them_all_compiling
Maddiaa0 Feb 27, 2024
1d26069
fix: error message
Maddiaa0 Feb 27, 2024
c4e0c5e
temp
Maddiaa0 Feb 14, 2024
543e093
fix: get e2e hashing working
Maddiaa0 Feb 14, 2024
800a774
sweep
Maddiaa0 Feb 14, 2024
1bd7952
sweep 2
Maddiaa0 Feb 14, 2024
2522293
feat: avm storage
Maddiaa0 Feb 16, 2024
94466de
fix: rebase
Maddiaa0 Feb 27, 2024
fdda301
Merge branch 'master' into md/02-16-feat_avm_storage
Maddiaa0 Feb 27, 2024
d16dd58
fix: merge
Maddiaa0 Feb 27, 2024
eded899
fixes: test and tag check alignment
Maddiaa0 Feb 27, 2024
025dc2a
align addressing
Maddiaa0 Feb 27, 2024
d633b8b
chore: working with bitsize hack
Maddiaa0 Feb 27, 2024
3a327dd
fix: dont meddle with ssa
Maddiaa0 Feb 27, 2024
a932153
Merge branch 'master' into md/02-16-feat_avm_storage
Maddiaa0 Feb 27, 2024
7525ee8
Merge branch 'master' into md/02-16-feat_avm_storage
Maddiaa0 Mar 4, 2024
4997ce4
exp
Maddiaa0 Mar 4, 2024
1ad0307
merge 2: electric boogaloo
Maddiaa0 Mar 4, 2024
d534d2d
fmt
Maddiaa0 Mar 4, 2024
85d7ce8
fix: allow memory mem tags to be less than or equal
Maddiaa0 Mar 4, 2024
1bde64b
chore: remove noir logs
Maddiaa0 Mar 4, 2024
ccd689f
chore: cleanup ts comments
Maddiaa0 Mar 4, 2024
4916061
Merge branch 'master' into md/02-16-feat_avm_storage
Maddiaa0 Mar 4, 2024
b90675d
Merge branch 'master' into md/02-16-feat_avm_storage
Maddiaa0 Mar 6, 2024
84670d6
tidy
Maddiaa0 Mar 6, 2024
332f333
sweep
Maddiaa0 Mar 6, 2024
957d0bd
tag name update
Maddiaa0 Mar 6, 2024
d0386d4
fix: lint, tag not used
Maddiaa0 Mar 6, 2024
51f8570
death and taxes
Maddiaa0 Mar 6, 2024
3499714
fmt
Maddiaa0 Mar 6, 2024
78de9b7
fix test
Maddiaa0 Mar 6, 2024
6cdd047
fix: test
Maddiaa0 Mar 6, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions avm-transpiler/src/instructions.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ use crate::opcodes::AvmOpcode;
pub const ALL_DIRECT: u8 = 0b00000000;
pub const ZEROTH_OPERAND_INDIRECT: u8 = 0b00000001;
pub const FIRST_OPERAND_INDIRECT: u8 = 0b00000010;
pub const SECOND_OPERAND_INDIRECT: u8 = 0b00000100;
pub const ZEROTH_FIRST_OPERANDS_INDIRECT: u8 = ZEROTH_OPERAND_INDIRECT | FIRST_OPERAND_INDIRECT;

/// A simple representation of an AVM instruction for the purpose
Expand Down
91 changes: 87 additions & 4 deletions avm-transpiler/src/transpile.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ use acvm::brillig_vm::brillig::{

use crate::instructions::{
AvmInstruction, AvmOperand, AvmTypeTag, ALL_DIRECT, FIRST_OPERAND_INDIRECT,
ZEROTH_OPERAND_INDIRECT,
SECOND_OPERAND_INDIRECT, ZEROTH_OPERAND_INDIRECT,
};
use crate::opcodes::AvmOpcode;
use crate::utils::{dbg_print_avm_program, dbg_print_brillig_program};
Expand Down Expand Up @@ -257,8 +257,10 @@ fn handle_foreign_call(
"avmOpcodePoseidon" => {
handle_single_field_hash_instruction(avm_instrs, function, destinations, inputs)
}
"storageWrite" => emit_storage_write(avm_instrs, destinations, inputs),
"storageRead" => emit_storage_read(avm_instrs, destinations, inputs),
fcarreiro marked this conversation as resolved.
Show resolved Hide resolved
// Getters.
_ if inputs.len() == 0 && destinations.len() == 1 => {
_ if inputs.is_empty() && destinations.len() == 1 => {
handle_getter_instruction(avm_instrs, function, destinations, inputs)
}
// Anything else.
Expand Down Expand Up @@ -361,7 +363,7 @@ fn handle_emit_note_hash_or_nullifier(
"EMITNOTEHASH"
};

if destinations.len() != 0 || inputs.len() != 1 {
if !destinations.is_empty() || inputs.len() != 1 {
panic!(
"Transpiler expects ForeignCall::{} to have 0 destinations and 1 input, got {} and {}",
function_name,
Expand Down Expand Up @@ -390,6 +392,87 @@ fn handle_emit_note_hash_or_nullifier(
});
}

/// Emit a storage write opcode
/// The current implementation writes an array of values into storage ( contiguous slots in memory )
fn emit_storage_write(
avm_instrs: &mut Vec<AvmInstruction>,
destinations: &Vec<ValueOrArray>,
inputs: &Vec<ValueOrArray>,
) {
assert!(inputs.len() == 2);
assert!(destinations.len() == 0);

let slot_offset_maybe = inputs[0];
let slot_offset = match slot_offset_maybe {
ValueOrArray::MemoryAddress(slot_offset) => slot_offset.0,
_ => panic!("ForeignCall address destination should be a single value"),
};

let src_offset_maybe = inputs[1];
let (src_offset, src_size) = match src_offset_maybe {
ValueOrArray::HeapArray(HeapArray { pointer, size }) => (pointer.0, size),
_ => panic!("Storage write address inputs should be an array of values"),
};

avm_instrs.push(AvmInstruction {
opcode: AvmOpcode::SSTORE,
indirect: Some(ZEROTH_OPERAND_INDIRECT),
operands: vec![
AvmOperand::U32 {
value: src_offset as u32,
},
AvmOperand::U32 {
value: src_size as u32,
},
AvmOperand::U32 {
value: slot_offset as u32,
},
],
..Default::default()
})
}

/// Emit a storage read opcode
/// The current implementation reads an array of values from storage ( contiguous slots in memory )
fn emit_storage_read(
avm_instrs: &mut Vec<AvmInstruction>,
destinations: &Vec<ValueOrArray>,
inputs: &Vec<ValueOrArray>,
) {
// For the foreign calls we want to handle, we do not want inputs, as they are getters
assert!(inputs.len() == 2); // output, len - but we dont use this len - its for the oracle
assert!(destinations.len() == 1);

let slot_offset_maybe = inputs[0];
let slot_offset = match slot_offset_maybe {
ValueOrArray::MemoryAddress(slot_offset) => slot_offset.0,
_ => panic!("ForeignCall address destination should be a single value"),
};

let dest_offset_maybe = destinations[0];
let (dest_offset, src_size) = match dest_offset_maybe {
ValueOrArray::HeapArray(HeapArray { pointer, size }) => (pointer.0, size),
_ => panic!("Storage write address inputs should be an array of values"),
};

avm_instrs.push(AvmInstruction {
opcode: AvmOpcode::SLOAD,
indirect: Some(SECOND_OPERAND_INDIRECT),
operands: vec![
AvmOperand::U32 {
value: slot_offset as u32,
},
AvmOperand::U32 {
value: src_size as u32,
},
AvmOperand::U32 {
value: dest_offset as u32,
},
],
..Default::default()
})
}

/// Handle an AVM NULLIFIEREXISTS instruction
/// (a nullifierExists brillig foreign call was encountered)
/// Adds the new instruction to the avm instructions list.
Expand Down Expand Up @@ -483,7 +566,7 @@ fn handle_send_l2_to_l1_msg(
destinations: &Vec<ValueOrArray>,
inputs: &Vec<ValueOrArray>,
) {
if destinations.len() != 0 || inputs.len() != 2 {
if !destinations.is_empty() || inputs.len() != 2 {
panic!(
"Transpiler expects ForeignCall::SENDL2TOL1MSG to have 0 destinations and 2 inputs, got {} and {}",
destinations.len(),
Expand Down
11 changes: 8 additions & 3 deletions noir-projects/aztec-nr/aztec/src/context.nr
Original file line number Diff line number Diff line change
Expand Up @@ -14,18 +14,23 @@ use avm::AVMContext;
struct Context {
private: Option<&mut PrivateContext>,
public: Option<&mut PublicContext>,
public_vm: Option<&mut AVMContext>,
}

impl Context {
pub fn private(context: &mut PrivateContext) -> Context {
Context { private: Option::some(context), public: Option::none() }
Context { private: Option::some(context), public: Option::none(), public_vm: Option::none() }
}

pub fn public(context: &mut PublicContext) -> Context {
Context { public: Option::some(context), private: Option::none() }
Context { public: Option::some(context), private: Option::none(), public_vm: Option::none() }
}

pub fn public_vm(context: &mut AVMContext) -> Context {
Context { public_vm: Option::some(context), public: Option::none(), private: Option::none() }
}

pub fn none() -> Context {
Context { public: Option::none(), private: Option::none() }
Context { public: Option::none(), private: Option::none(), public_vm: Option::none() }
}
}
4 changes: 2 additions & 2 deletions noir-projects/aztec-nr/aztec/src/oracle/storage.nr
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,8 @@ pub fn storage_read<N>(storage_slot: Field) -> [Field; N] {
}

#[oracle(storageWrite)]
fn storage_write_oracle<N>(_storage_slot: Field, _values: [Field; N]) -> [Field; N] {}
fn storage_write_oracle<N>(_storage_slot: Field, _values: [Field; N]) {}

unconstrained pub fn storage_write<N>(storage_slot: Field, fields: [Field; N]) {
let _hash = storage_write_oracle(storage_slot, fields);
storage_write_oracle(storage_slot, fields);
}
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
contract AvmTest {
// Libs
use dep::aztec::state_vars::PublicMutable;
use dep::aztec::protocol_types::{address::{AztecAddress, EthAddress}, constants::L1_TO_L2_MESSAGE_LENGTH};
use dep::compressed_string::CompressedString;

Expand All @@ -9,7 +10,21 @@ contract AvmTest {
#[aztec(private)]
fn constructor() {}

// Public-vm macro will prefix avm to the function name for transpilation
struct Storage {
owner: PublicMutable<AztecAddress>
}

#[aztec(public-vm)]
fn setAdmin() {
storage.owner.write(context.sender());
}

#[aztec(public-vm)]
fn setAndRead() -> pub AztecAddress {
storage.owner.write(context.sender());
storage.owner.read()
}

#[aztec(public-vm)]
fn addArgsReturn(argA: Field, argB: Field) -> pub Field {
argA + argB
Expand Down
8 changes: 7 additions & 1 deletion noir/noir-repo/aztec_macros/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -727,8 +727,14 @@ fn transform_function(
/// Transform a function to work with AVM bytecode
fn transform_vm_function(
func: &mut NoirFunction,
_storage_defined: bool,
storage_defined: bool,
) -> Result<(), AztecMacroError> {
// Create access to storage
if storage_defined {
let storage = abstract_storage("public_vm", true);
func.def.body.0.insert(0, storage);
}

// Push Avm context creation to the beginning of the function
let create_context = create_avm_context()?;
func.def.body.0.insert(0, create_context);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ use num_bigint::BigUint;
/// The Brillig VM does not apply a limit to the memory address space,
/// As a convention, we take use 64 bits. This means that we assume that
/// memory has 2^64 memory slots.
pub(crate) const BRILLIG_MEMORY_ADDRESSING_BIT_SIZE: u32 = 32;
pub(crate) const BRILLIG_MEMORY_ADDRESSING_BIT_SIZE: u32 = 64;

// Registers reserved in runtime for special purposes.
pub(crate) enum ReservedRegisters {
Expand Down Expand Up @@ -562,6 +562,7 @@ impl BrilligContext {
bit_size: u32,
) {
self.debug_show.const_instruction(result, constant);

self.push_opcode(BrilligOpcode::Const { destination: result, value: constant, bit_size });
}

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
[package]
name = "should_fail_with_mismatch"
type = "bin"
authors = [""]
[dependencies]
8 changes: 5 additions & 3 deletions yarn-project/simulator/src/acvm/oracle/oracle.ts
Original file line number Diff line number Diff line change
Expand Up @@ -253,9 +253,11 @@ export class Oracle {
return values.map(toACVMField);
}

async storageWrite([startStorageSlot]: ACVMField[], values: ACVMField[]): Promise<ACVMField[]> {
const newValues = await this.typedOracle.storageWrite(fromACVMField(startStorageSlot), values.map(fromACVMField));
return newValues.map(toACVMField);
storageWrite([startStorageSlot]: ACVMField[], values: ACVMField[]): ACVMField {
this.typedOracle.storageWrite(fromACVMField(startStorageSlot), values.map(fromACVMField));

// We return 0 here as we MUST return something, but the value is not used.
return '0';
}

emitEncryptedLog(
Expand Down
2 changes: 1 addition & 1 deletion yarn-project/simulator/src/acvm/oracle/typed_oracle.ts
Original file line number Diff line number Diff line change
Expand Up @@ -173,7 +173,7 @@ export abstract class TypedOracle {
throw new Error('Not available.');
}

storageWrite(_startStorageSlot: Fr, _values: Fr[]): Promise<Fr[]> {
storageWrite(_startStorageSlot: Fr, _values: Fr[]) {
throw new Error('Not available.');
}

Expand Down
6 changes: 6 additions & 0 deletions yarn-project/simulator/src/avm/avm_memory_types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -255,6 +255,12 @@ export class TaggedMemory {
}
}

public checkIsValidMemoryOffsetTag(offset: number) {
if (this.getTag(offset) > TypeTag.UINT64) {
throw TagCheckError.forOffset(offset, TypeTag[this.getTag(offset)], "UINT64");
}
}

public static checkIsIntegralTag(tag: TypeTag) {
if (![TypeTag.UINT8, TypeTag.UINT16, TypeTag.UINT32, TypeTag.UINT64, TypeTag.UINT128].includes(tag)) {
throw TagCheckError.forTag(TypeTag[tag], 'integral');
Expand Down
58 changes: 58 additions & 0 deletions yarn-project/simulator/src/avm/avm_simulator.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,64 @@ describe('AVM simulator', () => {
});
});

describe('Storage accesses', () => {
it('Should set a single value in storage', async () => {
// We are setting the owner
const slot = 1n;
const sender = AztecAddress.fromField(new Fr(1));
const address = AztecAddress.fromField(new Fr(420));

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

// Get contract function artifact
expect(results.reverted).toBe(false);

// Contract 420 - Storage slot 1 should contain the value 1
const worldState = context.persistableState.flush();

const storageSlot = worldState.currentStorageValue.get(address.toBigInt())!;
const adminSlotValue = storageSlot.get(slot);
expect(adminSlotValue).toEqual(sender.toField());
fcarreiro marked this conversation as resolved.
Show resolved Hide resolved

// Tracing
const storageTrace = worldState.storageWrites.get(address.toBigInt())!;
const slotTrace = storageTrace.get(slot);
expect(slotTrace).toEqual(sender.toField());
});

it('Should read a value from storage', async () => {
// We are setting the owner
const sender = AztecAddress.fromField(new Fr(1));
const address = AztecAddress.fromField(new Fr(420));

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

expect(results.reverted).toBe(false);

expect(results.output).toEqual([sender.toField()]);

const worldState = context.persistableState.flush();

// Test read trace
const storageReadTrace = worldState.storageReads.get(address.toBigInt())!;
const slotReadTrace = storageReadTrace.get(1n);
expect(slotReadTrace).toEqual(sender.toField());

// Test write trace
const storageWriteTrace = worldState.storageWrites.get(address.toBigInt())!;
const slotWriteTrace = storageWriteTrace.get(1n);
expect(slotWriteTrace).toEqual(sender.toField());
});
});

describe('Test env getters from noir contract', () => {
const testEnvGetter = async (valueName: string, value: any, functionName: string, globalVar: boolean = false) => {
// Execute
Expand Down
3 changes: 2 additions & 1 deletion yarn-project/simulator/src/avm/journal/trace.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ export class WorldStateAccessTrace {
}

public tracePublicStorageWrite(storageAddress: Fr, slot: Fr, value: Fr) {
// TODO(4805): check if some threshold is reached for max storage writes
// TODO: check if some threshold is reached for max storage writes
// (need access to parent length, or trace needs to be initialized with parent's contents)
//const traced: TracedPublicStorageWrite = {
// callPointer: Fr.ZERO,
Expand All @@ -57,6 +57,7 @@ export class WorldStateAccessTrace {
// endLifetime: Fr.ZERO,
//};
//this.publicStorageWrites.push(traced);

this.journalWrite(storageAddress, slot, value);
this.incrementAccessCounter();
}
Expand Down
5 changes: 3 additions & 2 deletions yarn-project/simulator/src/avm/opcodes/addressing_mode.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { strict as assert } from 'assert';

import { TaggedMemory, TypeTag } from '../avm_memory_types.js';
import { TaggedMemory } from '../avm_memory_types.js';

export enum AddressingMode {
DIRECT,
Expand Down Expand Up @@ -51,7 +51,8 @@ export class Addressing {
for (const [i, offset] of offsets.entries()) {
switch (this.modePerOperand[i]) {
case AddressingMode.INDIRECT:
mem.checkTag(TypeTag.UINT32, offset);
// NOTE(reviewer): less than equal is a deviation from the spec - i dont see why this shouldnt be possible!
mem.checkIsValidMemoryOffsetTag(offset);
resolved[i] = Number(mem.get(offset).toBigInt());
break;
case AddressingMode.DIRECT:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -70,7 +70,7 @@ describe('External Calls', () => {
const successOffset = 7;
const otherContextInstructionsBytecode = encodeToBytecode([
new CalldataCopy(/*indirect=*/ 0, /*csOffset=*/ 0, /*copySize=*/ argsSize, /*dstOffset=*/ 0),
new SStore(/*indirect=*/ 0, /*srcOffset=*/ 0, /*slotOffset=*/ 0),
new SStore(/*indirect=*/ 0, /*srcOffset=*/ 0, /*size=*/ 1, /*slotOffset=*/ 0),
new Return(/*indirect=*/ 0, /*retOffset=*/ 0, /*size=*/ 2),
]);

Expand Down Expand Up @@ -159,7 +159,7 @@ describe('External Calls', () => {

const otherContextInstructions: Instruction[] = [
new CalldataCopy(/*indirect=*/ 0, /*csOffset=*/ 0, /*copySize=*/ argsSize, /*dstOffset=*/ 0),
new SStore(/*indirect=*/ 0, /*srcOffset=*/ 1, /*slotOffset=*/ 0),
new SStore(/*indirect=*/ 0, /*srcOffset=*/ 1, /*size=*/ 1, /*slotOffset=*/ 0),
];

const otherContextInstructionsBytecode = encodeToBytecode(otherContextInstructions);
Expand Down
Loading
Loading