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: Integrate tracers and implement circuits tracer in vm2 #2653

Merged
merged 68 commits into from
Sep 4, 2024
Merged
Show file tree
Hide file tree
Changes from 63 commits
Commits
Show all changes
68 commits
Select commit Hold shift + click to select a range
cd59c72
integrate vm being generic over tracer type
montekki Aug 13, 2024
0f23f10
initial skeleton of tracers interface for vm2
montekki Aug 14, 2024
bb97637
wip code, some of the stats work some dont
montekki Aug 15, 2024
4c2c4c3
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 15, 2024
4e195f8
all except storage statistics is supposed to work
montekki Aug 16, 2024
7cb67fc
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 16, 2024
a9e9893
reset to main
montekki Aug 16, 2024
1591b7b
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 16, 2024
86fd8de
newer interface
montekki Aug 20, 2024
097fd1a
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 20, 2024
236ab63
per opcode statistics
montekki Aug 20, 2024
2c35dd1
update vm2 ref
joonazan Aug 20, 2024
e491ca0
vm2 version that fixes segfault
joonazan Aug 20, 2024
5da39a8
save pc in vm snapshots, not in integration
joonazan Aug 20, 2024
214b97a
bump vm2 to segfault fix
montekki Aug 21, 2024
f56ca44
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 21, 2024
5c64fe7
count precompile cycles
montekki Aug 21, 2024
3c7aaea
code decommitter stats
montekki Aug 21, 2024
cd4e4cd
updates to newest vm2 code
montekki Aug 21, 2024
66a02a1
storage stats
montekki Aug 21, 2024
4a565b0
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 21, 2024
adad0e9
fix clippy
montekki Aug 22, 2024
fd708cf
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 22, 2024
8198784
reduce the tracer to just increment one number
joonazan Aug 22, 2024
ad3613c
integrate newer changes
montekki Aug 23, 2024
57ea5d0
fix clippy
montekki Aug 23, 2024
746a788
increase timeout
montekki Aug 23, 2024
dde82ed
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 26, 2024
716ea2e
Revert "reduce the tracer to just increment one number"
montekki Aug 26, 2024
921b007
fix secpverify counter
montekki Aug 26, 2024
7c5afb2
rollback contracts config
montekki Aug 26, 2024
2061a44
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 26, 2024
374ae7e
adds a test contract for ret panic that doesnt yet work
montekki Aug 27, 2024
b66876f
update vm2
joonazan Aug 27, 2024
feef33d
remove debug print
joonazan Aug 27, 2024
7bc28a8
use gas
montekki Aug 28, 2024
1133e84
updates contract, triggers the ret revert
montekki Aug 28, 2024
6da030e
Merge remote-tracking branch 'origin/main' into fvs-vm-fast-circuits-…
montekki Aug 28, 2024
6000bf8
resolve warning
montekki Aug 28, 2024
7e61fe1
bring back debug derives
montekki Aug 28, 2024
08c608b
restore prover cargo.lock
montekki Aug 28, 2024
ce5c5fb
Merge branch 'main' into fvs-vm-fast-circuits-tracer
montekki Aug 28, 2024
fccc292
update vm2 dep
joonazan Aug 28, 2024
043cdec
update prover lockfile
joonazan Aug 28, 2024
3d80346
try rebased dep
joonazan Aug 28, 2024
d63190f
update prover deps
joonazan Aug 28, 2024
c40bd44
once again bump dep
joonazan Aug 29, 2024
5599280
Merge branch 'main' into fvs-vm-fast-circuits-tracer
joonazan Aug 29, 2024
1b90aa5
update to version that computes storage cycles afterwards
joonazan Aug 29, 2024
763c7ed
don't compare cycle counts
joonazan Aug 29, 2024
0f24763
point fixed and historycleaned version
joonazan Aug 29, 2024
e472c37
Merge branch 'main' into fvs-vm-fast-circuits-tracer
joonazan Aug 29, 2024
fbd7396
try not fixing bug
joonazan Aug 29, 2024
e4ceb34
verify that test pass before refactoring
joonazan Aug 29, 2024
28b0f12
don't count decommits
joonazan Aug 30, 2024
3d7a16d
Merge branch 'main' into fvs-vm-fast-circuits-tracer
joonazan Aug 30, 2024
25ddc89
count storage cycles on the fly because order matters
joonazan Aug 30, 2024
066cbda
integrate
joonazan Sep 3, 2024
4d73bc2
point at vm2 master
joonazan Sep 3, 2024
f19fa75
Merge branch 'main' into fvs-vm-fast-circuits-tracer
joonazan Sep 3, 2024
5d15f06
fix formatting
joonazan Sep 3, 2024
9c20371
move circuits tracer to module
joonazan Sep 3, 2024
64d8fc5
import constants
joonazan Sep 3, 2024
8b5992b
remove partialeq
joonazan Sep 3, 2024
2cc74ff
clean up test contract
joonazan Sep 3, 2024
36a222f
reduce visibility of CT
joonazan Sep 3, 2024
8ba28d6
Merge branch 'main' into fvs-vm-fast-circuits-tracer
joonazan Sep 3, 2024
2a7c41f
don't edit contracts lib just for a test
joonazan Sep 3, 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
11 changes: 10 additions & 1 deletion Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -217,7 +217,7 @@ zk_evm_1_4_1 = { package = "zk_evm", version = "0.141.0" }
zk_evm_1_5_0 = { package = "zk_evm", version = "=0.150.4" }

# New VM; pinned to a specific commit because of instability
vm2 = { git = "https://github.com/matter-labs/vm2.git", rev = "2276b7b5af520fca0477bdafe43781b51896d235" }
vm2 = { git = "https://github.com/matter-labs/vm2.git", rev = "4ef15d46410ffc11744771a3a6c7c09dd9470c90" }

# Consensus dependencies.
zksync_concurrency = "=0.1.0-rc.11"
Expand Down
15 changes: 15 additions & 0 deletions core/lib/contracts/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,8 @@ const LOADNEXT_CONTRACT_FILE: &str =
"etc/contracts-test-data/artifacts-zk/contracts/loadnext/loadnext_contract.sol/LoadnextContract.json";
const LOADNEXT_SIMPLE_CONTRACT_FILE: &str =
"etc/contracts-test-data/artifacts-zk/contracts/loadnext/loadnext_contract.sol/Foo.json";
const FAILEDCALL_CONTRACT_FILE: &str =
"etc/contracts-test-data/artifacts-zk/contracts/failed-call/failed_call.sol/FailedCall.json";

fn home_path() -> &'static Path {
workspace_dir_or_current_dir()
Expand Down Expand Up @@ -184,11 +186,24 @@ pub fn get_loadnext_contract() -> TestContract {
}
}

pub fn get_failedcall_contract() -> TestContract {
joonazan marked this conversation as resolved.
Show resolved Hide resolved
let bytecode = read_bytecode(FAILEDCALL_CONTRACT_FILE);
TestContract {
bytecode,
contract: failedcall_contract(),
factory_deps: vec![],
}
}

// Returns loadnext contract and its factory dependencies
fn loadnext_contract() -> Contract {
load_contract("etc/contracts-test-data/artifacts-zk/contracts/loadnext/loadnext_contract.sol/LoadnextContract.json")
}

fn failedcall_contract() -> Contract {
load_contract("etc/contracts-test-data/artifacts-zk/contracts/failed-call/failed_call.sol/FailedCall.json")
}

pub fn deployer_contract() -> Contract {
load_sys_contract("ContractDeployer")
}
Expand Down
5 changes: 5 additions & 0 deletions core/lib/multivm/src/versions/shadow.rs
Original file line number Diff line number Diff line change
Expand Up @@ -165,6 +165,11 @@ impl DivergenceErrors {
let shadow_logs = UniqueStorageLogs::new(&shadow_result.logs.storage_logs);
self.check_match("logs.storage_logs", &main_logs, &shadow_logs);
self.check_match("refunds", &main_result.refunds, &shadow_result.refunds);
self.check_match(
"statistics.circuit_statistic",
&main_result.statistics.circuit_statistic,
&shadow_result.statistics.circuit_statistic,
);
self.check_match(
"gas_remaining",
&main_result.statistics.gas_remaining,
Expand Down
157 changes: 157 additions & 0 deletions core/lib/multivm/src/versions/vm_fast/circuits_tracer.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,157 @@
use circuit_sequencer_api_1_5_0::{geometry_config::get_geometry_config, toolset::GeometryConfig};
use vm2::{CycleStats, Opcode, OpcodeType, StateInterface, Tracer};
use zksync_vm_interface::CircuitStatistic;

use crate::vm_latest::tracers::circuits_capacity::*;

#[derive(Debug, Default, Clone, PartialEq)]
pub struct CircuitsTracer {
joonazan marked this conversation as resolved.
Show resolved Hide resolved
main_vm_cycles: u32,
ram_permutation_cycles: u32,
storage_application_cycles: u32,
storage_sorter_cycles: u32,
code_decommitter_cycles: u32,
code_decommitter_sorter_cycles: u32,
log_demuxer_cycles: u32,
events_sorter_cycles: u32,
keccak256_cycles: u32,
ecrecover_cycles: u32,
sha256_cycles: u32,
secp256k1_verify_cycles: u32,
transient_storage_checker_cycles: u32,
}

impl Tracer for CircuitsTracer {
fn after_instruction<OP: OpcodeType, S: StateInterface>(&mut self, _state: &mut S) {
self.main_vm_cycles += 1;

match OP::VALUE {
Opcode::Nop
| Opcode::Add
| Opcode::Sub
| Opcode::Mul
| Opcode::Div
| Opcode::Jump
| Opcode::Xor
| Opcode::And
| Opcode::Or
| Opcode::ShiftLeft
| Opcode::ShiftRight
| Opcode::RotateLeft
| Opcode::RotateRight
| Opcode::PointerAdd
| Opcode::PointerSub
| Opcode::PointerPack
| Opcode::PointerShrink => {
self.ram_permutation_cycles += RICH_ADDRESSING_OPCODE_RAM_CYCLES;
}
Opcode::This
| Opcode::Caller
| Opcode::CodeAddress
| Opcode::ContextMeta
| Opcode::ErgsLeft
| Opcode::SP
| Opcode::ContextU128
| Opcode::SetContextU128
| Opcode::AuxMutating0
| Opcode::IncrementTxNumber
| Opcode::Ret(_)
| Opcode::NearCall => {
self.ram_permutation_cycles += AVERAGE_OPCODE_RAM_CYCLES;
}
Opcode::StorageRead => {
self.ram_permutation_cycles += STORAGE_READ_RAM_CYCLES;
self.log_demuxer_cycles += STORAGE_READ_LOG_DEMUXER_CYCLES;
self.storage_sorter_cycles += STORAGE_READ_STORAGE_SORTER_CYCLES;
}
Opcode::TransientStorageRead => {
self.ram_permutation_cycles += TRANSIENT_STORAGE_READ_RAM_CYCLES;
self.log_demuxer_cycles += TRANSIENT_STORAGE_READ_LOG_DEMUXER_CYCLES;
self.transient_storage_checker_cycles +=
TRANSIENT_STORAGE_READ_TRANSIENT_STORAGE_CHECKER_CYCLES;
}
Opcode::StorageWrite => {
self.ram_permutation_cycles += STORAGE_WRITE_RAM_CYCLES;
self.log_demuxer_cycles += STORAGE_WRITE_LOG_DEMUXER_CYCLES;
self.storage_sorter_cycles += STORAGE_WRITE_STORAGE_SORTER_CYCLES;
}
Opcode::TransientStorageWrite => {
self.ram_permutation_cycles += TRANSIENT_STORAGE_WRITE_RAM_CYCLES;
self.log_demuxer_cycles += TRANSIENT_STORAGE_WRITE_LOG_DEMUXER_CYCLES;
self.transient_storage_checker_cycles +=
TRANSIENT_STORAGE_WRITE_TRANSIENT_STORAGE_CHECKER_CYCLES;
}
Opcode::L2ToL1Message | Opcode::Event => {
self.ram_permutation_cycles += EVENT_RAM_CYCLES;
self.log_demuxer_cycles += EVENT_LOG_DEMUXER_CYCLES;
self.events_sorter_cycles += EVENT_EVENTS_SORTER_CYCLES;
}
Opcode::PrecompileCall => {
self.ram_permutation_cycles += PRECOMPILE_RAM_CYCLES;
self.log_demuxer_cycles += PRECOMPILE_LOG_DEMUXER_CYCLES;
}
Opcode::Decommit => {
// Note, that for decommit the log demuxer circuit is not used.
self.ram_permutation_cycles += LOG_DECOMMIT_RAM_CYCLES;
self.code_decommitter_sorter_cycles += LOG_DECOMMIT_DECOMMITTER_SORTER_CYCLES;
}
Opcode::FarCall(_) => {
self.ram_permutation_cycles += FAR_CALL_RAM_CYCLES;
self.code_decommitter_sorter_cycles += FAR_CALL_CODE_DECOMMITTER_SORTER_CYCLES;
self.storage_sorter_cycles += FAR_CALL_STORAGE_SORTER_CYCLES;
self.log_demuxer_cycles += FAR_CALL_LOG_DEMUXER_CYCLES;
}
Opcode::AuxHeapWrite | Opcode::HeapWrite /* StaticMemoryWrite */ => {
self.ram_permutation_cycles += UMA_WRITE_RAM_CYCLES;
}
Opcode::AuxHeapRead | Opcode::HeapRead | Opcode::PointerRead /* StaticMemoryRead */ => {
self.ram_permutation_cycles += UMA_READ_RAM_CYCLES;
}
}
}

fn on_extra_prover_cycles(&mut self, stats: CycleStats) {
match stats {
CycleStats::Keccak256(cycles) => self.keccak256_cycles += cycles,
CycleStats::Sha256(cycles) => self.sha256_cycles += cycles,
CycleStats::EcRecover(cycles) => self.ecrecover_cycles += cycles,
CycleStats::Secp256k1Verify(cycles) => self.secp256k1_verify_cycles += cycles,
CycleStats::Decommit(cycles) => self.code_decommitter_cycles += cycles,
CycleStats::StorageRead => self.storage_application_cycles += 1,
CycleStats::StorageWrite => self.storage_application_cycles += 2,
}
}
}

impl CircuitsTracer {
pub(crate) fn circuit_statistic(&self) -> CircuitStatistic {
CircuitStatistic {
main_vm: self.main_vm_cycles as f32 / GEOMETRY_CONFIG.cycles_per_vm_snapshot as f32,
ram_permutation: self.ram_permutation_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_ram_permutation as f32,
storage_application: self.storage_application_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_storage_application as f32,
storage_sorter: self.storage_sorter_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_storage_sorter as f32,
code_decommitter: self.code_decommitter_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_code_decommitter as f32,
code_decommitter_sorter: self.code_decommitter_sorter_cycles as f32
/ GEOMETRY_CONFIG.cycles_code_decommitter_sorter as f32,
log_demuxer: self.log_demuxer_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_log_demuxer as f32,
events_sorter: self.events_sorter_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_events_or_l1_messages_sorter as f32,
keccak256: self.keccak256_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_keccak256_circuit as f32,
ecrecover: self.ecrecover_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_ecrecover_circuit as f32,
sha256: self.sha256_cycles as f32 / GEOMETRY_CONFIG.cycles_per_sha256_circuit as f32,
secp256k1_verify: self.secp256k1_verify_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_secp256r1_verify_circuit as f32,
transient_storage_checker: self.transient_storage_checker_cycles as f32
/ GEOMETRY_CONFIG.cycles_per_transient_storage_sorter as f32,
}
}
}

const GEOMETRY_CONFIG: GeometryConfig = get_geometry_config();
1 change: 1 addition & 0 deletions core/lib/multivm/src/versions/vm_fast/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ pub use self::vm::Vm;

mod bootloader_state;
mod bytecode;
mod circuits_tracer;
mod events;
mod glue;
mod hook;
Expand Down
10 changes: 5 additions & 5 deletions core/lib/multivm/src/versions/vm_fast/tests/code_oracle.rs
Original file line number Diff line number Diff line change
Expand Up @@ -207,11 +207,11 @@ fn refunds_in_code_oracle() {

let account = &mut vm.rich_accounts[0];
if decommit {
let (_, is_fresh) = vm
.vm
.inner
.world_diff
.decommit_opcode(&mut vm.vm.world, h256_to_u256(normal_zkevm_bytecode_hash));
let (_, is_fresh) = vm.vm.inner.world_diff.decommit_opcode(
&mut vm.vm.world,
&mut vm.vm.tracer,
h256_to_u256(normal_zkevm_bytecode_hash),
);
assert!(is_fresh);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ use crate::{
VmExecutionMode, VmExecutionResultAndLogs, VmInterface, VmInterfaceExt,
VmInterfaceHistoryEnabled, VmRevertReason,
},
vm_fast::Vm,
vm_fast::{circuits_tracer::CircuitsTracer, vm::World, Vm},
};

#[derive(Debug, Clone)]
Expand Down Expand Up @@ -185,14 +185,14 @@ impl TransactionTestInfo {

// TODO this doesn't include all the state of ModifiedWorld
#[derive(Debug, PartialEq)]
struct VmStateDump {
state: vm2::State,
struct VmStateDump<S: PartialEq> {
joonazan marked this conversation as resolved.
Show resolved Hide resolved
state: vm2::State<CircuitsTracer, World<S, CircuitsTracer>>,
storage_writes: Vec<((H160, U256), U256)>,
events: Box<[vm2::Event]>,
}

impl<S: ReadStorage> Vm<S> {
fn dump_state(&self) -> VmStateDump {
impl<S: ReadStorage + Clone + PartialEq> Vm<S> {
fn dump_state(&self) -> VmStateDump<S> {
VmStateDump {
state: self.inner.state.clone(),
storage_writes: self
Expand Down
5 changes: 4 additions & 1 deletion core/lib/multivm/src/versions/vm_fast/tests/utils.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,10 @@ use crate::interface::storage::ReadStorage;
pub(crate) static BASE_SYSTEM_CONTRACTS: Lazy<BaseSystemContracts> =
Lazy::new(BaseSystemContracts::load_from_disk);

pub(crate) fn verify_required_memory(state: &State, required_values: Vec<(U256, HeapId, u32)>) {
pub(crate) fn verify_required_memory<T, W>(
state: &State<T, W>,
required_values: Vec<(U256, HeapId, u32)>,
) {
for (required_value, memory_page, cell) in required_values {
let current_value = state.heaps[memory_page].read_u256(cell * 32);
assert_eq!(current_value, required_value);
Expand Down
Loading
Loading