scuffle_h265/sps/
sps_scc_extension.rs1use std::io;
2
3use scuffle_bytes_util::{BitReader, range_check};
4use scuffle_expgolomb::BitReaderExpGolombExt;
5
6#[derive(Debug, Clone, PartialEq)]
13pub struct SpsSccExtension {
14 pub sps_curr_pic_ref_enabled_flag: bool,
20 pub palette_mode: Option<SpsSccExtensionPaletteMode>,
22 pub motion_vector_resolution_control_idc: u8,
27 pub intra_boundary_filtering_disabled_flag: bool,
32}
33
34impl SpsSccExtension {
35 pub(crate) fn parse<R: io::Read>(
36 bit_reader: &mut BitReader<R>,
37 chroma_format_idc: u8,
38 bit_depth_y: u8,
39 bit_depth_c: u8,
40 ) -> io::Result<Self> {
41 let sps_curr_pic_ref_enabled_flag = bit_reader.read_bit()?;
42
43 let mut palette_mode = None;
44 let palette_mode_enabled_flag = bit_reader.read_bit()?;
45 if palette_mode_enabled_flag {
46 let palette_max_size = bit_reader.read_exp_golomb()?;
47 let delta_palette_max_predictor_size = bit_reader.read_exp_golomb()?;
48
49 if palette_max_size == 0 && delta_palette_max_predictor_size != 0 {
50 return Err(io::Error::new(
51 io::ErrorKind::InvalidData,
52 "delta_palette_max_predictor_size must be 0 when palette_max_size is 0",
53 ));
54 }
55
56 let sps_palette_predictor_initializers_present_flag = bit_reader.read_bit()?;
57 if palette_max_size == 0 && !sps_palette_predictor_initializers_present_flag {
58 return Err(io::Error::new(
59 io::ErrorKind::InvalidData,
60 "sps_palette_predictor_initializers_present_flag must be 0 when palette_max_size is 0",
61 ));
62 }
63
64 let mut sps_palette_predictor_initializers = None;
65 if sps_palette_predictor_initializers_present_flag {
66 let sps_num_palette_predictor_initializers_minus1 = bit_reader.read_exp_golomb()?;
67
68 if sps_num_palette_predictor_initializers_minus1 >= palette_max_size {
69 return Err(io::Error::new(
70 io::ErrorKind::InvalidData,
71 "sps_num_palette_predictor_initializers_minus1 + 1 must be less than or equal to palette_max_size",
72 ));
73 }
74
75 let num_comps = if chroma_format_idc == 0 { 1 } else { 3 };
76
77 let mut initializers = vec![vec![0; sps_num_palette_predictor_initializers_minus1 as usize + 1]; num_comps];
78 for (comp, initializer) in initializers.iter_mut().enumerate().take(num_comps) {
79 for sps_palette_predictor_initializer in initializer.iter_mut() {
80 let bit_depth = if comp == 0 { bit_depth_y } else { bit_depth_c };
81 *sps_palette_predictor_initializer = bit_reader.read_bits(bit_depth)?;
82 }
83 }
84
85 sps_palette_predictor_initializers = Some(initializers);
86 }
87
88 palette_mode = Some(SpsSccExtensionPaletteMode {
89 palette_max_size,
90 delta_palette_max_predictor_size,
91 sps_palette_predictor_initializers,
92 });
93 }
94
95 let motion_vector_resolution_control_idc = bit_reader.read_bits(2)? as u8;
96 range_check!(motion_vector_resolution_control_idc, 0, 2)?; let intra_boundary_filtering_disabled_flag = bit_reader.read_bit()?;
99
100 Ok(Self {
101 sps_curr_pic_ref_enabled_flag,
102 palette_mode,
103 motion_vector_resolution_control_idc,
104 intra_boundary_filtering_disabled_flag,
105 })
106 }
107}
108
109#[derive(Debug, Clone, PartialEq)]
111pub struct SpsSccExtensionPaletteMode {
112 pub palette_max_size: u64,
114 pub delta_palette_max_predictor_size: u64,
118 pub sps_palette_predictor_initializers: Option<Vec<Vec<u64>>>,
130}
131
132impl SpsSccExtensionPaletteMode {
133 pub fn palette_max_predictor_size(&self) -> u64 {
137 self.palette_max_size + self.delta_palette_max_predictor_size
138 }
139}
140
141#[cfg(test)]
142#[cfg_attr(all(test, coverage_nightly), coverage(off))]
143mod tests {
144 use byteorder::WriteBytesExt;
145 use scuffle_bytes_util::BitWriter;
146 use scuffle_expgolomb::BitWriterExpGolombExt;
147
148 #[test]
149 fn test_parse() {
150 let mut data = Vec::new();
151 let mut bit_writer = BitWriter::new(&mut data);
152
153 bit_writer.write_bit(true).unwrap(); bit_writer.write_bit(true).unwrap(); bit_writer.write_exp_golomb(5).unwrap(); bit_writer.write_exp_golomb(2).unwrap(); bit_writer.write_bit(true).unwrap(); bit_writer.write_exp_golomb(1).unwrap(); bit_writer.write_u8(1).unwrap(); bit_writer.write_u8(2).unwrap(); bit_writer.write_u8(3).unwrap(); bit_writer.write_u8(4).unwrap(); bit_writer.write_u8(5).unwrap(); bit_writer.write_u8(6).unwrap(); bit_writer.write_bits(0, 2).unwrap(); bit_writer.write_bit(false).unwrap(); bit_writer.write_bits(0, 8).unwrap(); let scc_extension = super::SpsSccExtension::parse(
175 &mut scuffle_bytes_util::BitReader::new(&data[..]),
176 1, 8, 8, )
180 .unwrap();
181
182 assert!(scc_extension.sps_curr_pic_ref_enabled_flag);
183
184 assert!(scc_extension.palette_mode.is_some());
185 let palette_mode = scc_extension.palette_mode.unwrap();
186 assert_eq!(palette_mode.palette_max_size, 5);
187 assert_eq!(palette_mode.delta_palette_max_predictor_size, 2);
188 assert_eq!(palette_mode.palette_max_predictor_size(), 7);
189
190 assert!(palette_mode.sps_palette_predictor_initializers.is_some());
191 let initializers = palette_mode.sps_palette_predictor_initializers.unwrap();
192 assert_eq!(initializers.len(), 3);
193 assert_eq!(initializers[0].len(), 2);
194 assert_eq!(initializers[0][0], 1);
195 assert_eq!(initializers[0][1], 2);
196 assert_eq!(initializers[1].len(), 2);
197 assert_eq!(initializers[1][0], 3);
198 assert_eq!(initializers[1][1], 4);
199 assert_eq!(initializers[2].len(), 2);
200 assert_eq!(initializers[2][0], 5);
201 assert_eq!(initializers[2][1], 6);
202
203 assert_eq!(scc_extension.motion_vector_resolution_control_idc, 0);
204 assert!(!scc_extension.intra_boundary_filtering_disabled_flag);
205 }
206}