mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-24 16:18:16 +00:00
ges: Protect from Gst dynamic callbacks
The pad-added and no-more-pad signal can be emited from any thread so we have to protect our code from that
This commit is contained in:
parent
302e0ed5d7
commit
7e121ff167
2 changed files with 64 additions and 2 deletions
|
@ -37,6 +37,24 @@
|
||||||
|
|
||||||
#define DEFAULT_TIMELINE_MODE TIMELINE_MODE_PREVIEW
|
#define DEFAULT_TIMELINE_MODE TIMELINE_MODE_PREVIEW
|
||||||
|
|
||||||
|
/* lock to protect dynamic callbacks, like pad-added or nmp */
|
||||||
|
#define DYN_LOCK(pipeline) (&GES_TIMELINE_PIPELINE (pipeline)->priv->dyn_mutex)
|
||||||
|
#define LOCK_DYN(pipeline) G_STMT_START { \
|
||||||
|
GST_INFO_OBJECT (pipeline, "Getting dynamic lock from %p", \
|
||||||
|
g_thread_self()); \
|
||||||
|
g_mutex_lock (DYN_LOCK (pipeline)); \
|
||||||
|
GST_INFO_OBJECT (pipeline, "Got Dynamic lock from %p", \
|
||||||
|
g_thread_self()); \
|
||||||
|
} G_STMT_END
|
||||||
|
|
||||||
|
#define UNLOCK_DYN(pipeline) G_STMT_START { \
|
||||||
|
GST_INFO_OBJECT (pipeline, "Unlocking dynamic lock from %p", \
|
||||||
|
g_thread_self()); \
|
||||||
|
g_mutex_unlock (DYN_LOCK (pipeline)); \
|
||||||
|
GST_INFO_OBJECT (pipeline, "Unlocked Dynamic lock from %p", \
|
||||||
|
g_thread_self()); \
|
||||||
|
} G_STMT_END
|
||||||
|
|
||||||
/* Structure corresponding to a timeline - sink link */
|
/* Structure corresponding to a timeline - sink link */
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
|
@ -61,6 +79,7 @@ struct _GESTimelinePipelinePrivate
|
||||||
|
|
||||||
GESPipelineFlags mode;
|
GESPipelineFlags mode;
|
||||||
|
|
||||||
|
GMutex dyn_mutex;
|
||||||
GList *chains;
|
GList *chains;
|
||||||
|
|
||||||
GstEncodingProfile *profile;
|
GstEncodingProfile *profile;
|
||||||
|
@ -462,6 +481,7 @@ new_output_chain_for_track (GESTimelinePipeline * self, GESTrack * track)
|
||||||
return chain;
|
return chain;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Should be called with LOCK_DYN */
|
||||||
static OutputChain *
|
static OutputChain *
|
||||||
get_output_chain_for_track (GESTimelinePipeline * self, GESTrack * track)
|
get_output_chain_for_track (GESTimelinePipeline * self, GESTrack * track)
|
||||||
{
|
{
|
||||||
|
@ -560,6 +580,7 @@ pad_added_cb (GstElement * timeline, GstPad * pad, GESTimelinePipeline * self)
|
||||||
GstCaps *caps;
|
GstCaps *caps;
|
||||||
gboolean reconfigured = FALSE;
|
gboolean reconfigured = FALSE;
|
||||||
|
|
||||||
|
LOCK_DYN (self);
|
||||||
caps = gst_pad_query_caps (pad, NULL);
|
caps = gst_pad_query_caps (pad, NULL);
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (self, "new pad %s:%s , caps:%" GST_PTR_FORMAT,
|
GST_DEBUG_OBJECT (self, "new pad %s:%s , caps:%" GST_PTR_FORMAT,
|
||||||
|
@ -571,6 +592,7 @@ pad_added_cb (GstElement * timeline, GstPad * pad, GESTimelinePipeline * self)
|
||||||
|
|
||||||
if (G_UNLIKELY (!track)) {
|
if (G_UNLIKELY (!track)) {
|
||||||
GST_WARNING_OBJECT (self, "Couldn't find coresponding track !");
|
GST_WARNING_OBJECT (self, "Couldn't find coresponding track !");
|
||||||
|
UNLOCK_DYN (self);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -696,6 +718,7 @@ pad_added_cb (GstElement * timeline, GstPad * pad, GESTimelinePipeline * self)
|
||||||
if (!get_output_chain_for_track (self, track))
|
if (!get_output_chain_for_track (self, track))
|
||||||
self->priv->chains = g_list_append (self->priv->chains, chain);
|
self->priv->chains = g_list_append (self->priv->chains, chain);
|
||||||
|
|
||||||
|
UNLOCK_DYN (self);
|
||||||
GST_DEBUG ("done");
|
GST_DEBUG ("done");
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -707,6 +730,7 @@ error:
|
||||||
if (sinkpad)
|
if (sinkpad)
|
||||||
gst_object_unref (sinkpad);
|
gst_object_unref (sinkpad);
|
||||||
g_free (chain);
|
g_free (chain);
|
||||||
|
UNLOCK_DYN (self);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -717,16 +741,19 @@ pad_removed_cb (GstElement * timeline, GstPad * pad, GESTimelinePipeline * self)
|
||||||
GESTrack *track;
|
GESTrack *track;
|
||||||
GstPad *peer;
|
GstPad *peer;
|
||||||
|
|
||||||
|
LOCK_DYN (self);
|
||||||
GST_DEBUG_OBJECT (self, "pad removed %s:%s", GST_DEBUG_PAD_NAME (pad));
|
GST_DEBUG_OBJECT (self, "pad removed %s:%s", GST_DEBUG_PAD_NAME (pad));
|
||||||
|
|
||||||
if (G_UNLIKELY (!(track =
|
if (G_UNLIKELY (!(track =
|
||||||
ges_timeline_get_track_for_pad (self->priv->timeline, pad)))) {
|
ges_timeline_get_track_for_pad (self->priv->timeline, pad)))) {
|
||||||
GST_WARNING_OBJECT (self, "Couldn't find coresponding track !");
|
GST_WARNING_OBJECT (self, "Couldn't find coresponding track !");
|
||||||
|
UNLOCK_DYN (self);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (G_UNLIKELY (!(chain = get_output_chain_for_track (self, track)))) {
|
if (G_UNLIKELY (!(chain = get_output_chain_for_track (self, track)))) {
|
||||||
GST_DEBUG_OBJECT (self, "Pad wasn't used");
|
GST_DEBUG_OBJECT (self, "Pad wasn't used");
|
||||||
|
UNLOCK_DYN (self);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -765,6 +792,7 @@ pad_removed_cb (GstElement * timeline, GstPad * pad, GESTimelinePipeline * self)
|
||||||
|
|
||||||
self->priv->chains = g_list_remove (self->priv->chains, chain);
|
self->priv->chains = g_list_remove (self->priv->chains, chain);
|
||||||
g_free (chain);
|
g_free (chain);
|
||||||
|
UNLOCK_DYN (self);
|
||||||
|
|
||||||
GST_DEBUG ("done");
|
GST_DEBUG ("done");
|
||||||
}
|
}
|
||||||
|
@ -775,6 +803,7 @@ no_more_pads_cb (GstElement * timeline, GESTimelinePipeline * self)
|
||||||
GList *tmp;
|
GList *tmp;
|
||||||
|
|
||||||
GST_DEBUG ("received no-more-pads");
|
GST_DEBUG ("received no-more-pads");
|
||||||
|
LOCK_DYN (self);
|
||||||
for (tmp = self->priv->chains; tmp; tmp = g_list_next (tmp)) {
|
for (tmp = self->priv->chains; tmp; tmp = g_list_next (tmp)) {
|
||||||
OutputChain *chain = (OutputChain *) tmp->data;
|
OutputChain *chain = (OutputChain *) tmp->data;
|
||||||
|
|
||||||
|
@ -786,6 +815,7 @@ no_more_pads_cb (GstElement * timeline, GESTimelinePipeline * self)
|
||||||
chain->probe_id = 0;
|
chain->probe_id = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
UNLOCK_DYN (self);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -65,6 +65,24 @@ GST_DEBUG_CATEGORY_STATIC (ges_timeline_debug);
|
||||||
#undef GST_CAT_DEFAULT
|
#undef GST_CAT_DEFAULT
|
||||||
#define GST_CAT_DEFAULT ges_timeline_debug
|
#define GST_CAT_DEFAULT ges_timeline_debug
|
||||||
|
|
||||||
|
/* lock to protect dynamic callbacks, like pad-added */
|
||||||
|
#define DYN_LOCK(timeline) (&GES_TIMELINE (timeline)->priv->dyn_mutex)
|
||||||
|
#define LOCK_DYN(timeline) G_STMT_START { \
|
||||||
|
GST_INFO_OBJECT (timeline, "Getting dynamic lock from %p", \
|
||||||
|
g_thread_self()); \
|
||||||
|
g_rec_mutex_lock (DYN_LOCK (timeline)); \
|
||||||
|
GST_INFO_OBJECT (timeline, "Got Dynamic lock from %p", \
|
||||||
|
g_thread_self()); \
|
||||||
|
} G_STMT_END
|
||||||
|
|
||||||
|
#define UNLOCK_DYN(timeline) G_STMT_START { \
|
||||||
|
GST_INFO_OBJECT (timeline, "Unlocking dynamic lock from %p", \
|
||||||
|
g_thread_self()); \
|
||||||
|
g_rec_mutex_unlock (DYN_LOCK (timeline)); \
|
||||||
|
GST_INFO_OBJECT (timeline, "Unlocked Dynamic lock from %p", \
|
||||||
|
g_thread_self()); \
|
||||||
|
} G_STMT_END
|
||||||
|
|
||||||
typedef struct TrackObjIters
|
typedef struct TrackObjIters
|
||||||
{
|
{
|
||||||
GSequenceIter *iter_start;
|
GSequenceIter *iter_start;
|
||||||
|
@ -143,6 +161,7 @@ struct _GESTimelinePrivate
|
||||||
/* We keep 1 reference to our trackelement here */
|
/* We keep 1 reference to our trackelement here */
|
||||||
GSequence *tracksources; /* Source-s sorted by start/priorities */
|
GSequence *tracksources; /* Source-s sorted by start/priorities */
|
||||||
|
|
||||||
|
GRecMutex dyn_mutex;
|
||||||
GList *priv_tracks;
|
GList *priv_tracks;
|
||||||
/* FIXME: We should definitly offer an API over this,
|
/* FIXME: We should definitly offer an API over this,
|
||||||
* probably through a ges_layer_get_track_elements () method */
|
* probably through a ges_layer_get_track_elements () method */
|
||||||
|
@ -1966,6 +1985,7 @@ layer_object_removed_cb (GESLayer * layer, GESClip * clip,
|
||||||
|
|
||||||
/* FIXME Check if we should actually check that we control the
|
/* FIXME Check if we should actually check that we control the
|
||||||
* track in the new management of TrackElement context */
|
* track in the new management of TrackElement context */
|
||||||
|
LOCK_DYN (timeline);
|
||||||
if (G_LIKELY (g_list_find_custom (timeline->priv->priv_tracks, track,
|
if (G_LIKELY (g_list_find_custom (timeline->priv->priv_tracks, track,
|
||||||
(GCompareFunc) custom_find_track) || track == NULL)) {
|
(GCompareFunc) custom_find_track) || track == NULL)) {
|
||||||
GST_DEBUG ("Belongs to one of the tracks we control");
|
GST_DEBUG ("Belongs to one of the tracks we control");
|
||||||
|
@ -1973,6 +1993,7 @@ layer_object_removed_cb (GESLayer * layer, GESClip * clip,
|
||||||
ges_container_remove (GES_CONTAINER (clip),
|
ges_container_remove (GES_CONTAINER (clip),
|
||||||
GES_TIMELINE_ELEMENT (track_element));
|
GES_TIMELINE_ELEMENT (track_element));
|
||||||
}
|
}
|
||||||
|
UNLOCK_DYN (timeline);
|
||||||
}
|
}
|
||||||
g_signal_handlers_disconnect_by_func (clip, clip_track_element_added_cb,
|
g_signal_handlers_disconnect_by_func (clip, clip_track_element_added_cb,
|
||||||
timeline);
|
timeline);
|
||||||
|
@ -2127,6 +2148,7 @@ pad_added_cb (GESTrack * track, GstPad * pad, TrackPrivate * tr_priv)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Remember the pad */
|
/* Remember the pad */
|
||||||
|
LOCK_DYN (tr_priv->timeline);
|
||||||
GST_OBJECT_LOCK (track);
|
GST_OBJECT_LOCK (track);
|
||||||
tr_priv->pad = pad;
|
tr_priv->pad = pad;
|
||||||
|
|
||||||
|
@ -2153,6 +2175,7 @@ pad_added_cb (GESTrack * track, GstPad * pad, TrackPrivate * tr_priv)
|
||||||
GST_DEBUG ("Signaling no-more-pads");
|
GST_DEBUG ("Signaling no-more-pads");
|
||||||
gst_element_no_more_pads (GST_ELEMENT (tr_priv->timeline));
|
gst_element_no_more_pads (GST_ELEMENT (tr_priv->timeline));
|
||||||
}
|
}
|
||||||
|
UNLOCK_DYN (tr_priv->timeline);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -2477,8 +2500,10 @@ ges_timeline_add_track (GESTimeline * timeline, GESTrack * track)
|
||||||
tr_priv->track = track;
|
tr_priv->track = track;
|
||||||
|
|
||||||
/* Add the track to the list of tracks we track */
|
/* Add the track to the list of tracks we track */
|
||||||
|
LOCK_DYN (timeline);
|
||||||
timeline->priv->priv_tracks = g_list_append (timeline->priv->priv_tracks,
|
timeline->priv->priv_tracks = g_list_append (timeline->priv->priv_tracks,
|
||||||
tr_priv);
|
tr_priv);
|
||||||
|
UNLOCK_DYN (timeline);
|
||||||
timeline->tracks = g_list_append (timeline->tracks, track);
|
timeline->tracks = g_list_append (timeline->tracks, track);
|
||||||
|
|
||||||
/* Listen to pad-added/-removed */
|
/* Listen to pad-added/-removed */
|
||||||
|
@ -2551,14 +2576,17 @@ ges_timeline_remove_track (GESTimeline * timeline, GESTrack * track)
|
||||||
GST_DEBUG ("timeline:%p, track:%p", timeline, track);
|
GST_DEBUG ("timeline:%p, track:%p", timeline, track);
|
||||||
|
|
||||||
priv = timeline->priv;
|
priv = timeline->priv;
|
||||||
|
LOCK_DYN (timeline);
|
||||||
if (G_UNLIKELY (!(tmp = g_list_find_custom (priv->priv_tracks,
|
if (G_UNLIKELY (!(tmp = g_list_find_custom (priv->priv_tracks,
|
||||||
track, (GCompareFunc) custom_find_track)))) {
|
track, (GCompareFunc) custom_find_track)))) {
|
||||||
GST_WARNING ("Track doesn't belong to this timeline");
|
GST_WARNING ("Track doesn't belong to this timeline");
|
||||||
|
UNLOCK_DYN (timeline);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
tr_priv = tmp->data;
|
tr_priv = tmp->data;
|
||||||
priv->priv_tracks = g_list_remove (priv->priv_tracks, tr_priv);
|
priv->priv_tracks = g_list_remove (priv->priv_tracks, tr_priv);
|
||||||
|
UNLOCK_DYN (timeline);
|
||||||
timeline->tracks = g_list_remove (timeline->tracks, track);
|
timeline->tracks = g_list_remove (timeline->tracks, track);
|
||||||
|
|
||||||
ges_track_set_timeline (track, NULL);
|
ges_track_set_timeline (track, NULL);
|
||||||
|
@ -2616,11 +2644,15 @@ ges_timeline_get_track_for_pad (GESTimeline * timeline, GstPad * pad)
|
||||||
{
|
{
|
||||||
GList *tmp;
|
GList *tmp;
|
||||||
|
|
||||||
|
LOCK_DYN (timeline);
|
||||||
for (tmp = timeline->priv->priv_tracks; tmp; tmp = g_list_next (tmp)) {
|
for (tmp = timeline->priv->priv_tracks; tmp; tmp = g_list_next (tmp)) {
|
||||||
TrackPrivate *tr_priv = (TrackPrivate *) tmp->data;
|
TrackPrivate *tr_priv = (TrackPrivate *) tmp->data;
|
||||||
if (pad == tr_priv->ghostpad)
|
if (pad == tr_priv->ghostpad) {
|
||||||
|
UNLOCK_DYN (timeline);
|
||||||
return tr_priv->track;
|
return tr_priv->track;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
UNLOCK_DYN (timeline);
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue