use alloc::boxed::Box;
use core::{
marker::PhantomData,
ops::{Deref, DerefMut},
};
use crate::{
Control, ExitException, Machine, Opcode,
eval::*,
runtime::{GasState, RuntimeBackend, RuntimeConfig, RuntimeEnvironment, RuntimeState},
trap::CallCreateTrap,
};
pub trait Etable<H> {
type State;
type Trap;
fn eval(
&self,
machine: &mut Machine<Self::State>,
handle: &mut H,
position: usize,
) -> Control<Self::Trap>;
}
pub struct Chained<E1, E2>(pub E1, pub E2);
impl<S, H, Tr, E1, E2> Etable<H> for Chained<E1, E2>
where
E1: Etable<H, State = S, Trap = Tr>,
E2: Etable<H, State = S, Trap = Tr>,
{
type State = S;
type Trap = Tr;
fn eval(&self, machine: &mut Machine<S>, handle: &mut H, position: usize) -> Control<Tr> {
let ret1 = self.0.eval(machine, handle, position);
if let Control::NoAction = ret1 {
self.1.eval(machine, handle, position)
} else {
ret1
}
}
}
pub struct Tracing<EPre, E, EPost>(pub EPre, pub E, pub EPost);
impl<S, H, Tr, EPre, E, EPost> Etable<H> for Tracing<EPre, E, EPost>
where
EPre: Etable<H, State = S, Trap = Tr>,
E: Etable<H, State = S, Trap = Tr>,
EPost: Etable<H, State = S, Trap = Tr>,
{
type State = S;
type Trap = Tr;
fn eval(&self, machine: &mut Machine<S>, handle: &mut H, position: usize) -> Control<Tr> {
let _ = self.0.eval(machine, handle, position);
let ret = self.1.eval(machine, handle, position);
let _ = self.2.eval(machine, handle, position);
ret
}
}
pub type Efn<S, H, Tr> = fn(&mut Machine<S>, &mut H, usize) -> Control<Tr>;
pub struct Single<S, H, Tr, F = Efn<S, H, Tr>>(F, PhantomData<(S, H, Tr)>);
unsafe impl<S, H, Tr, F: Send> Send for Single<S, H, Tr, F> {}
unsafe impl<S, H, Tr, F: Sync> Sync for Single<S, H, Tr, F> {}
impl<S, H, Tr, F> Deref for Single<S, H, Tr, F> {
type Target = F;
fn deref(&self) -> &F {
&self.0
}
}
impl<S, H, Tr, F> DerefMut for Single<S, H, Tr, F> {
fn deref_mut(&mut self) -> &mut F {
&mut self.0
}
}
impl<S, H, Tr, F> Single<S, H, Tr, F> {
pub fn new(single: F) -> Self {
Self(single, PhantomData)
}
}
impl<S, H, Tr, F> Etable<H> for Single<S, H, Tr, F>
where
F: Fn(&mut Machine<S>, &mut H, usize) -> Control<Tr>,
{
type State = S;
type Trap = Tr;
fn eval(&self, machine: &mut Machine<S>, handle: &mut H, position: usize) -> Control<Tr> {
self.0(machine, handle, position)
}
}
pub enum MultiEfn<S, H, Tr, F = Efn<S, H, Tr>> {
Leaf(F),
Node(Box<MultiEtable<S, H, Tr, F>>),
}
unsafe impl<S, H, Tr, F: Send> Send for MultiEfn<S, H, Tr, F> {}
unsafe impl<S, H, Tr, F: Sync> Sync for MultiEfn<S, H, Tr, F> {}
pub struct MultiEtable<S, H, Tr, F = Efn<S, H, Tr>>(
[MultiEfn<S, H, Tr, F>; 256],
PhantomData<(S, H, Tr)>,
);
unsafe impl<S, H, Tr, F: Send> Send for MultiEtable<S, H, Tr, F> {}
unsafe impl<S, H, Tr, F: Sync> Sync for MultiEtable<S, H, Tr, F> {}
impl<S, H, Tr, F> Deref for MultiEtable<S, H, Tr, F> {
type Target = [MultiEfn<S, H, Tr, F>; 256];
fn deref(&self) -> &[MultiEfn<S, H, Tr, F>; 256] {
&self.0
}
}
impl<S, H, Tr, F> DerefMut for MultiEtable<S, H, Tr, F> {
fn deref_mut(&mut self) -> &mut [MultiEfn<S, H, Tr, F>; 256] {
&mut self.0
}
}
impl<S, H, Tr, F> From<DispatchEtable<S, H, Tr, F>> for MultiEtable<S, H, Tr, F> {
fn from(etable: DispatchEtable<S, H, Tr, F>) -> Self {
Self(etable.0.map(|v| MultiEfn::Leaf(v)), PhantomData)
}
}
impl<S, H, Tr, F> Etable<H> for MultiEtable<S, H, Tr, F>
where
F: Fn(&mut Machine<S>, &mut H, usize) -> Control<Tr>,
{
type State = S;
type Trap = Tr;
fn eval(&self, machine: &mut Machine<S>, handle: &mut H, position: usize) -> Control<Tr> {
let opcode = Opcode(machine.code()[position]);
match &self[opcode.as_usize()] {
MultiEfn::Leaf(f) => f(machine, handle, position),
MultiEfn::Node(n) => {
let nextpos = position + 1;
if nextpos >= machine.code.len() {
return Control::Exit(ExitException::PCUnderflow.into());
}
match n.eval(machine, handle, nextpos) {
Control::Continue(c) => Control::Continue(c + 1),
ctrl => ctrl,
}
}
}
}
}
pub struct DispatchEtable<S, H, Tr, F = Efn<S, H, Tr>>([F; 256], PhantomData<(S, H, Tr)>);
unsafe impl<S, H, Tr, F: Send> Send for DispatchEtable<S, H, Tr, F> {}
unsafe impl<S, H, Tr, F: Sync> Sync for DispatchEtable<S, H, Tr, F> {}
impl<S, H, Tr, F> Deref for DispatchEtable<S, H, Tr, F> {
type Target = [F; 256];
fn deref(&self) -> &[F; 256] {
&self.0
}
}
impl<S, H, Tr, F> DerefMut for DispatchEtable<S, H, Tr, F> {
fn deref_mut(&mut self) -> &mut [F; 256] {
&mut self.0
}
}
impl<S, H, Tr, F> From<Single<S, H, Tr, F>> for DispatchEtable<S, H, Tr, F>
where
F: Copy,
{
fn from(single: Single<S, H, Tr, F>) -> Self {
Self([single.0; 256], PhantomData)
}
}
impl<S, H, Tr, F> Etable<H> for DispatchEtable<S, H, Tr, F>
where
F: Fn(&mut Machine<S>, &mut H, usize) -> Control<Tr>,
{
type State = S;
type Trap = Tr;
fn eval(&self, machine: &mut Machine<S>, handle: &mut H, position: usize) -> Control<Tr> {
let opcode = Opcode(machine.code()[position]);
self[opcode.as_usize()](machine, handle, position)
}
}
impl<S, H, Tr, F> DispatchEtable<S, H, Tr, F>
where
F: Fn(&mut Machine<S>, &mut H, usize) -> Control<Tr>,
{
pub fn wrap<FW, FR>(self, wrapper: FW) -> DispatchEtable<S, H, Tr, FR>
where
FW: Fn(F, Opcode) -> FR,
FR: Fn(&mut Machine<S>, &mut H, usize) -> Control<Tr>,
{
let mut current_opcode = Opcode(0);
DispatchEtable(
self.0.map(|f| {
let fr = wrapper(f, current_opcode);
if current_opcode != Opcode(255) {
current_opcode.0 += 1;
}
fr
}),
PhantomData,
)
}
}
impl<S, H, Tr> DispatchEtable<S, H, Tr> {
#[must_use]
pub const fn none() -> Self {
Self([eval_unknown as _; 256], PhantomData)
}
#[must_use]
pub const fn pass() -> Self {
Self([eval_pass as _; 256], PhantomData)
}
#[must_use]
pub const fn core() -> Self {
let mut table = [eval_unknown as _; 256];
table[Opcode::STOP.as_usize()] = eval_stop as _;
table[Opcode::ADD.as_usize()] = eval_add as _;
table[Opcode::MUL.as_usize()] = eval_mul as _;
table[Opcode::SUB.as_usize()] = eval_sub as _;
table[Opcode::DIV.as_usize()] = eval_div as _;
table[Opcode::SDIV.as_usize()] = eval_sdiv as _;
table[Opcode::MOD.as_usize()] = eval_mod as _;
table[Opcode::SMOD.as_usize()] = eval_smod as _;
table[Opcode::ADDMOD.as_usize()] = eval_addmod as _;
table[Opcode::MULMOD.as_usize()] = eval_mulmod as _;
table[Opcode::EXP.as_usize()] = eval_exp as _;
table[Opcode::SIGNEXTEND.as_usize()] = eval_signextend as _;
table[Opcode::LT.as_usize()] = eval_lt as _;
table[Opcode::GT.as_usize()] = eval_gt as _;
table[Opcode::SLT.as_usize()] = eval_slt as _;
table[Opcode::SGT.as_usize()] = eval_sgt as _;
table[Opcode::EQ.as_usize()] = eval_eq as _;
table[Opcode::ISZERO.as_usize()] = eval_iszero as _;
table[Opcode::AND.as_usize()] = eval_and as _;
table[Opcode::OR.as_usize()] = eval_or as _;
table[Opcode::XOR.as_usize()] = eval_xor as _;
table[Opcode::NOT.as_usize()] = eval_not as _;
table[Opcode::BYTE.as_usize()] = eval_byte as _;
table[Opcode::SHL.as_usize()] = eval_shl as _;
table[Opcode::SHR.as_usize()] = eval_shr as _;
table[Opcode::SAR.as_usize()] = eval_sar as _;
table[Opcode::CALLDATALOAD.as_usize()] = eval_calldataload as _;
table[Opcode::CALLDATASIZE.as_usize()] = eval_calldatasize as _;
table[Opcode::CALLDATACOPY.as_usize()] = eval_calldatacopy as _;
table[Opcode::CODESIZE.as_usize()] = eval_codesize as _;
table[Opcode::CODECOPY.as_usize()] = eval_codecopy as _;
table[Opcode::POP.as_usize()] = eval_pop as _;
table[Opcode::MLOAD.as_usize()] = eval_mload as _;
table[Opcode::MSTORE.as_usize()] = eval_mstore as _;
table[Opcode::MSTORE8.as_usize()] = eval_mstore8 as _;
table[Opcode::JUMP.as_usize()] = eval_jump as _;
table[Opcode::JUMPI.as_usize()] = eval_jumpi as _;
table[Opcode::PC.as_usize()] = eval_pc as _;
table[Opcode::MSIZE.as_usize()] = eval_msize as _;
table[Opcode::JUMPDEST.as_usize()] = eval_jumpdest as _;
table[Opcode::MCOPY.as_usize()] = eval_mcopy as _;
table[Opcode::PUSH0.as_usize()] = eval_push0 as _;
table[Opcode::PUSH1.as_usize()] = eval_push1 as _;
table[Opcode::PUSH2.as_usize()] = eval_push2 as _;
table[Opcode::PUSH3.as_usize()] = eval_push3 as _;
table[Opcode::PUSH4.as_usize()] = eval_push4 as _;
table[Opcode::PUSH5.as_usize()] = eval_push5 as _;
table[Opcode::PUSH6.as_usize()] = eval_push6 as _;
table[Opcode::PUSH7.as_usize()] = eval_push7 as _;
table[Opcode::PUSH8.as_usize()] = eval_push8 as _;
table[Opcode::PUSH9.as_usize()] = eval_push9 as _;
table[Opcode::PUSH10.as_usize()] = eval_push10 as _;
table[Opcode::PUSH11.as_usize()] = eval_push11 as _;
table[Opcode::PUSH12.as_usize()] = eval_push12 as _;
table[Opcode::PUSH13.as_usize()] = eval_push13 as _;
table[Opcode::PUSH14.as_usize()] = eval_push14 as _;
table[Opcode::PUSH15.as_usize()] = eval_push15 as _;
table[Opcode::PUSH16.as_usize()] = eval_push16 as _;
table[Opcode::PUSH17.as_usize()] = eval_push17 as _;
table[Opcode::PUSH18.as_usize()] = eval_push18 as _;
table[Opcode::PUSH19.as_usize()] = eval_push19 as _;
table[Opcode::PUSH20.as_usize()] = eval_push20 as _;
table[Opcode::PUSH21.as_usize()] = eval_push21 as _;
table[Opcode::PUSH22.as_usize()] = eval_push22 as _;
table[Opcode::PUSH23.as_usize()] = eval_push23 as _;
table[Opcode::PUSH24.as_usize()] = eval_push24 as _;
table[Opcode::PUSH25.as_usize()] = eval_push25 as _;
table[Opcode::PUSH26.as_usize()] = eval_push26 as _;
table[Opcode::PUSH27.as_usize()] = eval_push27 as _;
table[Opcode::PUSH28.as_usize()] = eval_push28 as _;
table[Opcode::PUSH29.as_usize()] = eval_push29 as _;
table[Opcode::PUSH30.as_usize()] = eval_push30 as _;
table[Opcode::PUSH31.as_usize()] = eval_push31 as _;
table[Opcode::PUSH32.as_usize()] = eval_push32 as _;
table[Opcode::DUP1.as_usize()] = eval_dup1 as _;
table[Opcode::DUP2.as_usize()] = eval_dup2 as _;
table[Opcode::DUP3.as_usize()] = eval_dup3 as _;
table[Opcode::DUP4.as_usize()] = eval_dup4 as _;
table[Opcode::DUP5.as_usize()] = eval_dup5 as _;
table[Opcode::DUP6.as_usize()] = eval_dup6 as _;
table[Opcode::DUP7.as_usize()] = eval_dup7 as _;
table[Opcode::DUP8.as_usize()] = eval_dup8 as _;
table[Opcode::DUP9.as_usize()] = eval_dup9 as _;
table[Opcode::DUP10.as_usize()] = eval_dup10 as _;
table[Opcode::DUP11.as_usize()] = eval_dup11 as _;
table[Opcode::DUP12.as_usize()] = eval_dup12 as _;
table[Opcode::DUP13.as_usize()] = eval_dup13 as _;
table[Opcode::DUP14.as_usize()] = eval_dup14 as _;
table[Opcode::DUP15.as_usize()] = eval_dup15 as _;
table[Opcode::DUP16.as_usize()] = eval_dup16 as _;
table[Opcode::SWAP1.as_usize()] = eval_swap1 as _;
table[Opcode::SWAP2.as_usize()] = eval_swap2 as _;
table[Opcode::SWAP3.as_usize()] = eval_swap3 as _;
table[Opcode::SWAP4.as_usize()] = eval_swap4 as _;
table[Opcode::SWAP5.as_usize()] = eval_swap5 as _;
table[Opcode::SWAP6.as_usize()] = eval_swap6 as _;
table[Opcode::SWAP7.as_usize()] = eval_swap7 as _;
table[Opcode::SWAP8.as_usize()] = eval_swap8 as _;
table[Opcode::SWAP9.as_usize()] = eval_swap9 as _;
table[Opcode::SWAP10.as_usize()] = eval_swap10 as _;
table[Opcode::SWAP11.as_usize()] = eval_swap11 as _;
table[Opcode::SWAP12.as_usize()] = eval_swap12 as _;
table[Opcode::SWAP13.as_usize()] = eval_swap13 as _;
table[Opcode::SWAP14.as_usize()] = eval_swap14 as _;
table[Opcode::SWAP15.as_usize()] = eval_swap15 as _;
table[Opcode::SWAP16.as_usize()] = eval_swap16 as _;
table[Opcode::RETURN.as_usize()] = eval_return as _;
table[Opcode::REVERT.as_usize()] = eval_revert as _;
table[Opcode::INVALID.as_usize()] = eval_invalid as _;
Self(table, PhantomData)
}
}
impl<S, H: RuntimeEnvironment + RuntimeBackend, Tr> DispatchEtable<S, H, Tr>
where
S: AsRef<RuntimeState> + AsMut<RuntimeState> + AsRef<RuntimeConfig> + GasState,
Tr: From<CallCreateTrap>,
{
#[must_use]
pub const fn runtime() -> Self {
let mut table = Self::core();
table.0[Opcode::SHA3.as_usize()] = eval_sha3 as _;
table.0[Opcode::ADDRESS.as_usize()] = eval_address as _;
table.0[Opcode::BALANCE.as_usize()] = eval_balance as _;
table.0[Opcode::ORIGIN.as_usize()] = eval_origin as _;
table.0[Opcode::CALLER.as_usize()] = eval_caller as _;
table.0[Opcode::CALLVALUE.as_usize()] = eval_callvalue as _;
table.0[Opcode::GASPRICE.as_usize()] = eval_gasprice as _;
table.0[Opcode::EXTCODESIZE.as_usize()] = eval_extcodesize as _;
table.0[Opcode::EXTCODECOPY.as_usize()] = eval_extcodecopy as _;
table.0[Opcode::RETURNDATASIZE.as_usize()] = eval_returndatasize as _;
table.0[Opcode::RETURNDATACOPY.as_usize()] = eval_returndatacopy as _;
table.0[Opcode::EXTCODEHASH.as_usize()] = eval_extcodehash as _;
table.0[Opcode::BLOCKHASH.as_usize()] = eval_blockhash as _;
table.0[Opcode::COINBASE.as_usize()] = eval_coinbase as _;
table.0[Opcode::TIMESTAMP.as_usize()] = eval_timestamp as _;
table.0[Opcode::NUMBER.as_usize()] = eval_number as _;
table.0[Opcode::DIFFICULTY.as_usize()] = eval_difficulty as _;
table.0[Opcode::GASLIMIT.as_usize()] = eval_gaslimit as _;
table.0[Opcode::CHAINID.as_usize()] = eval_chainid as _;
table.0[Opcode::SELFBALANCE.as_usize()] = eval_selfbalance as _;
table.0[Opcode::BASEFEE.as_usize()] = eval_basefee as _;
table.0[Opcode::BLOBHASH.as_usize()] = eval_blobhash as _;
table.0[Opcode::BLOBBASEFEE.as_usize()] = eval_blobbasefee as _;
table.0[Opcode::SLOAD.as_usize()] = eval_sload as _;
table.0[Opcode::SSTORE.as_usize()] = eval_sstore as _;
table.0[Opcode::GAS.as_usize()] = eval_gas as _;
table.0[Opcode::TLOAD.as_usize()] = eval_tload as _;
table.0[Opcode::TSTORE.as_usize()] = eval_tstore as _;
table.0[Opcode::LOG0.as_usize()] = eval_log0 as _;
table.0[Opcode::LOG1.as_usize()] = eval_log1 as _;
table.0[Opcode::LOG2.as_usize()] = eval_log2 as _;
table.0[Opcode::LOG3.as_usize()] = eval_log3 as _;
table.0[Opcode::LOG4.as_usize()] = eval_log4 as _;
table.0[Opcode::CREATE.as_usize()] = eval_call_create_trap as _;
table.0[Opcode::CALL.as_usize()] = eval_call_create_trap as _;
table.0[Opcode::CALLCODE.as_usize()] = eval_call_create_trap as _;
table.0[Opcode::DELEGATECALL.as_usize()] = eval_call_create_trap as _;
table.0[Opcode::CREATE2.as_usize()] = eval_call_create_trap as _;
table.0[Opcode::STATICCALL.as_usize()] = eval_call_create_trap as _;
table.0[Opcode::SUICIDE.as_usize()] = eval_suicide as _;
table
}
}