fallbackswitch: properly handle GAP events

Handle GAP events from the active pad as activity and forward
downstream.
This commit is contained in:
Guillaume Desmottes 2022-03-02 15:27:07 +01:00
parent 7b109785be
commit 8389bff7d8

View file

@ -997,7 +997,57 @@ impl AggregatorImpl for FallbackSwitch {
match event.view() { match event.view() {
EventView::Gap(gap) => { EventView::Gap(gap) => {
if gap.gap_flags().contains(gst::GapFlags::DATA) { if gap.gap_flags().contains(gst::GapFlags::DATA) {
let settings = self.settings.lock().unwrap().clone();
let mut state = self.output_state.lock().unwrap();
let fallback_sinkpad = self.fallback_sinkpad.read().unwrap();
let on_active_sinkpad = {
let active_sinkpad = self.active_sinkpad.lock().unwrap();
active_sinkpad.as_ref() == Some(agg_pad.upcast_ref::<gst::Pad>())
};
// handle GAP as timed item
let clock = agg.clock();
let base_time = agg.base_time();
let cur_running_time = if let Some(clock) = clock {
clock.time().opt_checked_sub(base_time).ok().flatten()
} else {
gst::ClockTime::NONE
};
let mut forwarded = false;
if let Ok(Some((item, _, _))) = self.handle_main_timed_item(
agg,
&mut *state,
&settings,
TimedItem::GapEvent(event),
agg_pad,
&fallback_sinkpad.as_ref(),
cur_running_time,
) {
// push GAP downstream only if it's from the active sink pad
if on_active_sinkpad {
let event = item.event();
// FIXME: API to retrieve src pad from Aggregator?
let src_pad = agg.static_pad("src").unwrap();
gst::debug!(
CAT,
obj: agg_pad,
"Forwarding gap event downstream: {:?}",
event
);
src_pad.push_event(event);
forwarded = true;
}
}
if !forwarded {
gst::debug!(CAT, obj: agg_pad, "Dropping gap event"); gst::debug!(CAT, obj: agg_pad, "Dropping gap event");
}
Ok(gst::FlowSuccess::Ok) Ok(gst::FlowSuccess::Ok)
} else { } else {
self.parent_sink_event_pre_queue(agg, agg_pad, event) self.parent_sink_event_pre_queue(agg, agg_pad, event)
@ -1287,18 +1337,29 @@ impl AggregatorImpl for FallbackSwitch {
#[derive(Debug)] #[derive(Debug)]
enum TimedItem { enum TimedItem {
Buffer(gst::Buffer), Buffer(gst::Buffer),
// only used with feature v1_20
#[allow(dead_code)]
GapEvent(gst::Event),
} }
impl TimedItem { impl TimedItem {
fn name(&self) -> &str { fn name(&self) -> &str {
match self { match self {
TimedItem::Buffer(_) => "buffer", TimedItem::Buffer(_) => "buffer",
TimedItem::GapEvent(_) => "GAP event",
} }
} }
fn pts(&self) -> Option<gst::ClockTime> { fn pts(&self) -> Option<gst::ClockTime> {
match self { match self {
TimedItem::Buffer(buffer) => buffer.pts(), TimedItem::Buffer(buffer) => buffer.pts(),
TimedItem::GapEvent(event) => match event.view() {
gst::EventView::Gap(gap) => {
let (pts, _duration) = gap.get();
Some(pts)
}
_ => unreachable!(),
},
} }
} }
@ -1314,18 +1375,45 @@ impl TimedItem {
buffer.set_pts(running_time); buffer.set_pts(running_time);
buffer.set_dts(segment.to_running_time(buffer.dts())); buffer.set_dts(segment.to_running_time(buffer.dts()));
} }
TimedItem::GapEvent(event) => {
let new_event = match event.view() {
gst::EventView::Gap(gap) => {
let (pts, duration) = gap.get();
let builder = gst::event::Gap::builder(pts)
.duration(duration)
.seqnum(event.seqnum());
#[cfg(feature = "v1_20")]
let builder = builder.gap_flags(gap.gap_flags());
builder.build()
}
_ => unreachable!(),
};
*event = new_event;
}
} }
} }
fn is_keyframe(&self) -> bool { fn is_keyframe(&self) -> bool {
match self { match self {
TimedItem::Buffer(buffer) => !buffer.flags().contains(gst::BufferFlags::DELTA_UNIT), TimedItem::Buffer(buffer) => !buffer.flags().contains(gst::BufferFlags::DELTA_UNIT),
TimedItem::GapEvent(_) => false,
} }
} }
fn buffer(self) -> gst::Buffer { fn buffer(self) -> gst::Buffer {
match self { match self {
TimedItem::Buffer(buffer) => buffer, TimedItem::Buffer(buffer) => buffer,
_ => unreachable!(),
}
}
#[cfg(feature = "v1_20")]
fn event(self) -> gst::Event {
match self {
TimedItem::GapEvent(event) => event,
_ => unreachable!(),
} }
} }
} }