ethereum.forks.bpo4.vm.gas

Ethereum Virtual Machine (EVM) Gas.

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

Introduction

EVM gas constants and calculators.

GasCosts

Constant gas values for the EVM.

class GasCosts:

BASE

35
    BASE = Uint(2)

VERY_LOW

36
    VERY_LOW = Uint(3)

LOW

37
    LOW = Uint(5)

MID

38
    MID = Uint(8)

HIGH

39
    HIGH = Uint(10)

WARM_ACCESS

42
    WARM_ACCESS = Uint(100)

COLD_ACCOUNT_ACCESS

43
    COLD_ACCOUNT_ACCESS = Uint(2600)

COLD_STORAGE_ACCESS

44
    COLD_STORAGE_ACCESS = Uint(2100)

STORAGE_SET

47
    STORAGE_SET = Uint(20000)

COLD_STORAGE_WRITE

48
    COLD_STORAGE_WRITE = Uint(5000)

CALL_VALUE

51
    CALL_VALUE = Uint(9000)

CALL_STIPEND

52
    CALL_STIPEND = Uint(2300)

NEW_ACCOUNT

53
    NEW_ACCOUNT = Uint(25000)

CODE_DEPOSIT_PER_BYTE

56
    CODE_DEPOSIT_PER_BYTE = Uint(200)

CODE_INIT_PER_WORD

57
    CODE_INIT_PER_WORD = Uint(2)

AUTH_PER_EMPTY_ACCOUNT

60
    AUTH_PER_EMPTY_ACCOUNT = 25000

ZERO

63
    ZERO = Uint(0)

MEMORY_PER_WORD

64
    MEMORY_PER_WORD = Uint(3)

FAST_STEP

65
    FAST_STEP = Uint(5)

REFUND_STORAGE_CLEAR

68
    REFUND_STORAGE_CLEAR = 4800

PRECOMPILE_ECRECOVER

71
    PRECOMPILE_ECRECOVER = Uint(3000)

PRECOMPILE_P256VERIFY

72
    PRECOMPILE_P256VERIFY = Uint(6900)

PRECOMPILE_SHA256_BASE

73
    PRECOMPILE_SHA256_BASE = Uint(60)

PRECOMPILE_SHA256_PER_WORD

74
    PRECOMPILE_SHA256_PER_WORD = Uint(12)

PRECOMPILE_RIPEMD160_BASE

75
    PRECOMPILE_RIPEMD160_BASE = Uint(600)

PRECOMPILE_RIPEMD160_PER_WORD

76
    PRECOMPILE_RIPEMD160_PER_WORD = Uint(120)

PRECOMPILE_IDENTITY_BASE

77
    PRECOMPILE_IDENTITY_BASE = Uint(15)

PRECOMPILE_IDENTITY_PER_WORD

78
    PRECOMPILE_IDENTITY_PER_WORD = Uint(3)

PRECOMPILE_BLAKE2F_PER_ROUND

79
    PRECOMPILE_BLAKE2F_PER_ROUND = Uint(1)

PRECOMPILE_POINT_EVALUATION

80
    PRECOMPILE_POINT_EVALUATION = Uint(50000)

PRECOMPILE_BLS_G1ADD

81
    PRECOMPILE_BLS_G1ADD = Uint(375)

PRECOMPILE_BLS_G1MUL

82
    PRECOMPILE_BLS_G1MUL = Uint(12000)

PRECOMPILE_BLS_G1MAP

83
    PRECOMPILE_BLS_G1MAP = Uint(5500)

PRECOMPILE_BLS_G2ADD

84
    PRECOMPILE_BLS_G2ADD = Uint(600)

PRECOMPILE_BLS_G2MUL

85
    PRECOMPILE_BLS_G2MUL = Uint(22500)

PRECOMPILE_BLS_G2MAP

86
    PRECOMPILE_BLS_G2MAP = Uint(23800)

PRECOMPILE_ECADD

87
    PRECOMPILE_ECADD = Uint(150)

PRECOMPILE_ECMUL

88
    PRECOMPILE_ECMUL = Uint(6000)

PRECOMPILE_ECPAIRING_BASE

89
    PRECOMPILE_ECPAIRING_BASE = Uint(45000)

PRECOMPILE_ECPAIRING_PER_POINT

90
    PRECOMPILE_ECPAIRING_PER_POINT = Uint(34000)

PER_BLOB

93
    PER_BLOB = U64(2**17)

BLOB_SCHEDULE_TARGET

94
    BLOB_SCHEDULE_TARGET = U64(14)

BLOB_TARGET_GAS_PER_BLOCK

95
    BLOB_TARGET_GAS_PER_BLOCK = PER_BLOB * BLOB_SCHEDULE_TARGET

BLOB_BASE_COST

96
    BLOB_BASE_COST = Uint(2**13)

BLOB_SCHEDULE_MAX

97
    BLOB_SCHEDULE_MAX = U64(21)

BLOB_MIN_GASPRICE

98
    BLOB_MIN_GASPRICE = Uint(1)

BLOB_BASE_FEE_UPDATE_FRACTION

99
    BLOB_BASE_FEE_UPDATE_FRACTION = Uint(11684671)

TX_BASE

102
    TX_BASE = Uint(21000)

TX_CREATE

103
    TX_CREATE = Uint(32000)

TX_DATA_TOKEN_STANDARD

104
    TX_DATA_TOKEN_STANDARD = Uint(4)

TX_DATA_TOKEN_FLOOR

105
    TX_DATA_TOKEN_FLOOR = Uint(10)

TX_ACCESS_LIST_ADDRESS

106
    TX_ACCESS_LIST_ADDRESS = Uint(2400)

TX_ACCESS_LIST_STORAGE_KEY

107
    TX_ACCESS_LIST_STORAGE_KEY = Uint(1900)

LIMIT_ADJUSTMENT_FACTOR

110
    LIMIT_ADJUSTMENT_FACTOR = Uint(1024)

LIMIT_MINIMUM

111
    LIMIT_MINIMUM = Uint(5000)

OPCODE_ADD

114
    OPCODE_ADD = VERY_LOW

OPCODE_SUB

115
    OPCODE_SUB = VERY_LOW

OPCODE_MUL

116
    OPCODE_MUL = LOW

OPCODE_DIV

117
    OPCODE_DIV = LOW

OPCODE_SDIV

118
    OPCODE_SDIV = LOW

OPCODE_MOD

119
    OPCODE_MOD = LOW

OPCODE_SMOD

120
    OPCODE_SMOD = LOW

OPCODE_ADDMOD

121
    OPCODE_ADDMOD = MID

OPCODE_MULMOD

122
    OPCODE_MULMOD = MID

OPCODE_SIGNEXTEND

123
    OPCODE_SIGNEXTEND = LOW

OPCODE_LT

124
    OPCODE_LT = VERY_LOW

OPCODE_GT

125
    OPCODE_GT = VERY_LOW

OPCODE_SLT

126
    OPCODE_SLT = VERY_LOW

OPCODE_SGT

127
    OPCODE_SGT = VERY_LOW

OPCODE_EQ

128
    OPCODE_EQ = VERY_LOW

OPCODE_ISZERO

129
    OPCODE_ISZERO = VERY_LOW

OPCODE_AND

130
    OPCODE_AND = VERY_LOW

OPCODE_OR

131
    OPCODE_OR = VERY_LOW

OPCODE_XOR

132
    OPCODE_XOR = VERY_LOW

OPCODE_NOT

133
    OPCODE_NOT = VERY_LOW

OPCODE_BYTE

134
    OPCODE_BYTE = VERY_LOW

OPCODE_SHL

135
    OPCODE_SHL = VERY_LOW

OPCODE_SHR

136
    OPCODE_SHR = VERY_LOW

OPCODE_SAR

137
    OPCODE_SAR = VERY_LOW

OPCODE_CLZ

138
    OPCODE_CLZ = LOW

OPCODE_JUMP

139
    OPCODE_JUMP = MID

OPCODE_JUMPI

140
    OPCODE_JUMPI = HIGH

OPCODE_JUMPDEST

141
    OPCODE_JUMPDEST = Uint(1)

OPCODE_CALLDATALOAD

142
    OPCODE_CALLDATALOAD = VERY_LOW

OPCODE_BLOCKHASH

143
    OPCODE_BLOCKHASH = Uint(20)

OPCODE_COINBASE

144
    OPCODE_COINBASE = BASE

OPCODE_POP

145
    OPCODE_POP = BASE

OPCODE_MSIZE

146
    OPCODE_MSIZE = BASE

OPCODE_PC

147
    OPCODE_PC = BASE

OPCODE_GAS

148
    OPCODE_GAS = BASE

OPCODE_ADDRESS

149
    OPCODE_ADDRESS = BASE

OPCODE_ORIGIN

150
    OPCODE_ORIGIN = BASE

OPCODE_CALLER

151
    OPCODE_CALLER = BASE

OPCODE_CALLVALUE

152
    OPCODE_CALLVALUE = BASE

OPCODE_CALLDATASIZE

153
    OPCODE_CALLDATASIZE = BASE

OPCODE_CODESIZE

154
    OPCODE_CODESIZE = BASE

OPCODE_GASPRICE

155
    OPCODE_GASPRICE = BASE

OPCODE_TIMESTAMP

156
    OPCODE_TIMESTAMP = BASE

OPCODE_NUMBER

157
    OPCODE_NUMBER = BASE

OPCODE_GASLIMIT

158
    OPCODE_GASLIMIT = BASE

OPCODE_PREVRANDAO

159
    OPCODE_PREVRANDAO = BASE

OPCODE_RETURNDATASIZE

160
    OPCODE_RETURNDATASIZE = BASE

OPCODE_CHAINID

161
    OPCODE_CHAINID = BASE

OPCODE_BASEFEE

162
    OPCODE_BASEFEE = BASE

OPCODE_BLOBBASEFEE

163
    OPCODE_BLOBBASEFEE = BASE

OPCODE_BLOBHASH

164
    OPCODE_BLOBHASH = Uint(3)

OPCODE_PUSH

165
    OPCODE_PUSH = VERY_LOW

OPCODE_PUSH0

166
    OPCODE_PUSH0 = BASE

OPCODE_DUP

167
    OPCODE_DUP = VERY_LOW

OPCODE_SWAP

168
    OPCODE_SWAP = VERY_LOW

OPCODE_RETURNDATACOPY_BASE

171
    OPCODE_RETURNDATACOPY_BASE = VERY_LOW

OPCODE_RETURNDATACOPY_PER_WORD

172
    OPCODE_RETURNDATACOPY_PER_WORD = Uint(3)

OPCODE_CALLDATACOPY_BASE

173
    OPCODE_CALLDATACOPY_BASE = VERY_LOW

OPCODE_CODECOPY_BASE

174
    OPCODE_CODECOPY_BASE = VERY_LOW

OPCODE_MCOPY_BASE

175
    OPCODE_MCOPY_BASE = VERY_LOW

OPCODE_MLOAD_BASE

176
    OPCODE_MLOAD_BASE = VERY_LOW

OPCODE_MSTORE_BASE

177
    OPCODE_MSTORE_BASE = VERY_LOW

OPCODE_MSTORE8_BASE

178
    OPCODE_MSTORE8_BASE = VERY_LOW

OPCODE_COPY_PER_WORD

179
    OPCODE_COPY_PER_WORD = Uint(3)

OPCODE_CREATE_BASE

180
    OPCODE_CREATE_BASE = Uint(32000)

OPCODE_EXP_BASE

181
    OPCODE_EXP_BASE = Uint(10)

OPCODE_EXP_PER_BYTE

182
    OPCODE_EXP_PER_BYTE = Uint(50)

OPCODE_KECCAK256_BASE

183
    OPCODE_KECCAK256_BASE = Uint(30)

OPCODE_KECCACK256_PER_WORD

184
    OPCODE_KECCACK256_PER_WORD = Uint(6)

OPCODE_LOG_BASE

185
    OPCODE_LOG_BASE = Uint(375)

OPCODE_LOG_DATA_PER_BYTE

186
    OPCODE_LOG_DATA_PER_BYTE = Uint(8)

OPCODE_LOG_TOPIC

187
    OPCODE_LOG_TOPIC = Uint(375)

OPCODE_SELFDESTRUCT_BASE

188
    OPCODE_SELFDESTRUCT_BASE = Uint(5000)

OPCODE_SELFDESTRUCT_NEW_ACCOUNT

189
    OPCODE_SELFDESTRUCT_NEW_ACCOUNT = Uint(25000)

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

192
@dataclass
class ExtendMemory:

cost

203
    cost: Uint

expand_by

204
    expand_by: Uint

MessageCallGas

Define the gas cost and gas given to the sub-call for executing the call opcodes.

cost: ethereum.base_types.Uint The gas required to execute the call opcode, excludes memory expansion costs. sub_call: ethereum.base_types.Uint The portion of gas available to sub-calls that is refundable if not consumed.

207
@dataclass
class MessageCallGas:

cost

221
    cost: Uint

sub_call

222
    sub_call: 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:
226
    """
227
    Subtracts `amount` from `evm.gas_left`.
228
229
    Parameters
230
    ----------
231
    evm :
232
        The current EVM.
233
    amount :
234
        The amount of gas the current operation requires.
235
236
    """
237
    evm_trace(evm, GasAndRefund(int(amount)))
238
239
    if evm.gas_left < amount:
240
        raise OutOfGasError
241
    else:
242
        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:
246
    """
247
    Calculates the gas cost for allocating memory
248
    to the smallest multiple of 32 bytes,
249
    such that the allocated size is at least as big as the given size.
250
251
    Parameters
252
    ----------
253
    size_in_bytes :
254
        The size of the data in bytes.
255
256
    Returns
257
    -------
258
    total_gas_cost : `ethereum.base_types.Uint`
259
        The gas cost for storing data in memory.
260
261
    """
262
    size_in_words = ceil32(size_in_bytes) // Uint(32)
263
    linear_cost = size_in_words * GasCosts.MEMORY_PER_WORD
264
    quadratic_cost = size_in_words ** Uint(2) // Uint(512)
265
    total_gas_cost = linear_cost + quadratic_cost
266
    try:
267
        return total_gas_cost
268
    except ValueError as e:
269
        raise OutOfGasError from e

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:
275
    """
276
    Calculates the gas amount to extend memory.
277
278
    Parameters
279
    ----------
280
    memory :
281
        Memory contents of the EVM.
282
    extensions:
283
        List of extensions to be made to the memory.
284
        Consists of a tuple of start position and size.
285
286
    Returns
287
    -------
288
    extend_memory: `ExtendMemory`
289
290
    """
291
    size_to_extend = Uint(0)
292
    to_be_paid = Uint(0)
293
    current_size = ulen(memory)
294
    for start_position, size in extensions:
295
        if size == 0:
296
            continue
297
        before_size = ceil32(current_size)
298
        after_size = ceil32(Uint(start_position) + Uint(size))
299
        if after_size <= before_size:
300
            continue
301
302
        size_to_extend += after_size - before_size
303
        already_paid = calculate_memory_gas_cost(before_size)
304
        total_cost = calculate_memory_gas_cost(after_size)
305
        to_be_paid += total_cost - already_paid
306
307
        current_size = after_size
308
309
    return ExtendMemory(to_be_paid, size_to_extend)

calculate_message_call_gas

Calculates the MessageCallGas (cost and gas made available to the sub-call) 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:
320
    """
321
    Calculates the MessageCallGas (cost and gas made available to the sub-call)
322
    for executing call Opcodes.
323
324
    Parameters
325
    ----------
326
    value:
327
        The amount of `ETH` that needs to be transferred.
328
    gas :
329
        The amount of gas provided to the message-call.
330
    gas_left :
331
        The amount of gas left in the current frame.
332
    memory_cost :
333
        The amount needed to extend the memory in the current frame.
334
    extra_gas :
335
        The amount of gas needed for transferring value + creating a new
336
        account inside a message call.
337
    call_stipend :
338
        The amount of stipend provided to a message call to execute code while
339
        transferring value (ETH).
340
341
    Returns
342
    -------
343
    message_call_gas: `MessageCallGas`
344
345
    """
346
    call_stipend = Uint(0) if value == 0 else call_stipend
347
    if gas_left < extra_gas + memory_cost:
348
        return MessageCallGas(gas + extra_gas, gas + call_stipend)
349
350
    gas = min(gas, max_message_call_gas(gas_left - memory_cost - extra_gas))
351
352
    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:
356
    """
357
    Calculates the maximum gas that is allowed for making a message call.
358
359
    Parameters
360
    ----------
361
    gas :
362
        The amount of gas provided to the message-call.
363
364
    Returns
365
    -------
366
    max_allowed_message_call_gas: `ethereum.base_types.Uint`
367
        The maximum gas allowed for making the message-call.
368
369
    """
370
    return gas - (gas // Uint(64))

init_code_cost

Calculates the gas to be charged for the init code in CREATE* 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:
374
    """
375
    Calculates the gas to be charged for the init code in CREATE*
376
    opcodes as well as create transactions.
377
378
    Parameters
379
    ----------
380
    init_code_length :
381
        The length of the init code provided to the opcode
382
        or a create transaction
383
384
    Returns
385
    -------
386
    init_code_gas: `ethereum.base_types.Uint`
387
        The gas to be charged for the init code.
388
389
    """
390
    return GasCosts.CODE_INIT_PER_WORD * ceil32(init_code_length) // Uint(32)

calculate_excess_blob_gas

Calculates 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:
394
    """
395
    Calculates the excess blob gas for the current block based
396
    on the gas used in the parent block.
397
398
    Parameters
399
    ----------
400
    parent_header :
401
        The parent block of the current block.
402
403
    Returns
404
    -------
405
    excess_blob_gas: `ethereum.base_types.U64`
406
        The excess blob gas for the current block.
407
408
    """
409
    # At the fork block, these are defined as zero.
410
    excess_blob_gas = U64(0)
411
    blob_gas_used = U64(0)
412
    base_fee_per_gas = Uint(0)
413
414
    if isinstance(parent_header, Header):
415
        # After the fork block, read them from the parent header.
416
        excess_blob_gas = parent_header.excess_blob_gas
417
        blob_gas_used = parent_header.blob_gas_used
418
        base_fee_per_gas = parent_header.base_fee_per_gas
419
420
    parent_blob_gas = excess_blob_gas + blob_gas_used
421
    if parent_blob_gas < GasCosts.BLOB_TARGET_GAS_PER_BLOCK:
422
        return U64(0)
423
424
    target_blob_gas_price = Uint(GasCosts.PER_BLOB)
425
    target_blob_gas_price *= calculate_blob_gas_price(excess_blob_gas)
426
427
    base_blob_tx_price = GasCosts.BLOB_BASE_COST * base_fee_per_gas
428
    if base_blob_tx_price > target_blob_gas_price:
429
        blob_schedule_delta = (
430
            GasCosts.BLOB_SCHEDULE_MAX - GasCosts.BLOB_SCHEDULE_TARGET
431
        )
432
        return (
433
            excess_blob_gas
434
            + blob_gas_used * blob_schedule_delta // GasCosts.BLOB_SCHEDULE_MAX
435
        )
436
437
    return parent_blob_gas - GasCosts.BLOB_TARGET_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) -> U64:
441
    """
442
    Calculate the total blob gas for a transaction.
443
444
    Parameters
445
    ----------
446
    tx :
447
        The transaction for which the blob gas is to be calculated.
448
449
    Returns
450
    -------
451
    total_blob_gas: `ethereum.base_types.Uint`
452
        The total blob gas for the transaction.
453
454
    """
455
    if isinstance(tx, BlobTransaction):
456
        return GasCosts.PER_BLOB * U64(len(tx.blob_versioned_hashes))
457
    else:
458
        return U64(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:
462
    """
463
    Calculate the blob gasprice for a block.
464
465
    Parameters
466
    ----------
467
    excess_blob_gas :
468
        The excess blob gas for the block.
469
470
    Returns
471
    -------
472
    blob_gasprice: `Uint`
473
        The blob gasprice.
474
475
    """
476
    return taylor_exponential(
477
        GasCosts.BLOB_MIN_GASPRICE,
478
        Uint(excess_blob_gas),
479
        GasCosts.BLOB_BASE_FEE_UPDATE_FRACTION,
480
    )

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:
484
    """
485
    Calculate the blob data fee for a transaction.
486
487
    Parameters
488
    ----------
489
    excess_blob_gas :
490
        The excess_blob_gas for the execution.
491
    tx :
492
        The transaction for which the blob data fee is to be calculated.
493
494
    Returns
495
    -------
496
    data_fee: `Uint`
497
        The blob data fee.
498
499
    """
500
    return Uint(calculate_total_blob_gas(tx)) * calculate_blob_gas_price(
501
        excess_blob_gas
502
    )