mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 10:25:33 +00:00
wrappercamerabinsrc: Handle src state change to avoid losing timestamps
Camerabin2 uses state changes to force the source to renegotiate its caps to the capture formats. The state changes makes the source lose its clock and base_time, causing it to stop timestamping the buffers. We still need a proper way to make sources renegotiate its caps, so this patch is a hack to make the source continue timestamping buffers even after changing state. The patch works by getting the clock and base time before doing the state change to NULL and setting them back after putting it to PLAYING again. It also cares to drop the first new segment after this state change.
This commit is contained in:
parent
4a7dc81782
commit
1a62d9374b
2 changed files with 69 additions and 1 deletions
|
@ -165,12 +165,50 @@ static void
|
||||||
gst_wrapper_camera_bin_reset_video_src_caps (GstWrapperCameraBinSrc * self,
|
gst_wrapper_camera_bin_reset_video_src_caps (GstWrapperCameraBinSrc * self,
|
||||||
GstCaps * caps)
|
GstCaps * caps)
|
||||||
{
|
{
|
||||||
|
GstClock *clock;
|
||||||
|
gint64 base_time;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (self, "Resetting src caps to %" GST_PTR_FORMAT, caps);
|
GST_DEBUG_OBJECT (self, "Resetting src caps to %" GST_PTR_FORMAT, caps);
|
||||||
if (self->src_vid_src) {
|
if (self->src_vid_src) {
|
||||||
|
clock = gst_element_get_clock (self->src_vid_src);
|
||||||
|
base_time = gst_element_get_base_time (self->src_vid_src);
|
||||||
|
|
||||||
gst_element_set_state (self->src_vid_src, GST_STATE_NULL);
|
gst_element_set_state (self->src_vid_src, GST_STATE_NULL);
|
||||||
set_capsfilter_caps (self, caps);
|
set_capsfilter_caps (self, caps);
|
||||||
|
|
||||||
|
self->drop_newseg = TRUE;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (self, "Bringing source up");
|
GST_DEBUG_OBJECT (self, "Bringing source up");
|
||||||
gst_element_sync_state_with_parent (self->src_vid_src);
|
gst_element_sync_state_with_parent (self->src_vid_src);
|
||||||
|
|
||||||
|
gst_element_set_clock (self->src_vid_src, clock);
|
||||||
|
gst_element_set_base_time (self->src_vid_src, base_time);
|
||||||
|
|
||||||
|
if (GST_IS_BIN (self->src_vid_src)) {
|
||||||
|
GstIterator *it = gst_bin_iterate_elements (GST_BIN (self->src_vid_src));
|
||||||
|
gpointer item = NULL;
|
||||||
|
gboolean done = FALSE;
|
||||||
|
while (!done) {
|
||||||
|
switch (gst_iterator_next (it, &item)) {
|
||||||
|
case GST_ITERATOR_OK:
|
||||||
|
gst_element_set_base_time (GST_ELEMENT (item), base_time);
|
||||||
|
gst_object_unref (item);
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_RESYNC:
|
||||||
|
gst_iterator_resync (it);
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_ERROR:
|
||||||
|
done = TRUE;
|
||||||
|
break;
|
||||||
|
case GST_ITERATOR_DONE:
|
||||||
|
done = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
gst_iterator_free (it);
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_object_unref (clock);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -286,6 +324,30 @@ gst_wrapper_camera_bin_src_event (GstPad * pad, GstEvent * event)
|
||||||
return src->srcpad_event_func (pad, event);
|
return src->srcpad_event_func (pad, event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_wrapper_camera_src_src_event_probe (GstPad * pad, GstEvent * evt,
|
||||||
|
gpointer udata)
|
||||||
|
{
|
||||||
|
gboolean ret = TRUE;
|
||||||
|
GstWrapperCameraBinSrc *self = udata;
|
||||||
|
|
||||||
|
switch (GST_EVENT_TYPE (evt)) {
|
||||||
|
case GST_EVENT_EOS:
|
||||||
|
/* drop */
|
||||||
|
ret = FALSE;
|
||||||
|
break;
|
||||||
|
case GST_EVENT_NEWSEGMENT:
|
||||||
|
if (self->drop_newseg) {
|
||||||
|
ret = FALSE;
|
||||||
|
self->drop_newseg = FALSE;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* gst_wrapper_camera_bin_src_construct_pipeline:
|
* gst_wrapper_camera_bin_src_construct_pipeline:
|
||||||
* @bcamsrc: camerasrc object
|
* @bcamsrc: camerasrc object
|
||||||
|
@ -334,7 +396,7 @@ gst_wrapper_camera_bin_src_construct_pipeline (GstBaseCameraSrc * bcamsrc)
|
||||||
pad = gst_element_get_static_pad (self->src_vid_src, "src");
|
pad = gst_element_get_static_pad (self->src_vid_src, "src");
|
||||||
|
|
||||||
self->src_event_probe_id = gst_pad_add_event_probe (pad,
|
self->src_event_probe_id = gst_pad_add_event_probe (pad,
|
||||||
(GCallback) gst_camerabin_drop_eos_probe, NULL);
|
(GCallback) gst_wrapper_camera_src_src_event_probe, self);
|
||||||
gst_object_unref (pad);
|
gst_object_unref (pad);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -991,6 +1053,9 @@ gst_wrapper_camera_bin_src_change_state (GstElement * element,
|
||||||
goto end;
|
goto end;
|
||||||
|
|
||||||
switch (trans) {
|
switch (trans) {
|
||||||
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
||||||
|
self->drop_newseg = FALSE;
|
||||||
|
break;
|
||||||
case GST_STATE_CHANGE_READY_TO_NULL:
|
case GST_STATE_CHANGE_READY_TO_NULL:
|
||||||
gst_element_set_state (self->preview_pipeline->pipeline, GST_STATE_NULL);
|
gst_element_set_state (self->preview_pipeline->pipeline, GST_STATE_NULL);
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -88,6 +88,9 @@ struct _GstWrapperCameraBinSrc
|
||||||
|
|
||||||
GstPadEventFunction srcpad_event_func;
|
GstPadEventFunction srcpad_event_func;
|
||||||
|
|
||||||
|
/* For changing caps without losing timestamps */
|
||||||
|
gboolean drop_newseg;
|
||||||
|
|
||||||
/* Application configurable elements */
|
/* Application configurable elements */
|
||||||
GstElement *app_vid_src;
|
GstElement *app_vid_src;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue