258 lines
8.5 KiB
Nim

# Nimbus
# 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.
## Transaction Pool Descriptor
## ===========================
##
import
std/[times],
eth/eip1559,
../../common/common,
../../evm/state,
../../evm/types,
../../db/ledger,
../../constants,
../pow/header,
../eip4844,
../casper,
./tx_item,
./tx_tabs,
./tx_tabs/tx_sender
{.push raises: [].}
type
TxPoolFlags* = enum ##\
## Processing strategy selector symbols
autoUpdateBucketsDB ##\
## Automatically update the state buckets after running batch jobs if
## the `dirtyBuckets` flag is also set.
autoZombifyUnpacked ##\
## Automatically dispose *pending* or *staged* txs that were queued
## at least `lifeTime` ago.
TxPoolParam* = tuple ## Getter/setter accessible parameters
dirtyBuckets: bool ## Buckets need to be updated
doubleCheck: seq[TxItemRef] ## Check items after moving block chain head
flags: set[TxPoolFlags] ## Processing strategy symbols
TxPoolRef* = ref object of RootObj ##\
## Transaction pool descriptor
startDate: Time ## Start date (read-only)
param: TxPoolParam ## Getter/Setter parameters
vmState: BaseVMState
txDB: TxTabsRef ## Transaction lists & tables
lifeTime*: times.Duration ## Maximum life time of a tx in the system
priceBump*: uint ## Min precentage price when superseding
const
txItemLifeTime = ##\
## Maximum amount of time transactions can be held in the database\
## unless they are packed already for a block. This default is chosen\
## as found in core/tx_pool.go(184) of the geth implementation.
initDuration(hours = 3)
txPriceBump = ##\
## Minimum price bump percentage to replace an already existing\
## transaction (nonce). This default is chosen as found in\
## core/tx_pool.go(177) of the geth implementation.
10u
txPoolFlags = {autoUpdateBucketsDB,
autoZombifyUnpacked}
# ------------------------------------------------------------------------------
# Private functions
# ------------------------------------------------------------------------------
proc baseFeeGet(com: CommonRef; parent: BlockHeader): Opt[UInt256] =
## Calculates the `baseFee` of the head assuming this is the parent of a
## new block header to generate.
# Note that the baseFee is calculated for the next header
if not com.isLondonOrLater(parent.number+1):
return Opt.none(UInt256)
# If the new block is the first EIP-1559 block, return initial base fee.
if not com.isLondonOrLater(parent.number):
return Opt.some(EIP1559_INITIAL_BASE_FEE)
Opt.some calcEip1599BaseFee(
parent.gasLimit,
parent.gasUsed,
parent.baseFeePerGas.get(0.u256))
proc gasLimitsGet(com: CommonRef; parent: BlockHeader): GasInt =
if com.isLondonOrLater(parent.number+1):
var parentGasLimit = parent.gasLimit
if not com.isLondonOrLater(parent.number):
# Bump by 2x
parentGasLimit = parent.gasLimit * EIP1559_ELASTICITY_MULTIPLIER
calcGasLimit1559(parentGasLimit, desiredLimit = DEFAULT_GAS_LIMIT)
else:
computeGasLimit(
parent.gasUsed,
parent.gasLimit,
gasFloor = DEFAULT_GAS_LIMIT,
gasCeil = DEFAULT_GAS_LIMIT)
proc setupVMState(com: CommonRef; parent: BlockHeader): BaseVMState =
# do hardfork transition before
# BaseVMState querying any hardfork/consensus from CommonRef
let pos = com.pos
com.hardForkTransition(
parent.blockHash, parent.number+1, Opt.some(pos.timestamp))
let blockCtx = BlockContext(
timestamp : pos.timestamp,
gasLimit : gasLimitsGet(com, parent),
baseFeePerGas: baseFeeGet(com, parent),
prevRandao : pos.prevRandao,
difficulty : UInt256.zero(),
coinbase : pos.feeRecipient,
excessBlobGas: calcExcessBlobGas(parent),
)
BaseVMState.new(
parent = parent,
blockCtx = blockCtx,
com = com)
proc update(xp: TxPoolRef; parent: BlockHeader) =
xp.vmState = setupVMState(xp.vmState.com, parent)
# ------------------------------------------------------------------------------
# Public functions, constructor
# ------------------------------------------------------------------------------
proc init*(xp: TxPoolRef; com: CommonRef)
{.gcsafe,raises: [CatchableError].} =
## Constructor, returns new tx-pool descriptor.
xp.startDate = getTime().utc.toTime
xp.vmState = setupVMState(com, com.db.getCanonicalHead)
xp.txDB = TxTabsRef.new
xp.lifeTime = txItemLifeTime
xp.priceBump = txPriceBump
xp.param.reset
xp.param.flags = txPoolFlags
# ------------------------------------------------------------------------------
# Public functions
# ------------------------------------------------------------------------------
proc clearAccounts*(xp: TxPoolRef) =
## Reset transaction environment, e.g. before packing a new block
xp.update(xp.vmState.parent)
# ------------------------------------------------------------------------------
# Public functions, getters
# ------------------------------------------------------------------------------
func pFlags*(xp: TxPoolRef): set[TxPoolFlags] =
## Returns the set of algorithm strategy symbols for labelling items
## as`packed`
xp.param.flags
func pDirtyBuckets*(xp: TxPoolRef): bool =
## Getter, buckets need update
xp.param.dirtyBuckets
func pDoubleCheck*(xp: TxPoolRef): seq[TxItemRef] =
## Getter, cached block chain head was moved back
xp.param.doubleCheck
func startDate*(xp: TxPoolRef): Time =
## Getter
xp.startDate
func txDB*(xp: TxPoolRef): TxTabsRef =
## Getter, pool database
xp.txDB
func baseFee*(xp: TxPoolRef): GasInt =
## Getter, baseFee for the next bock header. This value is auto-generated
## when a new insertion point is set via `head=`.
if xp.vmState.blockCtx.baseFeePerGas.isSome:
xp.vmState.blockCtx.baseFeePerGas.get.truncate(GasInt)
else:
0.GasInt
func vmState*(xp: TxPoolRef): BaseVMState =
xp.vmState
func nextFork*(xp: TxPoolRef): EVMFork =
xp.vmState.fork
func gasLimit*(xp: TxPoolRef): GasInt =
xp.vmState.blockCtx.gasLimit
func excessBlobGas*(xp: TxPoolRef): GasInt =
xp.vmState.blockCtx.excessBlobGas
proc getBalance*(xp: TxPoolRef; account: EthAddress): UInt256 =
## Wrapper around `vmState.readOnlyStateDB.getBalance()` for a `vmState`
## descriptor positioned at the `dh.head`. This might differ from the
## `dh.vmState.readOnlyStateDB.getBalance()` which returnes the current
## balance relative to what has been accumulated by the current packing
## procedure.
xp.vmState.stateDB.getBalance(account)
proc getNonce*(xp: TxPoolRef; account: EthAddress): AccountNonce =
## Wrapper around `vmState.readOnlyStateDB.getNonce()` for a `vmState`
## descriptor positioned at the `dh.head`. This might differ from the
## `dh.vmState.readOnlyStateDB.getNonce()` which returnes the current balance
## relative to what has been accumulated by the current packing procedure.
xp.vmState.stateDB.getNonce(account)
func head*(xp: TxPoolRef): BlockHeader =
## Getter, cached block chain insertion point. Typocally, this should be the
## the same header as retrieved by the `getCanonicalHead()` (unless in the
## middle of a mining update.)
xp.vmState.parent
# ------------------------------------------------------------------------------
# Public functions, setters
# ------------------------------------------------------------------------------
func `pDirtyBuckets=`*(xp: TxPoolRef; val: bool) =
## Setter
xp.param.dirtyBuckets = val
func pDoubleCheckAdd*(xp: TxPoolRef; val: seq[TxItemRef]) =
## Pseudo setter
xp.param.doubleCheck.add val
func pDoubleCheckFlush*(xp: TxPoolRef) =
## Pseudo setter
xp.param.doubleCheck.setLen(0)
func `pFlags=`*(xp: TxPoolRef; val: set[TxPoolFlags]) =
## Install a set of algorithm strategy symbols for labelling items as`packed`
xp.param.flags = val
proc `head=`*(xp: TxPoolRef; val: BlockHeader)
{.gcsafe,raises: [].} =
## Setter, updates descriptor. This setter re-positions the `vmState` and
## account caches to a new insertion point on the block chain database.
xp.update(val)
# ------------------------------------------------------------------------------
# End
# ------------------------------------------------------------------------------