nimbus-eth1/nimbus/core/gaslimit.nim

96 lines
3.4 KiB
Nim
Raw Normal View History

# Nimbus
2024-05-30 14:54:03 +02:00
# Copyright (c) 2018-2024 Status Research & Development GmbH
# Licensed under either of
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or
# http://www.apache.org/licenses/LICENSE-2.0)
# * MIT license ([LICENSE-MIT](LICENSE-MIT) or
# http://opensource.org/licenses/MIT)
# at your option. This file may not be copied, modified, or distributed except
# according to those terms.
{.push raises: [].}
import
2022-12-02 11:35:41 +07:00
../common/common,
std/strformat,
2024-05-30 14:54:03 +02:00
results,
2022-12-02 11:35:41 +07:00
eth/[eip1559]
export
eip1559
2022-10-20 09:44:16 +07:00
# ------------------------------------------------------------------------------
# Pre Eip 1559 gas limit validation
# ------------------------------------------------------------------------------
proc validateGasLimit(header: Header; limit: GasInt): Result[void,string] =
let diff = if limit > header.gasLimit:
limit - header.gasLimit
else:
header.gasLimit - limit
let upperLimit = limit div GAS_LIMIT_ADJUSTMENT_FACTOR
if diff >= upperLimit:
return err(&"invalid gas limit: have {header.gasLimit}, want {limit} +-= {upperLimit-1}")
if header.gasLimit < GAS_LIMIT_MINIMUM:
return err("invalid gas limit below 5000")
ok()
# ------------------------------------------------------------------------------
# Eip 1559 support
# ------------------------------------------------------------------------------
# consensus/misc/eip1559.go(55): func CalcBaseFee(config [..]
proc calcEip1599BaseFee*(com: CommonRef; parent: Header): UInt256 =
## calculates the basefee of the header.
# If the current block is the first EIP-1559 block, return the
# initial base fee.
if com.isLondonOrLater(parent.number):
eip1559.calcEip1599BaseFee(parent.gasLimit, parent.gasUsed, parent.baseFeePerGas.get(0.u256))
else:
EIP1559_INITIAL_BASE_FEE
# consensus/misc/eip1559.go(32): func VerifyEip1559Header(config [..]
2022-12-02 11:35:41 +07:00
proc verifyEip1559Header(com: CommonRef;
parent, header: Header): Result[void, string]
{.raises: [].} =
## Verify that the gas limit remains within allowed bounds
let limit = if com.isLondonOrLater(parent.number):
parent.gasLimit
else:
parent.gasLimit * EIP1559_ELASTICITY_MULTIPLIER
let rc = header.validateGasLimit(limit)
if rc.isErr:
return rc
let headerBaseFee = header.baseFeePerGas.get(0.u256)
# Verify the header is not malformed
if headerBaseFee.isZero:
return err("Post EIP-1559 header expected to have base fee")
# Verify the baseFee is correct based on the parent header.
2022-12-02 11:35:41 +07:00
var expectedBaseFee = com.calcEip1599BaseFee(parent)
if headerBaseFee != expectedBaseFee:
return err(&"invalid baseFee: have {expectedBaseFee}, "&
&"want {header.baseFeePerGas}, " &
&"parent.baseFee {parent.baseFeePerGas}, "&
&"parent.gasUsed {parent.gasUsed}")
return ok()
2022-12-02 11:35:41 +07:00
proc validateGasLimitOrBaseFee*(com: CommonRef;
header, parent: Header): Result[void, string] =
if not com.isLondonOrLater(header.number):
# Verify BaseFee not present before EIP-1559 fork.
let baseFeePerGas = header.baseFeePerGas.get(0.u256)
if not baseFeePerGas.isZero:
return err("invalid baseFee before London fork: have " & $baseFeePerGas & ", want <0>")
?validateGasLimit(header, parent.gasLimit)
else:
?com.verifyEip1559Header(parent = parent, header = header)
return ok()