Merge pull request #2356 from nspcc-dev/rpc-big-numbers
Allow to use big integers in RPC
This commit is contained in:
commit
075fd05bfc
7 changed files with 263 additions and 56 deletions
|
@ -7,6 +7,7 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math/big"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
@ -115,7 +116,7 @@ func (p *Param) GetString() (string, error) {
|
||||||
if err == nil {
|
if err == nil {
|
||||||
p.cache = s
|
p.cache = s
|
||||||
} else {
|
} else {
|
||||||
var i int
|
var i int64
|
||||||
err = json.Unmarshal(p.RawMessage, &i)
|
err = json.Unmarshal(p.RawMessage, &i)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
p.cache = i
|
p.cache = i
|
||||||
|
@ -133,8 +134,8 @@ func (p *Param) GetString() (string, error) {
|
||||||
switch t := p.cache.(type) {
|
switch t := p.cache.(type) {
|
||||||
case string:
|
case string:
|
||||||
return t, nil
|
return t, nil
|
||||||
case int:
|
case int64:
|
||||||
return strconv.Itoa(t), nil
|
return strconv.FormatInt(t, 10), nil
|
||||||
case bool:
|
case bool:
|
||||||
if t {
|
if t {
|
||||||
return "true", nil
|
return "true", nil
|
||||||
|
@ -180,7 +181,7 @@ func (p *Param) GetBoolean() (bool, error) {
|
||||||
if err == nil {
|
if err == nil {
|
||||||
p.cache = s
|
p.cache = s
|
||||||
} else {
|
} else {
|
||||||
var i int
|
var i int64
|
||||||
err = json.Unmarshal(p.RawMessage, &i)
|
err = json.Unmarshal(p.RawMessage, &i)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
p.cache = i
|
p.cache = i
|
||||||
|
@ -195,7 +196,7 @@ func (p *Param) GetBoolean() (bool, error) {
|
||||||
return t, nil
|
return t, nil
|
||||||
case string:
|
case string:
|
||||||
return t != "", nil
|
return t != "", nil
|
||||||
case int:
|
case int64:
|
||||||
return t != 0, nil
|
return t != 0, nil
|
||||||
default:
|
default:
|
||||||
return false, errNotABool
|
return false, errNotABool
|
||||||
|
@ -210,20 +211,46 @@ func (p *Param) GetIntStrict() (int, error) {
|
||||||
if p.IsNull() {
|
if p.IsNull() {
|
||||||
return 0, errNotAnInt
|
return 0, errNotAnInt
|
||||||
}
|
}
|
||||||
if p.cache == nil {
|
value, err := p.fillIntCache()
|
||||||
var i int
|
|
||||||
err := json.Unmarshal(p.RawMessage, &i)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return i, errNotAnInt
|
return 0, err
|
||||||
}
|
}
|
||||||
p.cache = i
|
if i, ok := value.(int64); ok && i == int64(int(i)) {
|
||||||
}
|
return int(i), nil
|
||||||
if i, ok := p.cache.(int); ok {
|
|
||||||
return i, nil
|
|
||||||
}
|
}
|
||||||
return 0, errNotAnInt
|
return 0, errNotAnInt
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *Param) fillIntCache() (interface{}, error) {
|
||||||
|
if p.cache != nil {
|
||||||
|
return p.cache, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// We could also try unmarshalling to uint64, but JSON reliably supports numbers
|
||||||
|
// up to 53 bits in size.
|
||||||
|
var i int64
|
||||||
|
err := json.Unmarshal(p.RawMessage, &i)
|
||||||
|
if err == nil {
|
||||||
|
p.cache = i
|
||||||
|
return i, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var s string
|
||||||
|
err = json.Unmarshal(p.RawMessage, &s)
|
||||||
|
if err == nil {
|
||||||
|
p.cache = s
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var b bool
|
||||||
|
err = json.Unmarshal(p.RawMessage, &b)
|
||||||
|
if err == nil {
|
||||||
|
p.cache = b
|
||||||
|
return b, nil
|
||||||
|
}
|
||||||
|
return nil, errNotAnInt
|
||||||
|
}
|
||||||
|
|
||||||
// GetInt returns int value of the parameter or tries to cast parameter to an int value.
|
// GetInt returns int value of the parameter or tries to cast parameter to an int value.
|
||||||
func (p *Param) GetInt() (int, error) {
|
func (p *Param) GetInt() (int, error) {
|
||||||
if p == nil {
|
if p == nil {
|
||||||
|
@ -232,30 +259,16 @@ func (p *Param) GetInt() (int, error) {
|
||||||
if p.IsNull() {
|
if p.IsNull() {
|
||||||
return 0, errNotAnInt
|
return 0, errNotAnInt
|
||||||
}
|
}
|
||||||
if p.cache == nil {
|
value, err := p.fillIntCache()
|
||||||
var i int
|
if err != nil {
|
||||||
err := json.Unmarshal(p.RawMessage, &i)
|
return 0, err
|
||||||
if err == nil {
|
}
|
||||||
p.cache = i
|
switch t := value.(type) {
|
||||||
} else {
|
case int64:
|
||||||
var s string
|
if t == int64(int(t)) {
|
||||||
err = json.Unmarshal(p.RawMessage, &s)
|
return int(t), nil
|
||||||
if err == nil {
|
}
|
||||||
p.cache = s
|
|
||||||
} else {
|
|
||||||
var b bool
|
|
||||||
err = json.Unmarshal(p.RawMessage, &b)
|
|
||||||
if err == nil {
|
|
||||||
p.cache = b
|
|
||||||
} else {
|
|
||||||
return 0, errNotAnInt
|
return 0, errNotAnInt
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
switch t := p.cache.(type) {
|
|
||||||
case int:
|
|
||||||
return t, nil
|
|
||||||
case string:
|
case string:
|
||||||
return strconv.Atoi(t)
|
return strconv.Atoi(t)
|
||||||
case bool:
|
case bool:
|
||||||
|
@ -264,7 +277,38 @@ func (p *Param) GetInt() (int, error) {
|
||||||
}
|
}
|
||||||
return 0, nil
|
return 0, nil
|
||||||
default:
|
default:
|
||||||
return 0, errNotAnInt
|
panic("unreachable")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetBigInt returns big-interer value of the parameter.
|
||||||
|
func (p *Param) GetBigInt() (*big.Int, error) {
|
||||||
|
if p == nil {
|
||||||
|
return nil, errMissingParameter
|
||||||
|
}
|
||||||
|
if p.IsNull() {
|
||||||
|
return nil, errNotAnInt
|
||||||
|
}
|
||||||
|
value, err := p.fillIntCache()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
switch t := value.(type) {
|
||||||
|
case int64:
|
||||||
|
return big.NewInt(t), nil
|
||||||
|
case string:
|
||||||
|
bi, ok := new(big.Int).SetString(t, 10)
|
||||||
|
if !ok {
|
||||||
|
return nil, errNotAnInt
|
||||||
|
}
|
||||||
|
return bi, nil
|
||||||
|
case bool:
|
||||||
|
if t {
|
||||||
|
return big.NewInt(1), nil
|
||||||
|
}
|
||||||
|
return new(big.Int), nil
|
||||||
|
default:
|
||||||
|
panic("unreachable")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,8 @@ import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math"
|
||||||
|
"math/big"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
@ -190,6 +192,48 @@ func TestParam_UnmarshalJSON(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestGetBigInt(t *testing.T) {
|
||||||
|
maxUint64 := new(big.Int).SetUint64(math.MaxUint64)
|
||||||
|
minInt64 := big.NewInt(math.MinInt64)
|
||||||
|
testCases := []struct {
|
||||||
|
raw string
|
||||||
|
expected *big.Int
|
||||||
|
}{
|
||||||
|
{"true", big.NewInt(1)},
|
||||||
|
{"false", new(big.Int)},
|
||||||
|
{"42", big.NewInt(42)},
|
||||||
|
{`"` + minInt64.String() + `"`, minInt64},
|
||||||
|
{`"` + maxUint64.String() + `"`, maxUint64},
|
||||||
|
{`"` + minInt64.String() + `000"`, new(big.Int).Mul(minInt64, big.NewInt(1000))},
|
||||||
|
{`"` + maxUint64.String() + `000"`, new(big.Int).Mul(maxUint64, big.NewInt(1000))},
|
||||||
|
{`"abc"`, nil},
|
||||||
|
{`[]`, nil},
|
||||||
|
{`null`, nil},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range testCases {
|
||||||
|
var p Param
|
||||||
|
require.NoError(t, json.Unmarshal([]byte(tc.raw), &p))
|
||||||
|
|
||||||
|
actual, err := p.GetBigInt()
|
||||||
|
if tc.expected == nil {
|
||||||
|
require.Error(t, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, tc.expected, actual)
|
||||||
|
|
||||||
|
expected := tc.expected.Int64()
|
||||||
|
actualInt, err := p.GetInt()
|
||||||
|
if !actual.IsInt64() || int64(int(expected)) != expected {
|
||||||
|
require.Error(t, err)
|
||||||
|
} else {
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, int(expected), actualInt)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestGetWitness(t *testing.T) {
|
func TestGetWitness(t *testing.T) {
|
||||||
accountHash, err := util.Uint160DecodeStringLE("cadb3dc2faa3ef14a13b619c9a43124755aa2569")
|
accountHash, err := util.Uint160DecodeStringLE("cadb3dc2faa3ef14a13b619c9a43124755aa2569")
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
|
@ -63,11 +63,11 @@ func ExpandArrayIntoScript(script *io.BinWriter, slice []Param) error {
|
||||||
}
|
}
|
||||||
emit.Bytes(script, key.Bytes())
|
emit.Bytes(script, key.Bytes())
|
||||||
case smartcontract.IntegerType:
|
case smartcontract.IntegerType:
|
||||||
val, err := fp.Value.GetInt()
|
bi, err := fp.Value.GetBigInt()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
emit.Int(script, int64(val))
|
emit.BigInt(script, bi)
|
||||||
case smartcontract.BoolType:
|
case smartcontract.BoolType:
|
||||||
val, err := fp.Value.GetBoolean() // not GetBooleanStrict(), because that's the way C# code works
|
val, err := fp.Value.GetBoolean() // not GetBooleanStrict(), because that's the way C# code works
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -97,7 +97,7 @@ func ExpandArrayIntoScript(script *io.BinWriter, slice []Param) error {
|
||||||
return fmt.Errorf("parameter type %v is not supported", fp.Type)
|
return fmt.Errorf("parameter type %v is not supported", fp.Type)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return script.Err
|
||||||
}
|
}
|
||||||
|
|
||||||
// CreateFunctionInvocationScript creates a script to invoke given contract with
|
// CreateFunctionInvocationScript creates a script to invoke given contract with
|
||||||
|
|
|
@ -3,6 +3,7 @@ package request
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math/big"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/io"
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
||||||
|
@ -100,6 +101,10 @@ func TestInvocationScriptCreationBad(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestExpandArrayIntoScript(t *testing.T) {
|
func TestExpandArrayIntoScript(t *testing.T) {
|
||||||
|
bi := new(big.Int).Lsh(big.NewInt(1), 254)
|
||||||
|
rawInt := make([]byte, 32)
|
||||||
|
rawInt[31] = 0x40
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
Input []Param
|
Input []Param
|
||||||
Expected []byte
|
Expected []byte
|
||||||
|
@ -112,6 +117,10 @@ func TestExpandArrayIntoScript(t *testing.T) {
|
||||||
Input: []Param{{RawMessage: []byte(`{"type": "Array", "value": [{"type": "String", "value": "a"}]}`)}},
|
Input: []Param{{RawMessage: []byte(`{"type": "Array", "value": [{"type": "String", "value": "a"}]}`)}},
|
||||||
Expected: []byte{byte(opcode.PUSHDATA1), 1, byte('a'), byte(opcode.PUSH1), byte(opcode.PACK)},
|
Expected: []byte{byte(opcode.PUSHDATA1), 1, byte('a'), byte(opcode.PUSH1), byte(opcode.PACK)},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
Input: []Param{{RawMessage: []byte(`{"type": "Integer", "value": "` + bi.String() + `"}`)}},
|
||||||
|
Expected: append([]byte{byte(opcode.PUSHINT256)}, rawInt...),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
for _, c := range testCases {
|
for _, c := range testCases {
|
||||||
script := io.NewBufBinWriter()
|
script := io.NewBufBinWriter()
|
||||||
|
@ -126,6 +135,10 @@ func TestExpandArrayIntoScript(t *testing.T) {
|
||||||
{
|
{
|
||||||
{RawMessage: []byte(`{"type": "Array", "value": null}`)},
|
{RawMessage: []byte(`{"type": "Array", "value": null}`)},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
{RawMessage: []byte(`{"type": "Integer", "value": "` +
|
||||||
|
new(big.Int).Lsh(big.NewInt(1), 255).String() + `"}`)},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
for _, c := range errorCases {
|
for _, c := range errorCases {
|
||||||
script := io.NewBufBinWriter()
|
script := io.NewBufBinWriter()
|
||||||
|
|
|
@ -52,18 +52,43 @@ func padRight(s int, buf []byte) []byte {
|
||||||
|
|
||||||
// Int emits a int type to the given buffer.
|
// Int emits a int type to the given buffer.
|
||||||
func Int(w *io.BinWriter, i int64) {
|
func Int(w *io.BinWriter, i int64) {
|
||||||
|
if smallInt(w, i) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
bigInt(w, big.NewInt(i), false)
|
||||||
|
}
|
||||||
|
|
||||||
|
// BigInt emits big-integer to the given buffer.
|
||||||
|
func BigInt(w *io.BinWriter, n *big.Int) {
|
||||||
|
bigInt(w, n, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
func smallInt(w *io.BinWriter, i int64) bool {
|
||||||
switch {
|
switch {
|
||||||
case i == -1:
|
case i == -1:
|
||||||
Opcodes(w, opcode.PUSHM1)
|
Opcodes(w, opcode.PUSHM1)
|
||||||
case i >= 0 && i < 16:
|
case i >= 0 && i < 16:
|
||||||
val := opcode.Opcode(int(opcode.PUSH1) - 1 + int(i))
|
val := opcode.Opcode(int(opcode.PUSH0) + int(i))
|
||||||
Opcodes(w, val)
|
Opcodes(w, val)
|
||||||
default:
|
default:
|
||||||
bigInt(w, big.NewInt(i))
|
return false
|
||||||
}
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func bigInt(w *io.BinWriter, n *big.Int, trySmall bool) {
|
||||||
|
if w.Err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if trySmall && n.IsInt64() && smallInt(w, n.Int64()) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := stackitem.CheckIntegerSize(n); err != nil {
|
||||||
|
w.Err = err
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func bigInt(w *io.BinWriter, n *big.Int) {
|
|
||||||
buf := bigint.ToPreallocatedBytes(n, make([]byte, 0, 32))
|
buf := bigint.ToPreallocatedBytes(n, make([]byte, 0, 32))
|
||||||
if len(buf) == 0 {
|
if len(buf) == 0 {
|
||||||
Opcodes(w, opcode.PUSH0)
|
Opcodes(w, opcode.PUSH0)
|
||||||
|
@ -101,7 +126,7 @@ func Array(w *io.BinWriter, es ...interface{}) {
|
||||||
case int:
|
case int:
|
||||||
Int(w, int64(e))
|
Int(w, int64(e))
|
||||||
case *big.Int:
|
case *big.Int:
|
||||||
bigInt(w, e)
|
BigInt(w, e)
|
||||||
case string:
|
case string:
|
||||||
String(w, e)
|
String(w, e)
|
||||||
case util.Uint160:
|
case util.Uint160:
|
||||||
|
|
|
@ -86,6 +86,76 @@ func TestEmitInt(t *testing.T) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestEmitBigInt(t *testing.T) {
|
||||||
|
t.Run("biggest positive number", func(t *testing.T) {
|
||||||
|
buf := io.NewBufBinWriter()
|
||||||
|
bi := big.NewInt(1)
|
||||||
|
bi.Lsh(bi, 255)
|
||||||
|
bi.Sub(bi, big.NewInt(1))
|
||||||
|
|
||||||
|
// sanity check
|
||||||
|
require.NotPanics(t, func() { stackitem.NewBigInteger(bi) })
|
||||||
|
|
||||||
|
BigInt(buf.BinWriter, bi)
|
||||||
|
require.NoError(t, buf.Err)
|
||||||
|
|
||||||
|
expected := make([]byte, 33)
|
||||||
|
expected[0] = byte(opcode.PUSHINT256)
|
||||||
|
for i := 1; i < 32; i++ {
|
||||||
|
expected[i] = 0xFF
|
||||||
|
}
|
||||||
|
expected[32] = 0x7F
|
||||||
|
require.Equal(t, expected, buf.Bytes())
|
||||||
|
})
|
||||||
|
t.Run("smallest negative number", func(t *testing.T) {
|
||||||
|
buf := io.NewBufBinWriter()
|
||||||
|
bi := big.NewInt(-1)
|
||||||
|
bi.Lsh(bi, 255)
|
||||||
|
|
||||||
|
// sanity check
|
||||||
|
require.NotPanics(t, func() { stackitem.NewBigInteger(bi) })
|
||||||
|
|
||||||
|
BigInt(buf.BinWriter, bi)
|
||||||
|
require.NoError(t, buf.Err)
|
||||||
|
|
||||||
|
expected := make([]byte, 33)
|
||||||
|
expected[0] = byte(opcode.PUSHINT256)
|
||||||
|
expected[32] = 0x80
|
||||||
|
require.Equal(t, expected, buf.Bytes())
|
||||||
|
})
|
||||||
|
t.Run("biggest positive number plus 1", func(t *testing.T) {
|
||||||
|
buf := io.NewBufBinWriter()
|
||||||
|
bi := big.NewInt(1)
|
||||||
|
bi.Lsh(bi, 255)
|
||||||
|
|
||||||
|
// sanity check
|
||||||
|
require.Panics(t, func() { stackitem.NewBigInteger(bi) })
|
||||||
|
|
||||||
|
BigInt(buf.BinWriter, bi)
|
||||||
|
require.Error(t, buf.Err)
|
||||||
|
|
||||||
|
t.Run("do not clear previous error", func(t *testing.T) {
|
||||||
|
buf.Reset()
|
||||||
|
expected := errors.New("expected")
|
||||||
|
buf.Err = expected
|
||||||
|
BigInt(buf.BinWriter, bi)
|
||||||
|
require.Equal(t, expected, buf.Err)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
t.Run("smallest negative number minus 1", func(t *testing.T) {
|
||||||
|
buf := io.NewBufBinWriter()
|
||||||
|
bi := big.NewInt(-1)
|
||||||
|
bi.Lsh(bi, 255)
|
||||||
|
bi.Sub(bi, big.NewInt(1))
|
||||||
|
|
||||||
|
// sanity check
|
||||||
|
require.Panics(t, func() { stackitem.NewBigInteger(bi) })
|
||||||
|
|
||||||
|
BigInt(buf.BinWriter, bi)
|
||||||
|
require.Error(t, buf.Err)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func getSlice(n int) []byte {
|
func getSlice(n int) []byte {
|
||||||
data := make([]byte, n)
|
data := make([]byte, n)
|
||||||
for i := range data {
|
for i := range data {
|
||||||
|
|
|
@ -384,21 +384,32 @@ type BigInteger big.Int
|
||||||
|
|
||||||
// NewBigInteger returns an new BigInteger object.
|
// NewBigInteger returns an new BigInteger object.
|
||||||
func NewBigInteger(value *big.Int) *BigInteger {
|
func NewBigInteger(value *big.Int) *BigInteger {
|
||||||
// There are 2 cases, when `BitLen` differs from actual size:
|
if err := CheckIntegerSize(value); err != nil {
|
||||||
// 1. Positive integer with highest bit on byte boundary = 1.
|
panic(err)
|
||||||
// 2. Negative integer with highest bit on byte boundary = 1
|
|
||||||
// minus some value. (-0x80 -> 0x80, -0x7F -> 0x81, -0x81 -> 0x7FFF).
|
|
||||||
sz := value.BitLen()
|
|
||||||
if sz > MaxBigIntegerSizeBits {
|
|
||||||
panic(errTooBigInteger)
|
|
||||||
} else if sz == MaxBigIntegerSizeBits {
|
|
||||||
if value.Sign() == 1 || value.TrailingZeroBits() != MaxBigIntegerSizeBits-1 {
|
|
||||||
panic(errTooBigInteger)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return (*BigInteger)(value)
|
return (*BigInteger)(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// CheckIntegerSize checks that value size doesn't exceed VM limit for Interer.
|
||||||
|
func CheckIntegerSize(value *big.Int) error {
|
||||||
|
// There are 2 cases, when `BitLen` differs from 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.
|
// Big casts i to the big.Int type.
|
||||||
func (i *BigInteger) Big() *big.Int {
|
func (i *BigInteger) Big() *big.Int {
|
||||||
return (*big.Int)(i)
|
return (*big.Int)(i)
|
||||||
|
|
Loading…
Reference in a new issue