mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-27 04:26:07 +00:00
EVMC: Using {.show.}
trace all calls from EVM to host services
When `show_tx_calls` is manually set to true, show all the calls from the EVM to the host, including name, arguments and results. For example this shows each call to `setStorage`, the key, value and storage result. This output allows the externally-visible activity of an EVM to be seen, and it's been useful for guessing what went wrong when a test fails. In theory, if two EVMs show the same activity in this log, they should have the same effect on account states, gas, etc. and the same final `roothash` (which is the only value some tests check). ps. Ideally we'd use `{.push show.}`...`{.pop.}`, just like with `inline`. But we can't: https://github.com/nim-lang/Nim/issues/12867 Signed-off-by: Jamie Lokier <jamie@shareable.org>
This commit is contained in:
parent
43b66a3a05
commit
ffd34a69fe
@ -12,7 +12,7 @@ import
|
|||||||
sets, times, stint, chronicles,
|
sets, times, stint, chronicles,
|
||||||
eth/common/eth_types, ../db/accounts_cache, ../forks,
|
eth/common/eth_types, ../db/accounts_cache, ../forks,
|
||||||
".."/[vm_types, vm_state, vm_computation, vm_internals],
|
".."/[vm_types, vm_state, vm_computation, vm_internals],
|
||||||
./host_types
|
./host_types, ./host_trace
|
||||||
|
|
||||||
proc setupTxContext(host: TransactionHost) =
|
proc setupTxContext(host: TransactionHost) =
|
||||||
# Conversion issues:
|
# Conversion issues:
|
||||||
@ -74,7 +74,7 @@ const use_evmc_glue = defined(evmc_enabled)
|
|||||||
when use_evmc_glue:
|
when use_evmc_glue:
|
||||||
{.push inline.}
|
{.push inline.}
|
||||||
|
|
||||||
proc accountExists(host: TransactionHost, address: HostAddress): bool =
|
proc accountExists(host: TransactionHost, address: HostAddress): bool {.show.} =
|
||||||
if host.vmState.fork >= FkSpurious:
|
if host.vmState.fork >= FkSpurious:
|
||||||
not host.vmState.readOnlyStateDB.isDeadAccount(address)
|
not host.vmState.readOnlyStateDB.isDeadAccount(address)
|
||||||
else:
|
else:
|
||||||
@ -84,7 +84,7 @@ proc accountExists(host: TransactionHost, address: HostAddress): bool =
|
|||||||
# `selfDestruct`, if an EVM is only allowed to do these things to its own
|
# `selfDestruct`, if an EVM is only allowed to do these things to its own
|
||||||
# contract account and the host always knows which account?
|
# contract account and the host always knows which account?
|
||||||
|
|
||||||
proc getStorage(host: TransactionHost, address: HostAddress, key: HostKey): HostValue =
|
proc getStorage(host: TransactionHost, address: HostAddress, key: HostKey): HostValue {.show.} =
|
||||||
host.vmState.readOnlyStateDB.getStorage(address, key)
|
host.vmState.readOnlyStateDB.getStorage(address, key)
|
||||||
|
|
||||||
proc setStorage1(host: TransactionHost, address: HostAddress,
|
proc setStorage1(host: TransactionHost, address: HostAddress,
|
||||||
@ -111,7 +111,7 @@ proc setStorage1(host: TransactionHost, address: HostAddress,
|
|||||||
return EVMC_STORAGE_MODIFIED
|
return EVMC_STORAGE_MODIFIED
|
||||||
|
|
||||||
proc setStorage(host: TransactionHost, address: HostAddress,
|
proc setStorage(host: TransactionHost, address: HostAddress,
|
||||||
key: HostKey, value: HostValue): EvmcStorageStatus =
|
key: HostKey, value: HostValue): EvmcStorageStatus {.show.} =
|
||||||
let status = setStorage1(host, address, key, value)
|
let status = setStorage1(host, address, key, value)
|
||||||
let gasParam = GasParams(kind: Op.Sstore,
|
let gasParam = GasParams(kind: Op.Sstore,
|
||||||
s_status: status,
|
s_status: status,
|
||||||
@ -121,15 +121,15 @@ proc setStorage(host: TransactionHost, address: HostAddress,
|
|||||||
if gasRefund != 0:
|
if gasRefund != 0:
|
||||||
host.computation.gasMeter.refundGas(gasRefund)
|
host.computation.gasMeter.refundGas(gasRefund)
|
||||||
|
|
||||||
proc getBalance(host: TransactionHost, address: HostAddress): HostBalance =
|
proc getBalance(host: TransactionHost, address: HostAddress): HostBalance {.show.} =
|
||||||
host.vmState.readOnlyStateDB.getBalance(address)
|
host.vmState.readOnlyStateDB.getBalance(address)
|
||||||
|
|
||||||
proc getCodeSize(host: TransactionHost, address: HostAddress): HostSize =
|
proc getCodeSize(host: TransactionHost, address: HostAddress): HostSize {.show.} =
|
||||||
# TODO: Check this `HostSize`, it was copied as `uint` from other code.
|
# TODO: Check this `HostSize`, it was copied as `uint` from other code.
|
||||||
# Note: Old `evmc_host` uses `getCode(address).len` instead.
|
# Note: Old `evmc_host` uses `getCode(address).len` instead.
|
||||||
host.vmState.readOnlyStateDB.getCodeSize(address).HostSize
|
host.vmState.readOnlyStateDB.getCodeSize(address).HostSize
|
||||||
|
|
||||||
proc getCodeHash(host: TransactionHost, address: HostAddress): HostHash =
|
proc getCodeHash(host: TransactionHost, address: HostAddress): HostHash {.show.} =
|
||||||
let db = host.vmState.readOnlyStateDB
|
let db = host.vmState.readOnlyStateDB
|
||||||
# TODO: Copied from `Computation`, but check if that code is wrong with
|
# TODO: Copied from `Computation`, but check if that code is wrong with
|
||||||
# `FkSpurious`, as it has different calls from `accountExists` above.
|
# `FkSpurious`, as it has different calls from `accountExists` above.
|
||||||
@ -140,7 +140,7 @@ proc getCodeHash(host: TransactionHost, address: HostAddress): HostHash =
|
|||||||
|
|
||||||
proc copyCode(host: TransactionHost, address: HostAddress,
|
proc copyCode(host: TransactionHost, address: HostAddress,
|
||||||
code_offset: HostSize, buffer_data: ptr byte,
|
code_offset: HostSize, buffer_data: ptr byte,
|
||||||
buffer_size: HostSize): HostSize =
|
buffer_size: HostSize): HostSize {.show.} =
|
||||||
# We must handle edge cases carefully to prevent overflows. `len` is signed
|
# We must handle edge cases carefully to prevent overflows. `len` is signed
|
||||||
# type `int`, but `code_offset` and `buffer_size` are _unsigned_, and may
|
# type `int`, but `code_offset` and `buffer_size` are _unsigned_, and may
|
||||||
# have large values (deliberately if attacked) that exceed the range of `int`.
|
# have large values (deliberately if attacked) that exceed the range of `int`.
|
||||||
@ -168,7 +168,7 @@ proc copyCode(host: TransactionHost, address: HostAddress,
|
|||||||
copyMem(buffer_data, code[safe_offset].addr, safe_len)
|
copyMem(buffer_data, code[safe_offset].addr, safe_len)
|
||||||
return safe_len.HostSize
|
return safe_len.HostSize
|
||||||
|
|
||||||
proc selfDestruct(host: TransactionHost, address, beneficiary: HostAddress) =
|
proc selfDestruct(host: TransactionHost, address, beneficiary: HostAddress) {.show.} =
|
||||||
host.vmState.mutateStateDB:
|
host.vmState.mutateStateDB:
|
||||||
let closingBalance = db.getBalance(address)
|
let closingBalance = db.getBalance(address)
|
||||||
let beneficiaryBalance = db.getBalance(beneficiary)
|
let beneficiaryBalance = db.getBalance(beneficiary)
|
||||||
@ -184,23 +184,23 @@ proc selfDestruct(host: TransactionHost, address, beneficiary: HostAddress) =
|
|||||||
host.touchedAccounts.incl(beneficiary)
|
host.touchedAccounts.incl(beneficiary)
|
||||||
host.selfDestructs.incl(address)
|
host.selfDestructs.incl(address)
|
||||||
|
|
||||||
proc call(host: TransactionHost, msg: EvmcMessage): EvmcResult =
|
proc call(host: TransactionHost, msg: EvmcMessage): EvmcResult {.show.} =
|
||||||
echo "**** Nested call not implemented ****"
|
echo "**** Nested call not implemented ****"
|
||||||
return EvmcResult(status_code: EVMC_REJECTED)
|
return EvmcResult(status_code: EVMC_REJECTED)
|
||||||
|
|
||||||
proc getTxContext(host: TransactionHost): EvmcTxContext =
|
proc getTxContext(host: TransactionHost): EvmcTxContext {.show.} =
|
||||||
if not host.cachedTxContext:
|
if not host.cachedTxContext:
|
||||||
host.setupTxContext()
|
host.setupTxContext()
|
||||||
host.cachedTxContext = true
|
host.cachedTxContext = true
|
||||||
return host.txContext
|
return host.txContext
|
||||||
|
|
||||||
proc getBlockHash(host: TransactionHost, number: HostBlockNumber): HostHash =
|
proc getBlockHash(host: TransactionHost, number: HostBlockNumber): HostHash {.show.} =
|
||||||
# TODO: Clean up the different messy block number types.
|
# TODO: Clean up the different messy block number types.
|
||||||
host.vmState.getAncestorHash(number.toBlockNumber)
|
host.vmState.getAncestorHash(number.toBlockNumber)
|
||||||
|
|
||||||
proc emitLog(host: TransactionHost, address: HostAddress,
|
proc emitLog(host: TransactionHost, address: HostAddress,
|
||||||
data: ptr byte, data_size: HostSize,
|
data: ptr byte, data_size: HostSize,
|
||||||
topics: ptr HostTopic, topics_count: HostSize) =
|
topics: ptr HostTopic, topics_count: HostSize) {.show.} =
|
||||||
var log: Log
|
var log: Log
|
||||||
# Note, this assumes the EVM ensures `data_size` and `topics_count` cannot be
|
# Note, this assumes the EVM ensures `data_size` and `topics_count` cannot be
|
||||||
# unreasonably large values. Largest `topics_count` should be 4 according to
|
# unreasonably large values. Largest `topics_count` should be 4 according to
|
||||||
|
Loading…
x
Reference in New Issue
Block a user