libs/gst/base/gstbasesink.c: call ::unlock before taking the PREROLL_LOCK so we can safely handle elements that lock ...

Original commit message from CVS:
Patch by: Philippe Rouquier <philippero at libertysurf dot fr>
* libs/gst/base/gstbasesink.c: (gst_base_sink_set_flushing),
(gst_base_sink_change_state):
call ::unlock before taking the PREROLL_LOCK so we can safely
handle elements that lock in ::render.
Fixes #340174.
This commit is contained in:
Philippe Rouquier 2006-05-08 11:49:43 +00:00 committed by Wim Taymans
parent 462d27b23a
commit 246712327d
2 changed files with 26 additions and 9 deletions

View file

@ -1,3 +1,13 @@
2006-05-08 Wim Taymans <wim@fluendo.com>
Patch by: Philippe Rouquier <philippero at libertysurf dot fr>
* libs/gst/base/gstbasesink.c: (gst_base_sink_set_flushing),
(gst_base_sink_change_state):
call ::unlock before taking the PREROLL_LOCK so we can safely
handle elements that lock in ::render.
Fixes #340174.
2006-05-08 Edward Hervey <edward@fluendo.com> 2006-05-08 Edward Hervey <edward@fluendo.com>
* autogen.sh: (CONFIGURE_DEF_OPT): * autogen.sh: (CONFIGURE_DEF_OPT):

View file

@ -2023,23 +2023,27 @@ static gboolean
gst_base_sink_set_flushing (GstBaseSink * basesink, GstPad * pad, gst_base_sink_set_flushing (GstBaseSink * basesink, GstPad * pad,
gboolean flushing) gboolean flushing)
{ {
GST_PAD_PREROLL_LOCK (pad);
basesink->flushing = flushing;
if (flushing) { if (flushing) {
GstBaseSinkClass *bclass; GstBaseSinkClass *bclass;
bclass = GST_BASE_SINK_GET_CLASS (basesink); bclass = GST_BASE_SINK_GET_CLASS (basesink);
/* unlock any subclasses, we need to do this before grabbing the
* PREROLL_LOCK since we hold this lock before going into ::render. */
if (bclass->unlock)
bclass->unlock (basesink);
}
GST_PAD_PREROLL_LOCK (pad);
basesink->flushing = flushing;
if (flushing) {
/* step 1, unblock clock sync (if any) or any other blocking thing */ /* step 1, unblock clock sync (if any) or any other blocking thing */
basesink->need_preroll = TRUE; basesink->need_preroll = TRUE;
if (basesink->clock_id) { if (basesink->clock_id) {
gst_clock_id_unschedule (basesink->clock_id); gst_clock_id_unschedule (basesink->clock_id);
} }
/* unlock any subclasses */
if (bclass->unlock)
bclass->unlock (basesink);
/* flush out the data thread if it's locked in finish_preroll */ /* flush out the data thread if it's locked in finish_preroll */
GST_DEBUG_OBJECT (basesink, GST_DEBUG_OBJECT (basesink,
"flushing out data thread, need preroll to TRUE"); "flushing out data thread, need preroll to TRUE");
@ -2488,15 +2492,18 @@ gst_base_sink_change_state (GstElement * element, GstStateChange transition)
switch (transition) { switch (transition) {
case GST_STATE_CHANGE_PLAYING_TO_PAUSED: case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
GST_DEBUG_OBJECT (basesink, "PLAYING to PAUSED"); GST_DEBUG_OBJECT (basesink, "PLAYING to PAUSED");
/* we need to call ::unlock before locking PREROLL_LOCK
* since we lock it before going into ::render */
if (bclass->unlock)
bclass->unlock (basesink);
GST_PAD_PREROLL_LOCK (basesink->sinkpad); GST_PAD_PREROLL_LOCK (basesink->sinkpad);
basesink->need_preroll = TRUE; basesink->need_preroll = TRUE;
if (basesink->clock_id) { if (basesink->clock_id) {
gst_clock_id_unschedule (basesink->clock_id); gst_clock_id_unschedule (basesink->clock_id);
} }
if (bclass->unlock)
bclass->unlock (basesink);
/* if we don't have a preroll buffer we need to wait for a preroll and /* if we don't have a preroll buffer we need to wait for a preroll and
* return ASYNC. */ * return ASYNC. */
if (gst_base_sink_is_prerolled (basesink)) { if (gst_base_sink_is_prerolled (basesink)) {