2021-04-08 14:52:10 +00:00
|
|
|
# Nimbus
|
2023-01-31 12:38:08 +00:00
|
|
|
# Copyright (c) 2018-2023 Status Research & Development GmbH
|
2021-04-08 14:52:10 +00:00
|
|
|
# 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.
|
|
|
|
|
2023-02-14 20:27:17 +00:00
|
|
|
{.push raises: [].}
|
|
|
|
|
2021-04-08 14:52:10 +00:00
|
|
|
import
|
2023-08-02 10:17:40 +00:00
|
|
|
std/[options, sets, strformat, tables],
|
2023-01-31 12:38:08 +00:00
|
|
|
eth/[keys],
|
2021-07-27 11:28:05 +00:00
|
|
|
../../stateless/[witness_from_tree, witness_types],
|
2022-12-02 04:35:41 +00:00
|
|
|
../db/accounts_cache,
|
|
|
|
../common/[common, evmforks],
|
2023-03-10 22:16:42 +00:00
|
|
|
./async/data_sources,
|
2023-08-02 10:17:40 +00:00
|
|
|
./interpreter/op_codes,
|
2023-01-31 12:38:08 +00:00
|
|
|
./types
|
2021-07-27 11:28:05 +00:00
|
|
|
|
2022-01-18 16:19:32 +00:00
|
|
|
proc init(
|
2023-04-12 12:39:11 +00:00
|
|
|
self: BaseVMState;
|
|
|
|
ac: AccountsCache;
|
|
|
|
parent: BlockHeader;
|
|
|
|
timestamp: EthTime;
|
|
|
|
gasLimit: GasInt;
|
|
|
|
fee: Option[UInt256];
|
|
|
|
prevRandao: Hash256;
|
|
|
|
difficulty: UInt256;
|
|
|
|
miner: EthAddress;
|
|
|
|
com: CommonRef;
|
2023-08-02 10:17:40 +00:00
|
|
|
tracer: TracerRef,
|
2023-04-12 12:39:11 +00:00
|
|
|
asyncFactory: AsyncOperationFactory = AsyncOperationFactory(maybeDataSource: none[AsyncDataSource]()))
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe.} =
|
2022-01-18 16:19:32 +00:00
|
|
|
## Initialisation helper
|
2021-04-08 14:52:10 +00:00
|
|
|
self.prevHeaders = @[]
|
2022-01-18 16:19:32 +00:00
|
|
|
self.parent = parent
|
|
|
|
self.timestamp = timestamp
|
|
|
|
self.gasLimit = gasLimit
|
2023-04-11 08:28:45 +00:00
|
|
|
self.gasPool = gasLimit
|
2022-01-18 16:19:32 +00:00
|
|
|
self.fee = fee
|
2022-02-27 05:21:46 +00:00
|
|
|
self.prevRandao = prevRandao
|
2022-06-14 06:01:51 +00:00
|
|
|
self.blockDifficulty = difficulty
|
2022-12-02 04:35:41 +00:00
|
|
|
self.com = com
|
2022-01-18 16:19:32 +00:00
|
|
|
self.tracer = tracer
|
2021-10-28 09:42:39 +00:00
|
|
|
self.stateDB = ac
|
2022-01-18 16:19:32 +00:00
|
|
|
self.minerAddress = miner
|
2023-04-12 12:39:11 +00:00
|
|
|
self.asyncFactory = asyncFactory
|
2022-01-18 16:19:32 +00:00
|
|
|
|
|
|
|
# --------------
|
|
|
|
|
|
|
|
proc `$`*(vmState: BaseVMState): string
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe, raises: [ValueError].} =
|
2022-01-18 16:19:32 +00:00
|
|
|
if vmState.isNil:
|
|
|
|
result = "nil"
|
|
|
|
else:
|
2022-12-02 04:35:41 +00:00
|
|
|
result = &"VMState:"&
|
|
|
|
&"\n blockNumber: {vmState.parent.blockNumber + 1}"
|
2022-01-18 16:19:32 +00:00
|
|
|
|
|
|
|
proc new*(
|
|
|
|
T: type BaseVMState;
|
|
|
|
parent: BlockHeader; ## parent header, account sync position
|
|
|
|
timestamp: EthTime; ## tx env: time stamp
|
|
|
|
gasLimit: GasInt; ## tx env: gas limit
|
2022-04-08 04:54:11 +00:00
|
|
|
fee: Option[UInt256]; ## tx env: optional base fee
|
2022-02-27 05:21:46 +00:00
|
|
|
prevRandao: Hash256; ## tx env: POS block randomness
|
2022-06-14 06:01:51 +00:00
|
|
|
difficulty: UInt256, ## tx env: difficulty
|
2022-01-18 16:19:32 +00:00
|
|
|
miner: EthAddress; ## tx env: coinbase(PoW) or signer(PoA)
|
2022-12-02 04:35:41 +00:00
|
|
|
com: CommonRef; ## block chain config
|
2023-08-02 10:17:40 +00:00
|
|
|
tracer: TracerRef = nil): T
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe.} =
|
2022-01-18 16:19:32 +00:00
|
|
|
## Create a new `BaseVMState` descriptor from a parent block header. This
|
|
|
|
## function internally constructs a new account state cache rooted at
|
|
|
|
## `parent.stateRoot`
|
|
|
|
##
|
|
|
|
## This `new()` constructor and its variants (see below) provide a save
|
|
|
|
## `BaseVMState` environment where the account state cache is synchronised
|
|
|
|
## with the `parent` block header.
|
|
|
|
new result
|
|
|
|
result.init(
|
2023-08-04 11:10:09 +00:00
|
|
|
ac = AccountsCache.init(com.db, parent.stateRoot, com.pruneTrie),
|
2022-01-18 16:19:32 +00:00
|
|
|
parent = parent,
|
|
|
|
timestamp = timestamp,
|
|
|
|
gasLimit = gasLimit,
|
|
|
|
fee = fee,
|
2022-02-27 05:21:46 +00:00
|
|
|
prevRandao = prevRandao,
|
2022-06-14 06:01:51 +00:00
|
|
|
difficulty = difficulty,
|
2022-01-18 16:19:32 +00:00
|
|
|
miner = miner,
|
2022-12-02 04:35:41 +00:00
|
|
|
com = com,
|
2023-08-02 10:17:40 +00:00
|
|
|
tracer = tracer)
|
2022-01-18 16:19:32 +00:00
|
|
|
|
|
|
|
proc reinit*(self: BaseVMState; ## Object descriptor
|
|
|
|
parent: BlockHeader; ## parent header, account sync pos.
|
|
|
|
timestamp: EthTime; ## tx env: time stamp
|
|
|
|
gasLimit: GasInt; ## tx env: gas limit
|
2022-04-08 04:54:11 +00:00
|
|
|
fee: Option[UInt256]; ## tx env: optional base fee
|
2022-02-27 05:21:46 +00:00
|
|
|
prevRandao:Hash256; ## tx env: POS block randomness
|
2022-06-14 06:01:51 +00:00
|
|
|
difficulty:UInt256, ## tx env: difficulty
|
2022-01-18 16:19:32 +00:00
|
|
|
miner: EthAddress; ## tx env: coinbase(PoW) or signer(PoA)
|
2022-12-02 04:35:41 +00:00
|
|
|
): bool
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe.} =
|
2022-01-18 16:19:32 +00:00
|
|
|
## Re-initialise state descriptor. The `AccountsCache` database is
|
|
|
|
## re-initilaise only if its `rootHash` doe not point to `parent.stateRoot`,
|
|
|
|
## already. Accumulated state data are reset.
|
|
|
|
##
|
|
|
|
## This function returns `true` unless the `AccountsCache` database could be
|
|
|
|
## queries about its `rootHash`, i.e. `isTopLevelClean` evaluated `true`. If
|
|
|
|
## this function returns `false`, the function argument `self` is left
|
|
|
|
## untouched.
|
|
|
|
if self.stateDB.isTopLevelClean:
|
|
|
|
let
|
|
|
|
tracer = self.tracer
|
2022-12-02 04:35:41 +00:00
|
|
|
com = self.com
|
|
|
|
db = com.db
|
2022-01-18 16:19:32 +00:00
|
|
|
ac = if self.stateDB.rootHash == parent.stateRoot: self.stateDB
|
2023-08-04 11:10:09 +00:00
|
|
|
else: AccountsCache.init(db, parent.stateRoot, com.pruneTrie)
|
2022-01-18 16:19:32 +00:00
|
|
|
self[].reset
|
|
|
|
self.init(
|
|
|
|
ac = ac,
|
|
|
|
parent = parent,
|
|
|
|
timestamp = timestamp,
|
|
|
|
gasLimit = gasLimit,
|
|
|
|
fee = fee,
|
2022-02-27 05:21:46 +00:00
|
|
|
prevRandao = prevRandao,
|
2022-06-14 06:01:51 +00:00
|
|
|
difficulty = difficulty,
|
2022-01-18 16:19:32 +00:00
|
|
|
miner = miner,
|
2022-12-02 04:35:41 +00:00
|
|
|
com = com,
|
2022-01-18 16:19:32 +00:00
|
|
|
tracer = tracer)
|
|
|
|
return true
|
|
|
|
# else: false
|
|
|
|
|
|
|
|
proc reinit*(self: BaseVMState; ## Object descriptor
|
|
|
|
parent: BlockHeader; ## parent header, account sync pos.
|
|
|
|
header: BlockHeader; ## header with tx environment data fields
|
2022-12-02 04:35:41 +00:00
|
|
|
): bool
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe, raises: [CatchableError].} =
|
2022-01-18 16:19:32 +00:00
|
|
|
## Variant of `reinit()`. The `parent` argument is used to sync the accounts
|
|
|
|
## cache and the `header` is used as a container to pass the `timestamp`,
|
|
|
|
## `gasLimit`, and `fee` values.
|
|
|
|
##
|
|
|
|
## It requires the `header` argument properly initalised so that for PoA
|
|
|
|
## networks, the miner address is retrievable via `ecRecover()`.
|
2022-03-15 17:21:41 +00:00
|
|
|
result = self.reinit(
|
2022-01-18 16:19:32 +00:00
|
|
|
parent = parent,
|
|
|
|
timestamp = header.timestamp,
|
|
|
|
gasLimit = header.gasLimit,
|
|
|
|
fee = header.fee,
|
2022-02-27 05:21:46 +00:00
|
|
|
prevRandao= header.prevRandao,
|
2022-06-14 06:01:51 +00:00
|
|
|
difficulty= header.difficulty,
|
2022-12-02 04:35:41 +00:00
|
|
|
miner = self.com.minerAddress(header))
|
2022-01-18 16:19:32 +00:00
|
|
|
|
|
|
|
proc reinit*(self: BaseVMState; ## Object descriptor
|
|
|
|
header: BlockHeader; ## header with tx environment data fields
|
2022-12-02 04:35:41 +00:00
|
|
|
): bool
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe, raises: [CatchableError].} =
|
2022-01-18 16:19:32 +00:00
|
|
|
## This is a variant of the `reinit()` function above where the field
|
|
|
|
## `header.parentHash`, is used to fetch the `parent` BlockHeader to be
|
|
|
|
## used in the `update()` variant, above.
|
2022-07-21 12:14:41 +00:00
|
|
|
var parent: BlockHeader
|
2022-12-02 04:35:41 +00:00
|
|
|
if self.com.db.getBlockHeader(header.parentHash, parent):
|
2022-07-21 12:14:41 +00:00
|
|
|
return self.reinit(
|
|
|
|
parent = parent,
|
2022-12-02 04:35:41 +00:00
|
|
|
header = header)
|
2022-01-18 16:19:32 +00:00
|
|
|
|
|
|
|
proc init*(
|
2023-08-02 10:17:40 +00:00
|
|
|
self: BaseVMState; ## Object descriptor
|
|
|
|
parent: BlockHeader; ## parent header, account sync position
|
|
|
|
header: BlockHeader; ## header with tx environment data fields
|
|
|
|
com: CommonRef; ## block chain config
|
|
|
|
tracer: TracerRef = nil)
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe, raises: [CatchableError].} =
|
2022-01-18 16:19:32 +00:00
|
|
|
## Variant of `new()` constructor above for in-place initalisation. The
|
|
|
|
## `parent` argument is used to sync the accounts cache and the `header`
|
|
|
|
## is used as a container to pass the `timestamp`, `gasLimit`, and `fee`
|
|
|
|
## values.
|
|
|
|
##
|
|
|
|
## It requires the `header` argument properly initalised so that for PoA
|
|
|
|
## networks, the miner address is retrievable via `ecRecover()`.
|
2022-03-15 17:21:41 +00:00
|
|
|
self.init(
|
2023-08-04 11:10:09 +00:00
|
|
|
ac = AccountsCache.init(com.db, parent.stateRoot, com.pruneTrie),
|
2022-03-15 17:21:41 +00:00
|
|
|
parent = parent,
|
|
|
|
timestamp = header.timestamp,
|
|
|
|
gasLimit = header.gasLimit,
|
|
|
|
fee = header.fee,
|
|
|
|
prevRandao = header.prevRandao,
|
2022-06-14 06:01:51 +00:00
|
|
|
difficulty = header.difficulty,
|
2022-12-02 04:35:41 +00:00
|
|
|
miner = com.minerAddress(header),
|
|
|
|
com = com,
|
2023-08-02 10:17:40 +00:00
|
|
|
tracer = tracer)
|
2022-01-18 16:19:32 +00:00
|
|
|
|
|
|
|
proc new*(
|
2023-08-02 10:17:40 +00:00
|
|
|
T: type BaseVMState;
|
|
|
|
parent: BlockHeader; ## parent header, account sync position
|
|
|
|
header: BlockHeader; ## header with tx environment data fields
|
|
|
|
com: CommonRef; ## block chain config
|
|
|
|
tracer: TracerRef = nil): T
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe, raises: [CatchableError].} =
|
2022-01-18 16:19:32 +00:00
|
|
|
## This is a variant of the `new()` constructor above where the `parent`
|
|
|
|
## argument is used to sync the accounts cache and the `header` is used
|
|
|
|
## as a container to pass the `timestamp`, `gasLimit`, and `fee` values.
|
|
|
|
##
|
|
|
|
## It requires the `header` argument properly initalised so that for PoA
|
|
|
|
## networks, the miner address is retrievable via `ecRecover()`.
|
2021-04-08 14:52:10 +00:00
|
|
|
new result
|
2022-01-18 16:19:32 +00:00
|
|
|
result.init(
|
2023-08-02 10:17:40 +00:00
|
|
|
parent = parent,
|
|
|
|
header = header,
|
|
|
|
com = com,
|
|
|
|
tracer = tracer)
|
2022-01-18 16:19:32 +00:00
|
|
|
|
|
|
|
proc new*(
|
2023-08-02 10:17:40 +00:00
|
|
|
T: type BaseVMState;
|
|
|
|
header: BlockHeader; ## header with tx environment data fields
|
|
|
|
com: CommonRef; ## block chain config
|
|
|
|
tracer: TracerRef = nil): T
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe, raises: [CatchableError].} =
|
2022-01-18 16:19:32 +00:00
|
|
|
## This is a variant of the `new()` constructor above where the field
|
|
|
|
## `header.parentHash`, is used to fetch the `parent` BlockHeader to be
|
|
|
|
## used in the `new()` variant, above.
|
|
|
|
BaseVMState.new(
|
2023-08-02 10:17:40 +00:00
|
|
|
parent = com.db.getBlockHeader(header.parentHash),
|
|
|
|
header = header,
|
|
|
|
com = com,
|
|
|
|
tracer = tracer)
|
2022-01-18 16:19:32 +00:00
|
|
|
|
2022-07-21 12:14:41 +00:00
|
|
|
proc init*(
|
2023-08-02 10:17:40 +00:00
|
|
|
vmState: BaseVMState;
|
|
|
|
header: BlockHeader; ## header with tx environment data fields
|
|
|
|
com: CommonRef; ## block chain config
|
|
|
|
tracer: TracerRef = nil): bool
|
2023-01-31 12:38:08 +00:00
|
|
|
{.gcsafe, raises: [CatchableError].} =
|
2022-07-21 12:14:41 +00:00
|
|
|
## Variant of `new()` which does not throw an exception on a dangling
|
|
|
|
## `BlockHeader` parent hash reference.
|
|
|
|
var parent: BlockHeader
|
2022-12-02 04:35:41 +00:00
|
|
|
if com.db.getBlockHeader(header.parentHash, parent):
|
2022-07-21 12:14:41 +00:00
|
|
|
vmState.init(
|
2023-08-02 10:17:40 +00:00
|
|
|
parent = parent,
|
|
|
|
header = header,
|
|
|
|
com = com,
|
|
|
|
tracer = tracer)
|
2022-07-21 12:14:41 +00:00
|
|
|
return true
|
|
|
|
|
2023-04-12 12:39:11 +00:00
|
|
|
proc statelessInit*(
|
2023-05-10 16:04:35 +00:00
|
|
|
vmState: BaseVMState;
|
|
|
|
parent: BlockHeader; ## parent header, account sync position
|
|
|
|
header: BlockHeader; ## header with tx environment data fields
|
|
|
|
com: CommonRef; ## block chain config
|
|
|
|
asyncFactory: AsyncOperationFactory;
|
2023-08-02 10:17:40 +00:00
|
|
|
tracer: TracerRef = nil): bool
|
2023-05-10 16:04:35 +00:00
|
|
|
{.gcsafe, raises: [CatchableError].} =
|
2023-04-12 12:39:11 +00:00
|
|
|
vmState.init(
|
2023-08-04 11:10:09 +00:00
|
|
|
ac = AccountsCache.init(com.db, parent.stateRoot, com.pruneTrie),
|
2023-04-12 12:39:11 +00:00
|
|
|
parent = parent,
|
|
|
|
timestamp = header.timestamp,
|
|
|
|
gasLimit = header.gasLimit,
|
|
|
|
fee = header.fee,
|
|
|
|
prevRandao = header.prevRandao,
|
|
|
|
difficulty = header.difficulty,
|
|
|
|
miner = com.minerAddress(header),
|
|
|
|
com = com,
|
|
|
|
tracer = tracer,
|
|
|
|
asyncFactory = asyncFactory)
|
|
|
|
return true
|
|
|
|
|
2021-04-08 14:52:10 +00:00
|
|
|
method coinbase*(vmState: BaseVMState): EthAddress {.base, gcsafe.} =
|
|
|
|
vmState.minerAddress
|
|
|
|
|
|
|
|
method blockNumber*(vmState: BaseVMState): BlockNumber {.base, gcsafe.} =
|
|
|
|
# it should return current block number
|
|
|
|
# and not head.blockNumber
|
2022-01-18 16:19:32 +00:00
|
|
|
vmState.parent.blockNumber + 1
|
2021-04-08 14:52:10 +00:00
|
|
|
|
|
|
|
method difficulty*(vmState: BaseVMState): UInt256 {.base, gcsafe.} =
|
2022-12-02 04:35:41 +00:00
|
|
|
if vmState.com.consensus == ConsensusType.POS:
|
2022-02-05 09:15:50 +00:00
|
|
|
# EIP-4399/EIP-3675
|
2022-02-27 05:21:46 +00:00
|
|
|
UInt256.fromBytesBE(vmState.prevRandao.data, allowPadding = false)
|
2022-02-05 09:15:50 +00:00
|
|
|
else:
|
2022-06-14 06:01:51 +00:00
|
|
|
vmState.blockDifficulty
|
2021-04-08 14:52:10 +00:00
|
|
|
|
2021-06-27 13:19:22 +00:00
|
|
|
method baseFee*(vmState: BaseVMState): UInt256 {.base, gcsafe.} =
|
2022-01-18 16:19:32 +00:00
|
|
|
if vmState.fee.isSome:
|
|
|
|
vmState.fee.get
|
|
|
|
else:
|
|
|
|
0.u256
|
2021-06-27 13:19:22 +00:00
|
|
|
|
2021-04-08 14:52:10 +00:00
|
|
|
when defined(geth):
|
|
|
|
import db/geth_db
|
|
|
|
|
2023-01-31 12:38:08 +00:00
|
|
|
method getAncestorHash*(
|
|
|
|
vmState: BaseVMState, blockNumber: BlockNumber):
|
|
|
|
Hash256 {.base, gcsafe, raises: [CatchableError].} =
|
2022-12-02 04:35:41 +00:00
|
|
|
let db = vmState.com.db
|
2021-04-08 14:52:10 +00:00
|
|
|
when defined(geth):
|
2022-12-02 04:35:41 +00:00
|
|
|
result = db.headerHash(blockNumber.truncate(uint64))
|
2021-04-08 14:52:10 +00:00
|
|
|
else:
|
2022-12-02 04:35:41 +00:00
|
|
|
result = db.getBlockHash(blockNumber)
|
|
|
|
|
2021-04-08 14:52:10 +00:00
|
|
|
#TODO: should we use deque here?
|
|
|
|
# someday we may revive this code when
|
|
|
|
# we already have working miner
|
|
|
|
when false:
|
|
|
|
let idx = ancestorDepth.toInt
|
|
|
|
if idx >= vmState.prevHeaders.len:
|
|
|
|
return
|
|
|
|
|
|
|
|
var header = vmState.prevHeaders[idx]
|
|
|
|
result = header.hash
|
|
|
|
|
|
|
|
proc readOnlyStateDB*(vmState: BaseVMState): ReadOnlyStateDB {.inline.} =
|
2021-10-28 09:42:39 +00:00
|
|
|
ReadOnlyStateDB(vmState.stateDB)
|
2021-04-08 14:52:10 +00:00
|
|
|
|
|
|
|
template mutateStateDB*(vmState: BaseVMState, body: untyped) =
|
|
|
|
block:
|
2021-10-28 09:42:39 +00:00
|
|
|
var db {.inject.} = vmState.stateDB
|
2021-04-08 14:52:10 +00:00
|
|
|
body
|
|
|
|
|
|
|
|
proc getAndClearLogEntries*(vmState: BaseVMState): seq[Log] =
|
2023-03-20 11:51:09 +00:00
|
|
|
vmState.stateDB.getAndClearLogEntries()
|
2021-04-08 14:52:10 +00:00
|
|
|
|
2022-01-18 16:19:32 +00:00
|
|
|
proc status*(vmState: BaseVMState): bool =
|
2021-04-08 14:52:10 +00:00
|
|
|
ExecutionOK in vmState.flags
|
|
|
|
|
|
|
|
proc `status=`*(vmState: BaseVMState, status: bool) =
|
|
|
|
if status: vmState.flags.incl ExecutionOK
|
|
|
|
else: vmState.flags.excl ExecutionOK
|
|
|
|
|
2022-01-18 16:19:32 +00:00
|
|
|
proc generateWitness*(vmState: BaseVMState): bool =
|
2021-04-08 14:52:10 +00:00
|
|
|
GenerateWitness in vmState.flags
|
|
|
|
|
|
|
|
proc `generateWitness=`*(vmState: BaseVMState, status: bool) =
|
|
|
|
if status: vmState.flags.incl GenerateWitness
|
|
|
|
else: vmState.flags.excl GenerateWitness
|
|
|
|
|
2022-01-18 16:19:32 +00:00
|
|
|
proc buildWitness*(vmState: BaseVMState): seq[byte]
|
2023-01-31 12:38:08 +00:00
|
|
|
{.raises: [CatchableError].} =
|
2021-10-28 09:42:39 +00:00
|
|
|
let rootHash = vmState.stateDB.rootHash
|
|
|
|
let mkeys = vmState.stateDB.makeMultiKeys()
|
2022-04-08 04:54:11 +00:00
|
|
|
let flags = if vmState.fork >= FkSpurious: {wfEIP170} else: {}
|
2021-04-08 14:52:10 +00:00
|
|
|
|
|
|
|
# build witness from tree
|
2023-08-04 11:10:09 +00:00
|
|
|
var wb = initWitnessBuilder(vmState.com.db, rootHash, flags)
|
2023-02-14 20:27:17 +00:00
|
|
|
wb.buildWitness(mkeys)
|
2023-02-16 11:40:07 +00:00
|
|
|
|
|
|
|
func forkDeterminationInfoForVMState*(vmState: BaseVMState): ForkDeterminationInfo =
|
|
|
|
# FIXME-Adam: Is this timestamp right? Note that up above in blockNumber we add 1;
|
|
|
|
# should timestamp be adding 12 or something?
|
|
|
|
# Also, can I get the TD? Do I need to?
|
|
|
|
forkDeterminationInfo(vmState.blockNumber, vmState.timestamp)
|
2023-03-16 20:34:47 +00:00
|
|
|
|
2023-03-17 18:16:24 +00:00
|
|
|
func determineFork*(vmState: BaseVMState): EVMFork =
|
|
|
|
vmState.com.toEVMFork(vmState.forkDeterminationInfoForVMState)
|
2023-08-02 10:17:40 +00:00
|
|
|
|
|
|
|
func tracingEnabled*(vmState: BaseVMState): bool =
|
|
|
|
vmState.tracer.isNil.not
|
|
|
|
|
|
|
|
proc captureTxStart*(vmState: BaseVMState, gasLimit: GasInt) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.captureTxStart(gasLimit)
|
|
|
|
|
|
|
|
proc captureTxEnd*(vmState: BaseVMState, restGas: GasInt) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.captureTxEnd(restGas)
|
|
|
|
|
|
|
|
proc captureStart*(vmState: BaseVMState, c: Computation,
|
|
|
|
sender: EthAddress, to: EthAddress,
|
|
|
|
create: bool, input: openArray[byte],
|
|
|
|
gas: GasInt, value: UInt256) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.captureStart(c, sender, to, create, input, gas, value)
|
|
|
|
|
|
|
|
proc captureEnd*(vmState: BaseVMState, output: openArray[byte],
|
|
|
|
gasUsed: GasInt, error: Option[string]) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.captureEnd(output, gasUsed, error)
|
|
|
|
|
|
|
|
proc captureEnter*(vmState: BaseVMState, op: Op,
|
|
|
|
sender: EthAddress, to: EthAddress,
|
|
|
|
input: openArray[byte], gas: GasInt,
|
|
|
|
value: UInt256) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.captureEnter(op, sender, to, input, gas, value)
|
|
|
|
|
|
|
|
proc captureExit*(vmState: BaseVMState, output: openArray[byte],
|
|
|
|
gasUsed: GasInt, error: Option[string]) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.captureExit(output, gasUsed, error)
|
|
|
|
|
|
|
|
proc captureOpStart*(vmState: BaseVMState, pc: int,
|
|
|
|
op: Op, gas: GasInt,
|
|
|
|
depth: int): int =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
result = vmState.tracer.captureOpStart(pc, op, gas, depth)
|
|
|
|
|
|
|
|
proc captureOpEnd*(vmState: BaseVMState, pc: int,
|
|
|
|
op: Op, gas: GasInt, refund: GasInt,
|
|
|
|
rData: openArray[byte],
|
|
|
|
depth: int, opIndex: int) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.captureOpEnd(pc, op, gas, refund, rData, depth, opIndex)
|
|
|
|
|
|
|
|
proc captureFault*(vmState: BaseVMState, pc: int,
|
|
|
|
op: Op, gas: GasInt, refund: GasInt,
|
|
|
|
rData: openArray[byte],
|
|
|
|
depth: int, error: Option[string]) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.captureFault(pc, op, gas, refund, rData, depth, error)
|
|
|
|
|
|
|
|
proc capturePrepare*(vmState: BaseVMState, depth: int) =
|
|
|
|
if vmState.tracingEnabled:
|
|
|
|
vmState.tracer.capturePrepare(depth)
|