neo-go/pkg/vm/stackitem/item.go
Roman Khimov 93ecd61079 stackitem: microoptimize memory management of Make()
Preallocate as much as needed.

Signed-off-by: Roman Khimov <roman@nspcc.ru>
2024-08-27 08:24:52 +03:00

1266 lines
28 KiB
Go

package stackitem
import (
"bytes"
"encoding/hex"
"encoding/json"
"errors"
"fmt"
"math"
"math/big"
"reflect"
"slices"
"unicode/utf8"
"github.com/nspcc-dev/neo-go/pkg/crypto/hash"
"github.com/nspcc-dev/neo-go/pkg/encoding/bigint"
"github.com/nspcc-dev/neo-go/pkg/util"
)
const (
// MaxBigIntegerSizeBits is the maximum size of a BigInt item in bits.
MaxBigIntegerSizeBits = 32 * 8
// MaxSize is the maximum item size allowed in the VM.
MaxSize = math.MaxUint16 * 2
// MaxComparableNumOfItems is the maximum number of items that can be compared for structs.
MaxComparableNumOfItems = MaxDeserialized
// MaxClonableNumOfItems is the maximum number of items that can be cloned in structs.
MaxClonableNumOfItems = MaxDeserialized
// MaxByteArrayComparableSize is the maximum allowed length of a ByteArray for Equals method.
// It is set to be the maximum uint16 value + 1.
MaxByteArrayComparableSize = math.MaxUint16 + 1
// MaxKeySize is the maximum size of a map key.
MaxKeySize = 64
)
// Item represents the "real" value that is pushed on the stack.
type Item interface {
fmt.Stringer
Value() any
// Dup duplicates current Item.
Dup() Item
// TryBool converts Item to a boolean value.
TryBool() (bool, error)
// TryBytes converts Item to a byte slice. If the underlying type is a
// byte slice, it's returned as is without copying.
TryBytes() ([]byte, error)
// TryInteger converts Item to an integer.
TryInteger() (*big.Int, error)
// Equals checks if 2 StackItems are equal.
Equals(s Item) bool
// Type returns stack item type.
Type() Type
// Convert converts Item to another type.
Convert(Type) (Item, error)
}
// Convertible is something that can be converted to/from Item.
type Convertible interface {
ToStackItem() (Item, error)
FromStackItem(Item) error
}
// Equatable describes a special value of Interop that can be compared with
// value of some other Interop that implements Equatable.
type Equatable interface {
// Equals checks if two objects are equal.
Equals(other Equatable) bool
}
var (
// ErrInvalidConversion is returned upon an attempt to make an incorrect
// conversion between item types.
ErrInvalidConversion = errors.New("invalid conversion")
// ErrTooBig is returned when an item exceeds some size constraints, like
// the maximum allowed integer value of the number of elements in an array. It
// can also be returned by serialization functions if the resulting
// value exceeds MaxSize.
ErrTooBig = errors.New("too big")
// ErrReadOnly is returned on attempt to modify immutable stack item.
ErrReadOnly = errors.New("item is read-only")
errTooBigComparable = fmt.Errorf("%w: uncomparable", ErrTooBig)
errTooBigInteger = fmt.Errorf("%w: integer", ErrTooBig)
errTooBigKey = fmt.Errorf("%w: map key", ErrTooBig)
errTooBigSize = fmt.Errorf("%w: size", ErrTooBig)
errTooBigElements = fmt.Errorf("%w: many elements", ErrTooBig)
)
// mkInvConversion creates a conversion error with additional metadata (from and
// to types).
func mkInvConversion(from Item, to Type) error {
return fmt.Errorf("%w: %s/%s", ErrInvalidConversion, from, to)
}
// Make tries to make an appropriate stack item from the provided value.
// It will panic if it's not possible.
func Make(v any) Item {
switch val := v.(type) {
case int:
return (*BigInteger)(big.NewInt(int64(val)))
case int64:
return (*BigInteger)(big.NewInt(val))
case uint8:
return (*BigInteger)(big.NewInt(int64(val)))
case uint16:
return (*BigInteger)(big.NewInt(int64(val)))
case uint32:
return (*BigInteger)(big.NewInt(int64(val)))
case uint64:
return (*BigInteger)(new(big.Int).SetUint64(val))
case []byte:
return NewByteArray(val)
case string:
return NewByteArray([]byte(val))
case bool:
return Bool(val)
case []Item:
return &Array{
value: val,
}
case *big.Int:
return NewBigInteger(val)
case Item:
return val
case []int:
var res = make([]Item, len(val))
for i := range val {
res[i] = Make(val[i])
}
return Make(res)
case []string:
var res = make([]Item, len(val))
for i := range val {
res[i] = Make(val[i])
}
return Make(res)
case []any:
res := make([]Item, len(val))
for i := range val {
res[i] = Make(val[i])
}
return Make(res)
case util.Uint160:
return Make(val.BytesBE())
case util.Uint256:
return Make(val.BytesBE())
case *util.Uint160:
if val == nil {
return Null{}
}
return Make(*val)
case *util.Uint256:
if val == nil {
return Null{}
}
return Make(*val)
case nil:
return Null{}
default:
i64T := reflect.TypeOf(int64(0))
if reflect.TypeOf(val).ConvertibleTo(i64T) {
i64Val := reflect.ValueOf(val).Convert(i64T).Interface()
return Make(i64Val)
}
panic(
fmt.Sprintf(
"invalid stack item type: %v (%v)",
val,
reflect.TypeOf(val),
),
)
}
}
// ToString converts an Item to a string if it is a valid UTF-8.
func ToString(item Item) (string, error) {
bs, err := item.TryBytes()
if err != nil {
return "", err
}
if !utf8.Valid(bs) {
return "", fmt.Errorf("%w: not UTF-8", ErrInvalidValue)
}
return string(bs), nil
}
// convertPrimitive converts a primitive item to the specified type.
func convertPrimitive(item Item, typ Type) (Item, error) {
if item.Type() == typ {
return item, nil
}
switch typ {
case IntegerT:
bi, err := item.TryInteger()
if err != nil {
return nil, err
}
return NewBigInteger(bi), nil
case ByteArrayT, BufferT:
b, err := item.TryBytes()
if err != nil {
return nil, err
}
if typ == BufferT {
return NewBuffer(bytes.Clone(b)), nil
}
// ByteArray can't really be changed, so it's OK to reuse `b`.
return NewByteArray(b), nil
case BooleanT:
b, err := item.TryBool()
if err != nil {
return nil, err
}
return NewBool(b), nil
default:
return nil, mkInvConversion(item, typ)
}
}
// Struct represents a struct on the stack.
type Struct struct {
value []Item
rc
ro
}
// NewStruct returns a new Struct object.
func NewStruct(items []Item) *Struct {
return &Struct{
value: items,
}
}
// Value implements the Item interface.
func (i *Struct) Value() any {
return i.value
}
// Remove removes the element at `pos` index from the Struct value.
// It will panic if a bad index given.
func (i *Struct) Remove(pos int) {
if i.IsReadOnly() {
panic(ErrReadOnly)
}
i.value = append(i.value[:pos], i.value[pos+1:]...)
}
// Append adds an Item to the end of the Struct value.
func (i *Struct) Append(item Item) {
if i.IsReadOnly() {
panic(ErrReadOnly)
}
i.value = append(i.value, item)
}
// Clear removes all elements from the Struct item value.
func (i *Struct) Clear() {
if i.IsReadOnly() {
panic(ErrReadOnly)
}
i.value = i.value[:0]
}
// Len returns the length of the Struct value.
func (i *Struct) Len() int {
return len(i.value)
}
// String implements the Item interface.
func (i *Struct) String() string {
return "Struct"
}
// Dup implements the Item interface.
func (i *Struct) Dup() Item {
// it's a reference type, so no copying here.
return i
}
// TryBool implements the Item interface.
func (i *Struct) TryBool() (bool, error) { return true, nil }
// TryBytes implements the Item interface.
func (i *Struct) TryBytes() ([]byte, error) {
return nil, mkInvConversion(i, ByteArrayT)
}
// TryInteger implements the Item interface.
func (i *Struct) TryInteger() (*big.Int, error) {
return nil, mkInvConversion(i, IntegerT)
}
// Equals implements the Item interface.
func (i *Struct) Equals(s Item) bool {
if s == nil {
return false
}
val, ok := s.(*Struct)
if !ok {
return false
}
var limit = MaxComparableNumOfItems - 1 // 1 for current element.
return i.equalStruct(val, &limit)
}
func (i *Struct) equalStruct(s *Struct, limit *int) bool {
if i == s {
return true
} else if len(i.value) != len(s.value) {
return false
}
var maxComparableSize = MaxByteArrayComparableSize
for j := range i.value {
*limit--
if *limit == 0 {
panic(errTooBigElements)
}
arr, ok := i.value[j].(*ByteArray)
if ok {
if !arr.equalsLimited(s.value[j], &maxComparableSize) {
return false
}
} else {
if maxComparableSize == 0 {
panic(errTooBigComparable)
}
maxComparableSize--
sa, oka := i.value[j].(*Struct)
sb, okb := s.value[j].(*Struct)
if oka && okb {
if !sa.equalStruct(sb, limit) {
return false
}
} else if !i.value[j].Equals(s.value[j]) {
return false
}
}
}
return true
}
// Type implements the Item interface.
func (i *Struct) Type() Type { return StructT }
// Convert implements the Item interface.
func (i *Struct) Convert(typ Type) (Item, error) {
switch typ {
case StructT:
return i, nil
case ArrayT:
return NewArray(slices.Clone(i.value)), nil
case BooleanT:
return NewBool(true), nil
default:
return nil, mkInvConversion(i, typ)
}
}
// Clone returns a Struct with all Struct fields copied by the value.
// Array fields are still copied by reference.
func (i *Struct) Clone() (*Struct, error) {
var limit = MaxClonableNumOfItems - 1 // For this struct itself.
return i.clone(&limit)
}
func (i *Struct) clone(limit *int) (*Struct, error) {
ret := &Struct{value: make([]Item, len(i.value))}
for j := range i.value {
*limit--
if *limit < 0 {
return nil, ErrTooBig
}
switch t := i.value[j].(type) {
case *Struct:
var err error
ret.value[j], err = t.clone(limit)
if err != nil {
return nil, err
}
default:
ret.value[j] = t
}
}
return ret, nil
}
// Null represents null on the stack.
type Null struct{}
// String implements the Item interface.
func (i Null) String() string {
return "Null"
}
// Value implements the Item interface.
func (i Null) Value() any {
return nil
}
// Dup implements the Item interface.
// There is no need to perform a real copy here
// since Null has no internal state.
func (i Null) Dup() Item {
return i
}
// TryBool implements the Item interface.
func (i Null) TryBool() (bool, error) { return false, nil }
// TryBytes implements the Item interface.
func (i Null) TryBytes() ([]byte, error) {
return nil, mkInvConversion(i, ByteArrayT)
}
// TryInteger implements the Item interface.
func (i Null) TryInteger() (*big.Int, error) {
return nil, mkInvConversion(i, IntegerT)
}
// Equals implements the Item interface.
func (i Null) Equals(s Item) bool {
_, ok := s.(Null)
return ok
}
// Type implements the Item interface.
func (i Null) Type() Type { return AnyT }
// Convert implements the Item interface.
func (i Null) Convert(typ Type) (Item, error) {
if typ == AnyT || !typ.IsValid() {
return nil, mkInvConversion(i, typ)
}
return i, nil
}
// BigInteger represents a big integer on the stack.
type BigInteger big.Int
// NewBigInteger returns an new BigInteger object.
func NewBigInteger(value *big.Int) *BigInteger {
if err := CheckIntegerSize(value); err != nil {
panic(err)
}
return (*BigInteger)(value)
}
// CheckIntegerSize checks that the value size doesn't exceed the VM limit for Interer.
func CheckIntegerSize(value *big.Int) error {
// There are 2 cases when `BitLen` differs from the actual size:
// 1. Positive integer with the highest bit on byte boundary = 1.
// 2. Negative integer with the highest bit on byte boundary = 1
// minus some value. (-0x80 -> 0x80, -0x7F -> 0x81, -0x81 -> 0x7FFF).
sz := value.BitLen()
// This check is not required, just an optimization for the common case.
if sz < MaxBigIntegerSizeBits {
return nil
}
if sz > MaxBigIntegerSizeBits {
return errTooBigInteger
}
if value.Sign() == 1 || value.TrailingZeroBits() != MaxBigIntegerSizeBits-1 {
return errTooBigInteger
}
return nil
}
// Big casts i to the big.Int type.
func (i *BigInteger) Big() *big.Int {
return (*big.Int)(i)
}
// Bytes converts i to a slice of bytes.
func (i *BigInteger) Bytes() []byte {
return bigint.ToBytes(i.Big())
}
// TryBool implements the Item interface.
func (i *BigInteger) TryBool() (bool, error) {
return i.Big().Sign() != 0, nil
}
// TryBytes implements the Item interface.
func (i *BigInteger) TryBytes() ([]byte, error) {
return i.Bytes(), nil
}
// TryInteger implements the Item interface.
func (i *BigInteger) TryInteger() (*big.Int, error) {
return i.Big(), nil
}
// Equals implements the Item interface.
func (i *BigInteger) Equals(s Item) bool {
if i == s {
return true
} else if s == nil {
return false
}
val, ok := s.(*BigInteger)
return ok && i.Big().Cmp(val.Big()) == 0
}
// Value implements the Item interface.
func (i *BigInteger) Value() any {
return i.Big()
}
func (i *BigInteger) String() string {
return "BigInteger"
}
// Dup implements the Item interface.
func (i *BigInteger) Dup() Item {
n := new(big.Int)
return (*BigInteger)(n.Set(i.Big()))
}
// Type implements the Item interface.
func (i *BigInteger) Type() Type { return IntegerT }
// Convert implements the Item interface.
func (i *BigInteger) Convert(typ Type) (Item, error) {
return convertPrimitive(i, typ)
}
// MarshalJSON implements the json.Marshaler interface.
func (i *BigInteger) MarshalJSON() ([]byte, error) {
return json.Marshal(i.Big())
}
// Bool represents a boolean Item.
type Bool bool
// NewBool returns an new Bool object.
func NewBool(val bool) Bool {
return Bool(val)
}
// Value implements the Item interface.
func (i Bool) Value() any {
return bool(i)
}
// MarshalJSON implements the json.Marshaler interface.
func (i Bool) MarshalJSON() ([]byte, error) {
return json.Marshal(bool(i))
}
func (i Bool) String() string {
return "Boolean"
}
// Dup implements the Item interface.
func (i Bool) Dup() Item {
return i
}
// TryBool implements the Item interface.
func (i Bool) TryBool() (bool, error) { return bool(i), nil }
// Bytes converts Bool to bytes.
func (i Bool) Bytes() []byte {
if i {
return []byte{1}
}
return []byte{0}
}
// TryBytes implements the Item interface.
func (i Bool) TryBytes() ([]byte, error) {
return i.Bytes(), nil
}
// TryInteger implements the Item interface.
func (i Bool) TryInteger() (*big.Int, error) {
if i {
return big.NewInt(1), nil
}
return big.NewInt(0), nil
}
// Equals implements the Item interface.
func (i Bool) Equals(s Item) bool {
if i == s {
return true
} else if s == nil {
return false
}
val, ok := s.(Bool)
return ok && i == val
}
// Type implements the Item interface.
func (i Bool) Type() Type { return BooleanT }
// Convert implements the Item interface.
func (i Bool) Convert(typ Type) (Item, error) {
return convertPrimitive(i, typ)
}
// ByteArray represents a byte array on the stack.
type ByteArray []byte
// NewByteArray returns an new ByteArray object.
func NewByteArray(b []byte) *ByteArray {
return (*ByteArray)(&b)
}
// Value implements the Item interface.
func (i *ByteArray) Value() any {
return []byte(*i)
}
// MarshalJSON implements the json.Marshaler interface.
func (i *ByteArray) MarshalJSON() ([]byte, error) {
return json.Marshal(hex.EncodeToString(*i))
}
func (i *ByteArray) String() string {
return "ByteString"
}
// TryBool implements the Item interface.
func (i *ByteArray) TryBool() (bool, error) {
if len(*i) > MaxBigIntegerSizeBits/8 {
return false, errTooBigInteger
}
for _, b := range *i {
if b != 0 {
return true, nil
}
}
return false, nil
}
// TryBytes implements the Item interface.
func (i ByteArray) TryBytes() ([]byte, error) {
return i, nil
}
// TryInteger implements the Item interface.
func (i ByteArray) TryInteger() (*big.Int, error) {
if len(i) > MaxBigIntegerSizeBits/8 {
return nil, errTooBigInteger
}
return bigint.FromBytes(i), nil
}
// Equals implements the Item interface.
func (i *ByteArray) Equals(s Item) bool {
var limit = MaxByteArrayComparableSize
return i.equalsLimited(s, &limit)
}
// equalsLimited compares ByteArray with provided stackitem using the limit.
func (i *ByteArray) equalsLimited(s Item, limit *int) bool {
if i == nil {
return s == nil
}
lCurr := len(*i)
if lCurr > *limit || *limit == 0 {
panic(errTooBigComparable)
}
var comparedSize = 1
defer func() { *limit -= comparedSize }()
if s == nil {
return false
}
val, ok := s.(*ByteArray)
if !ok {
return false
}
lOther := len(*val)
comparedSize = max(lCurr, lOther)
if i == val {
return true
}
if lOther > *limit {
panic(errTooBigComparable)
}
return bytes.Equal(*i, *val)
}
// Dup implements the Item interface.
func (i *ByteArray) Dup() Item {
ba := bytes.Clone(*i)
return (*ByteArray)(&ba)
}
// Type implements the Item interface.
func (i *ByteArray) Type() Type { return ByteArrayT }
// Convert implements the Item interface.
func (i *ByteArray) Convert(typ Type) (Item, error) {
return convertPrimitive(i, typ)
}
// Array represents a new Array object.
type Array struct {
value []Item
rc
ro
}
// NewArray returns a new Array object.
func NewArray(items []Item) *Array {
return &Array{
value: items,
}
}
// Value implements the Item interface.
func (i *Array) Value() any {
return i.value
}
// Remove removes the element at `pos` index from Array value.
// It will panics on bad index.
func (i *Array) Remove(pos int) {
if i.IsReadOnly() {
panic(ErrReadOnly)
}
i.value = append(i.value[:pos], i.value[pos+1:]...)
}
// Append adds an Item to the end of the Array value.
func (i *Array) Append(item Item) {
if i.IsReadOnly() {
panic(ErrReadOnly)
}
i.value = append(i.value, item)
}
// Clear removes all elements from the Array item value.
func (i *Array) Clear() {
if i.IsReadOnly() {
panic(ErrReadOnly)
}
i.value = i.value[:0]
}
// Len returns length of Array value.
func (i *Array) Len() int {
return len(i.value)
}
// MarshalJSON implements the json.Marshaler interface.
func (i *Array) MarshalJSON() ([]byte, error) {
return json.Marshal(i.value)
}
func (i *Array) String() string {
return "Array"
}
// TryBool implements the Item interface.
func (i *Array) TryBool() (bool, error) { return true, nil }
// TryBytes implements the Item interface.
func (i *Array) TryBytes() ([]byte, error) {
return nil, mkInvConversion(i, ByteArrayT)
}
// TryInteger implements the Item interface.
func (i *Array) TryInteger() (*big.Int, error) {
return nil, mkInvConversion(i, IntegerT)
}
// Equals implements the Item interface.
func (i *Array) Equals(s Item) bool {
return i == s
}
// Dup implements the Item interface.
func (i *Array) Dup() Item {
// reference type
return i
}
// Type implements the Item interface.
func (i *Array) Type() Type { return ArrayT }
// Convert implements the Item interface.
func (i *Array) Convert(typ Type) (Item, error) {
switch typ {
case ArrayT:
return i, nil
case StructT:
return NewStruct(slices.Clone(i.value)), nil
case BooleanT:
return NewBool(true), nil
default:
return nil, mkInvConversion(i, typ)
}
}
// MapElement is a key-value pair of StackItems.
type MapElement struct {
Key Item
Value Item
}
// Map represents a Map object. It's ordered, so we use slice representation,
// which should be fine for maps with less than 32 or so elements. Given that
// our VM has quite low limit of overall stack items, it should be good enough,
// but it can be extended with a real map for fast random access in the future
// if needed.
type Map struct {
value []MapElement
rc
ro
}
// NewMap returns a new Map object.
func NewMap() *Map {
return &Map{
value: make([]MapElement, 0),
}
}
// NewMapWithValue returns a new Map object filled with the specified value.
func NewMapWithValue(value []MapElement) *Map {
if value != nil {
return &Map{
value: value,
}
}
return NewMap()
}
// Value implements the Item interface.
func (i *Map) Value() any {
return i.value
}
// Clear removes all elements from the Map item value.
func (i *Map) Clear() {
if i.IsReadOnly() {
panic(ErrReadOnly)
}
i.value = i.value[:0]
}
// Len returns the length of the Map value.
func (i *Map) Len() int {
return len(i.value)
}
// TryBool implements the Item interface.
func (i *Map) TryBool() (bool, error) { return true, nil }
// TryBytes implements the Item interface.
func (i *Map) TryBytes() ([]byte, error) {
return nil, mkInvConversion(i, ByteArrayT)
}
// TryInteger implements the Item interface.
func (i *Map) TryInteger() (*big.Int, error) {
return nil, mkInvConversion(i, IntegerT)
}
// Equals implements the Item interface.
func (i *Map) Equals(s Item) bool {
return i == s
}
func (i *Map) String() string {
return "Map"
}
// Index returns an index of the key in map.
func (i *Map) Index(key Item) int {
for k := range i.value {
if i.value[k].Key.Equals(key) {
return k
}
}
return -1
}
// Has checks if the map has the specified key.
func (i *Map) Has(key Item) bool {
return i.Index(key) >= 0
}
// Dup implements the Item interface.
func (i *Map) Dup() Item {
// reference type
return i
}
// Type implements the Item interface.
func (i *Map) Type() Type { return MapT }
// Convert implements the Item interface.
func (i *Map) Convert(typ Type) (Item, error) {
switch typ {
case MapT:
return i, nil
case BooleanT:
return NewBool(true), nil
default:
return nil, mkInvConversion(i, typ)
}
}
// Add adds a key-value pair to the map.
func (i *Map) Add(key, value Item) {
if err := IsValidMapKey(key); err != nil {
panic(err)
}
if i.IsReadOnly() {
panic(ErrReadOnly)
}
index := i.Index(key)
if index >= 0 {
i.value[index].Value = value
} else {
i.value = append(i.value, MapElement{key, value})
}
}
// Drop removes the given index from the map (no bounds check done here).
func (i *Map) Drop(index int) {
if i.IsReadOnly() {
panic(ErrReadOnly)
}
copy(i.value[index:], i.value[index+1:])
i.value = i.value[:len(i.value)-1]
}
// IsValidMapKey checks whether it's possible to use the given Item as a Map
// key.
func IsValidMapKey(key Item) error {
switch key.(type) {
case Bool, *BigInteger:
return nil
case *ByteArray:
size := len(key.Value().([]byte))
if size > MaxKeySize {
return errTooBigKey
}
return nil
default:
return fmt.Errorf("%w: %s map key", ErrInvalidType, key.Type())
}
}
// Interop represents interop data on the stack.
type Interop struct {
value any
}
// NewInterop returns a new Interop object.
func NewInterop(value any) *Interop {
return &Interop{
value: value,
}
}
// Value implements the Item interface.
func (i *Interop) Value() any {
return i.value
}
// String implements stringer interface.
func (i *Interop) String() string {
return "InteropInterface"
}
// Dup implements the Item interface.
func (i *Interop) Dup() Item {
// reference type
return i
}
// TryBool implements the Item interface.
func (i *Interop) TryBool() (bool, error) { return true, nil }
// TryBytes implements the Item interface.
func (i *Interop) TryBytes() ([]byte, error) {
return nil, mkInvConversion(i, ByteArrayT)
}
// TryInteger implements the Item interface.
func (i *Interop) TryInteger() (*big.Int, error) {
return nil, mkInvConversion(i, IntegerT)
}
// Equals implements the Item interface.
func (i *Interop) Equals(s Item) bool {
if i == s {
return true
} else if s == nil {
return false
}
val, ok := s.(*Interop)
if !ok {
return false
}
a, okA := i.value.(Equatable)
b, okB := val.value.(Equatable)
return (okA && okB && a.Equals(b)) || (!okA && !okB && i.value == val.value)
}
// Type implements the Item interface.
func (i *Interop) Type() Type { return InteropT }
// Convert implements the Item interface.
func (i *Interop) Convert(typ Type) (Item, error) {
switch typ {
case InteropT:
return i, nil
case BooleanT:
return NewBool(true), nil
default:
return nil, mkInvConversion(i, typ)
}
}
// MarshalJSON implements the json.Marshaler interface.
func (i *Interop) MarshalJSON() ([]byte, error) {
return json.Marshal(i.value)
}
// Pointer represents a VM-level instruction pointer.
type Pointer struct {
pos int
script []byte
hash util.Uint160
}
// NewPointer returns a new pointer on the specified position.
func NewPointer(pos int, script []byte) *Pointer {
return &Pointer{
pos: pos,
script: script,
hash: hash.Hash160(script),
}
}
// NewPointerWithHash returns a new pointer on the specified position of the
// specified script. It differs from NewPointer in that the script hash is being
// passed explicitly to save on hash calculation. This hash is then being used
// for pointer comparisons.
func NewPointerWithHash(pos int, script []byte, h util.Uint160) *Pointer {
return &Pointer{
pos: pos,
script: script,
hash: h,
}
}
// String implements the Item interface.
func (p *Pointer) String() string {
return "Pointer"
}
// Value implements the Item interface.
func (p *Pointer) Value() any {
return p.pos
}
// Dup implements the Item interface.
func (p *Pointer) Dup() Item {
return &Pointer{
pos: p.pos,
script: p.script,
hash: p.hash,
}
}
// TryBool implements the Item interface.
func (p *Pointer) TryBool() (bool, error) {
return true, nil
}
// TryBytes implements the Item interface.
func (p *Pointer) TryBytes() ([]byte, error) {
return nil, mkInvConversion(p, ByteArrayT)
}
// TryInteger implements the Item interface.
func (p *Pointer) TryInteger() (*big.Int, error) {
return nil, mkInvConversion(p, IntegerT)
}
// Equals implements the Item interface.
func (p *Pointer) Equals(s Item) bool {
if p == s {
return true
}
ptr, ok := s.(*Pointer)
return ok && p.pos == ptr.pos && p.hash == ptr.hash
}
// Type implements the Item interface.
func (p *Pointer) Type() Type {
return PointerT
}
// Convert implements the Item interface.
func (p *Pointer) Convert(typ Type) (Item, error) {
switch typ {
case PointerT:
return p, nil
case BooleanT:
return NewBool(true), nil
default:
return nil, mkInvConversion(p, typ)
}
}
// ScriptHash returns the pointer item hash.
func (p *Pointer) ScriptHash() util.Uint160 {
return p.hash
}
// Position returns the pointer item position.
func (p *Pointer) Position() int {
return p.pos
}
// Buffer represents represents a Buffer stack item.
type Buffer []byte
// NewBuffer returns a new Buffer object.
func NewBuffer(b []byte) *Buffer {
return (*Buffer)(&b)
}
// Value implements the Item interface.
func (i *Buffer) Value() any {
return []byte(*i)
}
// String implements the fmt.Stringer interface.
func (i *Buffer) String() string {
return "Buffer"
}
// TryBool implements the Item interface.
func (i *Buffer) TryBool() (bool, error) {
return true, nil
}
// TryBytes implements the Item interface.
func (i *Buffer) TryBytes() ([]byte, error) {
return *i, nil
}
// TryInteger implements the Item interface.
func (i *Buffer) TryInteger() (*big.Int, error) {
return nil, mkInvConversion(i, IntegerT)
}
// Equals implements the Item interface.
func (i *Buffer) Equals(s Item) bool {
return i == s
}
// Dup implements the Item interface.
func (i *Buffer) Dup() Item {
return i
}
// MarshalJSON implements the json.Marshaler interface.
func (i *Buffer) MarshalJSON() ([]byte, error) {
return json.Marshal(hex.EncodeToString(*i))
}
// Type implements the Item interface.
func (i *Buffer) Type() Type { return BufferT }
// Convert implements the Item interface.
func (i *Buffer) Convert(typ Type) (Item, error) {
switch typ {
case BooleanT:
return NewBool(true), nil
case BufferT:
return i, nil
case ByteArrayT:
return NewByteArray(bytes.Clone(*i)), nil
case IntegerT:
if len(*i) > MaxBigIntegerSizeBits/8 {
return nil, errTooBigInteger
}
return NewBigInteger(bigint.FromBytes(*i)), nil
default:
return nil, mkInvConversion(i, typ)
}
}
// Len returns the length of the Buffer value.
func (i *Buffer) Len() int {
return len(*i)
}
// DeepCopy returns a new deep copy of the provided item.
// Values of Interop items are not deeply copied.
// It does preserve duplicates only for non-primitive types.
func DeepCopy(item Item, asImmutable bool) Item {
seen := make(map[Item]Item, typicalNumOfItems)
return deepCopy(item, seen, asImmutable)
}
func deepCopy(item Item, seen map[Item]Item, asImmutable bool) Item {
if it := seen[item]; it != nil {
return it
}
switch it := item.(type) {
case Null:
return Null{}
case *Array:
arr := NewArray(make([]Item, len(it.value)))
seen[item] = arr
for i := range it.value {
arr.value[i] = deepCopy(it.value[i], seen, asImmutable)
}
arr.MarkAsReadOnly()
return arr
case *Struct:
arr := NewStruct(make([]Item, len(it.value)))
seen[item] = arr
for i := range it.value {
arr.value[i] = deepCopy(it.value[i], seen, asImmutable)
}
arr.MarkAsReadOnly()
return arr
case *Map:
m := NewMap()
seen[item] = m
for i := range it.value {
key := deepCopy(it.value[i].Key, seen,
false) // Key is always primitive and not a Buffer.
value := deepCopy(it.value[i].Value, seen, asImmutable)
m.Add(key, value)
}
m.MarkAsReadOnly()
return m
case *BigInteger:
bi := new(big.Int).Set(it.Big())
return (*BigInteger)(bi)
case *ByteArray:
return NewByteArray(bytes.Clone(*it))
case *Buffer:
if asImmutable {
return NewByteArray(bytes.Clone(*it))
}
return NewBuffer(bytes.Clone(*it))
case Bool:
return it
case *Pointer:
return NewPointerWithHash(it.pos, it.script, it.hash)
case *Interop:
return NewInterop(it.value)
default:
return nil
}
}