mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-18 05:16:05 +00:00
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:
parent
4093fc621a
commit
618cd5e65c
1 changed files with 50 additions and 18 deletions
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue