294 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			294 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GStreamer unit test for the gdkpixbufsink element
 | |
|  * Copyright (C) 2008 Tim-Philipp Müller <tim centricular net>
 | |
|  *
 | |
|  * 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.
 | |
|  */
 | |
| 
 | |
| #ifdef HAVE_CONFIG_H
 | |
| # include "config.h"
 | |
| #endif
 | |
| 
 | |
| #include <gst/check/gstcheck.h>
 | |
| #include <gst/video/video.h>
 | |
| 
 | |
| #include <gdk-pixbuf/gdk-pixbuf.h>
 | |
| 
 | |
| #define CAPS_RGB "video/x-raw, format=RGB"
 | |
| #define CAPS_RGBA "video/x-raw, format=RGBA"
 | |
| #define WxH ",width=(int)319,height=(int)241"
 | |
| 
 | |
| #define N_BUFFERS 5
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
|   GstElement *pipe;
 | |
|   GstElement *src;
 | |
|   GstElement *filter;
 | |
|   GstElement *sink;
 | |
| } GstGdkPixbufSinkTestContext;
 | |
| 
 | |
| static void
 | |
| gdkpixbufsink_init_test_context (GstGdkPixbufSinkTestContext * ctx,
 | |
|     const gchar * filter_caps_string, gint num_buffers)
 | |
| {
 | |
|   GstCaps *caps;
 | |
| 
 | |
|   fail_unless (ctx != NULL);
 | |
| 
 | |
|   ctx->pipe = gst_pipeline_new ("pipeline");
 | |
|   fail_unless (ctx->pipe != NULL);
 | |
|   ctx->src = gst_element_factory_make ("videotestsrc", "src");
 | |
|   fail_unless (ctx->src != NULL, "Failed to create videotestsrc element");
 | |
|   ctx->filter = gst_element_factory_make ("capsfilter", "filter");
 | |
|   fail_unless (ctx->filter != NULL, "Failed to create capsfilter element");
 | |
|   ctx->sink = gst_element_factory_make ("gdkpixbufsink", "sink");
 | |
|   fail_unless (ctx->sink != NULL, "Failed to create gdkpixbufsink element");
 | |
| 
 | |
|   caps = gst_caps_from_string (filter_caps_string);
 | |
|   fail_unless (caps != NULL);
 | |
|   g_object_set (ctx->filter, "caps", caps, NULL);
 | |
|   gst_caps_unref (caps);
 | |
| 
 | |
|   if (num_buffers > 0)
 | |
|     g_object_set (ctx->src, "num-buffers", num_buffers, NULL);
 | |
| 
 | |
|   fail_unless (gst_bin_add (GST_BIN (ctx->pipe), ctx->src));
 | |
|   fail_unless (gst_bin_add (GST_BIN (ctx->pipe), ctx->filter));
 | |
|   fail_unless (gst_bin_add (GST_BIN (ctx->pipe), ctx->sink));
 | |
|   fail_unless (gst_element_link (ctx->src, ctx->filter));
 | |
|   fail_unless (gst_element_link (ctx->filter, ctx->sink));
 | |
| }
 | |
| 
 | |
| static void
 | |
| gdkpixbufsink_unset_test_context (GstGdkPixbufSinkTestContext * ctx)
 | |
| {
 | |
|   gst_element_set_state (ctx->pipe, GST_STATE_NULL);
 | |
|   gst_object_unref (ctx->pipe);
 | |
|   memset (ctx, 0, sizeof (GstGdkPixbufSinkTestContext));
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| check_last_pixbuf (GstGdkPixbufSinkTestContext * ctx, gpointer pixbuf)
 | |
| {
 | |
|   gpointer last_pb = NULL;
 | |
|   gboolean ret;
 | |
| 
 | |
|   g_object_get (ctx->sink, "last-pixbuf", &last_pb, NULL);
 | |
| 
 | |
|   ret = (last_pb == pixbuf);
 | |
| 
 | |
|   if (last_pb)
 | |
|     g_object_unref (last_pb);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| /* doesn't return a ref to the pixbuf */
 | |
| static GdkPixbuf *
 | |
| check_message_pixbuf (GstMessage * msg, const gchar * name, gint channels,
 | |
|     gboolean has_alpha)
 | |
| {
 | |
|   GdkPixbuf *pixbuf;
 | |
|   const GstStructure *s;
 | |
| 
 | |
|   fail_unless (gst_message_get_structure (msg) != NULL);
 | |
| 
 | |
|   s = gst_message_get_structure (msg);
 | |
|   fail_unless_equals_string (gst_structure_get_name (s), name);
 | |
| 
 | |
|   fail_unless (gst_structure_has_field (s, "pixbuf"));
 | |
|   fail_unless (gst_structure_has_field_typed (s, "pixel-aspect-ratio",
 | |
|           GST_TYPE_FRACTION));
 | |
|   pixbuf =
 | |
|       GDK_PIXBUF (g_value_get_object (gst_structure_get_value (s, "pixbuf")));
 | |
|   fail_unless (GDK_IS_PIXBUF (pixbuf));
 | |
|   fail_unless_equals_int (gdk_pixbuf_get_n_channels (pixbuf), channels);
 | |
|   fail_unless_equals_int (gdk_pixbuf_get_has_alpha (pixbuf), has_alpha);
 | |
|   fail_unless_equals_int (gdk_pixbuf_get_width (pixbuf), 319);
 | |
|   fail_unless_equals_int (gdk_pixbuf_get_height (pixbuf), 241);
 | |
| 
 | |
|   return pixbuf;
 | |
| }
 | |
| 
 | |
| GST_START_TEST (test_rgb)
 | |
| {
 | |
|   GstGdkPixbufSinkTestContext ctx;
 | |
|   GstMessage *msg;
 | |
|   GdkPixbuf *pixbuf;
 | |
|   GstBus *bus;
 | |
|   gint i;
 | |
| 
 | |
|   gdkpixbufsink_init_test_context (&ctx, CAPS_RGB WxH, N_BUFFERS);
 | |
| 
 | |
|   fail_unless (check_last_pixbuf (&ctx, NULL));
 | |
| 
 | |
|   /* start prerolling */
 | |
|   fail_unless_equals_int (gst_element_set_state (ctx.pipe, GST_STATE_PAUSED),
 | |
|       GST_STATE_CHANGE_ASYNC);
 | |
| 
 | |
|   /* wait until prerolled */
 | |
|   fail_unless_equals_int (gst_element_get_state (ctx.pipe, NULL, NULL, -1),
 | |
|       GST_STATE_CHANGE_SUCCESS);
 | |
| 
 | |
|   bus = gst_element_get_bus (ctx.pipe);
 | |
| 
 | |
|   /* find element message from our gdkpixbufsink, ignore element messages from
 | |
|    * other elemements (which just seems prudent to do, we don't expect any) */
 | |
|   while (1) {
 | |
|     msg = gst_bus_pop_filtered (bus, GST_MESSAGE_ELEMENT);
 | |
|     fail_if (msg == NULL, "Expected element message from gdkpixbufsink");
 | |
| 
 | |
|     if (msg->src == GST_OBJECT (ctx.sink))
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   pixbuf = check_message_pixbuf (msg, "preroll-pixbuf", 3, FALSE);
 | |
|   fail_unless (check_last_pixbuf (&ctx, pixbuf));
 | |
|   gst_message_unref (msg);
 | |
|   pixbuf = NULL;
 | |
| 
 | |
|   /* and go! */
 | |
|   fail_unless_equals_int (gst_element_set_state (ctx.pipe, GST_STATE_PLAYING),
 | |
|       GST_STATE_CHANGE_SUCCESS);
 | |
| 
 | |
|   /* This is racy, supposed to make sure locking and refcounting works at
 | |
|    * least to some extent */
 | |
|   for (i = 0; i < 10000; ++i) {
 | |
|     gpointer obj = NULL;
 | |
| 
 | |
|     g_object_get (ctx.sink, "last-pixbuf", &obj, NULL);
 | |
|     fail_unless (obj == NULL || GDK_IS_PIXBUF (obj));
 | |
|     if (obj)
 | |
|       g_object_unref (obj);
 | |
|   }
 | |
| 
 | |
|   /* there should be as many pixbuf messages as buffers */
 | |
|   for (i = 0; i < N_BUFFERS; ++i) {
 | |
|     /* find element message from our gdkpixbufsink, ignore element messages from
 | |
|      * other elemements (which just seems prudent to do, we don't expect any) */
 | |
|     while (1) {
 | |
|       msg = gst_bus_timed_pop_filtered (bus, -1, GST_MESSAGE_ELEMENT);
 | |
|       fail_if (msg == NULL, "Expected element message from gdkpixbufsink");
 | |
|       if (msg->src == GST_OBJECT (ctx.sink))
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     pixbuf = check_message_pixbuf (msg, "pixbuf", 3, FALSE);
 | |
|     gst_message_unref (msg);
 | |
|   }
 | |
| 
 | |
|   /* note: we don't hold a ref to pixbuf any longer here, but it should be ok */
 | |
|   fail_unless (check_last_pixbuf (&ctx, pixbuf));
 | |
|   pixbuf = NULL;
 | |
| 
 | |
|   gdkpixbufsink_unset_test_context (&ctx);
 | |
|   gst_object_unref (bus);
 | |
| }
 | |
| 
 | |
| GST_END_TEST;
 | |
| 
 | |
| GST_START_TEST (test_rgba)
 | |
| {
 | |
|   GstGdkPixbufSinkTestContext ctx;
 | |
|   GstMessage *msg;
 | |
|   GdkPixbuf *pixbuf;
 | |
|   GstBus *bus;
 | |
|   gint i;
 | |
| 
 | |
|   gdkpixbufsink_init_test_context (&ctx, CAPS_RGBA WxH, N_BUFFERS);
 | |
| 
 | |
|   fail_unless (check_last_pixbuf (&ctx, NULL));
 | |
| 
 | |
|   /* start prerolling */
 | |
|   fail_unless_equals_int (gst_element_set_state (ctx.pipe, GST_STATE_PAUSED),
 | |
|       GST_STATE_CHANGE_ASYNC);
 | |
| 
 | |
|   /* wait until prerolled */
 | |
|   fail_unless_equals_int (gst_element_get_state (ctx.pipe, NULL, NULL, -1),
 | |
|       GST_STATE_CHANGE_SUCCESS);
 | |
| 
 | |
|   bus = gst_element_get_bus (ctx.pipe);
 | |
| 
 | |
|   /* find element message from our gdkpixbufsink, ignore element messages from
 | |
|    * other elemements (which just seems prudent to do, we don't expect any) */
 | |
|   while (1) {
 | |
|     msg = gst_bus_pop_filtered (bus, GST_MESSAGE_ELEMENT);
 | |
|     fail_if (msg == NULL, "Expected element message from gdkpixbufsink");
 | |
| 
 | |
|     if (msg->src == GST_OBJECT (ctx.sink))
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   pixbuf = check_message_pixbuf (msg, "preroll-pixbuf", 4, TRUE);
 | |
|   fail_unless (check_last_pixbuf (&ctx, pixbuf));
 | |
|   gst_message_unref (msg);
 | |
|   pixbuf = NULL;
 | |
| 
 | |
|   /* and go! */
 | |
|   fail_unless_equals_int (gst_element_set_state (ctx.pipe, GST_STATE_PLAYING),
 | |
|       GST_STATE_CHANGE_SUCCESS);
 | |
| 
 | |
|   /* This is racy, supposed to make sure locking and refcounting works at
 | |
|    * least to some extent */
 | |
|   for (i = 0; i < 10000; ++i) {
 | |
|     gpointer obj = NULL;
 | |
| 
 | |
|     g_object_get (ctx.sink, "last-pixbuf", &obj, NULL);
 | |
|     fail_unless (obj == NULL || GDK_IS_PIXBUF (obj));
 | |
|     if (obj)
 | |
|       g_object_unref (obj);
 | |
|   }
 | |
| 
 | |
|   /* there should be as many pixbuf messages as buffers */
 | |
|   for (i = 0; i < N_BUFFERS; ++i) {
 | |
|     /* find element message from our gdkpixbufsink, ignore element messages from
 | |
|      * other elemements (which just seems prudent to do, we don't expect any) */
 | |
|     while (1) {
 | |
|       msg = gst_bus_timed_pop_filtered (bus, -1, GST_MESSAGE_ELEMENT);
 | |
|       fail_if (msg == NULL, "Expected element message from gdkpixbufsink");
 | |
|       if (msg->src == GST_OBJECT (ctx.sink))
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     pixbuf = check_message_pixbuf (msg, "pixbuf", 4, TRUE);
 | |
|     gst_message_unref (msg);
 | |
|   }
 | |
| 
 | |
|   /* note: we don't hold a ref to pixbuf any longer here, but it should be ok */
 | |
|   fail_unless (check_last_pixbuf (&ctx, pixbuf));
 | |
|   pixbuf = NULL;
 | |
| 
 | |
|   gdkpixbufsink_unset_test_context (&ctx);
 | |
|   gst_object_unref (bus);
 | |
| }
 | |
| 
 | |
| GST_END_TEST;
 | |
| 
 | |
| static Suite *
 | |
| gdkpixbufsink_suite (void)
 | |
| {
 | |
|   Suite *s = suite_create ("gdkpixbufsink");
 | |
|   TCase *tc_chain = tcase_create ("general");
 | |
| 
 | |
|   suite_add_tcase (s, tc_chain);
 | |
|   tcase_add_test (tc_chain, test_rgb);
 | |
|   tcase_add_test (tc_chain, test_rgba);
 | |
| 
 | |
|   return s;
 | |
| }
 | |
| 
 | |
| GST_CHECK_MAIN (gdkpixbufsink);
 |