1#![deny(clippy::arithmetic_side_effects)]
4
5use bytemuck::AnyBitPattern;
6use font_types::FixedSize;
7
8use crate::read::{ComputeSize, FontReadWithArgs, ReadArgs, VarSize};
9use crate::{FontData, FontRead, ReadError};
10
11#[derive(Clone)]
20pub struct ComputedArray<'a, T: ReadArgs> {
21 item_len: usize,
23 len: usize,
24 data: FontData<'a>,
25 args: T::Args,
26}
27
28impl<'a, T: ComputeSize> ComputedArray<'a, T> {
29 pub fn new(data: FontData<'a>, args: T::Args) -> Result<Self, ReadError> {
30 let item_len = T::compute_size(&args)?;
31 let len = data.len().checked_div(item_len).unwrap_or(0);
32 Ok(ComputedArray {
33 item_len,
34 len,
35 data,
36 args,
37 })
38 }
39
40 pub fn len(&self) -> usize {
42 self.len
43 }
44
45 pub fn is_empty(&self) -> bool {
46 self.len == 0
47 }
48}
49
50impl<T: ReadArgs> ReadArgs for ComputedArray<'_, T> {
51 type Args = T::Args;
52}
53
54impl<'a, T> FontReadWithArgs<'a> for ComputedArray<'a, T>
55where
56 T: ComputeSize + FontReadWithArgs<'a>,
57 T::Args: Copy,
58{
59 fn read_with_args(data: FontData<'a>, args: &Self::Args) -> Result<Self, ReadError> {
60 Self::new(data, *args)
61 }
62}
63
64impl<T> Default for ComputedArray<'_, T>
65where
66 T: ReadArgs,
67 T::Args: Default,
68{
69 fn default() -> Self {
70 Self {
71 item_len: 0,
72 len: 0,
73 data: Default::default(),
74 args: Default::default(),
75 }
76 }
77}
78
79impl<'a, T> ComputedArray<'a, T>
80where
81 T: FontReadWithArgs<'a>,
82 T::Args: Copy + 'static,
83{
84 pub fn iter(&self) -> impl Iterator<Item = Result<T, ReadError>> + 'a {
85 let mut i = 0;
86 let data = self.data;
87 let args = self.args;
88 let item_len = self.item_len;
89 let len = self.len;
90
91 std::iter::from_fn(move || {
92 if i == len {
93 return None;
94 }
95 let item_start = item_len.checked_mul(i)?;
96 i = i.checked_add(1)?;
97 let data = data.split_off(item_start)?;
98 Some(T::read_with_args(data, &args))
99 })
100 }
101
102 #[inline]
103 pub fn get(&self, idx: usize) -> Result<T, ReadError> {
104 let item_start = idx
105 .checked_mul(self.item_len)
106 .ok_or(ReadError::OutOfBounds)?;
107 self.data
108 .split_off(item_start)
109 .ok_or(ReadError::OutOfBounds)
110 .and_then(|data| T::read_with_args(data, &self.args))
111 }
112}
113
114impl<T: ReadArgs> std::fmt::Debug for ComputedArray<'_, T> {
115 fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
116 f.debug_struct("DynSizedArray")
117 .field("bytes", &self.data)
118 .finish()
119 }
120}
121
122pub struct VarLenArray<'a, T> {
127 data: FontData<'a>,
128 phantom: std::marker::PhantomData<*const T>,
129}
130
131impl<'a, T: FontRead<'a> + VarSize> VarLenArray<'a, T> {
132 pub fn get(&self, idx: usize) -> Option<Result<T, ReadError>> {
145 if self.data.is_empty() {
146 return None;
147 }
148
149 let mut pos = 0usize;
150 for _ in 0..idx {
151 pos = pos.checked_add(T::read_len_at(self.data, pos)?)?;
152 }
153 self.data.split_off(pos).map(T::read)
154 }
155
156 pub fn iter(&self) -> impl Iterator<Item = Result<T, ReadError>> + 'a {
158 let mut data = self.data;
159 std::iter::from_fn(move || {
160 if data.is_empty() {
161 return None;
162 }
163
164 let item_len = T::read_len_at(data, 0)?;
165 if item_len == 0 {
170 return None;
171 }
172 let item_data = data.slice(..item_len)?;
173 let next = T::read(item_data);
174 data = data.split_off(item_len)?;
175 Some(next)
176 })
177 }
178}
179
180impl<'a, T> FontRead<'a> for VarLenArray<'a, T> {
181 fn read(data: FontData<'a>) -> Result<Self, ReadError> {
182 Ok(VarLenArray {
183 data,
184 phantom: core::marker::PhantomData,
185 })
186 }
187}
188
189impl<T> Default for VarLenArray<'_, T> {
190 fn default() -> Self {
191 Self {
192 data: Default::default(),
193 phantom: std::marker::PhantomData,
194 }
195 }
196}
197
198impl<T: AnyBitPattern> ReadArgs for &[T] {
199 type Args = u16;
200}
201
202impl<'a, T: AnyBitPattern + FixedSize> FontReadWithArgs<'a> for &'a [T] {
203 fn read_with_args(data: FontData<'a>, args: &u16) -> Result<Self, ReadError> {
204 let len = (*args as usize)
205 .checked_mul(T::RAW_BYTE_LEN)
206 .ok_or(ReadError::OutOfBounds)?;
207 data.read_array(0..len)
208 }
209}
210
211#[cfg(test)]
212mod tests {
213 use super::*;
214 use crate::codegen_test::records::VarLenItem;
215 use font_test_data::bebuffer::BeBuffer;
216
217 impl VarSize for VarLenItem<'_> {
218 type Size = u32;
219
220 fn read_len_at(data: FontData, pos: usize) -> Option<usize> {
221 data.read_at::<u32>(pos).ok().map(|len| len as usize)
222 }
223 }
224
225 #[test]
229 fn var_len_iter_with_zero_length_item() {
230 let mut buf = BeBuffer::new();
233 buf = buf.push(8u32).extend([0u8; 4]);
234 buf = buf.push(18u32).extend([0u8; 14]);
235 buf = buf.push(0u32);
236 let arr: VarLenArray<VarLenItem> = VarLenArray::read(FontData::new(buf.data())).unwrap();
237 assert_eq!(arr.iter().take(10).count(), 2);
241 }
242}