aurora-evm 2.2.1

Aurora Ethereum Virtual Machine implementation written in pure Rust
Documentation
//! Core layer for EVM.

#[cfg(not(feature = "std"))]
pub mod prelude {
    pub use alloc::{borrow::Cow, rc::Rc, vec, vec::Vec};
}
#[cfg(feature = "std")]
pub mod prelude {
    pub use std::{borrow::Cow, rc::Rc, vec::Vec};
}

mod error;
mod eval;
mod external;
mod memory;
mod opcode;
mod stack;
pub mod utils;
mod valids;

pub use error::{Capture, ExitError, ExitFatal, ExitReason, ExitRevert, ExitSucceed, Trap};
pub use external::ExternalOperation;
pub use memory::Memory;
pub use opcode::Opcode;
pub use stack::Stack;
pub use valids::Valids;

use crate::utils::U256_ZERO;
use core::ops::Range;
use eval::{eval, Control};
use prelude::*;
use primitive_types::{H160, U256};
use utils::USIZE_MAX;

/// Core execution layer for EVM.
pub struct Machine {
    /// Program data.
    data: Rc<Vec<u8>>,
    /// Program code.
    code: Rc<Vec<u8>>,
    /// Program counter.
    position: Result<usize, ExitReason>,
    /// Return value.
    return_range: Range<U256>,
    /// Code validity maps.
    valids: Valids,
    /// Memory.
    memory: Memory,
    /// Stack.
    stack: Stack,
}

/// EVM interpreter handler.
pub trait InterpreterHandler {
    /// # Errors
    /// Return `ExitError`
    fn before_bytecode(
        &mut self,
        opcode: Opcode,
        pc: usize,
        machine: &Machine,
        address: &H160,
    ) -> Result<(), ExitError>;

    // Only invoked for tracing
    #[cfg(feature = "tracing")]
    fn after_bytecode(&mut self, result: &Result<(), Capture<ExitReason, Trap>>, machine: &Machine);
}

impl Machine {
    /// Reference of machine stack.
    #[must_use]
    pub const fn stack(&self) -> &Stack {
        &self.stack
    }
    /// Mutable reference of machine stack.
    pub const fn stack_mut(&mut self) -> &mut Stack {
        &mut self.stack
    }
    /// Reference of machine memory.
    #[must_use]
    pub const fn memory(&self) -> &Memory {
        &self.memory
    }
    /// Mutable reference of machine memory.
    pub const fn memory_mut(&mut self) -> &mut Memory {
        &mut self.memory
    }
    /// Return a reference of the program counter.
    pub const fn position(&self) -> &Result<usize, ExitReason> {
        &self.position
    }

    /// Create a new machine with given code and data.
    #[must_use]
    pub fn new(
        code: Rc<Vec<u8>>,
        data: Rc<Vec<u8>>,
        stack_limit: usize,
        memory_limit: usize,
    ) -> Self {
        let valids = Valids::new(&code[..]);

        Self {
            data,
            code,
            position: Ok(0),
            return_range: U256_ZERO..U256_ZERO,
            valids,
            memory: Memory::new(memory_limit),
            stack: Stack::new(stack_limit),
        }
    }

    /// Explicit exit of the machine. Further step will return error.
    pub fn exit(&mut self, reason: ExitReason) {
        self.position = Err(reason);
    }

    /// Inspect the machine's next opcode and current stack.
    #[must_use]
    pub fn inspect(&self) -> Option<(Opcode, &Stack)> {
        let Ok(position) = self.position else {
            return None;
        };
        self.code.get(position).map(|v| (Opcode(*v), &self.stack))
    }

    /// Copy and get the return value of the machine, if any.
    #[must_use]
    pub fn return_value(&self) -> Vec<u8> {
        if self.return_range.start > USIZE_MAX {
            vec![0; (self.return_range.end - self.return_range.start).as_usize()]
        } else if self.return_range.end > USIZE_MAX {
            let mut ret = self.memory.get(
                self.return_range.start.as_usize(),
                usize::MAX - self.return_range.start.as_usize(),
            );

            let new_len = (self.return_range.end - self.return_range.start).as_usize();
            if ret.len() < new_len {
                ret.resize(new_len, 0);
            }
            ret
        } else {
            self.memory.get(
                self.return_range.start.as_usize(),
                (self.return_range.end - self.return_range.start).as_usize(),
            )
        }
    }

    /// Step the machine, executing until exit or trap.
    ///
    /// # Errors
    /// Return `Capture<ExitReason, Trap>`
    #[inline]
    pub fn step<H: InterpreterHandler>(
        &mut self,
        handler: &mut H,
        address: &H160,
    ) -> Result<(), Capture<ExitReason, Trap>> {
        let position = *self
            .position
            .as_ref()
            .map_err(|reason| Capture::Exit(reason.clone()))?;
        match eval(self, position, handler, address) {
            Control::Exit(e) => {
                self.position = Err(e.clone());
                Err(Capture::Exit(e))
            }
            Control::Trap(opcode) => Err(Capture::Trap(opcode)),
            Control::Continue(_) | Control::Jump(_) => Ok(()),
        }
    }
}