2018-04-06 16:52:10 +02:00
|
|
|
# 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.
|
|
|
|
|
2018-01-16 19:05:20 +02:00
|
|
|
import
|
2020-04-15 18:09:49 +07:00
|
|
|
chronicles, strformat, macros, options, times,
|
2020-07-21 13:15:06 +07:00
|
|
|
sets, eth/[common, keys],
|
2019-11-13 21:49:39 +07:00
|
|
|
../constants, ../errors, ../vm_state, ../vm_types,
|
Refactor interpreter dispatch (#65)
* move forks constants, rename errors
* Move vm/utils to vm/interpreter/utils
* initial opcodes refactoring
* Add refactored Comparison & Bitwise Logic Operations
* Add sha3 and address, simplify macro, support pop 0
* balance, origin, caller, callValue
* fix gas copy opcodes gas costs, add callDataLoad/Size/Copy, CodeSize/Copy and gas price opcode
* Update with 30s, 40s, 50s opcodes + impl of balance + stack improvement
* add push, dup, swap, log, create and call operations
* finish opcode implementation
* Add the new dispatching logic
* Pass the opcode test
* Make test_vm_json compile
* halt execution without exceptions for Return, Revert, selfdestruct (fix #62)
* Properly catch and recover from EVM exceptions (stack underflow ...)
* Fix byte op
* Fix jump regressions
* Update for latest devel, don't import old dispatch code as quasiBoolean macro is broken by latest devel
* Fix sha3 regression on empty memory slice and until end of range slice
* Fix padding / range error on expXY_success (gas computation left)
* update logging procs
* Add tracing - expXY_success is not a regression, sload stub was accidentally passing the test
* Reuse the same stub as OO implementation
* Delete previous opcode implementation
* Delete object oriented fork code
* Delete exceptions that were used as control flows
* delete base.nim :fire:, yet another OO remnants
* Delete opcode table
* Enable omputed gotos and compile-time gas fees
* Revert const gasCosts -> generates SIGSEGV
* inline push, swap and dup opcodes
* loggers are now template again, why does this pass new tests?
* Trigger CI rebuild after rocksdb fix https://github.com/status-im/nim-rocksdb/pull/5
* Address review comment on "push" + VMTests in debug mode (not release)
* Address review comment: don't tag fork by default, make opcode impl grepable
* Static compilation fixes after rebasing
* fix the initialization of the VM database
* add a missing import
* Deactivate balance and sload test following #59
* Reactivate stack check (deactivated in #59, necessary to pass tests)
* Merge remaining opcodes implementation from #59
* Merge callDataLoad and codeCopy fixes, todo simplify see #67
2018-07-06 09:52:31 +02:00
|
|
|
./interpreter/[opcode_values, gas_meter, gas_costs, vm_forks],
|
2020-05-30 10:14:59 +07:00
|
|
|
./code_stream, ./memory, ./message, ./stack, ../db/[accounts_cache, db_chain],
|
2020-07-21 13:15:06 +07:00
|
|
|
../utils/header, precompiles,
|
2020-01-20 19:55:29 +07:00
|
|
|
transaction_tracer, ../utils
|
2020-01-17 21:54:28 +07:00
|
|
|
|
2020-07-21 15:12:59 +07:00
|
|
|
when defined(chronicles_log_level):
|
|
|
|
import stew/byteutils
|
|
|
|
|
2020-01-17 21:54:28 +07:00
|
|
|
when defined(evmc_enabled):
|
2020-07-21 13:15:06 +07:00
|
|
|
import evmc/evmc, evmc_helpers, evmc_api, stew/ranges/ptr_arith
|
2018-05-24 12:01:59 +02:00
|
|
|
|
2018-08-22 21:38:00 -06:00
|
|
|
logScope:
|
|
|
|
topics = "vm computation"
|
|
|
|
|
2020-01-16 17:23:51 +07:00
|
|
|
const
|
2020-01-16 23:22:43 +07:00
|
|
|
evmc_enabled* = defined(evmc_enabled)
|
2020-01-16 17:23:51 +07:00
|
|
|
|
|
|
|
template getCoinbase*(c: Computation): EthAddress =
|
|
|
|
when evmc_enabled:
|
2020-01-22 14:47:27 +07:00
|
|
|
c.host.getTxContext().block_coinbase
|
2020-01-16 17:23:51 +07:00
|
|
|
else:
|
|
|
|
c.vmState.coinbase
|
|
|
|
|
|
|
|
template getTimestamp*(c: Computation): int64 =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.getTxContext().block_timestamp
|
|
|
|
else:
|
|
|
|
c.vmState.timestamp.toUnix
|
|
|
|
|
|
|
|
template getBlockNumber*(c: Computation): Uint256 =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.getTxContext().block_number.u256
|
|
|
|
else:
|
|
|
|
c.vmState.blockNumber.blockNumberToVmWord
|
|
|
|
|
|
|
|
template getDifficulty*(c: Computation): DifficultyInt =
|
|
|
|
when evmc_enabled:
|
|
|
|
Uint256.fromEvmc c.host.getTxContext().block_difficulty
|
|
|
|
else:
|
|
|
|
c.vmState.difficulty
|
|
|
|
|
|
|
|
template getGasLimit*(c: Computation): GasInt =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.getTxContext().block_gas_limit.GasInt
|
|
|
|
else:
|
|
|
|
c.vmState.gasLimit
|
|
|
|
|
|
|
|
template getChainId*(c: Computation): uint =
|
|
|
|
when evmc_enabled:
|
|
|
|
Uint256.fromEvmc(c.host.getTxContext().chain_id).truncate(uint)
|
|
|
|
else:
|
2021-02-13 16:32:48 +07:00
|
|
|
c.vmState.chaindb.config.chainId.uint
|
2020-01-16 17:23:51 +07:00
|
|
|
|
|
|
|
template getOrigin*(c: Computation): EthAddress =
|
|
|
|
when evmc_enabled:
|
2020-01-22 14:47:27 +07:00
|
|
|
c.host.getTxContext().tx_origin
|
2020-01-16 17:23:51 +07:00
|
|
|
else:
|
|
|
|
c.vmState.txOrigin
|
|
|
|
|
|
|
|
template getGasPrice*(c: Computation): GasInt =
|
|
|
|
when evmc_enabled:
|
|
|
|
Uint256.fromEvmc(c.host.getTxContext().tx_gas_price).truncate(GasInt)
|
|
|
|
else:
|
|
|
|
c.vmState.txGasPrice
|
|
|
|
|
|
|
|
template getBlockHash*(c: Computation, blockNumber: Uint256): Hash256 =
|
2020-01-16 21:34:16 +07:00
|
|
|
when evmc_enabled:
|
|
|
|
c.host.getBlockHash(blockNumber)
|
|
|
|
else:
|
2020-01-16 17:23:51 +07:00
|
|
|
c.vmState.getAncestorHash(blockNumber.vmWordToBlockNumber)
|
|
|
|
|
2020-01-16 21:43:29 +07:00
|
|
|
template accountExists*(c: Computation, address: EthAddress): bool =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.accountExists(address)
|
|
|
|
else:
|
2020-01-17 18:48:14 +07:00
|
|
|
if c.fork >= FkSpurious:
|
|
|
|
not c.vmState.readOnlyStateDB.isDeadAccount(address)
|
|
|
|
else:
|
|
|
|
c.vmState.readOnlyStateDB.accountExists(address)
|
2020-01-16 21:43:29 +07:00
|
|
|
|
2020-01-16 21:56:59 +07:00
|
|
|
template getStorage*(c: Computation, slot: Uint256): Uint256 =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.getStorage(c.msg.contractAddress, slot)
|
|
|
|
else:
|
2020-05-30 10:14:59 +07:00
|
|
|
c.vmState.readOnlyStateDB.getStorage(c.msg.contractAddress, slot)
|
2020-01-16 21:56:59 +07:00
|
|
|
|
2020-01-16 22:07:04 +07:00
|
|
|
template getBalance*(c: Computation, address: EthAddress): Uint256 =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.getBalance(address)
|
|
|
|
else:
|
|
|
|
c.vmState.readOnlyStateDB.getBalance(address)
|
|
|
|
|
2020-01-16 22:15:20 +07:00
|
|
|
template getCodeSize*(c: Computation, address: EthAddress): uint =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.getCodeSize(address)
|
|
|
|
else:
|
2020-05-30 10:14:59 +07:00
|
|
|
uint(c.vmState.readOnlyStateDB.getCodeSize(address))
|
2020-01-16 22:48:22 +07:00
|
|
|
|
|
|
|
template getCodeHash*(c: Computation, address: EthAddress): Hash256 =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.getCodeHash(address)
|
|
|
|
else:
|
|
|
|
let db = c.vmState.readOnlyStateDB
|
|
|
|
if not db.accountExists(address) or db.isEmptyAccount(address):
|
|
|
|
default(Hash256)
|
|
|
|
else:
|
|
|
|
db.getCodeHash(address)
|
2020-01-16 22:15:20 +07:00
|
|
|
|
2020-01-16 23:12:46 +07:00
|
|
|
template selfDestruct*(c: Computation, address: EthAddress) =
|
|
|
|
when evmc_enabled:
|
|
|
|
c.host.selfDestruct(c.msg.contractAddress, address)
|
|
|
|
else:
|
2020-01-22 11:48:02 +07:00
|
|
|
c.execSelfDestruct(address)
|
2020-01-16 23:12:46 +07:00
|
|
|
|
2020-04-20 20:12:44 +02:00
|
|
|
template getCode*(c: Computation, address: EthAddress): seq[byte] =
|
2020-01-16 23:12:46 +07:00
|
|
|
when evmc_enabled:
|
2020-04-20 20:12:44 +02:00
|
|
|
c.host.copyCode(address)
|
2020-01-16 23:12:46 +07:00
|
|
|
else:
|
|
|
|
c.vmState.readOnlyStateDB.getCode(address)
|
|
|
|
|
2020-01-31 08:08:44 +07:00
|
|
|
proc generateContractAddress(c: Computation, salt: Uint256): EthAddress =
|
2020-01-20 19:55:29 +07:00
|
|
|
if c.msg.kind == evmcCreate:
|
|
|
|
let creationNonce = c.vmState.readOnlyStateDb().getNonce(c.msg.sender)
|
|
|
|
result = generateAddress(c.msg.sender, creationNonce)
|
|
|
|
else:
|
2020-01-31 08:08:44 +07:00
|
|
|
result = generateSafeAddress(c.msg.sender, salt, c.msg.data)
|
2020-01-20 19:55:29 +07:00
|
|
|
|
2020-07-28 23:48:45 +07:00
|
|
|
import stew/byteutils
|
|
|
|
|
2020-01-31 08:08:44 +07:00
|
|
|
proc newComputation*(vmState: BaseVMState, message: Message, salt= 0.u256): Computation =
|
Refactor interpreter dispatch (#65)
* move forks constants, rename errors
* Move vm/utils to vm/interpreter/utils
* initial opcodes refactoring
* Add refactored Comparison & Bitwise Logic Operations
* Add sha3 and address, simplify macro, support pop 0
* balance, origin, caller, callValue
* fix gas copy opcodes gas costs, add callDataLoad/Size/Copy, CodeSize/Copy and gas price opcode
* Update with 30s, 40s, 50s opcodes + impl of balance + stack improvement
* add push, dup, swap, log, create and call operations
* finish opcode implementation
* Add the new dispatching logic
* Pass the opcode test
* Make test_vm_json compile
* halt execution without exceptions for Return, Revert, selfdestruct (fix #62)
* Properly catch and recover from EVM exceptions (stack underflow ...)
* Fix byte op
* Fix jump regressions
* Update for latest devel, don't import old dispatch code as quasiBoolean macro is broken by latest devel
* Fix sha3 regression on empty memory slice and until end of range slice
* Fix padding / range error on expXY_success (gas computation left)
* update logging procs
* Add tracing - expXY_success is not a regression, sload stub was accidentally passing the test
* Reuse the same stub as OO implementation
* Delete previous opcode implementation
* Delete object oriented fork code
* Delete exceptions that were used as control flows
* delete base.nim :fire:, yet another OO remnants
* Delete opcode table
* Enable omputed gotos and compile-time gas fees
* Revert const gasCosts -> generates SIGSEGV
* inline push, swap and dup opcodes
* loggers are now template again, why does this pass new tests?
* Trigger CI rebuild after rocksdb fix https://github.com/status-im/nim-rocksdb/pull/5
* Address review comment on "push" + VMTests in debug mode (not release)
* Address review comment: don't tag fork by default, make opcode impl grepable
* Static compilation fixes after rebasing
* fix the initialization of the VM database
* add a missing import
* Deactivate balance and sload test following #59
* Reactivate stack check (deactivated in #59, necessary to pass tests)
* Merge remaining opcodes implementation from #59
* Merge callDataLoad and codeCopy fixes, todo simplify see #67
2018-07-06 09:52:31 +02:00
|
|
|
new result
|
2018-01-16 19:05:20 +02:00
|
|
|
result.vmState = vmState
|
|
|
|
result.msg = message
|
|
|
|
result.memory = Memory()
|
|
|
|
result.stack = newStack()
|
2020-11-25 19:09:16 +07:00
|
|
|
result.returnStack = @[]
|
2018-07-18 15:18:17 +03:00
|
|
|
result.gasMeter.init(message.gas)
|
2020-01-09 13:25:53 +07:00
|
|
|
result.touchedAccounts = initHashSet[EthAddress]()
|
2019-12-19 23:37:10 +07:00
|
|
|
result.suicides = initHashSet[EthAddress]()
|
2020-01-16 17:23:51 +07:00
|
|
|
|
2020-01-20 19:55:29 +07:00
|
|
|
if result.msg.isCreate():
|
|
|
|
result.msg.contractAddress = result.generateContractAddress(salt)
|
2020-01-20 21:02:06 +07:00
|
|
|
result.code = newCodeStream(message.data)
|
|
|
|
message.data = @[]
|
|
|
|
else:
|
2020-04-20 20:12:44 +02:00
|
|
|
result.code = newCodeStream(vmState.readOnlyStateDb.getCode(message.codeAddress))
|
2020-01-20 19:55:29 +07:00
|
|
|
|
2020-01-16 17:23:51 +07:00
|
|
|
when evmc_enabled:
|
|
|
|
result.host.init(
|
|
|
|
nim_host_get_interface(),
|
|
|
|
cast[evmc_host_context](result)
|
|
|
|
)
|
|
|
|
|
2020-01-16 14:01:59 +07:00
|
|
|
template gasCosts*(c: Computation): untyped =
|
|
|
|
c.vmState.gasCosts
|
|
|
|
|
|
|
|
template fork*(c: Computation): untyped =
|
|
|
|
c.vmState.fork
|
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc isOriginComputation*(c: Computation): bool =
|
2018-01-16 19:05:20 +02:00
|
|
|
# Is this computation the computation initiated by a transaction
|
2020-01-16 13:36:58 +07:00
|
|
|
c.msg.sender == c.vmState.txOrigin
|
2018-01-16 19:05:20 +02:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
template isSuccess*(c: Computation): bool =
|
2018-01-16 19:05:20 +02:00
|
|
|
c.error.isNil
|
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
template isError*(c: Computation): bool =
|
2018-01-16 19:05:20 +02:00
|
|
|
not c.isSuccess
|
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
func shouldBurnGas*(c: Computation): bool =
|
2018-01-16 19:05:20 +02:00
|
|
|
c.isError and c.error.burnsGas
|
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc isSuicided*(c: Computation, address: EthAddress): bool =
|
2020-01-09 13:25:53 +07:00
|
|
|
result = address in c.suicides
|
2019-03-15 18:16:47 +07:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc snapshot*(c: Computation) =
|
2020-05-30 10:14:59 +07:00
|
|
|
c.savePoint = c.vmState.accountDb.beginSavePoint()
|
2019-02-17 07:30:02 +07:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc commit*(c: Computation) =
|
2020-05-30 10:14:59 +07:00
|
|
|
c.vmState.accountDb.commit(c.savePoint)
|
2018-01-16 19:05:20 +02:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc dispose*(c: Computation) {.inline.} =
|
2020-05-30 10:14:59 +07:00
|
|
|
c.vmState.accountDb.dispose(c.savePoint)
|
2019-02-18 18:45:18 +07:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc rollback*(c: Computation) =
|
2020-05-30 10:14:59 +07:00
|
|
|
c.vmState.accountDb.rollback(c.savePoint)
|
2019-04-01 08:54:02 +07:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc setError*(c: Computation, msg: string, burnsGas = false) {.inline.} =
|
|
|
|
c.error = Error(info: msg, burnsGas: burnsGas)
|
2019-04-01 08:54:02 +07:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc writeContract*(c: Computation, fork: Fork): bool {.gcsafe.} =
|
2019-03-19 22:19:08 +07:00
|
|
|
result = true
|
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
let contractCode = c.output
|
2019-03-19 22:19:08 +07:00
|
|
|
if contractCode.len == 0: return
|
|
|
|
|
|
|
|
if fork >= FkSpurious and contractCode.len >= EIP170_CODE_SIZE_LIMIT:
|
|
|
|
debug "Contract code size exceeds EIP170", limit=EIP170_CODE_SIZE_LIMIT, actual=contractCode.len
|
|
|
|
return false
|
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
let storageAddr = c.msg.contractAddress
|
|
|
|
if c.isSuicided(storageAddr): return
|
2019-03-19 22:19:08 +07:00
|
|
|
|
|
|
|
let gasParams = GasParams(kind: Create, cr_memLength: contractCode.len)
|
2020-01-10 08:26:17 +07:00
|
|
|
let codeCost = c.gasCosts[Create].c_handler(0.u256, gasParams).gasCost
|
|
|
|
if c.gasMeter.gasRemaining >= codeCost:
|
|
|
|
c.gasMeter.consumeGas(codeCost, reason = "Write contract code for CREATE")
|
|
|
|
c.vmState.mutateStateDb:
|
2020-04-20 20:12:44 +02:00
|
|
|
db.setCode(storageAddr, contractCode)
|
2019-03-19 22:19:08 +07:00
|
|
|
result = true
|
|
|
|
else:
|
2020-01-10 08:26:17 +07:00
|
|
|
if fork < FkHomestead or fork >= FkByzantium: c.output = @[]
|
2019-03-19 22:19:08 +07:00
|
|
|
result = false
|
|
|
|
|
2020-01-09 14:52:19 +07:00
|
|
|
proc initAddress(x: int): EthAddress {.compileTime.} = result[19] = x.byte
|
2020-02-04 18:18:30 +07:00
|
|
|
const ripemdAddr = initAddress(3)
|
2020-01-10 08:26:17 +07:00
|
|
|
proc executeOpcodes*(c: Computation) {.gcsafe.}
|
2019-03-19 23:30:35 +07:00
|
|
|
|
2020-02-03 12:37:33 +07:00
|
|
|
proc execCall*(c: Computation) =
|
|
|
|
c.snapshot()
|
|
|
|
defer:
|
|
|
|
c.dispose()
|
|
|
|
|
|
|
|
if c.msg.kind == evmcCall:
|
|
|
|
c.vmState.mutateStateDb:
|
|
|
|
db.subBalance(c.msg.sender, c.msg.value)
|
|
|
|
db.addBalance(c.msg.contractAddress, c.msg.value)
|
|
|
|
|
|
|
|
executeOpcodes(c)
|
|
|
|
|
2020-02-04 18:18:30 +07:00
|
|
|
## Collect all of the accounts that *may* need to be deleted based on EIP161
|
|
|
|
## https://github.com/ethereum/EIPs/blob/master/EIPS/eip-161.md
|
|
|
|
## also see: https://github.com/ethereum/EIPs/issues/716
|
|
|
|
|
2021-01-13 08:08:56 +07:00
|
|
|
if c.isError or c.fork >= FKByzantium:
|
2020-02-04 18:00:23 +07:00
|
|
|
if c.msg.contractAddress == ripemdAddr:
|
2020-02-04 18:18:30 +07:00
|
|
|
# Special case to account for geth+parity bug
|
2020-02-04 18:00:23 +07:00
|
|
|
c.vmState.touchedAccounts.incl c.msg.contractAddress
|
2020-02-04 18:18:30 +07:00
|
|
|
|
2020-02-03 12:37:33 +07:00
|
|
|
if c.isSuccess:
|
|
|
|
c.commit()
|
2020-02-04 18:00:23 +07:00
|
|
|
c.touchedAccounts.incl c.msg.contractAddress
|
2020-02-03 12:37:33 +07:00
|
|
|
else:
|
|
|
|
c.rollback()
|
|
|
|
|
2020-01-31 08:08:44 +07:00
|
|
|
proc execCreate*(c: Computation) =
|
|
|
|
c.vmState.mutateStateDB:
|
|
|
|
db.incNonce(c.msg.sender)
|
|
|
|
|
2020-12-11 17:51:17 +07:00
|
|
|
# We add this to the access list _before_ taking a snapshot.
|
|
|
|
# Even if the creation fails, the access-list change should not be rolled back
|
|
|
|
# EIP2929
|
|
|
|
if c.fork >= FkBerlin:
|
|
|
|
db.accessList(c.msg.contractAddress)
|
|
|
|
|
2020-01-31 08:08:44 +07:00
|
|
|
c.snapshot()
|
|
|
|
defer:
|
|
|
|
c.dispose()
|
|
|
|
|
|
|
|
if c.vmState.readOnlyStateDb().hasCodeOrNonce(c.msg.contractAddress):
|
|
|
|
c.setError("Address collision when creating contract address={c.msg.contractAddress.toHex}", true)
|
|
|
|
c.rollback()
|
|
|
|
return
|
|
|
|
|
|
|
|
c.vmState.mutateStateDb:
|
|
|
|
db.subBalance(c.msg.sender, c.msg.value)
|
|
|
|
db.addBalance(c.msg.contractAddress, c.msg.value)
|
|
|
|
db.clearStorage(c.msg.contractAddress)
|
|
|
|
if c.fork >= FkSpurious:
|
|
|
|
# EIP161 nonce incrementation
|
|
|
|
db.incNonce(c.msg.contractAddress)
|
|
|
|
|
|
|
|
executeOpcodes(c)
|
|
|
|
|
|
|
|
if c.isSuccess:
|
|
|
|
let fork = c.fork
|
|
|
|
let contractFailed = not c.writeContract(fork)
|
|
|
|
if contractFailed and fork >= FkHomestead:
|
|
|
|
c.setError(&"writeContract failed, depth={c.msg.depth}", true)
|
|
|
|
|
|
|
|
if c.isSuccess:
|
|
|
|
c.commit()
|
|
|
|
else:
|
|
|
|
c.rollback()
|
|
|
|
|
|
|
|
proc merge*(c, child: Computation) =
|
|
|
|
c.logEntries.add child.logEntries
|
|
|
|
c.gasMeter.refundGas(child.gasMeter.gasRefunded)
|
|
|
|
c.suicides.incl child.suicides
|
|
|
|
c.touchedAccounts.incl child.touchedAccounts
|
|
|
|
|
2020-01-22 11:48:02 +07:00
|
|
|
proc execSelfDestruct*(c: Computation, beneficiary: EthAddress) =
|
2020-01-17 18:58:03 +07:00
|
|
|
c.vmState.mutateStateDB:
|
|
|
|
let
|
|
|
|
localBalance = c.getBalance(c.msg.contractAddress)
|
|
|
|
beneficiaryBalance = c.getBalance(beneficiary)
|
|
|
|
|
|
|
|
# Transfer to beneficiary
|
|
|
|
db.setBalance(beneficiary, localBalance + beneficiaryBalance)
|
|
|
|
|
|
|
|
# Zero the balance of the address being deleted.
|
|
|
|
# This must come after sending to beneficiary in case the
|
|
|
|
# contract named itself as the beneficiary.
|
|
|
|
db.setBalance(c.msg.contractAddress, 0.u256)
|
|
|
|
|
|
|
|
trace "SELFDESTRUCT",
|
|
|
|
contractAddress = c.msg.contractAddress.toHex,
|
|
|
|
localBalance = localBalance.toString,
|
|
|
|
beneficiary = beneficiary.toHex
|
|
|
|
|
2020-01-09 13:25:53 +07:00
|
|
|
c.touchedAccounts.incl beneficiary
|
2020-01-17 18:58:03 +07:00
|
|
|
# Register the account to be deleted
|
2020-01-07 23:11:06 +07:00
|
|
|
c.suicides.incl(c.msg.contractAddress)
|
2018-01-16 19:05:20 +02:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc addLogEntry*(c: Computation, log: Log) {.inline.} =
|
2019-02-27 21:04:42 +07:00
|
|
|
c.logEntries.add(log)
|
2018-01-16 19:05:20 +02:00
|
|
|
|
2020-01-10 08:26:17 +07:00
|
|
|
proc getGasRefund*(c: Computation): GasInt =
|
2019-12-19 16:31:06 +07:00
|
|
|
if c.isSuccess:
|
|
|
|
result = c.gasMeter.gasRefunded
|
2018-01-16 19:05:20 +02:00
|
|
|
|
2020-01-10 18:18:36 +07:00
|
|
|
proc refundSelfDestruct*(c: Computation) =
|
|
|
|
let cost = gasFees[c.fork][RefundSelfDestruct]
|
|
|
|
c.gasMeter.refundGas(cost * c.suicides.len)
|
|
|
|
|
2020-06-06 10:05:11 +07:00
|
|
|
proc tracingEnabled*(c: Computation): bool {.inline.} =
|
|
|
|
EnableTracing in c.vmState.tracer.flags
|
2018-12-03 17:54:19 +07:00
|
|
|
|
2020-06-06 10:05:11 +07:00
|
|
|
proc traceOpCodeStarted*(c: Computation, op: Op): int {.inline.} =
|
2018-12-11 16:23:15 +07:00
|
|
|
c.vmState.tracer.traceOpCodeStarted(c, op)
|
2018-12-03 17:54:19 +07:00
|
|
|
|
2020-06-06 10:05:11 +07:00
|
|
|
proc traceOpCodeEnded*(c: Computation, op: Op, lastIndex: int) {.inline.} =
|
2019-02-21 15:17:43 +07:00
|
|
|
c.vmState.tracer.traceOpCodeEnded(c, op, lastIndex)
|
2018-12-03 23:22:08 +07:00
|
|
|
|
2020-06-06 10:05:11 +07:00
|
|
|
proc traceError*(c: Computation) {.inline.} =
|
2018-12-03 23:22:08 +07:00
|
|
|
c.vmState.tracer.traceError(c)
|
2019-02-25 20:02:16 +07:00
|
|
|
|
2020-06-06 10:05:11 +07:00
|
|
|
proc prepareTracer*(c: Computation) {.inline.} =
|
2019-02-25 20:02:16 +07:00
|
|
|
c.vmState.tracer.prepare(c.msg.depth)
|
2019-03-19 23:30:35 +07:00
|
|
|
|
|
|
|
include interpreter_dispatch
|
2020-01-17 21:54:28 +07:00
|
|
|
|
|
|
|
when defined(evmc_enabled):
|
|
|
|
include evmc_host
|