diff options
author | Tim-Philipp Müller <tim@centricular.net> | 2008-04-03 22:59:44 +0000 |
---|---|---|
committer | Tim-Philipp Müller <tim@centricular.net> | 2008-04-03 22:59:44 +0000 |
commit | 401159a255a6357de35bb2925179d6a1596f8903 (patch) | |
tree | 6aee145accc136c1cd4228d2132d5b3059a6e951 | |
parent | 27f764087c1222cd0ae1631bdb58098af2ab0e18 (diff) |
tests/check/: Add unit test for gdkpixbufsink element.
Original commit message from CVS:
* tests/check/Makefile.am:
* tests/check/elements/.cvsignore:
* tests/check/elements/gdkpixbufsink.c:
Add unit test for gdkpixbufsink element.
-rw-r--r-- | ChangeLog | 7 | ||||
-rw-r--r-- | tests/check/Makefile.am | 16 | ||||
-rw-r--r-- | tests/check/elements/.gitignore | 3 | ||||
-rw-r--r-- | tests/check/elements/gdkpixbufsink.c | 291 |
4 files changed, 316 insertions, 1 deletions
@@ -1,5 +1,12 @@ 2008-04-03 Tim-Philipp Müller <tim at centricular dot net> + * tests/check/Makefile.am: + * tests/check/elements/.cvsignore: + * tests/check/elements/gdkpixbufsink.c: + Add unit test for gdkpixbufsink element. + +2008-04-03 Tim-Philipp Müller <tim at centricular dot net> + * ext/gdk_pixbuf/Makefile.am: * ext/gdk_pixbuf/gstgdkpixbuf.c: (plugin_init): * ext/gdk_pixbuf/gstgdkpixbufsink.c: diff --git a/tests/check/Makefile.am b/tests/check/Makefile.am index bc1b3244..2e687569 100644 --- a/tests/check/Makefile.am +++ b/tests/check/Makefile.am @@ -26,6 +26,12 @@ else check_annodex = endif +if USE_GDK_PIXBUF +check_gdkpixbuf = elements/gdkpixbufsink +else +check_gdkpixbuf = +endif + if USE_SUNAUDIO check_sunaudio = elements/sunaudio else @@ -75,6 +81,7 @@ check_PROGRAMS = \ elements/videocrop \ elements/videofilter \ pipelines/simple-launch-lines \ + $(check_gdkpixbuf) \ $(check_sunaudio) \ $(check_taglib) \ $(check_wavpack) @@ -123,6 +130,15 @@ elements_sunaudio_LDADD = \ elements_videocrop_LDADD = $(LDADD) $(GST_BASE_LIBS) elements_videocrop_CFLAGS = $(GST_BASE_CFLAGS) $(CFLAGS) $(AM_CFLAGS) +# FIXME: configure should check for gdk-pixbuf not gtk +# only need video.h header, not the lib +elements_gdkpixbufsink_CFLAGS = \ + $(GST_PLUGINS_BASE_CFLAGS) $(CFLAGS) $(AM_CFLAGS) $(GTK_CFLAGS) +elements_gdkpixbufsink_LDADD = \ + $(LDADD) $(GTK_LIBS) + + pipelines_wavpack_LDADD = $(LDADD) $(GST_BASE_LIBS) pipelines_wavpack_CFLAGS = $(GST_BASE_CFLAGS) $(CFLAGS) $(AM_CFLAGS) + diff --git a/tests/check/elements/.gitignore b/tests/check/elements/.gitignore index 26531d99..4f6c15f1 100644 --- a/tests/check/elements/.gitignore +++ b/tests/check/elements/.gitignore @@ -12,9 +12,10 @@ audiopanorama autodetect avimux avisubtitle -equalizer cmmldec cmmlenc +equalizer +gdkpixbufsink icydemux id3demux id3v2mux diff --git a/tests/check/elements/gdkpixbufsink.c b/tests/check/elements/gdkpixbufsink.c new file mode 100644 index 00000000..242cf85f --- /dev/null +++ b/tests/check/elements/gdkpixbufsink.c @@ -0,0 +1,291 @@ +/* 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., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, 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 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; + GstStructure *s; + + fail_unless (msg->structure != NULL); + + s = msg->structure; + 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, GST_VIDEO_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, GST_VIDEO_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); |