bitwrap is a derive macro and trait to declare a struct data member with explicit size, in bits.
bits
attribute accept only one argument - field size in bits.
For example packet has next format:
| Field | Bits | |---|---| | flag1 | 1 | | flag2 | 1 | | data3 | 2 | | data4 | 12 |
```rust
use bitwrap::BitWrap;
struct Packet { #[bits(1)] flag1: u8, #[bits(1)] flag2: u8, #[bits(2)] data3: u8, #[bits(12)] data4: 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); ```
Some packets contains reserved or fixed bits.
This bits could be skiped with bits_skip
attribute.
For example packet has next format:
| Field | Bits | |---|---| | Data | 6 | | 0b00 | 2 | | 0b1111 | 4 | | Data | 4 |
bits_skip attribute accept next arguments:
```rust use bitwrap::BitWrap;
struct Packet { #[bits(6)] f1: u8, #[bitsskip(2)] #[bitsskip(4, 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: Vec
assert_eq!(&buffer[.. result], DATA); ```
Nested field is an object with BitWrap interface.
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;
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.ttl, 64); asserteq!(packet.protocol, 136); asserteq!(packet.checksum, 0x375D); asserteq!(packet.src, Ipv4Addr::new(192, 168, 200, 176)); assert_eq!(packet.dst, Ipv4Addr::new(192, 168, 200, 183));
let mut buffer: Vec
assert_eq!(&buffer[.. result], DATA); ```
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)
- numerical type will be u8
.
If the field type is bool
then conversion code will be appended automatically.
For other types or for value conversion you may use - bits_convert(1, from, into)
:
from
- method to convert field type from numeric typeinto
- method to convert numeric type into field typeIf conversion methods not defined - bits_convert(1)
will be used
From and Into traits implemented for the field type.
| Field | Bits | |---|---| | Reserved | 4 | | Coffee | 4 |
```rust use bitwrap::BitWrap;
enum Coffee { Water, Latte, Cappuccino, Espresso, Americano, }
impl Default for Coffee { fn default() -> Self { Coffee::Water } }
impl From
impl Into
struct Packet { #[bitsskip(4)] #[bitsconvert(4)] 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); ```