• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* GStreamer
2  * Copyright (C) 2014 Stefan Sauer <ensonic@users.sf.net>
3  *
4  * gstbufferpool.c: Unit test for GstBufferPool
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public
17  * License along with this library; if not, write to the
18  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  */
21 #ifdef HAVE_CONFIG_H
22 #include "config.h"
23 #endif
24 
25 #include <gst/check/gstcheck.h>
26 
27 static GstBufferPool *
create_pool(guint size,guint min_buf,guint max_buf)28 create_pool (guint size, guint min_buf, guint max_buf)
29 {
30   GstBufferPool *pool = gst_buffer_pool_new ();
31   GstStructure *conf = gst_buffer_pool_get_config (pool);
32   GstCaps *caps = gst_caps_new_empty_simple ("test/data");
33 
34   gst_buffer_pool_config_set_params (conf, caps, size, min_buf, max_buf);
35   gst_buffer_pool_set_config (pool, conf);
36   gst_caps_unref (caps);
37 
38   return pool;
39 }
40 
41 static void
buffer_destroy_notify(gpointer ptr)42 buffer_destroy_notify (gpointer ptr)
43 {
44   gint *counter = ptr;
45 
46   GST_DEBUG ("buffer destroyed");
47 
48   *counter += 1;
49 }
50 
51 /* Track when a buffer is destroyed. The counter will be increased if the
52  * buffer is finalized (but not if it was re-surrected in dispose and put
53  * back into the buffer pool. */
54 static void
buffer_track_destroy(GstBuffer * buf,gint * counter)55 buffer_track_destroy (GstBuffer * buf, gint * counter)
56 {
57   gst_mini_object_set_qdata (GST_MINI_OBJECT (buf),
58       g_quark_from_static_string ("TestTracker"),
59       counter, buffer_destroy_notify);
60 }
61 
GST_START_TEST(test_new_buffer_from_empty_pool)62 GST_START_TEST (test_new_buffer_from_empty_pool)
63 {
64   GstBufferPool *pool = create_pool (10, 0, 0);
65   GstBuffer *buf = NULL;
66 
67   gst_buffer_pool_set_active (pool, TRUE);
68   gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
69   fail_if (buf == NULL, "acquiring buffer returned NULL");
70 
71   gst_buffer_unref (buf);
72   gst_buffer_pool_set_active (pool, FALSE);
73   gst_object_unref (pool);
74 }
75 
76 GST_END_TEST;
77 
78 
GST_START_TEST(test_buffer_is_recycled)79 GST_START_TEST (test_buffer_is_recycled)
80 {
81   GstBufferPool *pool = create_pool (10, 0, 0);
82   GstBuffer *buf = NULL, *prev;
83   gint dcount = 0;
84 
85   gst_buffer_pool_set_active (pool, TRUE);
86   gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
87   prev = buf;
88   buffer_track_destroy (buf, &dcount);
89   gst_buffer_unref (buf);
90 
91   /* buffer should not have been freed, but have been recycled */
92   fail_unless (dcount == 0);
93 
94   gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
95   fail_unless (buf == prev, "got a fresh buffer instead of previous");
96 
97   gst_buffer_unref (buf);
98   gst_buffer_pool_set_active (pool, FALSE);
99   gst_object_unref (pool);
100 
101   /* buffer should now be gone */
102   fail_unless (dcount == 1);
103 }
104 
105 GST_END_TEST;
106 
107 
GST_START_TEST(test_buffer_out_of_order_reuse)108 GST_START_TEST (test_buffer_out_of_order_reuse)
109 {
110   GstBufferPool *pool = create_pool (10, 0, 0);
111   GstBuffer *buf1 = NULL, *buf2 = NULL, *prev;
112   gint dcount1 = 0, dcount2 = 0;
113 
114   gst_buffer_pool_set_active (pool, TRUE);
115   gst_buffer_pool_acquire_buffer (pool, &buf1, NULL);
116   buffer_track_destroy (buf1, &dcount1);
117   gst_buffer_pool_acquire_buffer (pool, &buf2, NULL);
118   buffer_track_destroy (buf2, &dcount2);
119   prev = buf2;
120   gst_buffer_unref (buf2);
121 
122   /* buffer should not have been freed, but have been recycled */
123   fail_unless (dcount2 == 0);
124 
125   gst_buffer_pool_acquire_buffer (pool, &buf2, NULL);
126   fail_unless (buf2 == prev, "got a fresh buffer instead of previous");
127 
128   gst_buffer_unref (buf1);
129   gst_buffer_unref (buf2);
130   gst_buffer_pool_set_active (pool, FALSE);
131   gst_object_unref (pool);
132 
133   fail_unless (dcount1 == 1);
134   fail_unless (dcount2 == 1);
135 }
136 
137 GST_END_TEST;
138 
139 
GST_START_TEST(test_pool_config_buffer_size)140 GST_START_TEST (test_pool_config_buffer_size)
141 {
142   GstBufferPool *pool = create_pool (10, 0, 0);
143   GstBuffer *buf = NULL;
144 
145   gst_buffer_pool_set_active (pool, TRUE);
146   gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
147   ck_assert_int_eq (gst_buffer_get_size (buf), 10);
148 
149   gst_buffer_unref (buf);
150   gst_buffer_pool_set_active (pool, FALSE);
151   gst_object_unref (pool);
152 }
153 
154 GST_END_TEST;
155 
156 
GST_START_TEST(test_inactive_pool_returns_flushing)157 GST_START_TEST (test_inactive_pool_returns_flushing)
158 {
159   GstBufferPool *pool = create_pool (10, 0, 0);
160   GstFlowReturn ret;
161   GstBuffer *buf = NULL;
162 
163   ret = gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
164   ck_assert_int_eq (ret, GST_FLOW_FLUSHING);
165 
166   gst_object_unref (pool);
167 }
168 
169 GST_END_TEST;
170 
GST_START_TEST(test_buffer_modify_discard)171 GST_START_TEST (test_buffer_modify_discard)
172 {
173 
174   GstBufferPool *pool = create_pool (10, 0, 0);
175   GstBuffer *buf = NULL, *prev;
176   GstMemory *mem;
177   gint dcount = 0;
178 
179   gst_buffer_pool_set_active (pool, TRUE);
180   gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
181   fail_unless (buf != NULL);
182   buffer_track_destroy (buf, &dcount);
183   /* remove all memory, pool should not reuse this buffer */
184   gst_buffer_remove_all_memory (buf);
185   gst_buffer_unref (buf);
186 
187   /* buffer should've been destroyed instead of going back into pool */
188   fail_unless_equals_int (dcount, 1);
189 
190   gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
191   buffer_track_destroy (buf, &dcount);
192   /* do resize, as we didn't modify the memory, pool should reuse this buffer */
193   gst_buffer_resize (buf, 8, 2);
194   gst_buffer_unref (buf);
195 
196   /* buffer should've gone back into pool */
197   fail_unless_equals_int (dcount, 1);
198 
199   gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
200   prev = buf;
201   fail_unless (buf == prev, "got a fresh buffer instead of previous");
202   /* keep ref to memory, not exclusive so pool should reuse this buffer */
203   mem = gst_buffer_get_memory (buf, 0);
204   gst_buffer_unref (buf);
205   gst_memory_unref (mem);
206 
207   /* buffer should not have been destroyed and gone back into pool */
208   fail_unless_equals_int (dcount, 1);
209 
210   gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
211   fail_unless (buf == prev, "got a fresh buffer instead of previous");
212   /* we're already did track_destroy on this buf, so no need to do it again */
213   mem = gst_buffer_get_memory (buf, 0);
214   /* exclusive lock so pool should not reuse this buffer */
215   gst_memory_lock (mem, GST_LOCK_FLAG_EXCLUSIVE);
216   gst_buffer_unref (buf);
217   gst_memory_unlock (mem, GST_LOCK_FLAG_EXCLUSIVE);
218   gst_memory_unref (mem);
219 
220   /* buffer should have been destroyed and not gone back into pool because
221    * of the exclusive lock */
222   fail_unless_equals_int (dcount, 2);
223 
224   gst_buffer_pool_set_active (pool, FALSE);
225   gst_object_unref (pool);
226 }
227 
228 GST_END_TEST;
229 
GST_START_TEST(test_pool_activation_and_config)230 GST_START_TEST (test_pool_activation_and_config)
231 {
232   GstBufferPool *pool = gst_buffer_pool_new ();
233   GstStructure *config = gst_buffer_pool_get_config (pool);
234   GstCaps *caps = gst_caps_new_empty_simple ("test/data");
235 
236   /* unconfigured pool cannot be activated */
237   fail_if (gst_buffer_pool_set_active (pool, TRUE));
238 
239   gst_buffer_pool_config_set_params (config, caps, 10, 10, 0);
240   fail_unless (gst_buffer_pool_set_config (pool, config));
241   fail_unless (gst_buffer_pool_set_active (pool, TRUE));
242 
243   /* setting the same config on an active pool is ok */
244   config = gst_buffer_pool_get_config (pool);
245   fail_unless (gst_buffer_pool_set_config (pool, config));
246 
247   /* setting a different config on active pool should fail */
248   config = gst_buffer_pool_get_config (pool);
249   gst_buffer_pool_config_set_params (config, caps, 12, 10, 0);
250   fail_if (gst_buffer_pool_set_config (pool, config));
251   fail_unless (gst_buffer_pool_is_active (pool));
252 
253   gst_buffer_pool_set_active (pool, FALSE);
254   gst_object_unref (pool);
255   gst_caps_unref (caps);
256 }
257 
258 GST_END_TEST;
259 
GST_START_TEST(test_pool_config_validate)260 GST_START_TEST (test_pool_config_validate)
261 {
262   GstBufferPool *pool = create_pool (5, 4, 30);
263   GstStructure *config = gst_buffer_pool_get_config (pool);
264   GstCaps *caps = gst_caps_new_empty_simple ("test/data");
265 
266   fail_unless (gst_buffer_pool_config_validate_params (config, caps, 5, 4, 0));
267   fail_unless (gst_buffer_pool_config_validate_params (config, caps, 5, 2, 0));
268   fail_unless (gst_buffer_pool_config_validate_params (config, caps, 4, 4, 0));
269   fail_if (gst_buffer_pool_config_validate_params (config, caps, 5, 6, 0));
270 
271   gst_caps_unref (caps);
272   caps = gst_caps_new_empty_simple ("test/data2");
273   fail_if (gst_buffer_pool_config_validate_params (config, caps, 5, 4, 0));
274 
275   gst_caps_unref (caps);
276   gst_structure_free (config);
277   gst_object_unref (pool);
278 }
279 
280 GST_END_TEST;
281 
GST_START_TEST(test_flushing_pool_returns_flushing)282 GST_START_TEST (test_flushing_pool_returns_flushing)
283 {
284   GstBufferPool *pool = create_pool (10, 0, 0);
285   GstFlowReturn ret;
286   GstBuffer *buf = NULL;
287 
288   gst_buffer_pool_set_active (pool, TRUE);
289   gst_buffer_pool_set_flushing (pool, TRUE);
290 
291   ret = gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
292   ck_assert_int_eq (ret, GST_FLOW_FLUSHING);
293 
294   gst_buffer_pool_set_flushing (pool, FALSE);
295   ret = gst_buffer_pool_acquire_buffer (pool, &buf, NULL);
296   ck_assert_int_eq (ret, GST_FLOW_OK);
297 
298   gst_buffer_unref (buf);
299   gst_object_unref (pool);
300 }
301 
302 GST_END_TEST;
303 
304 static gpointer
unref_buf(gpointer p)305 unref_buf (gpointer p)
306 {
307   GstBuffer *buf = (GstBuffer *) p;
308   /* remove all memory, pool should not reuse this buffer */
309   gst_buffer_remove_all_memory (buf);
310   gst_buffer_unref (buf);
311   return NULL;
312 }
313 
GST_START_TEST(test_no_deadlock_for_buffer_discard)314 GST_START_TEST (test_no_deadlock_for_buffer_discard)
315 {
316   GstBufferPool *pool;
317   GstBuffer *buf1, *buf2;
318   GThread *thread;
319 
320   pool = create_pool (1, 1, 1);
321   fail_unless (pool);
322   gst_buffer_pool_set_active (pool, TRUE);
323 
324   fail_unless (gst_buffer_pool_acquire_buffer (pool, &buf1,
325           NULL) == GST_FLOW_OK);
326   thread = g_thread_new (NULL, unref_buf, buf1);
327   fail_unless (thread);
328   /* we will be blocked here until buf1 unrefed */
329   fail_unless (gst_buffer_pool_acquire_buffer (pool, &buf2,
330           NULL) == GST_FLOW_OK);
331 
332   gst_buffer_unref (buf2);
333   g_thread_join (thread);
334   gst_object_unref (pool);
335 }
336 
337 GST_END_TEST;
338 
339 static Suite *
gst_buffer_pool_suite(void)340 gst_buffer_pool_suite (void)
341 {
342   Suite *s = suite_create ("GstBufferPool");
343   TCase *tc_chain = tcase_create ("buffer_pool tests");
344 
345   tcase_set_timeout (tc_chain, 0);
346 
347   suite_add_tcase (s, tc_chain);
348   tcase_add_test (tc_chain, test_new_buffer_from_empty_pool);
349   tcase_add_test (tc_chain, test_buffer_is_recycled);
350   tcase_add_test (tc_chain, test_buffer_out_of_order_reuse);
351   tcase_add_test (tc_chain, test_pool_config_buffer_size);
352   tcase_add_test (tc_chain, test_inactive_pool_returns_flushing);
353   tcase_add_test (tc_chain, test_buffer_modify_discard);
354   tcase_add_test (tc_chain, test_pool_activation_and_config);
355   tcase_add_test (tc_chain, test_pool_config_validate);
356   tcase_add_test (tc_chain, test_flushing_pool_returns_flushing);
357   tcase_add_test (tc_chain, test_no_deadlock_for_buffer_discard);
358 
359   return s;
360 }
361 
362 GST_CHECK_MAIN (gst_buffer_pool);
363