This crate provides a library for conversion between bytes/bits and heterogeneous lists (tuples).
Library features: - implements compile time type checking - neither declarative nor procedural macros exports - mixed endianness from single bytes array
Parse complex data structure ```rust use heterob::{P4, endianness::LeBytesInto, bit_numbering::LsbInto};
// Source is a [u8;6] bytes array let data = [0x00u8,0x11,0x22,0x33,0x44,0x55,0b1010_1001];
// Target struct
struct S {
byte: u8,
word: Option
// Parse bytes array as integers
let P4((byte, word, bytes, byte6)) = data.lebytesinto();
// Parse last byte as bitfield. Unit type () used as placeholder
let (isbyte6bit0, (), issomeword, byte6last4bits) =
P4::is_some_word
coerce to bool via let statement
let : bool = issome_word;
// Final structure let result = S { byte, word: issomeword.then(|| word), bytes, isbyte6bit0, byte6last4_bits, };
let sample = S { byte: 0x00, word: Some(0x2211), bytes: [0x33,0x44,0x55], isbyte6bit0: true, byte6last4_bits: 0b1010, };
assert_eq!(sample, result); ```
Mixed endians ```rust use heterob::{T3, endianness::{Be, Le}};
struct S { le: u16, be: u16, bytes: [u8;2] }
let data = [0x00,0x11,0x22,0x33,0x44,0x55];
let (Le(le),Be(be),bytes) = T3::from(data).into(); let s = S { le, be, bytes };
assert_eq!(S { le: 0x1100, be: 0x2233, bytes: [0x44,0x55] }, s, "{:x}", s.be); ```
The idea of compile time checks taken from issue comment of static_assertions crate. Unfortunately it's hard to find the error source using this type of compile time checks. With stabilized constant generics arithmetics it should be much easy.
There are two things checking at compile time
This check asserts that length of input array equal to sum of lengths of output arrays ```compile_fail
let data = [0u8; 13];
let _ = T2::<[u8;4], [u8;3]>::from(data);
Trying to split 13 bytes length array to 2 arrays with lengths 4 + 3 = 7 will throw an error:
text
error[E0080]: evaluation of <heterob::T2<[u8; 4], [u8; 3]> as heterob::ParamAndAssociatedConst<13_usize>>::LESS
failed
--> src/lib.rs:21:25
|
21 | const LESS: usize = Self::VALUE - N;
| ^^^^^^^^^^^^^^^ attempt to compute 7_usize - 13_usize
, which would overflow
note: the above error was encountered while instantiating fn <heterob::T2<[u8; 4], [u8; 3]> as std::convert::From<[u8; 13]>>::from
--> src/lib.rs:86:9
|
6 | let _ = T2::<[u8;4], [u8;3]>::from(data);
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
```
This check asserts that sum of bit indexes is less than value bits count ```compile_fail
let data = 0u16;
let ((),(),()) = P3::<_, 2, 11, 5>(data).lsbinto();
Trying to extract bits 12-17 from 16 bits value will throw an error:
text
error[E0080]: evaluation of <((), (), ()) as heterob::bit_numbering::FromLsb<heterob::P3<u16, 2_usize, 11_usize, 5_usize>>>::ASSERT_INDEX_IN_BOUNDS
failed
--> src/bitnumbering.rs:81:43
|
81 | const ASSERTINDEXINBOUNDS: usize = Self::BITS - Self::MAXBIT_INDEX;
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ attempt to compute 16_usize - 18_usize
, which would overflow
note: the above error was encountered while instantiating fn <((), (), ()) as heterob::bit_numbering::FromLsb<heterob::P3<u16, 2_usize, 11_usize, 5_usize>>>::from_lsb
--> src/bitnumbering.rs:97:9
|
97 | U::fromlsb(self)
| ^^^^^^^^^^^^^^^^^
```