Merge pull request #919 from nspcc-dev/feature/arith

vm: reorder arithmetic and bitwise opcodes
This commit is contained in:
Roman Khimov 2020-04-30 20:18:19 +03:00 committed by GitHub
commit 40e153030d
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
10 changed files with 424 additions and 1378 deletions

8
go.sum
View file

@ -129,16 +129,8 @@ github.com/nspcc-dev/dbft v0.0.0-20200117124306-478e5cfbf03a h1:ajvxgEe9qY4vvoSm
github.com/nspcc-dev/dbft v0.0.0-20200117124306-478e5cfbf03a/go.mod h1:/YFK+XOxxg0Bfm6P92lY5eDSLYfp06XOdL8KAVgXjVk= github.com/nspcc-dev/dbft v0.0.0-20200117124306-478e5cfbf03a/go.mod h1:/YFK+XOxxg0Bfm6P92lY5eDSLYfp06XOdL8KAVgXjVk=
github.com/nspcc-dev/dbft v0.0.0-20200219114139-199d286ed6c1 h1:yEx9WznS+rjE0jl0dLujCxuZSIb+UTjF+005TJu/nNI= github.com/nspcc-dev/dbft v0.0.0-20200219114139-199d286ed6c1 h1:yEx9WznS+rjE0jl0dLujCxuZSIb+UTjF+005TJu/nNI=
github.com/nspcc-dev/dbft v0.0.0-20200219114139-199d286ed6c1/go.mod h1:O0qtn62prQSqizzoagHmuuKoz8QMkU3SzBoKdEvm3aQ= github.com/nspcc-dev/dbft v0.0.0-20200219114139-199d286ed6c1/go.mod h1:O0qtn62prQSqizzoagHmuuKoz8QMkU3SzBoKdEvm3aQ=
github.com/nspcc-dev/dbft v0.0.0-20200303183127-36d3da79c682 h1:63OWUolW4GcjJR7cThq8hLnMLTwL+sjO3Qf4fo4sx8w=
github.com/nspcc-dev/dbft v0.0.0-20200303183127-36d3da79c682/go.mod h1:1FYQXSbb6/9HQIkoF8XO7W/S8N7AZRkBsgwbcXRvk0E=
github.com/nspcc-dev/dbft v0.0.0-20200427132226-05feeca847dd h1:4XKbXahJWlhjVx2cETQz9edHQfe3BQ2JjNdvSKFBelY= github.com/nspcc-dev/dbft v0.0.0-20200427132226-05feeca847dd h1:4XKbXahJWlhjVx2cETQz9edHQfe3BQ2JjNdvSKFBelY=
github.com/nspcc-dev/dbft v0.0.0-20200427132226-05feeca847dd/go.mod h1:1FYQXSbb6/9HQIkoF8XO7W/S8N7AZRkBsgwbcXRvk0E= github.com/nspcc-dev/dbft v0.0.0-20200427132226-05feeca847dd/go.mod h1:1FYQXSbb6/9HQIkoF8XO7W/S8N7AZRkBsgwbcXRvk0E=
github.com/nspcc-dev/dbft v0.0.0-20200427132226-15a7927772a4 h1:3cFSp4v2u9+S7K1GdLUOP1680EiGEHSBvSI6G2n8XzY=
github.com/nspcc-dev/dbft v0.0.0-20200427132226-15a7927772a4/go.mod h1:1FYQXSbb6/9HQIkoF8XO7W/S8N7AZRkBsgwbcXRvk0E=
github.com/nspcc-dev/dbft v0.0.0-20200427132226-342f23599814 h1:iNqBioi0RU2VX9UiGl/GfQKBbZrDWq5KSxQG+dgTaqo=
github.com/nspcc-dev/dbft v0.0.0-20200427132226-342f23599814/go.mod h1:1FYQXSbb6/9HQIkoF8XO7W/S8N7AZRkBsgwbcXRvk0E=
github.com/nspcc-dev/dbft v0.0.0-20200427132226-660464796c11 h1:sledsmRo0wzgWNCZir5/CeM0PjhHVP8khnGtOfBCFWk=
github.com/nspcc-dev/dbft v0.0.0-20200427132226-660464796c11/go.mod h1:1FYQXSbb6/9HQIkoF8XO7W/S8N7AZRkBsgwbcXRvk0E=
github.com/nspcc-dev/neo-go v0.73.1-pre.0.20200303142215-f5a1b928ce09/go.mod h1:pPYwPZ2ks+uMnlRLUyXOpLieaDQSEaf4NM3zHVbRjmg= github.com/nspcc-dev/neo-go v0.73.1-pre.0.20200303142215-f5a1b928ce09/go.mod h1:pPYwPZ2ks+uMnlRLUyXOpLieaDQSEaf4NM3zHVbRjmg=
github.com/nspcc-dev/neofs-crypto v0.2.0 h1:ftN+59WqxSWz/RCgXYOfhmltOOqU+udsNQSvN6wkFck= github.com/nspcc-dev/neofs-crypto v0.2.0 h1:ftN+59WqxSWz/RCgXYOfhmltOOqU+udsNQSvN6wkFck=
github.com/nspcc-dev/neofs-crypto v0.2.0/go.mod h1:F/96fUzPM3wR+UGsPi3faVNmFlA9KAEAUQR7dMxZmNA= github.com/nspcc-dev/neofs-crypto v0.2.0/go.mod h1:F/96fUzPM3wR+UGsPi3faVNmFlA9KAEAUQR7dMxZmNA=

View file

@ -616,8 +616,7 @@ func (c *codegen) Visit(node ast.Node) ast.Visitor {
case n.Op == token.NEQ: case n.Op == token.NEQ:
// VM has separate opcodes for number and string equality // VM has separate opcodes for number and string equality
if isStringType(c.typeInfo.Types[n.X].Type) { if isStringType(c.typeInfo.Types[n.X].Type) {
emit.Opcode(c.prog.BinWriter, opcode.EQUAL) emit.Opcode(c.prog.BinWriter, opcode.NOTEQUAL)
emit.Opcode(c.prog.BinWriter, opcode.NOT)
} else { } else {
emit.Opcode(c.prog.BinWriter, opcode.NUMNOTEQUAL) emit.Opcode(c.prog.BinWriter, opcode.NUMNOTEQUAL)
} }

View file

@ -47,18 +47,18 @@ type rpcTestCase struct {
check func(t *testing.T, e *executor, result interface{}) check func(t *testing.T, e *executor, result interface{})
} }
const testContractHash = "7858559a8405f4e4e1834fcff274ddf4f1b8f407" const testContractHash = "642e2f47cc6883f0c196cf0dd3dcbb2333300173"
var rpcTestCases = map[string][]rpcTestCase{ var rpcTestCases = map[string][]rpcTestCase{
"getapplicationlog": { "getapplicationlog": {
{ {
name: "positive", name: "positive",
params: `["fe1a3678b16eca35209acf85397708eb0f1668e4045ad4cd5d2453d3bc0a0a6d"]`, params: `["953b6e5ba85dd068c2eba5afd2ffdf8ff832b3a791976e2fd896baf5737b2616"]`,
result: func(e *executor) interface{} { return &result.ApplicationLog{} }, result: func(e *executor) interface{} { return &result.ApplicationLog{} },
check: func(t *testing.T, e *executor, acc interface{}) { check: func(t *testing.T, e *executor, acc interface{}) {
res, ok := acc.(*result.ApplicationLog) res, ok := acc.(*result.ApplicationLog)
require.True(t, ok) require.True(t, ok)
expectedTxHash, err := util.Uint256DecodeStringLE("fe1a3678b16eca35209acf85397708eb0f1668e4045ad4cd5d2453d3bc0a0a6d") expectedTxHash, err := util.Uint256DecodeStringLE("953b6e5ba85dd068c2eba5afd2ffdf8ff832b3a791976e2fd896baf5737b2616")
require.NoError(t, err) require.NoError(t, err)
assert.Equal(t, expectedTxHash, res.TxHash) assert.Equal(t, expectedTxHash, res.TxHash)
assert.Equal(t, 1, len(res.Executions)) assert.Equal(t, 1, len(res.Executions))

Binary file not shown.

Binary file not shown.

View file

@ -98,38 +98,39 @@ const (
RIGHT Opcode = 0x81 RIGHT Opcode = 0x81
// Bitwise logic // Bitwise logic
INVERT Opcode = 0x83 INVERT Opcode = 0x90
AND Opcode = 0x84 AND Opcode = 0x91
OR Opcode = 0x85 OR Opcode = 0x92
XOR Opcode = 0x86 XOR Opcode = 0x93
EQUAL Opcode = 0x87 EQUAL Opcode = 0x97
NOTEQUAL Opcode = 0x98
// Arithmetic // Arithmetic
INC Opcode = 0x8B SIGN Opcode = 0x99
DEC Opcode = 0x8C ABS Opcode = 0x9A
SIGN Opcode = 0x8D NEGATE Opcode = 0x9B
NEGATE Opcode = 0x8F INC Opcode = 0x9C
ABS Opcode = 0x90 DEC Opcode = 0x9D
NOT Opcode = 0x91 ADD Opcode = 0x9E
NZ Opcode = 0x92 SUB Opcode = 0x9F
ADD Opcode = 0x93 MUL Opcode = 0xA0
SUB Opcode = 0x94 DIV Opcode = 0xA1
MUL Opcode = 0x95 MOD Opcode = 0xA2
DIV Opcode = 0x96 SHL Opcode = 0xA8
MOD Opcode = 0x97 SHR Opcode = 0xA9
SHL Opcode = 0x98 NOT Opcode = 0xAA
SHR Opcode = 0x99 BOOLAND Opcode = 0xAB
BOOLAND Opcode = 0x9A BOOLOR Opcode = 0xAC
BOOLOR Opcode = 0x9B NZ Opcode = 0xB1
NUMEQUAL Opcode = 0x9C NUMEQUAL Opcode = 0xB3
NUMNOTEQUAL Opcode = 0x9E NUMNOTEQUAL Opcode = 0xB4
LT Opcode = 0x9F LT Opcode = 0xB5
GT Opcode = 0xA0 LTE Opcode = 0xB6
LTE Opcode = 0xA1 GT Opcode = 0xB7
GTE Opcode = 0xA2 GTE Opcode = 0xB8
MIN Opcode = 0xA3 MIN Opcode = 0xB9
MAX Opcode = 0xA4 MAX Opcode = 0xBA
WITHIN Opcode = 0xA5 WITHIN Opcode = 0xBB
// Advanced data structures (arrays, structures, maps) // Advanced data structures (arrays, structures, maps)
PACK Opcode = 0xC0 PACK Opcode = 0xC0

View file

@ -84,36 +84,36 @@ func _() {
_ = x[SUBSTR-127] _ = x[SUBSTR-127]
_ = x[LEFT-128] _ = x[LEFT-128]
_ = x[RIGHT-129] _ = x[RIGHT-129]
_ = x[INVERT-131] _ = x[INVERT-144]
_ = x[AND-132] _ = x[AND-145]
_ = x[OR-133] _ = x[OR-146]
_ = x[XOR-134] _ = x[XOR-147]
_ = x[EQUAL-135] _ = x[EQUAL-151]
_ = x[INC-139] _ = x[SIGN-153]
_ = x[DEC-140] _ = x[ABS-154]
_ = x[SIGN-141] _ = x[NEGATE-155]
_ = x[NEGATE-143] _ = x[INC-156]
_ = x[ABS-144] _ = x[DEC-157]
_ = x[NOT-145] _ = x[ADD-158]
_ = x[NZ-146] _ = x[SUB-159]
_ = x[ADD-147] _ = x[MUL-160]
_ = x[SUB-148] _ = x[DIV-161]
_ = x[MUL-149] _ = x[MOD-162]
_ = x[DIV-150] _ = x[SHL-168]
_ = x[MOD-151] _ = x[SHR-169]
_ = x[SHL-152] _ = x[NOT-170]
_ = x[SHR-153] _ = x[BOOLAND-171]
_ = x[BOOLAND-154] _ = x[BOOLOR-172]
_ = x[BOOLOR-155] _ = x[NZ-177]
_ = x[NUMEQUAL-156] _ = x[NUMEQUAL-179]
_ = x[NUMNOTEQUAL-158] _ = x[NUMNOTEQUAL-180]
_ = x[LT-159] _ = x[LT-181]
_ = x[GT-160] _ = x[LTE-182]
_ = x[LTE-161] _ = x[GT-183]
_ = x[GTE-162] _ = x[GTE-184]
_ = x[MIN-163] _ = x[MIN-185]
_ = x[MAX-164] _ = x[MAX-186]
_ = x[WITHIN-165] _ = x[WITHIN-187]
_ = x[PACK-192] _ = x[PACK-192]
_ = x[UNPACK-193] _ = x[UNPACK-193]
_ = x[NEWARRAY0-194] _ = x[NEWARRAY0-194]
@ -139,7 +139,7 @@ func _() {
_ = x[THROWIFNOT-241] _ = x[THROWIFNOT-241]
} }
const _Opcode_name = "PUSHINT8PUSHINT16PUSHINT32PUSHINT64PUSHINT128PUSHINT256PUSHNULLPUSHDATA1PUSHDATA2PUSHDATA4PUSHM1PUSH0PUSH1PUSH2PUSH3PUSH4PUSH5PUSH6PUSH7PUSH8PUSH9PUSH10PUSH11PUSH12PUSH13PUSH14PUSH15PUSH16NOPJMPJMPLJMPIFJMPIFLJMPIFNOTJMPIFNOTLJMPEQJMPEQLJMPNEJMPNELJMPGTJMPGTLJMPGEJMPGELJMPLTJMPLTLJMPLEJMPLELCALLCALLLOLDPUSH1RETAPPCALLSYSCALLTAILCALLDUPFROMALTSTACKTOALTSTACKFROMALTSTACKXDROPXSWAPXTUCKDEPTHDROPDUPNIPOVERPICKROLLROTSWAPTUCKCATSUBSTRLEFTRIGHTINVERTANDORXOREQUALINCDECSIGNNEGATEABSNOTNZADDSUBMULDIVMODSHLSHRBOOLANDBOOLORNUMEQUALNUMNOTEQUALLTGTLTEGTEMINMAXWITHINPACKUNPACKNEWARRAY0NEWARRAYNEWARRAYTNEWSTRUCT0NEWSTRUCTNEWMAPSIZEHASKEYKEYSVALUESPICKITEMAPPENDSETITEMREVERSEITEMSREMOVECLEARITEMSISNULLISTYPECONVERTTHROWTHROWIFNOT" const _Opcode_name = "PUSHINT8PUSHINT16PUSHINT32PUSHINT64PUSHINT128PUSHINT256PUSHNULLPUSHDATA1PUSHDATA2PUSHDATA4PUSHM1PUSH0PUSH1PUSH2PUSH3PUSH4PUSH5PUSH6PUSH7PUSH8PUSH9PUSH10PUSH11PUSH12PUSH13PUSH14PUSH15PUSH16NOPJMPJMPLJMPIFJMPIFLJMPIFNOTJMPIFNOTLJMPEQJMPEQLJMPNEJMPNELJMPGTJMPGTLJMPGEJMPGELJMPLTJMPLTLJMPLEJMPLELCALLCALLLOLDPUSH1RETAPPCALLSYSCALLTAILCALLDUPFROMALTSTACKTOALTSTACKFROMALTSTACKXDROPXSWAPXTUCKDEPTHDROPDUPNIPOVERPICKROLLROTSWAPTUCKCATSUBSTRLEFTRIGHTINVERTANDORXOREQUALSIGNABSNEGATEINCDECADDSUBMULDIVMODSHLSHRNOTBOOLANDBOOLORNZNUMEQUALNUMNOTEQUALLTLTEGTGTEMINMAXWITHINPACKUNPACKNEWARRAY0NEWARRAYNEWARRAYTNEWSTRUCT0NEWSTRUCTNEWMAPSIZEHASKEYKEYSVALUESPICKITEMAPPENDSETITEMREVERSEITEMSREMOVECLEARITEMSISNULLISTYPECONVERTTHROWTHROWIFNOT"
var _Opcode_map = map[Opcode]string{ var _Opcode_map = map[Opcode]string{
0: _Opcode_name[0:8], 0: _Opcode_name[0:8],
@ -216,36 +216,36 @@ var _Opcode_map = map[Opcode]string{
127: _Opcode_name[427:433], 127: _Opcode_name[427:433],
128: _Opcode_name[433:437], 128: _Opcode_name[433:437],
129: _Opcode_name[437:442], 129: _Opcode_name[437:442],
131: _Opcode_name[442:448], 144: _Opcode_name[442:448],
132: _Opcode_name[448:451], 145: _Opcode_name[448:451],
133: _Opcode_name[451:453], 146: _Opcode_name[451:453],
134: _Opcode_name[453:456], 147: _Opcode_name[453:456],
135: _Opcode_name[456:461], 151: _Opcode_name[456:461],
139: _Opcode_name[461:464], 153: _Opcode_name[461:465],
140: _Opcode_name[464:467], 154: _Opcode_name[465:468],
141: _Opcode_name[467:471], 155: _Opcode_name[468:474],
143: _Opcode_name[471:477], 156: _Opcode_name[474:477],
144: _Opcode_name[477:480], 157: _Opcode_name[477:480],
145: _Opcode_name[480:483], 158: _Opcode_name[480:483],
146: _Opcode_name[483:485], 159: _Opcode_name[483:486],
147: _Opcode_name[485:488], 160: _Opcode_name[486:489],
148: _Opcode_name[488:491], 161: _Opcode_name[489:492],
149: _Opcode_name[491:494], 162: _Opcode_name[492:495],
150: _Opcode_name[494:497], 168: _Opcode_name[495:498],
151: _Opcode_name[497:500], 169: _Opcode_name[498:501],
152: _Opcode_name[500:503], 170: _Opcode_name[501:504],
153: _Opcode_name[503:506], 171: _Opcode_name[504:511],
154: _Opcode_name[506:513], 172: _Opcode_name[511:517],
155: _Opcode_name[513:519], 177: _Opcode_name[517:519],
156: _Opcode_name[519:527], 179: _Opcode_name[519:527],
158: _Opcode_name[527:538], 180: _Opcode_name[527:538],
159: _Opcode_name[538:540], 181: _Opcode_name[538:540],
160: _Opcode_name[540:542], 182: _Opcode_name[540:543],
161: _Opcode_name[542:545], 183: _Opcode_name[543:545],
162: _Opcode_name[545:548], 184: _Opcode_name[545:548],
163: _Opcode_name[548:551], 185: _Opcode_name[548:551],
164: _Opcode_name[551:554], 186: _Opcode_name[551:554],
165: _Opcode_name[554:560], 187: _Opcode_name[554:560],
192: _Opcode_name[560:564], 192: _Opcode_name[560:564],
193: _Opcode_name[564:570], 193: _Opcode_name[564:570],
194: _Opcode_name[570:579], 194: _Opcode_name[570:579],

View file

@ -85,9 +85,7 @@ func makeStackItem(v interface{}) StackItem {
value: val, value: val,
} }
case *big.Int: case *big.Int:
return &BigIntegerItem{ return NewBigIntegerItem(val)
value: val,
}
case StackItem: case StackItem:
return val return val
case []int: case []int:
@ -311,6 +309,9 @@ type BigIntegerItem struct {
// NewBigIntegerItem returns an new BigIntegerItem object. // NewBigIntegerItem returns an new BigIntegerItem object.
func NewBigIntegerItem(value *big.Int) *BigIntegerItem { func NewBigIntegerItem(value *big.Int) *BigIntegerItem {
if value.BitLen() > MaxBigIntegerSizeBits {
panic("integer is too big")
}
return &BigIntegerItem{ return &BigIntegerItem{
value: value, value: value,
} }
@ -519,7 +520,11 @@ func (i *ByteArrayItem) TryBytes() ([]byte, error) {
// TryInteger implements StackItem interface. // TryInteger implements StackItem interface.
func (i *ByteArrayItem) TryInteger() (*big.Int, error) { func (i *ByteArrayItem) TryInteger() (*big.Int, error) {
return emit.BytesToInt(i.value), nil bi := emit.BytesToInt(i.value)
if bi.BitLen() > MaxBigIntegerSizeBits {
return nil, errors.New("integer is too big")
}
return bi, nil
} }
// Equals implements StackItem interface. // Equals implements StackItem interface.

View file

@ -726,17 +726,6 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
} }
v.estack.Pop() v.estack.Pop()
case opcode.EQUAL:
b := v.estack.Pop()
if b == nil {
panic("no top-level element found")
}
a := v.estack.Pop()
if a == nil {
panic("no second-to-the-top element found")
}
v.estack.PushVal(a.value.Equals(b.value))
// Bit operations. // Bit operations.
case opcode.INVERT: case opcode.INVERT:
// inplace // inplace
@ -759,50 +748,70 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
v.estack.PushVal(new(big.Int).Xor(b, a)) v.estack.PushVal(new(big.Int).Xor(b, a))
case opcode.EQUAL, opcode.NOTEQUAL:
b := v.estack.Pop()
if b == nil {
panic("no top-level element found")
}
a := v.estack.Pop()
if a == nil {
panic("no second-to-the-top element found")
}
v.estack.PushVal(a.value.Equals(b.value) == (op == opcode.EQUAL))
// Numeric operations. // Numeric operations.
case opcode.SIGN:
x := v.estack.Pop().BigInt()
v.estack.PushVal(x.Sign())
case opcode.ABS:
x := v.estack.Pop().BigInt()
v.estack.PushVal(x.Abs(x))
case opcode.NEGATE:
x := v.estack.Pop().BigInt()
v.estack.PushVal(x.Neg(x))
case opcode.INC:
x := v.estack.Pop().BigInt()
a := new(big.Int).Add(x, big.NewInt(1))
v.estack.PushVal(a)
case opcode.DEC:
x := v.estack.Pop().BigInt()
a := new(big.Int).Sub(x, big.NewInt(1))
v.estack.PushVal(a)
case opcode.ADD: case opcode.ADD:
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
v.checkBigIntSize(a)
b := v.estack.Pop().BigInt() b := v.estack.Pop().BigInt()
v.checkBigIntSize(b)
c := new(big.Int).Add(a, b) c := new(big.Int).Add(a, b)
v.checkBigIntSize(c)
v.estack.PushVal(c) v.estack.PushVal(c)
case opcode.SUB: case opcode.SUB:
b := v.estack.Pop().BigInt() b := v.estack.Pop().BigInt()
v.checkBigIntSize(b)
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
v.checkBigIntSize(a)
c := new(big.Int).Sub(a, b) c := new(big.Int).Sub(a, b)
v.checkBigIntSize(c) v.estack.PushVal(c)
case opcode.MUL:
a := v.estack.Pop().BigInt()
b := v.estack.Pop().BigInt()
c := new(big.Int).Mul(a, b)
v.estack.PushVal(c) v.estack.PushVal(c)
case opcode.DIV: case opcode.DIV:
b := v.estack.Pop().BigInt() b := v.estack.Pop().BigInt()
v.checkBigIntSize(b)
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
v.checkBigIntSize(a)
v.estack.PushVal(new(big.Int).Quo(a, b)) v.estack.PushVal(new(big.Int).Quo(a, b))
case opcode.MUL:
a := v.estack.Pop().BigInt()
v.checkBigIntSize(a)
b := v.estack.Pop().BigInt()
v.checkBigIntSize(b)
c := new(big.Int).Mul(a, b)
v.checkBigIntSize(c)
v.estack.PushVal(c)
case opcode.MOD: case opcode.MOD:
b := v.estack.Pop().BigInt() b := v.estack.Pop().BigInt()
v.checkBigIntSize(b)
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
v.checkBigIntSize(a)
v.estack.PushVal(new(big.Int).Rem(a, b)) v.estack.PushVal(new(big.Int).Rem(a, b))
@ -814,7 +823,6 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
panic(fmt.Sprintf("operand must be between %d and %d", 0, maxSHLArg)) panic(fmt.Sprintf("operand must be between %d and %d", 0, maxSHLArg))
} }
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
v.checkBigIntSize(a)
var item big.Int var item big.Int
if op == opcode.SHL { if op == opcode.SHL {
@ -823,9 +831,12 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
item.Rsh(a, uint(b)) item.Rsh(a, uint(b))
} }
v.checkBigIntSize(&item)
v.estack.PushVal(&item) v.estack.PushVal(&item)
case opcode.NOT:
x := v.estack.Pop().Bool()
v.estack.PushVal(!x)
case opcode.BOOLAND: case opcode.BOOLAND:
b := v.estack.Pop().Bool() b := v.estack.Pop().Bool()
a := v.estack.Pop().Bool() a := v.estack.Pop().Bool()
@ -836,6 +847,10 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
a := v.estack.Pop().Bool() a := v.estack.Pop().Bool()
v.estack.PushVal(a || b) v.estack.PushVal(a || b)
case opcode.NZ:
x := v.estack.Pop().BigInt()
v.estack.PushVal(x.Sign() != 0)
case opcode.NUMEQUAL: case opcode.NUMEQUAL:
b := v.estack.Pop().BigInt() b := v.estack.Pop().BigInt()
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
@ -851,16 +866,16 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
v.estack.PushVal(a.Cmp(b) == -1) v.estack.PushVal(a.Cmp(b) == -1)
case opcode.GT:
b := v.estack.Pop().BigInt()
a := v.estack.Pop().BigInt()
v.estack.PushVal(a.Cmp(b) == 1)
case opcode.LTE: case opcode.LTE:
b := v.estack.Pop().BigInt() b := v.estack.Pop().BigInt()
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
v.estack.PushVal(a.Cmp(b) <= 0) v.estack.PushVal(a.Cmp(b) <= 0)
case opcode.GT:
b := v.estack.Pop().BigInt()
a := v.estack.Pop().BigInt()
v.estack.PushVal(a.Cmp(b) == 1)
case opcode.GTE: case opcode.GTE:
b := v.estack.Pop().BigInt() b := v.estack.Pop().BigInt()
a := v.estack.Pop().BigInt() a := v.estack.Pop().BigInt()
@ -890,38 +905,6 @@ func (v *VM) execute(ctx *Context, op opcode.Opcode, parameter []byte) (err erro
x := v.estack.Pop().BigInt() x := v.estack.Pop().BigInt()
v.estack.PushVal(a.Cmp(x) <= 0 && x.Cmp(b) == -1) v.estack.PushVal(a.Cmp(x) <= 0 && x.Cmp(b) == -1)
case opcode.INC:
x := v.estack.Pop().BigInt()
a := new(big.Int).Add(x, big.NewInt(1))
v.checkBigIntSize(a)
v.estack.PushVal(a)
case opcode.DEC:
x := v.estack.Pop().BigInt()
a := new(big.Int).Sub(x, big.NewInt(1))
v.checkBigIntSize(a)
v.estack.PushVal(a)
case opcode.SIGN:
x := v.estack.Pop().BigInt()
v.estack.PushVal(x.Sign())
case opcode.NEGATE:
x := v.estack.Pop().BigInt()
v.estack.PushVal(x.Neg(x))
case opcode.ABS:
x := v.estack.Pop().BigInt()
v.estack.PushVal(x.Abs(x))
case opcode.NOT:
x := v.estack.Pop().Bool()
v.estack.PushVal(!x)
case opcode.NZ:
x := v.estack.Pop().BigInt()
v.estack.PushVal(x.Cmp(big.NewInt(0)) != 0)
// Object operations // Object operations
case opcode.NEWARRAY0: case opcode.NEWARRAY0:
v.estack.PushVal(&ArrayItem{[]StackItem{}}) v.estack.PushVal(&ArrayItem{[]StackItem{}})
@ -1527,12 +1510,6 @@ func (v *VM) checkInvocationStackSize() {
} }
} }
func (v *VM) checkBigIntSize(a *big.Int) {
if a.BitLen() > MaxBigIntegerSizeBits {
panic("big integer is too big")
}
}
// bytesToPublicKey is a helper deserializing keys using cache and panicing on // bytesToPublicKey is a helper deserializing keys using cache and panicing on
// error. // error.
func (v *VM) bytesToPublicKey(b []byte) *keys.PublicKey { func (v *VM) bytesToPublicKey(b []byte) *keys.PublicKey {

File diff suppressed because it is too large Load diff