Skip to content

Commit 5ca4741

Browse files
0xJepsenAutoparallelrakita
authoredJun 18, 2023
book 📝 (bluealloy#494)
* book 📝 initialization intro + chapters interpreter * Gas + Memory * added to examples and refactor * precompiles * Update primitives.md * added docs through `instructions` * book - utilities - revm - restructure book for subchapters * final touches * Update SUMMARY.md reorder topics Change order of book items --------- Co-authored-by: Colin Roberts <[email protected]> Co-authored-by: rakita <[email protected]>
1 parent 63f9460 commit 5ca4741

40 files changed

+1416
-0
lines changed
 

‎.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -10,3 +10,4 @@ pkg/
1010
bins/revme/temp_folder
1111
bins/revme/tests
1212
ethereumjs-util.js
13+
book

‎book.toml

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
[book]
2+
authors = ["Colin, Waylon Jepsen"]
3+
language = "en"
4+
multilingual = false
5+
src = "documentation/src"
6+
title = "Rust EVM"

‎documentation/bins/revm-test.md

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
# revm-test

‎documentation/bins/revme.md

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
# revme

‎documentation/src/SUMMARY.md

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
# Summary
2+
3+
- [Introduction](./introduction.md)
4+
- [Revm](./crates/revm.md)
5+
- [evm](./crates/revm/evm.md)
6+
- [evm_impl](./crates/revm/evm_impl.md)
7+
- [inspector](./crates/revm/inspector.md)
8+
- [journaled_state](./crates/revm/journaled_state.md)
9+
- [Interpreter](./crates/interpreter.md)
10+
- [gas](./crates/interpreter/gas.md)
11+
- [host](./crates/interpreter/host.md)
12+
- [inner_models](./crates/interpreter/inner_models.md)
13+
- [instruction_result](./crates/interpreter/instruction_result.md)
14+
- [instructions](./crates/interpreter/instructions.md)
15+
- [Examples](./examples.md)
16+
- [Primitives](./crates/primitives.md)
17+
- [database](./crates/primitives/database.md)
18+
- [result](./crates/primitives/result.md)
19+
- [environment](./crates/primitives/environment.md)
20+
- [specifications](./crates/primitives/specifications.md)
21+
- [bits](./crates/primitives/bits.md)
22+
- [bytecode](./crates/primitives/bytecode.md)
23+
- [constants](./crates/primitives/constants.md)
24+
- [log](./crates/primitives/log.md)
25+
- [precompile](./crates/primitives/precompile.md)
26+
- [state](./crates/primitives/state.md)
27+
- [utils](./crates/primitives/utils.md)
28+
- [Precompile](./crates/precompile.md)
29+
- [blake2](./crates/precompile/blake2.md)
30+
- [bn128 curve](./crates/precompile/bn128.md)
31+
- [Hash functions](./crates/precompile/hash.md)
32+
- [Identity function](./crates/precompile/identity.md)
33+
- [Modular Exponentiation](./crates/precompile/modexp.md)
34+
- [Secp256k1](./crates/precompile/secp256k1.md)

‎documentation/src/bins/revm-test.md

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
# revm-test

‎documentation/src/bins/revme.md

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
# revme
+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
# Interpreter
2+
3+
The `interpreter` crate is concerned with the execution of the EVM opcodes and serves as the event loop to step through the opcodes. The interpreter is concerned with attributes like gas, contracts, memory, stack, and returning execution results. It's structured as follows:
4+
5+
Modules:
6+
7+
- [gas](./interpreter/gas.md): This module deals with handling the gas mechanics in the EVM, such as calculating gas costs for operations.
8+
- [host](./interpreter/host.md): This module defines the `Host` trait, and any types or functions that the host machine (the machine running the EVM).
9+
- [inner_models](./interpreter/inner_models.md): Based on the name, this module could contain the inner data structures or models used in the EVM implementation.
10+
- [instruction_result](./interpreter/instruction_result.md): This module likely contains definitions related to the result of instruction execution.
11+
- [instructions](./interpreter/instructions.md): This module is expected to include the definitions of the EVM opcodes (instructions).
12+
- [interpreter](./interpreter/interpreter.md): This module would contain the Interpreter struct and related functionality for executing EVM instructions.
13+
14+
External Crates:
15+
16+
- alloc: The alloc crate is used to provide the ability to allocate memory on the heap. It's a part of Rust's standard library that can be used in environments without a full host OS.
17+
- core: The core crate is the dependency-free foundation of the Rust standard library. It includes fundamental types, macros, and traits.
18+
19+
Constants:
20+
21+
- `USE_GAS`: This constant determines whether gas measurement should be used. It's set to false if the no_gas_measuring feature is enabled.
22+
23+
Re-exported Types:
24+
Several types and functions are re-exported for easier access by users of this library, such as Gas, Host, InstructionResult, OpCode, Interpreter, Memory, Stack, and others. This allows users to import these items directly from the library root instead of from their individual modules.
25+
Re-exported Crate:
26+
revm_primitives: This crate is re-exported, likely providing primitive types or functionality used in the EVM implementation.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
# The `gas.rs` Module
2+
3+
The `gas.rs` module in this Rust EVM implementation manages the concept of "gas" within the Ethereum network. In Ethereum, "gas" signifies the computational effort needed to execute operations, whether a simple transfer of ether or the execution of a smart contract function. Each operation carries a gas cost, and transactions must specify the maximum amount of gas they are willing to consume.
4+
5+
## `Gas` Struct
6+
7+
The `Gas` struct represents the gas state for a particular operation or transaction. The struct is defined as follows:
8+
9+
```rust
10+
#[derive(Clone, Copy, Debug)]
11+
pub struct Gas {
12+
/// Gas Limit
13+
limit: u64,
14+
/// used+memory gas.
15+
all_used_gas: u64,
16+
/// Used gas without memory
17+
used: u64,
18+
/// Used gas for memory expansion
19+
memory: u64,
20+
/// Refunded gas. This gas is used only at the end of execution.
21+
refunded: i64,
22+
}
23+
```
24+
25+
### Fields in `Gas` Struct
26+
27+
- `limit`: The maximum amount of gas allowed for the operation or transaction.
28+
- `all_used_gas`: The total gas used, inclusive of memory expansion costs.
29+
- `used`: The gas used, excluding memory expansion costs.
30+
- `memory`: The gas used for memory expansion.
31+
- `refunded`: The gas refunded. Certain operations in Ethereum allow for gas refunds, up to half the gas used by a transaction.
32+
33+
## Methods of the `Gas` Struct
34+
35+
The `Gas` struct also includes several methods to manage the gas state. Here's a brief summary of their functions:
36+
37+
- `new`: Creates a new `Gas` instance with a specified gas limit and zero usage and refunds.
38+
- `limit`, `memory`, `refunded`, `spend`, `remaining`: These getters return the current state of the corresponding field.
39+
- `erase_cost`: Decreases the gas usage by a specified amount.
40+
- `record_refund`: Increases the refunded gas by a specified amount.
41+
- `record_cost`: Increases the used gas by a specified amount. It also checks for gas limit overflow. If the new total used gas would exceed the gas limit, it returns `false` and doesn't change the state.
42+
- `record_memory`: This method works similarly to `record_cost`, but specifically for memory expansion gas. It only updates the state if the new memory gas usage is greater than the current usage.
43+
- `gas_refund`: Increases the refunded gas by a specified amount.
44+
45+
## Importance of the `Gas` Struct
46+
47+
These features of the `Gas` struct allow for effective management and tracking of the gas cost associated with executing EVM operations. This is a key part of ensuring that smart contracts and transactions adhere to the resource constraints of the Ethereum network, since overconsumption of resources could potentially lead to network congestion.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
# The `host.rs` Module
2+
3+
The `host.rs` module in this Rust EVM implementation defines a crucial trait `Host`. The `Host` trait outlines an interface for the interaction of the EVM interpreter with its environment (or "host"), encompassing essential operations such as account and storage access, creating logs, and invoking transactions.
4+
5+
## `Host` Trait
6+
7+
The `Host` trait provides the definition for all the methods that any EVM host must implement. It's a crucial bridge between the EVM and its environment, allowing the EVM to interact with blockchain state.
8+
9+
```rust
10+
pub trait Host {
11+
fn step(&mut self, interpreter: &mut Interpreter) -> InstructionResult;
12+
fn step_end(
13+
&mut self,
14+
interpreter: &mut Interpreter,
15+
ret: InstructionResult,
16+
) -> InstructionResult;
17+
// ... (other methods omitted for brevity)
18+
}
19+
```
20+
21+
## Key Methods of the `Host` Trait
22+
23+
- `step` & `step_end`: These methods manage the execution of EVM opcodes. The `step` method is invoked before executing an opcode, while `step_end` is invoked after. These methods can modify the EVM state or halt execution based on certain conditions.
24+
25+
- `env`: This method provides access to the EVM environment, including information about the current block and transaction.
26+
27+
- `load_account`: Retrieves information about a given Ethereum account.
28+
29+
- `block_hash`: Retrieves the block hash for a given block number.
30+
31+
- `balance`, `code`, `code_hash`, `sload`: These methods retrieve specific information (balance, code, code hash, and specific storage value) for a given Ethereum account.
32+
33+
- `sstore`: This method sets the value of a specific storage slot in a given Ethereum account.
34+
35+
- `log`: Creates a log entry with the specified address, topics, and data. Log entries are used by smart contracts to emit events.
36+
37+
- `selfdestruct`: Marks an Ethereum account to be self-destructed, transferring its funds to a target account.
38+
39+
- `create` & `call`: These methods handle the creation of new smart contracts and the invocation of smart contract functions, respectively.
40+
41+
## Importance of the `Host` Trait
42+
43+
The `Host` trait provides a standard interface that any host environment for the EVM must implement. This abstraction allows the EVM code to interact with the state of the Ethereum network in a generic way, thereby enhancing modularity and interoperability. Different implementations of the `Host` trait can be used to simulate different environments for testing or for connecting to different Ethereum-like networks.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,94 @@
1+
# The `inner_models.rs` Module in the Rust Ethereum Virtual Machine (EVM)
2+
3+
The `inner_models.rs` module within this Rust EVM implementation encompasses a collection of structs and enums that are used as internal models within the EVM. These models represent various aspects of EVM operations such as call and create inputs, call context, value transfers, and the result of self-destruction operations.
4+
5+
## `CallInputs` Struct
6+
7+
The `CallInputs` struct is used to encapsulate the inputs to a smart contract call in the EVM.
8+
9+
```rust
10+
pub struct CallInputs {
11+
pub contract: B160,
12+
pub transfer: Transfer,
13+
pub input: Bytes,
14+
pub gas_limit: u64,
15+
pub context: CallContext,
16+
pub is_static: bool,
17+
}
18+
```
19+
20+
This struct includes the target contract address, the value to be transferred (if any), the input data, the gas limit for the call, the call context, and a boolean indicating if the call is a static call (a read-only operation).
21+
22+
## `CreateInputs` Struct
23+
24+
The `CreateInputs` struct encapsulates the inputs for creating a new smart contract.
25+
26+
```rust
27+
pub struct CreateInputs {
28+
pub caller: B160,
29+
pub scheme: CreateScheme,
30+
pub value: U256,
31+
pub init_code: Bytes,
32+
pub gas_limit: u64,
33+
}
34+
```
35+
36+
This includes the address of the creator, the creation scheme, the value to be transferred, the initialization code for the new contract, and the gas limit for the creation operation.
37+
38+
## `CallScheme` Enum
39+
40+
The `CallScheme` enum represents the type of call being made to a smart contract.
41+
42+
```rust
43+
pub enum CallScheme {
44+
Call,
45+
CallCode,
46+
DelegateCall,
47+
StaticCall,
48+
}
49+
```
50+
51+
The different types of calls (`CALL`, `CALLCODE`, `DELEGATECALL`, `STATICCALL`) represent different modes of interaction with a smart contract, each with its own semantics concerning the treatment of the message sender, value transfer, and the context in which the called code executes.
52+
53+
## `CallContext` Struct
54+
55+
The `CallContext` struct encapsulates the context of a smart contract call.
56+
57+
```rust
58+
pub struct CallContext {
59+
pub address: B160,
60+
pub caller: B160,
61+
pub code_address: B160,
62+
pub apparent_value: U256,
63+
pub scheme: CallScheme,
64+
}
65+
```
66+
67+
This includes the executing contract's address, the caller's address, the address from which the contract code was loaded, the apparent value of the call (for `DELEGATECALL` and `CALLCODE`), and the call scheme.
68+
69+
## `Transfer` Struct
70+
71+
The `Transfer` struct represents a value transfer between two accounts.
72+
73+
```rust
74+
pub struct Transfer {
75+
pub source: B160,
76+
pub target: B160,
77+
pub value: U256,
78+
}
79+
```
80+
81+
## `SelfDestructResult` Struct
82+
83+
Finally, the `SelfDestructResult` struct captures the result of a self-destruction operation on a contract.
84+
85+
```rust
86+
pub struct SelfDestructResult {
87+
pub had_value: bool,
88+
pub target_exists: bool,
89+
pub is_cold: bool,
90+
pub previously_destroyed: bool,
91+
}
92+
```
93+
94+
In summary, the `inner_models.rs` module provides several crucial data structures that facilitate the representation and handling of various EVM operations and their associated data within this Rust EVM implementation.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
# The `instruction_result.rs` Module
2+
3+
The `instruction_result.rs` module of this Rust EVM implementation includes the definitions of the enumerations `InstructionResult` and `SuccessOrHalt`, which represent the possible outcomes of EVM instruction execution, and functions to work with these types.
4+
5+
## `InstructionResult` Enum
6+
7+
The `InstructionResult` enum categorizes the different types of results that can arise from executing an EVM instruction. This enumeration uses the `#[repr(u8)]` attribute, meaning its variants have an explicit storage representation of an 8-bit unsigned integer.
8+
9+
```rust
10+
pub enum InstructionResult {
11+
Continue = 0x00,
12+
Stop = 0x01,
13+
Return = 0x02,
14+
SelfDestruct = 0x03,
15+
Revert = 0x20,
16+
CallTooDeep = 0x21,
17+
OutOfFund = 0x22,
18+
OutOfGas = 0x50,
19+
// more variants...
20+
}
21+
```
22+
23+
The different instruction results represent outcomes such as successful continuation, stop, return, self-destruction, reversion, deep call, out of funds, out of gas, and various error conditions.
24+
25+
## `SuccessOrHalt` Enum
26+
27+
The `SuccessOrHalt` enum represents the outcome of a transaction execution, distinguishing successful operations, reversion, halting conditions, fatal external errors, and internal continuation.
28+
29+
```rust
30+
pub enum SuccessOrHalt {
31+
Success(Eval),
32+
Revert,
33+
Halt(Halt),
34+
FatalExternalError,
35+
InternalContinue,
36+
}
37+
```
38+
39+
It also provides several methods to check the kind of result and to extract the value of the successful evaluation or halt.
40+
41+
## `From<InstructionResult> for SuccessOrHalt` Implementation
42+
43+
This implementation provides a way to convert an `InstructionResult` into a `SuccessOrHalt`. It maps each instruction result to the corresponding `SuccessOrHalt` variant.
44+
45+
```rust
46+
impl From<InstructionResult> for SuccessOrHalt {
47+
fn from(result: InstructionResult) -> Self {
48+
match result {
49+
InstructionResult::Continue => Self::InternalContinue,
50+
InstructionResult::Stop => Self::Success(Eval::Stop),
51+
// more match arms...
52+
}
53+
}
54+
}
55+
```
56+
57+
## Macros for returning instruction results
58+
59+
Finally, the module provides two macros, `return_ok!` and `return_revert!`, which simplify returning some common sets of instruction results.
60+
61+
```rust
62+
#[macro_export]
63+
macro_rules! return_ok {
64+
() => {
65+
InstructionResult::Continue
66+
| InstructionResult::Stop
67+
| InstructionResult::Return
68+
| InstructionResult::SelfDestruct
69+
};
70+
}
71+
72+
#[macro_export]
73+
macro_rules! return_revert {
74+
() => {
75+
InstructionResult::Revert | InstructionResult::CallTooDeep | InstructionResult::OutOfFund
76+
};
77+
}
78+
```
79+
80+
In summary, the `instruction_result.rs` module is essential to the interpretation of EVM instructions, as it outlines the possible outcomes and provides means to handle these outcomes within the Rust EVM implementation.

0 commit comments

Comments
 (0)
Please sign in to comment.