mirror of
https://github.com/status-im/nim-stint.git
synced 2025-02-19 18:38:13 +00:00
Use iterator for word size iteration.
This commit is contained in:
parent
a2220617a0
commit
4ed4252af2
107
src/private/as_words.nim
Normal file
107
src/private/as_words.nim
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
# Mpint
|
||||||
|
# Copyright 2018 Status Research & Development GmbH
|
||||||
|
# Licensed under either of
|
||||||
|
#
|
||||||
|
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or http://www.apache.org/licenses/LICENSE-2.0)
|
||||||
|
# * MIT license ([LICENSE-MIT](LICENSE-MIT) or http://opensource.org/licenses/MIT)
|
||||||
|
#
|
||||||
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
||||||
|
|
||||||
|
import ./uint_type, macros
|
||||||
|
|
||||||
|
macro optim(x: typed): untyped =
|
||||||
|
let size = getSize(x)
|
||||||
|
|
||||||
|
if size > 64:
|
||||||
|
result = quote do:
|
||||||
|
array[`size` div 64, uint64]
|
||||||
|
elif size == 64:
|
||||||
|
result = quote do:
|
||||||
|
uint64
|
||||||
|
elif size == 32:
|
||||||
|
result = quote do:
|
||||||
|
uint32
|
||||||
|
elif size == 16:
|
||||||
|
result = quote do:
|
||||||
|
uint16
|
||||||
|
elif size == 8:
|
||||||
|
result = quote do:
|
||||||
|
uint8
|
||||||
|
else:
|
||||||
|
error "Unreachable path reached"
|
||||||
|
|
||||||
|
# The following iterators allow efficient iteration on multiprecision integers internal representation.
|
||||||
|
# Note: in case the size in one, hopefully the compiler optimizes it.
|
||||||
|
# using a template is not more efficient due to having to allocate the injection variable.
|
||||||
|
# This can be rewritten using macros though
|
||||||
|
#
|
||||||
|
# template asWordsRawZip*(x, y: MpUintImpl, xw, yw: untyped, body: untyped): untyped =
|
||||||
|
# when optim(x) is array:
|
||||||
|
# let
|
||||||
|
# x_ptr = cast[ptr optim(x)](x.unsafeaddr)
|
||||||
|
# y_ptr = cast[ptr optim(y)](y.unsafeaddr)
|
||||||
|
|
||||||
|
# for i in 0..<x_ptr[].len:
|
||||||
|
# let
|
||||||
|
# xw{.inject.} = x_ptr[i]
|
||||||
|
# yw{.inject.} = y_ptr[i]
|
||||||
|
# body
|
||||||
|
# else:
|
||||||
|
# let
|
||||||
|
# xw{.inject.} = cast[optim(x)](x)
|
||||||
|
# yw{.inject.} = cast[optim(y)](y)
|
||||||
|
# body
|
||||||
|
|
||||||
|
iterator asWordsRaw*(n: MpUintImpl): auto =
|
||||||
|
## Iterates over n, as an array of words.
|
||||||
|
## Input:
|
||||||
|
## - n: The Multiprecision int
|
||||||
|
## - nw: A word of the multi-precision int
|
||||||
|
## - body: the operation you want to do
|
||||||
|
## Iteration is done from low to high, not taking endianness in account
|
||||||
|
when optim(`n`) is array:
|
||||||
|
for val in cast[optim(n)](n):
|
||||||
|
yield val
|
||||||
|
else:
|
||||||
|
yield cast[optim(n)](n)
|
||||||
|
|
||||||
|
iterator asWordsRawZip*(x, y: MpUintImpl): auto =
|
||||||
|
## Iterates over n, as an array of words.
|
||||||
|
## Input:
|
||||||
|
## - x, y: The multiprecision ints
|
||||||
|
## - xw, yw: a pair of word of the multi-precision ints
|
||||||
|
## - body: the operation you want to do
|
||||||
|
## Iteration is done from low to high, not taking endianness in account
|
||||||
|
when optim(x) is array:
|
||||||
|
{.pragma: restrict, codegenDecl: "$# __restrict__ $#".}
|
||||||
|
let
|
||||||
|
x_ptr{.restrict.} = cast[ptr optim(x)](x.unsafeaddr)
|
||||||
|
y_ptr{.restrict.} = cast[ptr optim(y)](y.unsafeaddr)
|
||||||
|
|
||||||
|
for i in 0..<x_ptr[].len:
|
||||||
|
yield (x_ptr[i], y_ptr[i])
|
||||||
|
else:
|
||||||
|
yield (cast[optim(x)](x), cast[optim(y)](y))
|
||||||
|
|
||||||
|
iterator asWordsZip*(x, y: MpUintImpl): auto =
|
||||||
|
## Iterates over n, as an array of words.
|
||||||
|
## Input:
|
||||||
|
## - x, y: The multiprecision ints
|
||||||
|
## - xw, yw: a pair of word of the multi-precision ints
|
||||||
|
## - body: the operation you want to do
|
||||||
|
## Iteration is done from Most significant byte to Least significant byte
|
||||||
|
## i.e. memory order for BigEndian, reverse for little endian
|
||||||
|
when optim(x) is array:
|
||||||
|
{.pragma: restrict, codegenDecl: "$# __restrict__ $#".}
|
||||||
|
let
|
||||||
|
x_ptr{.restrict.} = cast[ptr optim(x)](x.unsafeaddr)
|
||||||
|
y_ptr{.restrict.} = cast[ptr optim(y)](y.unsafeaddr)
|
||||||
|
|
||||||
|
when system.cpuEndian == bigEndian:
|
||||||
|
for i in 0..<x_ptr[].len:
|
||||||
|
yield (x_ptr[i], y_ptr[i])
|
||||||
|
else: # littleEndian, the most significant bytes are on the right
|
||||||
|
for i in countdown(x_ptr[].len - 1, 0):
|
||||||
|
yield (x_ptr[i], y_ptr[i])
|
||||||
|
else:
|
||||||
|
yield (cast[optim(x)](x), cast[optim(y)](y))
|
@ -7,94 +7,34 @@
|
|||||||
#
|
#
|
||||||
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
||||||
|
|
||||||
import ./uint_type, macros
|
import ./uint_type, ./as_words
|
||||||
|
|
||||||
macro optim(x: typed): untyped =
|
|
||||||
let size = getSize(x)
|
|
||||||
|
|
||||||
if size > 64:
|
|
||||||
result = quote do:
|
|
||||||
array[`size` div 64, uint64]
|
|
||||||
elif size == 64:
|
|
||||||
result = quote do:
|
|
||||||
uint64
|
|
||||||
elif size == 32:
|
|
||||||
result = quote do:
|
|
||||||
uint32
|
|
||||||
elif size == 16:
|
|
||||||
result = quote do:
|
|
||||||
uint16
|
|
||||||
elif size == 8:
|
|
||||||
result = quote do:
|
|
||||||
uint8
|
|
||||||
else:
|
|
||||||
error "Unreachable path reached"
|
|
||||||
|
|
||||||
func isZero*(n: SomeUnsignedInt): bool {.inline.} =
|
func isZero*(n: SomeUnsignedInt): bool {.inline.} =
|
||||||
n == 0
|
n == 0
|
||||||
|
|
||||||
func isZero*(n: MpUintImpl): bool {.inline.} =
|
func isZero*(n: MpUintImpl): bool {.inline.} =
|
||||||
|
for val in asWordsRaw(n):
|
||||||
|
if val != 0:
|
||||||
|
return false
|
||||||
|
return true
|
||||||
|
|
||||||
when optim(`n`) is array:
|
func `<`*(x, y: MpUintImpl): bool {.inline.}=
|
||||||
for val in cast[optim(n)](n):
|
# Lower comparison for multi-precision integers
|
||||||
if val != 0:
|
for xw, yw in asWordsZip(x, y):
|
||||||
return false
|
if xw != yw:
|
||||||
return true
|
return xw < yw
|
||||||
else:
|
return false # they're equal
|
||||||
cast[optim(n)](n) == 0
|
|
||||||
|
|
||||||
func `<`*(x, y: MpUintImpl): bool {.noInit, inline.}=
|
func `==`*(x, y: MpUintImpl): bool {.inline.}=
|
||||||
|
|
||||||
when optim(x) is array:
|
|
||||||
let
|
|
||||||
x_ptr = cast[ptr optim(x)](x.unsafeaddr)
|
|
||||||
y_ptr = cast[ptr optim(y)](y.unsafeaddr)
|
|
||||||
|
|
||||||
when system.cpuEndian == bigEndian:
|
|
||||||
for i in 0..<x_ptr[].len:
|
|
||||||
if x_ptr[i] != y_ptr[i]:
|
|
||||||
return x_ptr[i] < y_ptr[i]
|
|
||||||
return false # They're equal
|
|
||||||
else: # littleEndian, the most significant bytes are on the right
|
|
||||||
for i in countdown(x_ptr[].len - 1, 0):
|
|
||||||
if x_ptr[i] != y_ptr[i]:
|
|
||||||
return x_ptr[i] < y_ptr[i]
|
|
||||||
return false # They're equal
|
|
||||||
else:
|
|
||||||
cast[optim(x)](x) < cast[optim(y)](y)
|
|
||||||
|
|
||||||
func `==`*(x, y: MpUintImpl): bool {.noInit, inline.}=
|
|
||||||
# Equal comparison for multi-precision integers
|
# Equal comparison for multi-precision integers
|
||||||
|
for xw, yw in asWordsRawZip(x, y):
|
||||||
|
if xw != yw:
|
||||||
|
return false
|
||||||
|
return true # they're equal
|
||||||
|
|
||||||
when optim(x) is array:
|
func `<=`*(x, y: MpUintImpl): bool {.inline.}=
|
||||||
let
|
|
||||||
x_ptr = cast[ptr optim(x)](x.unsafeaddr)
|
|
||||||
y_ptr = cast[ptr optim(y)](y.unsafeaddr)
|
|
||||||
|
|
||||||
for i in 0..<x_ptr[].len:
|
|
||||||
if x_ptr[i] != y_ptr[i]:
|
|
||||||
return false
|
|
||||||
return true
|
|
||||||
else:
|
|
||||||
cast[optim(x)](x) < cast[optim(y)](y)
|
|
||||||
|
|
||||||
func `<=`*(x, y: MpUintImpl): bool {.noInit, inline.}=
|
|
||||||
# Lower or equal comparison for multi-precision integers
|
# Lower or equal comparison for multi-precision integers
|
||||||
|
for xw, yw in asWordsZip(x, y):
|
||||||
when optim(x) is array:
|
if xw != yw:
|
||||||
let
|
return xw < yw
|
||||||
x_ptr = cast[ptr optim(x)](x.unsafeaddr)
|
return true # they're equal
|
||||||
y_ptr = cast[ptr optim(y)](y.unsafeaddr)
|
|
||||||
|
|
||||||
when system.cpuEndian == bigEndian:
|
|
||||||
for i in 0..<x_ptr[].len:
|
|
||||||
if x_ptr[i] != y_ptr[i]:
|
|
||||||
return x_ptr[i] < y_ptr[i]
|
|
||||||
return true # They're equal
|
|
||||||
else: # littleEndian, the most significant bytes are on the right
|
|
||||||
for i in countdown(x_ptr[].len - 1, 0):
|
|
||||||
if x_ptr[i] != y_ptr[i]:
|
|
||||||
return x_ptr[i] < y_ptr[i]
|
|
||||||
return true # They're equal
|
|
||||||
else:
|
|
||||||
cast[optim(x)](x) <= cast[optim(y)](y)
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user