rtp: ac3: add some unit tests

Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs/-/merge_requests/1586>
This commit is contained in:
Tim-Philipp Müller 2024-05-27 12:51:03 +03:00 committed by GStreamer Marge Bot
parent 2b68920f82
commit ab2f5e3d8d
5 changed files with 383 additions and 0 deletions

View file

@ -4,3 +4,7 @@ pub mod depay;
pub mod pay; pub mod pay;
mod ac3_audio_utils; mod ac3_audio_utils;
#[allow(clippy::module_inception)]
#[cfg(test)]
mod tests;

1
net/rtp/src/ac3/tests/.gitattributes vendored Normal file
View file

@ -0,0 +1 @@
*.ac3 binary

Binary file not shown.

View file

@ -0,0 +1,3 @@
// SPDX-License-Identifier: MPL-2.0
mod tests;

View file

@ -0,0 +1,375 @@
// GStreamer RTP AC-3 Payloader / Depayloader - unit tests
//
// Copyright (C) 2024 Tim-Philipp Müller <tim centricular com>
//
// This Source Code Form is subject to the terms of the Mozilla Public License, v2.0.
// If a copy of the MPL was not distributed with this file, You can obtain one at
// <https://mozilla.org/MPL/2.0/>.
//
// SPDX-License-Identifier: MPL-2.0
use crate::ac3::ac3_audio_utils::*;
use crate::tests::{
run_test_pipeline, run_test_pipeline_full, ExpectedBuffer, ExpectedPacket, Liveness, Source,
};
fn init() {
use std::sync::Once;
static INIT: Once = Once::new();
INIT.call_once(|| {
gst::init().unwrap();
crate::plugin_register_static().expect("rtpac3 test");
});
}
// gst-launch-1.0 audiotestsrc samplesperbuffer=1536 wave=ticks num-buffers=3
// ! audio/x-raw,rate=48000 ! avenc_ac3 ! filesink
const AC3_DATA: &[u8] = include_bytes!("audiotestsrc-1ch-48kHz.ac3").as_slice();
pub(crate) fn parse_ac3_frames(data: &[u8]) -> Vec<&[u8]> {
let mut frames = vec![];
let mut data = &data[0..];
while !data.is_empty() {
let hdr = peek_frame_header(data).unwrap();
let size = hdr.frame_len;
// eprintln!("AC-3 frame {} of size {size}", frames.len());
frames.push(&data[0..][..size]);
data = &data[size..];
}
frames
}
fn make_buffer(
data: &'static [u8],
pts: gst::ClockTime,
duration: gst::ClockTime,
flags: gst::BufferFlags,
) -> gst::Buffer {
let mut buf = gst::Buffer::from_slice(data);
let buf_ref = buf.get_mut().unwrap();
buf_ref.set_pts(pts);
buf_ref.set_duration(duration);
buf_ref.set_flags(flags);
buf
}
// test_ac3_pay_depay
//
// Check basic payloading/depayloading, in live and non-live mode
//
#[test]
fn test_ac3_pay_depay() {
init();
fn run_ac3_pay_depay_test(liveness: Liveness) {
let frames = parse_ac3_frames(AC3_DATA);
let input_caps = gst::Caps::builder("audio/x-ac3")
.field("rate", 48000i32)
.field("channels", 1i32)
.field("framed", true)
.field("alignment", "frame")
.build();
let mut input_buffers = vec![];
for (i, frame) in frames.iter().enumerate() {
input_buffers.push(make_buffer(
frame,
gst::ClockTime::from_mseconds(32 * i as u64),
gst::ClockTime::from_mseconds(32),
if i == 0 {
gst::BufferFlags::DISCONT
} else {
gst::BufferFlags::empty()
},
));
}
// If upstream is non-live, the payloader should collect and pack as many AC-3 frames
// into each RTP packet as it can. With mtu=1400 and frame size of 384 bytes,
// that's 3 frames/packet.
let mut expected_pay = vec![];
if liveness == Liveness::NonLive {
expected_pay.push(vec![ExpectedPacket::builder()
.pts(gst::ClockTime::ZERO)
.flags(gst::BufferFlags::DISCONT | gst::BufferFlags::MARKER)
.pt(96)
.rtp_time(0)
.marker_bit(true)
.build()]);
expected_pay.push(vec![ExpectedPacket::builder()
.pts(gst::ClockTime::from_mseconds(96))
.flags(gst::BufferFlags::MARKER)
.pt(96)
.rtp_time(3 * 1536)
.marker_bit(true)
.build()]);
} else {
for (i, _frame) in frames.iter().enumerate() {
let discont_flag = if i == 0 {
gst::BufferFlags::DISCONT
} else {
gst::BufferFlags::empty()
};
expected_pay.push(vec![ExpectedPacket::builder()
.pts(gst::ClockTime::from_mseconds(32 * i as u64))
.flags(discont_flag | gst::BufferFlags::MARKER)
.pt(96)
.rtp_time(1536 * i as u32)
.marker_bit(true)
.build()]);
}
}
let expected_depay = vec![
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::ZERO)
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::DISCONT)
.build()],
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::from_mseconds(32))
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::empty())
.build()],
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::from_mseconds(64))
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::empty())
.build()],
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::from_mseconds(96))
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::empty())
.build()],
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::from_mseconds(128))
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::empty())
.build()],
];
run_test_pipeline_full(
Source::Buffers(input_caps, input_buffers),
"rtpac3pay2",
"rtpac3depay2",
expected_pay,
expected_depay,
None,
liveness,
);
}
println!("Testing non-live mode (should aggregate frames)..");
run_ac3_pay_depay_test(Liveness::NonLive);
println!("Testing live mode (should send out frames immediately)..");
run_ac3_pay_depay_test(Liveness::Live(20_000_000));
}
// test_ac3_pay_depay_fragmented
//
// Check basic payloading/depayloading with small MTU
//
#[test]
fn test_ac3_pay_depay_fragmented() {
init();
let frames = parse_ac3_frames(AC3_DATA);
let input_caps = gst::Caps::builder("audio/x-ac3")
.field("rate", 48000i32)
.field("channels", 1i32)
.field("framed", true)
.field("alignment", "frame")
.build();
let mut input_buffers = vec![];
let mut expected_pay = vec![];
for (i, frame) in frames.iter().enumerate() {
let discont_flag = if i == 0 {
gst::BufferFlags::DISCONT
} else {
gst::BufferFlags::empty()
};
input_buffers.push(make_buffer(
frame,
gst::ClockTime::from_mseconds(32 * i as u64),
gst::ClockTime::from_mseconds(32),
discont_flag,
));
// Each 384 byte AC-3 frame will be split into 2 RTP packets with mtu=250
expected_pay.push(vec![
ExpectedPacket::builder()
.pts(gst::ClockTime::from_mseconds(32 * i as u64))
.flags(discont_flag)
.pt(96)
.rtp_time(1536 * i as u32)
.marker_bit(false)
.build(),
ExpectedPacket::builder()
.pts(gst::ClockTime::from_mseconds(32 * i as u64))
.flags(gst::BufferFlags::MARKER)
.pt(96)
.rtp_time(1536 * i as u32)
.marker_bit(true)
.build(),
]);
}
let expected_depay = vec![
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::ZERO)
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::DISCONT)
.build()],
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::from_mseconds(32))
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::empty())
.build()],
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::from_mseconds(64))
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::empty())
.build()],
vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::from_mseconds(96))
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(gst::BufferFlags::empty())
.build()],
vec![ExpectedBuffer::builder()
.duration(gst::ClockTime::from_mseconds(32))
.pts(gst::ClockTime::from_mseconds(128))
.size(384)
.flags(gst::BufferFlags::empty())
.build()],
];
run_test_pipeline(
Source::Buffers(input_caps, input_buffers),
"rtpac3pay2 mtu=250",
"rtpac3depay2",
expected_pay,
expected_depay,
);
}
// test_ac3_pay_depay_fragmented_with_packet_loss
//
// Check basic payloading/depayloading with small MTU and some packet loss
//
#[test]
fn test_ac3_pay_depay_fragmented_with_packet_loss() {
init();
fn run_ac3_pay_depay_fragmented_with_packet_loss_with_drop_mask(
drop_mask: u32,
initial_seqnum: Option<u16>,
) {
let frames = parse_ac3_frames(AC3_DATA);
let input_caps = gst::Caps::builder("audio/x-ac3")
.field("rate", 48000i32)
.field("channels", 1i32)
.field("framed", true)
.field("alignment", "frame")
.build();
let mut input_buffers = vec![];
let mut expected_pay = vec![];
let mut expected_depay = vec![];
for (i, frame) in frames.iter().enumerate() {
let packet_mask = (drop_mask >> (2 * i)) & 0b11;
let discont_flag = if i == 0 {
gst::BufferFlags::DISCONT
} else {
gst::BufferFlags::empty()
};
input_buffers.push(make_buffer(
frame,
gst::ClockTime::from_mseconds(32 * i as u64),
gst::ClockTime::from_mseconds(32),
discont_flag,
));
// Each 384 byte AC-3 frame will be split into 2 RTP packets with mtu=250
expected_pay.push(vec![
ExpectedPacket::builder()
.pts(gst::ClockTime::from_mseconds(32 * i as u64))
.flags(discont_flag)
.pt(96)
.rtp_time(1536 * i as u32)
.marker_bit(false)
.drop((packet_mask & 0b0001) == 0b0001)
.build(),
ExpectedPacket::builder()
.pts(gst::ClockTime::from_mseconds(32 * i as u64))
.flags(gst::BufferFlags::MARKER)
.pt(96)
.rtp_time(1536 * i as u32)
.marker_bit(true)
.drop((packet_mask & 0b0010) == 0b0010)
.build(),
]);
// Expect discont on first packet and if previous packet got dropped
let expected_flags = if i == 0 || (drop_mask >> (2 * (i - 1))) & 0b11 != 0b0000 {
gst::BufferFlags::DISCONT
} else {
gst::BufferFlags::empty()
};
// If any of the fragments got dropped, we can't reconstruct the original payload
if packet_mask == 0b0000 {
expected_depay.push(vec![ExpectedBuffer::builder()
.pts(gst::ClockTime::from_mseconds(32 * i as u64))
.duration(gst::ClockTime::from_mseconds(32))
.size(384)
.flags(expected_flags)
.build()]);
}
}
let payloader = if let Some(seqnum_offset) = initial_seqnum {
format!("rtpac3pay2 mtu=250 seqnum-offset={seqnum_offset}")
} else {
"rtpac3pay2 mtu=250".to_string()
};
run_test_pipeline(
Source::Buffers(input_caps, input_buffers),
&payloader,
"rtpac3depay2",
expected_pay,
expected_depay,
);
}
for drop_mask in 0..(1 << (6 * 2)) {
run_ac3_pay_depay_fragmented_with_packet_loss_with_drop_mask(drop_mask, None);
if drop_mask % 3 == 3 {
run_ac3_pay_depay_fragmented_with_packet_loss_with_drop_mask(drop_mask, Some(65533));
}
}
}