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: decommit opcode #44

Merged
merged 12 commits into from
Jun 18, 2024
Merged
7 changes: 6 additions & 1 deletion src/decode.rs
Original file line number Diff line number Diff line change
Expand Up @@ -283,7 +283,12 @@ pub(crate) fn decode(raw: u64, is_bootloader: bool) -> Instruction {
out.try_into().unwrap(),
arguments,
),
x => unimplemented_instruction(zkevm_opcode_defs::Opcode::Log(x)),
zkevm_opcode_defs::LogOpcode::Decommit => Instruction::from_decommit(
src1.try_into().unwrap(),
src2,
out.try_into().unwrap(),
arguments,
),
},
zkevm_opcode_defs::Opcode::UMA(x) => {
let increment = parsed.variant.flags[UMA_INCREMENT_FLAG_IDX];
Expand Down
16 changes: 14 additions & 2 deletions src/decommit.rs
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,17 @@ impl WorldDiff {
Some((UnpaidDecommit { cost, code_key }, is_evm))
}

pub(crate) fn decommit_opcode(
&mut self,
world: &mut dyn World,
code_hash: U256,
) -> Option<Program> {
montekki marked this conversation as resolved.
Show resolved Hide resolved
let mut code_info_bytes = [0; 32];
code_hash.to_big_endian(&mut code_info_bytes);
self.decommitted_hashes.insert(code_hash, ());
Some(world.decommit(code_hash))
}

pub(crate) fn pay_for_decommit(
&mut self,
world: &mut dyn World,
Expand All @@ -93,9 +104,10 @@ impl WorldDiff {
}
}

#[derive(Debug)]
pub(crate) struct UnpaidDecommit {
cost: u32,
code_key: U256,
pub cost: u32,
joonazan marked this conversation as resolved.
Show resolved Hide resolved
pub code_key: U256,
}

/// May be used to load code when the VM first starts up.
Expand Down
1 change: 1 addition & 0 deletions src/fat_pointer.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
use crate::heap::HeapId;
use u256::U256;

#[derive(Debug)]
#[repr(C)]
pub struct FatPointer {
pub offset: u32,
Expand Down
5 changes: 5 additions & 0 deletions src/heap.rs
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,11 @@ impl HeapInterface for Heap {
}
result
}
fn memset(&mut self, src: &[U256]) {
for (i, word) in src.iter().enumerate() {
self.write_u256((i * 32) as u32, *word);
}
}
}

#[derive(Debug, Clone)]
Expand Down
83 changes: 83 additions & 0 deletions src/instruction_handlers/decommit.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
use u256::U256;
use zkevm_opcode_defs::{
BlobSha256Format, ContractCodeSha256Format, VersionedHashHeader, VersionedHashLen32,
};

use crate::{
addressing_modes::{Arguments, Destination, Register1, Register2, Source},
fat_pointer::FatPointer,
instruction::InstructionResult,
Instruction, VirtualMachine, World,
};

use super::{common::instruction_boilerplate_with_panic, HeapInterface};

fn decommit(
vm: &mut VirtualMachine,
instruction: *const Instruction,
world: &mut dyn World,
) -> InstructionResult {
instruction_boilerplate_with_panic(
vm,
instruction,
world,
|vm, args, world, continue_normally| {
let code_hash = Register1::get(args, &mut vm.state);
let extra_cost = Register2::get(args, &mut vm.state).low_u32();

let mut buffer = [0u8; 32];
code_hash.to_big_endian(&mut buffer);

let preimage_len_in_bytes =
zkevm_opcode_defs::system_params::NEW_KERNEL_FRAME_MEMORY_STIPEND;
let mut _decommit_header = VersionedHashHeader::default();
joonazan marked this conversation as resolved.
Show resolved Hide resolved

if vm.state.use_gas(extra_cost).is_err() {
montekki marked this conversation as resolved.
Show resolved Hide resolved
Register1::set(args, &mut vm.state, U256::zero());
return continue_normally;
}

if !ContractCodeSha256Format::is_valid(&buffer) && !BlobSha256Format::is_valid(&buffer)
{
Register1::set(args, &mut vm.state, U256::zero());
return continue_normally;
};

let Some(program) = vm.world_diff.decommit_opcode(world, code_hash) else {
Register1::set(args, &mut vm.state, U256::zero());
return continue_normally;
montekki marked this conversation as resolved.
Show resolved Hide resolved
};

let heap = vm.state.heaps.allocate();
montekki marked this conversation as resolved.
Show resolved Hide resolved
let decommited_memory = program.code_page().as_ref();
vm.state.heaps[heap].memset(decommited_memory);

let value = FatPointer {
offset: 0,
memory_page: heap,
start: 0,
length: preimage_len_in_bytes,
};
let value = value.into_u256();
Register1::set_fat_ptr(args, &mut vm.state, value);

continue_normally
},
)
}
impl Instruction {
pub fn from_decommit(
abi: Register1,
burn: Register2,
out: Register1,
arguments: Arguments,
) -> Self {
Self {
arguments: arguments
.write_source(&abi)
.write_source(&burn)
.write_destination(&out),
handler: decommit,
}
}
}
1 change: 1 addition & 0 deletions src/instruction_handlers/heap_access.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ pub trait HeapInterface {
fn read_u256_partially(&self, range: Range<u32>) -> U256;
fn write_u256(&mut self, start_address: u32, value: U256);
fn read_range_big_endian(&self, range: Range<u32>) -> Vec<u8>;
fn memset(&mut self, memory: &[U256]);
}

pub trait HeapFromState {
Expand Down
1 change: 1 addition & 0 deletions src/instruction_handlers/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ pub(crate) use ret::{free_panic, PANIC};
mod binop;
mod common;
mod context;
mod decommit;
mod event;
mod far_call;
mod heap_access;
Expand Down
13 changes: 12 additions & 1 deletion src/single_instruction_test/heap.rs
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,12 @@ impl HeapInterface for Heap {
// This is wrong, but this method is only used to get the final return value.
vec![]
}

fn memset(&mut self, src: &[U256]) {
for (i, word) in src.iter().enumerate() {
self.write_u256((i * 32) as u32, *word);
montekki marked this conversation as resolved.
Show resolved Hide resolved
}
}
}

impl<'a> Arbitrary<'a> for Heap {
Expand All @@ -47,6 +53,7 @@ impl<'a> Arbitrary<'a> for Heap {

#[derive(Debug, Clone, Arbitrary)]
joonazan marked this conversation as resolved.
Show resolved Hide resolved
pub struct Heaps {
heap_id: HeapId,
pub(crate) read: MockRead<HeapId, Heap>,
}

Expand All @@ -60,10 +67,14 @@ impl Heaps {
}

pub(crate) fn allocate(&mut self) -> HeapId {
HeapId(3)
self.heap_id
}

pub(crate) fn deallocate(&mut self, _: HeapId) {}

pub fn set_heap_id(&mut self, heap_id: HeapId) {
montekki marked this conversation as resolved.
Show resolved Hide resolved
self.heap_id = heap_id
}
}

impl Index<HeapId> for Heaps {
Expand Down
11 changes: 4 additions & 7 deletions src/single_instruction_test/vm.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use super::stack::StackPool;
use super::{heap::Heaps, stack::StackPool};
use crate::{
callframe::Callframe, fat_pointer::FatPointer, instruction::InstructionResult,
instruction_handlers::free_panic, HeapId, Instruction, Settings, State, VirtualMachine, World,
Expand Down Expand Up @@ -40,11 +40,6 @@ impl VirtualMachine {
}

pub fn instruction_is_not_precompile_call(&self) -> bool {
// TODO PLA-934 implement Decommit
if self.current_opcode() == 1093 {
return false;
}

// TODO PLA-972 implement StaticMemoryRead/Write
if (1096..=1103).contains(&self.current_opcode()) {
return false;
Expand Down Expand Up @@ -81,6 +76,8 @@ impl<'a> Arbitrary<'a> for VirtualMachine {
register_pointer_flags |= (is_pointer as u16) << i;
}

let mut heaps: Heaps = u.arbitrary()?;
heaps.set_heap_id(current_frame.heap);
Ok(Self {
state: State {
registers,
Expand All @@ -90,7 +87,7 @@ impl<'a> Arbitrary<'a> for VirtualMachine {
// Exiting the final frame is different in vm2 on purpose,
// so always generate two frames to avoid that.
previous_frames: vec![(0, Callframe::dummy())],
heaps: u.arbitrary()?,
heaps,
transaction_number: u.arbitrary()?,
joonazan marked this conversation as resolved.
Show resolved Hide resolved
context_u128: u.arbitrary()?,
},
Expand Down
Loading