Skip to content

Commit

Permalink
feat: store shared mutable hash (#7169)
Browse files Browse the repository at this point in the history
This PR changes public write functions so that they not only store the
new delay and/or value, but also a hash of the combined serialization of
them. This lets us produce smaller proofs in private, since we only need
to prove inclusion of the hash and not each of the individual 4 values.
This will result in even larger savings once we do #5491.

The only annoying bit is that producing the unconstrained hint so that
we can contrain the hash and prove inclusion involves performing reads,
and hence computation of storage slots. But because we cannot pass
mutable references to unconstrained functions, we cannot have methods
for the `&mut PrivateContext` impl, resulting in a bit of a hack to
create a dummy state variable that I can call methods on. This is all
going to go away regardless once #5492 is done.

I also restored the tests deleted in #6985, updating the existing ones
and adding new cases.

---------

Co-authored-by: LHerskind <[email protected]>
  • Loading branch information
nventuro and LHerskind authored Jul 1, 2024
1 parent 1de3746 commit 868606e
Show file tree
Hide file tree
Showing 11 changed files with 448 additions and 381 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -171,3 +171,9 @@ impl<INITIAL_DELAY> Deserialize<1> for ScheduledDelayChange<INITIAL_DELAY> {
}
}
}

impl<INITIAL_DELAY> Eq for ScheduledDelayChange<INITIAL_DELAY> {
fn eq(self, other: Self) -> bool {
(self.pre == other.pre) & (self.post == other.post) & (self.block_of_change == other.block_of_change)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ fn assert_equal_after_conversion(original: ScheduledDelayChange<TEST_INITIAL_DEL
// TODO: improve syntax once https://github.com/noir-lang/noir/issues/4710 is implemented.
let converted: ScheduledDelayChange<TEST_INITIAL_DELAY> = ScheduledDelayChange::deserialize((original).serialize());

assert_eq(original, converted); // This also tests the Eq impl
assert_eq(original.pre, converted.pre);
assert_eq(original.post, converted.post);
assert_eq(original.block_of_change, converted.block_of_change);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -152,3 +152,9 @@ impl<T> Deserialize<3> for ScheduledValueChange<T> {
}
}
}

impl<T> Eq for ScheduledValueChange<T> {
fn eq(self, other: Self) -> bool where T: Eq {
(self.pre == other.pre) & (self.post == other.post) & (self.block_of_change == other.block_of_change)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ fn test_serde() {
let original = ScheduledValueChange::new(pre, post, block_of_change);
let converted = ScheduledValueChange::deserialize((original).serialize());

assert_eq(original, converted); // This also tests the Eq impl
assert_eq(original.pre, converted.pre);
assert_eq(original.post, converted.post);
assert_eq(original.block_of_change, converted.block_of_change);
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +1,15 @@
use dep::protocol_types::{hash::pedersen_hash, traits::FromField};
use dep::protocol_types::{
hash::{pedersen_hash, poseidon2_hash}, header::Header, address::AztecAddress,
traits::{FromField, ToField}
};

use crate::context::{PrivateContext, PublicContext};
use crate::state_vars::{
storage::Storage,
shared_mutable::{scheduled_value_change::ScheduledValueChange, scheduled_delay_change::ScheduledDelayChange}
};
use crate::oracle::storage::storage_read;
use dep::std::unsafe::zeroed;

mod test;

Expand All @@ -18,42 +23,112 @@ struct SharedMutable<T, INITIAL_DELAY, Context> {
// - a ScheduledValueChange<T>, which requires 1 + 2 * M storage slots, where M is the serialization length of T
// - a ScheduledDelayChange, which requires another storage slot
//
// TODO https://github.com/AztecProtocol/aztec-packages/issues/5736: change the storage allocation scheme so that we
// TODO https://github.com/AztecProtocol/aztec-packages/issues/5736: change the storage allocation scheme so that we
// can actually use it here
impl<T, INITIAL_DELAY, Context> Storage<T> for SharedMutable<T, INITIAL_DELAY, Context> {}

// TODO: extract into a utils module once we can do arithmetic on generics, i.e. https://github.com/noir-lang/noir/issues/4784
fn concat_arrays<N, M, O>(arr_n: [Field; N], arr_m: [Field; M]) -> [Field; O] {
assert_eq(N + M, O);
let mut out: [Field; O] = [0; O];
for i in 0..N {
out[i] = arr_n[i];
}
for i in 0..M {
out[N+i] = arr_m[i];
}
out
}

// SharedMutable<T> stores a value of type T that is:
// - publicly known (i.e. unencrypted)
// - mutable in public
// - readable in private with no contention (i.e. multiple parties can all read the same value without blocking one
// another nor needing to coordinate)
// This is famously a hard problem to solve. SharedMutable makes it work by introducing a delay to public mutation:
// the value is not changed immediately but rather a value change is scheduled to happen in the future after some delay
// measured in blocks. Reads in private are only valid as long as they are included in a block not too far into the
// measured in blocks. Reads in private are only valid as long as they are included in a block not too far into the
// future, so that they can guarantee the value will not have possibly changed by then (because of the delay).
// The delay for changing a value is initially equal to INITIAL_DELAY, but can be changed by calling
// The delay for changing a value is initially equal to INITIAL_DELAY, but can be changed by calling
// `schedule_delay_change`.
impl<T, INITIAL_DELAY, Context> SharedMutable<T, INITIAL_DELAY, Context> {
pub fn new(context: Context, storage_slot: Field) -> Self {
assert(storage_slot != 0, "Storage slot 0 not allowed. Storage slots must start from 1.");
Self { context, storage_slot }
}

fn hash_scheduled_data(
value_change: ScheduledValueChange<T>,
delay_change: ScheduledDelayChange<INITIAL_DELAY>
) -> Field where T: ToField {
// TODO(#5491 and https://github.com/noir-lang/noir/issues/4784): update this so that we don't need to rely on
// ScheduledValueChange serializing to 3 and ScheduledDelayChange serializing to 1
let concatenated: [Field; 4] = concat_arrays(value_change.serialize(), delay_change.serialize());
poseidon2_hash(concatenated)
}

// Since we can't rely on the native storage allocation scheme, we hash the storage slot to get a unique location in
// which we can safely store as much data as we need.
// See https://github.com/AztecProtocol/aztec-packages/issues/5492 and
// which we can safely store as much data as we need.
// See https://github.com/AztecProtocol/aztec-packages/issues/5492 and
// https://github.com/AztecProtocol/aztec-packages/issues/5736
// We store three things in public storage:
// - a ScheduledValueChange
// - a ScheduledDelaChange
// - the hash of both of these (via `hash_scheduled_data`)
fn get_value_change_storage_slot(self) -> Field {
pedersen_hash([self.storage_slot, 0], 0)
}

fn get_delay_change_storage_slot(self) -> Field {
pedersen_hash([self.storage_slot, 1], 0)
}

fn get_hash_storage_slot(self) -> Field {
pedersen_hash([self.storage_slot, 2], 0)
}

// It may seem odd that we take a header and address instead of reading from e.g. a PrivateContext, but this lets us
// reuse this function in SharedMutablePrivateGetter.
fn historical_read_from_public_storage(
self,
header: Header,
address: AztecAddress
) -> (ScheduledValueChange<T>, ScheduledDelayChange<INITIAL_DELAY>, u32) where T: FromField + ToField + Eq {
let historical_block_number = header.global_variables.block_number as u32;

// We could simply produce historical inclusion proofs for both the ScheduledValueChange and
// ScheduledDelayChange, but that'd require one full sibling path per storage slot (since due to kernel siloing
// the storage is not contiguous), and in the best case in which T is a single field that'd be 4 slots.
// Instead, we get an oracle to provide us the correct values for both the value and delay changes, and instead
// prove inclusion of their hash, which is both a much smaller proof (a single slot), and also independent of
// the size of T.
let (value_change_hint, delay_change_hint) = get_public_storage_hints(address, self.storage_slot, historical_block_number);

// Ideally the following would be simply public_storage::read_historical, but we can't implement that yet.
let hash = header.public_storage_historical_read(self.get_hash_storage_slot(), address);

// @todo This is written strangely to bypass a formatting issue with the if that is breaking ci.
let (a, b, c) = if hash != 0 {
let a = SharedMutable::hash_scheduled_data(value_change_hint, delay_change_hint);
(a, value_change_hint, delay_change_hint)
} else {
// The hash slot can only hold a zero if it is uninitialized, meaning no value or delay change was ever
// scheduled. Therefore, the hints must then correspond to uninitialized scheduled changes.
let b = ScheduledValueChange::deserialize(zeroed());
let c = ScheduledDelayChange::deserialize(zeroed());
(hash, b, c)
};

assert_eq(hash, a, "Hint values do not match hash");
assert_eq(value_change_hint, b, "Non-zero value change for zero hash");
assert_eq(delay_change_hint, c, "Non-zero delay change for zero hash");

(value_change_hint, delay_change_hint, historical_block_number)
}
}

impl<T, INITIAL_DELAY> SharedMutable<T, INITIAL_DELAY, &mut PublicContext> {
pub fn schedule_value_change(self, new_value: T) {
pub fn schedule_value_change(self, new_value: T) where T: ToField {
let mut value_change = self.read_value_change();
let delay_change = self.read_delay_change();

Expand All @@ -65,17 +140,17 @@ impl<T, INITIAL_DELAY> SharedMutable<T, INITIAL_DELAY, &mut PublicContext> {
let block_of_change = block_number + current_delay;
value_change.schedule_change(new_value, block_number, current_delay, block_of_change);

self.write_value_change(value_change);
self.write(value_change, delay_change);
}

pub fn schedule_delay_change(self, new_delay: u32) {
pub fn schedule_delay_change(self, new_delay: u32) where T: ToField {
let mut delay_change = self.read_delay_change();

let block_number = self.context.block_number() as u32;

delay_change.schedule_change(new_delay, block_number);

self.write_delay_change(delay_change);
self.write(self.read_value_change(), delay_change);
}

pub fn get_current_value_in_public(self) -> T {
Expand Down Expand Up @@ -104,64 +179,64 @@ impl<T, INITIAL_DELAY> SharedMutable<T, INITIAL_DELAY, &mut PublicContext> {
self.context.storage_read(self.get_delay_change_storage_slot())
}

fn write_value_change(self, value_change: ScheduledValueChange<T>) {
fn write(
self,
value_change: ScheduledValueChange<T>,
delay_change: ScheduledDelayChange<INITIAL_DELAY>
) where T: ToField {
// Whenever we write to public storage, we write both the value change and delay change as well as the hash of
// them both. This guarantees that the hash is always kept up to date.
// While this makes for more costly writes, it also makes private proofs much simpler because they only need to
// produce a historical proof for the hash, which results in a single inclusion proof (as opposed to 4 in the
// best case scenario in which T is a single field). Private shared mutable reads are assumed to be much more
// frequent than public writes, so this tradeoff makes sense.
self.context.storage_write(self.get_value_change_storage_slot(), value_change);
}

fn write_delay_change(self, delay_change: ScheduledDelayChange<INITIAL_DELAY>) {
self.context.storage_write(self.get_delay_change_storage_slot(), delay_change);
self.context.storage_write(
self.get_hash_storage_slot(),
SharedMutable::hash_scheduled_data(value_change, delay_change)
);
}
}

impl<T, INITIAL_DELAY> SharedMutable<T, INITIAL_DELAY, &mut PrivateContext> {
pub fn get_current_value_in_private(self) -> T where T: FromField {
pub fn get_current_value_in_private(self) -> T where T: FromField + ToField + Eq {
// When reading the current value in private we construct a historical state proof for the public value.
// However, since this value might change, we must constrain the maximum transaction block number as this proof
// will only be valid for however many blocks we can ensure the value will not change, which will depend on the
// current delay and any scheduled delay changes.

let (value_change, delay_change, historical_block_number) = self.historical_read_from_public_storage(*self.context);
let (value_change, delay_change, historical_block_number) = self.historical_read_from_public_storage(self.context.get_header(), self.context.this_address());

// We use the effective minimum delay as opposed to the current delay at the historical block as this one also
// takes into consideration any scheduled delay changes.
// takes into consideration any scheduled delay changes.
// For example, consider a scenario in which at block 200 the current delay was 50. We may naively think that
// the earliest we could change the value would be at block 251 by scheduling immediately after the historical
// block, i.e. at block 201. But if there was a delay change scheduled for block 210 to reduce the delay to 20
// blocks, then if a value change was scheduled at block 210 it would go into effect at block 230, which is
// block, i.e. at block 201. But if there was a delay change scheduled for block 210 to reduce the delay to 20
// blocks, then if a value change was scheduled at block 210 it would go into effect at block 230, which is
// earlier than what we'd expect if we only considered the current delay.
let effective_minimum_delay = delay_change.get_effective_minimum_delay_at(historical_block_number);
let block_horizon = value_change.get_block_horizon(historical_block_number, effective_minimum_delay);

// We prevent this transaction from being included in any block after the block horizon, ensuring that the
// We prevent this transaction from being included in any block after the block horizon, ensuring that the
// historical public value matches the current one, since it can only change after the horizon.
self.context.set_tx_max_block_number(block_horizon);
value_change.get_current_at(historical_block_number)
}
}

fn historical_read_from_public_storage(
self,
context: PrivateContext
) -> (ScheduledValueChange<T>, ScheduledDelayChange<INITIAL_DELAY>, u32) where T: FromField {
let header = context.get_header();
// Ideally the following would be simply public_storage::read_historical, but we can't implement that yet.
let value_change_slot = self.get_value_change_storage_slot();
let mut raw_value_change_fields = [0; 3];
for i in 0..3 {
raw_value_change_fields[i] = header.public_storage_historical_read(
value_change_slot + i as Field,
context.this_address()
);
}

// Ideally the following would be simply public_storage::read_historical, but we can't implement that yet.
let delay_change_slot = self.get_delay_change_storage_slot();
let raw_delay_change_fields = [header.public_storage_historical_read(delay_change_slot, context.this_address())];

let value_change = ScheduledValueChange::deserialize(raw_value_change_fields);
let delay_change = ScheduledDelayChange::deserialize(raw_delay_change_fields);

let historical_block_number = context.historical_header.global_variables.block_number as u32;

(value_change, delay_change, historical_block_number)
}
unconstrained fn get_public_storage_hints<T, INITIAL_DELAY>(
address: AztecAddress,
storage_slot: Field,
block_number: u32
) -> (ScheduledValueChange<T>, ScheduledDelayChange<INITIAL_DELAY>) {
// This function cannot be part of the &mut PrivateContext impl because that'd mean that by passing `self` we'd also
// be passing a mutable reference to an unconstrained function, which is not allowed. We therefore create a dummy
// state variable here so that we can access the methods to compute storage slots. This will all be removed in the
// future once we do proper storage slot allocation (#5492).
let dummy = SharedMutable::new((), storage_slot);

(
storage_read(address, dummy.get_value_change_storage_slot(), block_number), storage_read(address, dummy.get_delay_change_storage_slot(), block_number)
)
}
Original file line number Diff line number Diff line change
@@ -1,9 +1,15 @@
use dep::protocol_types::{hash::pedersen_hash, traits::FromField, address::AztecAddress, header::Header};
use dep::protocol_types::{
hash::{pedersen_hash, poseidon2_hash}, traits::{FromField, ToField}, address::AztecAddress,
header::Header
};

use crate::context::PrivateContext;
use crate::state_vars::{
storage::Storage,
shared_mutable::{scheduled_delay_change::ScheduledDelayChange, scheduled_value_change::ScheduledValueChange}
shared_mutable::{
shared_mutable::SharedMutable, scheduled_delay_change::ScheduledDelayChange,
scheduled_value_change::ScheduledValueChange
}
};

struct SharedMutablePrivateGetter<T, INITIAL_DELAY> {
Expand All @@ -30,8 +36,12 @@ impl<T, INITIAL_DELAY> SharedMutablePrivateGetter<T, INITIAL_DELAY> {
Self { context, other_contract_address, storage_slot, _dummy: [0; INITIAL_DELAY] }
}

pub fn get_value_in_private(self, header: Header) -> T where T: FromField {
let (value_change, delay_change, historical_block_number) = self.historical_read_from_public_storage(header);
pub fn get_value_in_private(self, header: Header) -> T where T: FromField + ToField + Eq {
// We create a dummy SharedMutable state variable so that we can reuse its historical_read_from_public_storage
// method, greatly reducing code duplication.
let dummy: SharedMutable<T, INITIAL_DELAY, ()> = SharedMutable::new((), self.storage_slot);
let (value_change, delay_change, historical_block_number) = dummy.historical_read_from_public_storage(header, self.other_contract_address);

let effective_minimum_delay = delay_change.get_effective_minimum_delay_at(historical_block_number);
let block_horizon = value_change.get_block_horizon(historical_block_number, effective_minimum_delay);

Expand All @@ -44,36 +54,4 @@ impl<T, INITIAL_DELAY> SharedMutablePrivateGetter<T, INITIAL_DELAY> {

value_change.get_current_at(historical_block_number)
}

fn historical_read_from_public_storage(
self,
header: Header
) -> (ScheduledValueChange<T>, ScheduledDelayChange<INITIAL_DELAY>, u32) where T: FromField {
let value_change_slot = self.get_value_change_storage_slot();
let mut raw_value_change_fields = [0; 3];
for i in 0..3 {
raw_value_change_fields[i] = header.public_storage_historical_read(
value_change_slot + i as Field,
self.other_contract_address
);
}

let delay_change_slot = self.get_delay_change_storage_slot();
let raw_delay_change_fields = [header.public_storage_historical_read(delay_change_slot, self.other_contract_address)];

let value_change = ScheduledValueChange::deserialize(raw_value_change_fields);
let delay_change = ScheduledDelayChange::deserialize(raw_delay_change_fields);

let historical_block_number = header.global_variables.block_number as u32;

(value_change, delay_change, historical_block_number)
}

fn get_value_change_storage_slot(self) -> Field {
pedersen_hash([self.storage_slot, 0], 0)
}

fn get_delay_change_storage_slot(self) -> Field {
pedersen_hash([self.storage_slot, 1], 0)
}
}
Loading

0 comments on commit 868606e

Please sign in to comment.