mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-11 19:06:33 +00:00
output-selector: Make access to the active pad and last buffer thread-safe
Both can be modified from different threads at the same time.
This commit is contained in:
parent
0e132260a3
commit
485b15d8a9
1 changed files with 54 additions and 14 deletions
|
@ -178,14 +178,17 @@ gst_output_selector_init (GstOutputSelector * sel)
|
||||||
static void
|
static void
|
||||||
gst_output_selector_reset (GstOutputSelector * osel)
|
gst_output_selector_reset (GstOutputSelector * osel)
|
||||||
{
|
{
|
||||||
|
GST_OBJECT_LOCK (osel);
|
||||||
if (osel->pending_srcpad != NULL) {
|
if (osel->pending_srcpad != NULL) {
|
||||||
gst_object_unref (osel->pending_srcpad);
|
gst_object_unref (osel->pending_srcpad);
|
||||||
osel->pending_srcpad = NULL;
|
osel->pending_srcpad = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (osel->latest_buffer != NULL) {
|
if (osel->latest_buffer != NULL) {
|
||||||
gst_buffer_unref (osel->latest_buffer);
|
gst_buffer_unref (osel->latest_buffer);
|
||||||
osel->latest_buffer = NULL;
|
osel->latest_buffer = NULL;
|
||||||
}
|
}
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
gst_segment_init (&osel->segment, GST_FORMAT_UNDEFINED);
|
gst_segment_init (&osel->segment, GST_FORMAT_UNDEFINED);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -335,9 +338,13 @@ gst_output_selector_request_new_pad (GstElement * element,
|
||||||
gst_element_add_pad (GST_ELEMENT (osel), srcpad);
|
gst_element_add_pad (GST_ELEMENT (osel), srcpad);
|
||||||
|
|
||||||
/* Set the first requested src pad as active by default */
|
/* Set the first requested src pad as active by default */
|
||||||
|
GST_OBJECT_LOCK (osel);
|
||||||
if (osel->active_srcpad == NULL) {
|
if (osel->active_srcpad == NULL) {
|
||||||
osel->active_srcpad = srcpad;
|
osel->active_srcpad = srcpad;
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
g_object_notify (G_OBJECT (osel), "active-pad");
|
g_object_notify (G_OBJECT (osel), "active-pad");
|
||||||
|
} else {
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
}
|
}
|
||||||
g_free (padname);
|
g_free (padname);
|
||||||
|
|
||||||
|
@ -354,9 +361,13 @@ gst_output_selector_release_pad (GstElement * element, GstPad * pad)
|
||||||
GST_DEBUG_OBJECT (osel, "releasing pad");
|
GST_DEBUG_OBJECT (osel, "releasing pad");
|
||||||
|
|
||||||
/* Disable active pad if it's the to be removed pad */
|
/* Disable active pad if it's the to be removed pad */
|
||||||
|
GST_OBJECT_LOCK (osel);
|
||||||
if (osel->active_srcpad == pad) {
|
if (osel->active_srcpad == pad) {
|
||||||
osel->active_srcpad = NULL;
|
osel->active_srcpad = NULL;
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
g_object_notify (G_OBJECT (osel), "active-pad");
|
g_object_notify (G_OBJECT (osel), "active-pad");
|
||||||
|
} else {
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_pad_set_active (pad, FALSE);
|
gst_pad_set_active (pad, FALSE);
|
||||||
|
@ -370,50 +381,68 @@ gst_output_selector_switch (GstOutputSelector * osel)
|
||||||
gboolean res = FALSE;
|
gboolean res = FALSE;
|
||||||
GstEvent *ev = NULL;
|
GstEvent *ev = NULL;
|
||||||
GstSegment *seg = NULL;
|
GstSegment *seg = NULL;
|
||||||
|
GstPad *active_srcpad;
|
||||||
|
|
||||||
/* Switch */
|
/* Switch */
|
||||||
GST_OBJECT_LOCK (GST_OBJECT (osel));
|
GST_OBJECT_LOCK (osel);
|
||||||
GST_INFO_OBJECT (osel, "switching to pad %" GST_PTR_FORMAT,
|
GST_INFO_OBJECT (osel, "switching to pad %" GST_PTR_FORMAT,
|
||||||
osel->pending_srcpad);
|
osel->pending_srcpad);
|
||||||
|
if (!osel->pending_srcpad) {
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
if (gst_pad_is_linked (osel->pending_srcpad)) {
|
if (gst_pad_is_linked (osel->pending_srcpad)) {
|
||||||
osel->active_srcpad = osel->pending_srcpad;
|
osel->active_srcpad = osel->pending_srcpad;
|
||||||
res = TRUE;
|
res = TRUE;
|
||||||
}
|
}
|
||||||
gst_object_unref (osel->pending_srcpad);
|
gst_object_unref (osel->pending_srcpad);
|
||||||
osel->pending_srcpad = NULL;
|
osel->pending_srcpad = NULL;
|
||||||
GST_OBJECT_UNLOCK (GST_OBJECT (osel));
|
active_srcpad = res ? gst_object_ref (osel->active_srcpad) : NULL;
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
|
|
||||||
/* Send SEGMENT event and latest buffer if switching succeeded
|
/* Send SEGMENT event and latest buffer if switching succeeded
|
||||||
* and we already have a valid segment configured */
|
* and we already have a valid segment configured */
|
||||||
if (res) {
|
if (res) {
|
||||||
|
GstBuffer *latest_buffer;
|
||||||
|
|
||||||
g_object_notify (G_OBJECT (osel), "active-pad");
|
g_object_notify (G_OBJECT (osel), "active-pad");
|
||||||
|
|
||||||
|
GST_OBJECT_LOCK (osel);
|
||||||
|
latest_buffer =
|
||||||
|
osel->latest_buffer ? gst_buffer_ref (osel->latest_buffer) : NULL;
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
|
|
||||||
gst_pad_sticky_events_foreach (osel->sinkpad, forward_sticky_events,
|
gst_pad_sticky_events_foreach (osel->sinkpad, forward_sticky_events,
|
||||||
osel->active_srcpad);
|
active_srcpad);
|
||||||
|
|
||||||
/* update segment if required */
|
/* update segment if required */
|
||||||
if (osel->segment.format != GST_FORMAT_UNDEFINED) {
|
if (osel->segment.format != GST_FORMAT_UNDEFINED) {
|
||||||
/* Send SEGMENT to the pad we are going to switch to */
|
/* Send SEGMENT to the pad we are going to switch to */
|
||||||
seg = &osel->segment;
|
seg = &osel->segment;
|
||||||
/* If resending then mark segment start and position accordingly */
|
/* If resending then mark segment start and position accordingly */
|
||||||
if (osel->resend_latest && osel->latest_buffer &&
|
if (osel->resend_latest && latest_buffer &&
|
||||||
GST_BUFFER_TIMESTAMP_IS_VALID (osel->latest_buffer)) {
|
GST_BUFFER_TIMESTAMP_IS_VALID (latest_buffer)) {
|
||||||
seg->position = GST_BUFFER_TIMESTAMP (osel->latest_buffer);
|
seg->position = GST_BUFFER_TIMESTAMP (latest_buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
ev = gst_event_new_segment (seg);
|
ev = gst_event_new_segment (seg);
|
||||||
|
|
||||||
if (!gst_pad_push_event (osel->active_srcpad, ev)) {
|
if (!gst_pad_push_event (active_srcpad, ev)) {
|
||||||
GST_WARNING_OBJECT (osel,
|
GST_WARNING_OBJECT (osel,
|
||||||
"newsegment handling failed in %" GST_PTR_FORMAT,
|
"newsegment handling failed in %" GST_PTR_FORMAT, active_srcpad);
|
||||||
osel->active_srcpad);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Resend latest buffer to newly switched pad */
|
/* Resend latest buffer to newly switched pad */
|
||||||
if (osel->resend_latest && osel->latest_buffer) {
|
if (osel->resend_latest && latest_buffer) {
|
||||||
GST_INFO ("resending latest buffer");
|
GST_INFO ("resending latest buffer");
|
||||||
gst_pad_push (osel->active_srcpad, gst_buffer_ref (osel->latest_buffer));
|
gst_pad_push (active_srcpad, latest_buffer);
|
||||||
|
} else if (latest_buffer) {
|
||||||
|
gst_buffer_unref (latest_buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
gst_object_unref (active_srcpad);
|
||||||
} else {
|
} else {
|
||||||
GST_WARNING_OBJECT (osel, "switch failed, pad not linked");
|
GST_WARNING_OBJECT (osel, "switch failed, pad not linked");
|
||||||
}
|
}
|
||||||
|
@ -427,6 +456,7 @@ gst_output_selector_chain (GstPad * pad, GstObject * parent, GstBuffer * buf)
|
||||||
GstFlowReturn res;
|
GstFlowReturn res;
|
||||||
GstOutputSelector *osel;
|
GstOutputSelector *osel;
|
||||||
GstClockTime position, duration;
|
GstClockTime position, duration;
|
||||||
|
GstPad *active_srcpad;
|
||||||
|
|
||||||
osel = GST_OUTPUT_SELECTOR (parent);
|
osel = GST_OUTPUT_SELECTOR (parent);
|
||||||
|
|
||||||
|
@ -446,6 +476,14 @@ gst_output_selector_chain (GstPad * pad, GstObject * parent, GstBuffer * buf)
|
||||||
gst_output_selector_switch (osel);
|
gst_output_selector_switch (osel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
active_srcpad = gst_output_selector_get_active (osel);
|
||||||
|
if (!active_srcpad) {
|
||||||
|
GST_DEBUG_OBJECT (osel, "No active srcpad");
|
||||||
|
gst_buffer_unref (buf);
|
||||||
|
return GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_OBJECT_LOCK (osel);
|
||||||
if (osel->latest_buffer) {
|
if (osel->latest_buffer) {
|
||||||
gst_buffer_unref (osel->latest_buffer);
|
gst_buffer_unref (osel->latest_buffer);
|
||||||
osel->latest_buffer = NULL;
|
osel->latest_buffer = NULL;
|
||||||
|
@ -455,6 +493,7 @@ gst_output_selector_chain (GstPad * pad, GstObject * parent, GstBuffer * buf)
|
||||||
/* Keep reference to latest buffer to resend it after switch */
|
/* Keep reference to latest buffer to resend it after switch */
|
||||||
osel->latest_buffer = gst_buffer_ref (buf);
|
osel->latest_buffer = gst_buffer_ref (buf);
|
||||||
}
|
}
|
||||||
|
GST_OBJECT_UNLOCK (osel);
|
||||||
|
|
||||||
/* Keep track of last stop and use it in SEGMENT start after
|
/* Keep track of last stop and use it in SEGMENT start after
|
||||||
switching to a new src pad */
|
switching to a new src pad */
|
||||||
|
@ -469,9 +508,10 @@ gst_output_selector_chain (GstPad * pad, GstObject * parent, GstBuffer * buf)
|
||||||
osel->segment.position = position;
|
osel->segment.position = position;
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_LOG_OBJECT (osel, "pushing buffer to %" GST_PTR_FORMAT,
|
GST_LOG_OBJECT (osel, "pushing buffer to %" GST_PTR_FORMAT, active_srcpad);
|
||||||
osel->active_srcpad);
|
res = gst_pad_push (active_srcpad, buf);
|
||||||
res = gst_pad_push (osel->active_srcpad, buf);
|
|
||||||
|
gst_object_unref (active_srcpad);
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue