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

Zk server compatibility #221

Merged
merged 60 commits into from
Sep 4, 2024
Merged
Show file tree
Hide file tree
Changes from 55 commits
Commits
Show all changes
60 commits
Select commit Hold shift + click to select a range
5de8afb
Derive partial-eq for VmState
MarcosNicolau Aug 13, 2024
59470a7
Add initial gas param for VmState new
MarcosNicolau Aug 13, 2024
26c2b18
Update era-compiler-tester branch
MarcosNicolau Aug 13, 2024
e5cb98b
fix test bytecode_publishing
juan518munoz Aug 14, 2024
f0dce41
Store changes for get_used_contracts tests
MarcosNicolau Aug 14, 2024
c862649
adapt to zksync-era tests
juan518munoz Aug 16, 2024
bbac76d
Remove hash_map from Storage trait
MarcosNicolau Aug 16, 2024
5ca9edf
Fix increment_tx_number not cleaning transient_storage
MarcosNicolau Aug 20, 2024
01c6eb6
Fix decommit opcode
MarcosNicolau Aug 20, 2024
be5024f
Various far_call fixes
MarcosNicolau Aug 20, 2024
cdfcd92
Add stipend param to frame and decrease it in panics and reverts
MarcosNicolau Aug 20, 2024
5be7e35
Add no refund storage_read for far_call decommit_code_hash
MarcosNicolau Aug 20, 2024
29c184a
Remove rollbacks from main context
MarcosNicolau Aug 20, 2024
679d38d
Reimplement vec snapshots
MarcosNicolau Aug 20, 2024
1c9a4b2
Add external snapshot(whole vm)
MarcosNicolau Aug 20, 2024
f6f2829
Add storage cache and util functio to get storage changes from initial
MarcosNicolau Aug 20, 2024
8747004
Refactor state pub fields
MarcosNicolau Aug 20, 2024
2a28162
Add new function to get storage changes from snapshot or initial
MarcosNicolau Aug 20, 2024
6509f4a
Merge branch 'main' into zksync-era-integration-tests
MarcosNicolau Aug 20, 2024
41db1b5
Update zksync-era submodule
MarcosNicolau Aug 20, 2024
fe7a666
Update era-compiler-tester submodule
MarcosNicolau Aug 20, 2024
18c5199
Address clippy warnings
MarcosNicolau Aug 20, 2024
897eb16
More clippy warnings
MarcosNicolau Aug 20, 2024
4efc066
Update zksync-era submodule
MarcosNicolau Aug 20, 2024
b512b31
ci: add submodule step for era
fkrause98 Aug 20, 2024
a5700ea
Address review comments
MarcosNicolau Aug 20, 2024
c30ea73
Add cache to storage_read
MarcosNicolau Aug 21, 2024
cd451dd
Update zksync-era submodule
MarcosNicolau Aug 22, 2024
d0c59af
Add full tracers implementation
MarcosNicolau Aug 22, 2024
855d043
Refactor how the vm handles tracers
MarcosNicolau Aug 22, 2024
439f68e
Update era-compiler-tester submodule
MarcosNicolau Aug 22, 2024
e5ab604
Update zksync-era submodule
MarcosNicolau Aug 22, 2024
6d1682e
Address clippy warnings
MarcosNicolau Aug 22, 2024
0cc8684
Add default function in Tracer trait
MarcosNicolau Aug 23, 2024
081d615
Address clippy warnings
MarcosNicolau Aug 23, 2024
ee97d09
Address review comments
MarcosNicolau Aug 26, 2024
5baf370
Add docs to state
MarcosNicolau Aug 26, 2024
75434eb
Address review comments
MarcosNicolau Aug 26, 2024
66d412c
Remove optional from run with tracers
MarcosNicolau Aug 26, 2024
291e4c1
Update era-compiler-tester submodule
MarcosNicolau Aug 26, 2024
02e8a90
Update era-compiler-tester submodule
MarcosNicolau Aug 26, 2024
ed3784c
Address clippy warnings
MarcosNicolau Aug 26, 2024
aea62fe
Fix storage_read
MarcosNicolau Aug 26, 2024
e6b1c19
Merge remote-tracking branch 'origin' into zksync-era-integration-tests
MarcosNicolau Aug 26, 2024
0385534
Merge branch 'zksync-era-tests' into implement-tracers
MarcosNicolau Aug 26, 2024
caad06d
Merge branch 'zksync-era-integration-tests' into implement-tracers
MarcosNicolau Aug 26, 2024
c06ce65
Merge branch 'main' into implement-tracers
MarcosNicolau Aug 28, 2024
fc2855a
Implement statistics (#213)
MarcosNicolau Aug 29, 2024
c36cf47
Merge branch 'main' into implement-tracers
MarcosNicolau Aug 29, 2024
5de6b4d
Update zksync-era submodule
MarcosNicolau Aug 29, 2024
adc3c0e
Move storage param to vm run
MarcosNicolau Aug 30, 2024
c1ab599
Merge branch 'main' into zk_server_compatibility
gianbelinche Sep 3, 2024
05c6bb9
Update zksync era submodule
gianbelinche Sep 3, 2024
beb8ef0
Update compiler tester submodule
gianbelinche Sep 3, 2024
264c0d0
Lint
gianbelinche Sep 3, 2024
d9f581f
Remove print
gianbelinche Sep 3, 2024
1e6167a
Update zksync era submodule
gianbelinche Sep 3, 2024
2a213dc
Add heap size optimization
gianbelinche Sep 4, 2024
a44f3b8
Update submodules
gianbelinche Sep 4, 2024
c4dab0e
Address comments
gianbelinche Sep 4, 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
2 changes: 1 addition & 1 deletion era-compiler-tester
56 changes: 36 additions & 20 deletions src/op_handlers/far_call.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,11 +15,13 @@ use crate::{
rollbacks::Rollbackable,
state::VMState,
statistics::{VmStatistics, STORAGE_READ_STORAGE_APPLICATION_CYCLES},
store::{StorageError, StorageKey},
store::{Storage, StorageError, StorageKey},
utils::{address_into_u256, is_kernel},
value::{FatPointer, TaggedValue},
Opcode,
};

use super::ret::panic_from_far_call;
#[allow(dead_code)]
struct FarCallParams {
forward_memory: FatPointer,
Expand Down Expand Up @@ -134,14 +136,15 @@ fn decommit_code_hash(
default_aa_code_hash: [u8; 32],
evm_interpreter_code_hash: [u8; 32],
is_constructor_call: bool,
storage: &mut dyn Storage,
) -> Result<(U256, bool, u32), EraVmError> {
let mut is_evm = false;
let deployer_system_contract_address =
Address::from_low_u64_be(DEPLOYER_SYSTEM_CONTRACT_ADDRESS_LOW as u64);
let storage_key = StorageKey::new(deployer_system_contract_address, address_into_u256(address));

// reading when decommiting doesn't refund
let code_info = state.storage_read_with_no_refund(storage_key);
let code_info = state.storage_read_with_no_refund(storage_key, storage);
let mut code_info_bytes = [0; 32];
code_info.to_big_endian(&mut code_info_bytes);

Expand Down Expand Up @@ -216,6 +219,7 @@ pub fn far_call(
far_call: &FarCallOpcode,
state: &mut VMState,
statistics: &mut VmStatistics,
storage: &mut dyn Storage,
) -> Result<(), EraVmError> {
let (src0, src1) = address_operands_read(vm, opcode)?;
let contract_address = address_from_u256(&src1.value);
Expand All @@ -227,45 +231,57 @@ pub fn far_call(
abi.is_constructor_call = abi.is_constructor_call && vm.current_context()?.is_kernel();
abi.is_system_call = abi.is_system_call && is_kernel(&contract_address);

let FarCallParams {
ergs_passed,
forward_memory,
..
} = far_call_params_from_register(src0, vm)?;

let mut mandated_gas = if abi.is_system_call && src1.value == ADDRESS_MSG_VALUE.into() {
MSG_VALUE_SIMULATOR_ADDITIVE_COST
} else {
0
};

let (code_key, is_evm, decommit_cost) = decommit_code_hash(
state,
contract_address,
vm.default_aa_code_hash,
vm.evm_interpreter_code_hash,
abi.is_constructor_call,
storage,
)?;

// Unlike all other gas costs, this one is not paid if low on gas.
if decommit_cost <= vm.gas_left()? {
vm.decrease_gas(decommit_cost)?;
if vm.decrease_gas(mandated_gas).is_err() {
vm.set_gas_left(0)?;
mandated_gas = 0;
panic_from_far_call(vm, opcode)?;
} else {
return Err(EraVmError::DecommitFailed);
// Pay for decommit
// Unlike all other gas costs, this one is not paid if low on gas.
if decommit_cost <= vm.gas_left()? {
vm.decrease_gas(decommit_cost)?;
} else {
return Err(EraVmError::DecommitFailed);
}
}

let FarCallParams {
ergs_passed,
forward_memory,
..
} = far_call_params_from_register(src0, vm)?;

let mandated_gas = if abi.is_system_call && src1.value == ADDRESS_MSG_VALUE.into() {
MSG_VALUE_SIMULATOR_ADDITIVE_COST
} else {
0
};
let gas_left = vm.gas_left()?;
let maximum_gas =
gas_left / FAR_CALL_GAS_SCALAR_MODIFIER_DIVISOR * FAR_CALL_GAS_SCALAR_MODIFIER_DIVIDEND;
let ergs_passed = ergs_passed.min(maximum_gas);
vm.decrease_gas(ergs_passed)?;

// mandated gas can surprass the 63/64 limit
let ergs_passed = ergs_passed + mandated_gas;

vm.decrease_gas(ergs_passed)?;

let stipend = if is_evm { EVM_SIMULATOR_STIPEND } else { 0 };

let ergs_passed = (ergs_passed)
.checked_add(stipend)
.expect("stipend must not cause overflow");

let (program_code, was_decommited) = state.decommit(code_key);
let (program_code, was_decommited) = state.decommit(code_key, storage);

let program_code = program_code.ok_or(StorageError::KeyNotPresent)?;
if !was_decommited {
Expand Down
12 changes: 9 additions & 3 deletions src/op_handlers/log.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ use crate::{
VmStatistics, STORAGE_READ_STORAGE_APPLICATION_CYCLES,
STORAGE_WRITE_STORAGE_APPLICATION_CYCLES,
},
store::StorageKey,
store::{Storage, StorageKey},
value::TaggedValue,
Opcode,
};
Expand All @@ -16,6 +16,7 @@ pub fn storage_write(
opcode: &Opcode,
state: &mut VMState,
statistics: &mut VmStatistics,
storage: &mut dyn Storage,
) -> Result<(), EraVmError> {
let key_for_contract_storage = vm.get_register(opcode.src0_index).value;
let address = vm.current_context()?.contract_address;
Expand All @@ -24,7 +25,7 @@ pub fn storage_write(
statistics.storage_application_cycles += STORAGE_WRITE_STORAGE_APPLICATION_CYCLES;
}
let value = vm.get_register(opcode.src1_index).value;
let refund = state.storage_write(key, value);
let refund = state.storage_write(key, value, storage);
vm.increase_gas(refund)?;
Ok(())
}
Expand All @@ -34,6 +35,7 @@ pub fn storage_read(
opcode: &Opcode,
state: &mut VMState,
statistics: &mut VmStatistics,
storage: &mut dyn Storage,
) -> Result<(), EraVmError> {
let key_for_contract_storage = vm.get_register(opcode.src0_index).value;
let address = vm.current_context()?.contract_address;
Expand All @@ -43,7 +45,7 @@ pub fn storage_read(
if !state.read_storage_slots().contains(&key) && !state.written_storage_slots().contains(&key) {
statistics.storage_application_cycles += STORAGE_READ_STORAGE_APPLICATION_CYCLES;
}
let (value, refund) = state.storage_read(key);
let (value, refund) = state.storage_read(key, storage);
vm.increase_gas(refund)?;
vm.set_register(opcode.dst0_index, TaggedValue::new_raw_integer(value));
Ok(())
Expand Down Expand Up @@ -82,6 +84,10 @@ pub fn add_l2_to_l1_message(
) -> Result<(), EraVmError> {
let key = vm_state.get_register(opcode.src0_index).value;
let value = vm_state.get_register(opcode.src1_index).value;
println!(
"L1 TO L2 regusters indexes {} {}",
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

typo here

opcode.src0_index, opcode.src1_index
);
let is_service = opcode.imm0 == 1;
state.record_l2_to_l1_log(L2ToL1Log {
key,
Expand Down
4 changes: 3 additions & 1 deletion src/op_handlers/opcode_decommit.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ use crate::{
execution::Execution,
state::VMState,
statistics::{VmStatistics, STORAGE_READ_STORAGE_APPLICATION_CYCLES},
store::Storage,
value::{FatPointer, TaggedValue},
Opcode,
};
Expand All @@ -15,6 +16,7 @@ pub fn opcode_decommit(
opcode: &Opcode,
state: &mut VMState,
statistics: &mut VmStatistics,
storage: &mut dyn Storage,
) -> Result<(), EraVmError> {
let (src0, src1) = address_operands_read(vm, opcode)?;

Expand All @@ -34,7 +36,7 @@ pub fn opcode_decommit(
return Ok(());
}

let (code, was_decommited) = state.decommit(code_hash);
let (code, was_decommited) = state.decommit(code_hash, storage);
if was_decommited {
// refund it
vm.increase_gas(extra_cost)?;
Expand Down
60 changes: 34 additions & 26 deletions src/state.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,7 @@ use crate::{
},
store::{Storage, StorageKey},
};
use std::{
cell::RefCell,
collections::{HashMap, HashSet},
rc::Rc,
};
use std::collections::{HashMap, HashSet};
use u256::{H160, U256};
use zkevm_opcode_defs::system_params::{
STORAGE_ACCESS_COLD_READ_COST, STORAGE_ACCESS_COLD_WRITE_COST, STORAGE_ACCESS_WARM_READ_COST,
Expand Down Expand Up @@ -41,7 +37,6 @@ pub struct Event {

#[derive(Debug, Clone)]
pub struct VMState {
pub storage: Rc<RefCell<dyn Storage>>,
storage_changes: RollbackableHashMap<StorageKey, U256>,
transient_storage: RollbackableHashMap<StorageKey, U256>,
l2_to_l1_logs: RollbackableVec<L2ToL1Log>,
Expand All @@ -59,10 +54,15 @@ pub struct VMState {
decommitted_hashes: RollbackableHashSet<U256>,
}

impl Default for VMState {
fn default() -> Self {
Self::new()
}
}

impl VMState {
pub fn new(storage: Rc<RefCell<dyn Storage>>) -> Self {
pub fn new() -> Self {
Self {
storage,
storage_changes: RollbackableHashMap::<StorageKey, U256>::default(),
transient_storage: RollbackableHashMap::<StorageKey, U256>::default(),
l2_to_l1_logs: RollbackableVec::<L2ToL1Log>::default(),
Expand All @@ -78,7 +78,7 @@ impl VMState {
}

pub fn reset(&mut self) {
*self = Self::new(self.storage.clone());
*self = Self::new();
}

pub fn storage_changes(&self) -> &HashMap<StorageKey, U256> {
Expand Down Expand Up @@ -137,13 +137,11 @@ impl VMState {

// reads shouldn't be mutable, we should consider change it to a non-mutable reference
// though that would require a refactor in the integration with the operator
pub fn storage_read(&mut self, key: StorageKey) -> (U256, u32) {
pub fn storage_read(&mut self, key: StorageKey, storage: &mut dyn Storage) -> (U256, u32) {
let value = self
.storage_read_inner(&key)
.storage_read_inner(&key, storage)
.map_or_else(U256::zero, |val| val);

let storage = self.storage.borrow();

let refund =
if storage.is_free_storage_slot(&key) || self.read_storage_slots.map.contains(&key) {
WARM_READ_REFUND
Expand All @@ -158,11 +156,14 @@ impl VMState {
(value, refund)
}

pub fn storage_read_with_no_refund(&mut self, key: StorageKey) -> U256 {
pub fn storage_read_with_no_refund(
&mut self,
key: StorageKey,
storage: &mut dyn Storage,
) -> U256 {
let value = self
.storage_read_inner(&key)
.storage_read_inner(&key, storage)
.map_or_else(U256::zero, |val| val);
let storage = self.storage.borrow();

if !storage.is_free_storage_slot(&key) && !self.read_storage_slots.map.contains(&key) {
self.read_storage_slots.map.insert(key);
Expand All @@ -173,18 +174,21 @@ impl VMState {
value
}

fn storage_read_inner(&self, key: &StorageKey) -> Option<U256> {
fn storage_read_inner(&self, key: &StorageKey, storage: &mut dyn Storage) -> Option<U256> {
match self.storage_changes.map.get(key) {
None => self.storage.borrow_mut().storage_read(key),
None => storage.storage_read(key),
value => value.copied(),
}
}

pub fn storage_write(&mut self, key: StorageKey, value: U256) -> u32 {
pub fn storage_write(
&mut self,
key: StorageKey,
value: U256,
storage: &mut dyn Storage,
) -> u32 {
self.storage_changes.map.insert(key, value);

let mut storage = self.storage.borrow_mut();

if storage.is_free_storage_slot(&key) {
self.written_storage_slots.map.insert(key);
let refund = WARM_WRITE_REFUND;
Expand Down Expand Up @@ -257,9 +261,9 @@ impl VMState {
/// A tuple containing:
/// - `Option<Vec<U256>>`: the contract bytecode
/// - `bool`: A boolean flag indicating whether the hash was decommitted (`true` if it was newly decommitted, `false` if it had already been decommitted).
pub fn decommit(&mut self, hash: U256) -> (Option<Vec<U256>>, bool) {
pub fn decommit(&mut self, hash: U256, storage: &mut dyn Storage) -> (Option<Vec<U256>>, bool) {
let was_decommitted = !self.decommitted_hashes.map.insert(hash);
(self.storage.borrow_mut().decommit(hash), was_decommitted)
(storage.decommit(hash), was_decommitted)
}

pub fn decommitted_hashes(&self) -> &HashSet<U256> {
Expand All @@ -274,12 +278,15 @@ impl VMState {
/// - `StorageKey`: The key for the storage value.
/// - `Option<U256>`: The initial value from the storage.
/// - `U256`: The current value after the change.
pub fn get_storage_changes(&mut self) -> Vec<(StorageKey, Option<U256>, U256)> {
pub fn get_storage_changes(
&mut self,
storage: &mut dyn Storage,
) -> Vec<(StorageKey, Option<U256>, U256)> {
self.storage_changes
.map
.iter()
.filter_map(|(key, value)| {
let initial_value = self.storage.borrow_mut().storage_read(key);
let initial_value = storage.storage_read(key);
if initial_value.unwrap_or_default() == *value {
None
} else {
Expand All @@ -302,12 +309,13 @@ impl VMState {
pub fn get_storage_changes_from_snapshot(
&self,
snapshot: <RollbackableHashMap<StorageKey, U256> as Rollbackable>::Snapshot,
storage: &mut dyn Storage,
) -> Vec<(StorageKey, Option<U256>, U256, bool)> {
self.storage_changes
.get_logs_after_snapshot(snapshot)
.iter()
.map(|(key, (before, after))| {
let initial = self.storage.borrow_mut().storage_read(key);
let initial = storage.storage_read(key);
(*key, before.or(initial), *after, initial.is_none())
})
.collect()
Expand Down
Loading
Loading