downloadbuffer: release lock before posting msg

to avoid the deadlock in playbin2,
send msg after release the download buffer lock.

https://bugzilla.gnome.org/show_bug.cgi?id=749535
This commit is contained in:
eunhae choi 2015-05-20 21:18:08 +09:00 committed by Tim-Philipp Müller
parent 4093fc621a
commit 618cd5e65c

View file

@ -174,7 +174,7 @@ enum
G_DEFINE_TYPE_WITH_CODE (GstDownloadBuffer, gst_download_buffer, G_DEFINE_TYPE_WITH_CODE (GstDownloadBuffer, gst_download_buffer,
GST_TYPE_ELEMENT, _do_init); GST_TYPE_ELEMENT, _do_init);
static void update_buffering (GstDownloadBuffer * dlbuf); static GstMessage *update_buffering (GstDownloadBuffer * dlbuf);
static void gst_download_buffer_finalize (GObject * object); static void gst_download_buffer_finalize (GObject * object);
@ -390,25 +390,19 @@ reset_rate_timer (GstDownloadBuffer * dlbuf)
#define AVG_OUT(avg,val) ((avg) * 3.0 + (val)) / 4.0 #define AVG_OUT(avg,val) ((avg) * 3.0 + (val)) / 4.0
static void static void
update_time_level (GstDownloadBuffer * dlbuf) update_levels (GstDownloadBuffer * dlbuf, guint bytes)
{ {
dlbuf->cur_level.bytes = bytes;
if (dlbuf->byte_in_rate > 0.0) { if (dlbuf->byte_in_rate > 0.0) {
dlbuf->cur_level.time = dlbuf->cur_level.time =
dlbuf->cur_level.bytes / dlbuf->byte_in_rate * GST_SECOND; dlbuf->cur_level.bytes / dlbuf->byte_in_rate * GST_SECOND;
} }
GST_DEBUG ("levels: bytes %u/%u, time %" GST_TIME_FORMAT "/%" GST_TIME_FORMAT, GST_DEBUG ("levels: bytes %u/%u, time %" GST_TIME_FORMAT "/%" GST_TIME_FORMAT,
dlbuf->cur_level.bytes, dlbuf->max_level.bytes, dlbuf->cur_level.bytes, dlbuf->max_level.bytes,
GST_TIME_ARGS (dlbuf->cur_level.time), GST_TIME_ARGS (dlbuf->cur_level.time),
GST_TIME_ARGS (dlbuf->max_level.time)); GST_TIME_ARGS (dlbuf->max_level.time));
/* update the buffering */
update_buffering (dlbuf);
}
static void
update_levels (GstDownloadBuffer * dlbuf, guint bytes)
{
dlbuf->cur_level.bytes = bytes;
update_time_level (dlbuf);
} }
static void static void
@ -566,14 +560,15 @@ get_buffering_stats (GstDownloadBuffer * dlbuf, gint percent,
} }
} }
static void static GstMessage *
update_buffering (GstDownloadBuffer * dlbuf) update_buffering (GstDownloadBuffer * dlbuf)
{ {
gint percent; gint percent;
gboolean post = FALSE; gboolean post = FALSE;
GstMessage *message = NULL;
if (!get_buffering_percent (dlbuf, NULL, &percent)) if (!get_buffering_percent (dlbuf, NULL, &percent))
return; return NULL;
if (dlbuf->is_buffering) { if (dlbuf->is_buffering) {
post = TRUE; post = TRUE;
@ -597,7 +592,6 @@ update_buffering (GstDownloadBuffer * dlbuf)
} }
if (post) { if (post) {
GstMessage *message;
GstBufferingMode mode; GstBufferingMode mode;
gint avg_in, avg_out; gint avg_in, avg_out;
gint64 buffering_left; gint64 buffering_left;
@ -609,9 +603,9 @@ update_buffering (GstDownloadBuffer * dlbuf)
(gint) percent); (gint) percent);
gst_message_set_buffering_stats (message, mode, gst_message_set_buffering_stats (message, mode,
avg_in, avg_out, buffering_left); avg_in, avg_out, buffering_left);
gst_element_post_message (GST_ELEMENT_CAST (dlbuf), message);
} }
return message;
} }
static gboolean static gboolean
@ -1051,6 +1045,8 @@ gst_download_buffer_handle_sink_event (GstPad * pad, GstObject * parent,
} }
default: default:
if (GST_EVENT_IS_SERIALIZED (event)) { if (GST_EVENT_IS_SERIALIZED (event)) {
GstMessage *msg = NULL;
/* serialized events go in the buffer */ /* serialized events go in the buffer */
GST_DOWNLOAD_BUFFER_MUTEX_LOCK_CHECK (dlbuf, dlbuf->sinkresult, GST_DOWNLOAD_BUFFER_MUTEX_LOCK_CHECK (dlbuf, dlbuf->sinkresult,
out_flushing); out_flushing);
@ -1061,6 +1057,8 @@ gst_download_buffer_handle_sink_event (GstPad * pad, GstObject * parent,
* filled and we can read all data from the dlbuf. */ * filled and we can read all data from the dlbuf. */
/* update the buffering status */ /* update the buffering status */
update_levels (dlbuf, dlbuf->max_level.bytes); update_levels (dlbuf, dlbuf->max_level.bytes);
/* update the buffering */
msg = update_buffering (dlbuf);
/* wakeup the waiter and let it recheck */ /* wakeup the waiter and let it recheck */
GST_DOWNLOAD_BUFFER_SIGNAL_ADD (dlbuf, -1); GST_DOWNLOAD_BUFFER_SIGNAL_ADD (dlbuf, -1);
break; break;
@ -1078,6 +1076,8 @@ gst_download_buffer_handle_sink_event (GstPad * pad, GstObject * parent,
} }
gst_event_unref (event); gst_event_unref (event);
GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf); GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf);
if (msg != NULL)
gst_element_post_message (GST_ELEMENT_CAST (dlbuf), msg);
} else { } else {
/* non-serialized events are passed upstream. */ /* non-serialized events are passed upstream. */
ret = gst_pad_push_event (dlbuf->srcpad, event); ret = gst_pad_push_event (dlbuf->srcpad, event);
@ -1127,6 +1127,7 @@ gst_download_buffer_chain (GstPad * pad, GstObject * parent, GstBuffer * buffer)
guint64 offset; guint64 offset;
gsize res, available; gsize res, available;
GError *error = NULL; GError *error = NULL;
GstMessage *msg = NULL;
dlbuf = GST_DOWNLOAD_BUFFER (parent); dlbuf = GST_DOWNLOAD_BUFFER (parent);
@ -1208,8 +1209,14 @@ gst_download_buffer_chain (GstPad * pad, GstObject * parent, GstBuffer * buffer)
update_levels (dlbuf, 0); update_levels (dlbuf, 0);
} }
/* update the buffering */
msg = update_buffering (dlbuf);
GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf); GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf);
if (msg != NULL)
gst_element_post_message (GST_ELEMENT_CAST (dlbuf), msg);
return GST_FLOW_OK; return GST_FLOW_OK;
/* ERRORS */ /* ERRORS */
@ -1251,12 +1258,18 @@ completed:
dlbuf->write_pos = dlbuf->upstream_size; dlbuf->write_pos = dlbuf->upstream_size;
dlbuf->filling = FALSE; dlbuf->filling = FALSE;
update_levels (dlbuf, dlbuf->max_level.bytes); update_levels (dlbuf, dlbuf->max_level.bytes);
msg = update_buffering (dlbuf);
gst_element_post_message (GST_ELEMENT_CAST (dlbuf), gst_element_post_message (GST_ELEMENT_CAST (dlbuf),
gst_message_new_element (GST_OBJECT_CAST (dlbuf), gst_message_new_element (GST_OBJECT_CAST (dlbuf),
gst_structure_new ("GstCacheDownloadComplete", gst_structure_new ("GstCacheDownloadComplete",
"location", G_TYPE_STRING, dlbuf->temp_location, NULL))); "location", G_TYPE_STRING, dlbuf->temp_location, NULL)));
GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf); GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf);
if (msg != NULL)
gst_element_post_message (GST_ELEMENT_CAST (dlbuf), msg);
return GST_FLOW_EOS; return GST_FLOW_EOS;
} }
} }
@ -1269,6 +1282,7 @@ gst_download_buffer_loop (GstPad * pad)
GstDownloadBuffer *dlbuf; GstDownloadBuffer *dlbuf;
GstFlowReturn ret; GstFlowReturn ret;
GstBuffer *buffer = NULL; GstBuffer *buffer = NULL;
GstMessage *msg = NULL;
dlbuf = GST_DOWNLOAD_BUFFER (GST_PAD_PARENT (pad)); dlbuf = GST_DOWNLOAD_BUFFER (GST_PAD_PARENT (pad));
@ -1288,9 +1302,15 @@ gst_download_buffer_loop (GstPad * pad)
if (ret != GST_FLOW_OK) if (ret != GST_FLOW_OK)
goto out_flushing; goto out_flushing;
/* update the buffering */
msg = update_buffering (dlbuf);
g_atomic_int_set (&dlbuf->downstream_may_block, 1); g_atomic_int_set (&dlbuf->downstream_may_block, 1);
GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf); GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf);
if (msg != NULL)
gst_element_post_message (GST_ELEMENT_CAST (dlbuf), msg);
ret = gst_pad_push (dlbuf->srcpad, buffer); ret = gst_pad_push (dlbuf->srcpad, buffer);
g_atomic_int_set (&dlbuf->downstream_may_block, 0); g_atomic_int_set (&dlbuf->downstream_may_block, 0);
@ -1589,14 +1609,21 @@ gst_download_buffer_get_range (GstPad * pad, GstObject * parent, guint64 offset,
{ {
GstDownloadBuffer *dlbuf; GstDownloadBuffer *dlbuf;
GstFlowReturn ret; GstFlowReturn ret;
GstMessage *msg = NULL;
dlbuf = GST_DOWNLOAD_BUFFER_CAST (parent); dlbuf = GST_DOWNLOAD_BUFFER_CAST (parent);
GST_DOWNLOAD_BUFFER_MUTEX_LOCK_CHECK (dlbuf, dlbuf->srcresult, out_flushing); GST_DOWNLOAD_BUFFER_MUTEX_LOCK_CHECK (dlbuf, dlbuf->srcresult, out_flushing);
/* FIXME - function will block when the range is not yet available */ /* FIXME - function will block when the range is not yet available */
ret = gst_download_buffer_read_buffer (dlbuf, offset, length, buffer); ret = gst_download_buffer_read_buffer (dlbuf, offset, length, buffer);
/* update the buffering */
msg = update_buffering (dlbuf);
GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf); GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf);
if (msg != NULL)
gst_element_post_message (GST_ELEMENT_CAST (dlbuf), msg);
return ret; return ret;
/* ERRORS */ /* ERRORS */
@ -1835,6 +1862,7 @@ gst_download_buffer_set_property (GObject * object,
guint prop_id, const GValue * value, GParamSpec * pspec) guint prop_id, const GValue * value, GParamSpec * pspec)
{ {
GstDownloadBuffer *dlbuf = GST_DOWNLOAD_BUFFER (object); GstDownloadBuffer *dlbuf = GST_DOWNLOAD_BUFFER (object);
GstMessage *msg = NULL;
/* someone could change levels here, and since this /* someone could change levels here, and since this
* affects the get/put funcs, we need to lock for safety. */ * affects the get/put funcs, we need to lock for safety. */
@ -1843,11 +1871,11 @@ gst_download_buffer_set_property (GObject * object,
switch (prop_id) { switch (prop_id) {
case PROP_MAX_SIZE_BYTES: case PROP_MAX_SIZE_BYTES:
dlbuf->max_level.bytes = g_value_get_uint (value); dlbuf->max_level.bytes = g_value_get_uint (value);
CAPACITY_CHANGE (dlbuf); msg = CAPACITY_CHANGE (dlbuf);
break; break;
case PROP_MAX_SIZE_TIME: case PROP_MAX_SIZE_TIME:
dlbuf->max_level.time = g_value_get_uint64 (value); dlbuf->max_level.time = g_value_get_uint64 (value);
CAPACITY_CHANGE (dlbuf); msg = CAPACITY_CHANGE (dlbuf);
break; break;
case PROP_LOW_PERCENT: case PROP_LOW_PERCENT:
dlbuf->low_percent = g_value_get_int (value); dlbuf->low_percent = g_value_get_int (value);
@ -1867,6 +1895,10 @@ gst_download_buffer_set_property (GObject * object,
} }
GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf); GST_DOWNLOAD_BUFFER_MUTEX_UNLOCK (dlbuf);
if (msg != NULL)
gst_element_post_message (GST_ELEMENT_CAST (dlbuf), msg);
} }
static void static void