Skip to content

Commit

Permalink
feat: sstore
Browse files Browse the repository at this point in the history
  • Loading branch information
enitrat committed Jan 17, 2025
1 parent 170de09 commit 3563b9d
Show file tree
Hide file tree
Showing 4 changed files with 153 additions and 4 deletions.
2 changes: 1 addition & 1 deletion cairo/ethereum/cancun/vm.cairo
Original file line number Diff line number Diff line change
Expand Up @@ -297,7 +297,7 @@ namespace EvmImpl {
return ();
}

func set_refund_counter{evm: Evm}(new_refund_counter: Uint) {
func set_refund_counter{evm: Evm}(new_refund_counter: felt) {
tempvar evm = Evm(
new EvmStruct(
pc=evm.value.pc,
Expand Down
141 changes: 139 additions & 2 deletions cairo/ethereum/cancun/vm/instructions/storage.cairo
Original file line number Diff line number Diff line change
@@ -1,8 +1,15 @@
from ethereum.cancun.vm.stack import pop, push
from ethereum.cancun.vm import Evm, EvmImpl, Environment, EnvImpl
from ethereum.cancun.vm.exceptions import ExceptionalHalt, WriteInStaticContext
from ethereum.cancun.vm.exceptions import ExceptionalHalt, WriteInStaticContext, OutOfGasError
from ethereum.cancun.vm.gas import charge_gas, GasConstants
from ethereum.cancun.state import get_storage, get_transient_storage, set_transient_storage
from ethereum.utils.numeric import U256__eq__
from ethereum.cancun.state import (
get_storage,
get_storage_original,
set_storage,
get_transient_storage,
set_transient_storage,
)
from ethereum.cancun.fork_types import (
SetTupleAddressBytes32,
SetTupleAddressBytes32DictAccess,
Expand All @@ -21,6 +28,7 @@ from starkware.cairo.common.cairo_builtins import PoseidonBuiltin, BitwiseBuilti
from starkware.cairo.lang.compiler.lib.registers import get_fp_and_pc
from starkware.cairo.common.alloc import alloc
from starkware.cairo.common.dict_access import DictAccess
from starkware.cairo.common.math_cmp import is_le

// @notice Loads to the stack the value corresponding to a certain key from the
// storage of the current account.
Expand Down Expand Up @@ -96,6 +104,135 @@ func sload{range_check_ptr, bitwise_ptr: BitwiseBuiltin*, poseidon_ptr: Poseidon
return ok;
}

// @notice Stores a value at a certain key in the current context's storage.
func sstore{
range_check_ptr, bitwise_ptr: BitwiseBuiltin*, poseidon_ptr: PoseidonBuiltin*, evm: Evm
}() -> ExceptionalHalt* {
alloc_locals;
// STACK
let stack = evm.value.stack;
with stack {
let (key, err) = pop();
if (cast(err, felt) != 0) {
return err;
}
let (new_value, err) = pop();
if (cast(err, felt) != 0) {
return err;
}
}

let is_gas_left_not_enough = is_le(evm.value.gas_left.value, GasConstants.GAS_CALL_STIPEND);
if (is_gas_left_not_enough != 0) {
tempvar err = new ExceptionalHalt(OutOfGasError);
return err;
}

// Get storage values
let key_bytes32 = U256_to_be_bytes(key);
let state = evm.value.env.value.state;
let current_target = evm.value.message.value.current_target;
with state {
let original_value = get_storage_original(current_target, key_bytes32);
let current_value = get_storage(current_target, key_bytes32);
}

// Gas calculation
// Check accessed storage keys
tempvar accessed_tuple = TupleAddressBytes32(
new TupleAddressBytes32Struct(current_target, key_bytes32)
);
let (serialized_keys: felt*) = alloc();
assert serialized_keys[0] = accessed_tuple.value.address.value;
assert serialized_keys[1] = accessed_tuple.value.bytes32.value.low;
assert serialized_keys[2] = accessed_tuple.value.bytes32.value.high;
let dict_ptr = cast(evm.value.accessed_storage_keys.value.dict_ptr, DictAccess*);
with dict_ptr {
let (is_present) = hashdict_read(3, serialized_keys);
if (is_present == 0) {
hashdict_write(3, serialized_keys, 1);
tempvar gas_cost = GasConstants.GAS_COLD_SLOAD;
tempvar poseidon_ptr = poseidon_ptr;
tempvar dict_ptr = dict_ptr;
} else {
tempvar gas_cost = 0;
tempvar poseidon_ptr = poseidon_ptr;
tempvar dict_ptr = dict_ptr;
}
}
let gas_cost = [ap - 3];
let poseidon_ptr = cast([ap - 2], PoseidonBuiltin*);
let dict_ptr = cast([ap - 1], DictAccess*);

// Calculate storage gas cost
tempvar zero_u256 = U256(new U256Struct(0, 0));
let is_original_eq_current = U256__eq__(original_value, current_value);
let is_current_eq_new = U256__eq__(current_value, new_value);
let is_original_zero = U256__eq__(original_value, zero_u256);
if (is_original_eq_current.value != 0) {
if (is_current_eq_new.value == 0) {
if (is_original_zero.value != 0) {
tempvar gas_cost = gas_cost + GasConstants.GAS_STORAGE_SET;
} else {
tempvar gas_cost = gas_cost + (
GasConstants.GAS_STORAGE_UPDATE - GasConstants.GAS_COLD_SLOAD
);
}
}
} else {
tempvar gas_cost = GasConstants.GAS_WARM_ACCESS;
}
let gas_cost = [ap - 1];

tempvar refund_counter = evm.value.refund_counter;
let is_original_eq_new = U256__eq__(original_value, new_value);
// Refund calculation
if (is_current_eq_new.value == 0) {
let is_current_zero = U256__eq__(current_value, zero_u256);
let is_new_zero = U256__eq__(new_value, zero_u256);
if (is_original_zero.value == 0 and is_current_zero.value == 0 and is_new_zero.value != 0) {
refund_counter = refund_counter + GasConstants.GAS_STORAGE_CLEAR_REFUND;
}
if (is_original_zero.value == 0 and is_current_zero.value != 0) {
refund_counter = refund_counter - GasConstants.GAS_STORAGE_CLEAR_REFUND;
}
if (is_original_eq_new.value != 0) {
if (is_original_zero.value != 0) {
refund_counter = refund_counter +
(GasConstants.GAS_STORAGE_SET - GasConstants.GAS_WARM_ACCESS);
} else {
refund_counter = refund_counter +
(GasConstants.GAS_STORAGE_UPDATE - GasConstants.GAS_COLD_SLOAD - GasConstants.GAS_WARM_ACCESS);
}
}
}

// Charge gas
let err = charge_gas(Uint(gas_cost));
if (cast(err, felt) != 0) {
return err;
}
// Check static call
if (evm.value.message.value.is_static.value != 0) {
tempvar err = new ExceptionalHalt(WriteInStaticContext);
return err;
}

// Set storage
with state {
set_storage(current_target, key_bytes32, new_value);
}

// Update EVM state
let env = evm.value.env;
EnvImpl.set_state{env=env}(state);
EvmImpl.set_env(env);
EvmImpl.set_pc_stack(Uint(evm.value.pc.value + 1), stack);
EvmImpl.set_refund_counter(refund_counter);
let ok = cast(0, ExceptionalHalt*);
return ok;
}

// @notice Loads to the stack the value corresponding to a certain key from the
// transient storage of the current account.
func tload{range_check_ptr, bitwise_ptr: BitwiseBuiltin*, poseidon_ptr: PoseidonBuiltin*, evm: Evm}(
Expand Down
13 changes: 12 additions & 1 deletion cairo/tests/ethereum/cancun/vm/instructions/test_storage.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
from hypothesis import strategies as st
from hypothesis.strategies import composite

from ethereum.cancun.vm.instructions.storage import sload, tload, tstore
from ethereum.cancun.vm.instructions.storage import sload, sstore, tload, tstore
from tests.utils.args_gen import Evm
from tests.utils.errors import strict_raises
from tests.utils.evm_builder import EvmBuilder
Expand Down Expand Up @@ -45,6 +45,17 @@ def test_sload(self, cairo_run, evm: Evm):
sload(evm)
assert evm == cairo_evm

@given(evm=evm_with_accessed_storage_keys())
def test_sstore(self, cairo_run, evm: Evm):
try:
cairo_evm = cairo_run("sstore", evm)
except Exception as cairo_error:
with assert_raises_exactly(type(cairo_error)):
sstore(evm)
return
sstore(evm)
assert evm == cairo_evm

@given(evm=evm_with_accessed_storage_keys())
def test_tload(self, cairo_run, evm: Evm):
try:
Expand Down
1 change: 1 addition & 0 deletions cairo/tests/fixtures/runner.py
Original file line number Diff line number Diff line change
Expand Up @@ -460,6 +460,7 @@ def _run(entrypoint, *args, **kwargs):
except Exception as e:
if "An ASSERT_EQ instruction failed" in str(e):
raise AssertionError(e) from e
raise Exception(str(e)) from e

cumulative_retdata_offsets = serde.get_offsets(return_data_types)
first_return_data_offset = (
Expand Down

0 comments on commit 3563b9d

Please sign in to comment.