diff options
| author | Felix Kaaman <tmtu@tmtu.ee> | 2021-01-07 20:32:49 +0100 |
|---|---|---|
| committer | Felix Kaaman <tmtu@tmtu.ee> | 2021-01-07 20:32:49 +0100 |
| commit | 9e6e4962f2346a3fbd96ab3e6c331858ef6ec0d1 (patch) | |
| tree | 14b580295f71243def9c8df9c242c33f9f2c66a8 /src/boxes/avcc.rs | |
| download | fmp4-9e6e4962f2346a3fbd96ab3e6c331858ef6ec0d1.tar.gz fmp4-9e6e4962f2346a3fbd96ab3e6c331858ef6ec0d1.zip | |
Diffstat (limited to 'src/boxes/avcc.rs')
| -rw-r--r-- | src/boxes/avcc.rs | 76 |
1 files changed, 76 insertions, 0 deletions
diff --git a/src/boxes/avcc.rs b/src/boxes/avcc.rs new file mode 100644 index 0000000..e932543 --- /dev/null +++ b/src/boxes/avcc.rs @@ -0,0 +1,76 @@ +use byteorder::{BigEndian, WriteBytesExt}; + +use four_cc::FourCC; + +use bytes::{BufMut, BytesMut}; + +use crate::Mp4Box; +use crate::Mp4BoxError; + +use std::mem::size_of; + +pub struct AvcConfigurationBox { + pub config: AvcDecoderConfigurationRecord, +} + +impl Mp4Box for AvcConfigurationBox { + const NAME: FourCC = FourCC(*b"avcC"); + + fn content_size(&self) -> u64 { + self.config.size() + } + + fn write_box_contents(&self, writer: &mut BytesMut) -> Result<(), Mp4BoxError> { + self.config.write(writer)?; + + Ok(()) + } +} + +pub struct AvcDecoderConfigurationRecord { + pub profile_indication: u8, + pub profile_compatibility: u8, + pub level_indication: u8, + pub sequence_parameter_set: Vec<u8>, + pub picture_parameter_set: Vec<u8>, +} + +impl AvcDecoderConfigurationRecord { + fn size(&self) -> u64 { + size_of::<u8>() as u64 // configurationVersion + + size_of::<u8>() as u64 // AVCProfileIndication + + size_of::<u8>() as u64 // profile_compatibility + + size_of::<u8>() as u64 // AVCLevelIndication + + size_of::<u8>() as u64 // lengthSizeMinusOne + + size_of::<u8>() as u64 // numOfSequenceParameterSets + + size_of::<u16>() as u64 // sequenceParameterSetLength + + self.sequence_parameter_set.len() as u64 + + size_of::<u8>() as u64 // numOfPictureParameterSets + + size_of::<u16>() as u64 // pictureParameterSetLength + + self.picture_parameter_set.len() as u64 + } + + fn write(&self, writer: &mut BytesMut) -> Result<(), Mp4BoxError> { + let mut v = Vec::new(); + + v.push(1); + v.push(self.profile_indication); + v.push(self.profile_compatibility); + v.push(self.level_indication); + v.push(0b1111_1100 | 3); + + v.push(0b1110_0000 | 1); + v.write_u16::<BigEndian>(self.sequence_parameter_set.len() as u16)?; + v.extend(&self.sequence_parameter_set); + + v.push(1); + v.write_u16::<BigEndian>(self.picture_parameter_set.len() as u16)?; + v.extend(&self.picture_parameter_set); + + assert_eq!(self.size(), v.len() as u64); + + writer.put_slice(&v); + + Ok(()) + } +} |
