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-03-13 14:30:38 +00:00
|
|
|
import
|
|
|
|
tables,
|
2018-04-06 14:52:10 +00:00
|
|
|
constants, vm_state,
|
2018-05-07 12:41:54 +00:00
|
|
|
opcode_values, stint,
|
2018-03-13 14:30:38 +00:00
|
|
|
vm / [code_stream, gas_meter, memory, message, stack]
|
|
|
|
|
|
|
|
type
|
|
|
|
BaseComputation* = ref object of RootObj
|
|
|
|
# The execution computation
|
|
|
|
vmState*: BaseVMState
|
|
|
|
msg*: Message
|
|
|
|
memory*: Memory
|
|
|
|
stack*: Stack
|
|
|
|
gasMeter*: GasMeter
|
|
|
|
code*: CodeStream
|
|
|
|
children*: seq[BaseComputation]
|
|
|
|
rawOutput*: string
|
|
|
|
returnData*: string
|
|
|
|
error*: Error
|
|
|
|
logEntries*: seq[(string, seq[UInt256], string)]
|
|
|
|
shouldEraseReturnData*: bool
|
|
|
|
accountsToDelete*: Table[string, string]
|
|
|
|
opcodes*: Table[Op, Opcode] # TODO array[Op, Opcode]
|
|
|
|
precompiles*: Table[string, Opcode]
|
2018-05-24 10:01:59 +00:00
|
|
|
gasCosts*: GasCosts # TODO separate opcode processing and gas computation
|
2018-03-13 14:30:38 +00:00
|
|
|
|
|
|
|
Error* = ref object
|
|
|
|
info*: string
|
|
|
|
burnsGas*: bool
|
|
|
|
erasesReturnData*: bool
|
|
|
|
|
|
|
|
Opcode* = ref object of RootObj
|
|
|
|
kind*: Op
|
|
|
|
#of VARIABLE_GAS_COST_OPS:
|
|
|
|
# gasCostHandler*: proc(computation: var BaseComputation): UInt256
|
|
|
|
## so, we could have special logic that separates all gas cost calculations
|
|
|
|
## from actual opcode execution
|
|
|
|
## that's what parity does:
|
|
|
|
## it uses the peek methods of the stack and calculates the cost
|
|
|
|
## then it actually pops/pushes stuff in exec
|
2018-04-06 14:52:10 +00:00
|
|
|
## I followed the py-evm approach which does that in opcode logic
|
2018-05-24 10:01:59 +00:00
|
|
|
gasCostKind*: GasCostKind
|
2018-03-13 14:30:38 +00:00
|
|
|
runLogic*: proc(computation: var BaseComputation)
|
2018-05-24 10:01:59 +00:00
|
|
|
|
|
|
|
GasCostKind* = enum
|
|
|
|
GasZero
|
|
|
|
GasBase
|
|
|
|
GasVeryLow
|
|
|
|
GasLow
|
|
|
|
GasMid
|
|
|
|
GasHigh
|
|
|
|
GasSload
|
|
|
|
GasJumpDest
|
|
|
|
GasSset
|
|
|
|
GasSreset
|
|
|
|
GasExtCode
|
|
|
|
GasCoinbase
|
|
|
|
GasSelfDestruct
|
|
|
|
GasInHandler
|
|
|
|
GasRefundSclear
|
|
|
|
|
|
|
|
GasBalance
|
|
|
|
GasCall
|
|
|
|
GasExp
|
|
|
|
GasSHA3
|
|
|
|
|
|
|
|
GasCosts* = array[GasCostKind, UInt256]
|