gst/gstthread.c: Make sure no iteration happens while performing the state change as it could mess up the internal co...

Original commit message from CVS:
* gst/gstthread.c: (gst_thread_change_state),
(gst_thread_child_state_change):
Make sure no iteration happens while performing
the state change as it could mess up the internal
consistency of the thread state.
This commit is contained in:
Wim Taymans 2004-10-07 18:34:57 +00:00
parent c43a7670f0
commit 625722ecdc
2 changed files with 20 additions and 13 deletions

View file

@ -1,3 +1,11 @@
2004-10-07 Wim Taymans <wim at fluendo dot com>
* gst/gstthread.c: (gst_thread_change_state),
(gst_thread_child_state_change):
Make sure no iteration happens while performing
the state change as it could mess up the internal
consistency of the thread state.
2004-10-07 Wim Taymans <wim at fluendo dot com> 2004-10-07 Wim Taymans <wim at fluendo dot com>
* gst/gstthread.c: (gst_thread_dispose), (gst_thread_sync), * gst/gstthread.c: (gst_thread_dispose), (gst_thread_sync),

View file

@ -445,8 +445,6 @@ gst_thread_change_state (GstElement * element)
gst_element_state_get_name (GST_STATE (element)), gst_element_state_get_name (GST_STATE (element)),
gst_element_state_get_name (GST_STATE_PENDING (element))); gst_element_state_get_name (GST_STATE_PENDING (element)));
transition = GST_STATE_TRANSITION (element);
thread = GST_THREAD (element); thread = GST_THREAD (element);
/* boolean to check if we called the state change in the same thread as /* boolean to check if we called the state change in the same thread as
@ -458,12 +456,15 @@ gst_thread_change_state (GstElement * element)
gst_thread_sync (thread, is_self); gst_thread_sync (thread, is_self);
/* FIXME: (or GStreamers ideas about "threading"): the element variables are /* no iteration is allowed during this state change because an iteration
commonly accessed by multiple threads at the same time (see bug #111146 * can cause another state change conflicting with this one */
for an example) */ /* do not try to grab the lock if this method is called from the
if (transition != GST_STATE_TRANSITION (element)) { * same thread as the iterate thread, the lock might be held and we
g_warning ("inconsistent state information, fix threading please"); * might deadlock */
} if (!is_self)
g_mutex_lock (thread->iterate_lock);
transition = GST_STATE_TRANSITION (element);
switch (transition) { switch (transition) {
case GST_STATE_NULL_TO_READY: case GST_STATE_NULL_TO_READY:
@ -560,11 +561,6 @@ gst_thread_change_state (GstElement * element)
GST_LOG_OBJECT (thread, "unlocking lock"); GST_LOG_OBJECT (thread, "unlocking lock");
g_mutex_unlock (thread->lock); g_mutex_unlock (thread->lock);
/* do not try to grab the lock if this method is called from the
* same thread as the iterate thread, the lock might be held and we
* might deadlock */
if (!is_self)
g_mutex_lock (thread->iterate_lock);
if (GST_ELEMENT_CLASS (parent_class)->change_state) { if (GST_ELEMENT_CLASS (parent_class)->change_state) {
ret = GST_ELEMENT_CLASS (parent_class)->change_state (GST_ELEMENT (thread)); ret = GST_ELEMENT_CLASS (parent_class)->change_state (GST_ELEMENT (thread));
} else { } else {
@ -583,6 +579,9 @@ error_out:
g_mutex_unlock (thread->lock); g_mutex_unlock (thread->lock);
if (!is_self)
g_mutex_unlock (thread->iterate_lock);
return GST_STATE_FAILURE; return GST_STATE_FAILURE;
} }