nim-stew/stew/bitops2.nim

644 lines
23 KiB
Nim
Raw Normal View History

2019-05-08 14:05:16 -06:00
#
#
# Nim's Runtime Library
# (c) Copyright 2017 Nim Authors
2019-07-04 20:30:24 +02:00
# (c) Copyright 2019 Status Research
2019-05-08 14:05:16 -06:00
#
# See the file "copying.txt", included in this
# distribution, for details about the copyright.
#
## This module implements a series of low level methods for bit manipulation.
## By default, this module use compiler intrinsics to improve performance
## on supported compilers: ``GCC``, ``LLVM_GCC``, ``CLANG``, ``VCC``, ``ICC``.
##
## The module will fallback to pure nim procs incase the backend is not supported.
## You can also use the flag `noIntrinsicsBitOpts` to disable compiler intrinsics.
##
## This module is also compatible with other backends: ``Javascript``, ``Nimscript``
## as well as the ``compiletime VM``.
{.push raises: [].}
2020-06-01 19:49:46 +03:00
import
endians2
2019-07-04 20:30:24 +02:00
const
useBuiltins = not defined(noIntrinsicsBitOpts)
template bitsof*(T: typedesc[SomeInteger]): int = 8 * sizeof(T)
template bitsof*(x: SomeInteger): int = 8 * sizeof(x)
2019-05-08 14:05:16 -06:00
type BitIndexable = SomeUnsignedInt
2019-05-08 14:05:16 -06:00
# #### Pure Nim version ####
2019-07-04 20:30:24 +02:00
func nextPow2Nim(x: SomeUnsignedInt): SomeUnsignedInt =
var v = x - 1
# round down, make sure all bits are 1 below the threshold, then add 1
v = v or v shr 1
v = v or v shr 2
v = v or v shr 4
when bitsof(x) > 8:
v = v or v shr 8
when bitsof(x) > 16:
v = v or v shr 16
when bitsof(x) > 32:
v = v or v shr 32
v + 1
2019-05-08 14:05:16 -06:00
func firstOneNim(x: uint32): int =
## Returns the 1-based index of the least significant set bit of x, or if x is zero, returns zero.
# https://graphics.stanford.edu/%7Eseander/bithacks.html#ZerosOnRightMultLookup
const lookup = [0'u8, 1, 28, 2, 29, 14, 24, 3, 30, 22, 20, 15,
25, 17, 4, 8, 31, 27, 13, 23, 21, 19, 16, 7, 26, 12, 18, 6, 11, 5, 10, 9]
2019-07-04 20:30:24 +02:00
if x == 0:
0
else:
let k = not x + 1 # get two's complement
cast[int](1 + lookup[((x and k) * 0x077CB531'u32) shr 27])
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func firstOneNim(x: uint8|uint16): int = firstOneNim(x.uint32)
2019-05-08 14:05:16 -06:00
func firstOneNim(x: uint64): int =
## Returns the 1-based index of the least significant set bit of x, or if x is zero, returns zero.
# https://graphics.stanford.edu/%7Eseander/bithacks.html#ZerosOnRightMultLookup
if (x and uint32.high) == 0:
cast[int](32 + uint(firstOneNim(uint32(x shr 32'u32))))
2019-05-08 14:05:16 -06:00
else:
firstOneNim(uint32(x))
2019-07-04 20:30:24 +02:00
func log2truncNim(x: uint8|uint16|uint32): int =
2019-05-08 14:05:16 -06:00
## Quickly find the log base 2 of a 32-bit or less integer.
# https://graphics.stanford.edu/%7Eseander/bithacks.html#IntegerLogDeBruijn
# https://stackoverflow.com/questions/11376288/fast-computing-of-log2-for-64-bit-integers
const lookup: array[32, uint8] = [0'u8, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18,
22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31]
var v = x.uint32
v = v or v shr 1 # first round down to one less than a power of 2
v = v or v shr 2
v = v or v shr 4
v = v or v shr 8
v = v or v shr 16
cast[int](lookup[uint32(v * 0x07C4ACDD'u32) shr 27])
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func log2truncNim(x: uint64): int =
2019-05-08 14:05:16 -06:00
## Quickly find the log base 2 of a 64-bit integer.
# https://graphics.stanford.edu/%7Eseander/bithacks.html#IntegerLogDeBruijn
# https://stackoverflow.com/questions/11376288/fast-computing-of-log2-for-64-bit-integers
const lookup: array[64, uint8] = [0'u8, 58, 1, 59, 47, 53, 2, 60, 39, 48, 27, 54,
33, 42, 3, 61, 51, 37, 40, 49, 18, 28, 20, 55, 30, 34, 11, 43, 14, 22, 4, 62,
57, 46, 52, 38, 26, 32, 41, 50, 36, 17, 19, 29, 10, 13, 21, 56, 45, 25, 31,
35, 16, 9, 12, 44, 24, 15, 8, 23, 7, 6, 5, 63]
2019-07-04 20:30:24 +02:00
var v = x
2019-05-08 14:05:16 -06:00
v = v or v shr 1 # first round down to one less than a power of 2
v = v or v shr 2
v = v or v shr 4
v = v or v shr 8
v = v or v shr 16
v = v or v shr 32
cast[int](lookup[(v * 0x03F6EAF2CD271461'u64) shr 58])
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func countOnesNim(x: uint8|uint16|uint32): int =
2019-05-08 14:05:16 -06:00
## Counts the set bits in integer. (also called Hamming weight.)
# generic formula is from: https://graphics.stanford.edu/~seander/bithacks.html#CountBitsSetParallel
2019-07-04 20:30:24 +02:00
var v = x.uint32
2019-05-08 14:05:16 -06:00
v = v - ((v shr 1) and 0x55555555)
v = (v and 0x33333333) + ((v shr 2) and 0x33333333)
cast[int](((v + (v shr 4) and 0xF0F0F0F) * 0x1010101) shr 24)
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func countOnesNim(x: uint64): int =
2019-05-08 14:05:16 -06:00
## Counts the set bits in integer. (also called Hamming weight.)
# generic formula is from: https://graphics.stanford.edu/~seander/bithacks.html#CountBitsSetParallel
2019-07-04 20:30:24 +02:00
var v = x
2019-05-08 14:05:16 -06:00
v = v - ((v shr 1'u64) and 0x5555555555555555'u64)
v = (v and 0x3333333333333333'u64) + ((v shr 2'u64) and 0x3333333333333333'u64)
v = (v + (v shr 4'u64) and 0x0F0F0F0F0F0F0F0F'u64)
cast[int]((v * 0x0101010101010101'u64) shr 56'u64)
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func parityNim(x: SomeUnsignedInt): int =
2019-05-08 14:05:16 -06:00
# formula id from: https://graphics.stanford.edu/%7Eseander/bithacks.html#ParityParallel
2019-07-04 20:30:24 +02:00
var v = x
when sizeof(v) == 8:
2019-05-08 14:05:16 -06:00
v = v xor (v shr 32)
2019-07-04 20:30:24 +02:00
when sizeof(v) >= 4:
2019-05-08 14:05:16 -06:00
v = v xor (v shr 16)
2019-07-04 20:30:24 +02:00
when sizeof(v) >= 2:
2019-05-08 14:05:16 -06:00
v = v xor (v shr 8)
v = v xor (v shr 4)
v = v and 0xf
cast[int]((0x6996'u shr v) and 1)
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
when (defined(gcc) or defined(llvm_gcc) or defined(clang)) and useBuiltins:
2019-05-08 14:05:16 -06:00
# Returns the number of set 1-bits in value.
2019-07-04 20:30:24 +02:00
func builtin_popcount(x: cuint): cint {.importc: "__builtin_popcount", nodecl.}
func builtin_popcountll(x: culonglong): cint {.importc: "__builtin_popcountll", nodecl.}
2019-05-08 14:05:16 -06:00
# Returns the bit parity in value
2019-07-04 20:30:24 +02:00
func builtin_parity(x: cuint): cint {.importc: "__builtin_parity", nodecl.}
func builtin_parityll(x: culonglong): cint {.importc: "__builtin_parityll", nodecl.}
2019-05-08 14:05:16 -06:00
# Returns one plus the index of the least significant 1-bit of x, or if x is zero, returns zero.
2019-07-04 20:30:24 +02:00
func builtin_ffs(x: cint): cint {.importc: "__builtin_ffs", nodecl.}
func builtin_ffsll(x: clonglong): cint {.importc: "__builtin_ffsll", nodecl.}
2019-05-08 14:05:16 -06:00
# Returns the number of leading 0-bits in x, starting at the most significant bit position. If x is 0, the result is undefined.
2019-07-04 20:30:24 +02:00
func builtin_clz(x: cuint): cint {.importc: "__builtin_clz", nodecl.}
func builtin_clzll(x: culonglong): cint {.importc: "__builtin_clzll", nodecl.}
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func countOnesBuiltin(x: SomeUnsignedInt): int =
when bitsof(x) == bitsof(culonglong):
cast[int](builtin_popcountll(x.culonglong))
2019-07-04 20:30:24 +02:00
else:
cast[int](builtin_popcount(x.cuint))
2019-07-04 20:30:24 +02:00
func parityBuiltin(x: SomeUnsignedInt): int =
when bitsof(x) == bitsof(culonglong):
cast[int](builtin_parityll(x.culonglong))
2019-07-04 20:30:24 +02:00
else:
cast[int](builtin_parity(x.cuint))
2019-07-04 20:30:24 +02:00
func firstOneBuiltin(x: SomeUnsignedInt): int =
when bitsof(x) == bitsof(clonglong):
cast[int](builtin_ffsll(cast[clonglong](x)))
2019-07-04 20:30:24 +02:00
else:
cast[int](builtin_ffs(cast[cint](x.cuint)))
2019-05-08 14:05:16 -06:00
func log2truncBuiltin(v: uint8|uint16|uint32): int =
cast[int](31 - cast[cuint](builtin_clz(v.uint32)))
func log2truncBuiltin(v: uint64): int =
cast[int](63 - cast[cuint](builtin_clzll(v)))
2019-07-04 20:30:24 +02:00
elif defined(vcc) and useBuiltins:
const arch64 = sizeof(int) == 8
2019-05-08 14:05:16 -06:00
# Counts the number of one bits (population count) in a 16-, 32-, or 64-byte unsigned integer.
func builtin_popcnt16(a2: uint16): uint16 {.importc: "__popcnt16" header: "<intrin.h>".}
func builtin_popcnt32(a2: uint32): uint32 {.importc: "__popcnt" header: "<intrin.h>".}
# Search the mask data from most significant bit (MSB) to least significant bit (LSB) for a set bit (1).
func bitScanReverse(index: ptr culong, mask: culong): cuchar {.importc: "_BitScanReverse", header: "<intrin.h>".}
# Search the mask data from least significant bit (LSB) to the most significant bit (MSB) for a set bit (1).
func bitScanForward(index: ptr culong, mask: culong): cuchar {.importc: "_BitScanForward", header: "<intrin.h>".}
when arch64:
2019-07-04 20:30:24 +02:00
func builtin_popcnt64(a2: uint64): uint64 {.importc: "__popcnt64" header: "<intrin.h>".}
func bitScanReverse64(index: ptr culong, mask: uint64): cuchar {.importc: "_BitScanReverse64", header: "<intrin.h>".}
func bitScanForward64(index: ptr culong, mask: uint64): cuchar {.importc: "_BitScanForward64", header: "<intrin.h>".}
func countOnesBuiltin(v: uint8|uint16): int =
cast[int](builtin_popcnt16(v.uint16))
func countOnesBuiltin(v: uint32): int =
cast[int](builtin_popcnt32(v))
2019-07-04 20:30:24 +02:00
func countOnesBuiltin(v: uint64): int =
when arch64:
cast[int](builtin_popcnt64(v))
2019-07-04 20:30:24 +02:00
else:
cast[int](
builtin_popcnt32((v and uint32.high).uint32) +
builtin_popcnt32((v shr 32'u64).uint32))
2019-07-04 20:30:24 +02:00
template checkedScan(fnc: untyped, x: typed, def: typed): int =
var index{.noinit.}: culong
if fnc(index.addr, v) == cuchar(0): def
else: cast[int](index)
2019-07-04 20:30:24 +02:00
func firstOneBuiltin(v: uint8|uint16|uint32): int =
1 + checkedScan(bitScanForward, v.culong, -1)
func firstOneBuiltin(v: uint64): int =
when arch64:
1 + checkedScan(bitScanForward64, v.culonglong, -1)
2019-07-04 20:30:24 +02:00
else:
firstOneNim(v)
template bitScan(fnc: untyped, x: typed): int =
var index{.noinit.}: culong
if fnc(index.addr, v) == cuchar(0): 0
else: cast[int](index)
2019-07-04 20:30:24 +02:00
func log2truncBuiltin(v: uint8|uint16|uint32): int =
bitScan(bitScanReverse, v.culong)
func log2truncBuiltin(v: uint64): int =
when arch64:
2019-07-04 20:30:24 +02:00
bitScan(bitScanReverse64, v.culong)
else:
log2truncNim(v)
elif defined(icc) and useBuiltins:
const arch64 = sizeof(int) == 8
2019-05-08 14:05:16 -06:00
# Intel compiler intrinsics: http://fulla.fnal.gov/intel/compiler_c/main_cls/intref_cls/common/intref_allia_misc.htm
# see also: https://software.intel.com/en-us/node/523362
# Count the number of bits set to 1 in an integer a, and return that count in dst.
func builtin_popcnt32(x: cint): cint {.importc: "_popcnt32" header: "<immintrin.h>".}
2019-05-08 14:05:16 -06:00
# Returns the number of trailing 0-bits in x, starting at the least significant bit position. If x is 0, the result is undefined.
func bitScanForward(p: ptr uint32, b: uint32): cuchar {.importc: "_BitScanForward", header: "<immintrin.h>".}
# Returns the number of leading 0-bits in x, starting at the most significant bit position. If x is 0, the result is undefined.
func bitScanReverse(p: ptr uint32, b: uint32): cuchar {.importc: "_BitScanReverse", header: "<immintrin.h>".}
when arch64:
2019-07-04 20:30:24 +02:00
func builtin_popcnt64(x: uint64): cint {.importc: "_popcnt64" header: "<immintrin.h>".}
func bitScanForward64(p: ptr uint32, b: uint64): cuchar {.importc: "_BitScanForward64", header: "<immintrin.h>".}
func bitScanReverse64(p: ptr uint32, b: uint64): cuchar {.importc: "_BitScanReverse64", header: "<immintrin.h>".}
template checkedScan(fnc: untyped, x: typed, def: typed): int =
var index{.noinit.}: culong
if fnc(index.addr, v) == cuchar(0): def
else: cast[int](index)
2019-07-04 20:30:24 +02:00
template bitScan(fnc: untyped, x: typed): int =
var index{.noinit.}: culong
if fnc(index.addr, v) == cuchar(0): 0
else: cast[int](index)
2019-07-04 20:30:24 +02:00
func countOnesBuiltin(v: uint8|uint16|uint32): int =
cast[int](builtin_popcnt32(cast[cint](v)))
2019-07-04 20:30:24 +02:00
func countOnesBuiltin(v: uint64): int =
when arch64:
cast[int](builtin_popcnt64(v))
2019-07-04 20:30:24 +02:00
else:
cast[int](
builtin_popcnt32(cast[cint](v and 0xFFFFFFFF'u64)) +
builtin_popcnt32(cast[cint](v shr 32'u64)))
2019-07-04 20:30:24 +02:00
func firstOneBuiltin(v: uint8|uint16|uint32): int =
1 + checkedScan(bitScanForward, v.culong, -1)
func firstOneBuiltin(v: uint64): int =
when arch64:
2019-07-04 20:30:24 +02:00
1 + checkedScan(bitScanForward64, v.culong, -1)
else:
firstOneNim(v)
func log2truncBuiltin(v: uint8|uint16|uint32): int =
bitScan(bitScanReverse, v.culong)
func log2truncBuiltin(v: uint64): int =
when arch64:
2019-07-04 20:30:24 +02:00
bitScan(bitScanReverse64, v.culong)
else:
log2truncNim(v)
2019-05-08 14:05:16 -06:00
func countOnes*(x: SomeUnsignedInt): int {.inline.} =
## Counts the set bits in integer. (also called `Hamming weight`:idx:.)
##
## Example:
2019-07-04 20:30:24 +02:00
## doAssert countOnes(0b01000100'u8) == 2
2019-05-08 14:05:16 -06:00
when nimvm:
2019-07-04 20:30:24 +02:00
countOnesNim(x)
2019-05-08 14:05:16 -06:00
else:
when declared(countOnesBuiltin):
2019-07-04 20:30:24 +02:00
countOnesBuiltin(x)
2019-05-08 14:05:16 -06:00
else:
2019-07-04 20:30:24 +02:00
countOnesNim(x)
func countZeros*(x: SomeUnsignedInt): int {.inline.} =
bitsof(x) - countOnes(x)
2019-05-08 14:05:16 -06:00
func parity*(x: SomeUnsignedInt): int {.inline.} =
## Calculate the bit parity in integer. If number of 1-bit
## is odd parity is 1, otherwise 0.
##
## Example:
## doAssert parity(0b00000001'u8) == 1
# Can be used a base if creating ASM version.
# https://stackoverflow.com/questions/21617970/how-to-check-if-value-has-even-parity-of-bits-or-odd
when nimvm:
2019-07-04 20:30:24 +02:00
parityNim(x)
2019-05-08 14:05:16 -06:00
else:
when declared(parityBuiltin):
2019-07-04 20:30:24 +02:00
parityBuiltin(x)
2019-05-08 14:05:16 -06:00
else:
2019-07-04 20:30:24 +02:00
parityNim(x)
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func firstOne*(x: SomeUnsignedInt): int {.inline.} =
2019-05-08 14:05:16 -06:00
## Returns the 1-based index of the least significant set bit of x.
2019-07-04 20:30:24 +02:00
## If `x` is zero result is 0
##
## firstOne(x) == trailingZeros(x) + 1
2019-05-08 14:05:16 -06:00
##
## Example:
## doAssert firstOneBit(0b00000010'u8) == 2
##
when nimvm:
2019-07-04 20:30:24 +02:00
firstOneNim(x)
2019-05-08 14:05:16 -06:00
else:
when declared(firstOneBuiltin):
2019-07-04 20:30:24 +02:00
firstOneBuiltin(x)
2019-05-08 14:05:16 -06:00
else:
2019-07-04 20:30:24 +02:00
firstOneNim(x)
func log2trunc*(x: SomeUnsignedInt): int {.inline.} =
## Return the truncated base 2 logarithm of `x` - this is the zero-based
## index of the last set bit.
##
## If `x` is zero result is -1
##
## log2trunc(x) == bitsof(x) - leadingZeros(x) - 1.
2019-05-08 14:05:16 -06:00
##
## Example:
2019-07-04 20:30:24 +02:00
## doAssert log2trunc(0b01001000'u8) == 6
if x == 0: -1
2019-05-08 14:05:16 -06:00
else:
when nimvm:
2019-07-04 20:30:24 +02:00
log2truncNim(x)
2019-05-08 14:05:16 -06:00
else:
when declared(log2truncBuiltin):
2019-07-04 20:30:24 +02:00
log2truncBuiltin(x)
2019-05-08 14:05:16 -06:00
else:
2019-07-04 20:30:24 +02:00
log2truncNim(x)
2019-05-08 14:05:16 -06:00
2019-07-31 02:00:39 +03:00
template bitWidth*(x: SomeUnsignedInt): int =
## Returns the number of bits needed to write down the
## number `x` in binary. If `x` is zero, the result is 0.
log2trunc(x) + 1
2019-07-04 20:30:24 +02:00
func leadingZeros*(x: SomeInteger): int {.inline.} =
2019-05-08 14:05:16 -06:00
## Returns the number of leading zero bits in integer.
2019-07-04 20:30:24 +02:00
## If `x` is zero, result is bitsof(x)
2019-05-08 14:05:16 -06:00
##
## Example:
2019-07-04 20:30:24 +02:00
## doAssert leadingZeros(0b00000000'u8) == 8
## doAssert leadingZeros(0b00100000'u8) == 2
2019-05-08 14:05:16 -06:00
##
2019-07-04 20:30:24 +02:00
# Performance note:
# On recent x86_64 cpu's, this translates to the LZCNT instruction
bitsof(x) - 1 - log2trunc(x)
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func trailingZeros*(x: SomeUnsignedInt): int {.inline.} =
2019-05-08 14:05:16 -06:00
## Returns the number of trailing zeros in integer.
2019-07-04 20:30:24 +02:00
## If `x` is zero, result is sizeof(x) * 8
2019-05-08 14:05:16 -06:00
##
## Example:
2019-07-04 20:30:24 +02:00
## doAssert trailingZeros(0b00000010'u8) == 1
2019-05-08 14:05:16 -06:00
##
2019-07-04 20:30:24 +02:00
# Performance note:
# On recent x86_64 cpu's, this translates to the TZCNT instruction
if x == 0:
bitsof(x)
2019-05-08 14:05:16 -06:00
else:
2019-07-04 20:30:24 +02:00
firstOne(x) - 1
2019-05-08 14:05:16 -06:00
2019-07-04 20:30:24 +02:00
func nextPow2*(x: SomeUnsignedInt): SomeUnsignedInt {.inline.} =
## Calculate the next power-of-2 of x - wraps to 0
##
## Examples:
## doAssert nextPow2(3) == 4
## doAssert nextPow2(4) == 4
nextPow2Nim(x)
func rotateLeft*(v: SomeUnsignedInt, amount: SomeInteger):
SomeUnsignedInt {.inline.} =
## Left-rotate bits in an unsigned value
2019-05-08 14:05:16 -06:00
# using this form instead of the one below should handle any value
# out of range as well as negative values.
# taken from: https://en.wikipedia.org/wiki/Circular_shift#Implementing_circular_shifts
2019-07-04 20:30:24 +02:00
const mask = 8 * sizeof(v) - 1
let amount = int(amount and mask)
(v shl amount) or (v shr ( (-amount) and mask))
func rotateRight*(v: SomeUnsignedInt, amount: SomeInteger):
SomeUnsignedInt {.inline.} =
## Right-rotate bits in an unsigned value.
const mask = bitsof(v) - 1
let amount = int(amount and mask)
(v shr amount) or (v shl ( (-amount) and mask))
template mostSignificantBit(T: type): auto =
const res = 1 shl (sizeof(T) * 8 - 1)
T(res)
template getBit*(x: BitIndexable, bit: Natural): bool =
## reads a bit from `x`, assuming 0 to be the position of the
## least significant bit
type T = type(x)
2019-07-31 11:40:25 +03:00
((x shr bit) and T(1)) != 0
template getBitLE*(x: BitIndexable, bit: Natural): bool =
getBit(x, bit)
template getBitBE*(x: BitIndexable, bit: Natural): bool =
## Reads a bit from `x`, assuming 0 to be the position of
## the most significant bit.
##
## This indexing may be natural when you are considering the
## string representation of a bit field. For example, 72 can
## be written in binary as 0b01001000. The first bit here is
## zero, while the second bit is one.
##
## Since the string representation will depend on the size of
## the operand, using `getBitBE` with the same numeric value
## and a bit position may produce different results depending
## on the machine type used to store the value. For this reason,
## this indexing scheme is considered more error-prone and
## `getBitLE` is considering the default indexing scheme.
(x and mostSignificantBit(x.type) shr bit) != 0
func changeBit*(x: var BitIndexable, bit: Natural, val: bool) {.inline.} =
## changes a bit in `x` to val, assuming 0 to be the position of the
## least significant bit
type T = type(x)
x = (x and not (T(1) shl bit)) or (T(val) shl bit)
template changeBitLE*(x: var BitIndexable, bit: Natural, val: bool) =
setBit(x, bit, val)
func changeBitBE*(x: var BitIndexable, bit: Natural, val: bool) {.inline.} =
## changes a bit in `x` to val, assuming 0 to be the position of the
## most significant bit
changeBit(x, bitsof(x) - 1 - bit, val)
func setBit*(x: var BitIndexable, bit: Natural) {.inline.} =
## sets bit in `x`, assuming 0 to be the position of the
## least significant bit
type T = type(x)
let mask = T(1) shl bit
x = x or mask
template setBitLE*(x: var BitIndexable, bit: Natural) =
setBit(x, bit)
func setBitBE*(x: var BitIndexable, bit: Natural) {.inline.} =
## sets a bit in `x`, assuming 0 to be the position of the
## most significant bit
let mask = mostSignificantBit(x.type) shr bit
x = x or mask
func clearBit*(x: var BitIndexable, bit: Natural) {.inline.} =
## clears bit in a byte, assuming 0 to be the position of the
## least significant bit
type T = type(x)
let mask = T(1) shl bit
x = x and not mask
template clearBitLE*(x: var BitIndexable, bit: Natural) =
clearBit(x, bit)
func clearBitBE*(x: var BitIndexable, bit: Natural) {.inline.} =
## clears a bit in `x`, assuming 0 to be the position of the
## most significant bit
let mask = mostSignificantBit(x.type) shr bit
x = x and not mask
func toggleBit*(x: var BitIndexable, bit: Natural) {.inline.} =
## toggles (inverts) bit in `x`, assuming 0 to be the position of the
## least significant bit
type T = type(x)
let mask = T(1) shl bit
x = x xor mask
template toggleBitLE*(x: var BitIndexable, bit: Natural) =
toggleBit(x, bit)
func toggleBitBE*(x: var BitIndexable, bit: Natural) {.inline.} =
## toggles (inverts) a bit in `x`, assuming 0 to be the position of the
## most significant bit
let mask = mostSignificantBit(x.type) shr bit
x = x xor mask
template byteIndex(pos: Natural): int =
pos shr 3 # same as pos div 8
template bitIndex(pos: Natural): int =
pos and 0b111 # same as pos mod 8
2021-12-02 16:24:02 +01:00
func getBit*(bytes: openArray[byte], pos: Natural): bool {.inline.} =
getBit(bytes[byteIndex pos], bitIndex pos)
2021-12-02 16:24:02 +01:00
template getBitLE*(bytes: openArray[byte], pos: Natural): bool =
getBit(bytes, pos)
2021-12-02 16:24:02 +01:00
func getBitBE*(bytes: openArray[byte], pos: Natural): bool {.inline.} =
getBitBE(bytes[byteIndex pos], bitIndex pos)
2021-12-02 16:24:02 +01:00
func changeBit*(bytes: var openArray[byte], pos: Natural, value: bool) {.inline.} =
changeBit(bytes[byteIndex pos], bitIndex pos, value)
2021-12-02 16:24:02 +01:00
template changeBitLE*(bytes: var openArray[byte], pos: Natural, value: bool) =
changeBit(bytes, pos, value)
2021-12-02 16:24:02 +01:00
func changeBitBE*(bytes: var openArray[byte], pos: Natural, value: bool) {.inline.} =
changeBitBE(bytes[byteIndex pos], bitIndex pos, value)
2021-12-02 16:24:02 +01:00
func setBit*(bytes: var openArray[byte], pos: Natural) {.inline.} =
setBit(bytes[byteIndex pos], bitIndex pos)
2021-12-02 16:24:02 +01:00
template setBitLE*(bytes: var openArray[byte], pos: Natural) =
setBit(bytes, pos)
2021-12-02 16:24:02 +01:00
func setBitBE*(bytes: var openArray[byte], pos: Natural) {.inline.} =
setBitBE(bytes[byteIndex pos], bitIndex pos)
2021-12-02 16:24:02 +01:00
func clearBit*(bytes: var openArray[byte], pos: Natural) {.inline.} =
clearBit(bytes[byteIndex pos], bitIndex pos)
2021-12-02 16:24:02 +01:00
template clearBitLE*(bytes: var openArray[byte], pos: Natural) =
clearBit(bytes, pos)
2021-12-02 16:24:02 +01:00
func clearBitBE*(bytes: var openArray[byte], pos: Natural) {.inline.} =
clearBitBE(bytes[byteIndex pos], bitIndex pos)
2021-12-02 16:24:02 +01:00
func toggleBit*(bytes: var openArray[byte], pos: Natural) {.inline.} =
toggleBit(bytes[byteIndex pos], bitIndex pos)
2021-12-02 16:24:02 +01:00
template toggleBitLE*(bytes: var openArray[byte], pos: Natural) =
toggleBit(bytes, pos)
2021-12-02 16:24:02 +01:00
func toggleBitBE*(bytes: var openArray[byte], pos: Natural) {.inline.} =
toggleBitBE(bytes[byteIndex pos], bitIndex pos)
template setBit*(x: var BitIndexable, bit: Natural, val: bool) {.deprecated: "changeBit".} =
changeBit(x, bit, val)
template setBitLE*(x: var BitIndexable, bit: Natural, val: bool) {.deprecated: "changeBitLE".} =
changeBitLE(x, bit, val)
template setBitBE*(x: var BitIndexable, bit: Natural, val: bool) {.deprecated: "changeBitBE".} =
changeBitBE(x, bit, val)
template raiseBit*(x: var BitIndexable, bit: Natural) {.deprecated: "setBit".} =
setBit(x, bit)
template raiseBitLE*(x: var BitIndexable, bit: Natural) {.deprecated: "setBitLE".} =
setBitLE(x, bit)
template raiseBitBE*(x: var BitIndexable, bit: Natural) {.deprecated: "setBitBE".} =
setBitBE(x, bit)
func lowerBit*(x: var BitIndexable, bit: Natural) {.inline, deprecated: "clearBit".} =
clearBit(x, bit)
template lowerBitLE*(x: var BitIndexable, bit: Natural) {.deprecated: "clearBitLE".} =
clearBit(x, bit)
template lowerBitBE*(x: var BitIndexable, bit: Natural) {.deprecated: "clearBitBE".} =
clearBitBE(x, bit)
2021-12-02 16:24:02 +01:00
template setBit*(bytes: var openArray[byte], pos: Natural, val: bool) {.deprecated: "changeBit".} =
changeBit(bytes, pos, val)
2021-12-02 16:24:02 +01:00
template setBitLE*(bytes: var openArray[byte], pos: Natural, val: bool) {.deprecated: "changeBitLE".} =
changeBitLE(bytes, pos, val)
2021-12-02 16:24:02 +01:00
template setBitBE*(bytes: var openArray[byte], pos: Natural, val: bool) {.deprecated: "changeBitBE".} =
changeBitBE(bytes, pos, val)
2021-12-02 16:24:02 +01:00
template raiseBit*(bytes: var openArray[byte], pos: Natural) {.deprecated: "setBit".} =
setBit(bytes, pos)
2021-12-02 16:24:02 +01:00
template raiseBitLE*(bytes: var openArray[byte], pos: Natural) {.deprecated: "setBitLE".} =
setBitLE(bytes, pos)
2021-12-02 16:24:02 +01:00
template raiseBitBE*(bytes: var openArray[byte], pos: Natural) {.deprecated: "setBitBE".} =
setBitBE(bytes, pos)
2021-12-02 16:24:02 +01:00
template lowerBit*(bytes: var openArray[byte], pos: Natural) {.deprecated: "clearBit".} =
clearBit(bytes, pos)
2021-12-02 16:24:02 +01:00
template lowerBitLE*(bytes: var openArray[byte], pos: Natural) {.deprecated: "clearBitLE".} =
clearBitLE(bytes, pos)
2021-12-02 16:24:02 +01:00
template lowerBitBE*(bytes: var openArray[byte], pos: Natural) {.deprecated: "clearBitBE".} =
clearBitBE(bytes, pos)
2020-06-01 19:49:46 +03:00
2021-12-02 16:24:02 +01:00
func getBitsBE*(data: openArray[byte], slice: HSlice, T: type[SomeUnsignedInt]): T =
2020-06-01 19:49:46 +03:00
## Treats `data` as an unsigned big endian integer and returns a slice of bits
## extracted from it, assuming 0 to be the possition of the most significant bit.
let totalBits = data.len * 8
template normalizeIdx(idx): int =
when idx is BackwardsIndex: totalBits - int(idx)
else: int(idx)
let
a = normalizeIdx(slice.a)
b = normalizeIdx(slice.b) + 1
sliceLen = b - a
const resultBits = sizeof(result) * 8
doAssert a < b and sliceLen <= resultBits and b <= totalBits
let limbs = cast[ptr UncheckedArray[T]](unsafeAddr data[0])
template readLimb(idx: int): auto =
when cpuEndian == bigEndian or sizeof(result) == 1:
limbs[][idx]
else:
swapBytes(limbs[][idx])
let
firstLimbIdx = a div resultBits
firstLimbUnusedBits = (a mod resultBits)
firstLimbUsedBits = resultBits - firstLimbUnusedBits
firstLimb = readLimb firstLimbIdx
if sliceLen > firstLimbUsedBits:
let
bitsFromSecondLimb = sliceLen - firstLimbUsedBits
secondLimb = readLimb(firstLimbIdx + 1)
((firstLimb shl firstLimbUnusedBits) shr (firstLimbUnusedBits - bitsFromSecondLimb)) or
(secondLimb shr (resultBits - bitsFromSecondLimb))
else:
(firstLimb shl firstLimbUnusedBits) shr (resultBits - sliceLen)
2021-12-02 16:24:02 +01:00
template getBitsBE*(data: openArray[byte], slice: HSlice): BiggestUInt =
2020-06-01 19:49:46 +03:00
getBitsBE(data, slice, BiggestUInt)