bitwrap

docs

Intro

BitWrap is a derive macro and trait to declare a struct data member with explicit size, in bits.

BitWrap Trait

BitWrap trait declares 2 methods:

rust,ignore fn pack(&self, dst: &mut [u8]) -> Result<usize, BitWrapError>

pack method puts value into dst array

rust,ignore fn unpack(&mut self, src: &[u8]) -> Result<usize, BitWrapError>

unpack method extracts value from src array

BitWrap Macro

BitWrap Macro implements BitWrap trait for object. pack and unpack code generates automatically according to the fields arguments.

Option format - bits(N) where:

For example packet has next format:

| Field | Bits | |---|---| | flag1 | 1 | | flag2 | 1 | | data3 | 2 | | data4 | 12 |

```rust

![no_std]

use bitwrap::BitWrap;

[derive(Default, BitWrap)]

struct Packet { #[bits(1)] flag_1: u8,

#[bits(1)]
flag_2: u8,

#[bits(2)]
data_3: u8,

#[bits(12)]
data_4: u16,

}

const DATA: &[u8] = &[0xA2, 0x34];

let mut packet = Packet::default(); packet.unpack(DATA).unwrap();

asserteq!(packet.flag1, 1); asserteq!(packet.flag2, 0); asserteq!(packet.data3, 2); asserteq!(packet.data4, 0x0234);

let mut buffer: [u8; 2] = [0; 2]; let result = packet.pack(&mut buffer).unwrap();

asserteq!(result, DATA.len()); asserteq!(buffer, DATA); ```

Skip bits

Some packets contains reserved or fixed bits. This field could be skiped with skip option.

Option format - bits(N, skip = V) where:

For example packet has next format:

| Field | Bits | |---|---| | Data | 6 | | Reserved 0b00 | 2 | | Reserved 0b1111 | 4 | | Data | 4 |

```rust use bitwrap::BitWrap;

[derive(Default, BitWrap)]

struct Packet { #[bits(6)] f1: u8,

#[bits(2, skip = 0)]
#[bits(4, skip = 0b1111)]
#[bits(4)]
f2: u8,

}

const DATA: &[u8] = &[0xAC, 0xF5];

let mut packet = Packet::default(); packet.unpack(DATA).unwrap();

asserteq!(packet.f1, 0x2B); asserteq!(packet.f2, 0x05);

let mut buffer: [u8; 2] = [0; 2]; let result = packet.pack(&mut buffer).unwrap();

assert_eq!(&buffer[.. result], DATA); ```

Nested objects

Nested field is an object with implemented BitWrap trait.

For example part of IPv4 packet:

| Field | Bits | |---|---| | ttl | 8 | | protocol | 8 | | checksum | 16 | | src | 32 | | dst | 32 |

```rust use std::net::Ipv4Addr; use bitwrap::BitWrap;

[derive(BitWrap)]

struct IP4 { #[bits(8)] ttl: u8,

#[bits(8)]
protocol: u8,

#[bits(16)]
checksum: u16,

#[bits]
src: Ipv4Addr,

#[bits]
dst: Ipv4Addr,

}

const DATA: &[u8] = &[ 0x40, 0x88, 0x37, 0x5D, 0xC0, 0xA8, 0xC8, 0xB0, 0xC0, 0xA8, 0xC8, 0xB7, ];

let mut packet = IP4 { ttl: 0, protocol: 0, checksum: 0, src: Ipv4Addr::new(0, 0, 0, 0), dst: Ipv4Addr::new(0, 0, 0, 0), };

packet.unpack(DATA).unwrap();

asserteq!(packet.src, Ipv4Addr::new(192, 168, 200, 176)); asserteq!(packet.dst, Ipv4Addr::new(192, 168, 200, 183));

let mut buffer: [u8; 16] = [0; 16]; let result = packet.pack(&mut buffer).unwrap();

assert_eq!(&buffer[.. result], DATA); ```

Types converter

This feature converts numerical type into the field type. Here is numerical type means an unsigned number that enough to contain all data. Field type means a type of the struct field.

For example bits(1) - the numerical type will be u8. If the field type is bool then conversion code will be appended automatically.

For other types has next options - bits(N, from = F, into = I) where:

| Field | Bits | |---|---| | Reserved 0b0000 | 4 | | Coffee | 4 |

```rust use bitwrap::BitWrap;

[derive(Debug, PartialEq, Clone, Copy)]

enum Coffee { Water, Latte, Cappuccino, Espresso, Americano, }

impl Default for Coffee { fn default() -> Self { Coffee::Water } }

impl From for Coffee { fn from(value: u8) -> Self { match value { 0 => Coffee::Water, 1 => Coffee::Latte, 2 => Coffee::Cappuccino, 3 => Coffee::Espresso, 4 => Coffee::Americano, _ => unreachable!(), } } }

impl Into for Coffee { fn into(self) -> u8 { match self { Coffee::Water => 0, Coffee::Latte => 1, Coffee::Cappuccino => 2, Coffee::Espresso => 3, Coffee::Americano => 4, } } }

[derive(Default, BitWrap)]

struct Packet { #[bits(4, skip = 0)] #[bits(4, from = Coffee::from, into = Coffee::into)] coffee: Coffee, }

const DATA: &[u8] = &[0x01];

let mut packet = Packet::default(); packet.unpack(DATA).unwrap();

assert_eq!(packet.coffee, Coffee::Latte);

let mut buffer: [u8; 1] = [0; 1]; let result = packet.pack(&mut buffer).unwrap();

assert_eq!(&buffer[.. result], DATA); ```