/bitseqs

bitseqs - a bit of bit manipulation

Primary LanguageNimMIT LicenseMIT

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 and BitSeqs (basically masks)
  • Use std/sequtils on BitSeqs (concat, map, filter, ...)
  • Flip and set bits in BitSeqs
  • Get data back from a BitSeq and convert it back to uint8 - uint64 (which you can cast later)
  • Rotations & shifts (std/sequtils or std/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]