bin-proto 0.12.6

Conversion to/from binary for arbitrary types
Documentation
use bitstream_io::{BitRead, BitWrite, Endianness};

use crate::{util, BitDecode, BitEncode, Result};
use core::mem::MaybeUninit;

struct PartialGuard<T> {
    ptr: *mut T,
    len: usize,
}

impl<T> Drop for PartialGuard<T> {
    fn drop(&mut self) {
        unsafe {
            core::ptr::drop_in_place(core::ptr::slice_from_raw_parts_mut(self.ptr, self.len));
        }
    }
}

impl<Ctx, T, const N: usize> BitDecode<Ctx> for [T; N]
where
    T: BitDecode<Ctx>,
{
    fn decode<R, E>(read: &mut R, ctx: &mut Ctx, (): ()) -> Result<Self>
    where
        R: BitRead,
        E: Endianness,
    {
        let mut array: MaybeUninit<[T; N]> = MaybeUninit::uninit();
        let mut guard = PartialGuard {
            ptr: array.as_mut_ptr().cast::<T>(),
            len: 0,
        };
        while guard.len < N {
            let item = T::decode::<_, E>(read, ctx, ())?;
            unsafe {
                guard.ptr.add(guard.len).write(item);
            }
            guard.len += 1;
        }
        core::mem::forget(guard);
        Ok(unsafe { array.assume_init() })
    }
}

impl<Ctx, T, const N: usize> BitEncode<Ctx> for [T; N]
where
    T: BitEncode<Ctx> + Sized,
{
    fn encode<W, E>(&self, write: &mut W, ctx: &mut Ctx, (): ()) -> Result<()>
    where
        W: BitWrite,
        E: Endianness,
    {
        util::encode_items::<_, E, _, _>(self.iter(), write, ctx)
    }
}

test_codec!([u8; 4]; [0, 1, 2, 3] => [0x00, 0x01, 0x02, 0x03]);
test_roundtrip!([u8; 4]);

#[cfg(test)]
mod test {
    use core::cell::RefCell;

    use bitstream_io::BigEndian;

    use crate::{BitDecodeExt, Error};

    use super::*;

    #[derive(Default)]
    struct MustDropState {
        decoded: bool,
        dropped: bool,
    }

    struct Ctx<'a>(&'a RefCell<MustDropState>);

    struct MustDrop<'a>(&'a RefCell<MustDropState>);

    impl<'a> Drop for MustDrop<'a> {
        fn drop(&mut self) {
            self.0.borrow_mut().dropped = true;
        }
    }

    impl<'a> BitDecode<Ctx<'a>> for MustDrop<'a> {
        fn decode<R, E>(_: &mut R, ctx: &mut Ctx<'a>, (): ()) -> Result<Self>
        where
            R: BitRead,
            E: Endianness,
        {
            let mut state = ctx.0.borrow_mut();
            if state.decoded {
                Err(Error::Other(""))
            } else {
                state.decoded = true;
                Ok(Self(ctx.0))
            }
        }
    }

    #[test]
    fn partial_result_dropped() {
        let state = RefCell::new(MustDropState::default());
        assert!(<[MustDrop; 2]>::decode_bytes_ctx(&[], BigEndian, &mut Ctx(&state), ()).is_err());
        assert!(state.borrow().dropped);
    }
}