mirror of
https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs.git
synced 2024-12-23 02:26:35 +00:00
jitterbuffer: handle flush-start/stop
Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs/-/merge_requests/1426>
This commit is contained in:
parent
66306e32f2
commit
48e7a2ed06
2 changed files with 108 additions and 2 deletions
|
@ -167,7 +167,9 @@ impl futures::stream::Stream for JitterBufferStream {
|
|||
let mut lowest_wait = None;
|
||||
|
||||
let mut jitterbuffer_store = self.store.lock().unwrap();
|
||||
match jitterbuffer_store.jitterbuffer.poll(now) {
|
||||
let ret = jitterbuffer_store.jitterbuffer.poll(now);
|
||||
gst::trace!(CAT, "jitterbuffer poll ret: {ret:?}");
|
||||
match ret {
|
||||
jitterbuffer::PollResult::Flushing => {
|
||||
return Poll::Ready(None);
|
||||
}
|
||||
|
@ -176,6 +178,7 @@ impl futures::stream::Stream for JitterBufferStream {
|
|||
.store
|
||||
.remove(&id)
|
||||
.unwrap_or_else(|| panic!("Buffer with id {id} not in store!"));
|
||||
cx.waker().wake_by_ref();
|
||||
}
|
||||
jitterbuffer::PollResult::Forward { id, discont } => {
|
||||
let mut item = jitterbuffer_store
|
||||
|
@ -1397,6 +1400,9 @@ impl RtpBin2 {
|
|||
jitterbuffer_store
|
||||
.store
|
||||
.insert(id, JitterBufferItem::Event(event.clone()));
|
||||
if let Some(waker) = jitterbuffer_store.waker.take() {
|
||||
waker.wake();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1498,10 +1504,48 @@ impl RtpBin2 {
|
|||
waker.wake();
|
||||
}
|
||||
}
|
||||
drop(state);
|
||||
// FIXME: may need to delay sending eos under some circumstances
|
||||
self.rtp_recv_sink_queue_serialized_event(id, event);
|
||||
true
|
||||
}
|
||||
// TODO: need to handle FlushStart/FlushStop through the jitterbuffer queue
|
||||
gst::EventView::FlushStart(_fs) => {
|
||||
let state = self.state.lock().unwrap();
|
||||
let mut pause_tasks = vec![];
|
||||
if let Some(session) = state.session_by_id(id) {
|
||||
let session = session.inner.lock().unwrap();
|
||||
for recv_pad in session.rtp_recv_srcpads.iter() {
|
||||
let mut store = recv_pad.jitter_buffer_store.lock().unwrap();
|
||||
store.jitterbuffer.set_flushing(true);
|
||||
if let Some(waker) = store.waker.take() {
|
||||
waker.wake();
|
||||
}
|
||||
pause_tasks.push(recv_pad.pad.clone());
|
||||
}
|
||||
}
|
||||
drop(state);
|
||||
for pad in pause_tasks {
|
||||
let _ = pad.pause_task();
|
||||
}
|
||||
gst::Pad::event_default(pad, Some(&*self.obj()), event)
|
||||
}
|
||||
gst::EventView::FlushStop(_fs) => {
|
||||
let state = self.state.lock().unwrap();
|
||||
if let Some(session) = state.session_by_id(id) {
|
||||
let mut session = session.inner.lock().unwrap();
|
||||
let pads = session
|
||||
.rtp_recv_srcpads
|
||||
.iter()
|
||||
.map(|r| r.pad.clone())
|
||||
.collect::<Vec<_>>();
|
||||
for pad in pads {
|
||||
// Will reset flushing to false and ensure task is woken up
|
||||
let _ = session.start_rtp_recv_task(&pad);
|
||||
}
|
||||
}
|
||||
drop(state);
|
||||
self.rtp_recv_sink_queue_serialized_event(id, event)
|
||||
}
|
||||
_ => {
|
||||
if event.is_serialized() {
|
||||
self.rtp_recv_sink_queue_serialized_event(id, event)
|
||||
|
|
|
@ -177,3 +177,65 @@ fn test_receive() {
|
|||
TEST_SSRC as i32
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_receive_flush() {
|
||||
init();
|
||||
|
||||
let h = Arc::new(Mutex::new(Harness::with_padnames(
|
||||
"rtpbin2",
|
||||
Some("rtp_recv_sink_0"),
|
||||
None,
|
||||
)));
|
||||
let weak_h = Arc::downgrade(&h);
|
||||
let mut inner = h.lock().unwrap();
|
||||
inner
|
||||
.element()
|
||||
.unwrap()
|
||||
.connect_pad_added(move |_elem, pad| {
|
||||
weak_h
|
||||
.upgrade()
|
||||
.unwrap()
|
||||
.lock()
|
||||
.unwrap()
|
||||
.add_element_src_pad(pad)
|
||||
});
|
||||
inner.play();
|
||||
|
||||
let caps = Caps::builder("application/x-rtp")
|
||||
.field("media", "audio")
|
||||
.field("payload", TEST_PT as i32)
|
||||
.field("clock-rate", TEST_CLOCK_RATE as i32)
|
||||
.field("encoding-name", "custom-test")
|
||||
.build();
|
||||
inner.set_src_caps(caps);
|
||||
|
||||
// Cannot push with harness lock as the 'pad-added' handler needs to add the newly created pad to
|
||||
// the harness and needs to also take the harness lock. Workaround by pushing from the
|
||||
// internal harness pad directly.
|
||||
let push_pad = inner
|
||||
.element()
|
||||
.unwrap()
|
||||
.static_pad("rtp_recv_sink_0")
|
||||
.unwrap()
|
||||
.peer()
|
||||
.unwrap();
|
||||
drop(inner);
|
||||
push_pad.push(generate_rtp_buffer(500, 20, 9)).unwrap();
|
||||
push_pad.push(generate_rtp_buffer(501, 30, 11)).unwrap();
|
||||
let mut inner = h.lock().unwrap();
|
||||
let seqnum = gst::Seqnum::next();
|
||||
inner.push_event(gst::event::FlushStart::builder().seqnum(seqnum).build());
|
||||
inner.push_event(gst::event::FlushStop::builder(false).seqnum(seqnum).build());
|
||||
|
||||
let event = inner.pull_event().unwrap();
|
||||
let gst::EventView::FlushStart(fs) = event.view() else {
|
||||
unreachable!();
|
||||
};
|
||||
assert_eq!(fs.seqnum(), seqnum);
|
||||
let event = inner.pull_event().unwrap();
|
||||
let gst::EventView::FlushStop(fs) = event.view() else {
|
||||
unreachable!();
|
||||
};
|
||||
assert_eq!(fs.seqnum(), seqnum);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue