mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-08 00:16:13 +00:00
4b9f2966fb
The test assumed that if a buffer has the same pointer address as before it is in fact the same mini object and has been re-used by the pool. This seems to be mostly true, but not always. The buffer might be destroyed and when a new buffer is created the allocator might return the same memory that we just freed. Instead attach a qdata with destroy notify function to buffer instances we want to track to make sure the buffer actually gets finalized rather than resurrected and put back into the pool.
325 lines
9.5 KiB
C
325 lines
9.5 KiB
C
/* GStreamer
|
|
* Copyright (C) 2014 Stefan Sauer <ensonic@users.sf.net>
|
|
*
|
|
* gstbufferpool.c: Unit test for GstBufferPool
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include <gst/check/gstcheck.h>
|
|
|
|
static GstBufferPool *
|
|
create_pool (guint size, guint min_buf, guint max_buf)
|
|
{
|
|
GstBufferPool *pool = gst_buffer_pool_new ();
|
|
GstStructure *conf = gst_buffer_pool_get_config (pool);
|
|
GstCaps *caps = gst_caps_new_empty_simple ("test/data");
|
|
|
|
gst_buffer_pool_config_set_params (conf, caps, size, min_buf, max_buf);
|
|
gst_buffer_pool_set_config (pool, conf);
|
|
gst_caps_unref (caps);
|
|
|
|
return pool;
|
|
}
|
|
|
|
static void
|
|
buffer_destroy_notify (gpointer ptr)
|
|
{
|
|
gint *counter = ptr;
|
|
|
|
GST_DEBUG ("buffer destroyed");
|
|
|
|
*counter += 1;
|
|
}
|
|
|
|
/* Track when a buffer is destroyed. The counter will be increased if the
|
|
* buffer is finalized (but not if it was re-surrected in dispose and put
|
|
* back into the buffer pool. */
|
|
static void
|
|
buffer_track_destroy (GstBuffer * buf, gint * counter)
|
|
{
|
|
gst_mini_object_set_qdata (GST_MINI_OBJECT (buf),
|
|
g_quark_from_static_string ("TestTracker"),
|
|
counter, buffer_destroy_notify);
|
|
}
|
|
|
|
GST_START_TEST (test_new_buffer_from_empty_pool)
|
|
{
|
|
GstBufferPool *pool = create_pool (10, 0, 0);
|
|
GstBuffer *buf = NULL;
|
|
|
|
gst_buffer_pool_set_active (pool, TRUE);
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
fail_if (buf == NULL, "acquiring buffer returned NULL");
|
|
|
|
gst_buffer_unref (buf);
|
|
gst_buffer_pool_set_active (pool, FALSE);
|
|
gst_object_unref (pool);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
|
|
GST_START_TEST (test_buffer_is_recycled)
|
|
{
|
|
GstBufferPool *pool = create_pool (10, 0, 0);
|
|
GstBuffer *buf = NULL, *prev;
|
|
gint dcount = 0;
|
|
|
|
gst_buffer_pool_set_active (pool, TRUE);
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
prev = buf;
|
|
buffer_track_destroy (buf, &dcount);
|
|
gst_buffer_unref (buf);
|
|
|
|
/* buffer should not have been freed, but have been recycled */
|
|
fail_unless (dcount == 0);
|
|
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
fail_unless (buf == prev, "got a fresh buffer instead of previous");
|
|
|
|
gst_buffer_unref (buf);
|
|
gst_buffer_pool_set_active (pool, FALSE);
|
|
gst_object_unref (pool);
|
|
|
|
/* buffer should now be gone */
|
|
fail_unless (dcount == 1);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
|
|
GST_START_TEST (test_buffer_out_of_order_reuse)
|
|
{
|
|
GstBufferPool *pool = create_pool (10, 0, 0);
|
|
GstBuffer *buf1 = NULL, *buf2 = NULL, *prev;
|
|
gint dcount1 = 0, dcount2 = 0;
|
|
|
|
gst_buffer_pool_set_active (pool, TRUE);
|
|
gst_buffer_pool_acquire_buffer (pool, &buf1, NULL);
|
|
buffer_track_destroy (buf1, &dcount1);
|
|
gst_buffer_pool_acquire_buffer (pool, &buf2, NULL);
|
|
buffer_track_destroy (buf2, &dcount2);
|
|
prev = buf2;
|
|
gst_buffer_unref (buf2);
|
|
|
|
/* buffer should not have been freed, but have been recycled */
|
|
fail_unless (dcount2 == 0);
|
|
|
|
gst_buffer_pool_acquire_buffer (pool, &buf2, NULL);
|
|
fail_unless (buf2 == prev, "got a fresh buffer instead of previous");
|
|
|
|
gst_buffer_unref (buf1);
|
|
gst_buffer_unref (buf2);
|
|
gst_buffer_pool_set_active (pool, FALSE);
|
|
gst_object_unref (pool);
|
|
|
|
fail_unless (dcount1 == 1);
|
|
fail_unless (dcount2 == 1);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
|
|
GST_START_TEST (test_pool_config_buffer_size)
|
|
{
|
|
GstBufferPool *pool = create_pool (10, 0, 0);
|
|
GstBuffer *buf = NULL;
|
|
|
|
gst_buffer_pool_set_active (pool, TRUE);
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
ck_assert_int_eq (gst_buffer_get_size (buf), 10);
|
|
|
|
gst_buffer_unref (buf);
|
|
gst_buffer_pool_set_active (pool, FALSE);
|
|
gst_object_unref (pool);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
|
|
GST_START_TEST (test_inactive_pool_returns_flushing)
|
|
{
|
|
GstBufferPool *pool = create_pool (10, 0, 0);
|
|
GstFlowReturn ret;
|
|
GstBuffer *buf = NULL;
|
|
|
|
ret = gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
ck_assert_int_eq (ret, GST_FLOW_FLUSHING);
|
|
|
|
gst_object_unref (pool);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_buffer_modify_discard)
|
|
{
|
|
|
|
GstBufferPool *pool = create_pool (10, 0, 0);
|
|
GstBuffer *buf = NULL, *prev;
|
|
GstMemory *mem;
|
|
gint dcount = 0;
|
|
|
|
gst_buffer_pool_set_active (pool, TRUE);
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
fail_unless (buf != NULL);
|
|
prev = buf;
|
|
buffer_track_destroy (buf, &dcount);
|
|
/* remove all memory, pool should not reuse this buffer */
|
|
gst_buffer_remove_all_memory (buf);
|
|
gst_buffer_unref (buf);
|
|
|
|
/* buffer should've been destroyed instead of going back into pool */
|
|
fail_unless_equals_int (dcount, 1);
|
|
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
prev = buf;
|
|
buffer_track_destroy (buf, &dcount);
|
|
/* do resize, pool should not reuse this buffer */
|
|
gst_buffer_resize (buf, 5, 2);
|
|
gst_buffer_unref (buf);
|
|
|
|
/* buffer should've been destroyed instead of going back into pool */
|
|
fail_unless_equals_int (dcount, 2);
|
|
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
prev = buf;
|
|
buffer_track_destroy (buf, &dcount);
|
|
/* keep ref to memory, not exclusive so pool should reuse this buffer */
|
|
mem = gst_buffer_get_memory (buf, 0);
|
|
gst_buffer_unref (buf);
|
|
gst_memory_unref (mem);
|
|
|
|
/* buffer should not have been destroyed and gone back into pool */
|
|
fail_unless_equals_int (dcount, 2);
|
|
|
|
gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
fail_unless (buf == prev, "got a fresh buffer instead of previous");
|
|
/* we're already did track_destroy on this buf, so no need to do it again */
|
|
mem = gst_buffer_get_memory (buf, 0);
|
|
/* exclusive lock so pool should not reuse this buffer */
|
|
gst_memory_lock (mem, GST_LOCK_FLAG_EXCLUSIVE);
|
|
gst_buffer_unref (buf);
|
|
gst_memory_unlock (mem, GST_LOCK_FLAG_EXCLUSIVE);
|
|
gst_memory_unref (mem);
|
|
|
|
/* buffer should have been destroyed and not gone back into pool because
|
|
* of the exclusive lock */
|
|
fail_unless_equals_int (dcount, 3);
|
|
|
|
gst_buffer_pool_set_active (pool, FALSE);
|
|
gst_object_unref (pool);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_pool_activation_and_config)
|
|
{
|
|
GstBufferPool *pool = gst_buffer_pool_new ();
|
|
GstStructure *config = gst_buffer_pool_get_config (pool);
|
|
GstCaps *caps = gst_caps_new_empty_simple ("test/data");
|
|
|
|
/* unconfigured pool cannot be activated */
|
|
fail_if (gst_buffer_pool_set_active (pool, TRUE));
|
|
|
|
gst_buffer_pool_config_set_params (config, caps, 10, 10, 0);
|
|
fail_unless (gst_buffer_pool_set_config (pool, config));
|
|
fail_unless (gst_buffer_pool_set_active (pool, TRUE));
|
|
|
|
/* setting the same config on an active pool is ok */
|
|
config = gst_buffer_pool_get_config (pool);
|
|
fail_unless (gst_buffer_pool_set_config (pool, config));
|
|
|
|
/* setting a different config on active pool should fail */
|
|
config = gst_buffer_pool_get_config (pool);
|
|
gst_buffer_pool_config_set_params (config, caps, 12, 10, 0);
|
|
fail_if (gst_buffer_pool_set_config (pool, config));
|
|
fail_unless (gst_buffer_pool_is_active (pool));
|
|
|
|
gst_buffer_pool_set_active (pool, FALSE);
|
|
gst_object_unref (pool);
|
|
gst_caps_unref (caps);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_pool_config_validate)
|
|
{
|
|
GstBufferPool *pool = create_pool (5, 4, 30);
|
|
GstStructure *config = gst_buffer_pool_get_config (pool);
|
|
GstCaps *caps = gst_caps_new_empty_simple ("test/data");
|
|
|
|
fail_unless (gst_buffer_pool_config_validate_params (config, caps, 5, 4, 0));
|
|
fail_unless (gst_buffer_pool_config_validate_params (config, caps, 5, 2, 0));
|
|
fail_unless (gst_buffer_pool_config_validate_params (config, caps, 4, 4, 0));
|
|
fail_if (gst_buffer_pool_config_validate_params (config, caps, 5, 6, 0));
|
|
|
|
gst_caps_unref (caps);
|
|
caps = gst_caps_new_empty_simple ("test/data2");
|
|
fail_if (gst_buffer_pool_config_validate_params (config, caps, 5, 4, 0));
|
|
|
|
gst_caps_unref (caps);
|
|
gst_structure_free (config);
|
|
gst_object_unref (pool);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_flushing_pool_returns_flushing)
|
|
{
|
|
GstBufferPool *pool = create_pool (10, 0, 0);
|
|
GstFlowReturn ret;
|
|
GstBuffer *buf = NULL;
|
|
|
|
gst_buffer_pool_set_active (pool, TRUE);
|
|
gst_buffer_pool_set_flushing (pool, TRUE);
|
|
|
|
ret = gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
ck_assert_int_eq (ret, GST_FLOW_FLUSHING);
|
|
|
|
gst_buffer_pool_set_flushing (pool, FALSE);
|
|
ret = gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
|
|
ck_assert_int_eq (ret, GST_FLOW_OK);
|
|
|
|
gst_buffer_unref (buf);
|
|
gst_object_unref (pool);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static Suite *
|
|
gst_buffer_pool_suite (void)
|
|
{
|
|
Suite *s = suite_create ("GstBufferPool");
|
|
TCase *tc_chain = tcase_create ("buffer_pool tests");
|
|
|
|
tcase_set_timeout (tc_chain, 0);
|
|
|
|
suite_add_tcase (s, tc_chain);
|
|
tcase_add_test (tc_chain, test_new_buffer_from_empty_pool);
|
|
tcase_add_test (tc_chain, test_buffer_is_recycled);
|
|
tcase_add_test (tc_chain, test_buffer_out_of_order_reuse);
|
|
tcase_add_test (tc_chain, test_pool_config_buffer_size);
|
|
tcase_add_test (tc_chain, test_inactive_pool_returns_flushing);
|
|
tcase_add_test (tc_chain, test_buffer_modify_discard);
|
|
tcase_add_test (tc_chain, test_pool_activation_and_config);
|
|
tcase_add_test (tc_chain, test_pool_config_validate);
|
|
tcase_add_test (tc_chain, test_flushing_pool_returns_flushing);
|
|
|
|
return s;
|
|
}
|
|
|
|
GST_CHECK_MAIN (gst_buffer_pool);
|