H.264 fixes, testing, packetizer
Fuzz testing found a few bugs in the new H.264 depacketizer. Also add a H.264 packetizer. Currently it's just used for testing.
This commit is contained in:
parent
af7e8a77fb
commit
1279dec0c3
6
fuzz/Cargo.lock
generated
6
fuzz/Cargo.lock
generated
@ -335,10 +335,12 @@ checksum = "0e4075386626662786ddb0ec9081e7c7eeb1ba31951f447ca780ef9f5d568189"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "h264-reader"
|
name = "h264-reader"
|
||||||
version = "0.4.0"
|
version = "0.5.0"
|
||||||
source = "git+https://github.com/scottlamb/h264-reader?branch=pr-sps-overflow#7d347e160ff73dcb8ae4680513db2618efe9c28d"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "c8d87669bdeca3d51902f1bf1f2c71c8f514a8f3011d9b81e63719b374091da1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bitreader",
|
"bitreader",
|
||||||
|
"log",
|
||||||
"memchr",
|
"memchr",
|
||||||
"rfc6381-codec",
|
"rfc6381-codec",
|
||||||
]
|
]
|
||||||
|
@ -25,3 +25,10 @@ name = "depacketize_h264"
|
|||||||
path = "fuzz_targets/depacketize_h264.rs"
|
path = "fuzz_targets/depacketize_h264.rs"
|
||||||
test = false
|
test = false
|
||||||
doc = false
|
doc = false
|
||||||
|
|
||||||
|
[[bin]]
|
||||||
|
name = "roundtrip_h264"
|
||||||
|
path = "fuzz_targets/roundtrip_h264.rs"
|
||||||
|
test = false
|
||||||
|
doc = false
|
||||||
|
|
||||||
|
55
fuzz/fuzz_targets/roundtrip_h264.rs
Normal file
55
fuzz/fuzz_targets/roundtrip_h264.rs
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
// Copyright (C) 2021 Scott Lamb <slamb@slamb.org>
|
||||||
|
// SPDX-License-Identifier: MIT OR Apache-2.0
|
||||||
|
|
||||||
|
//! Test a roundtrip through the H.264 packetizer and depacketizer with an arbitrary
|
||||||
|
//! input packet size and frame. Ensures the following:
|
||||||
|
//! * there are no crashes.
|
||||||
|
//! * the round trip produces an error or identical data.
|
||||||
|
|
||||||
|
#![no_main]
|
||||||
|
use bytes::Bytes;
|
||||||
|
use libfuzzer_sys::fuzz_target;
|
||||||
|
use std::num::NonZeroU32;
|
||||||
|
|
||||||
|
fuzz_target!(|data: &[u8]| {
|
||||||
|
if data.len() < 2 {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let max_payload_size = u16::from_be_bytes([data[0], data[1]]);
|
||||||
|
let mut p = match retina::codec::h264::Packetizer::new(max_payload_size, 0, 0) {
|
||||||
|
Ok(p) => p,
|
||||||
|
Err(_) => return,
|
||||||
|
};
|
||||||
|
let mut d = retina::codec::Depacketizer::new(
|
||||||
|
"video", "h264", 90_000, None,
|
||||||
|
Some("packetization-mode=1;sprop-parameter-sets=J01AHqkYGwe83gDUBAQG2wrXvfAQ,KN4JXGM4"),
|
||||||
|
).unwrap();
|
||||||
|
let timestamp = retina::Timestamp::new(0, NonZeroU32::new(90_000).unwrap(), 0).unwrap();
|
||||||
|
if p.push(timestamp, Bytes::copy_from_slice(&data[2..])).is_err() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let frame = loop {
|
||||||
|
match p.pull() {
|
||||||
|
Ok(Some(pkt)) => {
|
||||||
|
let mark = pkt.mark;
|
||||||
|
if d.push(pkt).is_err() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
match d.pull() {
|
||||||
|
Err(_) => return,
|
||||||
|
Ok(Some(retina::codec::CodecItem::VideoFrame(f))) => {
|
||||||
|
assert!(mark);
|
||||||
|
break f
|
||||||
|
},
|
||||||
|
Ok(Some(_)) => panic!(),
|
||||||
|
Ok(None) => assert!(!mark),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(None) => panic!("packetizer ran out of packets before depacketizer produced frame"),
|
||||||
|
Err(_) => return,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
assert_eq!(&data[2..], &frame.data()[..]);
|
||||||
|
assert!(matches!(d.pull(), Ok(None)));
|
||||||
|
assert!(matches!(p.pull(), Ok(None)));
|
||||||
|
});
|
@ -10,7 +10,7 @@ use failure::{bail, format_err, Error};
|
|||||||
use h264_reader::nal::{NalHeader, UnitType};
|
use h264_reader::nal::{NalHeader, UnitType};
|
||||||
use log::debug;
|
use log::debug;
|
||||||
|
|
||||||
use crate::client::rtp::Packet;
|
use crate::{Timestamp, client::rtp::Packet};
|
||||||
|
|
||||||
use super::VideoFrame;
|
use super::VideoFrame;
|
||||||
|
|
||||||
@ -134,16 +134,18 @@ impl Depacketizer {
|
|||||||
}
|
}
|
||||||
DepacketizerInputState::PreMark(mut access_unit) => {
|
DepacketizerInputState::PreMark(mut access_unit) => {
|
||||||
if pkt.loss > 0 {
|
if pkt.loss > 0 {
|
||||||
|
self.nals.clear();
|
||||||
|
self.pieces.clear();
|
||||||
if access_unit.timestamp.timestamp == pkt.timestamp.timestamp {
|
if access_unit.timestamp.timestamp == pkt.timestamp.timestamp {
|
||||||
// Loss within this access unit. Ignore until mark or new timestamp.
|
// Loss within this access unit. Ignore until mark or new timestamp.
|
||||||
self.nals.clear();
|
|
||||||
self.pieces.clear();
|
|
||||||
self.input_state = if pkt.mark {
|
self.input_state = if pkt.mark {
|
||||||
DepacketizerInputState::PostMark {
|
DepacketizerInputState::PostMark {
|
||||||
timestamp: pkt.timestamp,
|
timestamp: pkt.timestamp,
|
||||||
loss: pkt.loss,
|
loss: pkt.loss,
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
self.pieces.clear();
|
||||||
|
self.nals.clear();
|
||||||
DepacketizerInputState::Loss {
|
DepacketizerInputState::Loss {
|
||||||
timestamp: pkt.timestamp,
|
timestamp: pkt.timestamp,
|
||||||
pkts: pkt.loss,
|
pkts: pkt.loss,
|
||||||
@ -155,7 +157,7 @@ impl Depacketizer {
|
|||||||
// A prefix of the new one may have been lost; try parsing.
|
// A prefix of the new one may have been lost; try parsing.
|
||||||
AccessUnit::start(&pkt, 0)
|
AccessUnit::start(&pkt, 0)
|
||||||
} else if access_unit.timestamp.timestamp != pkt.timestamp.timestamp {
|
} else if access_unit.timestamp.timestamp != pkt.timestamp.timestamp {
|
||||||
if !access_unit.in_fu_a {
|
if access_unit.in_fu_a {
|
||||||
bail!("Timestamp changed from {} to {} in the middle of a fragmented NAL at seq={:04x} {:#?}", access_unit.timestamp, pkt.timestamp, seq, &pkt.rtsp_ctx);
|
bail!("Timestamp changed from {} to {} in the middle of a fragmented NAL at seq={:04x} {:#?}", access_unit.timestamp, pkt.timestamp, seq, &pkt.rtsp_ctx);
|
||||||
}
|
}
|
||||||
access_unit.end_ctx = pkt.rtsp_ctx;
|
access_unit.end_ctx = pkt.rtsp_ctx;
|
||||||
@ -225,9 +227,9 @@ impl Depacketizer {
|
|||||||
24 => {
|
24 => {
|
||||||
// STAP-A. https://tools.ietf.org/html/rfc6184#section-5.7.1
|
// STAP-A. https://tools.ietf.org/html/rfc6184#section-5.7.1
|
||||||
loop {
|
loop {
|
||||||
if data.remaining() < 2 {
|
if data.remaining() < 3 {
|
||||||
bail!(
|
bail!(
|
||||||
"STAP-A has {} remaining bytes while expecting 2-byte length",
|
"STAP-A has {} remaining bytes; expecting 2-byte length, non-empty NAL",
|
||||||
data.remaining()
|
data.remaining()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -276,7 +278,7 @@ impl Depacketizer {
|
|||||||
28 => {
|
28 => {
|
||||||
// FU-A. https://tools.ietf.org/html/rfc6184#section-5.8
|
// FU-A. https://tools.ietf.org/html/rfc6184#section-5.8
|
||||||
if data.len() < 2 {
|
if data.len() < 2 {
|
||||||
bail!("FU-A is too short at seq {:04x} {:#?}", seq, &pkt.rtsp_ctx);
|
bail!("FU-A len {} too short at seq {:04x} {:#?}", data.len(), seq, &pkt.rtsp_ctx);
|
||||||
}
|
}
|
||||||
let fu_header = data[0];
|
let fu_header = data[0];
|
||||||
let start = (fu_header & 0b10000000) != 0;
|
let start = (fu_header & 0b10000000) != 0;
|
||||||
@ -294,6 +296,9 @@ impl Depacketizer {
|
|||||||
&pkt.rtsp_ctx
|
&pkt.rtsp_ctx
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
if !end && pkt.mark {
|
||||||
|
bail!("FU-A pkt with MARK && !END at seq {:04x} {:#?}", seq, &pkt.rtsp_ctx);
|
||||||
|
}
|
||||||
let u32_len = u32::try_from(data.len()).expect("RTP packet len must be < u16::MAX");
|
let u32_len = u32::try_from(data.len()).expect("RTP packet len must be < u16::MAX");
|
||||||
match (start, access_unit.in_fu_a) {
|
match (start, access_unit.in_fu_a) {
|
||||||
(true, true) => bail!(
|
(true, true) => bail!(
|
||||||
@ -336,6 +341,8 @@ impl Depacketizer {
|
|||||||
}
|
}
|
||||||
(false, false) => {
|
(false, false) => {
|
||||||
if pkt.loss > 0 {
|
if pkt.loss > 0 {
|
||||||
|
self.pieces.clear();
|
||||||
|
self.nals.clear();
|
||||||
self.input_state = DepacketizerInputState::Loss {
|
self.input_state = DepacketizerInputState::Loss {
|
||||||
timestamp: pkt.timestamp,
|
timestamp: pkt.timestamp,
|
||||||
pkts: pkt.loss,
|
pkts: pkt.loss,
|
||||||
@ -655,14 +662,240 @@ fn to_bytes(hdr: NalHeader, len: u32, pieces: &[Bytes]) -> Bytes {
|
|||||||
out.into()
|
out.into()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// A simple packetizer, currently only for testing/benchmarking. Unstable.
|
||||||
|
///
|
||||||
|
/// Only uses plain NALs and FU-As, never STAP-A.
|
||||||
|
/// Expects data to be NALs separated by 4-byte prefixes.
|
||||||
|
#[doc(hidden)]
|
||||||
|
pub struct Packetizer {
|
||||||
|
max_payload_size: u16,
|
||||||
|
next_sequence_number: u16,
|
||||||
|
stream_id: usize,
|
||||||
|
state: PacketizerState,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Packetizer {
|
||||||
|
pub fn new(max_payload_size: u16, stream_id: usize, initial_sequence_number: u16) -> Result<Self, Error> {
|
||||||
|
if max_payload_size < 3 { // minimum size to make progress with FU-A packets.
|
||||||
|
bail!("max_payload_size must be > 3");
|
||||||
|
}
|
||||||
|
Ok(Self {
|
||||||
|
max_payload_size,
|
||||||
|
stream_id,
|
||||||
|
next_sequence_number: initial_sequence_number,
|
||||||
|
state: PacketizerState::Idle,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn push(&mut self, timestamp: Timestamp, data: Bytes) -> Result<(), Error> {
|
||||||
|
assert!(matches!(self.state, PacketizerState::Idle));
|
||||||
|
self.state = PacketizerState::HaveData {
|
||||||
|
timestamp,
|
||||||
|
data,
|
||||||
|
};
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn pull(&mut self) -> Result<Option<Packet>, Error> {
|
||||||
|
let max_payload_size = usize::from(self.max_payload_size);
|
||||||
|
match std::mem::replace(&mut self.state, PacketizerState::Idle) {
|
||||||
|
PacketizerState::Idle => return Ok(None),
|
||||||
|
PacketizerState::HaveData { timestamp, mut data } => {
|
||||||
|
if data.len() < 5 {
|
||||||
|
bail!("have only {} bytes; expected 4-byte length + non-empty NAL", data.len());
|
||||||
|
}
|
||||||
|
let len = data.get_u32();
|
||||||
|
let usize_len = usize::try_from(len).expect("u32 fits in usize");
|
||||||
|
if data.len() < usize_len || len == 0 {
|
||||||
|
bail!("bad length of {} bytes; expected [1, {}]", len, data.len());
|
||||||
|
}
|
||||||
|
let sequence_number = self.next_sequence_number;
|
||||||
|
self.next_sequence_number = self.next_sequence_number.wrapping_add(1);
|
||||||
|
let hdr = NalHeader::new(data[0]).map_err(|_| format_err!("F bit in NAL header"))?;
|
||||||
|
if matches!(hdr.nal_unit_type(), UnitType::Unspecified(_)) {
|
||||||
|
// This can clash with fragmentation/aggregation NAL types.
|
||||||
|
bail!("bad NAL header {:?}", hdr);
|
||||||
|
}
|
||||||
|
if usize_len > max_payload_size { // start a FU-A.
|
||||||
|
data.advance(1);
|
||||||
|
let mut payload = Vec::with_capacity(max_payload_size);
|
||||||
|
let fu_indicator = (hdr.nal_ref_idc() << 5) | 28;
|
||||||
|
let fu_header = 0b100_00000 | hdr.nal_unit_type().id(); // START bit set.
|
||||||
|
payload.extend_from_slice(&[fu_indicator, fu_header]);
|
||||||
|
payload.extend_from_slice(&data[..max_payload_size - 2]);
|
||||||
|
data.advance(max_payload_size - 2);
|
||||||
|
self.state = PacketizerState::InFragment {
|
||||||
|
timestamp,
|
||||||
|
hdr,
|
||||||
|
left: len + 1 - u32::from(self.max_payload_size),
|
||||||
|
data,
|
||||||
|
};
|
||||||
|
return Ok(Some(Packet {
|
||||||
|
rtsp_ctx: crate::Context::dummy(),
|
||||||
|
stream_id: self.stream_id,
|
||||||
|
timestamp,
|
||||||
|
sequence_number,
|
||||||
|
loss: 0,
|
||||||
|
mark: false,
|
||||||
|
payload: Bytes::from(payload),
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send a plain NAL packet. (TODO: consider using STAP-A.)
|
||||||
|
let mark;
|
||||||
|
if data.len() == usize_len {
|
||||||
|
mark = true;
|
||||||
|
} else {
|
||||||
|
self.state = PacketizerState::HaveData {
|
||||||
|
timestamp,
|
||||||
|
data: data.split_off(usize_len),
|
||||||
|
};
|
||||||
|
mark = false;
|
||||||
|
}
|
||||||
|
Ok(Some(Packet {
|
||||||
|
rtsp_ctx: crate::Context::dummy(),
|
||||||
|
stream_id: self.stream_id,
|
||||||
|
timestamp,
|
||||||
|
sequence_number,
|
||||||
|
loss: 0,
|
||||||
|
mark,
|
||||||
|
payload: data,
|
||||||
|
}))
|
||||||
|
},
|
||||||
|
PacketizerState::InFragment { timestamp, hdr, left, mut data } => {
|
||||||
|
let sequence_number = self.next_sequence_number;
|
||||||
|
self.next_sequence_number = self.next_sequence_number.wrapping_add(1);
|
||||||
|
let mut payload;
|
||||||
|
let mark;
|
||||||
|
if left > u32::from(self.max_payload_size) - 2 {
|
||||||
|
mark = false;
|
||||||
|
payload = Vec::with_capacity(max_payload_size);
|
||||||
|
let fu_indicator = (hdr.nal_ref_idc() << 5) | 28;
|
||||||
|
let fu_header = hdr.nal_unit_type().id(); // neither START nor END bits set.
|
||||||
|
payload.extend_from_slice(&[fu_indicator, fu_header]);
|
||||||
|
payload.extend_from_slice(&data[..max_payload_size - 2]);
|
||||||
|
data.advance(max_payload_size - 2);
|
||||||
|
self.state = PacketizerState::InFragment {
|
||||||
|
timestamp,
|
||||||
|
hdr,
|
||||||
|
left: left + 2 - u32::from(self.max_payload_size),
|
||||||
|
data,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
let usize_left = usize::try_from(left).expect("u32 fits in usize");
|
||||||
|
payload = Vec::with_capacity(usize_left + 2);
|
||||||
|
let fu_indicator = (hdr.nal_ref_idc() << 5) | 28;
|
||||||
|
let fu_header = 0b010_00000 | hdr.nal_unit_type().id(); // END bit set.
|
||||||
|
payload.extend_from_slice(&[fu_indicator, fu_header]);
|
||||||
|
payload.extend_from_slice(&data[..usize_left]);
|
||||||
|
if data.len() == usize_left {
|
||||||
|
mark = true;
|
||||||
|
self.state = PacketizerState::Idle;
|
||||||
|
} else {
|
||||||
|
mark = false;
|
||||||
|
data.advance(usize_left);
|
||||||
|
self.state = PacketizerState::HaveData {
|
||||||
|
timestamp,
|
||||||
|
data,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(Some(Packet {
|
||||||
|
rtsp_ctx: crate::Context::dummy(),
|
||||||
|
stream_id: self.stream_id,
|
||||||
|
timestamp,
|
||||||
|
sequence_number,
|
||||||
|
loss: 0,
|
||||||
|
mark,
|
||||||
|
payload: Bytes::from(payload),
|
||||||
|
}))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum PacketizerState {
|
||||||
|
Idle,
|
||||||
|
|
||||||
|
/// Have NALs to send; not in the middle of a fragmented packet.
|
||||||
|
HaveData {
|
||||||
|
timestamp: Timestamp,
|
||||||
|
|
||||||
|
/// Positioned before the length of a NAL.
|
||||||
|
data: Bytes,
|
||||||
|
},
|
||||||
|
InFragment {
|
||||||
|
timestamp: Timestamp,
|
||||||
|
hdr: NalHeader,
|
||||||
|
|
||||||
|
/// The number of non-header payload bytes to send in this NAL.
|
||||||
|
left: u32,
|
||||||
|
|
||||||
|
/// Positioned at the next non-header payload byte of this NAL.
|
||||||
|
data: Bytes,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
|
use bytes::Bytes;
|
||||||
use std::num::NonZeroU32;
|
use std::num::NonZeroU32;
|
||||||
|
|
||||||
use bytes::Bytes;
|
|
||||||
|
|
||||||
use crate::{client::rtp::Packet, codec::CodecItem};
|
use crate::{client::rtp::Packet, codec::CodecItem};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This test requires
|
||||||
|
* 1. a hacked version of the "mp4" crate to fix a couple bugs
|
||||||
|
* 2. a copy of a .mp4 or .mov file
|
||||||
|
* so it's disabled.
|
||||||
|
#[test]
|
||||||
|
fn roundtrip_using_mp4() {
|
||||||
|
use crate::Timestamp;
|
||||||
|
use pretty_hex::PrettyHex;
|
||||||
|
use std::convert::TryFrom;
|
||||||
|
let mut p = super::Packetizer::new(1400, 0, 0).unwrap();
|
||||||
|
let mut d = super::Depacketizer::new(
|
||||||
|
90_000,
|
||||||
|
Some("packetization-mode=1;sprop-parameter-sets=J01AHqkYGwe83gDUBAQG2wrXvfAQ,KN4JXGM4"))
|
||||||
|
.unwrap();
|
||||||
|
let mut f = mp4::read_mp4(std::fs::File::open("src/codec/testdata/big_buck_bunny_480p_h264.mov").unwrap()).unwrap();
|
||||||
|
let h264_track = f.tracks().iter().find_map(|t| {
|
||||||
|
if matches!(t.media_type(), Ok(mp4::MediaType::H264)) {
|
||||||
|
println!("sps: {:?}", t.sequence_parameter_set().unwrap().hex_dump());
|
||||||
|
println!("pps: {:?}", t.picture_parameter_set().unwrap().hex_dump());
|
||||||
|
Some(t.track_id())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}).unwrap();
|
||||||
|
let samples = f.sample_count(h264_track).unwrap();
|
||||||
|
for i in 1..=samples {
|
||||||
|
let sample = f.read_sample(h264_track, i).unwrap().unwrap();
|
||||||
|
//println!("packetizing {:#?}", sample.bytes.hex_dump());
|
||||||
|
println!("\n\npacketizing frame");
|
||||||
|
let mut frame = None;
|
||||||
|
p.push(Timestamp::new(i64::try_from(sample.start_time).unwrap(), NonZeroU32::new(90_000).unwrap(), 0).unwrap(), sample.bytes.clone()).unwrap();
|
||||||
|
while let Some(pkt) = p.pull().unwrap() {
|
||||||
|
assert!(frame.is_none());
|
||||||
|
d.push(pkt).unwrap();
|
||||||
|
assert!(frame.is_none());
|
||||||
|
loop {
|
||||||
|
if let Some(f) = d.pull().unwrap() {
|
||||||
|
assert!(frame.is_none());
|
||||||
|
frame = Some(match f {
|
||||||
|
CodecItem::VideoFrame(f) => f,
|
||||||
|
_ => panic!(),
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
assert_eq!(frame.unwrap().data(), &sample.bytes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn depacketize() {
|
fn depacketize() {
|
||||||
let mut d = super::Depacketizer::new(90_000, Some("packetization-mode=1;profile-level-id=64001E;sprop-parameter-sets=Z2QAHqwsaoLA9puCgIKgAAADACAAAAMD0IAA,aO4xshsA")).unwrap();
|
let mut d = super::Depacketizer::new(90_000, Some("packetization-mode=1;profile-level-id=64001E;sprop-parameter-sets=Z2QAHqwsaoLA9puCgIKgAAADACAAAAMD0IAA,aO4xshsA")).unwrap();
|
||||||
@ -787,8 +1020,10 @@ mod tests {
|
|||||||
assert_eq!(p.pixel_dimensions(), (640, 480));
|
assert_eq!(p.pixel_dimensions(), (640, 480));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Tests parsing parameters from GW Security camera, which erroneously puts
|
||||||
|
/// an Annex B NAL separator at the end of each of the `sprop-parameter-sets` NALs.
|
||||||
#[test]
|
#[test]
|
||||||
fn gw_security() {
|
fn gw_security_params() {
|
||||||
let params = super::InternalParameters::parse_format_specific_params(
|
let params = super::InternalParameters::parse_format_specific_params(
|
||||||
"packetization-mode=1;\
|
"packetization-mode=1;\
|
||||||
profile-level-id=5046302;\
|
profile-level-id=5046302;\
|
||||||
|
@ -16,7 +16,10 @@ use pretty_hex::PrettyHex;
|
|||||||
|
|
||||||
pub(crate) mod aac;
|
pub(crate) mod aac;
|
||||||
pub(crate) mod g723;
|
pub(crate) mod g723;
|
||||||
pub(crate) mod h264;
|
|
||||||
|
#[doc(hidden)]
|
||||||
|
pub mod h264;
|
||||||
|
|
||||||
pub(crate) mod onvif;
|
pub(crate) mod onvif;
|
||||||
pub(crate) mod simple_audio;
|
pub(crate) mod simple_audio;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user