summaryrefslogtreecommitdiff
path: root/tests/check/elements/textoverlay.c
diff options
context:
space:
mode:
Diffstat (limited to 'tests/check/elements/textoverlay.c')
-rw-r--r--tests/check/elements/textoverlay.c751
1 files changed, 0 insertions, 751 deletions
diff --git a/tests/check/elements/textoverlay.c b/tests/check/elements/textoverlay.c
deleted file mode 100644
index 63dd31d5..00000000
--- a/tests/check/elements/textoverlay.c
+++ /dev/null
@@ -1,751 +0,0 @@
-/* GStreamer unit tests for textoverlay
- *
- * Copyright (C) 2006 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.
- */
-
-#include <unistd.h>
-
-#include <gst/check/gstcheck.h>
-
-#define I420_Y_ROWSTRIDE(width) (GST_ROUND_UP_4(width))
-#define I420_U_ROWSTRIDE(width) (GST_ROUND_UP_8(width)/2)
-#define I420_V_ROWSTRIDE(width) ((GST_ROUND_UP_8(I420_Y_ROWSTRIDE(width)))/2)
-
-#define I420_Y_OFFSET(w,h) (0)
-#define I420_U_OFFSET(w,h) (I420_Y_OFFSET(w,h)+(I420_Y_ROWSTRIDE(w)*GST_ROUND_UP_2(h)))
-#define I420_V_OFFSET(w,h) (I420_U_OFFSET(w,h)+(I420_U_ROWSTRIDE(w)*GST_ROUND_UP_2(h)/2))
-
-#define I420_SIZE(w,h) (I420_V_OFFSET(w,h)+(I420_V_ROWSTRIDE(w)*GST_ROUND_UP_2(h)/2))
-
-#define WIDTH 240
-#define HEIGHT 120
-
-/* For ease of programming we use globals to keep refs for our floating
- * src and sink pads we create; otherwise we always have to do get_pad,
- * get_peer, and then remove references in every test function */
-static GstPad *myvideosrcpad, *mytextsrcpad, *mysinkpad;
-
-#define VIDEO_CAPS_STRING \
- "video/x-raw-yuv, " \
- "format = (fourcc) I420, " \
- "framerate = (fraction) 1/1, " \
- "width = (int) 240, " \
- "height = (int) 120"
-
-#define VIDEO_CAPS_TEMPLATE_STRING \
- "video/x-raw-yuv, " \
- "format = (fourcc) I420"
-
-static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
- GST_PAD_SINK,
- GST_PAD_ALWAYS,
- GST_STATIC_CAPS (VIDEO_CAPS_TEMPLATE_STRING)
- );
-static GstStaticPadTemplate text_srctemplate = GST_STATIC_PAD_TEMPLATE ("src",
- GST_PAD_SRC,
- GST_PAD_ALWAYS,
- GST_STATIC_CAPS ("text/plain")
- );
-
-static GstStaticPadTemplate video_srctemplate = GST_STATIC_PAD_TEMPLATE ("src",
- GST_PAD_SRC,
- GST_PAD_ALWAYS,
- GST_STATIC_CAPS (VIDEO_CAPS_TEMPLATE_STRING)
- );
-
-/* much like gst_check_setup_src_pad(), but with possibility to give a hint
- * which sink template of the element to use, if there are multiple ones */
-static GstPad *
-notgst_check_setup_src_pad2 (GstElement * element,
- GstStaticPadTemplate * template, GstCaps * caps,
- const gchar * sink_template_name)
-{
- GstPad *srcpad, *sinkpad;
-
- if (sink_template_name == NULL)
- sink_template_name = "sink";
-
- /* sending pad */
- srcpad = gst_pad_new_from_static_template (template, "src");
- GST_DEBUG_OBJECT (element, "setting up sending pad %p", srcpad);
- fail_if (srcpad == NULL, "Could not create a srcpad");
- ASSERT_OBJECT_REFCOUNT (srcpad, "srcpad", 1);
-
- if (!(sinkpad = gst_element_get_static_pad (element, sink_template_name)))
- sinkpad = gst_element_get_request_pad (element, sink_template_name);
- fail_if (sinkpad == NULL, "Could not get sink pad from %s",
- GST_ELEMENT_NAME (element));
- ASSERT_OBJECT_REFCOUNT (sinkpad, "sinkpad", 2);
- if (caps)
- fail_unless (gst_pad_set_caps (srcpad, caps));
- fail_unless (gst_pad_link (srcpad, sinkpad) == GST_PAD_LINK_OK,
- "Could not link source and %s sink pads", GST_ELEMENT_NAME (element));
- gst_object_unref (sinkpad); /* because we got it higher up */
- ASSERT_OBJECT_REFCOUNT (sinkpad, "sinkpad", 1);
-
- return srcpad;
-}
-
-static void
-notgst_check_teardown_src_pad2 (GstElement * element,
- const gchar * sink_template_name)
-{
- GstPad *srcpad, *sinkpad;
-
- if (sink_template_name == NULL)
- sink_template_name = "sink";
-
- /* clean up floating src pad */
- if (!(sinkpad = gst_element_get_static_pad (element, sink_template_name)))
- sinkpad = gst_element_get_request_pad (element, sink_template_name);
- ASSERT_OBJECT_REFCOUNT (sinkpad, "sinkpad", 2);
- srcpad = gst_pad_get_peer (sinkpad);
-
- gst_pad_unlink (srcpad, sinkpad);
-
- /* caps could have been set, make sure they get unset */
- gst_pad_set_caps (srcpad, NULL);
-
- /* pad refs held by both creator and this function (through _get) */
- ASSERT_OBJECT_REFCOUNT (sinkpad, "element sinkpad", 2);
- gst_object_unref (sinkpad);
- /* one more ref is held by element itself */
-
- /* pad refs held by both creator and this function (through _get_peer) */
- ASSERT_OBJECT_REFCOUNT (srcpad, "check srcpad", 2);
- gst_object_unref (srcpad);
- gst_object_unref (srcpad);
-}
-
-static GstElement *
-setup_textoverlay (gboolean video_only_no_text)
-{
- GstElement *textoverlay;
-
- GST_DEBUG ("setup_textoverlay");
- textoverlay = gst_check_setup_element ("textoverlay");
- mysinkpad = gst_check_setup_sink_pad (textoverlay, &sinktemplate, NULL);
- myvideosrcpad =
- notgst_check_setup_src_pad2 (textoverlay, &video_srctemplate, NULL,
- "video_sink");
-
- if (!video_only_no_text) {
- mytextsrcpad =
- notgst_check_setup_src_pad2 (textoverlay, &text_srctemplate, NULL,
- "text_sink");
- gst_pad_set_active (mytextsrcpad, TRUE);
- } else {
- mytextsrcpad = NULL;
- }
-
- gst_pad_set_active (myvideosrcpad, TRUE);
- gst_pad_set_active (mysinkpad, TRUE);
-
- return textoverlay;
-}
-
-static gboolean
-buffer_is_all_black (GstBuffer * buf)
-{
- GstStructure *s;
- gint x, y, w, h;
-
- fail_unless (buf != NULL);
- fail_unless (GST_BUFFER_CAPS (buf) != NULL);
- s = gst_caps_get_structure (GST_BUFFER_CAPS (buf), 0);
- fail_unless (s != NULL);
- fail_unless (gst_structure_get_int (s, "width", &w));
- fail_unless (gst_structure_get_int (s, "height", &h));
-
- for (y = 0; y < h; ++y) {
- guint8 *data = GST_BUFFER_DATA (buf) + (y * GST_ROUND_UP_4 (w));
-
- for (x = 0; x < w; ++x) {
- if (data[x] != 0x00) {
- GST_LOG ("non-black pixel at (x,y) %d,%d", x, y);
- return FALSE;
- }
- }
- }
-
- return TRUE;
-}
-
-static GstBuffer *
-create_black_buffer (const gchar * caps_string)
-{
- GstStructure *s;
- GstBuffer *buffer;
- GstCaps *caps;
- gint w, h, size;
-
- fail_unless (caps_string != NULL);
-
- caps = gst_caps_from_string (caps_string);
- fail_unless (caps != NULL);
- fail_unless (gst_caps_is_fixed (caps));
-
- s = gst_caps_get_structure (caps, 0);
- fail_unless (gst_structure_get_int (s, "width", &w));
- fail_unless (gst_structure_get_int (s, "height", &h));
-
- GST_LOG ("creating buffer (%dx%d)", w, h);
- size = I420_SIZE (w, h);
- buffer = gst_buffer_new_and_alloc (size);
- /* we're only checking the Y plane later, so just zero it all out,
- * even if it's not the blackest black there is */
- memset (GST_BUFFER_DATA (buffer), 0, size);
-
- gst_buffer_set_caps (buffer, caps);
- gst_caps_unref (caps);
-
- /* double check to make sure it's been created right */
- fail_unless (buffer_is_all_black (buffer));
-
- return buffer;
-}
-
-static GstBuffer *
-create_text_buffer (const gchar * txt, GstClockTime ts, GstClockTime duration)
-{
- GstBuffer *buffer;
- GstCaps *caps;
- guint txt_len;
-
- fail_unless (txt != NULL);
-
- txt_len = strlen (txt);
-
- buffer = gst_buffer_new_and_alloc (txt_len);
- memcpy (GST_BUFFER_DATA (buffer), txt, txt_len);
-
- GST_BUFFER_TIMESTAMP (buffer) = ts;
- GST_BUFFER_DURATION (buffer) = duration;
-
- caps = gst_caps_new_simple ("text/plain", NULL);
- gst_buffer_set_caps (buffer, caps);
- gst_caps_unref (caps);
-
- return buffer;
-}
-
-static void
-cleanup_textoverlay (GstElement * textoverlay)
-{
- GST_DEBUG ("cleanup_textoverlay");
-
- g_list_foreach (buffers, (GFunc) gst_mini_object_unref, NULL);
- g_list_free (buffers);
- buffers = NULL;
-
- gst_element_set_state (textoverlay, GST_STATE_NULL);
- gst_element_get_state (textoverlay, NULL, NULL, GST_CLOCK_TIME_NONE);
- gst_pad_set_active (myvideosrcpad, FALSE);
- gst_pad_set_active (mysinkpad, FALSE);
- notgst_check_teardown_src_pad2 (textoverlay, "video_sink");
- if (mytextsrcpad) {
- notgst_check_teardown_src_pad2 (textoverlay, "text_sink");
- }
- gst_check_teardown_sink_pad (textoverlay);
- gst_check_teardown_element (textoverlay);
-}
-
-GST_START_TEST (test_video_passthrough)
-{
- GstElement *textoverlay;
- GstBuffer *inbuffer;
-
- textoverlay = setup_textoverlay (TRUE);
- fail_unless (gst_element_set_state (textoverlay,
- GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS,
- "could not set to playing");
-
- inbuffer = create_black_buffer (VIDEO_CAPS_STRING);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
-
- /* ========== (1) video buffer without timestamp => should be dropped ==== */
-
- /* take additional ref to keep it alive */
- gst_buffer_ref (inbuffer);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
-
- /* pushing gives away one of the two references we have ... */
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* should have been discarded as out-of-segment since it has no timestamp */
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
- fail_unless_equals_int (g_list_length (buffers), 0);
-
- /* ========== (2) buffer with 0 timestamp => simple passthrough ========== */
-
- /* now try again, this time with timestamp (segment defaults to 0 start) */
- GST_BUFFER_TIMESTAMP (inbuffer) = 0;
- GST_BUFFER_DURATION (inbuffer) = GST_CLOCK_TIME_NONE;
-
- /* take additional ref to keep it alive */
- gst_buffer_ref (inbuffer);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
-
- /* pushing gives away one of the two references we have ... */
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* text pad is not linked, timestamp is in segment, no static text to
- * render, should have gone through right away without modification */
- fail_unless_equals_int (g_list_length (buffers), 1);
- fail_unless (GST_BUFFER_CAST (buffers->data) == inbuffer);
- fail_unless (buffer_is_all_black (inbuffer));
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
-
- /* and clean up */
- g_list_foreach (buffers, (GFunc) gst_mini_object_unref, NULL);
- g_list_free (buffers);
- buffers = NULL;
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
-
- /* ========== (3) buffer with 0 timestamp and no duration, with the
- * segment starting from 1sec => should be discarded */
-
- gst_pad_push_event (myvideosrcpad,
- gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_TIME, 1 * GST_SECOND,
- -1, 0));
-
- GST_BUFFER_TIMESTAMP (inbuffer) = 0;
- GST_BUFFER_DURATION (inbuffer) = GST_CLOCK_TIME_NONE;
-
- /* take additional ref to keep it alive */
- gst_buffer_ref (inbuffer);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
-
- /* pushing gives away one of the two references we have ... */
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* should have been discarded as out-of-segment */
- fail_unless_equals_int (g_list_length (buffers), 0);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
-
- /* ========== (4) buffer with 0 timestamp and small defined duration, with
- * segment starting from 1sec => should be discarded */
-
- gst_pad_push_event (myvideosrcpad,
- gst_event_new_new_segment (FALSE, 1.0, 1 * GST_FORMAT_TIME, GST_SECOND,
- -1, 0));
-
- GST_BUFFER_DURATION (inbuffer) = GST_SECOND / 10;
-
- /* take additional ref to keep it alive */
- gst_buffer_ref (inbuffer);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
-
- /* pushing gives away one of the two references we have ... */
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* should have been discareded as out-of-segment since it has no timestamp */
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
- fail_unless_equals_int (g_list_length (buffers), 0);
-
- /* ========== (5) buffer partially overlapping into the segment => should
- * be pushed through, but with adjusted stamp values */
-
- gst_pad_push_event (myvideosrcpad,
- gst_event_new_new_segment (FALSE, 1.0, 1 * GST_FORMAT_TIME, GST_SECOND,
- -1, 0));
-
- GST_BUFFER_TIMESTAMP (inbuffer) = GST_SECOND / 4;
- GST_BUFFER_DURATION (inbuffer) = GST_SECOND;
-
- /* take additional ref to keep it alive */
- gst_buffer_ref (inbuffer);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
-
- /* pushing gives away one of the two references we have ... */
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* should be the parent for a new subbuffer for the stamp fix-up */
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
- fail_unless_equals_int (g_list_length (buffers), 1);
- fail_unless (GST_BUFFER_CAST (buffers->data) != inbuffer);
- fail_unless (GST_BUFFER_TIMESTAMP (GST_BUFFER_CAST (buffers->data)) ==
- GST_SECOND);
- fail_unless (GST_BUFFER_DURATION (GST_BUFFER_CAST (buffers->data)) ==
- (GST_SECOND / 4));
- fail_unless (buffer_is_all_black (GST_BUFFER_CAST (buffers->data)));
- /* and clean up */
- g_list_foreach (buffers, (GFunc) gst_mini_object_unref, NULL);
- g_list_free (buffers);
- buffers = NULL;
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
-
- /* cleanup */
- cleanup_textoverlay (textoverlay);
- gst_buffer_unref (inbuffer);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_video_render_static_text)
-{
- GstElement *textoverlay;
- GstBuffer *inbuffer;
-
- textoverlay = setup_textoverlay (TRUE);
-
- /* set static text to render */
- g_object_set (textoverlay, "text", "XLX", NULL);
-
- fail_unless (gst_element_set_state (textoverlay,
- GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS,
- "could not set to playing");
-
- inbuffer = create_black_buffer (VIDEO_CAPS_STRING);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
-
- GST_BUFFER_TIMESTAMP (inbuffer) = 0;
- GST_BUFFER_DURATION (inbuffer) = GST_SECOND / 10;
-
- /* take additional ref to keep it alive */
- gst_buffer_ref (inbuffer);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
-
- /* pushing gives away one of the two references we have ... */
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* should have been dropped in favour of a new writable buffer */
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
- fail_unless_equals_int (g_list_length (buffers), 1);
- fail_unless (GST_BUFFER_CAST (buffers->data) != inbuffer);
-
- /* there should be text rendered */
- fail_unless (buffer_is_all_black (GST_BUFFER_CAST (buffers->data)) == FALSE);
-
- fail_unless (GST_BUFFER_TIMESTAMP (GST_BUFFER_CAST (buffers->data)) == 0);
- fail_unless (GST_BUFFER_DURATION (GST_BUFFER_CAST (buffers->data)) ==
- (GST_SECOND / 10));
-
- /* and clean up */
- g_list_foreach (buffers, (GFunc) gst_mini_object_unref, NULL);
- g_list_free (buffers);
- buffers = NULL;
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
-
- /* cleanup */
- cleanup_textoverlay (textoverlay);
- gst_buffer_unref (inbuffer);
-}
-
-GST_END_TEST;
-
-static gpointer
-test_video_waits_for_text_send_text_newsegment_thread (gpointer data)
-{
- g_usleep (1 * G_USEC_PER_SEC);
-
- /* send an update newsegment; the video buffer should now be pushed through
- * even though there is no text buffer queued at the moment */
- GST_INFO ("Sending newsegment update on text pad");
- gst_pad_push_event (mytextsrcpad,
- gst_event_new_new_segment (TRUE, 1.0, GST_FORMAT_TIME,
- 35 * GST_SECOND, -1, 35 * GST_SECOND));
-
- return NULL;
-}
-
-static gpointer
-test_video_waits_for_text_shutdown_element (gpointer data)
-{
- g_usleep (1 * G_USEC_PER_SEC);
-
- GST_INFO ("Trying to shut down textoverlay element ...");
- /* set to NULL state to make sure we can shut it down while it's
- * blocking in the video chain function waiting for a text buffer */
- gst_element_set_state (GST_ELEMENT (data), GST_STATE_NULL);
- GST_INFO ("Done.");
-
- return NULL;
-}
-
-GST_START_TEST (test_video_waits_for_text)
-{
- GstElement *textoverlay;
- GstBuffer *inbuffer, *tbuf;
- GThread *thread;
-
- textoverlay = setup_textoverlay (FALSE);
-
- fail_unless (gst_element_set_state (textoverlay,
- GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS,
- "could not set to playing");
-
- tbuf = create_text_buffer ("XLX", 1 * GST_SECOND, 5 * GST_SECOND);
- gst_buffer_ref (tbuf);
- ASSERT_BUFFER_REFCOUNT (tbuf, "tbuf", 2);
-
- GST_LOG ("pushing text buffer");
- fail_unless (gst_pad_push (mytextsrcpad, tbuf) == GST_FLOW_OK);
-
- /* it should be stuck in textoverlay until it gets a text buffer or a
- * newsegment event that indicates it's not needed any longer */
- fail_unless_equals_int (g_list_length (buffers), 0);
-
- inbuffer = create_black_buffer (VIDEO_CAPS_STRING);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
-
- GST_BUFFER_TIMESTAMP (inbuffer) = 0;
- GST_BUFFER_DURATION (inbuffer) = GST_SECOND / 2;
-
- /* take additional ref to keep it alive */
- gst_buffer_ref (inbuffer);
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 2);
-
- /* pushing gives away one of the two references we have ... */
- GST_LOG ("pushing video buffer 1");
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* video buffer should have gone through untainted, since the text is later */
- fail_unless_equals_int (g_list_length (buffers), 1);
-
- /* text should still be stuck in textoverlay */
- ASSERT_BUFFER_REFCOUNT (tbuf, "tbuf", 2);
-
- /* there should be no text rendered */
- fail_unless (buffer_is_all_black (GST_BUFFER_CAST (buffers->data)));
-
- /* now, another video buffer */
- inbuffer = gst_buffer_make_metadata_writable (inbuffer);
- GST_BUFFER_TIMESTAMP (inbuffer) = GST_SECOND;
- GST_BUFFER_DURATION (inbuffer) = GST_SECOND / 2;
-
- /* pushing gives away one of the two references we have ... */
- GST_LOG ("pushing video buffer 2");
- gst_buffer_ref (inbuffer);
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* video buffer should have gone right away, with text rendered on it */
- fail_unless_equals_int (g_list_length (buffers), 2);
-
- /* text should still be stuck in textoverlay */
- ASSERT_BUFFER_REFCOUNT (tbuf, "tbuf", 2);
-
- /* there should be text rendered */
- fail_unless (buffer_is_all_black (GST_BUFFER_CAST (buffers->next->data)) ==
- FALSE);
-
- /* a third video buffer */
- inbuffer = gst_buffer_make_metadata_writable (inbuffer);
- GST_BUFFER_TIMESTAMP (inbuffer) = 30 * GST_SECOND;
- GST_BUFFER_DURATION (inbuffer) = GST_SECOND / 2;
-
- /* video buffer #3: should not go through, it should discard the current
- * text buffer as too old and then wait for the next text buffer (or a
- * newsegment event to arrive); we spawn a background thread to send such
- * a newsegment event after a second or so so we get back control */
- thread =
- g_thread_create (test_video_waits_for_text_send_text_newsegment_thread,
- NULL, FALSE, NULL);
- fail_unless (thread != NULL);
-
- GST_LOG ("pushing video buffer 3");
- gst_buffer_ref (inbuffer);
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_OK);
-
- /* but the text should no longer be stuck in textoverlay */
- ASSERT_BUFFER_REFCOUNT (tbuf, "tbuf", 1);
-
- /* video buffer should have gone through after newsegment event */
- fail_unless_equals_int (g_list_length (buffers), 3);
-
- /* ... and there should not be any text rendered on it */
- fail_unless (buffer_is_all_black (GST_BUFFER_CAST (buffers->next->next->
- data)));
-
- /* a fourth video buffer */
- inbuffer = gst_buffer_make_metadata_writable (inbuffer);
- GST_BUFFER_TIMESTAMP (inbuffer) = 35 * GST_SECOND;
- GST_BUFFER_DURATION (inbuffer) = GST_SECOND;
-
- /* video buffer #4: should not go through, it should wait for the next
- * text buffer (or a newsegment event) to arrive; we spawn a background
- * thread to shut down the element while it's waiting to make sure that
- * works ok */
- thread = g_thread_create (test_video_waits_for_text_shutdown_element,
- textoverlay, FALSE, NULL);
- fail_unless (thread != NULL);
-
- GST_LOG ("pushing video buffer 4");
- gst_buffer_ref (inbuffer);
- fail_unless (gst_pad_push (myvideosrcpad, inbuffer) == GST_FLOW_WRONG_STATE);
-
- /* and clean up */
- g_list_foreach (buffers, (GFunc) gst_mini_object_unref, NULL);
- g_list_free (buffers);
- buffers = NULL;
- ASSERT_BUFFER_REFCOUNT (inbuffer, "inbuffer", 1);
-
- /* cleanup */
- cleanup_textoverlay (textoverlay);
- gst_buffer_unref (inbuffer);
-
- /* give up our ref, textoverlay should've cleared its queued buffer by now */
- ASSERT_BUFFER_REFCOUNT (tbuf, "tbuf", 1);
- gst_buffer_unref (tbuf);
-}
-
-GST_END_TEST;
-
-static gpointer
-test_render_continuity_push_video_buffers_thread (gpointer data)
-{
- /* push video buffers at 1fps */
- guint frame_count = 0;
-
- do {
- GstBuffer *vbuf;
-
- vbuf = create_black_buffer (VIDEO_CAPS_STRING);
- ASSERT_BUFFER_REFCOUNT (vbuf, "vbuf", 1);
-
- GST_BUFFER_TIMESTAMP (vbuf) = frame_count * GST_SECOND;
- GST_BUFFER_DURATION (vbuf) = GST_SECOND;
-
- /* pushing gives away one of the two references we have ... */
- GST_LOG ("pushing video buffer %u @ %" GST_TIME_FORMAT, frame_count,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (vbuf)));
- fail_unless (gst_pad_push (myvideosrcpad, vbuf) == GST_FLOW_OK);
-
- ++frame_count;
- } while (frame_count < 15);
-
- return NULL;
-}
-
-
-GST_START_TEST (test_render_continuity)
-{
- GThread *thread;
- GstElement *textoverlay;
- GstBuffer *tbuf;
-
- textoverlay = setup_textoverlay (FALSE);
-
- fail_unless (gst_element_set_state (textoverlay,
- GST_STATE_PLAYING) == GST_STATE_CHANGE_SUCCESS,
- "could not set to playing");
-
- thread = g_thread_create (test_render_continuity_push_video_buffers_thread,
- NULL, FALSE, NULL);
- fail_unless (thread != NULL);
-
- tbuf = create_text_buffer ("XLX", 2 * GST_SECOND, GST_SECOND);
- GST_LOG ("pushing text buffer @ %" GST_TIME_FORMAT,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (tbuf)));
- fail_unless (gst_pad_push (mytextsrcpad, tbuf) == GST_FLOW_OK);
-
- tbuf = create_text_buffer ("XLX", 3 * GST_SECOND, 2 * GST_SECOND);
- GST_LOG ("pushing text buffer @ %" GST_TIME_FORMAT,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (tbuf)));
- fail_unless (gst_pad_push (mytextsrcpad, tbuf) == GST_FLOW_OK);
-
- tbuf = create_text_buffer ("XLX", 7 * GST_SECOND, GST_SECOND);
- GST_LOG ("pushing text buffer @ %" GST_TIME_FORMAT,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (tbuf)));
- fail_unless (gst_pad_push (mytextsrcpad, tbuf) == GST_FLOW_OK);
-
- tbuf = create_text_buffer ("XLX", 8 * GST_SECOND, GST_SECOND);
- GST_LOG ("pushing text buffer @ %" GST_TIME_FORMAT,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (tbuf)));
- fail_unless (gst_pad_push (mytextsrcpad, tbuf) == GST_FLOW_OK);
-
- tbuf = create_text_buffer ("XLX", 9 * GST_SECOND, GST_SECOND);
- GST_LOG ("pushing text buffer @ %" GST_TIME_FORMAT,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (tbuf)));
- fail_unless (gst_pad_push (mytextsrcpad, tbuf) == GST_FLOW_OK);
-
- tbuf = create_text_buffer ("XLX", 10 * GST_SECOND, 30 * GST_SECOND);
- GST_LOG ("pushing text buffer @ %" GST_TIME_FORMAT,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (tbuf)));
- fail_unless (gst_pad_push (mytextsrcpad, tbuf) == GST_FLOW_OK);
-
- GST_LOG ("give the other thread some time to push through the remaining"
- "video buffers");
- g_usleep (G_USEC_PER_SEC);
- GST_LOG ("done");
-
- /* we should have 15 buffers each with one second length now */
- fail_unless_equals_int (g_list_length (buffers), 15);
-
- /* buffers 0 + 1 should be black */
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers, 0))));
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers, 1))));
-
- /* buffers 2 - 4 should have text */
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 2))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 3))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 4))) == FALSE);
-
- /* buffers 5 + 6 should be black */
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers, 5))));
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers, 6))));
-
- /* buffers 7 - last should have text */
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 7))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 8))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 9))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 10))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 11))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 12))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 13))) == FALSE);
- fail_unless (buffer_is_all_black (GST_BUFFER (g_list_nth_data (buffers,
- 14))) == FALSE);
-
- /* and clean up */
- g_list_foreach (buffers, (GFunc) gst_mini_object_unref, NULL);
- g_list_free (buffers);
- buffers = NULL;
-
- /* cleanup */
- cleanup_textoverlay (textoverlay);
-}
-
-GST_END_TEST;
-
-static Suite *
-textoverlay_suite (void)
-{
- Suite *s = suite_create ("textoverlay");
- TCase *tc_chain = tcase_create ("general");
-
- suite_add_tcase (s, tc_chain);
-
- tcase_add_test (tc_chain, test_video_passthrough);
- tcase_add_test (tc_chain, test_video_render_static_text);
- tcase_add_test (tc_chain, test_render_continuity);
- tcase_add_test (tc_chain, test_video_waits_for_text);
-
- return s;
-}
-
-GST_CHECK_MAIN (textoverlay);