use core::marker::PhantomData;
use std::io::{
Read,
Result,
Seek,
SeekFrom,
};
use crate::ReadExt;
use crate::codec::{
BigEndian,
BinaryCodec,
ByteOrder,
ByteOrderSpec,
LittleEndian,
};
use crate::util::read_utf8_payload;
pub struct BinaryReader<R, O = BigEndian> {
inner: R,
buffer: [u8; 16],
marker: PhantomData<fn() -> O>,
}
impl<R, O> BinaryReader<R, O>
where
O: ByteOrderSpec,
{
#[must_use]
#[inline]
pub const fn new(inner: R) -> Self {
Self {
inner,
buffer: [0; 16],
marker: PhantomData,
}
}
#[must_use]
#[inline]
pub const fn byte_order(&self) -> ByteOrder {
O::ORDER
}
#[must_use]
#[inline]
pub const fn get_ref(&self) -> &R {
&self.inner
}
#[must_use]
#[inline]
pub fn get_mut(&mut self) -> &mut R {
&mut self.inner
}
#[must_use]
#[inline]
pub fn into_inner(self) -> R {
self.inner
}
}
macro_rules! impl_value_read {
($order:ty, $method:ident, $ty:ty, $doc:literal) => {
#[doc = $doc]
#[inline]
pub fn $method(&mut self) -> Result<$ty> {
type Codec = BinaryCodec<$ty, $order>;
const LEN: usize = Codec::REQUIRED_MIN_BUFFER_LEN;
unsafe {
ReadExt::read_exact_unchecked(&mut self.inner, &mut self.buffer, 0, LEN)?;
Ok(Codec::read_unchecked(&self.buffer, 0))
}
}
};
}
macro_rules! impl_for_order {
($order:ty) => {
impl<R> BinaryReader<R, $order>
where
R: Read,
{
impl_value_read!($order, read_u8, u8, "Reads an unsigned 8-bit integer.");
impl_value_read!($order, read_i8, i8, "Reads a signed 8-bit integer.");
impl_value_read!($order, read_u16, u16, "Reads an unsigned 16-bit integer.");
impl_value_read!($order, read_u32, u32, "Reads an unsigned 32-bit integer.");
impl_value_read!($order, read_u64, u64, "Reads an unsigned 64-bit integer.");
impl_value_read!($order, read_u128, u128, "Reads an unsigned 128-bit integer.");
impl_value_read!($order, read_i16, i16, "Reads a signed 16-bit integer.");
impl_value_read!($order, read_i32, i32, "Reads a signed 32-bit integer.");
impl_value_read!($order, read_i64, i64, "Reads a signed 64-bit integer.");
impl_value_read!($order, read_i128, i128, "Reads a signed 128-bit integer.");
impl_value_read!($order, read_f32, f32, "Reads a 32-bit float.");
impl_value_read!($order, read_f64, f64, "Reads a 64-bit float.");
#[inline]
pub fn read_utf8_string_u16(&mut self, max_len: usize) -> Result<String> {
let len = usize::from(self.read_u16()?);
read_utf8_payload(&mut self.inner, len, max_len)
}
#[inline]
pub fn read_utf8_string_u32(&mut self, max_len: usize) -> Result<String> {
let len = self.read_u32()? as usize;
read_utf8_payload(&mut self.inner, len, max_len)
}
}
};
}
impl_for_order!(BigEndian);
impl_for_order!(LittleEndian);
impl<R, O> Read for BinaryReader<R, O>
where
R: Read,
{
#[inline]
fn read(&mut self, buffer: &mut [u8]) -> Result<usize> {
self.inner.read(buffer)
}
}
impl<R, O> Seek for BinaryReader<R, O>
where
R: Seek,
{
#[inline]
fn seek(&mut self, position: SeekFrom) -> Result<u64> {
self.inner.seek(position)
}
}