ethereum.cancun.vm.gas

Ethereum Virtual Machine (EVM) Gas ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

.. contents:: Table of Contents :backlinks: none :local:

Introduction

EVM gas constants and calculators.

GAS_JUMPDEST

27
GAS_JUMPDEST = Uint(1)

GAS_BASE

28
GAS_BASE = Uint(2)

GAS_VERY_LOW

29
GAS_VERY_LOW = Uint(3)

GAS_STORAGE_SET

30
GAS_STORAGE_SET = Uint(20000)

GAS_STORAGE_UPDATE

31
GAS_STORAGE_UPDATE = Uint(5000)

GAS_STORAGE_CLEAR_REFUND

32
GAS_STORAGE_CLEAR_REFUND = Uint(4800)

GAS_LOW

33
GAS_LOW = Uint(5)

GAS_MID

34
GAS_MID = Uint(8)

GAS_HIGH

35
GAS_HIGH = Uint(10)

GAS_EXPONENTIATION

36
GAS_EXPONENTIATION = Uint(10)

GAS_EXPONENTIATION_PER_BYTE

37
GAS_EXPONENTIATION_PER_BYTE = Uint(50)

GAS_MEMORY

38
GAS_MEMORY = Uint(3)

GAS_KECCAK256

39
GAS_KECCAK256 = Uint(30)

GAS_KECCAK256_WORD

40
GAS_KECCAK256_WORD = Uint(6)

GAS_COPY

41
GAS_COPY = Uint(3)

GAS_BLOCK_HASH

42
GAS_BLOCK_HASH = Uint(20)

GAS_LOG

43
GAS_LOG = Uint(375)

GAS_LOG_DATA

44
GAS_LOG_DATA = Uint(8)

GAS_LOG_TOPIC

45
GAS_LOG_TOPIC = Uint(375)

GAS_CREATE

46
GAS_CREATE = Uint(32000)

GAS_CODE_DEPOSIT

47
GAS_CODE_DEPOSIT = Uint(200)

GAS_ZERO

48
GAS_ZERO = Uint(0)

GAS_NEW_ACCOUNT

49
GAS_NEW_ACCOUNT = Uint(25000)

GAS_CALL_VALUE

50
GAS_CALL_VALUE = Uint(9000)

GAS_CALL_STIPEND

51
GAS_CALL_STIPEND = Uint(2300)

GAS_SELF_DESTRUCT

52
GAS_SELF_DESTRUCT = Uint(5000)

GAS_SELF_DESTRUCT_NEW_ACCOUNT

53
GAS_SELF_DESTRUCT_NEW_ACCOUNT = Uint(25000)

GAS_ECRECOVER

54
GAS_ECRECOVER = Uint(3000)

GAS_SHA256

55
GAS_SHA256 = Uint(60)

GAS_SHA256_WORD

56
GAS_SHA256_WORD = Uint(12)

GAS_RIPEMD160

57
GAS_RIPEMD160 = Uint(600)

GAS_RIPEMD160_WORD

58
GAS_RIPEMD160_WORD = Uint(120)

GAS_IDENTITY

59
GAS_IDENTITY = Uint(15)

GAS_IDENTITY_WORD

60
GAS_IDENTITY_WORD = Uint(3)

GAS_RETURN_DATA_COPY

61
GAS_RETURN_DATA_COPY = Uint(3)

GAS_FAST_STEP

62
GAS_FAST_STEP = Uint(5)

GAS_BLAKE2_PER_ROUND

63
GAS_BLAKE2_PER_ROUND = Uint(1)

GAS_COLD_SLOAD

64
GAS_COLD_SLOAD = Uint(2100)

GAS_COLD_ACCOUNT_ACCESS

65
GAS_COLD_ACCOUNT_ACCESS = Uint(2600)

GAS_WARM_ACCESS

66
GAS_WARM_ACCESS = Uint(100)

GAS_INIT_CODE_WORD_COST

67
GAS_INIT_CODE_WORD_COST = Uint(2)

GAS_BLOBHASH_OPCODE

68
GAS_BLOBHASH_OPCODE = Uint(3)

GAS_POINT_EVALUATION

69
GAS_POINT_EVALUATION = Uint(50000)

TARGET_BLOB_GAS_PER_BLOCK

71
TARGET_BLOB_GAS_PER_BLOCK = U64(393216)

GAS_PER_BLOB

72
GAS_PER_BLOB = Uint(2**17)

MIN_BLOB_GASPRICE

73
MIN_BLOB_GASPRICE = Uint(1)

BLOB_GASPRICE_UPDATE_FRACTION

74
BLOB_GASPRICE_UPDATE_FRACTION = Uint(3338477)

ExtendMemory

Define the parameters for memory extension in opcodes

cost: ethereum.base_types.Uint The gas required to perform the extension expand_by: ethereum.base_types.Uint The size by which the memory will be extended

77
@dataclass
class ExtendMemory:

cost

88
    cost: Uint

expand_by

89
    expand_by: Uint

MessageCallGas

Define the gas cost and stipend for executing the call opcodes.

cost: ethereum.base_types.Uint The non-refundable portion of gas reserved for executing the call opcode. stipend: ethereum.base_types.Uint The portion of gas available to sub-calls that is refundable if not consumed

92
@dataclass
class MessageCallGas:

cost

105
    cost: Uint

stipend

106
    stipend: Uint

charge_gas

Subtracts amount from evm.gas_left.

Parameters

evm : The current EVM. amount : The amount of gas the current operation requires.

def charge_gas(evm: Evm, ​​amount: Uint) -> None:
110
    """
111
    Subtracts `amount` from `evm.gas_left`.
112
113
    Parameters
114
    ----------
115
    evm :
116
        The current EVM.
117
    amount :
118
        The amount of gas the current operation requires.
119
120
    """
121
    evm_trace(evm, GasAndRefund(int(amount)))
122
123
    if evm.gas_left < amount:
124
        raise OutOfGasError
125
    else:
126
        evm.gas_left -= amount

calculate_memory_gas_cost

Calculates the gas cost for allocating memory to the smallest multiple of 32 bytes, such that the allocated size is at least as big as the given size.

Parameters

size_in_bytes : The size of the data in bytes.

Returns

total_gas_cost : ethereum.base_types.Uint The gas cost for storing data in memory.

def calculate_memory_gas_cost(size_in_bytes: Uint) -> Uint:
130
    """
131
    Calculates the gas cost for allocating memory
132
    to the smallest multiple of 32 bytes,
133
    such that the allocated size is at least as big as the given size.
134
135
    Parameters
136
    ----------
137
    size_in_bytes :
138
        The size of the data in bytes.
139
140
    Returns
141
    -------
142
    total_gas_cost : `ethereum.base_types.Uint`
143
        The gas cost for storing data in memory.
144
    """
145
    size_in_words = ceil32(size_in_bytes) // Uint(32)
146
    linear_cost = size_in_words * GAS_MEMORY
147
    quadratic_cost = size_in_words ** Uint(2) // Uint(512)
148
    total_gas_cost = linear_cost + quadratic_cost
149
    try:
150
        return total_gas_cost
151
    except ValueError:
152
        raise OutOfGasError

calculate_gas_extend_memory

Calculates the gas amount to extend memory

Parameters

memory : Memory contents of the EVM. extensions: List of extensions to be made to the memory. Consists of a tuple of start position and size.

Returns

extend_memory: ExtendMemory

def calculate_gas_extend_memory(memory: bytearray, ​​extensions: List[Tuple[U256, U256]]) -> ExtendMemory:
158
    """
159
    Calculates the gas amount to extend memory
160
161
    Parameters
162
    ----------
163
    memory :
164
        Memory contents of the EVM.
165
    extensions:
166
        List of extensions to be made to the memory.
167
        Consists of a tuple of start position and size.
168
169
    Returns
170
    -------
171
    extend_memory: `ExtendMemory`
172
    """
173
    size_to_extend = Uint(0)
174
    to_be_paid = Uint(0)
175
    current_size = Uint(len(memory))
176
    for start_position, size in extensions:
177
        if size == 0:
178
            continue
179
        before_size = ceil32(current_size)
180
        after_size = ceil32(Uint(start_position) + Uint(size))
181
        if after_size <= before_size:
182
            continue
183
184
        size_to_extend += after_size - before_size
185
        already_paid = calculate_memory_gas_cost(before_size)
186
        total_cost = calculate_memory_gas_cost(after_size)
187
        to_be_paid += total_cost - already_paid
188
189
        current_size = after_size
190
191
    return ExtendMemory(to_be_paid, size_to_extend)

calculate_message_call_gas

Calculates the MessageCallGas (cost and stipend) for executing call Opcodes.

Parameters

value: The amount of ETH that needs to be transferred. gas : The amount of gas provided to the message-call. gas_left : The amount of gas left in the current frame. memory_cost : The amount needed to extend the memory in the current frame. extra_gas : The amount of gas needed for transferring value + creating a new account inside a message call. call_stipend : The amount of stipend provided to a message call to execute code while transferring value(ETH).

Returns

message_call_gas: MessageCallGas

def calculate_message_call_gas(value: U256, ​​gas: Uint, ​​gas_left: Uint, ​​memory_cost: Uint, ​​extra_gas: Uint, ​​call_stipend: Uint) -> MessageCallGas:
202
    """
203
    Calculates the MessageCallGas (cost and stipend) for
204
    executing call Opcodes.
205
206
    Parameters
207
    ----------
208
    value:
209
        The amount of `ETH` that needs to be transferred.
210
    gas :
211
        The amount of gas provided to the message-call.
212
    gas_left :
213
        The amount of gas left in the current frame.
214
    memory_cost :
215
        The amount needed to extend the memory in the current frame.
216
    extra_gas :
217
        The amount of gas needed for transferring value + creating a new
218
        account inside a message call.
219
    call_stipend :
220
        The amount of stipend provided to a message call to execute code while
221
        transferring value(ETH).
222
223
    Returns
224
    -------
225
    message_call_gas: `MessageCallGas`
226
    """
227
    call_stipend = Uint(0) if value == 0 else call_stipend
228
    if gas_left < extra_gas + memory_cost:
229
        return MessageCallGas(gas + extra_gas, gas + call_stipend)
230
231
    gas = min(gas, max_message_call_gas(gas_left - memory_cost - extra_gas))
232
233
    return MessageCallGas(gas + extra_gas, gas + call_stipend)

max_message_call_gas

Calculates the maximum gas that is allowed for making a message call

Parameters

gas : The amount of gas provided to the message-call.

Returns

max_allowed_message_call_gas: ethereum.base_types.Uint The maximum gas allowed for making the message-call.

def max_message_call_gas(gas: Uint) -> Uint:
237
    """
238
    Calculates the maximum gas that is allowed for making a message call
239
240
    Parameters
241
    ----------
242
    gas :
243
        The amount of gas provided to the message-call.
244
245
    Returns
246
    -------
247
    max_allowed_message_call_gas: `ethereum.base_types.Uint`
248
        The maximum gas allowed for making the message-call.
249
    """
250
    return gas - (gas // Uint(64))

init_code_cost

Calculates the gas to be charged for the init code in CREAT* opcodes as well as create transactions.

Parameters

init_code_length : The length of the init code provided to the opcode or a create transaction

Returns

init_code_gas: ethereum.base_types.Uint The gas to be charged for the init code.

def init_code_cost(init_code_length: Uint) -> Uint:
254
    """
255
    Calculates the gas to be charged for the init code in CREAT*
256
    opcodes as well as create transactions.
257
258
    Parameters
259
    ----------
260
    init_code_length :
261
        The length of the init code provided to the opcode
262
        or a create transaction
263
264
    Returns
265
    -------
266
    init_code_gas: `ethereum.base_types.Uint`
267
        The gas to be charged for the init code.
268
    """
269
    return GAS_INIT_CODE_WORD_COST * ceil32(init_code_length) // Uint(32)

calculate_excess_blob_gas

Calculated the excess blob gas for the current block based on the gas used in the parent block.

Parameters

parent_header : The parent block of the current block.

Returns

excess_blob_gas: ethereum.base_types.U64 The excess blob gas for the current block.

def calculate_excess_blob_gas(parent_header: Header) -> U64:
273
    """
274
    Calculated the excess blob gas for the current block based
275
    on the gas used in the parent block.
276
277
    Parameters
278
    ----------
279
    parent_header :
280
        The parent block of the current block.
281
282
    Returns
283
    -------
284
    excess_blob_gas: `ethereum.base_types.U64`
285
        The excess blob gas for the current block.
286
    """
287
    # At the fork block, these are defined as zero.
288
    excess_blob_gas = U64(0)
289
    blob_gas_used = U64(0)
290
291
    if isinstance(parent_header, Header):
292
        # After the fork block, read them from the parent header.
293
        excess_blob_gas = parent_header.excess_blob_gas
294
        blob_gas_used = parent_header.blob_gas_used
295
296
    parent_blob_gas = excess_blob_gas + blob_gas_used
297
    if parent_blob_gas < TARGET_BLOB_GAS_PER_BLOCK:
298
        return U64(0)
299
    else:
300
        return parent_blob_gas - TARGET_BLOB_GAS_PER_BLOCK

calculate_total_blob_gas

Calculate the total blob gas for a transaction.

Parameters

tx : The transaction for which the blob gas is to be calculated.

Returns

total_blob_gas: ethereum.base_types.Uint The total blob gas for the transaction.

def calculate_total_blob_gas(tx: Transaction) -> Uint:
304
    """
305
    Calculate the total blob gas for a transaction.
306
307
    Parameters
308
    ----------
309
    tx :
310
        The transaction for which the blob gas is to be calculated.
311
312
    Returns
313
    -------
314
    total_blob_gas: `ethereum.base_types.Uint`
315
        The total blob gas for the transaction.
316
    """
317
    if isinstance(tx, BlobTransaction):
318
        return GAS_PER_BLOB * Uint(len(tx.blob_versioned_hashes))
319
    else:
320
        return Uint(0)

calculate_blob_gas_price

Calculate the blob gasprice for a block.

Parameters

excess_blob_gas : The excess blob gas for the block.

Returns

blob_gasprice: Uint The blob gasprice.

def calculate_blob_gas_price(excess_blob_gas: U64) -> Uint:
324
    """
325
    Calculate the blob gasprice for a block.
326
327
    Parameters
328
    ----------
329
    excess_blob_gas :
330
        The excess blob gas for the block.
331
332
    Returns
333
    -------
334
    blob_gasprice: `Uint`
335
        The blob gasprice.
336
    """
337
    return taylor_exponential(
338
        MIN_BLOB_GASPRICE,
339
        Uint(excess_blob_gas),
340
        BLOB_GASPRICE_UPDATE_FRACTION,
341
    )

calculate_data_fee

Calculate the blob data fee for a transaction.

Parameters

excess_blob_gas : The excess_blob_gas for the execution. tx : The transaction for which the blob data fee is to be calculated.

Returns

data_fee: Uint The blob data fee.

def calculate_data_fee(excess_blob_gas: U64, ​​tx: Transaction) -> Uint:
345
    """
346
    Calculate the blob data fee for a transaction.
347
348
    Parameters
349
    ----------
350
    excess_blob_gas :
351
        The excess_blob_gas for the execution.
352
    tx :
353
        The transaction for which the blob data fee is to be calculated.
354
355
    Returns
356
    -------
357
    data_fee: `Uint`
358
        The blob data fee.
359
    """
360
    return calculate_total_blob_gas(tx) * calculate_blob_gas_price(
361
        excess_blob_gas
362
    )