|
Packit |
971217 |
/* GStreamer unit tests for streamsynchronizer
|
|
Packit |
971217 |
*
|
|
Packit |
971217 |
* Copyright (C) 2012 Edward Hervey <edward@collabora.com>, Collabora Ltd
|
|
Packit |
971217 |
*
|
|
Packit |
971217 |
* This library is free software; you can redistribute it and/or
|
|
Packit |
971217 |
* modify it under the terms of the GNU Library General Public
|
|
Packit |
971217 |
* License as published by the Free Software Foundation; either
|
|
Packit |
971217 |
* version 2 of the License, or (at your option) any later version.
|
|
Packit |
971217 |
*
|
|
Packit |
971217 |
* This library is distributed in the hope that it will be useful,
|
|
Packit |
971217 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
971217 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Packit |
971217 |
* Library General Public License for more details.
|
|
Packit |
971217 |
*
|
|
Packit |
971217 |
* You should have received a copy of the GNU Library General Public
|
|
Packit |
971217 |
* License along with this library; if not, write to the
|
|
Packit |
971217 |
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
Packit |
971217 |
* Boston, MA 02110-1301, USA.
|
|
Packit |
971217 |
*/
|
|
Packit |
971217 |
|
|
Packit |
971217 |
#ifdef HAVE_CONFIG_H
|
|
Packit |
971217 |
# include <config.h>
|
|
Packit |
971217 |
#endif
|
|
Packit |
971217 |
|
|
Packit |
971217 |
#define GLIB_DISABLE_DEPRECATION_WARNINGS
|
|
Packit |
971217 |
|
|
Packit |
971217 |
#undef GST_CAT_DEFAULT
|
|
Packit |
971217 |
#include <gst/check/gstcheck.h>
|
|
Packit |
971217 |
|
|
Packit |
971217 |
static GstStaticPadTemplate mysinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
|
|
Packit |
971217 |
GST_PAD_SINK,
|
|
Packit |
971217 |
GST_PAD_ALWAYS,
|
|
Packit |
971217 |
GST_STATIC_CAPS_ANY);
|
|
Packit |
971217 |
static GstStaticPadTemplate mysrctemplate = GST_STATIC_PAD_TEMPLATE ("src",
|
|
Packit |
971217 |
GST_PAD_SRC,
|
|
Packit |
971217 |
GST_PAD_ALWAYS,
|
|
Packit |
971217 |
GST_STATIC_CAPS_ANY);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
typedef struct
|
|
Packit |
971217 |
{
|
|
Packit |
971217 |
GstPad *pad;
|
|
Packit |
971217 |
GList *to_push;
|
|
Packit |
971217 |
} MyPushInfo;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
GMutex push_mutex;
|
|
Packit |
971217 |
GCond push_cond;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
static GstPad *
|
|
Packit |
971217 |
get_other_pad (GstPad * pad)
|
|
Packit |
971217 |
{
|
|
Packit |
971217 |
GstIterator *it;
|
|
Packit |
971217 |
GValue item = G_VALUE_INIT;
|
|
Packit |
971217 |
GstPad *otherpad;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
it = gst_pad_iterate_internal_links (pad);
|
|
Packit |
971217 |
fail_unless (gst_iterator_next (it, &item) == GST_ITERATOR_OK);
|
|
Packit |
971217 |
otherpad = g_value_dup_object (&item);
|
|
Packit |
971217 |
g_value_unset (&item);
|
|
Packit |
971217 |
gst_iterator_free (it);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
return otherpad;
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
static GstFlowReturn
|
|
Packit |
971217 |
my_sink_chain (GstPad * pad, GstObject * parent, GstBuffer * buf)
|
|
Packit |
971217 |
{
|
|
Packit |
971217 |
GList **expected = GST_PAD_ELEMENT_PRIVATE (pad);
|
|
Packit |
971217 |
GList *next;
|
|
Packit |
971217 |
GstBuffer *exp;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
fail_if (*expected == NULL,
|
|
Packit |
971217 |
"streamsynchronizer pushed a buffer/event but we didn't expect any");
|
|
Packit |
971217 |
|
|
Packit |
971217 |
next = (*expected)->next;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
fail_if (GST_IS_EVENT ((*expected)->data),
|
|
Packit |
971217 |
"Expected an event (%s) but got a buffer instead",
|
|
Packit |
971217 |
GST_EVENT_TYPE_NAME (GST_EVENT ((*expected)->data)));
|
|
Packit |
971217 |
|
|
Packit |
971217 |
exp = GST_BUFFER ((*expected)->data);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
fail_unless_equals_uint64 (GST_BUFFER_TIMESTAMP (buf),
|
|
Packit |
971217 |
GST_BUFFER_TIMESTAMP (exp));
|
|
Packit |
971217 |
|
|
Packit |
971217 |
GST_DEBUG ("Properly received expected buffer");
|
|
Packit |
971217 |
gst_buffer_unref (exp);
|
|
Packit |
971217 |
gst_buffer_unref (buf);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
g_list_free1 (*expected);
|
|
Packit |
971217 |
*expected = next;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* When done signal main thread */
|
|
Packit |
971217 |
if (next == NULL) {
|
|
Packit |
971217 |
g_mutex_lock (&push_mutex);
|
|
Packit |
971217 |
g_cond_signal (&push_cond);
|
|
Packit |
971217 |
g_mutex_unlock (&push_mutex);
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
return GST_FLOW_OK;
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
static gboolean
|
|
Packit |
971217 |
my_sink_event (GstPad * pad, GstObject * parent, GstEvent * event)
|
|
Packit |
971217 |
{
|
|
Packit |
971217 |
GList **expected = GST_PAD_ELEMENT_PRIVATE (pad);
|
|
Packit |
971217 |
GList *next;
|
|
Packit |
971217 |
GstEvent *exp;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
fail_if (*expected == NULL,
|
|
Packit |
971217 |
"streamsynchronizer pushed a buffer/event but we didn't expect any");
|
|
Packit |
971217 |
|
|
Packit |
971217 |
next = (*expected)->next;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
fail_unless (GST_IS_EVENT ((*expected)->data),
|
|
Packit |
971217 |
"We were not expecting an event (But got an event of type %s)",
|
|
Packit |
971217 |
GST_EVENT_TYPE_NAME (event));
|
|
Packit |
971217 |
exp = GST_EVENT ((*expected)->data);
|
|
Packit |
971217 |
fail_unless (GST_EVENT_TYPE (event) == GST_EVENT_TYPE (exp),
|
|
Packit |
971217 |
"Got event of type %s but expected of type %s",
|
|
Packit |
971217 |
GST_EVENT_TYPE_NAME (event), GST_EVENT_TYPE_NAME (exp));
|
|
Packit |
971217 |
fail_unless_equals_int (GST_EVENT_SEQNUM (event), GST_EVENT_SEQNUM (exp));
|
|
Packit |
971217 |
/* FIXME : Check more types */
|
|
Packit |
971217 |
|
|
Packit |
971217 |
switch (GST_EVENT_TYPE (event)) {
|
|
Packit |
971217 |
case GST_EVENT_SEGMENT:
|
|
Packit |
971217 |
{
|
|
Packit |
971217 |
const GstSegment *recvseg, *expectseg;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* Compare segment values */
|
|
Packit |
971217 |
gst_event_parse_segment (event, &recvseg);
|
|
Packit |
971217 |
gst_event_parse_segment (exp, &expectseg);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
fail_unless_equals_int (recvseg->format, expectseg->format);
|
|
Packit |
971217 |
fail_unless_equals_uint64 (recvseg->base, expectseg->base);
|
|
Packit |
971217 |
fail_unless_equals_uint64 (recvseg->offset, expectseg->offset);
|
|
Packit |
971217 |
fail_unless_equals_uint64 (recvseg->start, expectseg->start);
|
|
Packit |
971217 |
fail_unless_equals_uint64 (recvseg->stop, expectseg->stop);
|
|
Packit |
971217 |
fail_unless_equals_uint64 (recvseg->time, expectseg->time);
|
|
Packit |
971217 |
fail_unless_equals_uint64 (recvseg->position, expectseg->position);
|
|
Packit |
971217 |
fail_unless_equals_uint64 (recvseg->duration, expectseg->duration);
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
break;
|
|
Packit |
971217 |
default:
|
|
Packit |
971217 |
break;
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
GST_DEBUG ("Properly received expected event %s", GST_EVENT_TYPE_NAME (exp));
|
|
Packit |
971217 |
|
|
Packit |
971217 |
gst_event_unref (exp);
|
|
Packit |
971217 |
gst_event_unref (event);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
g_list_free1 (*expected);
|
|
Packit |
971217 |
*expected = next;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* When done signal main thread */
|
|
Packit |
971217 |
if (next == NULL) {
|
|
Packit |
971217 |
g_mutex_lock (&push_mutex);
|
|
Packit |
971217 |
g_cond_signal (&push_cond);
|
|
Packit |
971217 |
g_mutex_unlock (&push_mutex);
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
return TRUE;
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
static gpointer
|
|
Packit |
971217 |
my_push_thread (MyPushInfo * pushinfo)
|
|
Packit |
971217 |
{
|
|
Packit |
971217 |
GList *tmp;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* FIXME : Do this in a thread */
|
|
Packit |
971217 |
for (tmp = pushinfo->to_push; tmp; tmp = tmp->next) {
|
|
Packit |
971217 |
if (GST_IS_EVENT (tmp->data))
|
|
Packit |
971217 |
gst_pad_push_event (pushinfo->pad, GST_EVENT (tmp->data));
|
|
Packit |
971217 |
else
|
|
Packit |
971217 |
gst_pad_push (pushinfo->pad, GST_BUFFER (tmp->data));
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
GST_INFO ("leaving thread");
|
|
Packit |
971217 |
return NULL;
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
GST_START_TEST (test_basic)
|
|
Packit |
971217 |
{
|
|
Packit |
971217 |
GstElement *synchr;
|
|
Packit |
971217 |
GstPad *sinkpad, *srcpad;
|
|
Packit |
971217 |
GstPad *mysrcpad, *mysinkpad;
|
|
Packit |
971217 |
GList *to_push = NULL, *expected = NULL;
|
|
Packit |
971217 |
GstEvent *event;
|
|
Packit |
971217 |
GstBuffer *buf;
|
|
Packit |
971217 |
GThread *thread;
|
|
Packit |
971217 |
MyPushInfo pushinfo;
|
|
Packit |
971217 |
guint i;
|
|
Packit |
971217 |
GstSegment segment;
|
|
Packit |
971217 |
guint32 seqnum;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
synchr = gst_element_factory_make ("streamsynchronizer", NULL);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* Get sinkpad/srcpad */
|
|
Packit |
971217 |
sinkpad = gst_element_get_request_pad (synchr, "sink_%u");
|
|
Packit |
971217 |
fail_unless (sinkpad != NULL);
|
|
Packit |
971217 |
srcpad = get_other_pad (sinkpad);
|
|
Packit |
971217 |
fail_unless (srcpad != NULL);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
gst_element_set_state (synchr, GST_STATE_PLAYING);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
mysrcpad = gst_pad_new_from_static_template (&mysrctemplate, "src");
|
|
Packit |
971217 |
fail_if (mysrcpad == NULL);
|
|
Packit |
971217 |
fail_unless (gst_pad_link (mysrcpad, sinkpad) == GST_PAD_LINK_OK);
|
|
Packit |
971217 |
fail_unless (gst_pad_set_active (mysrcpad, TRUE));
|
|
Packit |
971217 |
|
|
Packit |
971217 |
mysinkpad = gst_pad_new_from_static_template (&mysinktemplate, "sink");
|
|
Packit |
971217 |
gst_pad_set_chain_function (mysinkpad, my_sink_chain);
|
|
Packit |
971217 |
gst_pad_set_event_function (mysinkpad, my_sink_event);
|
|
Packit |
971217 |
fail_if (mysinkpad == NULL);
|
|
Packit |
971217 |
fail_unless (gst_pad_link (srcpad, mysinkpad) == GST_PAD_LINK_OK);
|
|
Packit |
971217 |
fail_unless (gst_pad_set_active (mysinkpad, TRUE));
|
|
Packit |
971217 |
GST_PAD_ELEMENT_PRIVATE (mysinkpad) = &expecte;;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* Start with a stream START and a new segment */
|
|
Packit |
971217 |
event = gst_event_new_stream_start ("lala");
|
|
Packit |
971217 |
to_push = g_list_append (to_push, event);
|
|
Packit |
971217 |
expected = g_list_append (expected, gst_event_ref (event));
|
|
Packit |
971217 |
|
|
Packit |
971217 |
gst_segment_init (&segment, GST_FORMAT_TIME);
|
|
Packit |
971217 |
event = gst_event_new_segment (&segment);
|
|
Packit |
971217 |
to_push = g_list_append (to_push, event);
|
|
Packit |
971217 |
expected = g_list_append (expected, gst_event_ref (event));
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* Then 10 buffers */
|
|
Packit |
971217 |
for (i = 0; i < 10; i++) {
|
|
Packit |
971217 |
buf = gst_buffer_new ();
|
|
Packit |
971217 |
GST_BUFFER_TIMESTAMP (buf) = i * GST_SECOND;
|
|
Packit |
971217 |
GST_BUFFER_DURATION (buf) = GST_SECOND;
|
|
Packit |
971217 |
to_push = g_list_append (to_push, buf);
|
|
Packit |
971217 |
expected = g_list_append (expected, gst_buffer_ref (buf));
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* Then a new stream start */
|
|
Packit |
971217 |
event = gst_event_new_stream_start ("lala again");
|
|
Packit |
971217 |
to_push = g_list_append (to_push, event);
|
|
Packit |
971217 |
expected = g_list_append (expected, gst_event_ref (event));
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* This newsegment will be updated */
|
|
Packit |
971217 |
gst_segment_init (&segment, GST_FORMAT_TIME);
|
|
Packit |
971217 |
event = gst_event_new_segment (&segment);
|
|
Packit |
971217 |
seqnum = gst_event_get_seqnum (event);
|
|
Packit |
971217 |
to_push = g_list_append (to_push, event);
|
|
Packit |
971217 |
/* The received segment's base should be updated by streamsynchronizer to
|
|
Packit |
971217 |
* take into account the amount of data played before (i.e. 10s) */
|
|
Packit |
971217 |
segment.base = 10 * GST_SECOND;
|
|
Packit |
971217 |
event = gst_event_new_segment (&segment);
|
|
Packit |
971217 |
gst_event_set_seqnum (event, seqnum);
|
|
Packit |
971217 |
expected = g_list_append (expected, event);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* Then 10 buffers */
|
|
Packit |
971217 |
for (i = 0; i < 10; i++) {
|
|
Packit |
971217 |
buf = gst_buffer_new ();
|
|
Packit |
971217 |
GST_BUFFER_TIMESTAMP (buf) = i * GST_SECOND;
|
|
Packit |
971217 |
GST_BUFFER_DURATION (buf) = GST_SECOND;
|
|
Packit |
971217 |
to_push = g_list_append (to_push, buf);
|
|
Packit |
971217 |
expected = g_list_append (expected, gst_buffer_ref (buf));
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
g_mutex_init (&push_mutex);
|
|
Packit |
971217 |
g_cond_init (&push_cond);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
pushinfo.pad = mysrcpad;
|
|
Packit |
971217 |
pushinfo.to_push = to_push;
|
|
Packit |
971217 |
g_mutex_lock (&push_mutex);
|
|
Packit |
971217 |
thread = g_thread_new ("pushthread", (GThreadFunc) my_push_thread, &pushinfo);
|
|
Packit |
971217 |
fail_unless (thread != NULL);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
g_cond_wait (&push_cond, &push_mutex);
|
|
Packit |
971217 |
g_mutex_unlock (&push_mutex);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
fail_if (expected != NULL);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* wait for thread to exit before freeing things */
|
|
Packit |
971217 |
g_thread_join (thread);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
/* Cleanup */
|
|
Packit |
971217 |
g_list_free (to_push);
|
|
Packit |
971217 |
gst_element_release_request_pad (synchr, sinkpad);
|
|
Packit |
971217 |
gst_object_unref (srcpad);
|
|
Packit |
971217 |
gst_object_unref (sinkpad);
|
|
Packit |
971217 |
gst_object_unref (mysinkpad);
|
|
Packit |
971217 |
gst_object_unref (mysrcpad);
|
|
Packit |
971217 |
gst_element_set_state (synchr, GST_STATE_NULL);
|
|
Packit |
971217 |
gst_object_unref (synchr);
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
GST_END_TEST;
|
|
Packit |
971217 |
|
|
Packit |
971217 |
static Suite *
|
|
Packit |
971217 |
streamsynchronizer_suite (void)
|
|
Packit |
971217 |
{
|
|
Packit |
971217 |
Suite *s = suite_create ("streamsynchronizer");
|
|
Packit |
971217 |
TCase *tc_chain = tcase_create ("general");
|
|
Packit |
971217 |
|
|
Packit |
971217 |
suite_add_tcase (s, tc_chain);
|
|
Packit |
971217 |
tcase_add_test (tc_chain, test_basic);
|
|
Packit |
971217 |
|
|
Packit |
971217 |
return s;
|
|
Packit |
971217 |
}
|
|
Packit |
971217 |
|
|
Packit |
971217 |
GST_CHECK_MAIN (streamsynchronizer);
|