aurora-evm 2.2.1

Aurora Ethereum Virtual Machine implementation written in pure Rust
Documentation
#[macro_use]
mod macros;
mod system;

use crate::core::utils::{U256_ONE, U256_ZERO};
use crate::prelude::*;
use crate::{CallScheme, ExitReason, Handler, Opcode, Runtime};
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::PREVRANDAO => 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::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::SELFDESTRUCT => system::selfdestruct(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),
        Opcode::BLOBBASEFEE => system::blob_base_fee(state, handler),
        Opcode::BLOBHASH => system::blob_hash(state, handler),
        Opcode::TLOAD => system::tload(state, handler),
        Opcode::TSTORE => system::tstore(state, handler),
        Opcode::MCOPY => system::mcopy(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(Into::into).unwrap_or_default();

    match reason {
        ExitReason::Succeed(_) => {
            runtime
                .machine
                .stack_mut()
                .push(U256::from_big_endian(&create_address[..]))?;
            Ok(())
        }
        ExitReason::Revert(_) => {
            runtime.machine.stack_mut().push(U256_ZERO)?;
            Ok(())
        }
        ExitReason::Error(_) => {
            runtime.machine.stack_mut().push(U256_ZERO)?;
            Ok(())
        }
        ExitReason::Fatal(e) => {
            runtime.machine.stack_mut().push(U256_ZERO)?;
            Err(e.into())
        }
    }
}

pub fn finish_call(
    runtime: &mut Runtime,
    out_len: usize,
    out_offset: usize,
    reason: ExitReason,
    return_data: Vec<u8>,
) -> Result<(), ExitReason> {
    runtime.return_data_buffer = return_data;
    let target_len = min(out_len, runtime.return_data_buffer.len());

    match reason {
        ExitReason::Succeed(_) => {
            let value_to_push = if runtime
                .machine
                .memory_mut()
                .copy_data(
                    out_offset,
                    U256_ZERO,
                    target_len,
                    &runtime.return_data_buffer[..],
                )
                .is_ok()
            {
                U256_ONE
            } else {
                U256_ZERO
            };
            runtime
                .machine
                .stack_mut()
                .push(value_to_push)
                .map_err(Into::into)
        }
        ExitReason::Revert(_) => {
            runtime.machine.stack_mut().push(U256_ZERO)?;

            let _ = runtime.machine.memory_mut().copy_data(
                out_offset,
                U256_ZERO,
                target_len,
                &runtime.return_data_buffer[..],
            );

            Ok(())
        }
        ExitReason::Error(_) => {
            runtime.machine.stack_mut().push(U256_ZERO)?;

            Ok(())
        }
        ExitReason::Fatal(e) => {
            runtime.machine.stack_mut().push(U256_ZERO)?;

            Err(e.into())
        }
    }
}