forked from TrueCloudLab/neoneo-go
Merge pull request #1122 from nspcc-dev/fix/printops
Extend PrintOps for JMP* and INIT* instructions
This commit is contained in:
commit
637f75d9ef
1 changed files with 31 additions and 10 deletions
41
pkg/vm/vm.go
41
pkg/vm/vm.go
|
@ -197,12 +197,25 @@ func (v *VM) PrintOps() {
|
||||||
var desc = ""
|
var desc = ""
|
||||||
if parameter != nil {
|
if parameter != nil {
|
||||||
switch instr {
|
switch instr {
|
||||||
case opcode.JMP, opcode.JMPIF, opcode.JMPIFNOT, opcode.CALL:
|
case opcode.JMP, opcode.JMPIF, opcode.JMPIFNOT, opcode.CALL,
|
||||||
offset := int16(binary.LittleEndian.Uint16(parameter))
|
opcode.JMPEQ, opcode.JMPNE,
|
||||||
desc = fmt.Sprintf("%d (%d/%x)", ctx.ip+int(offset), offset, parameter)
|
opcode.JMPGT, opcode.JMPGE, opcode.JMPLE, opcode.JMPLT,
|
||||||
|
opcode.JMPL, opcode.JMPIFL, opcode.JMPIFNOTL, opcode.CALLL,
|
||||||
|
opcode.JMPEQL, opcode.JMPNEL,
|
||||||
|
opcode.JMPGTL, opcode.JMPGEL, opcode.JMPLEL, opcode.JMPLTL:
|
||||||
|
offset, rOffset, err := v.calcJumpOffset(ctx, parameter)
|
||||||
|
if err != nil {
|
||||||
|
desc = fmt.Sprintf("ERROR: %v", err)
|
||||||
|
} else {
|
||||||
|
desc = fmt.Sprintf("%d (%d/%x)", offset, rOffset, parameter)
|
||||||
|
}
|
||||||
case opcode.PUSHA:
|
case opcode.PUSHA:
|
||||||
offset := int32(binary.LittleEndian.Uint32(parameter))
|
offset := int32(binary.LittleEndian.Uint32(parameter))
|
||||||
desc = fmt.Sprintf("%d (%x)", offset, parameter)
|
desc = fmt.Sprintf("%d (%x)", offset, parameter)
|
||||||
|
case opcode.INITSSLOT:
|
||||||
|
desc = fmt.Sprint(parameter[0])
|
||||||
|
case opcode.INITSLOT:
|
||||||
|
desc = fmt.Sprintf("%d local, %d arg", parameter[0], parameter[1])
|
||||||
case opcode.SYSCALL:
|
case opcode.SYSCALL:
|
||||||
desc = fmt.Sprintf("%q", parameter)
|
desc = fmt.Sprintf("%q", parameter)
|
||||||
default:
|
default:
|
||||||
|
@ -1218,7 +1231,7 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
|
||||||
opcode.JMPEQ, opcode.JMPEQL, opcode.JMPNE, opcode.JMPNEL,
|
opcode.JMPEQ, opcode.JMPEQL, opcode.JMPNE, opcode.JMPNEL,
|
||||||
opcode.JMPGT, opcode.JMPGTL, opcode.JMPGE, opcode.JMPGEL,
|
opcode.JMPGT, opcode.JMPGTL, opcode.JMPGE, opcode.JMPGEL,
|
||||||
opcode.JMPLT, opcode.JMPLTL, opcode.JMPLE, opcode.JMPLEL:
|
opcode.JMPLT, opcode.JMPLTL, opcode.JMPLE, opcode.JMPLEL:
|
||||||
offset := v.getJumpOffset(ctx, parameter, 0)
|
offset := v.getJumpOffset(ctx, parameter)
|
||||||
cond := true
|
cond := true
|
||||||
switch op {
|
switch op {
|
||||||
case opcode.JMP, opcode.JMPL:
|
case opcode.JMP, opcode.JMPL:
|
||||||
|
@ -1240,7 +1253,7 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
|
||||||
newCtx.rvcount = -1
|
newCtx.rvcount = -1
|
||||||
v.istack.PushVal(newCtx)
|
v.istack.PushVal(newCtx)
|
||||||
|
|
||||||
offset := v.getJumpOffset(newCtx, parameter, 0)
|
offset := v.getJumpOffset(newCtx, parameter)
|
||||||
v.jumpIf(newCtx, offset, true)
|
v.jumpIf(newCtx, offset, true)
|
||||||
|
|
||||||
case opcode.CALLA:
|
case opcode.CALLA:
|
||||||
|
@ -1423,7 +1436,15 @@ func (v *VM) jumpIf(ctx *Context, offset int, cond bool) {
|
||||||
// to a which JMP should be performed.
|
// to a which JMP should be performed.
|
||||||
// parameter should have length either 1 or 4 and
|
// parameter should have length either 1 or 4 and
|
||||||
// is interpreted as little-endian.
|
// is interpreted as little-endian.
|
||||||
func (v *VM) getJumpOffset(ctx *Context, parameter []byte, mod int) int {
|
func (v *VM) getJumpOffset(ctx *Context, parameter []byte) int {
|
||||||
|
offset, _, err := v.calcJumpOffset(ctx, parameter)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
return offset
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *VM) calcJumpOffset(ctx *Context, parameter []byte) (int, int, error) {
|
||||||
var rOffset int32
|
var rOffset int32
|
||||||
switch l := len(parameter); l {
|
switch l := len(parameter); l {
|
||||||
case 1:
|
case 1:
|
||||||
|
@ -1431,14 +1452,14 @@ func (v *VM) getJumpOffset(ctx *Context, parameter []byte, mod int) int {
|
||||||
case 4:
|
case 4:
|
||||||
rOffset = int32(binary.LittleEndian.Uint32(parameter))
|
rOffset = int32(binary.LittleEndian.Uint32(parameter))
|
||||||
default:
|
default:
|
||||||
panic(fmt.Sprintf("invalid JMP* parameter length: %d", l))
|
return 0, 0, fmt.Errorf("invalid JMP* parameter length: %d", l)
|
||||||
}
|
}
|
||||||
offset := ctx.ip + int(rOffset) + mod
|
offset := ctx.ip + int(rOffset)
|
||||||
if offset < 0 || offset > len(ctx.prog) {
|
if offset < 0 || offset > len(ctx.prog) {
|
||||||
panic(fmt.Sprintf("JMP: invalid offset %d ip at %d", offset, ctx.ip))
|
return 0, 0, fmt.Errorf("invalid offset %d ip at %d", offset, ctx.ip)
|
||||||
}
|
}
|
||||||
|
|
||||||
return offset
|
return offset, int(rOffset), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// CheckMultisigPar checks if sigs contains sufficient valid signatures.
|
// CheckMultisigPar checks if sigs contains sufficient valid signatures.
|
||||||
|
|
Loading…
Reference in a new issue