| 1 | use alloc::vec; |
| 2 | use alloc::vec::Vec; |
| 3 | use core::mem; |
| 4 | use crate::decoder::MAX_COMPONENTS; |
| 5 | use crate::error::Result; |
| 6 | use crate::idct::dequantize_and_idct_block; |
| 7 | use crate::alloc::sync::Arc; |
| 8 | use crate::parser::Component; |
| 9 | use super::{RowData, Worker}; |
| 10 | |
| 11 | pub struct ImmediateWorker { |
| 12 | offsets: [usize; MAX_COMPONENTS], |
| 13 | results: Vec<Vec<u8>>, |
| 14 | components: Vec<Option<Component>>, |
| 15 | quantization_tables: Vec<Option<Arc<[u16; 64]>>>, |
| 16 | } |
| 17 | |
| 18 | impl Default for ImmediateWorker { |
| 19 | fn default() -> Self { |
| 20 | ImmediateWorker { |
| 21 | offsets: [0; MAX_COMPONENTS], |
| 22 | results: vec![Vec::new(); MAX_COMPONENTS], |
| 23 | components: vec![None; MAX_COMPONENTS], |
| 24 | quantization_tables: vec![None; MAX_COMPONENTS], |
| 25 | } |
| 26 | } |
| 27 | } |
| 28 | |
| 29 | impl ImmediateWorker { |
| 30 | pub fn start_immediate(&mut self, data: RowData) { |
| 31 | assert!(self.results[data.index].is_empty()); |
| 32 | |
| 33 | self.offsets[data.index] = 0; |
| 34 | self.results[data.index].resize(data.component.block_size.width as usize * data.component.block_size.height as usize * data.component.dct_scale * data.component.dct_scale, 0u8); |
| 35 | self.components[data.index] = Some(data.component); |
| 36 | self.quantization_tables[data.index] = Some(data.quantization_table); |
| 37 | } |
| 38 | |
| 39 | pub fn append_row_immediate(&mut self, (index, data): (usize, Vec<i16>)) { |
| 40 | // Convert coefficients from a MCU row to samples. |
| 41 | |
| 42 | let component = self.components[index].as_ref().unwrap(); |
| 43 | let quantization_table = self.quantization_tables[index].as_ref().unwrap(); |
| 44 | let block_count = component.block_size.width as usize * component.vertical_sampling_factor as usize; |
| 45 | let line_stride = component.block_size.width as usize * component.dct_scale; |
| 46 | |
| 47 | assert_eq!(data.len(), block_count * 64); |
| 48 | |
| 49 | for i in 0..block_count { |
| 50 | let x = (i % component.block_size.width as usize) * component.dct_scale; |
| 51 | let y = (i / component.block_size.width as usize) * component.dct_scale; |
| 52 | |
| 53 | let coefficients = data[i * 64..(i + 1) * 64].try_into().unwrap(); |
| 54 | let output = &mut self.results[index][self.offsets[index] + y * line_stride + x..]; |
| 55 | |
| 56 | dequantize_and_idct_block(component.dct_scale, coefficients, quantization_table, line_stride, output); |
| 57 | } |
| 58 | |
| 59 | self.offsets[index] += block_count * component.dct_scale * component.dct_scale; |
| 60 | } |
| 61 | |
| 62 | pub fn get_result_immediate(&mut self, index: usize) -> Vec<u8> { |
| 63 | mem::take(&mut self.results[index]) |
| 64 | } |
| 65 | } |
| 66 | |
| 67 | impl Worker for ImmediateWorker { |
| 68 | fn start(&mut self, data: RowData) -> Result<()> { |
| 69 | self.start_immediate(data); |
| 70 | Ok(()) |
| 71 | } |
| 72 | fn append_row(&mut self, row: (usize, Vec<i16>)) -> Result<()> { |
| 73 | self.append_row_immediate(row); |
| 74 | Ok(()) |
| 75 | } |
| 76 | fn get_result(&mut self, index: usize) -> Result<Vec<u8>> { |
| 77 | Ok(self.get_result_immediate(index)) |
| 78 | } |
| 79 | } |
| 80 | |