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
|
2018-11-28 20:02:21 +01:00
|
|
|
chronicles, strformat, strutils, sequtils, macros, terminal, math, tables, options,
|
2019-04-17 11:03:52 +07:00
|
|
|
sets, eth/[common, keys], eth/trie/db as triedb,
|
2018-08-22 21:38:00 -06:00
|
|
|
../constants, ../errors, ../validation, ../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],
|
2018-09-14 18:03:26 +01:00
|
|
|
./code_stream, ./memory, ./message, ./stack, ../db/[state_db, db_chain],
|
2019-07-07 12:12:01 +02:00
|
|
|
../utils/header, stew/[byteutils, ranges], precompiles,
|
2019-05-02 12:59:10 +07:00
|
|
|
transaction_tracer, eth/trie/trie_defs
|
2018-05-24 12:01:59 +02:00
|
|
|
|
2018-08-22 21:38:00 -06:00
|
|
|
logScope:
|
|
|
|
topics = "vm computation"
|
|
|
|
|
2019-07-10 09:40:46 +02:00
|
|
|
proc newBaseComputation*(vmState: BaseVMState, blockNumber: BlockNumber, message: Message, forkOverride=none(Fork)): BaseComputation =
|
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()
|
2018-07-18 15:18:17 +03:00
|
|
|
result.gasMeter.init(message.gas)
|
2018-01-16 19:05:20 +02:00
|
|
|
result.children = @[]
|
2018-05-30 19:11:15 +03:00
|
|
|
result.accountsToDelete = initTable[EthAddress, EthAddress]()
|
2018-09-06 20:05:22 +03:00
|
|
|
result.code = newCodeStream(message.code)
|
2018-07-06 17:08:31 +02:00
|
|
|
# result.rawOutput = "0x"
|
2018-11-28 20:02:21 +01:00
|
|
|
result.gasCosts =
|
|
|
|
if forkOverride.isSome:
|
|
|
|
forkOverride.get.forkToSchedule
|
|
|
|
else:
|
|
|
|
blockNumber.toFork.forkToSchedule
|
|
|
|
result.forkOverride = forkOverride
|
2019-04-04 11:44:35 +07:00
|
|
|
# a dummy/terminus continuation proc
|
|
|
|
result.nextProc = proc() =
|
|
|
|
discard
|
2018-01-24 15:31:24 +02:00
|
|
|
|
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
|
|
|
proc isOriginComputation*(c: BaseComputation): bool =
|
2018-01-16 19:05:20 +02:00
|
|
|
# Is this computation the computation initiated by a transaction
|
|
|
|
c.msg.isOrigin
|
|
|
|
|
|
|
|
template isSuccess*(c: BaseComputation): bool =
|
|
|
|
c.error.isNil
|
|
|
|
|
|
|
|
template isError*(c: BaseComputation): bool =
|
|
|
|
not c.isSuccess
|
|
|
|
|
2018-09-07 17:18:46 +01:00
|
|
|
func shouldBurnGas*(c: BaseComputation): bool =
|
2018-01-16 19:05:20 +02:00
|
|
|
c.isError and c.error.burnsGas
|
|
|
|
|
2018-09-07 17:18:46 +01:00
|
|
|
func shouldEraseReturnData*(c: BaseComputation): bool =
|
2018-01-16 19:05:20 +02:00
|
|
|
c.isError and c.error.erasesReturnData
|
|
|
|
|
2018-07-06 17:08:31 +02:00
|
|
|
func bytesToHex(x: openarray[byte]): string {.inline.} =
|
|
|
|
## TODO: use seq[byte] for raw data and delete this proc
|
|
|
|
foldl(x, a & b.int.toHex(2).toLowerAscii, "0x")
|
|
|
|
|
2018-09-07 17:18:46 +01:00
|
|
|
func output*(c: BaseComputation): seq[byte] =
|
|
|
|
if c.shouldEraseReturnData:
|
|
|
|
@[]
|
|
|
|
else:
|
|
|
|
c.rawOutput
|
|
|
|
|
2019-04-04 10:20:00 +07:00
|
|
|
func `output=`*(c: BaseComputation, value: openarray[byte]) =
|
2018-09-07 17:18:46 +01:00
|
|
|
c.rawOutput = @value
|
|
|
|
|
|
|
|
proc outputHex*(c: BaseComputation): string =
|
|
|
|
if c.shouldEraseReturnData:
|
|
|
|
return "0x"
|
|
|
|
c.rawOutput.bytesToHex
|
|
|
|
|
2019-04-04 10:20:00 +07:00
|
|
|
proc isSuicided*(c: BaseComputation, address: EthAddress): bool =
|
2019-03-15 18:16:47 +07:00
|
|
|
result = address in c.accountsToDelete
|
|
|
|
|
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
|
|
|
proc prepareChildMessage*(
|
2019-04-04 10:20:00 +07:00
|
|
|
c: BaseComputation,
|
2018-05-25 12:25:19 +02:00
|
|
|
gas: GasInt,
|
2018-05-30 19:11:15 +03:00
|
|
|
to: EthAddress,
|
2018-02-20 19:27:43 +02:00
|
|
|
value: UInt256,
|
2018-01-29 19:40:22 +02:00
|
|
|
data: seq[byte],
|
2018-07-06 17:08:31 +02:00
|
|
|
code: seq[byte],
|
2019-04-18 12:42:37 +07:00
|
|
|
contractCreation: bool,
|
2018-01-16 19:05:20 +02:00
|
|
|
options: MessageOptions = newMessageOptions()): Message =
|
|
|
|
|
|
|
|
var childOptions = options
|
2018-01-30 20:12:05 +02:00
|
|
|
childOptions.depth = c.msg.depth + 1
|
2018-01-16 19:05:20 +02:00
|
|
|
result = newMessage(
|
|
|
|
gas,
|
|
|
|
c.msg.gasPrice,
|
|
|
|
to,
|
2018-10-18 17:47:54 +01:00
|
|
|
c.msg.origin,
|
2018-01-16 19:05:20 +02:00
|
|
|
value,
|
|
|
|
data,
|
2018-09-06 20:05:22 +03:00
|
|
|
code,
|
2019-04-18 12:42:37 +07:00
|
|
|
contractCreation,
|
2018-01-16 19:05:20 +02:00
|
|
|
childOptions)
|
|
|
|
|
2019-03-20 17:52:42 +07:00
|
|
|
proc snapshot*(comp: BaseComputation) =
|
|
|
|
comp.dbsnapshot.transaction = comp.vmState.chaindb.db.beginTransaction()
|
|
|
|
comp.dbsnapshot.intermediateRoot = comp.vmState.accountDb.rootHash
|
|
|
|
comp.vmState.blockHeader.stateRoot = comp.vmState.accountDb.rootHash
|
2019-02-17 07:30:02 +07:00
|
|
|
|
2019-03-20 17:52:42 +07:00
|
|
|
proc commit*(comp: BaseComputation) =
|
|
|
|
comp.dbsnapshot.transaction.commit()
|
|
|
|
comp.vmState.accountDb.rootHash = comp.vmState.blockHeader.stateRoot
|
2018-01-16 19:05:20 +02:00
|
|
|
|
2019-03-20 17:52:42 +07:00
|
|
|
proc dispose*(comp: BaseComputation) {.inline.} =
|
|
|
|
comp.dbsnapshot.transaction.dispose()
|
2019-02-18 18:45:18 +07:00
|
|
|
|
2019-04-01 08:54:02 +07:00
|
|
|
proc rollback*(comp: BaseComputation) =
|
|
|
|
comp.dbsnapshot.transaction.rollback()
|
|
|
|
comp.vmState.accountDb.rootHash = comp.dbsnapshot.intermediateRoot
|
|
|
|
comp.vmState.blockHeader.stateRoot = comp.dbsnapshot.intermediateRoot
|
|
|
|
|
|
|
|
proc setError*(comp: BaseComputation, msg: string, burnsGas = false) {.inline.} =
|
|
|
|
comp.error = Error(info: msg, burnsGas: burnsGas)
|
|
|
|
|
2019-03-19 22:19:08 +07:00
|
|
|
proc getFork*(computation: BaseComputation): Fork =
|
|
|
|
result =
|
|
|
|
if computation.forkOverride.isSome:
|
|
|
|
computation.forkOverride.get
|
|
|
|
else:
|
|
|
|
computation.vmState.blockNumber.toFork
|
|
|
|
|
2019-04-26 00:18:51 +02:00
|
|
|
proc writeContract*(computation: BaseComputation, fork: Fork): bool {.gcsafe.} =
|
2019-03-19 22:19:08 +07:00
|
|
|
result = true
|
|
|
|
|
|
|
|
let contractCode = computation.output
|
|
|
|
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
|
|
|
|
|
|
|
|
let storageAddr = computation.msg.storageAddress
|
|
|
|
if computation.isSuicided(storageAddr): return
|
|
|
|
|
|
|
|
let gasParams = GasParams(kind: Create, cr_memLength: contractCode.len)
|
|
|
|
let codeCost = computation.gasCosts[Create].c_handler(0.u256, gasParams).gasCost
|
|
|
|
if computation.gasMeter.gasRemaining >= codeCost:
|
|
|
|
computation.gasMeter.consumeGas(codeCost, reason = "Write contract code for CREATE")
|
|
|
|
computation.vmState.mutateStateDb:
|
|
|
|
db.setCode(storageAddr, contractCode.toRange)
|
|
|
|
result = true
|
|
|
|
else:
|
2019-05-02 13:39:55 +07:00
|
|
|
if fork < FkHomestead or fork >= FkByzantium: computation.output = @[]
|
2019-03-19 22:19:08 +07:00
|
|
|
result = false
|
|
|
|
|
2019-04-04 10:20:00 +07:00
|
|
|
proc transferBalance(computation: BaseComputation, opCode: static[Op]) =
|
2019-03-13 22:06:32 +07:00
|
|
|
let senderBalance = computation.vmState.readOnlyStateDb().
|
|
|
|
getBalance(computation.msg.sender)
|
2018-09-14 18:03:26 +01:00
|
|
|
|
2019-03-13 22:06:32 +07:00
|
|
|
if senderBalance < computation.msg.value:
|
2019-04-01 08:54:02 +07:00
|
|
|
computation.setError(&"insufficient funds available={senderBalance}, needed={computation.msg.value}")
|
|
|
|
return
|
2019-02-20 18:02:14 +07:00
|
|
|
|
2019-03-16 22:54:24 +07:00
|
|
|
when opCode in {Call, Create}:
|
|
|
|
computation.vmState.mutateStateDb:
|
|
|
|
db.subBalance(computation.msg.sender, computation.msg.value)
|
|
|
|
db.addBalance(computation.msg.storageAddress, computation.msg.value)
|
2018-09-14 18:03:26 +01:00
|
|
|
|
2019-04-04 10:50:25 +07:00
|
|
|
template continuation*(comp: BaseComputation, body: untyped) =
|
2019-04-04 11:44:35 +07:00
|
|
|
# this is a helper template to implement continuation
|
|
|
|
# passing and convert all recursion into tail call
|
2019-04-04 10:50:25 +07:00
|
|
|
var tmpNext = comp.nextProc
|
2019-04-26 00:18:51 +02:00
|
|
|
comp.nextProc = proc() {.gcsafe.} =
|
2019-04-04 10:50:25 +07:00
|
|
|
body
|
|
|
|
tmpNext()
|
|
|
|
|
2019-04-26 00:18:51 +02:00
|
|
|
proc postExecuteVM(computation: BaseComputation, opCode: static[Op]) {.gcsafe.} =
|
2019-04-17 12:15:15 +07:00
|
|
|
when opCode == Create:
|
|
|
|
if computation.isSuccess:
|
|
|
|
let fork = computation.getFork
|
|
|
|
let contractFailed = not computation.writeContract(fork)
|
|
|
|
if contractFailed and fork >= FkHomestead:
|
|
|
|
computation.setError(&"writeContract failed, depth={computation.msg.depth}", true)
|
2019-04-04 10:50:25 +07:00
|
|
|
|
|
|
|
if computation.isSuccess:
|
|
|
|
computation.commit()
|
|
|
|
else:
|
|
|
|
computation.rollback()
|
|
|
|
|
2019-04-04 10:20:00 +07:00
|
|
|
proc executeOpcodes*(computation: BaseComputation) {.gcsafe.}
|
2019-03-19 23:30:35 +07:00
|
|
|
|
2019-04-04 10:20:00 +07:00
|
|
|
proc applyMessage*(computation: BaseComputation, opCode: static[Op]) =
|
2019-04-10 10:11:25 +07:00
|
|
|
if computation.msg.depth > MaxCallDepth:
|
|
|
|
computation.setError(&"Stack depth limit reached depth={computation.msg.depth}")
|
|
|
|
computation.nextProc()
|
|
|
|
return
|
|
|
|
|
2019-03-20 17:52:42 +07:00
|
|
|
computation.snapshot()
|
2019-04-04 11:44:35 +07:00
|
|
|
defer:
|
|
|
|
computation.dispose()
|
2019-02-18 18:45:18 +07:00
|
|
|
|
2019-04-17 12:15:15 +07:00
|
|
|
# EIP161 nonce incrementation
|
2019-05-07 10:23:27 +07:00
|
|
|
when opCode in {Create, Create2}:
|
2019-04-17 12:15:15 +07:00
|
|
|
if computation.getFork >= FkSpurious:
|
|
|
|
computation.vmState.mutateStateDb:
|
2019-05-02 13:39:55 +07:00
|
|
|
db.incNonce(computation.msg.storageAddress)
|
2019-05-02 12:59:10 +07:00
|
|
|
if computation.getFork >= FkByzantium:
|
|
|
|
# RevertInCreateInInit.json
|
|
|
|
db.setStorageRoot(computation.msg.storageAddress, emptyRlpHash)
|
2019-04-17 12:15:15 +07:00
|
|
|
|
2019-03-16 22:54:24 +07:00
|
|
|
when opCode in {CallCode, Call, Create}:
|
2019-04-01 08:54:02 +07:00
|
|
|
computation.transferBalance(opCode)
|
|
|
|
if computation.isError():
|
|
|
|
computation.rollback()
|
2019-04-04 11:44:35 +07:00
|
|
|
computation.nextProc()
|
2019-02-20 18:02:14 +07:00
|
|
|
return
|
2019-03-19 23:54:54 +07:00
|
|
|
|
2019-02-25 17:03:43 +07:00
|
|
|
if computation.gasMeter.gasRemaining < 0:
|
2019-03-20 17:52:42 +07:00
|
|
|
computation.commit()
|
2019-04-04 11:44:35 +07:00
|
|
|
computation.nextProc()
|
2019-02-20 11:09:01 +07:00
|
|
|
return
|
2018-09-18 02:56:10 +03:00
|
|
|
|
2019-04-04 11:44:35 +07:00
|
|
|
continuation(computation):
|
2019-04-17 12:15:15 +07:00
|
|
|
postExecuteVM(computation, opCode)
|
2018-09-14 18:03:26 +01:00
|
|
|
|
2019-04-04 11:44:35 +07:00
|
|
|
executeOpcodes(computation)
|
2019-04-04 12:13:33 +07:00
|
|
|
|
2019-04-04 10:20:00 +07:00
|
|
|
proc addChildComputation*(computation: BaseComputation, child: BaseComputation) =
|
2018-09-07 17:18:46 +01:00
|
|
|
if child.isError:
|
|
|
|
if child.msg.isCreate:
|
|
|
|
computation.returnData = child.output
|
|
|
|
elif child.shouldBurnGas:
|
|
|
|
computation.returnData = @[]
|
|
|
|
else:
|
|
|
|
computation.returnData = child.output
|
|
|
|
else:
|
|
|
|
if child.msg.isCreate:
|
|
|
|
computation.returnData = @[]
|
|
|
|
else:
|
|
|
|
computation.returnData = child.output
|
2019-02-27 21:04:42 +07:00
|
|
|
computation.logEntries.add child.logEntries
|
2018-09-07 17:18:46 +01:00
|
|
|
|
2019-04-02 12:13:18 +07:00
|
|
|
if not child.shouldBurnGas:
|
|
|
|
computation.gasMeter.returnGas(child.gasMeter.gasRemaining)
|
|
|
|
computation.children.add(child)
|
2018-01-16 19:05:20 +02:00
|
|
|
|
2019-04-04 10:20:00 +07:00
|
|
|
proc registerAccountForDeletion*(c: BaseComputation, beneficiary: EthAddress) =
|
2018-01-16 19:05:20 +02:00
|
|
|
if c.msg.storageAddress in c.accountsToDelete:
|
|
|
|
raise newException(ValueError,
|
|
|
|
"invariant: should be impossible for an account to be " &
|
|
|
|
"registered for deletion multiple times")
|
|
|
|
c.accountsToDelete[c.msg.storageAddress] = beneficiary
|
|
|
|
|
2019-04-04 10:20:00 +07:00
|
|
|
proc addLogEntry*(c: BaseComputation, log: Log) {.inline.} =
|
2019-02-27 21:04:42 +07:00
|
|
|
c.logEntries.add(log)
|
2018-01-16 19:05:20 +02:00
|
|
|
|
2018-03-13 16:30:38 +02:00
|
|
|
# many methods are basically TODO, but they still return valid values
|
|
|
|
# in order to test some existing code
|
2019-03-20 00:11:35 +07:00
|
|
|
iterator accountsForDeletion*(c: BaseComputation): EthAddress =
|
2019-04-18 13:23:04 +07:00
|
|
|
var stack = @[c]
|
|
|
|
var deletedAccounts = initSet[EthAddress]()
|
|
|
|
while stack.len > 0:
|
|
|
|
let comp = stack.pop()
|
|
|
|
if comp.isError: continue
|
|
|
|
for account in comp.accountsToDelete.keys:
|
|
|
|
if account notin deletedAccounts:
|
|
|
|
deletedAccounts.incl account
|
|
|
|
yield account
|
|
|
|
stack.add comp.children
|
2018-01-16 19:05:20 +02:00
|
|
|
|
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
|
|
|
proc getGasRefund*(c: BaseComputation): GasInt =
|
2018-01-16 19:05:20 +02:00
|
|
|
if c.isError:
|
2018-05-25 12:25:19 +02:00
|
|
|
result = 0
|
2018-01-16 19:05:20 +02:00
|
|
|
else:
|
2018-05-25 12:25:19 +02:00
|
|
|
result = c.gasMeter.gasRefunded + c.children.mapIt(it.getGasRefund()).foldl(a + b, 0'i64)
|
2018-01-16 19:05:20 +02:00
|
|
|
|
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
|
|
|
proc getGasUsed*(c: BaseComputation): GasInt =
|
2018-01-16 19:05:20 +02:00
|
|
|
if c.shouldBurnGas:
|
|
|
|
result = c.msg.gas
|
|
|
|
else:
|
2018-05-25 12:25:19 +02:00
|
|
|
result = max(0, c.msg.gas - c.gasMeter.gasRemaining)
|
2018-01-16 19:05:20 +02:00
|
|
|
|
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
|
|
|
proc getGasRemaining*(c: BaseComputation): GasInt =
|
2018-01-16 19:05:20 +02:00
|
|
|
if c.shouldBurnGas:
|
2018-05-25 12:25:19 +02:00
|
|
|
result = 0
|
2018-01-16 19:05:20 +02:00
|
|
|
else:
|
|
|
|
result = c.gasMeter.gasRemaining
|
2018-12-03 17:54:19 +07:00
|
|
|
|
2019-04-17 11:03:52 +07:00
|
|
|
proc collectTouchedAccounts*(c: BaseComputation, output: var HashSet[EthAddress]) =
|
|
|
|
## 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
|
|
|
|
|
|
|
|
proc cmpThree(address: EthAddress): bool =
|
|
|
|
for i in 0..18:
|
|
|
|
if address[i] != 0: return
|
|
|
|
result = address[19] == byte(3)
|
|
|
|
|
|
|
|
for _, beneficiary in c.accountsToDelete:
|
|
|
|
if c.isError and c.isOriginComputation:
|
|
|
|
# Special case to account for geth+parity bug
|
|
|
|
# https://github.com/ethereum/EIPs/issues/716
|
|
|
|
if beneficiary.cmpThree:
|
|
|
|
output.incl beneficiary
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
output.incl beneficiary
|
|
|
|
|
|
|
|
if not c.msg.isCreate:
|
|
|
|
if c.isError and c.isOriginComputation:
|
|
|
|
# Special case to account for geth+parity bug
|
|
|
|
# https://github.com/ethereum/EIPs/issues/716
|
2019-04-18 10:21:20 +07:00
|
|
|
if cmpThree(c.msg.storageAddress):
|
|
|
|
output.incl c.msg.storageAddress
|
2019-04-17 11:03:52 +07:00
|
|
|
else:
|
2019-04-18 10:21:20 +07:00
|
|
|
output.incl c.msg.storageAddress
|
2019-04-17 11:03:52 +07:00
|
|
|
|
2019-05-01 21:47:30 +07:00
|
|
|
if not c.isError:
|
2019-04-17 11:03:52 +07:00
|
|
|
for child in c.children:
|
|
|
|
child.collectTouchedAccounts(output)
|
|
|
|
|
2018-12-03 23:22:08 +07:00
|
|
|
proc tracingEnabled*(c: BaseComputation): bool =
|
2018-12-03 17:54:19 +07:00
|
|
|
c.vmState.tracingEnabled
|
|
|
|
|
2019-02-21 15:17:43 +07:00
|
|
|
proc traceOpCodeStarted*(c: BaseComputation, op: Op): int =
|
2018-12-11 16:23:15 +07:00
|
|
|
c.vmState.tracer.traceOpCodeStarted(c, op)
|
2018-12-03 17:54:19 +07:00
|
|
|
|
2019-02-21 15:17:43 +07:00
|
|
|
proc traceOpCodeEnded*(c: BaseComputation, op: Op, lastIndex: int) =
|
|
|
|
c.vmState.tracer.traceOpCodeEnded(c, op, lastIndex)
|
2018-12-03 23:22:08 +07:00
|
|
|
|
|
|
|
proc traceError*(c: BaseComputation) =
|
|
|
|
c.vmState.tracer.traceError(c)
|
2019-02-25 20:02:16 +07:00
|
|
|
|
|
|
|
proc prepareTracer*(c: BaseComputation) =
|
|
|
|
c.vmState.tracer.prepare(c.msg.depth)
|
2019-03-19 23:30:35 +07:00
|
|
|
|
|
|
|
include interpreter_dispatch
|