nimbus-eth1/nimbus/utils/header.nim

104 lines
3.6 KiB
Nim

# Nimbus
# Copyright (c) 2018 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.
import
strformat, times, options,
eth/[common, rlp],
./difficulty, ../constants,
../chain_config
export BlockHeader
proc hasUncles*(header: BlockHeader): bool = header.ommersHash != EMPTY_UNCLE_HASH
proc `$`*(header: BlockHeader): string =
result = &"BlockHeader(timestamp: {header.timestamp} difficulty: {header.difficulty} blockNumber: {header.blockNumber} gasLimit: {header.gasLimit})"
# CalcGasLimit computes the gas limit of the next block after parent. It aims
# to keep the baseline gas above the provided floor, and increase it towards the
# ceil if the blocks are full. If the ceil is exceeded, it will always decrease
# the gas allowance.
func computeGasLimit*(parentGasUsed, parentGasLimit, gasFloor, gasCeil: GasInt): GasInt =
# contrib = (parentGasUsed * 3 / 2) / 1024
let contrib = (parentGasUsed + parentGasUsed div 2) div GAS_LIMIT_ADJUSTMENT_FACTOR
# decay = parentGasLimit / 1024 -1
let decay = parentGasLimit div GAS_LIMIT_ADJUSTMENT_FACTOR - 1
#[
strategy: gasLimit of block-to-mine is set based on parent's
gasUsed value. if parentGasUsed > parentGasLimit * (2/3) then we
increase it, otherwise lower it (or leave it unchanged if it's right
at that usage) the amount increased/decreased depends on how far away
from parentGasLimit * (2/3) parentGasUsed is.
]#
var limit = parentGasLimit - decay + contrib
if limit < GAS_LIMIT_MINIMUM:
limit = GAS_LIMIT_MINIMUM
# If we're outside our allowed gas range, we try to hone towards them
if limit < gasFloor:
limit = parentGasLimit + decay
if limit > gasFloor:
limit = gasFloor
elif limit > gasCeil:
limit = parentGasLimit - decay
if limit < gasCeil:
limit = gasCeil
return limit
proc generateHeaderFromParentHeader*(config: ChainConfig, parent: BlockHeader,
coinbase: EthAddress, timestamp: Option[EthTime],
gasLimit: GasInt, extraData: Blob, baseFee: Option[Uint256]): BlockHeader =
var lcTimestamp: EthTime
if timestamp.isNone:
lcTimeStamp = max(getTime(), parent.timestamp + 1.milliseconds) # Note: Py-evm uses +1 second, not ms
else:
lcTimestamp = timestamp.get()
if lcTimestamp <= parent.timestamp:
raise newException(ValueError, "header.timestamp should be higher than parent.timestamp")
result = BlockHeader(
timestamp: lcTimestamp,
blockNumber: (parent.blockNumber + 1),
difficulty: config.calcDifficulty(lcTimestamp, parent),
gasLimit: gasLimit,
stateRoot: parent.stateRoot,
coinbase: coinbase,
extraData: extraData,
fee: baseFee
)
# CalcGasLimit1559 calculates the next block gas limit under 1559 rules.
func calcGasLimit1559*(parentGasLimit, desiredLimit: GasInt): GasInt =
let delta = parentGasLimit div GAS_LIMIT_ADJUSTMENT_FACTOR - 1.GasInt
var limit = parentGasLimit
var desiredLimit = desiredLimit
if desiredLimit < GAS_LIMIT_MINIMUM:
desiredLimit = GAS_LIMIT_MINIMUM
# If we're outside our allowed gas range, we try to hone towards them
if limit < desiredLimit:
limit = parentGasLimit + delta
if limit > desiredLimit:
limit = desiredLimit
return limit
if limit > desiredLimit:
limit = parentGasLimit - delta
if limit < desiredLimit:
limit = desiredLimit
return limit