revm_context_interface/cfg/
gas.rs1use crate::{cfg::GasParams, transaction::AccessListItemTr as _, Transaction, TransactionType};
4use primitives::hardfork::SpecId;
5
6pub const ZERO: u64 = 0;
8pub const BASE: u64 = 2;
10
11pub const VERYLOW: u64 = 3;
13pub const DATA_LOADN_GAS: u64 = 3;
15
16pub const CONDITION_JUMP_GAS: u64 = 4;
18pub const RETF_GAS: u64 = 3;
20pub const DATA_LOAD_GAS: u64 = 4;
22
23pub const LOW: u64 = 5;
25pub const MID: u64 = 8;
27pub const HIGH: u64 = 10;
29pub const JUMPDEST: u64 = 1;
31pub const SELFDESTRUCT_REFUND: i64 = 24000;
33pub const CREATE: u64 = 32000;
35pub const CALLVALUE: u64 = 9000;
37pub const NEWACCOUNT: u64 = 25000;
39pub const EXP: u64 = 10;
41pub const MEMORY: u64 = 3;
43pub const LOG: u64 = 375;
45pub const LOGDATA: u64 = 8;
47pub const LOGTOPIC: u64 = 375;
49pub const KECCAK256: u64 = 30;
51pub const KECCAK256WORD: u64 = 6;
53pub const COPY: u64 = 3;
55pub const BLOCKHASH: u64 = 20;
57pub const CODEDEPOSIT: u64 = 200;
59
60pub const ISTANBUL_SLOAD_GAS: u64 = 800;
62pub const SSTORE_SET: u64 = 20000;
64pub const SSTORE_RESET: u64 = 5000;
66pub const REFUND_SSTORE_CLEARS: i64 = 15000;
68
69pub const STANDARD_TOKEN_COST: u64 = 4;
71pub const NON_ZERO_BYTE_DATA_COST: u64 = 68;
73pub const NON_ZERO_BYTE_MULTIPLIER: u64 = NON_ZERO_BYTE_DATA_COST / STANDARD_TOKEN_COST;
75pub const NON_ZERO_BYTE_DATA_COST_ISTANBUL: u64 = 16;
77pub const NON_ZERO_BYTE_MULTIPLIER_ISTANBUL: u64 =
79 NON_ZERO_BYTE_DATA_COST_ISTANBUL / STANDARD_TOKEN_COST;
80pub const TOTAL_COST_FLOOR_PER_TOKEN: u64 = 10;
82
83pub const EOF_CREATE_GAS: u64 = 32000;
85
86pub const ACCESS_LIST_ADDRESS: u64 = 2400;
89pub const ACCESS_LIST_STORAGE_KEY: u64 = 1900;
91
92pub const COLD_SLOAD_COST: u64 = 2100;
94pub const COLD_ACCOUNT_ACCESS_COST: u64 = 2600;
96pub const COLD_ACCOUNT_ACCESS_COST_ADDITIONAL: u64 =
98 COLD_ACCOUNT_ACCESS_COST - WARM_STORAGE_READ_COST;
99pub const WARM_STORAGE_READ_COST: u64 = 100;
101pub const WARM_SSTORE_RESET: u64 = SSTORE_RESET - COLD_SLOAD_COST;
103
104pub const INITCODE_WORD_COST: u64 = 2;
106
107pub const CALL_STIPEND: u64 = 2300;
109
110#[derive(Clone, Copy, Debug, Default, PartialEq, Eq)]
112#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
113pub struct InitialAndFloorGas {
114 pub initial_gas: u64,
116 pub floor_gas: u64,
119}
120
121impl InitialAndFloorGas {
122 #[inline]
124 pub const fn new(initial_gas: u64, floor_gas: u64) -> Self {
125 Self {
126 initial_gas,
127 floor_gas,
128 }
129 }
130}
131
132pub fn calculate_initial_tx_gas(
140 spec_id: SpecId,
141 input: &[u8],
142 is_create: bool,
143 access_list_accounts: u64,
144 access_list_storages: u64,
145 authorization_list_num: u64,
146) -> InitialAndFloorGas {
147 GasParams::new_spec(spec_id).initial_tx_gas(
148 input,
149 is_create,
150 access_list_accounts,
151 access_list_storages,
152 authorization_list_num,
153 )
154}
155
156pub fn calculate_initial_tx_gas_for_tx(tx: impl Transaction, spec: SpecId) -> InitialAndFloorGas {
164 let mut accounts = 0;
165 let mut storages = 0;
166 if tx.tx_type() != TransactionType::Legacy {
168 (accounts, storages) = tx
169 .access_list()
170 .map(|al| {
171 al.fold((0, 0), |(mut num_accounts, mut num_storage_slots), item| {
172 num_accounts += 1;
173 num_storage_slots += item.storage_slots().count();
174
175 (num_accounts, num_storage_slots)
176 })
177 })
178 .unwrap_or_default();
179 }
180
181 calculate_initial_tx_gas(
182 spec,
183 tx.input(),
184 tx.kind().is_create(),
185 accounts as u64,
186 storages as u64,
187 tx.authorization_list_len() as u64,
188 )
189}
190
191#[inline]
193pub fn get_tokens_in_calldata_istanbul(input: &[u8]) -> u64 {
194 get_tokens_in_calldata(input, NON_ZERO_BYTE_MULTIPLIER_ISTANBUL)
195}
196
197#[inline]
199pub fn get_tokens_in_calldata(input: &[u8], non_zero_data_multiplier: u64) -> u64 {
200 let zero_data_len = input.iter().filter(|v| **v == 0).count() as u64;
201 let non_zero_data_len = input.len() as u64 - zero_data_len;
202 zero_data_len + non_zero_data_len * non_zero_data_multiplier
203}