1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100
#![cfg_attr(feature = "nightly", feature(array_try_map))]
#![doc = include_str!("../README.md")]
mod cursor;
pub mod lencoder;
mod record;
mod types;
use core::convert::TryInto;
use core::mem::{size_of, MaybeUninit};
use core::ptr;
use ErrorKind::*;
pub use cursor::*;
pub use derive::*;
pub use record::*;
/// Shortcut for `Result<T, bin_layout::ErrorKind>`
pub type Result<T> = core::result::Result<T, ErrorKind>;
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum ErrorKind {
InsufficientBytes,
InvalidLength,
InvalidInput,
LenOverflow,
Unsupported,
InvalidType,
InvalidData,
InvalidChar,
InvalidUtf8,
Other,
}
/// A trait for serialize and deserialize data for binary format.
///
/// All [primitive types](https://doc.rust-lang.org/stable/rust-by-example/primitives.html) implement this trait.
///
/// `Vec`, `String`, `&[T]`, `&str` etc.. are encoded with their length value first, Following by each entry.
pub trait DataType<'de>: Sized {
/// Serialize the data to binary format.
fn serialize(self, _: &mut Cursor<impl AsMut<[u8]>>) -> Result<()>;
/// Deserialize the data from binary format.
fn deserialize(_: &mut Cursor<&'de [u8]>) -> Result<Self>;
/// Shortcut for `DataType::serialize(self, &mut View::new(bytes.as_mut()))`
///
/// ### Example
///
/// ```
/// use bin_layout::DataType;
///
/// #[derive(DataType)]
/// struct FooBar {
/// foo: u8,
/// bar: [u8; 2],
/// }
///
/// let mut bytes = [0; 3];
/// FooBar { foo: 1, bar: [2, 3] }.encode(&mut bytes);
/// assert_eq!(bytes, [1, 2, 3]);
/// ```
#[inline]
fn encode(self, data: &mut [u8]) -> Result<()> {
self.serialize(&mut Cursor::from(data))
}
/// Shortcut for `DataType::deserialize(&mut View::new(bytes.as_ref()))`
///
/// ### Example
///
/// ```
/// use bin_layout::DataType;
///
/// #[derive(DataType, PartialEq, Debug)]
/// struct FooBar {
/// foo: u8,
/// bar: [u8; 2],
/// }
///
/// let foobar = FooBar::decode(&[1, 2, 3]).unwrap();
/// assert_eq!(foobar, FooBar { foo: 1, bar: [2, 3] });
/// ```
#[inline]
fn decode(data: &'de [u8]) -> Result<Self> {
Self::deserialize(&mut Cursor::from(data))
}
}
// -----------------------------------------------------------------------
macro_rules! ret_err_or_add {
[($offset:expr; + $count:expr) > $len:expr] => {
let total_len = $offset + $count;
if total_len > $len { return Err(InsufficientBytes); }
$offset = total_len;
};
}
pub(crate) use ret_err_or_add;