2021-04-12 17:13:30 +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.
|
|
|
|
|
|
|
|
## EVM Opcode Handler Tables
|
|
|
|
## =========================
|
|
|
|
##
|
|
|
|
|
|
|
|
const
|
|
|
|
noisy {.intdefine.}: int = 0
|
|
|
|
isNoisy {.used.} = noisy > 0
|
|
|
|
|
2021-04-21 17:04:54 +00:00
|
|
|
kludge {.intdefine.}: int = 0
|
|
|
|
breakCircularDependency {.used.} = kludge > 0
|
|
|
|
|
2021-04-12 17:13:30 +00:00
|
|
|
import
|
2021-04-13 16:23:39 +00:00
|
|
|
strformat,
|
2021-04-21 17:04:54 +00:00
|
|
|
./forks_list,
|
2021-04-12 17:13:30 +00:00
|
|
|
./op_codes,
|
|
|
|
./op_handlers/[oph_defs,
|
2021-04-14 10:40:55 +00:00
|
|
|
oph_arithmetic, oph_hash, oph_envinfo, oph_blockdata,
|
2021-04-15 13:40:31 +00:00
|
|
|
oph_memory, oph_push, oph_dup, oph_swap, oph_log,
|
2021-04-16 15:06:02 +00:00
|
|
|
oph_create, oph_call, oph_sysops]
|
2021-04-12 17:13:30 +00:00
|
|
|
|
2021-04-21 17:04:54 +00:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Kludge BEGIN
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
when not breakCircularDependency:
|
|
|
|
const
|
|
|
|
useExecCreate = vm2OpExecCreate
|
|
|
|
useExecCall = vm2OpExecCall
|
|
|
|
|
|
|
|
else:
|
|
|
|
# note: oph_create/call are always imported to check for syntactic corretness,
|
|
|
|
# at the moment, it would not match the other handler lists due to the
|
|
|
|
# fake Computation object definition.
|
|
|
|
const
|
|
|
|
useExecCreate: seq[Vm2OpExec] = @[]
|
|
|
|
useExecCall: seq[Vm2OpExec] = @[]
|
|
|
|
|
|
|
|
ignoreVm2OpExecCreate {.used.} = vm2OpExecCreate
|
|
|
|
ignoreVm2OpExecCall {.used.} = vm2OpExecCall
|
|
|
|
{.warning: "*** Ignoring tables from <oph_create> and <oph_call>".}
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Kludge END
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
const
|
|
|
|
allHandlersList = @[
|
|
|
|
(vm2OpExecArithmetic, "Arithmetic"),
|
|
|
|
(vm2OpExecHash, "Hash"),
|
|
|
|
(vm2OpExecEnvInfo, "EnvInfo"),
|
|
|
|
(vm2OpExecBlockData, "BlockData"),
|
|
|
|
(vm2OpExecMemory, "Memory"),
|
|
|
|
(vm2OpExecPush, "Push"),
|
|
|
|
(vm2OpExecDup, "Dup"),
|
|
|
|
(vm2OpExecSwap, "Swap"),
|
|
|
|
(vm2OpExecLog, "Log"),
|
|
|
|
(useExecCreate, "Create"),
|
|
|
|
(useExecCall, "Call"),
|
|
|
|
(vm2OpExecSysOp, "SysOp")]
|
|
|
|
|
2021-04-12 17:13:30 +00:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Helper
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2021-04-21 17:04:54 +00:00
|
|
|
proc mkOpTable(selected: Fork): array[Op,Vm2OpExec] {.compileTime.} =
|
|
|
|
|
|
|
|
# Collect selected <fork> entries
|
|
|
|
for (subList,subName) in allHandlersList:
|
|
|
|
for w in subList:
|
|
|
|
if selected notin w.forks:
|
|
|
|
continue
|
|
|
|
# definitions must be mutually exclusive
|
|
|
|
var prvInfo = result[w.opCode].info
|
|
|
|
if prvInfo != "" or 0 < result[w.opCode].forks.card:
|
|
|
|
echo &"*** {subName}: duplicate <{w.opCode}> entry: ",
|
|
|
|
&"\"{prvInfo}\" vs. \"{w.info}\""
|
|
|
|
doAssert result[w.opCode].info == ""
|
|
|
|
doAssert result[w.opCode].forks.card == 0
|
|
|
|
result[w.opCode] = w
|
2021-04-14 16:34:24 +00:00
|
|
|
|
2021-04-21 17:04:54 +00:00
|
|
|
# Initialise unused entries
|
2021-04-12 17:13:30 +00:00
|
|
|
for op in Op:
|
2021-04-21 17:04:54 +00:00
|
|
|
if selected notin result[op].forks:
|
2021-04-14 16:34:24 +00:00
|
|
|
result[op] = result[Invalid]
|
|
|
|
result[op].opCode = op
|
2021-04-19 09:15:35 +00:00
|
|
|
if op == Stop:
|
|
|
|
result[op].name = "toBeReplacedByBreak"
|
2021-04-12 17:13:30 +00:00
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
2021-04-20 11:14:43 +00:00
|
|
|
# Public functions
|
2021-04-12 17:13:30 +00:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
2021-04-20 11:22:48 +00:00
|
|
|
#const
|
|
|
|
# vm2OpHandlers* = block:
|
|
|
|
# var rc: array[Fork, array[Op, Vm2OpExec]]
|
|
|
|
# for w in Fork:
|
|
|
|
# rc[w] = w.mkOpTable
|
|
|
|
# rc
|
2021-04-12 17:13:30 +00:00
|
|
|
|
2021-04-20 11:14:43 +00:00
|
|
|
type
|
|
|
|
hdlRec = tuple
|
|
|
|
name: string
|
|
|
|
info: string
|
|
|
|
run: Vm2OpFn
|
2021-04-15 13:40:31 +00:00
|
|
|
|
2021-04-20 11:14:43 +00:00
|
|
|
const
|
|
|
|
# Pack handler record.
|
|
|
|
#
|
|
|
|
# Important:
|
|
|
|
# As of NIM 1.2.10, this mapping to another record is crucial for
|
|
|
|
#
|
|
|
|
# vmOpHandlers[fork][op].run
|
|
|
|
#
|
|
|
|
# to pick right function when <op> is a variable . Using
|
|
|
|
#
|
|
|
|
# vm2OpHandlers[fork][op].exec.run
|
|
|
|
#
|
|
|
|
# only works when <op> is a constant. There seems to be some optimisation
|
|
|
|
# that garbles the <exec> sub-structures elements <prep>, <run>, and <post>.
|
|
|
|
# Moreover, <post> is seen NULL under the debugger. It is untested yet
|
|
|
|
# under what circumstances the vm2OpHandlers[] matrix is set up correctly.
|
|
|
|
# Linearising/flattening the index has no effect here.
|
|
|
|
#
|
|
|
|
vmOpHandlers = block:
|
|
|
|
var rc: array[Fork, array[Op, hdlRec]]
|
|
|
|
for fork in Fork:
|
|
|
|
var tab = fork.mkOpTable
|
|
|
|
for op in Op:
|
|
|
|
rc[fork][op].name = tab[op].name
|
|
|
|
rc[fork][op].info = tab[op].info
|
|
|
|
rc[fork][op].run = tab[op].exec.run
|
|
|
|
rc
|
2021-04-15 13:40:31 +00:00
|
|
|
|
2021-04-20 12:07:01 +00:00
|
|
|
proc opHandlersRun*(fork: Fork; op: Op; d: var Vm2Ctx) {.inline.} =
|
2021-04-20 11:14:43 +00:00
|
|
|
## Given a particular `fork` and an `op`-code, run the associated handler
|
|
|
|
vmOpHandlers[fork][op].run(d)
|
2021-04-15 13:40:31 +00:00
|
|
|
|
2021-04-20 11:14:43 +00:00
|
|
|
proc opHandlersName*(fork: Fork; op: Op): string =
|
|
|
|
## Get name (or ID) of op handler
|
|
|
|
vmOpHandlers[fork][op].name
|
2021-04-15 13:40:31 +00:00
|
|
|
|
2021-04-20 11:14:43 +00:00
|
|
|
proc opHandlersInfo*(fork: Fork; op: Op): string =
|
|
|
|
## Get some op handler info
|
|
|
|
vmOpHandlers[fork][op].info
|
2021-04-14 16:34:24 +00:00
|
|
|
|
2021-04-20 11:14:43 +00:00
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# Debugging ...
|
|
|
|
# ------------------------------------------------------------------------------
|
2021-04-14 16:34:24 +00:00
|
|
|
|
2021-04-20 11:14:43 +00:00
|
|
|
when isMainModule and isNoisy:
|
|
|
|
echo ">>> berlin[shl]: ", FkBerlin.opHandlersInfo(Shl)
|
|
|
|
echo ">>> berlin[push32]: ", FkBerlin.opHandlersInfo(Push32)
|
|
|
|
echo ">>> berlin[dup16]: ", FkBerlin.opHandlersInfo(Dup16)
|
|
|
|
echo ">>> berlin[swap16]: ", FkBerlin.opHandlersInfo(Swap16)
|
|
|
|
echo ">>> berlin[log4]: ", FkBerlin.opHandlersInfo(Log4)
|
|
|
|
|
|
|
|
echo ">>> frontier[sstore]: ", FkFrontier.opHandlersInfo(Sstore)
|
|
|
|
echo ">>> constantinople[sstore]: ", FkConstantinople.opHandlersInfo(Sstore)
|
|
|
|
echo ">>> berlin[sstore]: ", FkBerlin.opHandlersInfo(Sstore)
|
2021-04-12 17:13:30 +00:00
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
# End
|
|
|
|
# ------------------------------------------------------------------------------
|