1use std::collections::HashMap;
6
7use super::messages::{Dii, DownloadDataBlock};
8
9#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
11#[cfg_attr(feature = "serde", derive(serde::Serialize))]
12pub struct ModuleKey {
13 pub download_id: u32,
15 pub module_id: u16,
17 pub module_version: u8,
19}
20
21#[derive(Debug, Clone, PartialEq, Eq)]
23#[cfg_attr(feature = "serde", derive(serde::Serialize))]
24pub struct Module {
25 pub key: ModuleKey,
27 pub data: Vec<u8>,
29}
30
31type SlotKey = (u32, u16); struct Slot {
40 module_version: u8,
41 block_size: usize,
42 data: Vec<u8>,
43 received: Vec<u64>,
44 n_blocks: usize,
45 remaining: usize,
46}
47
48impl Slot {
49 fn is_received(&self, n: usize) -> bool {
50 (self.received[n >> 6] >> (n & 63)) & 1 != 0
51 }
52 fn mark_received(&mut self, n: usize) {
53 self.received[n >> 6] |= 1 << (n & 63);
54 }
55}
56
57pub const DEFAULT_MAX_MODULE_SIZE: u32 = 64 * 1024 * 1024;
59pub const DEFAULT_MAX_TOTAL_BYTES: usize = 256 * 1024 * 1024;
63pub const DEFAULT_MAX_SLOTS: usize = 16 * 1024;
70
71pub struct ModuleReassembler {
95 slots: HashMap<SlotKey, Slot>,
96 max_module_size: u32,
97 max_total_bytes: usize,
98 max_slots: usize,
99 total_bytes: usize,
100}
101
102impl Default for ModuleReassembler {
103 fn default() -> Self {
104 Self::new()
105 }
106}
107
108impl ModuleReassembler {
109 #[must_use]
112 pub fn new() -> Self {
113 Self::with_limits(DEFAULT_MAX_MODULE_SIZE, DEFAULT_MAX_TOTAL_BYTES)
114 }
115
116 #[must_use]
119 pub fn with_max_module_size(max_module_size: u32) -> Self {
120 Self::with_limits(max_module_size, DEFAULT_MAX_TOTAL_BYTES)
121 }
122
123 #[must_use]
126 pub fn with_limits(max_module_size: u32, max_total_bytes: usize) -> Self {
127 Self {
128 slots: HashMap::new(),
129 max_module_size,
130 max_total_bytes,
131 max_slots: DEFAULT_MAX_SLOTS,
132 total_bytes: 0,
133 }
134 }
135
136 #[must_use]
141 pub fn with_max_slots(mut self, max_slots: usize) -> Self {
142 self.max_slots = max_slots;
143 self
144 }
145
146 pub fn note_dii(&mut self, dii: &Dii<'_>) {
153 for m in &dii.modules {
154 if m.module_size == 0 || m.module_size > self.max_module_size || dii.block_size == 0 {
155 continue;
156 }
157 let key: SlotKey = (dii.download_id, m.module_id);
158 if let Some(existing) = self.slots.get(&key) {
159 if existing.module_version == m.module_version {
160 continue; }
162 let s = self.slots.remove(&key).expect("just found");
164 self.total_bytes -= s.data.len();
165 }
166 let size = m.module_size as usize;
167 if self.total_bytes + size > self.max_total_bytes || self.slots.len() >= self.max_slots
168 {
169 continue; }
171 let block_size = dii.block_size as usize;
172 let n_blocks = size.div_ceil(block_size).max(1);
173 self.total_bytes += size;
174 self.slots.insert(
175 key,
176 Slot {
177 module_version: m.module_version,
178 block_size,
179 data: vec![0u8; size],
180 received: vec![0u64; n_blocks.div_ceil(64)],
181 n_blocks,
182 remaining: n_blocks,
183 },
184 );
185 }
186 }
187
188 pub fn feed_ddb(&mut self, ddb: &DownloadDataBlock<'_>) -> Option<Module> {
193 let key: SlotKey = (ddb.download_id, ddb.module_id);
194 let slot = self.slots.get_mut(&key)?;
195 let n = ddb.block_number as usize;
196 if slot.module_version != ddb.module_version || n >= slot.n_blocks || slot.is_received(n) {
197 return None;
198 }
199 let offset = n * slot.block_size;
200 let expected = (slot.data.len() - offset).min(slot.block_size);
201 if ddb.block_data.len() != expected {
202 return None; }
204 slot.data[offset..offset + expected].copy_from_slice(ddb.block_data);
205 slot.mark_received(n);
206 slot.remaining -= 1;
207 if slot.remaining > 0 {
208 return None;
209 }
210 let slot = self.slots.remove(&key).expect("slot exists");
211 self.total_bytes -= slot.data.len();
212 Some(Module {
213 key: ModuleKey {
214 download_id: ddb.download_id,
215 module_id: ddb.module_id,
216 module_version: slot.module_version,
217 },
218 data: slot.data,
219 })
220 }
221
222 #[must_use]
224 pub fn pending(&self) -> usize {
225 self.slots.len()
226 }
227
228 #[must_use]
237 pub fn pending_bytes(&self) -> usize {
238 self.total_bytes
239 }
240}
241
242#[cfg(test)]
243mod tests {
244 use super::super::messages::DiiModule;
245 use super::*;
246
247 fn dii(download_id: u32, block_size: u16, modules: Vec<DiiModule<'static>>) -> Dii<'static> {
248 Dii {
249 transaction_id: 0x8000_0002,
250 adaptation: &[],
251 download_id,
252 block_size,
253 window_size: 0,
254 ack_period: 0,
255 t_c_download_window: 0,
256 t_c_download_scenario: 0,
257 compatibility_descriptor: &[],
258 modules,
259 private_data: &[],
260 }
261 }
262
263 fn module(module_id: u16, module_size: u32, module_version: u8) -> DiiModule<'static> {
264 DiiModule {
265 module_id,
266 module_size,
267 module_version,
268 module_info: &[],
269 }
270 }
271
272 fn ddb(
273 download_id: u32,
274 module_id: u16,
275 module_version: u8,
276 block_number: u16,
277 block_data: &[u8],
278 ) -> DownloadDataBlock<'_> {
279 DownloadDataBlock {
280 download_id,
281 adaptation: &[],
282 module_id,
283 module_version,
284 block_number,
285 block_data,
286 }
287 }
288
289 #[test]
290 fn two_block_module_completes() {
291 let mut r = ModuleReassembler::new();
292 r.note_dii(&dii(1, 4, vec![module(7, 6, 0)]));
293 assert!(r.feed_ddb(&ddb(1, 7, 0, 0, &[1, 2, 3, 4])).is_none());
294 let m = r.feed_ddb(&ddb(1, 7, 0, 1, &[5, 6])).expect("complete");
295 assert_eq!(m.key.module_id, 7);
296 assert_eq!(m.data, vec![1, 2, 3, 4, 5, 6]);
297 assert_eq!(r.pending(), 0);
298 }
299
300 #[test]
301 fn out_of_order_blocks_complete() {
302 let mut r = ModuleReassembler::new();
303 r.note_dii(&dii(1, 2, vec![module(1, 4, 0)]));
304 assert!(r.feed_ddb(&ddb(1, 1, 0, 1, &[3, 4])).is_none());
305 let m = r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2])).expect("complete");
306 assert_eq!(m.data, vec![1, 2, 3, 4]);
307 }
308
309 #[test]
310 fn ddb_before_dii_is_ignored() {
311 let mut r = ModuleReassembler::new();
312 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2])).is_none());
313 r.note_dii(&dii(1, 2, vec![module(1, 2, 0)]));
315 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2])).is_some());
316 }
317
318 #[test]
319 fn version_mismatch_ignored_and_new_version_restarts() {
320 let mut r = ModuleReassembler::new();
321 r.note_dii(&dii(1, 2, vec![module(1, 4, 0)]));
322 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2])).is_none());
323 assert!(r.feed_ddb(&ddb(1, 1, 3, 1, &[9, 9])).is_none());
325 r.note_dii(&dii(1, 2, vec![module(1, 4, 3)]));
327 assert_eq!(r.pending(), 1);
328 assert!(r.feed_ddb(&ddb(1, 1, 3, 0, &[5, 6])).is_none());
329 let m = r.feed_ddb(&ddb(1, 1, 3, 1, &[7, 8])).expect("complete");
330 assert_eq!(m.key.module_version, 3);
331 assert_eq!(m.data, vec![5, 6, 7, 8]);
332 }
333
334 #[test]
335 fn repeated_dii_keeps_progress() {
336 let mut r = ModuleReassembler::new();
337 let d = dii(1, 2, vec![module(1, 4, 0)]);
338 r.note_dii(&d);
339 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2])).is_none());
340 r.note_dii(&d); let m = r.feed_ddb(&ddb(1, 1, 0, 1, &[3, 4])).expect("complete");
342 assert_eq!(m.data, vec![1, 2, 3, 4]);
343 }
344
345 #[test]
346 fn duplicate_and_out_of_range_blocks_ignored() {
347 let mut r = ModuleReassembler::new();
348 r.note_dii(&dii(1, 2, vec![module(1, 4, 0)]));
349 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2])).is_none());
350 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2])).is_none()); assert!(r.feed_ddb(&ddb(1, 1, 0, 9, &[9, 9])).is_none()); assert_eq!(r.pending(), 1);
353 }
354
355 #[test]
356 fn wrong_block_length_ignored() {
357 let mut r = ModuleReassembler::new();
358 r.note_dii(&dii(1, 4, vec![module(1, 6, 0)]));
359 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2, 3])).is_none());
361 assert!(r.feed_ddb(&ddb(1, 1, 0, 1, &[5, 6, 7])).is_none());
362 assert_eq!(r.pending(), 1);
363 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[1, 2, 3, 4])).is_none());
364 assert!(r.feed_ddb(&ddb(1, 1, 0, 1, &[5, 6])).is_some());
365 }
366
367 #[test]
368 fn oversize_module_skipped() {
369 let mut r = ModuleReassembler::with_max_module_size(8);
370 r.note_dii(&dii(1, 4, vec![module(1, 9, 0), module(2, 8, 0)]));
371 assert_eq!(r.pending(), 1); }
373
374 #[test]
375 fn zero_block_size_skipped() {
376 let mut r = ModuleReassembler::new();
377 r.note_dii(&dii(1, 0, vec![module(1, 4, 0)]));
378 assert_eq!(r.pending(), 0);
379 }
380
381 #[test]
385 fn zero_size_module_announcement_ignored() {
386 let mut r = ModuleReassembler::new();
387 r.note_dii(&dii(1, 4, vec![module(1, 0, 0)]));
388 assert_eq!(r.pending(), 0);
389 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[])).is_none());
390 }
391
392 #[test]
397 fn slot_count_capped() {
398 let mut r = ModuleReassembler::new().with_max_slots(3);
399 let modules: Vec<_> = (0..5).map(|i| module(i, 1, 0)).collect();
400 r.note_dii(&dii(1, 4, modules));
401 assert_eq!(r.pending(), 3); assert!(r.feed_ddb(&ddb(1, 0, 0, 0, &[0xAA])).is_some());
404 assert_eq!(r.pending(), 2);
405 r.note_dii(&dii(1, 4, vec![module(4, 1, 0)]));
407 assert_eq!(r.pending(), 3);
408 }
409
410 #[test]
414 fn aggregate_budget_bounds_total_memory() {
415 let mut r = ModuleReassembler::with_limits(8, 10);
416 r.note_dii(&dii(1, 4, vec![module(1, 8, 0)]));
417 assert_eq!(r.pending_bytes(), 8);
418 r.note_dii(&dii(2, 4, vec![module(1, 8, 0)]));
421 assert_eq!(r.pending(), 1);
422 assert_eq!(r.pending_bytes(), 8);
423 assert!(r.feed_ddb(&ddb(1, 1, 0, 0, &[0; 4])).is_none());
425 assert!(r.feed_ddb(&ddb(1, 1, 0, 1, &[0; 4])).is_some());
426 assert_eq!(r.pending_bytes(), 0);
427 r.note_dii(&dii(2, 4, vec![module(1, 8, 0)]));
429 assert_eq!(r.pending(), 1);
430 assert_eq!(r.pending_bytes(), 8);
431 }
432
433 #[test]
435 fn version_replacement_releases_budget() {
436 let mut r = ModuleReassembler::with_limits(8, 8);
437 r.note_dii(&dii(1, 4, vec![module(1, 8, 0)]));
438 assert_eq!(r.pending_bytes(), 8);
439 r.note_dii(&dii(1, 4, vec![module(1, 8, 1)]));
440 assert_eq!(r.pending(), 1);
441 assert_eq!(r.pending_bytes(), 8); }
443
444 #[test]
447 fn block_size_one_uses_bitset() {
448 let mut r = ModuleReassembler::new();
449 r.note_dii(&dii(1, 1, vec![module(1, 130, 0)]));
450 for i in 0..129u16 {
451 assert!(r.feed_ddb(&ddb(1, 1, 0, i, &[i as u8])).is_none());
452 assert!(r.feed_ddb(&ddb(1, 1, 0, i, &[i as u8])).is_none());
454 }
455 let m = r.feed_ddb(&ddb(1, 1, 0, 129, &[0x81])).expect("complete");
456 assert_eq!(m.data.len(), 130);
457 assert_eq!(m.data[129], 0x81);
458 }
459}