use byteorder::{BigEndian, WriteBytesExt}; use serde::Serialize; use std::io::Write; use std::mem::size_of; use crate::mp4box::*; #[derive(Debug, Clone, PartialEq, Eq, Default, Serialize)] pub struct StszBox { pub version: u8, pub flags: u32, pub sample_size: u32, pub sample_count: u32, #[serde(skip_serializing)] pub sample_sizes: Vec, } impl StszBox { pub fn get_type(&self) -> BoxType { BoxType::StszBox } pub fn get_size(&self) -> u64 { HEADER_SIZE + HEADER_EXT_SIZE + 8 + (4 * self.sample_sizes.len() as u64) } } impl Mp4Box for StszBox { const TYPE: BoxType = BoxType::StszBox; fn box_size(&self) -> u64 { self.get_size() } fn to_json(&self) -> Result { Ok(serde_json::to_string(&self).unwrap()) } fn summary(&self) -> Result { let s = format!( "sample_size={} sample_count={} sample_sizes={}", self.sample_size, self.sample_count, self.sample_sizes.len() ); Ok(s) } } impl BlockReader for StszBox { fn read_block<'a>(reader: &mut impl Reader<'a>) -> Result { let (version, flags) = read_box_header_ext(reader); let sample_size = reader.get_u32(); let stsz_item_size = if sample_size == 0 { size_of::() // entry_size } else { 0 }; let sample_count = reader.get_u32(); let mut sample_sizes = Vec::new(); if sample_size == 0 { if sample_count as usize > reader.remaining() / stsz_item_size { return Err(Error::InvalidData( "stsz sample_count indicates more values than could fit in the box", )); } sample_sizes.reserve(sample_count as usize); for _ in 0..sample_count { let sample_number = reader.get_u32(); sample_sizes.push(sample_number); } } Ok(StszBox { version, flags, sample_size, sample_count, sample_sizes, }) } fn size_hint() -> usize { 12 } } impl WriteBox<&mut W> for StszBox { fn write_box(&self, writer: &mut W) -> Result { let size = self.box_size(); BoxHeader::new(Self::TYPE, size).write(writer)?; write_box_header_ext(writer, self.version, self.flags)?; writer.write_u32::(self.sample_size)?; writer.write_u32::(self.sample_count)?; if self.sample_size == 0 { if self.sample_count != self.sample_sizes.len() as u32 { return Err(Error::InvalidData("sample count out of sync")); } for sample_number in self.sample_sizes.iter() { writer.write_u32::(*sample_number)?; } } Ok(size) } } #[cfg(test)] mod tests { use super::*; use crate::mp4box::BoxHeader; #[test] fn test_stsz_same_size() { let src_box = StszBox { version: 0, flags: 0, sample_size: 1165, sample_count: 12, sample_sizes: vec![], }; let mut buf = Vec::new(); src_box.write_box(&mut buf).unwrap(); assert_eq!(buf.len(), src_box.box_size() as usize); let mut reader = buf.as_slice(); let header = BoxHeader::read_sync(&mut reader).unwrap().unwrap(); assert_eq!(header.kind, BoxType::StszBox); assert_eq!(src_box.box_size(), header.size); let dst_box = StszBox::read_block(&mut reader).unwrap(); assert_eq!(src_box, dst_box); } #[test] fn test_stsz_many_sizes() { let src_box = StszBox { version: 0, flags: 0, sample_size: 0, sample_count: 9, sample_sizes: vec![1165, 11, 11, 8545, 10126, 10866, 9643, 9351, 7730], }; let mut buf = Vec::new(); src_box.write_box(&mut buf).unwrap(); assert_eq!(buf.len(), src_box.box_size() as usize); let mut reader = buf.as_slice(); let header = BoxHeader::read_sync(&mut reader).unwrap().unwrap(); assert_eq!(header.kind, BoxType::StszBox); assert_eq!(src_box.box_size(), header.size); let dst_box = StszBox::read_block(&mut reader).unwrap(); assert_eq!(src_box, dst_box); } }