use crate::bitwidth::BitWidth::{W16, W32, W64, W8};
use bytes::{BufMut, BytesMut};
#[derive(
Debug,
Clone,
Copy,
PartialEq,
Eq,
PartialOrd,
Serialize,
Deserialize,
Ord,
num_enum::TryFromPrimitive,
)]
#[repr(u8)]
#[derive(Default)]
pub enum BitWidth {
#[default]
W8 = 0,
W16 = 1,
W32 = 2,
W64 = 3,
}
pub const BIT_WIDTHS: [BitWidth; 4] = [W8, W16, W32, W64];
impl BitWidth {
#[must_use]
pub fn n_bytes(self) -> usize {
1 << self as usize
}
pub fn from_nbytes(n: impl Into<usize>) -> Option<Self> {
match n.into() {
1 => Some(W8),
2 => Some(W16),
4 => Some(W32),
8 => Some(W64),
_ => None,
}
}
}
macro_rules! impl_bitwidth_from {
($from: ident, $w64: ident, $w32: ident, $w16: ident, $w8: ident) => {
#[allow(clippy::cast_lossless, clippy::checked_conversions)]
impl From<$from> for BitWidth {
fn from(x: $from) -> BitWidth {
let x = x as $w64;
if x >= $w8::MIN as $w64 && x <= $w8::MAX as $w64 {
return W8;
}
if x >= $w16::MIN as $w64 && x <= $w16::MAX as $w64 {
return W16;
}
if x >= $w32::MIN as $w64 && x <= $w32::MAX as $w64 {
return W32;
}
W64
}
}
};
}
impl_bitwidth_from!(u64, u64, u32, u16, u8);
impl_bitwidth_from!(usize, u64, u32, u16, u8);
impl_bitwidth_from!(i64, i64, i32, i16, i8);
#[allow(clippy::float_cmp, clippy::cast_possible_truncation)]
impl From<f64> for BitWidth {
fn from(x: f64) -> BitWidth {
if x == f64::from(x as f32) { W32 } else { W64 }
}
}
impl From<f32> for BitWidth {
fn from(_: f32) -> BitWidth {
W32
}
}
pub fn align(buffer: &mut BytesMut, width: BitWidth) {
let bytes = 1 << width as u8;
let mut alignment = (bytes - buffer.len() % bytes) % bytes;
loop {
if alignment == 0 {
break;
}
buffer.put_u8(0);
alignment -= 1;
}
}