Architecture and API
REVM is a flexible implementation of the Ethereum Virtual Machine (EVM). It follows the rules of the Ethereum mainnet and stays up to date with changes through hardforks as defined in the official Ethereum execution specs.
You can use REVM in two main ways:
- Run regular Ethereum transactions using an Execution API
- Create your own custom version of the EVM (for Layer 2 solutions or other chains) using the EVM framework
To see usage examples you can check the examples folder. Other than documentation, examples are the main resource to see and learn about Revm.
The main revm library combines all crates into one package and reexports them, standalone libraries are useful if there is a need to import functionality with smaller scope. You can see an overview of all revm crates in the crates folder.
REVM works in no_std environments which means it can be used in zero-knowledge virtual machines (zkVMs) and it is the standard library in that use case. It also has very few external dependencies.
Execution API
The Execution API provides the primary interface for running Ethereum transactions and interacting with the EVM. Whether you're building a blockchain client, testing framework, or analysis tool, this API offers multiple execution modes to suit your needs.
The API is designed around four key execution patterns:
- Basic execution: Run transactions and get results
- Execution with commit: Run transactions and automatically persist state changes
- Execution with inspection: Run transactions with detailed tracing and observation
Evm the main structure for executing mainnet ethereum transaction is built with a Context and a builder, code for it looks like this:
let mut evm = Context::mainnet().with_block(block).build_mainnet();
let out = evm.transact(tx);
Evm struct contains:
Context- Environment and evm state.Instructions- EVM opcode implementationsPrecompiles- Built-in contract implementationsInspector- Used for tracing.
And Context contains data used in execution:
- Environment data, the data that is known before execution starts are
Transaction,Block,Cfg. Journalis the place where internal state is stored. Internal state is returned after execution ends.- And
Databaseis an interface that allows fetching external data that is needed at runtime. That data are account, storage and bytecode. When loaded they are stored inJournal
- And
REVM provides four ways to execute transactions through traits (API):
transact(tx)andreplay()are function ofExecuteEvmtrait that allow execution transactions. They return the status of execution with reason, changed state and in case of failed execution an error.transact_commit(tx)andreplay_commit()are part ofExecuteCommitEvmthat internally commits the state diff to the database and returns status of execution. Database is required to supportDatabaseCommittrait.inspect(),inspect_replay(tx)and a few others are part ofInspectEvmtrait that allow execution with inspection. This is how tracers are called.inspect_commit(),inspect_replay_commit(tx)are part of theInspectCommitEvmtrait that extendsInspectEvmto allow committing state diff after tracing.
For inspection API to be enabled, Evm needs to be created with inspector.
let mut evm = Context::mainnet().with_block(block).build_mainnet().with_inspector(inspector);
let _ = evm.inspect_tx(tx);
Inspector - EVM Execution Tracing
The Inspector trait is REVM's powerful mechanism for observing EVM execution. It provides hooks into every aspect of transaction execution, enabling sophisticated debugging, tracing and custom tooling.
Key capabilities include:
- Step-by-step execution tracing: Hook into every opcode before and after execution
- State monitoring: Track stack, memory, and storage changes in real-time
- Call and creation tracing: Observe contract interactions and deployments
- Event capture: Record logs, self-destructs, and other EVM events
- Execution override: Optionally modify execution flow and outcomes
The Inspector is ideal for building debuggers, gas analyzers, security tools, testing frameworks, and any application that needs deep visibility into EVM execution. For detailed usage examples and advanced features, see the Inspector documentation.
EVM Framework
To learn how to build your own custom EVM:
- Check out the example-my-evm guide
- Look at op-revm to see how Optimism uses REVM
Each trait needed to build custom EVM has detailed documentation explaining how it works and is worth reading.
In summary, REVM is built around several key traits that enable customizable EVM functionality. The core traits include:
EvmTr: The core EVM trait that provides access toContext,Instruction,Precompiles:ContextTr: Accessed through EvmTr, defines the execution environment including Tx/Block/Journal/Db.Handler: Implements the core execution logic, taking an EvmTr implementation. The default implementation follows Ethereum consensus.
And traits that provide support for inspection and tracing:
InspectorEvmTr: Extends EvmTr to enable inspection mode execution with an associatedInspectortypeInspectorHandler: Extends Handler with inspection-enabled execution paths that makeInspectorcallbacksInspector: User-implementable trait for EVM inspection/tracing