forked from rust-ethereum/evm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhandler.rs
126 lines (118 loc) · 4.18 KB
/
handler.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
use crate::{Capture, Context, CreateScheme, ExitError, ExitReason, Machine, Opcode, Stack};
use alloc::vec::Vec;
use primitive_types::{H160, H256, U256};
/// Transfer from source to target, with given value.
#[derive(Clone, Debug)]
pub struct Transfer {
/// Source address.
pub source: H160,
/// Target address.
pub target: H160,
/// Transfer value.
pub value: U256,
}
/// EVM context handler.
#[auto_impl::auto_impl(&mut, Box)]
pub trait Handler {
/// Type of `CREATE` interrupt.
type CreateInterrupt;
/// Feedback value for `CREATE` interrupt.
type CreateFeedback;
/// Type of `CALL` interrupt.
type CallInterrupt;
/// Feedback value of `CALL` interrupt.
type CallFeedback;
/// Get balance of address.
fn balance(&self, address: H160) -> U256;
/// Get code size of address.
fn code_size(&self, address: H160) -> U256;
/// Get code hash of address.
fn code_hash(&self, address: H160) -> H256;
/// Get code of address.
fn code(&self, address: H160) -> Vec<u8>;
/// Get storage value of address at index.
fn storage(&self, address: H160, index: H256) -> H256;
/// Get original storage value of address at index.
fn original_storage(&self, address: H160, index: H256) -> H256;
/// Get the gas left value.
fn gas_left(&self) -> U256;
/// Get the gas price value.
fn gas_price(&self) -> U256;
/// Get execution origin.
fn origin(&self) -> H160;
/// Get environmental block hash.
fn block_hash(&self, number: U256) -> H256;
/// Get environmental block number.
fn block_number(&self) -> U256;
/// Get environmental coinbase.
fn block_coinbase(&self) -> H160;
/// Get environmental block timestamp.
fn block_timestamp(&self) -> U256;
/// Get environmental block difficulty.
fn block_difficulty(&self) -> U256;
/// Get environmental block randomness.
fn block_randomness(&self) -> Option<H256>;
/// Get environmental gas limit.
fn block_gas_limit(&self) -> U256;
/// Environmental block base fee.
fn block_base_fee_per_gas(&self) -> U256;
/// Get environmental chain ID.
fn chain_id(&self) -> U256;
/// Check whether an address exists.
fn exists(&self, address: H160) -> bool;
/// Check whether an address has already been deleted.
fn deleted(&self, address: H160) -> bool;
/// Checks if the address or (address, index) pair has been previously accessed
/// (or set in `accessed_addresses` / `accessed_storage_keys` via an access list
/// transaction).
/// References:
/// * <https://eips.ethereum.org/EIPS/eip-2929>
/// * <https://eips.ethereum.org/EIPS/eip-2930>
fn is_cold(&mut self, address: H160, index: Option<H256>) -> Result<bool, ExitError>;
/// Set storage value of address at index.
fn set_storage(&mut self, address: H160, index: H256, value: H256) -> Result<(), ExitError>;
/// Create a log owned by address with given topics and data.
fn log(&mut self, address: H160, topics: Vec<H256>, data: Vec<u8>) -> Result<(), ExitError>;
/// Mark an address to be deleted, with funds transferred to target.
fn mark_delete(&mut self, address: H160, target: H160) -> Result<(), ExitError>;
/// Invoke a create operation.
fn create(
&mut self,
caller: H160,
scheme: CreateScheme,
value: U256,
init_code: Vec<u8>,
target_gas: Option<u64>,
) -> Capture<(ExitReason, Option<H160>, Vec<u8>), Self::CreateInterrupt>;
/// Feed in create feedback.
fn create_feedback(&mut self, _feedback: Self::CreateFeedback) -> Result<(), ExitError> {
Ok(())
}
/// Invoke a call operation.
fn call(
&mut self,
code_address: H160,
transfer: Option<Transfer>,
input: Vec<u8>,
target_gas: Option<u64>,
is_static: bool,
context: Context,
) -> Capture<(ExitReason, Vec<u8>), Self::CallInterrupt>;
/// Feed in call feedback.
fn call_feedback(&mut self, _feedback: Self::CallFeedback) -> Result<(), ExitError> {
Ok(())
}
/// Pre-validation step for the runtime.
fn pre_validate(
&mut self,
context: &Context,
opcode: Opcode,
stack: &Stack,
) -> Result<(), ExitError>;
/// Handle other unknown external opcodes.
fn other(&mut self, opcode: Opcode, _stack: &mut Machine) -> Result<(), ExitError> {
Err(ExitError::InvalidCode(opcode))
}
/// Records some associated `ExternalOperation`.
fn record_external_operation(&mut self, op: crate::ExternalOperation) -> Result<(), ExitError>;
}