#[macro_use]
mod macros;
mod system;
use crate::{CallScheme, ExitReason, Handler, Opcode, Runtime};
use alloc::vec::Vec;
use core::cmp::min;
use primitive_types::{H160, H256, U256};
pub enum Control<H: Handler> {
Continue,
CallInterrupt(H::CallInterrupt),
CreateInterrupt(H::CreateInterrupt),
Exit(ExitReason),
}
fn handle_other<H: Handler>(state: &mut Runtime, opcode: Opcode, handler: &mut H) -> Control<H> {
match handler.other(opcode, &mut state.machine) {
Ok(()) => Control::Continue,
Err(e) => Control::Exit(e.into()),
}
}
pub fn eval<H: Handler>(state: &mut Runtime, opcode: Opcode, handler: &mut H) -> Control<H> {
match opcode {
Opcode::SHA3 => system::sha3(state),
Opcode::ADDRESS => system::address(state),
Opcode::BALANCE => system::balance(state, handler),
Opcode::SELFBALANCE => system::selfbalance(state, handler),
Opcode::ORIGIN => system::origin(state, handler),
Opcode::CALLER => system::caller(state),
Opcode::CALLVALUE => system::callvalue(state),
Opcode::GASPRICE => system::gasprice(state, handler),
Opcode::EXTCODESIZE => system::extcodesize(state, handler),
Opcode::EXTCODEHASH => system::extcodehash(state, handler),
Opcode::EXTCODECOPY => system::extcodecopy(state, handler),
Opcode::RETURNDATASIZE => system::returndatasize(state),
Opcode::RETURNDATACOPY => system::returndatacopy(state),
Opcode::BLOCKHASH => system::blockhash(state, handler),
Opcode::COINBASE => system::coinbase(state, handler),
Opcode::TIMESTAMP => system::timestamp(state, handler),
Opcode::NUMBER => system::number(state, handler),
Opcode::DIFFICULTY => system::prevrandao(state, handler),
Opcode::GASLIMIT => system::gaslimit(state, handler),
Opcode::SLOAD => system::sload(state, handler),
Opcode::SSTORE => system::sstore(state, handler),
Opcode::GAS => system::gas(state, handler),
Opcode::TLOAD => system::tload(state, handler),
Opcode::TSTORE => system::tstore(state, handler),
Opcode::LOG0 => system::log(state, 0, handler),
Opcode::LOG1 => system::log(state, 1, handler),
Opcode::LOG2 => system::log(state, 2, handler),
Opcode::LOG3 => system::log(state, 3, handler),
Opcode::LOG4 => system::log(state, 4, handler),
Opcode::SUICIDE => system::suicide(state, handler),
Opcode::CREATE => system::create(state, false, handler),
Opcode::CREATE2 => system::create(state, true, handler),
Opcode::CALL => system::call(state, CallScheme::Call, handler),
Opcode::CALLCODE => system::call(state, CallScheme::CallCode, handler),
Opcode::DELEGATECALL => system::call(state, CallScheme::DelegateCall, handler),
Opcode::STATICCALL => system::call(state, CallScheme::StaticCall, handler),
Opcode::CHAINID => system::chainid(state, handler),
Opcode::BASEFEE => system::base_fee(state, handler),
_ => handle_other(state, opcode, handler),
}
}
pub fn finish_create(
runtime: &mut Runtime,
reason: ExitReason,
address: Option<H160>,
return_data: Vec<u8>,
) -> Result<(), ExitReason> {
runtime.return_data_buffer = return_data;
let create_address: H256 = address.map(|a| a.into()).unwrap_or_default();
match reason {
ExitReason::Succeed(_) => {
runtime.machine.stack_mut().push(create_address)?;
Ok(())
}
ExitReason::Revert(_) => {
runtime.machine.stack_mut().push(H256::default())?;
Ok(())
}
ExitReason::Error(_) => {
runtime.machine.stack_mut().push(H256::default())?;
Ok(())
}
ExitReason::Fatal(e) => {
runtime.machine.stack_mut().push(H256::default())?;
Err(e.into())
}
}
}
pub fn finish_call(
runtime: &mut Runtime,
out_len: U256,
out_offset: U256,
reason: ExitReason,
return_data: Vec<u8>,
) -> Result<(), ExitReason> {
runtime.return_data_buffer = return_data;
let target_len = min(out_len, U256::from(runtime.return_data_buffer.len()));
match reason {
ExitReason::Succeed(_) => {
match runtime.machine.memory_mut().copy_large(
out_offset,
U256::zero(),
target_len,
&runtime.return_data_buffer[..],
) {
Ok(()) => {
let value = H256::from(U256::one().to_big_endian());
runtime.machine.stack_mut().push(value)?;
Ok(())
}
Err(_) => {
runtime.machine.stack_mut().push(H256::default())?;
Ok(())
}
}
}
ExitReason::Revert(_) => {
runtime.machine.stack_mut().push(H256::default())?;
let _ = runtime.machine.memory_mut().copy_large(
out_offset,
U256::zero(),
target_len,
&runtime.return_data_buffer[..],
);
Ok(())
}
ExitReason::Error(_) => {
runtime.machine.stack_mut().push(H256::default())?;
Ok(())
}
ExitReason::Fatal(e) => {
runtime.machine.stack_mut().push(H256::default())?;
Err(e.into())
}
}
}