flowly_mp4/mp4box/
vpcc.rs1use crate::mp4box::*;
2use crate::Mp4Box;
3use serde::Serialize;
4
5#[derive(Debug, Clone, PartialEq, Eq, Default, Serialize)]
6pub struct VpccBox {
7 pub version: u8,
8 pub flags: u32,
9 pub profile: u8,
10 pub level: u8,
11 pub bit_depth: u8,
12 pub chroma_subsampling: u8,
13 pub video_full_range_flag: bool,
14 pub color_primaries: u8,
15 pub transfer_characteristics: u8,
16 pub matrix_coefficients: u8,
17 pub codec_initialization_data_size: u16,
18}
19
20impl VpccBox {
21 pub const DEFAULT_VERSION: u8 = 1;
22 pub const DEFAULT_BIT_DEPTH: u8 = 8;
23}
24
25impl Mp4Box for VpccBox {
26 const TYPE: BoxType = BoxType::VpccBox;
27
28 fn box_size(&self) -> u64 {
29 HEADER_SIZE + HEADER_EXT_SIZE + 8
30 }
31
32 fn to_json(&self) -> Result<String, Error> {
33 Ok(serde_json::to_string(&self).unwrap())
34 }
35
36 fn summary(&self) -> Result<String, Error> {
37 Ok(format!("{self:?}"))
38 }
39}
40
41impl BlockReader for VpccBox {
42 fn read_block<'a>(reader: &mut impl Reader<'a>) -> Result<Self, Error> {
43 let (version, flags) = read_box_header_ext(reader);
44
45 let profile = reader.get_u8();
46 let level = reader.get_u8();
47 let (bit_depth, chroma_subsampling, video_full_range_flag) = {
48 let b = reader.get_u8();
49 (b >> 4, b << 4 >> 5, b & 0x01 == 1)
50 };
51
52 let transfer_characteristics = reader.get_u8();
53 let matrix_coefficients = reader.get_u8();
54 let codec_initialization_data_size = reader.get_u16();
55
56 Ok(Self {
57 version,
58 flags,
59 profile,
60 level,
61 bit_depth,
62 chroma_subsampling,
63 video_full_range_flag,
64 color_primaries: 0,
65 transfer_characteristics,
66 matrix_coefficients,
67 codec_initialization_data_size,
68 })
69 }
70
71 fn size_hint() -> usize {
72 11
73 }
74}
75
76impl<W: Write> WriteBox<&mut W> for VpccBox {
77 fn write_box(&self, writer: &mut W) -> Result<u64, Error> {
78 let size = self.box_size();
79 BoxHeader::new(Self::TYPE, size).write(writer)?;
80
81 write_box_header_ext(writer, self.version, self.flags)?;
82
83 writer.write_u8(self.profile)?;
84 writer.write_u8(self.level)?;
85 writer.write_u8(
86 (self.bit_depth << 4)
87 | (self.chroma_subsampling << 1)
88 | (self.video_full_range_flag as u8),
89 )?;
90 writer.write_u8(self.color_primaries)?;
91 writer.write_u8(self.transfer_characteristics)?;
92 writer.write_u8(self.matrix_coefficients)?;
93 writer.write_u16::<BigEndian>(self.codec_initialization_data_size)?;
94
95 Ok(size)
96 }
97}
98
99#[cfg(test)]
100mod tests {
101 use super::*;
102 use crate::mp4box::BoxHeader;
103
104 #[tokio::test]
105 async fn test_vpcc() {
106 let src_box = VpccBox {
107 version: VpccBox::DEFAULT_VERSION,
108 flags: 0,
109 profile: 0,
110 level: 0x1F,
111 bit_depth: VpccBox::DEFAULT_BIT_DEPTH,
112 chroma_subsampling: 0,
113 video_full_range_flag: false,
114 color_primaries: 0,
115 transfer_characteristics: 0,
116 matrix_coefficients: 0,
117 codec_initialization_data_size: 0,
118 };
119 let mut buf = Vec::new();
120 src_box.write_box(&mut buf).unwrap();
121 assert_eq!(buf.len(), src_box.box_size() as usize);
122
123 let mut reader = buf.as_slice();
124 let header = BoxHeader::read(&mut reader, &mut 0).await.unwrap().unwrap();
125 assert_eq!(header.kind, BoxType::VpccBox);
126 assert_eq!(src_box.box_size(), header.size);
127
128 let dst_box = VpccBox::read_block(&mut reader).unwrap();
129 assert_eq!(src_box, dst_box);
130 }
131}