1use crate::{Capture, Context, CreateScheme, ExitError, ExitReason, Machine, Opcode, Stack};
2use alloc::vec::Vec;
3use primitive_types::{H160, H256, U256};
4
5#[derive(Clone, Debug)]
7pub struct Transfer {
8 pub source: H160,
10 pub target: H160,
12 pub value: U256,
14}
15
16#[auto_impl::auto_impl(&mut, Box)]
18pub trait Handler {
19 type CreateInterrupt;
21 type CreateFeedback;
23 type CallInterrupt;
25 type CallFeedback;
27
28 fn balance(&self, address: H160) -> U256;
30 fn code_size(&self, address: H160) -> U256;
32 fn code_hash(&self, address: H160) -> H256;
34 fn code(&self, address: H160) -> Vec<u8>;
36 fn delegated_code(&self, address: H160) -> Option<Vec<u8>> {
38 let code = self.code(address);
39 evm_core::extract_delegation_address(&code)
40 .map(|delegated_address| self.code(delegated_address))
41 }
42 fn storage(&self, address: H160, index: H256) -> H256;
44 fn transient_storage(&self, address: H160, index: H256) -> H256;
46
47 fn original_storage(&self, address: H160, index: H256) -> H256;
49
50 fn gas_left(&self) -> U256;
52 fn gas_price(&self) -> U256;
54 fn origin(&self) -> H160;
56 fn block_hash(&self, number: U256) -> H256;
58 fn block_number(&self) -> U256;
60 fn block_coinbase(&self) -> H160;
62 fn block_timestamp(&self) -> U256;
64 fn block_difficulty(&self) -> U256;
66 fn block_randomness(&self) -> Option<H256>;
68 fn block_gas_limit(&self) -> U256;
70 fn block_base_fee_per_gas(&self) -> U256;
72 fn chain_id(&self) -> U256;
74
75 fn exists(&self, address: H160) -> bool;
77 fn deleted(&self, address: H160) -> bool;
79 fn is_cold(&mut self, address: H160, index: Option<H256>) -> Result<bool, ExitError>;
86
87 fn set_storage(&mut self, address: H160, index: H256, value: H256) -> Result<(), ExitError>;
89 fn set_transient_storage(&mut self, address: H160, index: H256, value: H256);
91 fn log(&mut self, address: H160, topics: Vec<H256>, data: Vec<u8>) -> Result<(), ExitError>;
93 fn mark_delete(&mut self, address: H160, target: H160) -> Result<(), ExitError>;
95 fn create(
97 &mut self,
98 caller: H160,
99 scheme: CreateScheme,
100 value: U256,
101 init_code: Vec<u8>,
102 target_gas: Option<u64>,
103 ) -> Capture<(ExitReason, Option<H160>, Vec<u8>), Self::CreateInterrupt>;
104 fn create_feedback(&mut self, _feedback: Self::CreateFeedback) -> Result<(), ExitError> {
106 Ok(())
107 }
108 fn call(
110 &mut self,
111 code_address: H160,
112 transfer: Option<Transfer>,
113 input: Vec<u8>,
114 target_gas: Option<u64>,
115 is_static: bool,
116 context: Context,
117 ) -> Capture<(ExitReason, Vec<u8>), Self::CallInterrupt>;
118 fn call_feedback(&mut self, _feedback: Self::CallFeedback) -> Result<(), ExitError> {
120 Ok(())
121 }
122
123 fn pre_validate(
125 &mut self,
126 context: &Context,
127 opcode: Opcode,
128 stack: &Stack,
129 ) -> Result<(), ExitError>;
130 fn other(&mut self, opcode: Opcode, _stack: &mut Machine) -> Result<(), ExitError> {
132 Err(ExitError::InvalidCode(opcode))
133 }
134
135 fn record_external_operation(&mut self, op: crate::ExternalOperation) -> Result<(), ExitError>;
137}