use std::{ops::Deref, sync::Arc};
use assembly_core::buffer::{CastError, MinimallyAligned, Repr};
use self::buffer::Buffer;
use super::file::ArrayHeader;
pub mod buffer;
pub mod handle;
pub mod slice;
pub type ArcHandle<B, T> = BaseHandle<Arc<B>, T>;
impl<B: AsRef<[u8]>> ArcHandle<B, ()> {
pub fn new_arc(inner: B) -> Self {
Self::new(Arc::new(inner))
}
}
impl<B: AsRef<[u8]>, T: Copy> ArcHandle<B, T> {
pub fn as_bytes_handle(&self) -> Handle<T> {
BaseHandle {
mem: Buffer::new(self.mem.as_ref().as_ref()),
raw: self.raw,
}
}
}
#[derive(Clone, Debug)]
pub struct BaseHandle<P: Deref, T>
where
<P as Deref>::Target: AsRef<[u8]>,
{
pub(super) mem: P,
pub(super) raw: T,
}
impl<P, T> Copy for BaseHandle<P, T>
where
P: Deref + Copy,
T: Copy,
<P as Deref>::Target: AsRef<[u8]>,
{
}
impl<P: Deref> BaseHandle<P, ()>
where
<P as Deref>::Target: AsRef<[u8]>,
{
pub fn new(mem: P) -> Self {
Self { mem, raw: () }
}
}
impl<T, P: Deref> BaseHandle<P, Option<T>>
where
<P as Deref>::Target: AsRef<[u8]>,
{
pub fn transpose(self) -> Option<BaseHandle<P, T>> {
if let Some(raw) = self.raw {
Some(BaseHandle { mem: self.mem, raw })
} else {
None
}
}
}
impl<P: Deref, T> BaseHandle<P, T>
where
<P as Deref>::Target: AsRef<[u8]>,
{
pub fn raw(&self) -> &T {
&self.raw
}
pub fn raw_mut(&mut self) -> &mut T {
&mut self.raw
}
pub fn as_bytes(&self) -> &[u8] {
self.mem.deref().as_ref()
}
pub fn replace<O>(self, raw: O) -> BaseHandle<P, O> {
BaseHandle { mem: self.mem, raw }
}
}
pub type Handle<'a, T> = BaseHandle<Buffer<'a>, T>;
impl<'a, T> Handle<'a, T> {
pub fn buf(self) -> Buffer<'a> {
self.mem
}
pub fn into_raw(self) -> T {
self.raw
}
pub(crate) fn wrap<R>(&self, raw: R) -> Handle<'a, R> {
Handle { mem: self.mem, raw }
}
pub(crate) fn try_map_cast<R: MinimallyAligned>(
&self,
offset: u32,
) -> Result<RefHandle<'a, R>, CastError> {
let raw: &'a R = self.mem.try_cast(offset)?;
Ok(self.wrap(raw))
}
pub(crate) fn try_map_cast_slice<R: MinimallyAligned>(
&self,
offset: u32,
count: u32,
) -> Result<RefHandle<'a, [R]>, CastError> {
let raw: &'a [R] = self.mem.try_cast_slice(offset, count)?;
Ok(self.wrap(raw))
}
pub(crate) fn try_map_cast_array<R: MinimallyAligned>(
&self,
array: ArrayHeader,
) -> Result<RefHandle<'a, [R]>, CastError> {
let raw: &'a [R] = self.mem.try_cast_slice(array.base_offset, array.count)?;
Ok(self.wrap(raw))
}
pub fn map<X>(self, mapper: impl Fn(Buffer<'a>, T) -> X) -> Handle<'a, X> {
let raw = mapper(self.mem, self.raw);
Handle { mem: self.mem, raw }
}
pub fn map_val<X>(self, mapper: impl Fn(T) -> X) -> Handle<'a, X> {
let raw = mapper(self.raw);
Handle { mem: self.mem, raw }
}
pub fn try_map<X, E>(
self,
mapper: impl Fn(Buffer<'a>, T) -> Result<X, E>,
) -> Result<Handle<'a, X>, E> {
let raw = mapper(self.mem, self.raw)?;
Ok(Handle { mem: self.mem, raw })
}
}
impl<'a, T> RefHandle<'a, [T]> {
pub fn get(self, index: usize) -> Option<RefHandle<'a, T>> {
self.raw.get(index).map(|raw| self.wrap(raw))
}
}
pub type RefHandle<'a, T> = Handle<'a, &'a T>;
impl<'a, T: Repr> RefHandle<'a, T> {
pub fn map_extract(self) -> Handle<'a, T::Value> {
self.wrap(self.raw.extract())
}
}