use ::core::mem::MaybeUninit;
pub unsafe trait DataBuf {
type Inner: Pod;
fn as_ref(&self) -> &[MaybeUninit<Self::Inner>];
fn as_mut(&mut self) -> &mut [MaybeUninit<Self::Inner>];
fn extend(&mut self, len: usize) -> Result<(), ()>;
fn round_to_words(bytes: usize) -> usize {
crate::round_to_words::<Self::Inner>(bytes)
}
}
pub unsafe trait Pod: Copy {
fn default() -> Self;
}
macro_rules! impl_pod {
( $($t:ty),* ) => {
$( unsafe impl Pod for $t { fn default() -> Self { 0 } } )*
}
}
impl_pod! { u8, u16, u32, u64, u128, usize }
unsafe impl<T> Pod for *const T {
fn default() -> Self { ::core::ptr::null() }
}
unsafe impl<T, U> DataBuf for &mut T
where
U: Pod,
T: DataBuf<Inner=U>,
{
type Inner = T::Inner;
fn as_ref(&self) -> &[MaybeUninit<Self::Inner>] {
(**self).as_ref()
}
fn as_mut(&mut self) -> &mut [MaybeUninit<Self::Inner>] {
(**self).as_mut()
}
fn extend(&mut self, len: usize) -> Result<(), ()> {
(**self).extend(len)
}
}
#[cfg(not(feature = "const_generics"))]
macro_rules! impl_databuf_array {
( $($n:expr),* ) => {
$(unsafe impl<T: Pod> DataBuf for [MaybeUninit<T>; $n] {
type Inner = T;
fn as_ref(&self) -> &[MaybeUninit<Self::Inner>] {
self
}
fn as_mut(&mut self) -> &mut [MaybeUninit<Self::Inner>] {
self
}
fn extend(&mut self, len: usize) -> Result<(), ()> {
if len > $n {
Err( () )
}
else {
Ok( () )
}
}
})*
}
}
#[cfg(not(feature = "const_generics"))]
impl_databuf_array! {
0, 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,48,
64,96,
128,192,
256
}
#[cfg(feature = "const_generics")]
unsafe impl<T: Pod, const N: usize> DataBuf for [MaybeUninit<T>; N] {
type Inner = T;
fn as_ref(&self) -> &[MaybeUninit<Self::Inner>] {
self
}
fn as_mut(&mut self) -> &mut [MaybeUninit<Self::Inner>] {
self
}
fn extend(&mut self, len: usize) -> Result<(), ()> {
if len > N {
Err(())
}
else {
Ok(())
}
}
}
#[cfg(all(feature = "alloc"))]
unsafe impl<T: Pod> crate::DataBuf for ::alloc::vec::Vec< MaybeUninit<T> > {
type Inner = T;
fn as_ref(&self) -> &[MaybeUninit<Self::Inner>] {
self
}
fn as_mut(&mut self) -> &mut [MaybeUninit<Self::Inner>] {
self
}
fn extend(&mut self, len: usize) -> Result<(), ()> {
if len > self.len() {
self.resize(len, MaybeUninit::uninit());
let cap = self.capacity();
self.resize(cap, MaybeUninit::uninit());
}
Ok(())
}
}