planus 1.1.1

Planus is an alternative compiler for flatbuffers, an efficient cross platform serialization library.
Documentation
use core::mem::MaybeUninit;

use crate::{
    builder::Builder, errors::ErrorKind, slice_helpers::SliceWithStartOffset, traits::*, Cursor,
};

macro_rules! unsafe_gen_primitive_types {
    ($ty:ty, $size:expr) => {
        /// # Safety
        /// `ALIGNMENT` and `SIZE` should match the actual alignment and size of the type.
        unsafe impl Primitive for $ty {
            const ALIGNMENT: usize = $size;
            const SIZE: usize = $size;
        }

        impl WriteAsPrimitive<$ty> for $ty {
            #[inline]
            fn write<const N: usize>(&self, cursor: Cursor<'_, N>, _buffer_position: u32) {
                cursor.assert_size().finish(self.to_le_bytes());
            }
        }

        impl WriteAs<$ty> for $ty {
            type Prepared = Self;
            #[inline]
            fn prepare(&self, _builder: &mut Builder) -> Self {
                *self
            }
        }

        impl WriteAsDefault<$ty, $ty> for $ty {
            type Prepared = Self;
            #[inline]
            fn prepare(&self, _builder: &mut Builder, default: &$ty) -> Option<Self> {
                #[allow(clippy::float_cmp)]
                if self == default {
                    None
                } else {
                    Some(*self)
                }
            }
        }

        impl WriteAsOptional<$ty> for $ty {
            type Prepared = Self;
            #[inline]
            fn prepare(&self, _builder: &mut Builder) -> Option<Self> {
                Some(*self)
            }
        }

        impl<'buf> TableRead<'buf> for $ty {
            #[inline]
            fn from_buffer(
                buffer: SliceWithStartOffset<'buf>,
                offset: usize,
            ) -> core::result::Result<$ty, ErrorKind> {
                let buffer = buffer.advance_as_array(offset)?.as_array();
                Ok(<$ty>::from_le_bytes(*buffer))
            }
        }
    };
}

macro_rules! unsafe_gen_primitive_types_with_vectors {
    ($ty:ty, $size:expr) => {
        unsafe_gen_primitive_types!($ty, $size);
        impl<'buf> VectorRead<'buf> for $ty {
            const STRIDE: usize = $size;
            #[inline]
            unsafe fn from_buffer(buffer: SliceWithStartOffset<'buf>, offset: usize) -> $ty {
                let buffer = buffer.unchecked_advance_as_array(offset).as_array();
                <$ty>::from_le_bytes(*buffer)
            }
        }

        /// # Safety
        /// write_values initalizes bytes.
        unsafe impl VectorWrite<$ty> for $ty {
            const STRIDE: usize = $size;
            type Value = $ty;
            #[inline]
            fn prepare(&self, _builder: &mut Builder) -> Self::Value {
                *self
            }

            #[inline]
            unsafe fn write_values(
                values: &[$ty],
                bytes: *mut MaybeUninit<u8>,
                buffer_position: u32,
            ) {
                let bytes = bytes as *mut [MaybeUninit<u8>; $size];
                for (i, v) in values.iter().enumerate() {
                    v.write(
                        Cursor::new(&mut *bytes.add(i)),
                        buffer_position - ($size * i) as u32,
                    );
                }
            }
        }
    };
}

unsafe_gen_primitive_types!(i8, core::mem::size_of::<i8>());
unsafe_gen_primitive_types!(u8, core::mem::size_of::<u8>());
unsafe_gen_primitive_types_with_vectors!(i16, core::mem::size_of::<i16>());
unsafe_gen_primitive_types_with_vectors!(u16, core::mem::size_of::<u16>());
unsafe_gen_primitive_types_with_vectors!(i32, core::mem::size_of::<i32>());
unsafe_gen_primitive_types_with_vectors!(u32, core::mem::size_of::<u32>());
unsafe_gen_primitive_types_with_vectors!(i64, core::mem::size_of::<i64>());
unsafe_gen_primitive_types_with_vectors!(u64, core::mem::size_of::<u64>());
unsafe_gen_primitive_types_with_vectors!(f32, core::mem::size_of::<f32>());
unsafe_gen_primitive_types_with_vectors!(f64, core::mem::size_of::<f64>());