1
0
Fork 0
mirror of https://github.com/alfg/mp4-rust.git synced 2025-01-17 23:35:35 +00:00

hev1 box parser (#101)

Co-authored-by: Alfred Gutierrez <alfg@users.noreply.github.com>
This commit is contained in:
Andrey Tkachenko 2023-08-03 07:20:03 +03:00 committed by GitHub
parent 19e4eaa3c8
commit 35560e94f5
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 162 additions and 10 deletions

View file

@ -158,15 +158,33 @@ impl<W: Write> WriteBox<&mut W> for Hev1Box {
} }
} }
#[derive(Debug, Clone, PartialEq, Eq, Default, Serialize)] #[derive(Default, Debug, Clone, PartialEq, Eq, Serialize)]
pub struct HvcCBox { pub struct HvcCBox {
pub configuration_version: u8, pub configuration_version: u8,
pub general_profile_space: u8,
pub general_tier_flag: bool,
pub general_profile_idc: u8,
pub general_profile_compatibility_flags: u32,
pub general_constraint_indicator_flag: u64,
pub general_level_idc: u8,
pub min_spatial_segmentation_idc: u16,
pub parallelism_type: u8,
pub chroma_format_idc: u8,
pub bit_depth_luma_minus8: u8,
pub bit_depth_chroma_minus8: u8,
pub avg_frame_rate: u16,
pub constant_frame_rate: u8,
pub num_temporal_layers: u8,
pub temporal_id_nested: bool,
pub length_size_minus_one: u8,
pub arrays: Vec<HvcCArray>,
} }
impl HvcCBox { impl HvcCBox {
pub fn new() -> Self { pub fn new() -> Self {
Self { Self {
configuration_version: 1, configuration_version: 1,
..Default::default()
} }
} }
} }
@ -177,7 +195,13 @@ impl Mp4Box for HvcCBox {
} }
fn box_size(&self) -> u64 { fn box_size(&self) -> u64 {
HEADER_SIZE + 1 HEADER_SIZE
+ 23
+ self
.arrays
.iter()
.map(|a| 3 + a.nalus.iter().map(|x| 2 + x.data.len() as u64).sum::<u64>())
.sum::<u64>()
} }
fn to_json(&self) -> Result<String> { fn to_json(&self) -> Result<String> {
@ -185,21 +209,108 @@ impl Mp4Box for HvcCBox {
} }
fn summary(&self) -> Result<String> { fn summary(&self) -> Result<String> {
let s = format!("configuration_version={}", self.configuration_version); Ok(format!("configuration_version={} general_profile_space={} general_tier_flag={} general_profile_idc={} general_profile_compatibility_flags={} general_constraint_indicator_flag={} general_level_idc={} min_spatial_segmentation_idc={} parallelism_type={} chroma_format_idc={} bit_depth_luma_minus8={} bit_depth_chroma_minus8={} avg_frame_rate={} constant_frame_rate={} num_temporal_layers={} temporal_id_nested={} length_size_minus_one={}",
Ok(s) self.configuration_version,
self.general_profile_space,
self.general_tier_flag,
self.general_profile_idc,
self.general_profile_compatibility_flags,
self.general_constraint_indicator_flag,
self.general_level_idc,
self.min_spatial_segmentation_idc,
self.parallelism_type,
self.chroma_format_idc,
self.bit_depth_luma_minus8,
self.bit_depth_chroma_minus8,
self.avg_frame_rate,
self.constant_frame_rate,
self.num_temporal_layers,
self.temporal_id_nested,
self.length_size_minus_one
))
} }
} }
#[derive(Debug, Clone, PartialEq, Eq, Default, Serialize)]
pub struct HvcCArrayNalu {
pub size: u16,
pub data: Vec<u8>,
}
#[derive(Debug, Clone, PartialEq, Eq, Default, Serialize)]
pub struct HvcCArray {
pub completeness: bool,
pub nal_unit_type: u8,
pub nalus: Vec<HvcCArrayNalu>,
}
impl<R: Read + Seek> ReadBox<&mut R> for HvcCBox { impl<R: Read + Seek> ReadBox<&mut R> for HvcCBox {
fn read_box(reader: &mut R, size: u64) -> Result<Self> { fn read_box(reader: &mut R, _size: u64) -> Result<Self> {
let start = box_start(reader)?;
let configuration_version = reader.read_u8()?; let configuration_version = reader.read_u8()?;
let params = reader.read_u8()?;
let general_profile_space = params & 0b11000000 >> 6;
let general_tier_flag = (params & 0b00100000 >> 5) > 0;
let general_profile_idc = params & 0b00011111;
skip_bytes_to(reader, start + size)?; let general_profile_compatibility_flags = reader.read_u32::<BigEndian>()?;
let general_constraint_indicator_flag = reader.read_u48::<BigEndian>()?;
let general_level_idc = reader.read_u8()?;
let min_spatial_segmentation_idc = reader.read_u16::<BigEndian>()? & 0x0FFF;
let parallelism_type = reader.read_u8()? & 0b11;
let chroma_format_idc = reader.read_u8()? & 0b11;
let bit_depth_luma_minus8 = reader.read_u8()? & 0b111;
let bit_depth_chroma_minus8 = reader.read_u8()? & 0b111;
let avg_frame_rate = reader.read_u16::<BigEndian>()?;
let params = reader.read_u8()?;
let constant_frame_rate = params & 0b11000000 >> 6;
let num_temporal_layers = params & 0b00111000 >> 3;
let temporal_id_nested = (params & 0b00000100 >> 2) > 0;
let length_size_minus_one = params & 0b000011;
let num_of_arrays = reader.read_u8()?;
let mut arrays = Vec::with_capacity(num_of_arrays as _);
for _ in 0..num_of_arrays {
let params = reader.read_u8()?;
let num_nalus = reader.read_u16::<BigEndian>()?;
let mut nalus = Vec::with_capacity(num_nalus as usize);
for _ in 0..num_nalus {
let size = reader.read_u16::<BigEndian>()?;
let mut data = vec![0; size as usize];
reader.read_exact(&mut data)?;
nalus.push(HvcCArrayNalu { size, data })
}
arrays.push(HvcCArray {
completeness: (params & 0b10000000) > 0,
nal_unit_type: params & 0b111111,
nalus,
});
}
Ok(HvcCBox { Ok(HvcCBox {
configuration_version, configuration_version,
general_profile_space,
general_tier_flag,
general_profile_idc,
general_profile_compatibility_flags,
general_constraint_indicator_flag,
general_level_idc,
min_spatial_segmentation_idc,
parallelism_type,
chroma_format_idc,
bit_depth_luma_minus8,
bit_depth_chroma_minus8,
avg_frame_rate,
constant_frame_rate,
num_temporal_layers,
temporal_id_nested,
length_size_minus_one,
arrays,
}) })
} }
} }
@ -210,6 +321,40 @@ impl<W: Write> WriteBox<&mut W> for HvcCBox {
BoxHeader::new(self.box_type(), size).write(writer)?; BoxHeader::new(self.box_type(), size).write(writer)?;
writer.write_u8(self.configuration_version)?; writer.write_u8(self.configuration_version)?;
let general_profile_space = (self.general_profile_space & 0b11) << 6;
let general_tier_flag = u8::from(self.general_tier_flag) << 5;
let general_profile_idc = self.general_profile_idc & 0b11111;
writer.write_u8(general_profile_space | general_tier_flag | general_profile_idc)?;
writer.write_u32::<BigEndian>(self.general_profile_compatibility_flags)?;
writer.write_u48::<BigEndian>(self.general_constraint_indicator_flag)?;
writer.write_u8(self.general_level_idc)?;
writer.write_u16::<BigEndian>(self.min_spatial_segmentation_idc & 0x0FFF)?;
writer.write_u8(self.parallelism_type & 0b11)?;
writer.write_u8(self.chroma_format_idc & 0b11)?;
writer.write_u8(self.bit_depth_luma_minus8 & 0b111)?;
writer.write_u8(self.bit_depth_chroma_minus8 & 0b111)?;
writer.write_u16::<BigEndian>(self.avg_frame_rate)?;
let constant_frame_rate = (self.constant_frame_rate & 0b11) << 6;
let num_temporal_layers = (self.num_temporal_layers & 0b111) << 3;
let temporal_id_nested = u8::from(self.temporal_id_nested) << 2;
let length_size_minus_one = self.length_size_minus_one & 0b11;
writer.write_u8(
constant_frame_rate | num_temporal_layers | temporal_id_nested | length_size_minus_one,
)?;
writer.write_u8(self.arrays.len() as u8)?;
for arr in &self.arrays {
writer.write_u8((arr.nal_unit_type & 0b111111) | u8::from(arr.completeness) << 7)?;
writer.write_u16::<BigEndian>(arr.nalus.len() as _)?;
for nalu in &arr.nalus {
writer.write_u16::<BigEndian>(nalu.size)?;
writer.write_all(&nalu.data)?;
}
}
Ok(size) Ok(size)
} }
} }
@ -232,6 +377,7 @@ mod tests {
depth: 24, depth: 24,
hvcc: HvcCBox { hvcc: HvcCBox {
configuration_version: 1, configuration_version: 1,
..Default::default()
}, },
}; };
let mut buf = Vec::new(); let mut buf = Vec::new();

View file

@ -657,15 +657,21 @@ pub fn creation_time(creation_time: u64) -> u64 {
} }
} }
#[derive(Debug, Default, Clone, PartialEq, Eq, Serialize)] #[derive(Debug, Clone, PartialEq, Eq, Serialize)]
pub enum DataType { pub enum DataType {
#[default]
Binary = 0x000000, Binary = 0x000000,
Text = 0x000001, Text = 0x000001,
Image = 0x00000D, Image = 0x00000D,
TempoCpil = 0x000015, TempoCpil = 0x000015,
} }
#[allow(clippy::derivable_impls)]
impl std::default::Default for DataType {
fn default() -> Self {
DataType::Binary
}
}
impl TryFrom<u32> for DataType { impl TryFrom<u32> for DataType {
type Error = Error; type Error = Error;
fn try_from(value: u32) -> Result<DataType> { fn try_from(value: u32) -> Result<DataType> {