bitseqs - a bit of bit manipulation
nimble install bitseqs
This library has no dependencies other than the Nim standard library.
About
type
Bit* = range[0'u8..1'u8]
BitSeq* = seq[Bit]
This package creates and performs operations on BitSeq
type. Since BitSeq
is just a seq[Bit]
it is possible to use it with Nim's std/sequtils
. Consider also using std/bitops
.
Features
- Get a single bit from a number or iterate over all bits
- Construct a
BitSeq
and manipulate it - Perform logical/bitwise operations on
Bits
andBitSeqs
(basically masks) - Use
std/sequtils
onBitSeqs
(concat
,map
,filter
, ...) - Flip and set bits in
BitSeqs
- Get data back from a
BitSeq
and convert it back touint8
-uint64
(which you cancast
later) - Rotations & shifts (
std/sequtils
orstd/deques
can be used instead?)
Example
import std/[sequtils, sugar]
import bitseqs
# MSB -> 0101 1100 <- LSB
let number: uint8 = 0b0101_1100
# Get a `Bit` (range[0'u8..1'u8]) from a number
# Indexed from LSB (right side)
# 7654 3210
# MSB -> 0101 1100 <- LSB
assert number.bit(0) == 0.Bit
assert number.bit(3) == 1.Bit
assert number.bit(7) == 0.Bit
assert number.bits().toSeq() == @[0.Bit, 1, 0, 1, 1, 1, 0, 0] # `toSeq` is from std/sequtils
# Perform bitwise operations on bits
# 0 or 1 == 1
assert (number.bit(1) or number.bit(2)) == 1.Bit
# Create `BitSeq` (seq[Bit])
# Indexed from MSB (left side)
# 0123 4567
# MSB -> 0101 1100 <- LSB
let bf = number.toBitSeq()
assert bf == @[0.Bit, 1, 0, 1, 1, 1, 0, 0]
# Get data back from `BitSeq` as uint8 - uint64
assert fromBitSeq[uint8](bf) == 0b0101_1100
# Use any seq operations on `BitSeq`
assert bf.filterIt(it == 1.Bit).len == 4
# Get a seq of indexes of bits set to 1/0
# Indexed from MSB (left side)
# 1 3 45
# MSB -> 0101 1100 <- LSB
assert number.bitsIdxOnSeq() == @[1, 3, 4, 5]
assert bf.bitsIdxOn.toSeq() == @[1, 3, 4, 5]
# 0 2 67
# MSB -> 0101 1100 <- LSB
assert bf.bitsIdxOffSeq() == @[0, 2, 6, 7]
# Create a new empty `BitSeq`
var bf2 = newBitSeq(uint8) # == newBitSeq(8)
for i in 0..<4:
bf2[i] = 1.Bit
assert bf2 == @[1.Bit, 1, 1, 1, 0, 0, 0, 0]
# Perform bitwise operations on `BitSeqs`
assert (bf and bf2) == @[0.Bit, 1, 0, 1, 0, 0, 0, 0]
assert (bf xor bf2) == @[1.Bit, 0, 1, 0, 1, 1, 0, 0]
var bf3 = not bf2
assert bf3 == @[0.Bit, 0, 0, 0, 1, 1, 1, 1]
# Flip `Bits` in `BitSeqs`
bf2.bitFlip(0)
assert bf2 == @[0.Bit, 1, 1, 1, 0, 0, 0, 0]
bf2.bitFlip(0..3)
assert bf2 == @[1.Bit, 0, 0, 0, 0, 0, 0, 0]
bf2.bitFlip(^2)
assert bf2 == @[1.Bit, 0, 0, 0, 0, 0, 1, 0]
bf3.bitFlip(3..^1)
assert bf3 == @[0.Bit, 0, 0, 1, 0, 0, 0, 0]
# With `dup()` from std/sugar it doesn't mutate original
assert bf3.dup(bitFlip(0)) == @[1.Bit, 0, 0, 1, 0, 0, 0, 0]
# Set `Bits` in `BitSeqs`
bf3.bitSet(0, 1)
assert bf3 == @[1.Bit, 0, 0, 1, 0, 0, 0, 0]
bf3.bitSet(1..3, 1)
assert bf3 == @[1.Bit, 1, 1, 1, 0, 0, 0, 0]
bf3.bitSet(^3..^1, 1)
assert bf3 == @[1.Bit, 1, 1, 1, 0, 1, 1, 1]
# Resize `BitSeqs`
# If `size` is smaller than `bf.len`, the most significant bits are dropped
# If `size` is larger than `bf.len`, the most significant bits are padded with 0
bf3.resize(4)
assert bf3 == @[0.Bit, 1, 1, 1]
bf3.resize(uint8)
assert bf3 == @[0.Bit, 0, 0, 0, 0, 1, 1, 1]