mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-02 21:48:55 +00:00
rtpjitterbuffer: Properly free internal packets queue in finalize()
As we override the GLib item with our own structure, we cannot use any
function from GList or GQueue that would try to free the RTPJitterBufferItem.
In this patch, we move away from g_queue_new() which forces using
g_queue_free(). This this function could use g_slice_free() if there is any items
left in the queue. Passing the wrong size to GSLice may cause data corruption
and crash.
A better approach would be to use a proper intrusive linked list
implementation but that's left as an exercise for the next person
running into crashes caused by this.
Be ware that this regression was introduced 6 years ago in the following
commit [0], the call to flush() looked useless, as there was a g_queue_free()
afterward.
Signed-off-by: Nicolas Dufresne <nicolas.dufresne@collabora.com>
[0] 479c7642fd
Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-good/-/merge_requests/573>
This commit is contained in:
parent
ea1797ccb5
commit
7b22397cf5
2 changed files with 18 additions and 15 deletions
|
@ -87,7 +87,7 @@ rtp_jitter_buffer_init (RTPJitterBuffer * jbuf)
|
||||||
{
|
{
|
||||||
g_mutex_init (&jbuf->clock_lock);
|
g_mutex_init (&jbuf->clock_lock);
|
||||||
|
|
||||||
jbuf->packets = g_queue_new ();
|
g_queue_init (&jbuf->packets);
|
||||||
jbuf->mode = RTP_JITTER_BUFFER_MODE_SLAVE;
|
jbuf->mode = RTP_JITTER_BUFFER_MODE_SLAVE;
|
||||||
|
|
||||||
rtp_jitter_buffer_reset_skew (jbuf);
|
rtp_jitter_buffer_reset_skew (jbuf);
|
||||||
|
@ -112,7 +112,10 @@ rtp_jitter_buffer_finalize (GObject * object)
|
||||||
if (jbuf->pipeline_clock)
|
if (jbuf->pipeline_clock)
|
||||||
gst_object_unref (jbuf->pipeline_clock);
|
gst_object_unref (jbuf->pipeline_clock);
|
||||||
|
|
||||||
g_queue_free (jbuf->packets);
|
/* We cannot use g_queue_clear() as it would pass the wrong size to
|
||||||
|
* g_slice_free() which may lead to data corruption in the slice allocator.
|
||||||
|
*/
|
||||||
|
rtp_jitter_buffer_flush (jbuf, NULL, NULL);
|
||||||
|
|
||||||
g_mutex_clear (&jbuf->clock_lock);
|
g_mutex_clear (&jbuf->clock_lock);
|
||||||
|
|
||||||
|
@ -385,7 +388,7 @@ get_buffer_level (RTPJitterBuffer * jbuf)
|
||||||
guint64 level;
|
guint64 level;
|
||||||
|
|
||||||
/* first buffer with timestamp */
|
/* first buffer with timestamp */
|
||||||
high_buf = (RTPJitterBufferItem *) g_queue_peek_tail_link (jbuf->packets);
|
high_buf = (RTPJitterBufferItem *) g_queue_peek_tail_link (&jbuf->packets);
|
||||||
while (high_buf) {
|
while (high_buf) {
|
||||||
if (high_buf->dts != -1 || high_buf->pts != -1)
|
if (high_buf->dts != -1 || high_buf->pts != -1)
|
||||||
break;
|
break;
|
||||||
|
@ -393,7 +396,7 @@ get_buffer_level (RTPJitterBuffer * jbuf)
|
||||||
high_buf = (RTPJitterBufferItem *) g_list_previous (high_buf);
|
high_buf = (RTPJitterBufferItem *) g_list_previous (high_buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
low_buf = (RTPJitterBufferItem *) g_queue_peek_head_link (jbuf->packets);
|
low_buf = (RTPJitterBufferItem *) g_queue_peek_head_link (&jbuf->packets);
|
||||||
while (low_buf) {
|
while (low_buf) {
|
||||||
if (low_buf->dts != -1 || low_buf->pts != -1)
|
if (low_buf->dts != -1 || low_buf->pts != -1)
|
||||||
break;
|
break;
|
||||||
|
@ -678,7 +681,7 @@ no_skew:
|
||||||
static void
|
static void
|
||||||
queue_do_insert (RTPJitterBuffer * jbuf, GList * list, GList * item)
|
queue_do_insert (RTPJitterBuffer * jbuf, GList * list, GList * item)
|
||||||
{
|
{
|
||||||
GQueue *queue = jbuf->packets;
|
GQueue *queue = &jbuf->packets;
|
||||||
|
|
||||||
/* It's more likely that the packet was inserted at the tail of the queue */
|
/* It's more likely that the packet was inserted at the tail of the queue */
|
||||||
if (G_LIKELY (list)) {
|
if (G_LIKELY (list)) {
|
||||||
|
@ -999,7 +1002,7 @@ rtp_jitter_buffer_insert (RTPJitterBuffer * jbuf, RTPJitterBufferItem * item,
|
||||||
g_return_val_if_fail (jbuf != NULL, FALSE);
|
g_return_val_if_fail (jbuf != NULL, FALSE);
|
||||||
g_return_val_if_fail (item != NULL, FALSE);
|
g_return_val_if_fail (item != NULL, FALSE);
|
||||||
|
|
||||||
list = jbuf->packets->tail;
|
list = jbuf->packets.tail;
|
||||||
|
|
||||||
/* no seqnum, simply append then */
|
/* no seqnum, simply append then */
|
||||||
if (item->seqnum == -1)
|
if (item->seqnum == -1)
|
||||||
|
@ -1236,7 +1239,7 @@ rtp_jitter_buffer_pop (RTPJitterBuffer * jbuf, gint * percent)
|
||||||
|
|
||||||
g_return_val_if_fail (jbuf != NULL, NULL);
|
g_return_val_if_fail (jbuf != NULL, NULL);
|
||||||
|
|
||||||
queue = jbuf->packets;
|
queue = &jbuf->packets;
|
||||||
|
|
||||||
item = queue->head;
|
item = queue->head;
|
||||||
if (item) {
|
if (item) {
|
||||||
|
@ -1277,7 +1280,7 @@ rtp_jitter_buffer_peek (RTPJitterBuffer * jbuf)
|
||||||
{
|
{
|
||||||
g_return_val_if_fail (jbuf != NULL, NULL);
|
g_return_val_if_fail (jbuf != NULL, NULL);
|
||||||
|
|
||||||
return (RTPJitterBufferItem *) jbuf->packets->head;
|
return (RTPJitterBufferItem *) jbuf->packets.head;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1299,7 +1302,7 @@ rtp_jitter_buffer_flush (RTPJitterBuffer * jbuf, GFunc free_func,
|
||||||
if (free_func == NULL)
|
if (free_func == NULL)
|
||||||
free_func = (GFunc) rtp_jitter_buffer_free_item;
|
free_func = (GFunc) rtp_jitter_buffer_free_item;
|
||||||
|
|
||||||
while ((item = g_queue_pop_head_link (jbuf->packets)))
|
while ((item = g_queue_pop_head_link (&jbuf->packets)))
|
||||||
free_func ((RTPJitterBufferItem *) item, user_data);
|
free_func ((RTPJitterBufferItem *) item, user_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1371,7 +1374,7 @@ rtp_jitter_buffer_num_packets (RTPJitterBuffer * jbuf)
|
||||||
{
|
{
|
||||||
g_return_val_if_fail (jbuf != NULL, 0);
|
g_return_val_if_fail (jbuf != NULL, 0);
|
||||||
|
|
||||||
return jbuf->packets->length;
|
return jbuf->packets.length;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1392,8 +1395,8 @@ rtp_jitter_buffer_get_ts_diff (RTPJitterBuffer * jbuf)
|
||||||
|
|
||||||
g_return_val_if_fail (jbuf != NULL, 0);
|
g_return_val_if_fail (jbuf != NULL, 0);
|
||||||
|
|
||||||
high_buf = (RTPJitterBufferItem *) g_queue_peek_tail_link (jbuf->packets);
|
high_buf = (RTPJitterBufferItem *) g_queue_peek_tail_link (&jbuf->packets);
|
||||||
low_buf = (RTPJitterBufferItem *) g_queue_peek_head_link (jbuf->packets);
|
low_buf = (RTPJitterBufferItem *) g_queue_peek_head_link (&jbuf->packets);
|
||||||
|
|
||||||
if (!high_buf || !low_buf || high_buf == low_buf)
|
if (!high_buf || !low_buf || high_buf == low_buf)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1429,8 +1432,8 @@ rtp_jitter_buffer_get_seqnum_diff (RTPJitterBuffer * jbuf)
|
||||||
|
|
||||||
g_return_val_if_fail (jbuf != NULL, 0);
|
g_return_val_if_fail (jbuf != NULL, 0);
|
||||||
|
|
||||||
high_buf = (RTPJitterBufferItem *) g_queue_peek_tail_link (jbuf->packets);
|
high_buf = (RTPJitterBufferItem *) g_queue_peek_tail_link (&jbuf->packets);
|
||||||
low_buf = (RTPJitterBufferItem *) g_queue_peek_head_link (jbuf->packets);
|
low_buf = (RTPJitterBufferItem *) g_queue_peek_head_link (&jbuf->packets);
|
||||||
|
|
||||||
while (high_buf && high_buf->seqnum == -1)
|
while (high_buf && high_buf->seqnum == -1)
|
||||||
high_buf = (RTPJitterBufferItem *) high_buf->prev;
|
high_buf = (RTPJitterBufferItem *) high_buf->prev;
|
||||||
|
|
|
@ -73,7 +73,7 @@ GType rtp_jitter_buffer_mode_get_type (void);
|
||||||
struct _RTPJitterBuffer {
|
struct _RTPJitterBuffer {
|
||||||
GObject object;
|
GObject object;
|
||||||
|
|
||||||
GQueue *packets;
|
GQueue packets;
|
||||||
|
|
||||||
RTPJitterBufferMode mode;
|
RTPJitterBufferMode mode;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue