2018-04-06 14:52:10 +00: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 17:05:20 +00:00
|
|
|
import
|
2018-08-23 03:38:00 +00:00
|
|
|
chronicles, strformat, strutils, sequtils, macros, terminal, math, tables,
|
2018-07-06 15:08:31 +00:00
|
|
|
eth_common,
|
2018-08-23 03:38:00 +00: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 07:52:31 +00:00
|
|
|
./interpreter/[opcode_values, gas_meter, gas_costs, vm_forks],
|
2018-09-14 17:03:26 +00:00
|
|
|
./code_stream, ./memory, ./message, ./stack, ../db/[state_db, db_chain],
|
2018-12-03 10:54:19 +00:00
|
|
|
../utils/header, byteutils, ranges, eth_keys, precompiles,
|
|
|
|
transaction_tracer
|
2018-05-24 10:01:59 +00:00
|
|
|
|
2018-08-23 03:38:00 +00:00
|
|
|
logScope:
|
|
|
|
topics = "vm 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 07:52:31 +00:00
|
|
|
proc newBaseComputation*(vmState: BaseVMState, blockNumber: UInt256, message: Message): BaseComputation =
|
|
|
|
new result
|
2018-01-16 17:05:20 +00:00
|
|
|
result.vmState = vmState
|
|
|
|
result.msg = message
|
|
|
|
result.memory = Memory()
|
|
|
|
result.stack = newStack()
|
2018-07-18 12:18:17 +00:00
|
|
|
result.gasMeter.init(message.gas)
|
2018-01-16 17:05:20 +00:00
|
|
|
result.children = @[]
|
2018-05-30 16:11:15 +00:00
|
|
|
result.accountsToDelete = initTable[EthAddress, EthAddress]()
|
2018-01-16 17:05:20 +00:00
|
|
|
result.logEntries = @[]
|
2018-09-06 17:05:22 +00:00
|
|
|
result.code = newCodeStream(message.code)
|
2018-07-06 15:08:31 +00:00
|
|
|
# result.rawOutput = "0x"
|
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 07:52:31 +00:00
|
|
|
result.gasCosts = blockNumber.toFork.forkToSchedule
|
2018-01-24 13:31:24 +00: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 07:52:31 +00:00
|
|
|
proc isOriginComputation*(c: BaseComputation): bool =
|
2018-01-16 17:05:20 +00: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 16:18:46 +00:00
|
|
|
func shouldBurnGas*(c: BaseComputation): bool =
|
2018-01-16 17:05:20 +00:00
|
|
|
c.isError and c.error.burnsGas
|
|
|
|
|
2018-09-07 16:18:46 +00:00
|
|
|
func shouldEraseReturnData*(c: BaseComputation): bool =
|
2018-01-16 17:05:20 +00:00
|
|
|
c.isError and c.error.erasesReturnData
|
|
|
|
|
2018-07-06 15:08:31 +00: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 16:18:46 +00:00
|
|
|
func output*(c: BaseComputation): seq[byte] =
|
|
|
|
if c.shouldEraseReturnData:
|
|
|
|
@[]
|
|
|
|
else:
|
|
|
|
c.rawOutput
|
|
|
|
|
|
|
|
func `output=`*(c: var BaseComputation, value: openarray[byte]) =
|
|
|
|
c.rawOutput = @value
|
|
|
|
|
|
|
|
proc outputHex*(c: BaseComputation): string =
|
|
|
|
if c.shouldEraseReturnData:
|
|
|
|
return "0x"
|
|
|
|
c.rawOutput.bytesToHex
|
|
|
|
|
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 07:52:31 +00:00
|
|
|
proc prepareChildMessage*(
|
2018-01-16 17:05:20 +00:00
|
|
|
c: var BaseComputation,
|
2018-05-25 10:25:19 +00:00
|
|
|
gas: GasInt,
|
2018-05-30 16:11:15 +00:00
|
|
|
to: EthAddress,
|
2018-02-20 17:27:43 +00:00
|
|
|
value: UInt256,
|
2018-01-29 17:40:22 +00:00
|
|
|
data: seq[byte],
|
2018-07-06 15:08:31 +00:00
|
|
|
code: seq[byte],
|
2018-01-16 17:05:20 +00:00
|
|
|
options: MessageOptions = newMessageOptions()): Message =
|
|
|
|
|
|
|
|
var childOptions = options
|
2018-01-30 18:12:05 +00:00
|
|
|
childOptions.depth = c.msg.depth + 1
|
2018-01-16 17:05:20 +00:00
|
|
|
result = newMessage(
|
|
|
|
gas,
|
|
|
|
c.msg.gasPrice,
|
|
|
|
to,
|
2018-10-18 16:47:54 +00:00
|
|
|
c.msg.origin,
|
2018-01-16 17:05:20 +00:00
|
|
|
value,
|
|
|
|
data,
|
2018-09-06 17:05:22 +00:00
|
|
|
code,
|
2018-01-16 17:05:20 +00:00
|
|
|
childOptions)
|
|
|
|
|
2018-10-02 15:07:16 +00:00
|
|
|
proc applyMessage(computation: var BaseComputation, opCode: static[Op]) =
|
2018-09-17 23:56:10 +00:00
|
|
|
var transaction = computation.vmState.beginTransaction()
|
|
|
|
defer: transaction.dispose()
|
2018-01-16 17:05:20 +00:00
|
|
|
|
2018-09-14 17:03:26 +00:00
|
|
|
if computation.msg.depth > STACK_DEPTH_LIMIT:
|
|
|
|
raise newException(StackDepthError, "Stack depth limit reached")
|
2018-07-06 15:08:31 +00:00
|
|
|
|
2018-09-14 17:03:26 +00:00
|
|
|
if computation.msg.value != 0:
|
|
|
|
let senderBalance =
|
|
|
|
computation.vmState.chainDb.getStateDb(
|
|
|
|
computation.vmState.blockHeader.hash, false).
|
|
|
|
getBalance(computation.msg.sender)
|
2018-10-02 15:07:16 +00:00
|
|
|
var newBalance = senderBalance
|
2018-09-14 17:03:26 +00:00
|
|
|
|
|
|
|
if sender_balance < computation.msg.value:
|
2018-09-17 23:56:10 +00:00
|
|
|
raise newException(InsufficientFunds,
|
2018-09-14 17:03:26 +00:00
|
|
|
&"Insufficient funds: {senderBalance} < {computation.msg.value}"
|
|
|
|
)
|
2018-10-02 15:07:16 +00:00
|
|
|
when opCode in {Call, CallCode}:
|
|
|
|
let
|
|
|
|
insufficientFunds = senderBalance < computation.msg.value
|
|
|
|
stackTooDeep = computation.msg.depth >= MaxCallDepth
|
|
|
|
|
|
|
|
if insufficientFunds or stackTooDeep:
|
|
|
|
computation.returnData = @[]
|
|
|
|
var errMessage: string
|
|
|
|
if insufficientFunds:
|
|
|
|
errMessage = &"Insufficient Funds: have: {$senderBalance} need: {$computation.msg.value}"
|
|
|
|
elif stackTooDeep:
|
|
|
|
errMessage = "Stack Limit Reached"
|
|
|
|
else:
|
|
|
|
raise newException(VMError, "Invariant: Unreachable code path")
|
|
|
|
|
|
|
|
debug "Computation failure", msg = errMessage
|
|
|
|
computation.gasMeter.returnGas(computation.msg.gas)
|
|
|
|
push: 0
|
|
|
|
return
|
2018-12-03 10:54:19 +00:00
|
|
|
|
2018-10-02 15:07:16 +00:00
|
|
|
newBalance = senderBalance - computation.msg.value
|
|
|
|
computation.vmState.mutateStateDb:
|
|
|
|
db.setBalance(computation.msg.sender, newBalance)
|
|
|
|
db.addBalance(computation.msg.storage_address, computation.msg.value)
|
|
|
|
|
|
|
|
debug "Value transferred",
|
|
|
|
source = computation.msg.sender,
|
|
|
|
dest = computation.msg.storage_address,
|
|
|
|
value = computation.msg.value,
|
|
|
|
oldSenderBalance = senderBalance,
|
|
|
|
newSenderBalance = newBalance,
|
|
|
|
gasPrice = computation.msg.gasPrice,
|
|
|
|
gas = computation.msg.gas
|
2018-09-14 17:03:26 +00:00
|
|
|
|
|
|
|
debug "Apply message",
|
|
|
|
value = computation.msg.value,
|
2018-10-02 15:07:16 +00:00
|
|
|
senderBalance = newBalance,
|
2018-09-14 17:03:26 +00:00
|
|
|
sender = computation.msg.sender.toHex,
|
2018-10-02 15:07:16 +00:00
|
|
|
address = computation.msg.storage_address.toHex,
|
|
|
|
gasPrice = computation.msg.gasPrice,
|
|
|
|
gas = computation.msg.gas
|
2018-09-14 17:03:26 +00:00
|
|
|
|
2018-10-02 15:07:16 +00:00
|
|
|
# Run code
|
2018-11-08 16:40:40 +00:00
|
|
|
# We cannot use the normal dispatching function `executeOpcodes`
|
|
|
|
# within `interpreter_dispatch.nim` due to a cyclic dependency.
|
2018-10-02 15:07:16 +00:00
|
|
|
if not computation.execPrecompiles:
|
|
|
|
computation.opcodeExec(computation)
|
2018-09-14 17:03:26 +00:00
|
|
|
|
2018-09-17 23:56:10 +00:00
|
|
|
if not computation.isError:
|
2018-10-02 15:07:16 +00:00
|
|
|
debug "Computation committed"
|
2018-09-17 23:56:10 +00:00
|
|
|
transaction.commit()
|
2018-10-02 15:07:16 +00:00
|
|
|
else:
|
|
|
|
debug "Computation rolled back due to error"
|
2018-09-14 17:03:26 +00:00
|
|
|
|
2018-10-02 15:07:16 +00:00
|
|
|
proc applyCreateMessage(fork: Fork, computation: var BaseComputation, opCode: static[Op]) =
|
|
|
|
computation.applyMessage(opCode)
|
2018-09-17 23:56:10 +00:00
|
|
|
|
|
|
|
var transaction: DbTransaction
|
|
|
|
defer: transaction.safeDispose()
|
|
|
|
|
2018-09-17 18:35:07 +00:00
|
|
|
if fork >= FkFrontier:
|
2018-09-17 23:56:10 +00:00
|
|
|
transaction = computation.vmState.beginTransaction()
|
2018-09-14 17:03:26 +00:00
|
|
|
|
|
|
|
if computation.isError:
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
let contractCode = computation.output
|
|
|
|
if contractCode.len > 0:
|
2018-09-17 18:35:07 +00:00
|
|
|
if fork >= FkSpurious and contractCode.len >= EIP170_CODE_SIZE_LIMIT:
|
|
|
|
raise newException(OutOfGas, &"Contract code size exceeds EIP170 limit of {EIP170_CODE_SIZE_LIMIT}. Got code of size: {contractCode.len}")
|
|
|
|
|
2018-09-14 17:03:26 +00:00
|
|
|
try:
|
|
|
|
computation.gasMeter.consumeGas(
|
|
|
|
computation.gasCosts[Create].m_handler(0, 0, contractCode.len),
|
2018-09-17 18:35:07 +00:00
|
|
|
reason = "Write contract code for CREATE")
|
2018-09-17 23:56:10 +00:00
|
|
|
|
|
|
|
let storageAddr = computation.msg.storage_address
|
|
|
|
debug "SETTING CODE",
|
|
|
|
address = storageAddr.toHex,
|
|
|
|
length = len(contract_code),
|
|
|
|
hash = contractCode.rlpHash
|
|
|
|
|
|
|
|
computation.vmState.mutateStateDb:
|
|
|
|
db.setCode(storageAddr, contractCode.toRange)
|
|
|
|
|
|
|
|
if transaction != nil:
|
|
|
|
transaction.commit()
|
|
|
|
|
2018-09-14 17:03:26 +00:00
|
|
|
except OutOfGas:
|
2018-09-17 18:35:07 +00:00
|
|
|
if fork == FkFrontier:
|
|
|
|
computation.output = @[]
|
|
|
|
else:
|
|
|
|
# Different from Frontier:
|
|
|
|
# Reverts state on gas failure while writing contract code.
|
2018-10-02 15:07:16 +00:00
|
|
|
# Transaction are reverted automatically by safeDispose.
|
2018-09-17 18:35:07 +00:00
|
|
|
discard
|
2018-09-14 17:03:26 +00:00
|
|
|
else:
|
2018-09-17 23:56:10 +00:00
|
|
|
if transaction != nil:
|
|
|
|
transaction.commit()
|
2018-09-14 17:03:26 +00:00
|
|
|
|
2018-10-02 15:07:16 +00:00
|
|
|
proc generateChildComputation*(fork: Fork, computation: BaseComputation, childMsg: Message, opCode: static[Op]): BaseComputation =
|
2018-09-14 17:03:26 +00:00
|
|
|
var childComp = newBaseComputation(
|
2018-09-07 16:18:46 +00:00
|
|
|
computation.vmState,
|
|
|
|
computation.vmState.blockHeader.blockNumber,
|
2018-09-11 17:48:38 +00:00
|
|
|
childMsg)
|
2018-12-03 10:54:19 +00:00
|
|
|
|
2018-09-18 12:03:22 +00:00
|
|
|
# Copy the fork op code executor proc (assumes child computation is in the same fork)
|
|
|
|
childComp.opCodeExec = computation.opCodeExec
|
2018-12-03 10:54:19 +00:00
|
|
|
|
2018-09-11 17:48:38 +00:00
|
|
|
if childMsg.isCreate:
|
2018-10-02 15:07:16 +00:00
|
|
|
fork.applyCreateMessage(childComp, opCode)
|
2018-09-07 16:18:46 +00:00
|
|
|
else:
|
2018-10-02 15:07:16 +00:00
|
|
|
applyMessage(childComp, opCode)
|
2018-09-14 17:03:26 +00:00
|
|
|
return childComp
|
2018-09-07 16:18:46 +00:00
|
|
|
|
2018-09-14 17:03:26 +00:00
|
|
|
proc addChildComputation(fork: Fork, computation: BaseComputation, child: BaseComputation) =
|
2018-09-07 16:18:46 +00: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
|
|
|
|
computation.children.add(child)
|
|
|
|
|
2018-10-02 15:07:16 +00:00
|
|
|
proc applyChildComputation*(computation: BaseComputation, childMsg: Message, opCode: static[Op]): BaseComputation =
|
2018-09-07 16:18:46 +00:00
|
|
|
## Apply the vm message childMsg as a child computation.
|
2018-09-14 17:03:26 +00:00
|
|
|
let fork = computation.vmState.blockHeader.blockNumber.toFork
|
2018-10-02 15:07:16 +00:00
|
|
|
result = fork.generateChildComputation(computation, childMsg, opCode)
|
2018-09-14 17:03:26 +00:00
|
|
|
fork.addChildComputation(computation, result)
|
2018-01-16 17:05:20 +00: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 07:52:31 +00:00
|
|
|
proc registerAccountForDeletion*(c: var BaseComputation, beneficiary: EthAddress) =
|
2018-01-16 17:05:20 +00: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
|
|
|
|
|
2018-07-06 15:08:31 +00:00
|
|
|
proc addLogEntry*(c: var BaseComputation, account: EthAddress, topics: seq[UInt256], data: seq[byte]) =
|
2018-01-16 17:05:20 +00:00
|
|
|
c.logEntries.add((account, topics, data))
|
|
|
|
|
2018-03-13 14:30:38 +00:00
|
|
|
# many methods are basically TODO, but they still return valid values
|
|
|
|
# in order to test some existing code
|
2018-09-14 13:43:40 +00:00
|
|
|
func getAccountsForDeletion*(c: BaseComputation): seq[EthAddress] =
|
2018-01-16 17:05:20 +00:00
|
|
|
# TODO
|
|
|
|
if c.isError:
|
|
|
|
result = @[]
|
|
|
|
else:
|
2018-09-18 14:07:04 +00:00
|
|
|
result = @[]
|
|
|
|
for account in c.accountsToDelete.keys:
|
|
|
|
result.add(account)
|
2018-01-16 17:05:20 +00: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 07:52:31 +00:00
|
|
|
proc getLogEntries*(c: BaseComputation): seq[(string, seq[UInt256], string)] =
|
2018-01-16 17:05:20 +00:00
|
|
|
# TODO
|
|
|
|
if c.isError:
|
|
|
|
result = @[]
|
|
|
|
else:
|
|
|
|
result = @[]
|
2018-01-29 17:40:22 +00: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 07:52:31 +00:00
|
|
|
proc getGasRefund*(c: BaseComputation): GasInt =
|
2018-01-16 17:05:20 +00:00
|
|
|
if c.isError:
|
2018-05-25 10:25:19 +00:00
|
|
|
result = 0
|
2018-01-16 17:05:20 +00:00
|
|
|
else:
|
2018-05-25 10:25:19 +00:00
|
|
|
result = c.gasMeter.gasRefunded + c.children.mapIt(it.getGasRefund()).foldl(a + b, 0'i64)
|
2018-01-16 17:05:20 +00: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 07:52:31 +00:00
|
|
|
proc getGasUsed*(c: BaseComputation): GasInt =
|
2018-01-16 17:05:20 +00:00
|
|
|
if c.shouldBurnGas:
|
|
|
|
result = c.msg.gas
|
|
|
|
else:
|
2018-05-25 10:25:19 +00:00
|
|
|
result = max(0, c.msg.gas - c.gasMeter.gasRemaining)
|
2018-01-16 17:05:20 +00: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 07:52:31 +00:00
|
|
|
proc getGasRemaining*(c: BaseComputation): GasInt =
|
2018-01-16 17:05:20 +00:00
|
|
|
if c.shouldBurnGas:
|
2018-05-25 10:25:19 +00:00
|
|
|
result = 0
|
2018-01-16 17:05:20 +00:00
|
|
|
else:
|
|
|
|
result = c.gasMeter.gasRemaining
|
2018-12-03 10:54:19 +00:00
|
|
|
|
2018-12-03 16:22:08 +00:00
|
|
|
proc tracingEnabled*(c: BaseComputation): bool =
|
2018-12-03 10:54:19 +00:00
|
|
|
c.vmState.tracingEnabled
|
|
|
|
|
2018-12-03 16:22:08 +00:00
|
|
|
proc traceOpCodeStarted*(c: BaseComputation, op: string) =
|
2018-12-03 10:54:19 +00:00
|
|
|
traceOpCodeStarted(c.vmState.tracer, c, op)
|
|
|
|
|
|
|
|
proc traceOpCodeEnded*(c: BaseComputation) =
|
|
|
|
c.vmState.tracer.traceOpCodeEnded(c)
|
2018-12-03 16:22:08 +00:00
|
|
|
|
|
|
|
proc traceError*(c: BaseComputation) =
|
|
|
|
c.vmState.tracer.traceError(c)
|