mirror of
https://github.com/status-im/nim-stint.git
synced 2025-02-19 18:38:13 +00:00
improve io test coverage + compile time test
This commit is contained in:
parent
523b74d0d2
commit
0cd6b2a052
100
stint/io.nim
100
stint/io.nim
@ -47,6 +47,9 @@ func stint*[T: SomeInteger](n: T, bits: static[int]): StInt[bits] {.inline.}=
|
||||
static_check_size(T, bits)
|
||||
when T is SomeSignedInt:
|
||||
if n < 0:
|
||||
# TODO: when bits >= 128, cannot create from
|
||||
# low(int8-64)
|
||||
# see: status-im/nim-stint/issues/92
|
||||
assignLo(result.data, -n)
|
||||
result = -result
|
||||
else:
|
||||
@ -71,19 +74,8 @@ func truncate*(num: Stint or StUint, T: typedesc[SomeInteger]): T {.inline.}=
|
||||
doAssert bitsof(T) <= bitsof(num.data.leastSignificantWord)
|
||||
|
||||
when nimvm:
|
||||
const bits = bitsof(T)
|
||||
let data = num.data.leastSignificantWord
|
||||
type DT = type data
|
||||
|
||||
# we use esoteric type juggling here to trick the Nim VM
|
||||
when bits == 64:
|
||||
cast[T](uint64(data and 0xFFFFFFFF_FFFFFFFF.DT))
|
||||
elif bits == 32:
|
||||
cast[T](uint32(data and 0xFFFFFFFF.DT))
|
||||
elif bits == 16:
|
||||
cast[T](uint16(data and 0xFFFF.DT))
|
||||
else:
|
||||
cast[T](uint8(data and 0xFF.DT))
|
||||
vmIntCast[T](data)
|
||||
else:
|
||||
cast[T](num.data.leastSignificantWord)
|
||||
|
||||
@ -187,9 +179,9 @@ func parse*[bits: static[int]](input: string, T: typedesc[Stint[bits]], radix: s
|
||||
|
||||
# TODO: we can't create the lowest int this way
|
||||
if isNeg:
|
||||
result = -cast[Stint[bits]](no_overflow)
|
||||
result = -convert[T](no_overflow)
|
||||
else:
|
||||
result = cast[Stint[bits]](no_overflow)
|
||||
result = convert[T](no_overflow)
|
||||
|
||||
func fromHex*(T: type StUint, s: string): T {.inline.} =
|
||||
## Convert an hex string to the corresponding unsigned integer
|
||||
@ -215,7 +207,10 @@ func toString*[bits: static[int]](num: StUint[bits], radix: static[uint8] = 10):
|
||||
var (q, r) = divmod(num, base)
|
||||
|
||||
while true:
|
||||
result.add hexChars[r.truncate(int)]
|
||||
when bitsof(r.data) <= 64:
|
||||
result.add hexChars[r.data.int]
|
||||
else:
|
||||
result.add hexChars[r.truncate(int)]
|
||||
if q.isZero:
|
||||
break
|
||||
(q, r) = divmod(q, base)
|
||||
@ -232,23 +227,27 @@ func toString*[bits: static[int]](num: Stint[bits], radix: static[int8] = 10): s
|
||||
# TODO: use static[range[2 .. 16]], not supported at the moment (2018-04-26)
|
||||
|
||||
const hexChars = "0123456789abcdef"
|
||||
const base = radix.int8.stint(bits)
|
||||
const base = radix.int8.stuint(bits)
|
||||
|
||||
result = ""
|
||||
|
||||
type T = Stuint[bits]
|
||||
let isNeg = num.isNegative
|
||||
let num = if radix == 10 and isNeg: -num
|
||||
else: num
|
||||
let num = convert[T](if radix == 10 and isNeg: -num
|
||||
else: num)
|
||||
|
||||
var (q, r) = divmod(num, base)
|
||||
|
||||
while true:
|
||||
result.add hexChars[r.truncate(int)]
|
||||
when bitsof(r.data) <= 64:
|
||||
result.add hexChars[r.data.int]
|
||||
else:
|
||||
result.add hexChars[r.truncate(int)]
|
||||
if q.isZero:
|
||||
break
|
||||
(q, r) = divmod(q, base)
|
||||
|
||||
if isNeg:
|
||||
if isNeg and radix == 10:
|
||||
result.add '-'
|
||||
|
||||
reverse(result)
|
||||
@ -318,24 +317,38 @@ proc initFromBytesBE*[bits: static[int]](val: var Stuint[bits], ba: openarray[by
|
||||
doAssert(ba.len == N)
|
||||
else:
|
||||
doAssert ba.len <= N
|
||||
|
||||
{.pragma: restrict, codegenDecl: "$# __restrict $#".}
|
||||
let r_ptr {.restrict.} = cast[ptr array[N, byte]](val.addr)
|
||||
|
||||
when system.cpuEndian == bigEndian:
|
||||
# TODO: due to https://github.com/status-im/nim-stint/issues/38
|
||||
# We can't cast a stack byte array to stuint with a convenient proc signature.
|
||||
when allowPadding:
|
||||
when system.cpuEndian == bigEndian:
|
||||
let baseIdx = N - val.len
|
||||
for i, b in ba: r_ptr[baseIdx + i] = b
|
||||
else:
|
||||
for i, b in ba: r_ptr[i] = b
|
||||
else:
|
||||
when allowPadding:
|
||||
let baseIdx = ba.len - 1
|
||||
for i, b in ba: r_ptr[baseIdx - i] = b
|
||||
|
||||
when nimvm:
|
||||
when system.cpuEndian == bigEndian:
|
||||
when allowPadding:
|
||||
for i, b in ba: val.data.setByte(baseIdx + i, b)
|
||||
else:
|
||||
for i, b in ba: val.data.setByte(i, b)
|
||||
else:
|
||||
for i, b in ba: r_ptr[N-1 - i] = b
|
||||
when allowPadding:
|
||||
for i, b in ba: val.data.setByte(baseIdx - i, b)
|
||||
else:
|
||||
for i, b in ba: val.data.setByte(N-1 - i, b)
|
||||
else:
|
||||
{.pragma: restrict, codegenDecl: "$# __restrict $#".}
|
||||
let r_ptr {.restrict.} = cast[ptr array[N, byte]](val.addr)
|
||||
|
||||
when system.cpuEndian == bigEndian:
|
||||
# TODO: due to https://github.com/status-im/nim-stint/issues/38
|
||||
# We can't cast a stack byte array to stuint with a convenient proc signature.
|
||||
when allowPadding:
|
||||
for i, b in ba: r_ptr[baseIdx + i] = b
|
||||
else:
|
||||
for i, b in ba: r_ptr[i] = b
|
||||
else:
|
||||
when allowPadding:
|
||||
for i, b in ba: r_ptr[baseIdx - i] = b
|
||||
else:
|
||||
for i, b in ba: r_ptr[N-1 - i] = b
|
||||
|
||||
func significantBytesBE*(val: openarray[byte]): int {.deprecated.}=
|
||||
## Returns the number of significant trailing bytes in a big endian
|
||||
@ -370,10 +383,17 @@ func toByteArrayBE*[bits: static[int]](n: StUint[bits]): array[bits div 8, byte]
|
||||
|
||||
const N = bits div 8
|
||||
|
||||
when system.cpuEndian == bigEndian:
|
||||
result = cast[type result](n)
|
||||
else:
|
||||
{.pragma: restrict, codegenDecl: "$# __restrict $#".}
|
||||
let n_ptr {.restrict.} = cast[ptr array[N, byte]](n.unsafeAddr)
|
||||
when nimvm:
|
||||
for i in 0 ..< N:
|
||||
result[N-1 - i] = n_ptr[i]
|
||||
when system.cpuEndian == bigEndian:
|
||||
result[i] = n.data.getByte(i)
|
||||
else:
|
||||
result[i] = n.data.getByte(N - 1 - i)
|
||||
else:
|
||||
when system.cpuEndian == bigEndian:
|
||||
result = cast[type result](n)
|
||||
else:
|
||||
{.pragma: restrict, codegenDecl: "$# __restrict $#".}
|
||||
let n_ptr {.restrict.} = cast[ptr array[N, byte]](n.unsafeAddr)
|
||||
for i in 0 ..< N:
|
||||
result[N-1 - i] = n_ptr[i]
|
||||
|
@ -7,7 +7,10 @@ func convertImpl[T: IntImpl|UintImpl](x: IntImpl|UintImpl): T {.compileTime.} =
|
||||
result.hi = convertImpl[type(result.hi)](x.hi)
|
||||
result.lo = x.lo
|
||||
|
||||
template convert*[T](x: UintImpl|IntImpl|SomeInteger): T =
|
||||
func convertImpl[T: Stuint|Stint](x: StUint|StInt): T {.compileTime.} =
|
||||
result.data = convertImpl[type(result.data)](x.data)
|
||||
|
||||
template convert*[T](x: Stuint|Stint|UintImpl|IntImpl|SomeInteger): T =
|
||||
when nimvm:
|
||||
# this is a workaround Nim VM inability to cast
|
||||
# something non integer
|
||||
@ -17,11 +20,17 @@ template convert*[T](x: UintImpl|IntImpl|SomeInteger): T =
|
||||
|
||||
func getByte*(x: SomeInteger, pos: int): byte {.compileTime.} =
|
||||
type DT = type x
|
||||
byte((x shr (pos * 8)) and 0xFF.DT)
|
||||
when bitsof(DT) == 8:
|
||||
cast[byte](x)
|
||||
else:
|
||||
byte((x shr (pos * 8)) and 0xFF.DT)
|
||||
|
||||
func getByte*(x: UintImpl | IntImpl, pos: int): byte {.compileTime.} =
|
||||
type DT = type x.leastSignificantWord
|
||||
byte((x shr (pos * 8)).leastSignificantWord and 0xFF.DT)
|
||||
when bitsof(DT) == 8:
|
||||
cast[byte](x.leastSignificantWord)
|
||||
else:
|
||||
byte((x shr (pos * 8)).leastSignificantWord and 0xFF.DT)
|
||||
|
||||
proc setByte*(x: var SomeInteger, pos: int, b: byte) {.compileTime.} =
|
||||
type DT = type x
|
||||
@ -57,3 +66,31 @@ func copyFromArray*(x: var SomeInteger, data: openArray[byte]) {.compileTime.} =
|
||||
doAssert data.len >= size
|
||||
for i in 0 ..< size:
|
||||
x.setByte(i, data[i])
|
||||
|
||||
template vmIntCast*[T](data: SomeInteger): T =
|
||||
type DT = type data
|
||||
const
|
||||
bits = bitsof(T)
|
||||
DTbits = bitsof(DT)
|
||||
|
||||
# we use esoteric type juggling here to trick the Nim VM
|
||||
when bits == 64:
|
||||
when DTbits == 64:
|
||||
cast[T](data)
|
||||
else:
|
||||
cast[T](uint64(data and DT(0xFFFFFFFF_FFFFFFFF)))
|
||||
elif bits == 32:
|
||||
when DTbits == 32:
|
||||
cast[T](data)
|
||||
else:
|
||||
cast[T](uint32(data and DT(0xFFFFFFFF)))
|
||||
elif bits == 16:
|
||||
when DTbits == 16:
|
||||
cast[T](data)
|
||||
else:
|
||||
cast[T](uint16(data and DT(0xFFFF)))
|
||||
else:
|
||||
when DTBits == 8:
|
||||
cast[T](data)
|
||||
else:
|
||||
cast[T](uint8(data and DT(0xFF)))
|
||||
|
@ -7,7 +7,7 @@
|
||||
#
|
||||
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
||||
|
||||
import ./datatypes, ./int_negabs, ./uint_div, ./int_comparison
|
||||
import ./datatypes, ./int_negabs, ./uint_div, ./int_comparison, ./compiletime_helpers
|
||||
|
||||
# Here are the expected signs for division/modulo by opposite signs and both negative numbers
|
||||
# in EVM
|
||||
@ -39,10 +39,17 @@ func divmod*(x, y: SomeSignedInt): tuple[quot, rem: SomeSignedInt] {.inline.}=
|
||||
proc divmod*[T, T2](x, y: IntImpl[T, T2]): tuple[quot, rem: IntImpl[T, T2]] =
|
||||
## Divmod operation for multi-precision signed integer
|
||||
|
||||
result = cast[type result](divmod(
|
||||
cast[UintImpl[T2]](x.abs),
|
||||
cast[UintImpl[T2]](y.abs)
|
||||
))
|
||||
when nimvm:
|
||||
let res = divmod(
|
||||
convert[UintImpl[T2]](x.abs),
|
||||
convert[UintImpl[T2]](y.abs))
|
||||
result.quot = convert[type result.quot](res.quot)
|
||||
result.rem = convert[type result.rem](res.rem)
|
||||
else:
|
||||
result = cast[type result](divmod(
|
||||
cast[UintImpl[T2]](x.abs),
|
||||
cast[UintImpl[T2]](y.abs)
|
||||
))
|
||||
|
||||
if (x.isNegative xor y.isNegative):
|
||||
# If opposite signs
|
||||
|
@ -0,0 +1,5 @@
|
||||
import ../stint, unittest
|
||||
|
||||
suite "int boundchecks":
|
||||
test "to be implemented":
|
||||
discard
|
1404
tests/test_io.nim
1404
tests/test_io.nim
File diff suppressed because it is too large
Load Diff
Loading…
x
Reference in New Issue
Block a user