1use assembly_core::buffer::{Buffer, CastError, MinimallyAligned, Repr};
2use assembly_fdb_core::file::ArrayHeader;
3use latin1str::Latin1Str;
4use std::{convert::TryFrom, mem::size_of, ops::Deref, result::Result};
5
6#[derive(Clone, Debug)]
13pub struct BaseHandle<P: Deref, T>
14where
15 <P as Deref>::Target: AsRef<[u8]>,
16{
17 pub(super) mem: P,
19 pub(super) raw: T,
21}
22
23impl<P, T> Copy for BaseHandle<P, T>
24where
25 P: Deref + Copy,
26 T: Copy,
27 <P as Deref>::Target: AsRef<[u8]>,
28{
29}
30
31impl<P: Deref> BaseHandle<P, ()>
32where
33 <P as Deref>::Target: AsRef<[u8]>,
34{
35 pub fn new(mem: P) -> Self {
37 Self { mem, raw: () }
38 }
39}
40
41impl<T, P: Deref> BaseHandle<P, Option<T>>
42where
43 <P as Deref>::Target: AsRef<[u8]>,
44{
45 pub fn transpose(self) -> Option<BaseHandle<P, T>> {
47 if let Some(raw) = self.raw {
48 Some(BaseHandle { mem: self.mem, raw })
49 } else {
50 None
51 }
52 }
53}
54
55impl<P: Deref, T> BaseHandle<P, T>
56where
57 <P as Deref>::Target: AsRef<[u8]>,
58{
59 pub fn raw(&self) -> &T {
61 &self.raw
62 }
63
64 pub fn raw_mut(&mut self) -> &mut T {
66 &mut self.raw
67 }
68
69 pub fn as_bytes(&self) -> &[u8] {
71 self.mem.deref().as_ref()
72 }
73
74 pub fn replace<O>(self, raw: O) -> BaseHandle<P, O> {
76 BaseHandle { mem: self.mem, raw }
77 }
78}
79
80pub type Handle<'a, T> = BaseHandle<&'a [u8], T>;
82
83impl<'a, T> Handle<'a, T> {
84 pub fn buf(self) -> &'a [u8] {
86 self.mem
87 }
88
89 pub fn into_raw(self) -> T {
91 self.raw
92 }
93
94 pub(crate) fn wrap<R>(&self, raw: R) -> Handle<'a, R> {
96 Handle { mem: self.mem, raw }
97 }
98
99 pub(crate) fn try_map_cast<R: MinimallyAligned>(
101 &self,
102 offset: u32,
103 ) -> Result<RefHandle<'a, R>, CastError> {
104 let raw: &'a R = self.mem.try_cast(offset)?;
105 Ok(self.wrap(raw))
106 }
107
108 pub(crate) fn try_map_cast_slice<R: MinimallyAligned>(
110 &self,
111 offset: u32,
112 count: u32,
113 ) -> Result<RefHandle<'a, [R]>, CastError> {
114 let raw: &'a [R] = self.mem.try_cast_slice(offset, count)?;
115 Ok(self.wrap(raw))
116 }
117
118 pub(crate) fn try_map_cast_array<R: MinimallyAligned>(
120 &self,
121 array: ArrayHeader,
122 ) -> Result<RefHandle<'a, [R]>, CastError> {
123 let raw: &'a [R] = self.mem.try_cast_slice(array.base_offset, array.count)?;
124 Ok(self.wrap(raw))
125 }
126
127 pub fn map<X>(self, mapper: impl Fn(&'a [u8], T) -> X) -> Handle<'a, X> {
129 let raw = mapper(self.mem, self.raw);
130 Handle { mem: self.mem, raw }
131 }
132
133 pub fn map_val<X>(self, mapper: impl Fn(T) -> X) -> Handle<'a, X> {
135 let raw = mapper(self.raw);
136 Handle { mem: self.mem, raw }
137 }
138
139 pub fn try_map<X, E>(
141 self,
142 mapper: impl Fn(&'a [u8], T) -> Result<X, E>,
143 ) -> Result<Handle<'a, X>, E> {
144 let raw = mapper(self.mem, self.raw)?;
145 Ok(Handle { mem: self.mem, raw })
146 }
147}
148
149impl<'a, T> Iterator for Handle<'a, T>
150where
151 T: Iterator,
152{
153 type Item = Handle<'a, T::Item>;
154
155 fn next(&mut self) -> Option<Self::Item> {
157 self.raw.next().map(|raw| Handle { mem: self.mem, raw })
158 }
159}
160
161pub trait TryFromHandle<'a, T>: Sized {
163 type Error;
165 fn try_from(h: Handle<'a, T>) -> Result<Self, Self::Error>;
167}
168
169pub type RefHandle<'a, T> = Handle<'a, &'a T>;
171
172impl<'a, T> RefHandle<'a, [T]> {
173 pub fn get(self, index: usize) -> Option<RefHandle<'a, T>> {
175 self.raw.get(index).map(|raw| self.wrap(raw))
176 }
177}
178
179impl<'a, T: Repr> RefHandle<'a, T> {
180 pub fn map_extract(self) -> Handle<'a, T::Value> {
182 self.wrap(self.raw.extract())
183 }
184}
185
186pub type SliceIterHandle<'a, T> = Handle<'a, std::slice::Iter<'a, T>>;
188
189pub fn get_string(buf: &[u8], offset: u32) -> Result<&Latin1Str, CastError> {
191 let start = offset as usize;
192 let buf = buf.get(start..).ok_or(CastError::OutOfBounds { offset })?;
193 Ok(Latin1Str::from_bytes_until_nul(buf))
194}
195
196pub fn get_i64(buf: &[u8], addr: u32) -> Result<i64, CastError> {
198 let start = addr as usize;
199 let end = start + size_of::<u64>();
200 if end > buf.len() {
201 Err(CastError::OutOfBounds { offset: addr })
202 } else {
203 let (_, base) = buf.split_at(start);
204 let (bytes, _) = base.split_at(size_of::<u64>());
205 let val = i64::from_le_bytes(<[u8; 8]>::try_from(bytes).unwrap());
206 Ok(val)
207 }
208}