nimbus-eth1/src/vm/code_stream.nim

113 lines
2.8 KiB
Nim
Raw Normal View History

import
strformat, strutils, sequtils, sets, macros,
../logging, ../constants, ../opcode_values
type
CodeStream* = ref object
bytes: seq[byte]
depthProcessed: int
invalidPositions: HashSet[int]
pc*: int
logger: Logger
proc `$`*(b: byte): string =
$(b.int)
proc newCodeStream*(codeBytes: seq[byte]): CodeStream =
new(result)
result.bytes = codeBytes
result.pc = 0
result.invalidPositions = initSet[int]()
result.depthProcessed = 0
result.logger = logging.getLogger("vm.code_stream")
proc newCodeStream*(codeBytes: string): CodeStream =
newCodeStream(codeBytes.mapIt(it.byte))
proc read*(c: var CodeStream, size: int): seq[byte] =
if c.pc + size - 1 < c.bytes.len:
result = c.bytes[c.pc .. c.pc + size - 1]
c.pc += size
else:
result = @[]
c.pc = c.bytes.len
proc len*(c: CodeStream): int =
len(c.bytes)
2018-01-23 18:36:23 +00:00
proc next*(c: var CodeStream): Op =
var nextOpcode = c.read(1)
if nextOpcode.len != 0:
2018-01-23 18:36:23 +00:00
return Op(nextOpcode[0])
else:
2018-01-23 18:36:23 +00:00
return Op.STOP
2018-01-23 18:36:23 +00:00
iterator items*(c: var CodeStream): Op =
var nextOpcode = c.next()
2018-01-23 18:36:23 +00:00
while nextOpcode != Op.STOP:
yield nextOpcode
nextOpcode = c.next()
proc `[]`*(c: CodeStream, offset: int): Op =
Op(c.bytes[offset])
2018-01-23 18:36:23 +00:00
proc peek*(c: var CodeStream): Op =
var currentPc = c.pc
result = c.next()
c.pc = currentPc
proc updatePc*(c: var CodeStream, value: int) =
c.pc = min(value, len(c))
macro seek*(c: var CodeStream, pc: int, handler: untyped): untyped =
let c2 = ident("c")
result = quote:
var anchorPc = `c`.pc
`c`.pc = `pc`
try:
var `c2` = `c`
`handler`
finally:
`c`.pc = anchorPc
proc isValidOpcode*(c: var CodeStream, position: int): bool =
if position >= len(c):
return false
if position in c.invalidPositions:
return false
if position <= c.depthProcessed:
return true
else:
var i = c.depthProcessed
while i <= position:
2018-01-23 18:36:23 +00:00
var opcode = Op(c[i])
if opcode >= Op.PUSH1 and opcode <= Op.PUSH32:
var leftBound = (i + 1)
var rightBound = leftBound + (opcode.int - 95)
for z in leftBound ..< rightBound:
c.invalidPositions.incl(z)
i = rightBound
else:
c.depthProcessed = i
i += 1
if position in c.invalidPositions:
return false
else:
return true
proc decompile*(original: CodeStream): seq[(int, Op, string)] =
# behave as https://etherscan.io/opcode-tool
# TODO
result = @[]
var c = newCodeStream(original.bytes)
while true:
var op = c.next
if op >= PUSH1 and op <= PUSH32:
let bytes = c.read(op.int - 95)
result.add((c.pc - 1, op, "0x" & bytes.mapIt($(it.BiggestInt.toHex(2))).join("")))
elif op != Op.Stop:
result.add((c.pc - 1, op, ""))
else:
result.add((-1, Op.STOP, ""))
break