--- /dev/null
+/* GStreamer
+ *
+ * tests for the ipcpipelinesrc/ipcpipelinesink elements
+ *
+ * Copyright (C) 2015-2017 YouView TV Ltd
+ * Author: Vincent Penquerc'h <vincent.penquerch@collabora.co.uk>
+ * Author: George Kiagiadakis <george.kiagiadakis@collabora.com>
+ *
+ * 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.
+ */
+
+#define _GNU_SOURCE /* See feature_test_macros(7) */
+#include <unistd.h>
+#include <stdio.h>
+#include <fcntl.h>
+#include <sys/wait.h>
+#include <sys/file.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <gst/check/gstcheck.h>
+#include <string.h>
+
+/* This enum contains flags that are used to configure the setup that
+ * test_base() will do internally */
+typedef enum
+{
+ /* Features related to the multi-process setup */
+ TEST_FEATURE_SPLIT_SINKS = 0x1, /* separate audio and video sink processes */
+ TEST_FEATURE_RECOVERY_SLAVE_PROCESS = 0x2,
+ TEST_FEATURE_RECOVERY_MASTER_PROCESS = 0x4,
+
+ TEST_FEATURE_HAS_VIDEO = 0x10,
+ TEST_FEATURE_LIVE = 0x20, /* sets is-live=true in {audio,video}testsrc */
+ TEST_FEATURE_ASYNC_SINK = 0x40, /* sets sync=false in fakesink */
+ TEST_FEATURE_ERROR_SINK = 0x80, /* generates error message in the slave */
+ TEST_FEATURE_LONG_DURATION = 0x100, /* bigger num-buffers in {audio,video}testsrc */
+ TEST_FEATURE_FILTER_SINK_CAPS = 0x200, /* plugs capsfilter before fakesink */
+
+ /* Source selection; Use only one of those, do not combine! */
+ TEST_FEATURE_TEST_SOURCE = 0x400,
+ TEST_FEATURE_WAV_SOURCE = 0x800,
+ TEST_FEATURE_MPEGTS_SOURCE = 0x1000 | TEST_FEATURE_HAS_VIDEO,
+ TEST_FEATURE_LIVE_A_SOURCE =
+ TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_LIVE | TEST_FEATURE_ASYNC_SINK,
+ TEST_FEATURE_LIVE_AV_SOURCE =
+ TEST_FEATURE_LIVE_A_SOURCE | TEST_FEATURE_HAS_VIDEO,
+} TestFeatures;
+
+/* This is the data structure that each function of the each test receives
+ * in user_data. It contains pointers to stack-allocated, test-specific
+ * structures that contain the test parameters (input data), the runtime
+ * data of the master (source) process (master data) and the runtime data
+ * of the slave (sink) process (slave data) */
+typedef struct
+{
+ gpointer id; /* input data struct */
+ gpointer md; /* master data struct */
+ gpointer sd; /* slave data struct */
+
+ TestFeatures features; /* the features that this test is running with */
+
+ /* whether there is both an audio and a video stream
+ * in this process'es pipeline */
+ gboolean two_streams;
+
+ /* the pipeline of this process; could be either master or slave */
+ GstElement *p;
+
+ /* this callback will be called in the master process when
+ * the master gets STATE_CHANGED with the new state being state_target */
+ void (*state_changed_cb) (gpointer);
+ GstState state_target;
+
+} test_data;
+
+/* All pipelines do not start buffers at exactly zero, so we consider
+ timestamps within a small tolerance to be zero */
+#define CLOSE_ENOUGH_TO_ZERO (GST_SECOND / 5)
+
+/* milliseconds */
+#define STEP_AT 100
+#define PAUSE_AT 500
+#define SEEK_AT 700
+#define QUERY_AT 600
+#define MESSAGE_AT 600
+#define CRASH_AT 600
+#define STOP_AT 600
+
+/* Rough duration of the sample files we use */
+#define MPEGTS_SAMPLE_ROUGH_DURATION (GST_SECOND * 64 / 10)
+#define WAV_SAMPLE_ROUGH_DURATION (GST_SECOND * 65 / 10)
+
+enum
+{
+ MSG_ACK = 0,
+ MSG_START = 1
+};
+
+static GMainLoop *loop;
+static gboolean child_dead;
+static int pipesfa[2], pipesba[2], pipesfv[2], pipesbv[2];
+static int ctlsock[2];
+static int recovery_pid = 0;
+static int check_fd = -1;
+static GList *weak_refs = NULL;
+
+/* lock helpers */
+
+#define FAIL_IF(x) do { lock_check (); fail_if(x); unlock_check (); } while(0)
+#define FAIL_UNLESS(x) do { lock_check (); fail_unless(x); unlock_check (); } while(0)
+#define FAIL_UNLESS_EQUALS_INT(x,y) do { lock_check (); fail_unless_equals_int(x,y); unlock_check (); } while(0)
+#define FAIL() do { lock_check (); fail(); unlock_check (); } while(0)
+
+static void
+lock_check (void)
+{
+ flock (check_fd, LOCK_EX);
+}
+
+static void
+unlock_check (void)
+{
+ flock (check_fd, LOCK_UN);
+}
+
+static void
+setup_lock (void)
+{
+ gchar *name = NULL;
+ check_fd = g_file_open_tmp (NULL, &name, NULL);
+ unlink (name);
+ g_free (name);
+}
+
+/* tracking for ipcpipeline elements; this is used mainly to detect leaks,
+ * but also to provide a method for calling "disconnect" on all of them
+ * in the tests that require it */
+
+static void
+remove_weak_ref (GstElement * element)
+{
+ weak_refs = g_list_remove (weak_refs, element);
+}
+
+static void
+add_weak_ref (GstElement * element)
+{
+ weak_refs = g_list_append (weak_refs, element);
+ g_object_weak_ref (G_OBJECT (element), (GWeakNotify) remove_weak_ref,
+ element);
+}
+
+static void
+disconnect_ipcpipeline_elements (void)
+{
+ GList *l;
+
+ for (l = weak_refs; l; l = l->next) {
+ g_signal_emit_by_name (G_OBJECT (l->data), "disconnect", NULL);
+ }
+}
+
+/* helper functions */
+
+static void
+cleanup_bus (GstElement * pipeline)
+{
+ gst_bus_remove_watch (GST_ELEMENT_BUS (pipeline));
+ gst_bus_set_flushing (GST_ELEMENT_BUS (pipeline), TRUE);
+}
+
+static void
+setup_log (const char *logfile, int append)
+{
+ FILE *f;
+
+ f = fopen (logfile, append ? "a+" : "w");
+ gst_debug_add_log_function (gst_debug_log_default, f, NULL);
+}
+
+static GstElement *
+create_pipeline (const char *type)
+{
+ GstElement *pipeline;
+
+ pipeline = gst_element_factory_make (type, NULL);
+ FAIL_UNLESS (pipeline);
+
+ return pipeline;
+}
+
+static GQuark
+to_be_removed_quark (void)
+{
+ static GQuark q = 0;
+ if (!q)
+ q = g_quark_from_static_string ("to_be_removed");
+ return q;
+}
+
+static gboolean
+are_caps_audio (const GstCaps * caps)
+{
+ GstStructure *structure;
+ const char *name;
+
+ structure = gst_caps_get_structure (caps, 0);
+ name = gst_structure_get_name (structure);
+ return g_str_has_prefix (name, "audio/");
+}
+
+static gboolean
+are_caps_video (const GstCaps * caps)
+{
+ GstStructure *structure;
+ const char *name;
+
+ structure = gst_caps_get_structure (caps, 0);
+ name = gst_structure_get_name (structure);
+ return (g_str_has_prefix (name, "video/")
+ && strcmp (name, "video/x-dvd-subpicture"));
+}
+
+static int
+caps2idx (GstCaps * caps, gboolean two_streams)
+{
+ int idx;
+
+ if (!two_streams)
+ return 0;
+
+ if (are_caps_audio (caps)) {
+ idx = 0;
+ } else if (are_caps_video (caps)) {
+ idx = 1;
+ } else {
+ FAIL_IF (1);
+ idx = 0;
+ }
+ return idx;
+}
+
+static int
+pad2idx (GstPad * pad, gboolean two_streams)
+{
+ GstCaps *caps;
+ int idx;
+
+ if (!two_streams)
+ return 0;
+
+ caps = gst_pad_get_current_caps (pad);
+ if (!caps)
+ caps = gst_pad_get_pad_template_caps (pad);
+ FAIL_UNLESS (caps);
+
+ idx = caps2idx (caps, two_streams);
+
+ gst_caps_unref (caps);
+ return idx;
+}
+
+static gboolean
+stop_pipeline (gpointer user_data)
+{
+ GstElement *pipeline = user_data;
+ GstStateChangeReturn ret;
+
+ ret = gst_element_set_state (pipeline, GST_STATE_NULL);
+ FAIL_IF (ret == GST_STATE_CHANGE_FAILURE);
+ gst_object_unref (pipeline);
+ g_main_loop_quit (loop);
+ return FALSE;
+}
+
+/* the master process'es async GstBus callback */
+static gboolean
+master_bus_msg (GstBus * bus, GstMessage * message, gpointer user_data)
+{
+ test_data *td = user_data;
+
+ switch (GST_MESSAGE_TYPE (message)) {
+ case GST_MESSAGE_ERROR:{
+ GError *err;
+ gchar *dbg;
+
+ /* elements we are removing might error out as they are taken out
+ of the pipeline, and fail to push. We don't care about those. */
+ if (g_object_get_qdata (G_OBJECT (GST_MESSAGE_SRC (message)),
+ to_be_removed_quark ()))
+ break;
+
+ gst_message_parse_error (message, &err, &dbg);
+ g_printerr ("ERROR: %s\n", err->message);
+ if (dbg != NULL)
+ g_printerr ("ERROR debug information: %s\n", dbg);
+ g_error_free (err);
+ g_free (dbg);
+ g_assert_not_reached ();
+ break;
+ }
+ case GST_MESSAGE_WARNING:{
+ GError *err;
+ gchar *dbg;
+
+ gst_message_parse_warning (message, &err, &dbg);
+ g_printerr ("WARNING: %s\n", err->message);
+ if (dbg != NULL)
+ g_printerr ("WARNING debug information: %s\n", dbg);
+ g_error_free (err);
+ g_free (dbg);
+ g_assert_not_reached ();
+ break;
+ }
+ case GST_MESSAGE_EOS:
+ g_main_loop_quit (loop);
+ break;
+ case GST_MESSAGE_STATE_CHANGED:
+ if (GST_MESSAGE_SRC (message) == GST_OBJECT_CAST (td->p)
+ && td->state_changed_cb) {
+ GstState state;
+ gst_message_parse_state_changed (message, NULL, &state, NULL);
+ if (state == td->state_target)
+ td->state_changed_cb (td);
+ }
+ break;
+ default:
+ break;
+ }
+ return TRUE;
+}
+
+/* source construction functions */
+
+static GstElement *
+create_wavparse_source_loc (const char *loc, int fdina, int fdouta)
+{
+ GstElement *sbin, *pipeline, *filesrc, *ipcpipelinesink;
+ GError *e = NULL;
+
+ pipeline = create_pipeline ("pipeline");
+ sbin =
+ gst_parse_bin_from_description ("pushfilesrc name=filesrc ! wavparse",
+ TRUE, &e);
+ FAIL_IF (e || !sbin);
+ gst_element_set_name (sbin, "source");
+ filesrc = gst_bin_get_by_name (GST_BIN (sbin), "filesrc");
+ FAIL_UNLESS (filesrc);
+ g_object_set (filesrc, "location", loc, NULL);
+ gst_object_unref (filesrc);
+ ipcpipelinesink =
+ gst_element_factory_make ("ipcpipelinesink", "ipcpipelinesink");
+ add_weak_ref (ipcpipelinesink);
+ g_object_set (ipcpipelinesink, "fdin", fdina, "fdout", fdouta, NULL);
+ gst_bin_add_many (GST_BIN (pipeline), sbin, ipcpipelinesink, NULL);
+ FAIL_UNLESS (gst_element_link_many (sbin, ipcpipelinesink, NULL));
+
+ return pipeline;
+}
+
+static void
+on_pad_added (GstElement * element, GstPad * pad, gpointer data)
+{
+ GstCaps *caps;
+ GstElement *next;
+ GstBin *pipeline = data;
+ GstPad *sink_pad;
+
+ caps = gst_pad_get_current_caps (pad);
+ if (!caps)
+ caps = gst_pad_get_pad_template_caps (pad);
+
+ if (are_caps_video (caps)) {
+ next = gst_bin_get_by_name (GST_BIN (pipeline), "vqueue");
+ } else if (are_caps_audio (caps)) {
+ next = gst_bin_get_by_name (GST_BIN (pipeline), "aqueue");
+ } else {
+ gst_caps_unref (caps);
+ return;
+ }
+ gst_caps_unref (caps);
+
+ FAIL_UNLESS (next);
+ sink_pad = gst_element_get_static_pad (next, "sink");
+ FAIL_UNLESS (sink_pad);
+ FAIL_UNLESS (gst_pad_link (pad, sink_pad) == GST_PAD_LINK_OK);
+ gst_object_unref (sink_pad);
+
+ gst_object_unref (next);
+}
+
+static GstElement *
+create_mpegts_source_loc (const char *loc, int fdina, int fdouta, int fdinv,
+ int fdoutv)
+{
+ GstElement *pipeline, *filesrc, *tsdemux, *aqueue, *vqueue, *aipcpipelinesink,
+ *vipcpipelinesink;
+
+ pipeline = create_pipeline ("pipeline");
+ filesrc = gst_element_factory_make ("filesrc", NULL);
+ g_object_set (filesrc, "location", loc, NULL);
+ tsdemux = gst_element_factory_make ("tsdemux", NULL);
+ g_signal_connect (tsdemux, "pad-added", G_CALLBACK (on_pad_added), pipeline);
+ aqueue = gst_element_factory_make ("queue", "aqueue");
+ aipcpipelinesink = gst_element_factory_make ("ipcpipelinesink", NULL);
+ add_weak_ref (aipcpipelinesink);
+ g_object_set (aipcpipelinesink, "fdin", fdina, "fdout", fdouta, NULL);
+ vqueue = gst_element_factory_make ("queue", "vqueue");
+ vipcpipelinesink = gst_element_factory_make ("ipcpipelinesink", NULL);
+ add_weak_ref (vipcpipelinesink);
+ g_object_set (vipcpipelinesink, "fdin", fdinv, "fdout", fdoutv, NULL);
+ gst_bin_add_many (GST_BIN (pipeline), filesrc, tsdemux, aqueue,
+ aipcpipelinesink, vqueue, vipcpipelinesink, NULL);
+ FAIL_UNLESS (gst_element_link_many (filesrc, tsdemux, NULL));
+ FAIL_UNLESS (gst_element_link_many (aqueue, aipcpipelinesink, NULL));
+ FAIL_UNLESS (gst_element_link_many (vqueue, vipcpipelinesink, NULL));
+
+ return pipeline;
+}
+
+static GstElement *
+create_test_source (gboolean live, int fdina, int fdouta, int fdinv, int fdoutv,
+ gboolean audio, gboolean video, gboolean Long)
+{
+ GstElement *pipeline, *audiotestsrc, *aipcpipelinesink;
+ GstElement *videotestsrc, *vipcpipelinesink;
+ int L = Long ? 2 : 1;
+
+ pipeline = create_pipeline ("pipeline");
+
+ if (audio) {
+ audiotestsrc = gst_element_factory_make ("audiotestsrc", "audiotestsrc");
+ g_object_set (audiotestsrc, "is-live", live, "num-buffers",
+ live ? 270 * L : 600, NULL);
+ aipcpipelinesink = gst_element_factory_make ("ipcpipelinesink",
+ "aipcpipelinesink");
+ add_weak_ref (aipcpipelinesink);
+ g_object_set (aipcpipelinesink, "fdin", fdina, "fdout", fdouta, NULL);
+ gst_bin_add_many (GST_BIN (pipeline), audiotestsrc, aipcpipelinesink, NULL);
+ FAIL_UNLESS (gst_element_link_many (audiotestsrc, aipcpipelinesink, NULL));
+ }
+
+ if (video) {
+ videotestsrc = gst_element_factory_make ("videotestsrc", "videotestsrc");
+ g_object_set (videotestsrc, "is-live", live, "num-buffers",
+ live ? 190 * L : 600, NULL);
+ vipcpipelinesink =
+ gst_element_factory_make ("ipcpipelinesink", "vipcpipelinesink");
+ add_weak_ref (vipcpipelinesink);
+ g_object_set (vipcpipelinesink, "fdin", fdinv, "fdout", fdoutv, NULL);
+ gst_bin_add_many (GST_BIN (pipeline), videotestsrc, vipcpipelinesink, NULL);
+ FAIL_UNLESS (gst_element_link_many (videotestsrc, vipcpipelinesink, NULL));
+ }
+
+ return pipeline;
+}
+
+static GstElement *
+create_source (TestFeatures features, int fdina, int fdouta, int fdinv,
+ int fdoutv, test_data * td)
+{
+ GstElement *pipeline = NULL;
+ gboolean live = ! !(features & TEST_FEATURE_LIVE);
+ gboolean longdur = ! !(features & TEST_FEATURE_LONG_DURATION);
+ gboolean has_video = ! !(features & TEST_FEATURE_HAS_VIDEO);
+
+ if (features & TEST_FEATURE_TEST_SOURCE) {
+
+ pipeline = create_test_source (live, fdina, fdouta, fdinv, fdoutv, TRUE,
+ has_video, longdur);
+ } else if (features & TEST_FEATURE_WAV_SOURCE) {
+ pipeline = create_wavparse_source_loc ("../../tests/files/sine.wav", fdina,
+ fdouta);
+ } else if (features & TEST_FEATURE_MPEGTS_SOURCE) {
+ pipeline = create_mpegts_source_loc ("../../tests/files/test.ts", fdina,
+ fdouta, fdinv, fdoutv);
+ } else {
+ g_assert_not_reached ();
+ }
+
+ td->two_streams = has_video;
+ td->p = pipeline;
+
+ if (pipeline)
+ gst_bus_add_watch (GST_ELEMENT_BUS (pipeline), master_bus_msg, td);
+
+ return pipeline;
+}
+
+/* sink construction */
+
+static GstElement *
+create_sink (TestFeatures features, GstElement ** slave_pipeline,
+ int fdin, int fdout, const char *filter_caps)
+{
+ GstElement *ipcpipelinesrc, *fakesink, *identity, *capsfilter, *endpoint;
+ GstCaps *caps;
+
+ if (!*slave_pipeline)
+ *slave_pipeline = create_pipeline ("ipcslavepipeline");
+ else
+ gst_object_ref (*slave_pipeline);
+ ipcpipelinesrc = gst_element_factory_make ("ipcpipelinesrc", NULL);
+ add_weak_ref (ipcpipelinesrc);
+ g_object_set (ipcpipelinesrc, "fdin", fdin, "fdout", fdout, NULL);
+ fakesink = gst_element_factory_make ("fakesink", NULL);
+ g_object_set (fakesink, "sync", !(features & TEST_FEATURE_ASYNC_SINK), NULL);
+ gst_bin_add_many (GST_BIN (*slave_pipeline), ipcpipelinesrc, fakesink, NULL);
+ endpoint = ipcpipelinesrc;
+
+ if (features & TEST_FEATURE_ERROR_SINK &&
+ !g_strcmp0 (filter_caps, "audio/x-raw")) {
+ identity = gst_element_factory_make ("identity", "error-element");
+ g_object_set (identity, "error-after", 5, NULL);
+ gst_bin_add (GST_BIN (*slave_pipeline), identity);
+ FAIL_UNLESS (gst_element_link_many (endpoint, identity, NULL));
+ endpoint = identity;
+ }
+
+ if ((features & TEST_FEATURE_FILTER_SINK_CAPS) && filter_caps) {
+ capsfilter = gst_element_factory_make ("capsfilter", NULL);
+ caps = gst_caps_from_string (filter_caps);
+ FAIL_UNLESS (caps);
+ g_object_set (capsfilter, "caps", caps, NULL);
+ gst_caps_unref (caps);
+ gst_bin_add (GST_BIN (*slave_pipeline), capsfilter);
+ FAIL_UNLESS (gst_element_link_many (endpoint, capsfilter, NULL));
+ endpoint = capsfilter;
+ }
+ FAIL_UNLESS (gst_element_link_many (endpoint, fakesink, NULL));
+
+ return *slave_pipeline;
+}
+
+static void
+ensure_sink_setup (GstElement * sink, void (*setup_sink) (GstElement *, void *),
+ gpointer user_data)
+{
+ static GQuark setup_done = 0;
+ test_data *td = user_data;
+
+ if (!setup_done)
+ setup_done = g_quark_from_static_string ("setup_done");
+
+ if (sink)
+ td->p = sink;
+
+ if (sink && setup_sink && !g_object_get_qdata (G_OBJECT (sink), setup_done)) {
+ g_object_set_qdata (G_OBJECT (sink), setup_done, GINT_TO_POINTER (1));
+ setup_sink (sink, user_data);
+ }
+}
+
+/* GstCheck multi-process setup helpers */
+
+static void
+on_child_exit (int signal)
+{
+ int status = 0;
+ if (waitpid (-1, &status, 0) > 0 && status) {
+ FAIL ();
+ exit (status);
+ } else {
+ child_dead = TRUE;
+ }
+}
+
+static void
+die_on_child_death (void)
+{
+ struct sigaction sa;
+
+ memset (&sa, 0, sizeof (sa));
+ sa.sa_handler = on_child_exit;
+ sigaction (SIGCHLD, &sa, NULL);
+}
+
+static void
+wait_for_recovery (void)
+{
+ int value;
+
+ FAIL_UNLESS (ctlsock[1]);
+ FAIL_UNLESS (read (ctlsock[1], &value, sizeof (int)) == sizeof (int));
+ FAIL_UNLESS (value == MSG_START);
+}
+
+static void
+ack_recovery (void)
+{
+ int value = MSG_ACK;
+ FAIL_UNLESS (ctlsock[1]);
+ FAIL_UNLESS (write (ctlsock[1], &value, sizeof (int)) == sizeof (int));
+}
+
+static void
+recreate_crashed_slave_process (void)
+{
+ int value = MSG_START;
+ /* We don't recreate, because there seems to be some subtle issues
+ with forking after gst has started running. So we create a new
+ recovery process at start, and wake it up after the current
+ slave dies, so it can take its place. It's a bit hacky, but it
+ works. The spare process waits for SIGUSR2 to setup a replacement
+ pipeline and connect to the master. */
+ FAIL_UNLESS (recovery_pid);
+ FAIL_UNLESS (ctlsock[0]);
+ FAIL_UNLESS (write (ctlsock[0], &value, sizeof (int)) == sizeof (int));
+ FAIL_UNLESS (read (ctlsock[0], &value, sizeof (int)) == sizeof (int));
+ FAIL_UNLESS (value == MSG_ACK);
+}
+
+static gboolean
+crash (gpointer user_data)
+{
+ _exit (0);
+}
+
+static gboolean
+unwind (gpointer user_data)
+{
+ g_main_loop_quit (loop);
+ return FALSE;
+}
+
+static void
+on_unwind (int signal)
+{
+ g_idle_add (unwind, NULL);
+}
+
+static void
+listen_for_unwind (void)
+{
+ struct sigaction sa;
+
+ memset (&sa, 0, sizeof (sa));
+ sa.sa_handler = on_unwind;
+ sigaction (SIGUSR1, &sa, NULL);
+}
+
+static void
+stop_listening_for_unwind (void)
+{
+ struct sigaction sa;
+
+ memset (&sa, 0, sizeof (sa));
+ sa.sa_handler = SIG_DFL;
+ sigaction (SIGUSR1, &sa, NULL);
+}
+
+#define TEST_BASE(...) test_base(__FUNCTION__,##__VA_ARGS__)
+
+/*
+ * This is the main function driving the tests. All tests configure it
+ * by way of all the function pointers it takes as arguments, which have
+ * self-explanatory names.
+ * Most tests are run over a number of different pipelines with the same
+ * configuration (eg, a wavparse based pipeline, a live pipeline with
+ * test audio/video, etc). Those pipelines that have more than one sink
+ * (eg, MPEG-TS source demuxing audio and video) have a version with a
+ * single slave pipeline and process, and a version with the audio and
+ * video sinks in two different processes, each with its slave pipeline.
+ * The master and slave crash tests are also run via this function, and
+ * have specific code (grep for recovery).
+ * There is a fair amount of hairy stuff to do with letting the main
+ * check process when a subprocess has failed. Best not to look at it
+ * and let it do its thing.
+ * To add new tests, duplicate a set of tests, eg the *_end_of_stream
+ * ones, and s/_end_of_stream/new_test_name/g. Then do the same for
+ * the functions they pass as parameters to test_base. Typically, the
+ * source creation sets a message hook to catch things like async-done
+ * messages. Sink creation typically adds a probe to check that events,
+ * buffers, etc, come through as expected. The two success functions
+ * check all went well for the source and sink. Note that since all of
+ * these functions take the same user data structure, and the process
+ * will fork, writing something from one process will not be reflected
+ * in the other, so there is usually a subset of data relevant to the
+ * source, and another to the sink. But some have data relevant to both,
+ * it depends on the test and what you are doing.
+ * New tests do not have to use this framework, it just avoids spending
+ * more time and effort on multi process handling.
+ */
+static void
+test_base (const char *name, TestFeatures features,
+ void (*run_source) (GstElement *, void *),
+ void (*setup_sink) (GstElement *, void *),
+ void (*check_success_source) (void *),
+ void (*check_success_sink) (void *),
+ gpointer input_data, gpointer master_data, gpointer slave_data)
+{
+ GstElement *source = NULL, *asink = NULL, *vsink = NULL;
+ GstElement *slave_pipeline = NULL;
+ GstStateChangeReturn ret;
+ gboolean c_src, c_sink;
+ pid_t pid = 0;
+ unsigned char x;
+ int master_recovery_pid_comm[2] = { -1, -1 };
+ test_data td = { input_data, master_data, slave_data, features, FALSE, NULL,
+ NULL, GST_STATE_NULL
+ };
+
+ g_print ("Testing: %s\n", name);
+
+ weak_refs = NULL;
+
+ FAIL_IF (pipe2 (pipesfa, O_NONBLOCK) < 0);
+ FAIL_IF (pipe2 (pipesba, O_NONBLOCK) < 0);
+ FAIL_IF (pipe2 (pipesfv, O_NONBLOCK) < 0);
+ FAIL_IF (pipe2 (pipesbv, O_NONBLOCK) < 0);
+ FAIL_IF (socketpair (PF_UNIX, SOCK_STREAM, 0, ctlsock) < 0);
+
+ FAIL_IF (pipesfa[0] < 0);
+ FAIL_IF (pipesfa[1] < 0);
+ FAIL_IF (pipesba[0] < 0);
+ FAIL_IF (pipesba[1] < 0);
+ FAIL_IF (pipesfv[0] < 0);
+ FAIL_IF (pipesfv[1] < 0);
+ FAIL_IF (pipesbv[0] < 0);
+ FAIL_IF (pipesbv[1] < 0);
+
+ gst_debug_remove_log_function (gst_debug_log_default);
+
+ listen_for_unwind ();
+ child_dead = FALSE;
+
+ if (features & TEST_FEATURE_RECOVERY_MASTER_PROCESS) {
+ /* the other master will let us know its child's PID so we can unwind
+ it when we're finished */
+ FAIL_IF (pipe2 (master_recovery_pid_comm, O_NONBLOCK) < 0);
+
+ recovery_pid = fork ();
+ if (recovery_pid > 0) {
+ /* we're the main process that libcheck waits for */
+ die_on_child_death ();
+ while (!child_dead)
+ g_usleep (1000);
+ /* leave some time for the slave to timeout (1 second), record error, etc */
+ g_usleep (1500 * 1000);
+
+ /* Discard anything that was sent to the previous process when it died */
+ while (read (pipesba[0], &x, 1) == 1);
+
+ FAIL_UNLESS (read (master_recovery_pid_comm[0], &pid,
+ sizeof (pid)) == sizeof (pid));
+
+ setup_log ("gstsrc.log", TRUE);
+ source = create_source (features, pipesba[0], pipesfa[1], pipesbv[0],
+ pipesfv[1], &td);
+ FAIL_UNLESS (source);
+ if (run_source)
+ run_source (source, &td);
+ goto setup_done;
+ }
+ }
+
+ if (features & TEST_FEATURE_RECOVERY_SLAVE_PROCESS) {
+ recovery_pid = fork ();
+ if (!recovery_pid) {
+ wait_for_recovery ();
+
+ /* Discard anything that was sent to the previous process when it died */
+ while (read (pipesfa[0], &x, 1) == 1);
+
+ setup_log ("gstasink.log", TRUE);
+ asink = create_sink (features, &slave_pipeline, pipesfa[0], pipesba[1],
+ "audio/x-raw");
+ FAIL_UNLESS (asink);
+ ensure_sink_setup (asink, setup_sink, &td);
+ ack_recovery ();
+ goto setup_done;
+ }
+ }
+
+ pid = fork ();
+ FAIL_IF (pid < 0);
+ if (pid) {
+ if (features & TEST_FEATURE_RECOVERY_MASTER_PROCESS) {
+ FAIL_UNLESS (write (master_recovery_pid_comm[1], &pid,
+ sizeof (pid)) == sizeof (pid));
+ }
+ die_on_child_death ();
+ if (features & TEST_FEATURE_SPLIT_SINKS) {
+ pid = fork ();
+ FAIL_IF (pid < 0);
+ if (pid) {
+ die_on_child_death ();
+ }
+ c_src = ! !pid;
+ c_sink = !pid;
+ } else {
+ c_src = TRUE;
+ c_sink = FALSE;
+ }
+ if (c_src) {
+ setup_log ("gstsrc.log", FALSE);
+ source = create_source (features, pipesba[0], pipesfa[1], pipesbv[0],
+ pipesfv[1], &td);
+ FAIL_UNLESS (source);
+ run_source (source, &td);
+ }
+ if (c_sink) {
+ setup_log ("gstasink.log", FALSE);
+ asink = create_sink (features, &slave_pipeline, pipesfa[0], pipesba[1],
+ "audio/x-raw");
+ FAIL_UNLESS (asink);
+ }
+ } else {
+ td.two_streams = (features & TEST_FEATURE_HAS_VIDEO) &&
+ !(features & TEST_FEATURE_SPLIT_SINKS);
+
+ if (features & TEST_FEATURE_HAS_VIDEO) {
+ setup_log ("gstvsink.log", FALSE);
+ vsink = create_sink (features, &slave_pipeline, pipesfv[0], pipesbv[1],
+ "video/x-raw");
+ FAIL_UNLESS (vsink);
+ }
+ if (!(features & TEST_FEATURE_SPLIT_SINKS)) {
+ setup_log ("gstasink.log", FALSE);
+ asink = create_sink (features, &slave_pipeline, pipesfa[0], pipesba[1],
+ "audio/x-raw");
+ FAIL_UNLESS (asink);
+ }
+ }
+
+setup_done:
+ ensure_sink_setup (asink, setup_sink, &td);
+ ensure_sink_setup (vsink, setup_sink, &td);
+
+ loop = g_main_loop_new (NULL, FALSE);
+ g_main_loop_run (loop);
+
+ /* tell the child process to unwind too */
+ stop_listening_for_unwind ();
+
+ if (source) {
+ ret = gst_element_set_state (source, GST_STATE_NULL);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_SUCCESS
+ || ret == GST_STATE_CHANGE_ASYNC);
+ }
+
+ if (pid)
+ kill (pid, SIGUSR1);
+
+ g_main_loop_unref (loop);
+
+ if (source) {
+ cleanup_bus (source);
+ if (check_success_source)
+ check_success_source (&td);
+ } else {
+ if (asink)
+ cleanup_bus (asink);
+ if (vsink)
+ cleanup_bus (vsink);
+ if (check_success_sink)
+ check_success_sink (&td);
+ }
+
+ disconnect_ipcpipeline_elements ();
+
+ close (pipesfa[0]);
+ close (pipesfa[1]);
+ close (pipesba[0]);
+ close (pipesba[1]);
+ close (pipesfv[0]);
+ close (pipesfv[1]);
+ close (pipesbv[0]);
+ close (pipesbv[1]);
+
+ /* If we have a child, we must now wait for it to be finished.
+ We can't just waitpid, because this child might be still doing
+ its shutdown, and might assert, and the die_on_child_death
+ function will exit with the right exit code if so. So we wait
+ for the child_dead boolean to be set, which die_on_child_death
+ sets if the child dies normally. */
+ if (pid) {
+ while (!child_dead)
+ g_usleep (1000);
+ }
+
+ if (source) {
+ FAIL_UNLESS_EQUALS_INT (GST_OBJECT_REFCOUNT_VALUE (source), 1);
+ gst_object_unref (source);
+ }
+ /* asink and vsink may be the same object, so refcount is not sure to be 1 */
+ if (asink)
+ gst_object_unref (asink);
+ if (vsink)
+ gst_object_unref (vsink);
+
+ /* cleanup tasks a bit earlier to make sure all weak refs are gone */
+ gst_task_cleanup_all ();
+
+ /* all ipcpipeline elements we created should now be destroyed */
+ if (weak_refs) {
+#if 1
+ /* to make it easier to see what leaks */
+ GList *l;
+ for (l = weak_refs; l; l = l->next) {
+ g_print ("%s has %u refs\n", GST_ELEMENT_NAME (l->data),
+ GST_OBJECT_REFCOUNT_VALUE (l->data));
+ }
+#endif
+ FAIL_UNLESS (0);
+ }
+}
+
+/**** play-pause test ****/
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing[2];
+ gboolean got_state_changed_to_paused;
+} play_pause_master_data;
+
+typedef struct
+{
+ gboolean got_caps[2];
+ gboolean got_segment[2];
+ gboolean got_buffer[2];
+} play_pause_slave_data;
+
+static gboolean
+idlenull (gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ ret = gst_element_set_state (td->p, GST_STATE_NULL);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_SUCCESS);
+ gst_object_unref (td->p);
+ g_main_loop_quit (loop);
+ return G_SOURCE_REMOVE;
+}
+
+static gboolean idleplay (gpointer user_data);
+static gboolean
+idlepause (gpointer user_data)
+{
+ test_data *td = user_data;
+ play_pause_master_data *d = td->md;
+ GstStateChangeReturn ret;
+
+ ret = gst_element_set_state (td->p, GST_STATE_PAUSED);
+ FAIL_IF (ret == GST_STATE_CHANGE_FAILURE);
+ if (ret == GST_STATE_CHANGE_SUCCESS || ret == GST_STATE_CHANGE_NO_PREROLL) {
+ /* if the state change is not async, we won't get an aync-done, but
+ this is expected, so set the flag here */
+ d->got_state_changed_to_paused = TRUE;
+ td->state_target = GST_STATE_PLAYING;
+ g_timeout_add (STEP_AT, idleplay, user_data);
+ return G_SOURCE_REMOVE;
+ }
+ gst_object_unref (td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static gboolean
+idleplay (gpointer user_data)
+{
+ test_data *td = user_data;
+ play_pause_master_data *d = td->md;
+ GstStateChangeReturn ret;
+
+ ret = gst_element_set_state (td->p, GST_STATE_PLAYING);
+ FAIL_IF (ret == GST_STATE_CHANGE_FAILURE);
+ if (ret == GST_STATE_CHANGE_SUCCESS || ret == GST_STATE_CHANGE_NO_PREROLL) {
+ /* if the state change is not async, we won't get an aync-done, but
+ this is expected, so set the flag here */
+ d->got_state_changed_to_playing[1] = TRUE;
+ td->state_target = GST_STATE_NULL;
+ g_timeout_add (STEP_AT, idlenull, user_data);
+ return G_SOURCE_REMOVE;
+ }
+ gst_object_unref (td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static void
+play_pause_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ play_pause_master_data *d = td->md;
+ GstStateChangeReturn ret;
+
+ if (d->got_state_changed_to_paused) {
+ d->got_state_changed_to_playing[1] = TRUE;
+ td->state_target = GST_STATE_NULL;
+ ret = gst_element_set_state (td->p, GST_STATE_NULL);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_SUCCESS);
+ g_main_loop_quit (loop);
+ } else if (d->got_state_changed_to_playing[0]) {
+ d->got_state_changed_to_paused = TRUE;
+ td->state_target = GST_STATE_PLAYING;
+ gst_object_ref (td->p);
+ g_timeout_add (STEP_AT, (GSourceFunc) idleplay, td);
+ } else {
+ d->got_state_changed_to_playing[0] = TRUE;
+ td->state_target = GST_STATE_PAUSED;
+ gst_object_ref (td->p);
+ g_timeout_add (STEP_AT, (GSourceFunc) idlepause, td);
+ }
+}
+
+static void
+play_pause_source (GstElement * source, void *user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = play_pause_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static GstPadProbeReturn
+play_pause_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ play_pause_slave_data *d = td->sd;
+ GstCaps *caps;
+
+ if (GST_IS_BUFFER (info->data)) {
+ d->got_buffer[pad2idx (pad, td->two_streams)] = TRUE;
+ } else if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_CAPS) {
+ gst_event_parse_caps (info->data, &caps);
+ d->got_caps[caps2idx (caps, td->two_streams)] = TRUE;
+ } else if (GST_EVENT_TYPE (info->data) == GST_EVENT_SEGMENT) {
+ d->got_segment[pad2idx (pad, td->two_streams)] = TRUE;
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_probe_types (const GValue * sinkv, GstPadProbeCallback probe,
+ unsigned int types, gpointer user_data)
+{
+ GstElement *sink;
+ GstPad *pad;
+
+ sink = g_value_get_object (sinkv);
+ FAIL_UNLESS (sink);
+ pad = gst_element_get_static_pad (sink, "sink");
+ FAIL_UNLESS (pad);
+ gst_pad_add_probe (pad, types, probe, user_data, NULL);
+ gst_object_unref (pad);
+}
+
+static void
+hook_probe (const GValue * sinkv, GstPadProbeCallback probe, gpointer user_data)
+{
+ hook_probe_types (sinkv, probe,
+ GST_PAD_PROBE_TYPE_DATA_DOWNSTREAM | GST_PAD_PROBE_TYPE_EVENT_FLUSH |
+ GST_PAD_PROBE_TYPE_QUERY_DOWNSTREAM, user_data);
+}
+
+static void
+hook_play_pause_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, play_pause_probe, user_data);
+}
+
+static void
+setup_sink_play_pause (GstElement * sink, void *user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_play_pause_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_play_pause (void *user_data)
+{
+ test_data *td = user_data;
+ play_pause_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing[0]);
+ FAIL_UNLESS (d->got_state_changed_to_playing[1]);
+ FAIL_UNLESS (d->got_state_changed_to_paused);
+}
+
+static void
+check_success_sink_play_pause (void *user_data)
+{
+ test_data *td = user_data;
+ play_pause_slave_data *d = td->sd;
+ int idx;
+
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ FAIL_UNLESS (d->got_caps[idx]);
+ FAIL_UNLESS (d->got_segment[idx]);
+ FAIL_UNLESS (d->got_buffer[idx]);
+ }
+}
+
+GST_START_TEST (test_empty_play_pause)
+{
+ play_pause_master_data md = { 0 };
+ play_pause_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, play_pause_source, setup_sink_play_pause,
+ check_success_source_play_pause, check_success_sink_play_pause, NULL, &md,
+ &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_play_pause)
+{
+ play_pause_master_data md = { 0 };
+ play_pause_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, play_pause_source, setup_sink_play_pause,
+ check_success_source_play_pause, check_success_sink_play_pause, NULL, &md,
+ &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_play_pause)
+{
+ play_pause_master_data md = { 0 };
+ play_pause_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, play_pause_source,
+ setup_sink_play_pause, check_success_source_play_pause,
+ check_success_sink_play_pause, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_play_pause)
+{
+ play_pause_master_data md = { 0 };
+ play_pause_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ play_pause_source, setup_sink_play_pause, check_success_source_play_pause,
+ check_success_sink_play_pause, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_play_pause)
+{
+ play_pause_master_data md = { 0 };
+ play_pause_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE, play_pause_source,
+ setup_sink_play_pause, check_success_source_play_pause,
+ check_success_sink_play_pause, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_play_pause)
+{
+ play_pause_master_data md = { 0 };
+ play_pause_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, play_pause_source,
+ setup_sink_play_pause, check_success_source_play_pause,
+ check_success_sink_play_pause, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_play_pause)
+{
+ play_pause_master_data md = { 0 };
+ play_pause_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ play_pause_source, setup_sink_play_pause, check_success_source_play_pause,
+ check_success_sink_play_pause, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** flushing seek test ****/
+
+typedef struct
+{
+ gboolean segment_seek;
+ gboolean pause;
+} flushing_seek_input_data;
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ gboolean got_segment_done;
+ gboolean seek_sent;
+} flushing_seek_master_data;
+
+typedef struct
+{
+ GstClockTime first_ts[2];
+ gboolean got_caps[2];
+ gboolean got_buffer_before_seek[2];
+ gboolean got_buffer_after_seek[2];
+ gboolean first_buffer_after_seek_has_timestamp_0[2];
+ gboolean got_segment_after_seek[2];
+ gboolean got_flush_start[2];
+ gboolean got_flush_stop[2];
+} flushing_seek_slave_data;
+
+static gboolean
+send_flushing_seek (gpointer user_data)
+{
+ test_data *td = user_data;
+ const flushing_seek_input_data *i = td->id;
+ flushing_seek_master_data *d = td->md;
+ GstEvent *seek_event;
+
+ if (i->segment_seek) {
+ GST_INFO_OBJECT (td->p, "Sending segment seek");
+ seek_event =
+ gst_event_new_seek (1.0, GST_FORMAT_TIME,
+ GST_SEEK_FLAG_SEGMENT | GST_SEEK_FLAG_FLUSH, GST_SEEK_TYPE_SET, 0,
+ GST_SEEK_TYPE_SET, 1 * GST_SECOND);
+ FAIL_UNLESS (gst_element_send_event (td->p, seek_event));
+ } else {
+ GST_INFO_OBJECT (td->p, "Sending flushing seek");
+ gst_element_seek_simple (td->p, GST_FORMAT_TIME, GST_SEEK_FLAG_FLUSH, 0);
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline,
+ gst_object_ref (td->p));
+ }
+ d->seek_sent = TRUE;
+ return G_SOURCE_REMOVE;
+}
+
+static gboolean
+pause_before_seek (gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ ret = gst_element_set_state (td->p, GST_STATE_PAUSED);
+ FAIL_IF (ret == GST_STATE_CHANGE_FAILURE);
+
+ return G_SOURCE_REMOVE;
+}
+
+static gboolean
+flushing_seek_bus_msg (GstBus * bus, GstMessage * message, gpointer user_data)
+{
+ test_data *td = user_data;
+ flushing_seek_master_data *d = td->md;
+
+ if (GST_IS_PIPELINE (GST_MESSAGE_SRC (message))) {
+ if (GST_MESSAGE_TYPE (message) == GST_MESSAGE_SEGMENT_DONE) {
+ d->got_segment_done = TRUE;
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline,
+ gst_object_ref (td->p));
+ }
+ }
+ return master_bus_msg (bus, message, user_data);
+}
+
+static void
+flushing_seek_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ const flushing_seek_input_data *i = td->id;
+ flushing_seek_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing) {
+ d->got_state_changed_to_playing = TRUE;
+ if (i->pause)
+ g_timeout_add (PAUSE_AT, (GSourceFunc) pause_before_seek, td);
+ g_timeout_add (SEEK_AT, (GSourceFunc) send_flushing_seek, td);
+ }
+}
+
+static void
+flushing_seek_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ /* we're on the source, there's already the basic master_bus_msg watch,
+ and gst doesn't want more than one watch, so we remove the watch and
+ call it directly when done in the new watch */
+ gst_bus_remove_watch (GST_ELEMENT_BUS (source));
+ gst_bus_add_watch (GST_ELEMENT_BUS (source), flushing_seek_bus_msg,
+ user_data);
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = flushing_seek_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static GstPadProbeReturn
+flushing_seek_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ flushing_seek_slave_data *d = td->sd;
+ GstClockTime ts;
+ int idx;
+ GstCaps *caps;
+
+ if (GST_IS_BUFFER (info->data)) {
+ idx = pad2idx (pad, td->two_streams);
+ if (d->got_flush_stop[idx]) {
+ if (!d->got_buffer_after_seek[idx]) {
+ ts = GST_BUFFER_TIMESTAMP (info->data);
+ d->first_buffer_after_seek_has_timestamp_0[idx] =
+ (ts < d->first_ts[idx] + 10 * GST_MSECOND);
+ d->got_buffer_after_seek[idx] = TRUE;
+ }
+ } else if (!d->got_buffer_before_seek[idx]) {
+ d->got_buffer_before_seek[idx] = TRUE;
+ d->first_ts[idx] = GST_BUFFER_TIMESTAMP (info->data);
+ }
+ } else if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_CAPS) {
+ gst_event_parse_caps (info->data, &caps);
+ if (are_caps_audio (caps) || are_caps_video (caps)) {
+ idx = caps2idx (caps, td->two_streams);
+ d->got_caps[idx] = TRUE;
+ }
+ } else if (GST_EVENT_TYPE (info->data) == GST_EVENT_SEGMENT) {
+ /* from the sink pipeline, we don't know whether the master issued a seek,
+ as the seek_sent memory location isn't directly accesible to us, so we
+ look for a segment after a buffer to mean a seek was sent */
+ idx = pad2idx (pad, td->two_streams);
+ if (d->got_buffer_before_seek[idx])
+ d->got_segment_after_seek[idx] = TRUE;
+ } else if (GST_EVENT_TYPE (info->data) == GST_EVENT_FLUSH_START) {
+ idx = pad2idx (pad, td->two_streams);
+ d->got_flush_start[idx] = TRUE;
+ } else if (GST_EVENT_TYPE (info->data) == GST_EVENT_FLUSH_STOP) {
+ idx = pad2idx (pad, td->two_streams);
+ if (d->got_buffer_before_seek[idx])
+ d->got_flush_stop[idx] = TRUE;
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_flushing_seek_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, flushing_seek_probe, user_data);
+}
+
+static void
+setup_sink_flushing_seek (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_flushing_seek_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_flushing_seek (gpointer user_data)
+{
+ test_data *td = user_data;
+ const flushing_seek_input_data *i = td->id;
+ flushing_seek_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ FAIL_UNLESS (d->seek_sent);
+ FAIL_UNLESS (d->got_segment_done == i->segment_seek);
+}
+
+static void
+check_success_sink_flushing_seek (gpointer user_data)
+{
+ test_data *td = user_data;
+ flushing_seek_slave_data *d = td->sd;
+ gint idx;
+
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ FAIL_UNLESS (d->got_caps[idx]);
+ FAIL_UNLESS (d->got_buffer_before_seek[idx]);
+ FAIL_UNLESS (d->got_buffer_after_seek[idx]);
+ FAIL_UNLESS (d->got_segment_after_seek[idx]);
+ FAIL_UNLESS (d->got_flush_start[idx]);
+ FAIL_UNLESS (d->got_flush_stop[idx]);
+ FAIL_UNLESS (d->first_buffer_after_seek_has_timestamp_0[idx]);
+ }
+}
+
+GST_START_TEST (test_empty_flushing_seek)
+{
+ flushing_seek_input_data id = { FALSE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_flushing_seek)
+{
+ flushing_seek_input_data id = { FALSE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_flushing_seek)
+{
+ flushing_seek_input_data id = { FALSE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_flushing_seek)
+{
+ flushing_seek_input_data id = { FALSE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ flushing_seek_source, setup_sink_flushing_seek,
+ check_success_source_flushing_seek, check_success_sink_flushing_seek, &id,
+ &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_flushing_seek)
+{
+ flushing_seek_input_data id = { FALSE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_flushing_seek)
+{
+ flushing_seek_input_data id = { FALSE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_flushing_seek)
+{
+ flushing_seek_input_data id = { FALSE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ flushing_seek_source, setup_sink_flushing_seek,
+ check_success_source_flushing_seek, check_success_sink_flushing_seek, &id,
+ &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_empty_flushing_seek_in_pause)
+{
+ flushing_seek_input_data id = { FALSE, TRUE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_flushing_seek_in_pause)
+{
+ flushing_seek_input_data id = { FALSE, TRUE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_flushing_seek_in_pause)
+{
+ flushing_seek_input_data id = { FALSE, TRUE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_flushing_seek_in_pause)
+{
+ flushing_seek_input_data id = { FALSE, TRUE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ flushing_seek_source, setup_sink_flushing_seek,
+ check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_empty_segment_seek)
+{
+ flushing_seek_input_data id = { TRUE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_segment_seek)
+{
+ flushing_seek_input_data id = { TRUE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, flushing_seek_source,
+ setup_sink_flushing_seek, check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_segment_seek)
+{
+ flushing_seek_input_data id = { TRUE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE,
+ flushing_seek_source, setup_sink_flushing_seek,
+ check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_segment_seek)
+{
+ flushing_seek_input_data id = { TRUE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE,
+ flushing_seek_source, setup_sink_flushing_seek,
+ check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_segment_seek)
+{
+ flushing_seek_input_data id = { TRUE, FALSE };
+ flushing_seek_master_data md = { 0 };
+ flushing_seek_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ flushing_seek_source, setup_sink_flushing_seek,
+ check_success_source_flushing_seek,
+ check_success_sink_flushing_seek, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** seek stress test ****/
+
+typedef struct
+{
+ gint n_flushing_seeks;
+ gint n_paused_seeks;
+ gint n_segment_seeks;
+} seek_stress_input_data;
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ gboolean got_eos;
+ gboolean seek_sent;
+ guint64 t0;
+} seek_stress_master_data;
+
+static gboolean
+send_seek_stress (gpointer user_data)
+{
+ test_data *td = user_data;
+ seek_stress_input_data *i = td->id;
+ seek_stress_master_data *d = td->md;
+ GstEvent *seek_event;
+ unsigned int available, seekidx;
+ GstClockTime t, base;
+
+ /* Live streams don't like to be seeked too far away from the
+ "current" time, since they're live, so always seek near the
+ "real" time, so we still exercise seeking to another position
+ but still land somewhere close enough to "live" position. */
+ t = (g_get_monotonic_time () - d->t0) * 1000;
+ base = t > GST_SECOND / 2 ? t - GST_SECOND / 2 : 0;
+ t = base + g_random_int_range (0, GST_SECOND);
+
+ /* pick a random seek type among the ones we have left */
+ available = i->n_flushing_seeks + i->n_paused_seeks + i->n_segment_seeks;
+ if (available == 0) {
+ GST_DEBUG_BIN_TO_DOT_FILE_WITH_TS (GST_BIN (td->p),
+ GST_DEBUG_GRAPH_SHOW_ALL, "inter.test.toplaying");
+ FAIL_UNLESS (gst_element_set_state (td->p,
+ GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE);
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline,
+ gst_object_ref (td->p));
+ gst_object_unref (td->p);
+ return G_SOURCE_REMOVE;
+ }
+
+ seekidx = rand () % available;
+ if (seekidx < i->n_flushing_seeks) {
+ GST_INFO_OBJECT (td->p, "Sending flushing seek to %" GST_TIME_FORMAT,
+ GST_TIME_ARGS (t));
+ FAIL_UNLESS (gst_element_set_state (td->p,
+ GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE);
+ FAIL_UNLESS (gst_element_seek_simple (td->p, GST_FORMAT_TIME,
+ GST_SEEK_FLAG_FLUSH, t));
+ --i->n_flushing_seeks;
+ return G_SOURCE_CONTINUE;
+ }
+ seekidx -= i->n_flushing_seeks;
+
+ if (seekidx < i->n_paused_seeks) {
+ GST_INFO_OBJECT (td->p,
+ "Sending flushing seek in paused to %" GST_TIME_FORMAT,
+ GST_TIME_ARGS (t));
+ FAIL_UNLESS (gst_element_set_state (td->p,
+ GST_STATE_PAUSED) != GST_STATE_CHANGE_FAILURE);
+ FAIL_UNLESS (gst_element_seek_simple (td->p, GST_FORMAT_TIME,
+ GST_SEEK_FLAG_FLUSH, t));
+ --i->n_paused_seeks;
+ return G_SOURCE_CONTINUE;
+ }
+ seekidx -= i->n_paused_seeks;
+
+ GST_INFO_OBJECT (td->p, "Sending segment seek to %" GST_TIME_FORMAT,
+ GST_TIME_ARGS (t));
+ seek_event =
+ gst_event_new_seek (1.0, GST_FORMAT_TIME,
+ GST_SEEK_FLAG_SEGMENT | GST_SEEK_FLAG_FLUSH, GST_SEEK_TYPE_SET, t,
+ GST_SEEK_TYPE_SET, t + 5 * GST_SECOND);
+ FAIL_UNLESS (gst_element_send_event (td->p, seek_event));
+ --i->n_segment_seeks;
+ return G_SOURCE_CONTINUE;
+}
+
+static gboolean
+seek_stress_bus_msg (GstBus * bus, GstMessage * message, gpointer user_data)
+{
+ test_data *td = user_data;
+ seek_stress_master_data *d = td->md;
+
+ if (GST_IS_PIPELINE (GST_MESSAGE_SRC (message))) {
+ if (GST_MESSAGE_TYPE (message) == GST_MESSAGE_EOS ||
+ GST_MESSAGE_TYPE (message) == GST_MESSAGE_SEGMENT_DONE) {
+ d->got_eos = TRUE;
+ }
+ }
+ return master_bus_msg (bus, message, user_data);
+}
+
+static void
+seek_stress_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ seek_stress_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing) {
+ d->got_state_changed_to_playing = TRUE;
+ d->t0 = g_get_monotonic_time ();
+ gst_object_ref (td->p);
+ g_timeout_add (10, (GSourceFunc) send_seek_stress, td);
+ }
+}
+
+static void
+seek_stress_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ /* we're on the source, there's already the basic master_bus_msg watch,
+ and gst doesn't want more than one watch, so we remove the watch and
+ call it directly when done in the new watch */
+ gst_bus_remove_watch (GST_ELEMENT_BUS (source));
+ gst_bus_add_watch (GST_ELEMENT_BUS (source), seek_stress_bus_msg, user_data);
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = seek_stress_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static void
+check_success_source_seek_stress (gpointer user_data)
+{
+ test_data *td = user_data;
+ seek_stress_input_data *i = td->id;
+ seek_stress_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ FAIL_UNLESS_EQUALS_INT (i->n_flushing_seeks, 0);
+ FAIL_UNLESS_EQUALS_INT (i->n_paused_seeks, 0);
+ FAIL_UNLESS_EQUALS_INT (i->n_segment_seeks, 0);
+ FAIL_IF (d->got_eos);
+}
+
+GST_START_TEST (test_empty_seek_stress)
+{
+ seek_stress_input_data id = { 100, 100, 100 };
+ seek_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, seek_stress_source, NULL,
+ check_success_source_seek_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_seek_stress)
+{
+ seek_stress_input_data id = { 100, 100, 100 };
+ seek_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, seek_stress_source, NULL,
+ check_success_source_seek_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_seek_stress)
+{
+ seek_stress_input_data id = { 100, 100, 0 };
+ seek_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, seek_stress_source, NULL,
+ check_success_source_seek_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_seek_stress)
+{
+ seek_stress_input_data id = { 100, 100, 0 };
+ seek_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ seek_stress_source, NULL, check_success_source_seek_stress, NULL, &id,
+ &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_seek_stress)
+{
+ seek_stress_input_data id = { 100, 0, 100 };
+ seek_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE | TEST_FEATURE_LONG_DURATION,
+ seek_stress_source, NULL, check_success_source_seek_stress, NULL, &id,
+ &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_seek_stress)
+{
+ seek_stress_input_data id = { 100, 0, 100 };
+ seek_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_LONG_DURATION,
+ seek_stress_source, NULL, check_success_source_seek_stress, NULL, &id,
+ &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_seek_stress)
+{
+ seek_stress_input_data id = { 100, 0, 100 };
+ seek_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_LONG_DURATION |
+ TEST_FEATURE_SPLIT_SINKS,
+ seek_stress_source, NULL, check_success_source_seek_stress, NULL, &id,
+ &md, NULL);
+}
+
+GST_END_TEST;
+
+/**** upstream query test ****/
+
+typedef struct
+{
+ GstClockTime expected_duration;
+
+ /* In this test, the source does a position query (in the source pipeline
+ process), and must check its return against the last buffer timestamp
+ in the sink pipeline process. We open a pipe to let the sink send us
+ the timestamps it receives so the source can make the comparison. */
+ gint ts_pipes[2];
+} upstream_query_input_data;
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ gboolean got_correct_position;
+ gboolean got_correct_duration;
+ GstClockTime last_buffer_ts;
+} upstream_query_master_data;
+
+typedef struct
+{
+ gboolean got_caps[2];
+ gboolean got_buffer[2];
+ GstClockTime last_buffer_ts;
+} upstream_query_slave_data;
+
+static gboolean
+send_upstream_queries (gpointer user_data)
+{
+ test_data *td = user_data;
+ upstream_query_input_data *i = td->id;
+ upstream_query_master_data *d = td->md;
+ gint64 pos, dur, last;
+
+ FAIL_UNLESS (gst_element_query_position (td->p, GST_FORMAT_TIME, &pos));
+
+ /* read up the buffer ts sent by the sink process till the last one */
+ while (read (i->ts_pipes[0], &last, sizeof (last)) == sizeof (last)) {
+ /* timestamps may not be increasing because we are getting ts from
+ * both the audio and video streams; the position query will report
+ * the higher */
+ if (last > d->last_buffer_ts)
+ d->last_buffer_ts = last;
+ }
+ if (ABS ((gint64) (pos - d->last_buffer_ts)) <= CLOSE_ENOUGH_TO_ZERO)
+ d->got_correct_position = TRUE;
+
+ FAIL_UNLESS (gst_element_query_duration (td->p, GST_FORMAT_TIME, &dur));
+ if (GST_CLOCK_TIME_IS_VALID (i->expected_duration)) {
+ GstClockTimeDiff diff = GST_CLOCK_DIFF (dur, i->expected_duration);
+ if (diff >= -CLOSE_ENOUGH_TO_ZERO && diff <= CLOSE_ENOUGH_TO_ZERO)
+ d->got_correct_duration = TRUE;
+ } else {
+ if (!GST_CLOCK_TIME_IS_VALID (dur))
+ d->got_correct_duration = TRUE;
+ }
+
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline, td->p);
+ return FALSE;
+}
+
+static void
+upstream_query_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ upstream_query_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing) {
+ d->got_state_changed_to_playing = TRUE;
+ gst_object_ref (td->p);
+ g_timeout_add (QUERY_AT, (GSourceFunc) send_upstream_queries, td);
+ }
+}
+
+static void
+upstream_query_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ td->state_changed_cb = upstream_query_on_state_changed;
+ td->state_target = GST_STATE_PLAYING;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static GstPadProbeReturn
+upstream_query_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ upstream_query_input_data *i = td->id;
+ upstream_query_slave_data *d = td->sd;
+ GstCaps *caps;
+
+ if (GST_IS_BUFFER (info->data)) {
+ d->got_buffer[pad2idx (pad, td->two_streams)] = TRUE;
+ if (GST_BUFFER_TIMESTAMP_IS_VALID (info->data)) {
+ d->last_buffer_ts = GST_BUFFER_TIMESTAMP (info->data);
+ FAIL_UNLESS (write (i->ts_pipes[1], &d->last_buffer_ts,
+ sizeof (d->last_buffer_ts)) == sizeof (d->last_buffer_ts));
+ }
+ } else if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_CAPS) {
+ gst_event_parse_caps (info->data, &caps);
+ d->got_caps[caps2idx (caps, td->two_streams)] = TRUE;
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_upstream_query_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, upstream_query_probe, user_data);
+}
+
+static void
+setup_sink_upstream_query (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_upstream_query_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_upstream_query (gpointer user_data)
+{
+ test_data *td = user_data;
+ upstream_query_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ FAIL_UNLESS (d->got_correct_position);
+ FAIL_UNLESS (d->got_correct_duration);
+}
+
+static void
+check_success_sink_upstream_query (gpointer user_data)
+{
+ test_data *td = user_data;
+ upstream_query_slave_data *d = td->sd;
+ int idx;
+
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); ++idx) {
+ FAIL_UNLESS (d->got_caps[idx]);
+ FAIL_UNLESS (d->got_buffer[idx]);
+ }
+}
+
+GST_START_TEST (test_empty_upstream_query)
+{
+ upstream_query_input_data id = { GST_CLOCK_TIME_NONE, };
+ upstream_query_master_data md = { 0 };
+ upstream_query_slave_data sd = { 0 };
+
+ FAIL_UNLESS (pipe2 (id.ts_pipes, O_NONBLOCK) == 0);
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, upstream_query_source,
+ setup_sink_upstream_query, check_success_source_upstream_query,
+ check_success_sink_upstream_query, &id, &md, &sd);
+ close (id.ts_pipes[0]);
+ close (id.ts_pipes[1]);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_upstream_query)
+{
+ upstream_query_input_data id = { WAV_SAMPLE_ROUGH_DURATION, };
+ upstream_query_master_data md = { 0 };
+ upstream_query_slave_data sd = { 0 };
+
+ FAIL_UNLESS (pipe2 (id.ts_pipes, O_NONBLOCK) == 0);
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, upstream_query_source,
+ setup_sink_upstream_query, check_success_source_upstream_query,
+ check_success_sink_upstream_query, &id, &md, &sd);
+ close (id.ts_pipes[0]);
+ close (id.ts_pipes[1]);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_upstream_query)
+{
+ upstream_query_input_data id = { MPEGTS_SAMPLE_ROUGH_DURATION, };
+ upstream_query_master_data md = { 0 };
+ upstream_query_slave_data sd = { 0 };
+
+ FAIL_UNLESS (pipe2 (id.ts_pipes, O_NONBLOCK) == 0);
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, upstream_query_source,
+ setup_sink_upstream_query, check_success_source_upstream_query,
+ check_success_sink_upstream_query, &id, &md, &sd);
+ close (id.ts_pipes[0]);
+ close (id.ts_pipes[1]);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_upstream_query)
+{
+ upstream_query_input_data id = { MPEGTS_SAMPLE_ROUGH_DURATION, };
+ upstream_query_master_data md = { 0 };
+ upstream_query_slave_data sd = { 0 };
+
+ FAIL_UNLESS (pipe2 (id.ts_pipes, O_NONBLOCK) == 0);
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ upstream_query_source, setup_sink_upstream_query,
+ check_success_source_upstream_query, check_success_sink_upstream_query,
+ &id, &md, &sd);
+ close (id.ts_pipes[0]);
+ close (id.ts_pipes[1]);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_upstream_query)
+{
+ upstream_query_input_data id = { GST_CLOCK_TIME_NONE, };
+ upstream_query_master_data md = { 0 };
+ upstream_query_slave_data sd = { 0 };
+
+ FAIL_UNLESS (pipe2 (id.ts_pipes, O_NONBLOCK) == 0);
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE,
+ upstream_query_source, setup_sink_upstream_query,
+ check_success_source_upstream_query, check_success_sink_upstream_query,
+ &id, &md, &sd);
+ close (id.ts_pipes[0]);
+ close (id.ts_pipes[1]);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_upstream_query)
+{
+ upstream_query_input_data id = { GST_CLOCK_TIME_NONE, };
+ upstream_query_master_data md = { 0 };
+ upstream_query_slave_data sd = { 0 };
+
+ FAIL_UNLESS (pipe2 (id.ts_pipes, O_NONBLOCK) == 0);
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE,
+ upstream_query_source, setup_sink_upstream_query,
+ check_success_source_upstream_query, check_success_sink_upstream_query,
+ &id, &md, &sd);
+ close (id.ts_pipes[0]);
+ close (id.ts_pipes[1]);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_upstream_query)
+{
+ upstream_query_input_data id = { GST_CLOCK_TIME_NONE, };
+ upstream_query_master_data md = { 0 };
+ upstream_query_slave_data sd = { 0 };
+
+ FAIL_UNLESS (pipe2 (id.ts_pipes, O_NONBLOCK) == 0);
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ upstream_query_source, setup_sink_upstream_query,
+ check_success_source_upstream_query, check_success_sink_upstream_query,
+ &id, &md, &sd);
+ close (id.ts_pipes[0]);
+ close (id.ts_pipes[1]);
+}
+
+GST_END_TEST;
+
+/**** message test ****/
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ guint8 num_got_message;
+ guint8 num_sent_message;
+} message_master_data;
+
+static void
+send_ipcpipeline_test_message_event (const GValue * v, gpointer user_data)
+{
+ test_data *td = user_data;
+ message_master_data *d = td->md;
+ GstElement *element = g_value_get_object (v);
+ GstMessage *msg;
+ gboolean ret;
+
+ d->num_sent_message++;
+
+ msg = gst_message_new_element (GST_OBJECT (element),
+ gst_structure_new_empty ("ipcpipeline-test"));
+ ret = gst_element_send_event (element,
+ gst_event_new_sink_message ("ipcpipeline-test", msg));
+ FAIL_UNLESS (ret);
+ gst_message_unref (msg);
+}
+
+static gboolean
+send_sink_message (gpointer user_data)
+{
+ test_data *td = user_data;
+ GstIterator *it;
+
+ it = gst_bin_iterate_sources (GST_BIN (td->p));
+ while (gst_iterator_foreach (it, send_ipcpipeline_test_message_event, td))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ gst_object_unref (td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static gboolean
+message_bus_msg (GstBus * bus, GstMessage * message, gpointer user_data)
+{
+ test_data *td = user_data;
+ message_master_data *d = td->md;
+ const GstStructure *structure;
+
+ if (GST_MESSAGE_TYPE (message) == GST_MESSAGE_ELEMENT) {
+ structure = gst_message_get_structure (message);
+ FAIL_UNLESS (structure);
+ if (gst_structure_has_name (structure, "ipcpipeline-test")) {
+ d->num_got_message++;
+ if (d->num_got_message == d->num_sent_message)
+ g_main_loop_quit (loop);
+ }
+ }
+ return master_bus_msg (bus, message, user_data);
+}
+
+static void
+message_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ message_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing) {
+ d->got_state_changed_to_playing = TRUE;
+ gst_object_ref (td->p);
+ g_timeout_add (MESSAGE_AT, (GSourceFunc) send_sink_message, td);
+ }
+}
+
+static void
+message_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ /* we're on the source, there's already the basic master_bus_msg watch,
+ and gst doesn't want more than one watch, so we remove the watch and
+ call it directly when done in the new watch */
+ gst_bus_remove_watch (GST_ELEMENT_BUS (source));
+ gst_bus_add_watch (GST_ELEMENT_BUS (source), message_bus_msg, user_data);
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = message_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static void
+check_success_source_message (gpointer user_data)
+{
+ test_data *td = user_data;
+ message_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ FAIL_UNLESS_EQUALS_INT (d->num_got_message, d->num_sent_message);
+}
+
+GST_START_TEST (test_empty_message)
+{
+ message_master_data md = { 0 };
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, message_source, NULL,
+ check_success_source_message, NULL, NULL, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_message)
+{
+ message_master_data md = { 0 };
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, message_source, NULL,
+ check_success_source_message, NULL, NULL, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_message)
+{
+ message_master_data md = { 0 };
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE, message_source, NULL,
+ check_success_source_message, NULL, NULL, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_message)
+{
+ message_master_data md = { 0 };
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, message_source, NULL,
+ check_success_source_message, NULL, NULL, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_message)
+{
+ message_master_data md = { 0 };
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ message_source, NULL, check_success_source_message, NULL, NULL, &md,
+ NULL);
+}
+
+GST_END_TEST;
+
+/**** end of stream test ****/
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+} end_of_stream_master_data;
+
+typedef struct
+{
+ gboolean got_buffer[2];
+ gboolean got_eos[2];
+} end_of_stream_slave_data;
+
+static void
+end_of_stream_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ end_of_stream_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing)
+ d->got_state_changed_to_playing = TRUE;
+}
+
+static void
+end_of_stream_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ td->state_changed_cb = end_of_stream_on_state_changed;
+ td->state_target = GST_STATE_PLAYING;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static GstPadProbeReturn
+end_of_stream_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ end_of_stream_slave_data *d = td->sd;
+
+ if (GST_IS_BUFFER (info->data)) {
+ d->got_buffer[pad2idx (pad, td->two_streams)] = TRUE;
+ } else if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_EOS) {
+ d->got_eos[pad2idx (pad, td->two_streams)] = TRUE;
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_end_of_stream_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, end_of_stream_probe, user_data);
+}
+
+static void
+setup_sink_end_of_stream (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_end_of_stream_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_end_of_stream (gpointer user_data)
+{
+ test_data *td = user_data;
+ end_of_stream_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+}
+
+static void
+check_success_sink_end_of_stream (gpointer user_data)
+{
+ test_data *td = user_data;
+ end_of_stream_slave_data *d = td->sd;
+ int idx;
+
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ FAIL_UNLESS (d->got_buffer[idx]);
+ FAIL_UNLESS (d->got_eos[idx]);
+ }
+}
+
+GST_START_TEST (test_empty_end_of_stream)
+{
+ end_of_stream_master_data md = { 0 };
+ end_of_stream_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_ASYNC_SINK,
+ end_of_stream_source, setup_sink_end_of_stream,
+ check_success_source_end_of_stream, check_success_sink_end_of_stream,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_end_of_stream)
+{
+ end_of_stream_master_data md = { 0 };
+ end_of_stream_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE | TEST_FEATURE_ASYNC_SINK,
+ end_of_stream_source, setup_sink_end_of_stream,
+ check_success_source_end_of_stream, check_success_sink_end_of_stream,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_end_of_stream)
+{
+ end_of_stream_master_data md = { 0 };
+ end_of_stream_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_ASYNC_SINK,
+ end_of_stream_source, setup_sink_end_of_stream,
+ check_success_source_end_of_stream, check_success_sink_end_of_stream,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_end_of_stream)
+{
+ end_of_stream_master_data md = { 0 };
+ end_of_stream_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS |
+ TEST_FEATURE_ASYNC_SINK,
+ end_of_stream_source, setup_sink_end_of_stream,
+ check_success_source_end_of_stream, check_success_sink_end_of_stream,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_end_of_stream)
+{
+ end_of_stream_master_data md = { 0 };
+ end_of_stream_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE,
+ end_of_stream_source, setup_sink_end_of_stream,
+ check_success_source_end_of_stream, check_success_sink_end_of_stream,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_end_of_stream)
+{
+ end_of_stream_master_data md = { 0 };
+ end_of_stream_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE,
+ end_of_stream_source, setup_sink_end_of_stream,
+ check_success_source_end_of_stream, check_success_sink_end_of_stream,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_end_of_stream)
+{
+ end_of_stream_master_data md = { 0 };
+ end_of_stream_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ end_of_stream_source, setup_sink_end_of_stream,
+ check_success_source_end_of_stream, check_success_sink_end_of_stream,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** reverse playback test ****/
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ gboolean seek_sent;
+} reverse_playback_master_data;
+
+typedef struct
+{
+ gboolean got_segment_with_negative_rate;
+ gboolean got_buffer_after_segment_with_negative_rate;
+ GstClockTime first_backward_buffer_timestamp;
+ gboolean got_buffer_one_second_early;
+} reverse_playback_slave_data;
+
+static gboolean
+play_backwards (gpointer user_data)
+{
+ test_data *td = user_data;
+ reverse_playback_master_data *d = td->md;
+ gint64 pos;
+ gboolean ret;
+
+ FAIL_UNLESS (gst_element_query_position (td->p, GST_FORMAT_TIME, &pos));
+
+ ret =
+ gst_element_seek (td->p, -0.5, GST_FORMAT_TIME, 0, GST_SEEK_TYPE_SET, 0,
+ GST_SEEK_TYPE_SET, pos);
+ FAIL_UNLESS (ret);
+ d->seek_sent = TRUE;
+
+ gst_object_unref (td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static void
+reverse_playback_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ reverse_playback_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing) {
+ d->got_state_changed_to_playing = TRUE;
+ gst_object_ref (td->p);
+ g_timeout_add (2000, (GSourceFunc) play_backwards, td);
+ }
+}
+
+static void
+reverse_playback_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = reverse_playback_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static GstPadProbeReturn
+reverse_playback_probe (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ reverse_playback_slave_data *d = td->sd;
+
+ if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_SEGMENT) {
+ const GstSegment *s;
+ gst_event_parse_segment (GST_EVENT (info->data), &s);
+ if (s->rate < 0)
+ d->got_segment_with_negative_rate = TRUE;
+ }
+ } else if (GST_IS_BUFFER (info->data)) {
+ GstClockTime ts = GST_BUFFER_TIMESTAMP (info->data);
+ if (GST_CLOCK_TIME_IS_VALID (ts)) {
+ if (d->got_segment_with_negative_rate) {
+ if (d->got_buffer_after_segment_with_negative_rate) {
+ /* We test for 1 second, not just earlier, to make sure we don't
+ just see B frames, or whatever else */
+ if (ts < d->first_backward_buffer_timestamp - GST_SECOND) {
+ d->got_buffer_one_second_early = TRUE;
+ }
+ } else {
+ d->got_buffer_after_segment_with_negative_rate = TRUE;
+ d->first_backward_buffer_timestamp = ts;
+ }
+ }
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_reverse_playback_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, reverse_playback_probe, user_data);
+}
+
+static void
+setup_sink_reverse_playback (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_reverse_playback_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_reverse_playback (gpointer user_data)
+{
+ test_data *td = user_data;
+ reverse_playback_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ FAIL_UNLESS (d->seek_sent);
+}
+
+static void
+check_success_sink_reverse_playback (gpointer user_data)
+{
+ test_data *td = user_data;
+ reverse_playback_slave_data *d = td->sd;
+
+ FAIL_UNLESS (d->got_segment_with_negative_rate);
+ FAIL_UNLESS (d->got_buffer_after_segment_with_negative_rate);
+ FAIL_UNLESS (GST_CLOCK_TIME_IS_VALID (d->first_backward_buffer_timestamp));
+ FAIL_UNLESS (d->first_backward_buffer_timestamp >= GST_SECOND);
+ FAIL_UNLESS (d->got_buffer_one_second_early);
+}
+
+GST_START_TEST (test_a_reverse_playback)
+{
+ reverse_playback_master_data md = { 0 };
+ reverse_playback_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE,
+ reverse_playback_source, setup_sink_reverse_playback,
+ check_success_source_reverse_playback,
+ check_success_sink_reverse_playback, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_av_reverse_playback)
+{
+ reverse_playback_master_data md = { 0 };
+ reverse_playback_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_HAS_VIDEO,
+ reverse_playback_source, setup_sink_reverse_playback,
+ check_success_source_reverse_playback,
+ check_success_sink_reverse_playback, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_av_2_reverse_playback)
+{
+ reverse_playback_master_data md = { 0 };
+ reverse_playback_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_HAS_VIDEO |
+ TEST_FEATURE_SPLIT_SINKS,
+ reverse_playback_source, setup_sink_reverse_playback,
+ check_success_source_reverse_playback,
+ check_success_sink_reverse_playback, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** tags test ****/
+
+enum
+{
+ TEST_TAG_EMPTY,
+ TEST_TAG_TWO_TAGS,
+ N_TEST_TAGS
+};
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ gboolean tags_sent[N_TEST_TAGS];
+} tags_master_data;
+
+typedef struct
+{
+ gboolean tags_received[N_TEST_TAGS];
+} tags_slave_data;
+
+static void
+send_tags_on_element (const GValue * v, gpointer user_data)
+{
+ test_data *td = user_data;
+ tags_master_data *d = td->md;
+ GstElement *sink;
+ GstPad *pad;
+ GstEvent *e;
+
+ sink = g_value_get_object (v);
+ FAIL_UNLESS (sink);
+ pad = gst_element_get_static_pad (sink, "sink");
+ FAIL_UNLESS (pad);
+
+ e = gst_event_new_tag (gst_tag_list_new_empty ());
+ FAIL_UNLESS (gst_pad_send_event (pad, e));
+ d->tags_sent[TEST_TAG_EMPTY] = TRUE;
+
+ e = gst_event_new_tag (gst_tag_list_new (GST_TAG_TITLE, "title",
+ GST_TAG_BITRATE, 56000, NULL));
+ FAIL_UNLESS (gst_pad_send_event (pad, e));
+ d->tags_sent[TEST_TAG_TWO_TAGS] = TRUE;
+
+ gst_object_unref (pad);
+}
+
+static gboolean
+send_tags (gpointer user_data)
+{
+ test_data *td = user_data;
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (td->p));
+ while (gst_iterator_foreach (it, send_tags_on_element, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline, td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static void
+tags_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ tags_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing) {
+ d->got_state_changed_to_playing = TRUE;
+ gst_object_ref (td->p);
+ g_timeout_add (STEP_AT, (GSourceFunc) send_tags, td);
+ }
+}
+
+static void
+tags_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = tags_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static GstPadProbeReturn
+tags_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ tags_slave_data *d = td->sd;
+ guint funsigned;
+ gchar *fstring = NULL;
+
+ if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_TAG) {
+ GstTagList *taglist = NULL;
+ gst_event_parse_tag (GST_EVENT (info->data), &taglist);
+ FAIL_UNLESS (taglist);
+ if (gst_tag_list_is_empty (taglist)) {
+ d->tags_received[TEST_TAG_EMPTY] = TRUE;
+ } else if (gst_tag_list_get_string (taglist, GST_TAG_TITLE, &fstring)
+ && !strcmp (fstring, "title")
+ && gst_tag_list_get_uint (taglist, GST_TAG_BITRATE, &funsigned)
+ && funsigned == 56000) {
+ d->tags_received[TEST_TAG_TWO_TAGS] = TRUE;
+ }
+ }
+ }
+ g_free (fstring);
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_tags_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, tags_probe, user_data);
+}
+
+static void
+setup_sink_tags (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_tags_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_tags (gpointer user_data)
+{
+ test_data *td = user_data;
+ tags_master_data *d = td->md;
+ gint n;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ for (n = 0; n < N_TEST_TAGS; ++n) {
+ FAIL_UNLESS (d->tags_sent[n]);
+ }
+}
+
+static void
+check_success_sink_tags (gpointer user_data)
+{
+ test_data *td = user_data;
+ tags_slave_data *d = td->sd;
+ gint n;
+
+ for (n = 0; n < N_TEST_TAGS; ++n) {
+ FAIL_UNLESS (d->tags_received[n]);
+ }
+}
+
+GST_START_TEST (test_empty_tags)
+{
+ tags_master_data md = { 0 };
+ tags_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, tags_source, setup_sink_tags,
+ check_success_source_tags, check_success_sink_tags, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_tags)
+{
+ tags_master_data md = { 0 };
+ tags_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, tags_source, setup_sink_tags,
+ check_success_source_tags, check_success_sink_tags, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_tags)
+{
+ tags_master_data md = { 0 };
+ tags_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, tags_source, setup_sink_tags,
+ check_success_source_tags, check_success_sink_tags, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_tags)
+{
+ tags_master_data md = { 0 };
+ tags_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS, tags_source,
+ setup_sink_tags, check_success_source_tags, check_success_sink_tags, NULL,
+ &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_tags)
+{
+ tags_master_data md = { 0 };
+ tags_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE, tags_source, setup_sink_tags,
+ check_success_source_tags, check_success_sink_tags, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_tags)
+{
+ tags_master_data md = { 0 };
+ tags_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, tags_source, setup_sink_tags,
+ check_success_source_tags, check_success_sink_tags, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_tags)
+{
+ tags_master_data md = { 0 };
+ tags_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ tags_source, setup_sink_tags, check_success_source_tags,
+ check_success_sink_tags, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** nagivation test ****/
+
+enum
+{
+ TEST_NAV_MOUSE_MOVE,
+ TEST_NAV_KEY_PRESS,
+ N_NAVIGATION_EVENTS
+};
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ gboolean navigation_received[N_NAVIGATION_EVENTS];
+} navigation_master_data;
+
+typedef struct
+{
+ gboolean started;
+ gboolean navigation_sent[N_NAVIGATION_EVENTS];
+ gint step;
+} navigation_slave_data;
+
+static GstPadProbeReturn
+navigation_probe_source (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ navigation_master_data *d = td->md;
+ const GstStructure *s;
+ const gchar *string, *key;
+ double x, y;
+
+ if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_NAVIGATION) {
+ s = gst_event_get_structure (info->data);
+ FAIL_UNLESS (s);
+
+ /* mouse-move */
+ string = gst_structure_get_string (s, "event");
+ if (string && !strcmp (string, "mouse-move")) {
+ if (gst_structure_get_double (s, "pointer_x", &x) && x == 4.7) {
+ if (gst_structure_get_double (s, "pointer_y", &y) && y == 0.1) {
+ d->navigation_received[TEST_NAV_MOUSE_MOVE] = TRUE;
+ }
+ }
+ }
+
+ /* key-press */
+ string = gst_structure_get_string (s, "event");
+ if (string && !strcmp (string, "key-press")) {
+ key = gst_structure_get_string (s, "key");
+ if (key && !strcmp (key, "Left")) {
+ d->navigation_received[TEST_NAV_KEY_PRESS] = TRUE;
+ }
+ }
+
+ /* drop at this point to imply successful handling; the upstream filesrc
+ * does not know how to handle navigation events and returns FALSE,
+ * which makes the test fail */
+ return GST_PAD_PROBE_DROP;
+ }
+ }
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_navigation_probe_source (const GValue * v, gpointer user_data)
+{
+ hook_probe_types (v, navigation_probe_source,
+ GST_PAD_PROBE_TYPE_EVENT_UPSTREAM, user_data);
+}
+
+static void
+navigation_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ navigation_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing)
+ d->got_state_changed_to_playing = TRUE;
+}
+
+static void
+navigation_source (GstElement * source, void *user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (source));
+ while (gst_iterator_foreach (it, hook_navigation_probe_source, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = navigation_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static void
+send_navigation_event (const GValue * v, gpointer user_data)
+{
+ test_data *td = user_data;
+ navigation_slave_data *d = td->sd;
+ GstElement *sink;
+ GstPad *pad, *peer;
+ GstStructure *s;
+ GstEvent *e = NULL;
+
+ sink = g_value_get_object (v);
+ FAIL_UNLESS (sink);
+ pad = gst_element_get_static_pad (sink, "sink");
+ FAIL_UNLESS (pad);
+ peer = gst_pad_get_peer (pad);
+ FAIL_UNLESS (peer);
+ gst_object_unref (pad);
+
+ switch (d->step) {
+ case TEST_NAV_MOUSE_MOVE:
+ s = gst_structure_new ("application/x-gst-navigation", "event",
+ G_TYPE_STRING, "mouse-move", "button", G_TYPE_INT, 0, "pointer_x",
+ G_TYPE_DOUBLE, 4.7, "pointer_y", G_TYPE_DOUBLE, 0.1, NULL);
+ e = gst_event_new_navigation (s);
+ break;
+ case TEST_NAV_KEY_PRESS:
+ s = gst_structure_new ("application/x-gst-navigation", "event",
+ G_TYPE_STRING, "key-press", "key", G_TYPE_STRING, "Left", NULL);
+ e = gst_event_new_navigation (s);
+ break;
+ }
+
+ FAIL_UNLESS (e);
+ FAIL_UNLESS (gst_pad_send_event (peer, e));
+ d->navigation_sent[d->step] = TRUE;
+
+ gst_object_unref (peer);
+}
+
+static gboolean
+step_navigation (gpointer user_data)
+{
+ test_data *td = user_data;
+ navigation_slave_data *d = td->sd;
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (td->p));
+ while (gst_iterator_foreach (it, send_navigation_event, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ if (++d->step < N_NAVIGATION_EVENTS)
+ return G_SOURCE_CONTINUE;
+
+ /* we are in the slave; send EOS to force the master to stop the pipeline */
+ gst_element_post_message (GST_ELEMENT (td->p),
+ gst_message_new_eos (GST_OBJECT (td->p)));
+
+ gst_object_unref (td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static GstPadProbeReturn
+navigation_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ navigation_slave_data *d = td->sd;
+ GstClockTime ts;
+
+ if (GST_IS_BUFFER (info->data)) {
+ ts = GST_BUFFER_TIMESTAMP (info->data);
+ if (GST_CLOCK_TIME_IS_VALID (ts) && ts > STEP_AT * GST_MSECOND) {
+ if (!d->started) {
+ d->started = TRUE;
+ gst_object_ref (td->p);
+ g_timeout_add (50, step_navigation, td);
+ }
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_navigation_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, navigation_probe, user_data);
+}
+
+static void
+setup_sink_navigation (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_navigation_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_navigation (gpointer user_data)
+{
+ test_data *td = user_data;
+ navigation_master_data *d = td->md;
+ gint n;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ for (n = 0; n < N_NAVIGATION_EVENTS; ++n) {
+ FAIL_UNLESS (d->navigation_received[n]);
+ }
+}
+
+static void
+check_success_sink_navigation (gpointer user_data)
+{
+ test_data *td = user_data;
+ navigation_slave_data *d = td->sd;
+ gint n;
+
+ FAIL_UNLESS (d->started);
+ for (n = 0; n < N_NAVIGATION_EVENTS; ++n) {
+ FAIL_UNLESS (d->navigation_sent[n]);
+ }
+}
+
+GST_START_TEST (test_non_live_av_navigation)
+{
+ navigation_master_data md = { 0 };
+ navigation_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, navigation_source,
+ setup_sink_navigation, check_success_source_navigation,
+ check_success_sink_navigation, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_non_live_av_2_navigation)
+{
+ navigation_master_data md = { 0 };
+ navigation_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ navigation_source, setup_sink_navigation, check_success_source_navigation,
+ check_success_sink_navigation, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_navigation)
+{
+ navigation_master_data md = { 0 };
+ navigation_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, navigation_source,
+ setup_sink_navigation, check_success_source_navigation,
+ check_success_sink_navigation, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_navigation)
+{
+ navigation_master_data md = { 0 };
+ navigation_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ navigation_source, setup_sink_navigation, check_success_source_navigation,
+ check_success_sink_navigation, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** reconfigure test ****/
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ gboolean reconfigure_sent[2];
+} reconfigure_master_data;
+
+typedef struct
+{
+ gboolean reconfigure_scheduled;
+ gboolean reconfigure_sent[2];
+ gboolean got_caps[2][2];
+} reconfigure_slave_data;
+
+static GstPadProbeReturn
+reconfigure_source_probe (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ reconfigure_master_data *d = td->md;
+
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_RECONFIGURE) {
+ gint idx = pad2idx (pad, td->two_streams);
+ d->reconfigure_sent[idx] = TRUE;
+ if (!td->two_streams || d->reconfigure_sent[idx ? 0 : 1]) {
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline,
+ gst_object_ref (td->p));
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_reconfigure_source_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe_types (v, reconfigure_source_probe,
+ GST_PAD_PROBE_TYPE_EVENT_UPSTREAM, user_data);
+}
+
+static void
+reconfigure_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ reconfigure_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing)
+ d->got_state_changed_to_playing = TRUE;
+}
+
+static void
+reconfigure_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (source));
+ while (gst_iterator_foreach (it, hook_reconfigure_source_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = reconfigure_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static void
+send_reconfigure_on_element (const GValue * v, gpointer user_data)
+{
+ test_data *td = user_data;
+ reconfigure_slave_data *d = td->sd;
+ GstElement *sink, *capsfilter;
+ GstPad *pad, *peer;
+ GstCaps *caps = NULL;
+
+ sink = g_value_get_object (v);
+ FAIL_UNLESS (sink);
+ pad = gst_element_get_static_pad (sink, "sink");
+ FAIL_UNLESS (pad);
+
+ // look for the previous element, change caps if a capsfilter
+ peer = gst_pad_get_peer (pad);
+ FAIL_UNLESS (peer);
+ capsfilter = GST_ELEMENT (gst_pad_get_parent (peer));
+ g_object_get (capsfilter, "caps", &caps, NULL);
+ FAIL_UNLESS (caps);
+ caps = gst_caps_make_writable (caps);
+ if (!strcmp (gst_structure_get_name (gst_caps_get_structure (caps, 0)),
+ "audio/x-raw")) {
+ gst_caps_set_simple (caps, "rate", G_TYPE_INT, 48000, NULL);
+ } else {
+ gst_caps_set_simple (caps, "width", G_TYPE_INT, 320, "height", G_TYPE_INT,
+ 200, NULL);
+ }
+ g_object_set (capsfilter, "caps", caps, NULL);
+ FAIL_UNLESS (capsfilter);
+
+ gst_object_unref (capsfilter);
+ gst_object_unref (peer);
+
+ d->reconfigure_sent[caps2idx (caps, td->two_streams)] = TRUE;
+
+ gst_caps_unref (caps);
+ gst_object_unref (pad);
+}
+
+static gboolean
+send_reconfigure (gpointer user_data)
+{
+ test_data *td = user_data;
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (td->p));
+ while (gst_iterator_foreach (it, send_reconfigure_on_element, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ gst_object_unref (td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static GstPadProbeReturn
+reconfigure_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ reconfigure_slave_data *d = td->sd;
+ GstClockTime ts;
+ GstCaps *caps;
+ int idx;
+
+ if (GST_IS_BUFFER (info->data)) {
+ ts = GST_BUFFER_TIMESTAMP (info->data);
+ if (GST_CLOCK_TIME_IS_VALID (ts) && ts >= STEP_AT * GST_MSECOND) {
+ if (!d->reconfigure_scheduled) {
+ d->reconfigure_scheduled = TRUE;
+ gst_object_ref (td->p);
+ g_idle_add ((GSourceFunc) send_reconfigure, td);
+ }
+ }
+ } else if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_CAPS) {
+ gst_event_parse_caps (GST_EVENT (info->data), &caps);
+ idx = caps2idx (caps, td->two_streams);
+ if (d->reconfigure_sent[idx]) {
+ d->got_caps[idx][1] = TRUE;
+ } else {
+ d->got_caps[idx][0] = TRUE;
+ }
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_reconfigure_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, reconfigure_probe, user_data);
+}
+
+static void
+setup_sink_reconfigure (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_reconfigure_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_reconfigure (gpointer user_data)
+{
+ test_data *td = user_data;
+ reconfigure_master_data *d = td->md;
+ gint idx;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ FAIL_UNLESS (d->reconfigure_sent[idx]);
+ }
+}
+
+static void
+check_success_sink_reconfigure (gpointer user_data)
+{
+ test_data *td = user_data;
+ reconfigure_slave_data *d = td->sd;
+ gint idx;
+
+ FAIL_UNLESS (d->reconfigure_scheduled);
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ FAIL_UNLESS (d->reconfigure_sent[idx]);
+ FAIL_UNLESS (d->got_caps[idx][0]);
+ FAIL_UNLESS (d->got_caps[idx][1]);
+ }
+}
+
+GST_START_TEST (test_non_live_a_reconfigure)
+{
+ reconfigure_master_data md = { 0 };
+ reconfigure_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_FILTER_SINK_CAPS,
+ reconfigure_source, setup_sink_reconfigure,
+ check_success_source_reconfigure, check_success_sink_reconfigure, NULL,
+ &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_non_live_av_reconfigure)
+{
+ reconfigure_master_data md = { 0 };
+ reconfigure_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_HAS_VIDEO |
+ TEST_FEATURE_FILTER_SINK_CAPS,
+ reconfigure_source, setup_sink_reconfigure,
+ check_success_source_reconfigure, check_success_sink_reconfigure, NULL,
+ &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_reconfigure)
+{
+ reconfigure_master_data md = { 0 };
+ reconfigure_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE | TEST_FEATURE_FILTER_SINK_CAPS,
+ reconfigure_source, setup_sink_reconfigure,
+ check_success_source_reconfigure, check_success_sink_reconfigure, NULL,
+ &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_reconfigure)
+{
+ reconfigure_master_data md = { 0 };
+ reconfigure_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_FILTER_SINK_CAPS,
+ reconfigure_source, setup_sink_reconfigure,
+ check_success_source_reconfigure, check_success_sink_reconfigure, NULL,
+ &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** state changes test ****/
+
+typedef struct
+{
+ gint step;
+ GHashTable *fdin, *fdout;
+ gboolean waiting_state_change;
+} state_changes_master_data;
+
+typedef struct
+{
+ gint n_null;
+ gint n_ready;
+ gint n_paused;
+ gint n_playing;
+ gboolean got_eos;
+ GThread *thread;
+ gint refcount;
+} state_changes_slave_data;
+
+static void
+set_fdin (gpointer key, gpointer value, gpointer user_data)
+{
+ g_object_set (key, "fdin", GPOINTER_TO_INT (value), NULL);
+}
+
+static void
+set_fdout (gpointer key, gpointer value, gpointer user_data)
+{
+ g_object_set (key, "fdout", GPOINTER_TO_INT (value), NULL);
+}
+
+/*
+ * NULL
+ * 0: READY NULL READY PAUSED READY PAUSED READY NULL
+ * 8: READY PAUSED PLAYING PAUSED PLAYING PAUSED READY PAUSED READY NULL
+ * 18: disconnect
+ * 19: READY NULL READY PAUSED READY PAUSED READY NULL
+ * 27: READY PAUSED PLAYING PAUSED PLAYING PAUSED READY PAUSED READY NULL
+ * 37: reconnect
+ * 38: READY NULL READY PAUSED READY PAUSED READY NULL
+ * 46: READY PAUSED PLAYING PAUSED PLAYING
+ * 51: EOS
+ */
+static gboolean
+step_state_changes (gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_master_data *d = td->md;
+ gboolean ret = G_SOURCE_CONTINUE;
+ GstStateChangeReturn scret = GST_STATE_CHANGE_FAILURE;
+ GList *l;
+ int fdin, fdout;
+
+ if (d->waiting_state_change)
+ goto done;
+
+ switch (d->step++) {
+ case 1:
+ case 7:
+ case 17:
+ case 20:
+ case 26:
+ case 36:
+ case 39:
+ case 45:
+ scret = gst_element_set_state (td->p, GST_STATE_NULL);
+ FAIL_UNLESS_EQUALS_INT (scret, GST_STATE_CHANGE_SUCCESS);
+ break;
+ case 0:
+ case 2:
+ case 4:
+ case 6:
+ case 8:
+ case 14:
+ case 16:
+ case 38:
+ case 40:
+ case 42:
+ case 44:
+ case 46:
+ scret = gst_element_set_state (td->p, GST_STATE_READY);
+ FAIL_UNLESS_EQUALS_INT (scret, GST_STATE_CHANGE_SUCCESS);
+ break;
+ case 19:
+ case 21:
+ case 23:
+ case 25:
+ case 27:
+ case 33:
+ case 35:
+ /* while we are disconnected, we can't do NULL -> READY */
+ scret = gst_element_set_state (td->p, GST_STATE_READY);
+ FAIL_UNLESS_EQUALS_INT (scret, GST_STATE_CHANGE_FAILURE);
+ break;
+ case 3:
+ case 5:
+ case 9:
+ case 11:
+ case 13:
+ case 15:
+ case 41:
+ case 43:
+ case 47:
+ case 49:
+ td->state_target = GST_STATE_PAUSED;
+ scret = gst_element_set_state (td->p, GST_STATE_PAUSED);
+ FAIL_IF (scret == GST_STATE_CHANGE_FAILURE);
+ break;
+ case 22:
+ case 24:
+ case 28:
+ case 30:
+ case 32:
+ case 34:
+ /* while we are disconnected, we can't do NULL -> READY */
+ scret = gst_element_set_state (td->p, GST_STATE_PAUSED);
+ FAIL_UNLESS_EQUALS_INT (scret, GST_STATE_CHANGE_FAILURE);
+ break;
+ case 10:
+ case 12:
+ case 48:
+ case 50:
+ td->state_target = GST_STATE_PLAYING;
+ scret = gst_element_set_state (td->p, GST_STATE_PLAYING);
+ FAIL_IF (scret == GST_STATE_CHANGE_FAILURE);
+ break;
+ case 29:
+ case 31:
+ /* while we are disconnected, we can't do NULL -> READY */
+ scret = gst_element_set_state (td->p, GST_STATE_PLAYING);
+ FAIL_UNLESS_EQUALS_INT (scret, GST_STATE_CHANGE_FAILURE);
+ break;
+ case 18:
+ d->fdin = g_hash_table_new (g_direct_hash, g_direct_equal);
+ d->fdout = g_hash_table_new (g_direct_hash, g_direct_equal);
+ for (l = weak_refs; l; l = l->next) {
+ g_object_get (l->data, "fdin", &fdin, "fdout", &fdout, NULL);
+ g_hash_table_insert (d->fdin, (gpointer) l->data,
+ GINT_TO_POINTER (fdin));
+ g_hash_table_insert (d->fdout, (gpointer) l->data,
+ GINT_TO_POINTER (fdout));
+ g_signal_emit_by_name (G_OBJECT (l->data), "disconnect", NULL);
+ }
+ break;
+ case 37:
+ g_hash_table_foreach (d->fdin, set_fdin, NULL);
+ g_hash_table_foreach (d->fdout, set_fdout, NULL);
+ g_hash_table_destroy (d->fdin);
+ g_hash_table_destroy (d->fdout);
+ break;
+ case 51:
+ /* send EOS early to avoid waiting for the actual end of the file */
+ gst_element_send_event (td->p, gst_event_new_eos ());
+ gst_object_unref (td->p);
+ ret = G_SOURCE_REMOVE;
+ break;
+ }
+
+ if (scret == GST_STATE_CHANGE_ASYNC)
+ d->waiting_state_change = TRUE;
+
+done:
+ return ret;
+}
+
+static void
+state_changes_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_master_data *d = td->md;
+
+ d->waiting_state_change = FALSE;
+}
+
+static void
+state_changes_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_master_data *d = td->md;
+
+ gst_object_ref (source);
+ g_timeout_add (STEP_AT, (GSourceFunc) step_state_changes, td);
+
+ d->waiting_state_change = FALSE;
+ td->state_changed_cb = state_changes_state_changed;
+}
+
+static GstPadProbeReturn
+state_changes_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_slave_data *d = td->sd;
+
+ if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_EOS) {
+ d->got_eos = TRUE;
+ if (g_atomic_int_dec_and_test (&d->refcount)) {
+ g_thread_join (d->thread);
+ gst_object_unref (td->p);
+ }
+ return GST_PAD_PROBE_REMOVE;
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_state_changes_probe (const GValue * v, gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_slave_data *d = td->sd;
+
+ d->refcount++;
+ hook_probe (v, state_changes_probe, user_data);
+}
+
+static gpointer
+state_changes_watcher (gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_slave_data *d = td->sd;
+ GstState state = GST_STATE_VOID_PENDING, prev_state = GST_STATE_VOID_PENDING;
+ GstStateChangeReturn ret;
+
+ while (!d->got_eos) {
+ ret = gst_element_get_state (td->p, &state, NULL, GST_CLOCK_TIME_NONE);
+ if (ret == GST_STATE_CHANGE_SUCCESS && state != GST_STATE_VOID_PENDING) {
+ if (state != prev_state) {
+ switch (state) {
+ case GST_STATE_NULL:
+ d->n_null++;
+ break;
+ case GST_STATE_READY:
+ d->n_ready++;
+ break;
+ case GST_STATE_PAUSED:
+ d->n_paused++;
+ break;
+ case GST_STATE_PLAYING:
+ d->n_playing++;
+ break;
+ default:
+ fail_if (1);
+ }
+ prev_state = state;
+ }
+ }
+ g_usleep (STEP_AT * 1000 / 4);
+ }
+ return NULL;
+}
+
+static void
+setup_sink_state_changes (GstElement * sink, gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_slave_data *d = td->sd;
+ GstIterator *it;
+
+ gst_object_ref (sink);
+ d->refcount = 0;
+ d->thread =
+ g_thread_new ("state-changes-watcher",
+ (GThreadFunc) state_changes_watcher, td);
+ FAIL_UNLESS (d->thread);
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_state_changes_probe, td))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_state_changes (gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_master_data *d = td->md;
+
+ FAIL_UNLESS_EQUALS_INT (d->step, 52);
+}
+
+static void
+check_success_sink_state_changes (gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_slave_data *d = td->sd;
+
+ FAIL_UNLESS (d->got_eos);
+ FAIL_UNLESS_EQUALS_INT (d->n_null, 6);
+ FAIL_UNLESS_EQUALS_INT (d->n_ready, 12);
+ FAIL_UNLESS_EQUALS_INT (d->n_paused, 10);
+ FAIL_UNLESS_EQUALS_INT (d->n_playing, 4);
+}
+
+GST_START_TEST (test_empty_state_changes)
+{
+ state_changes_master_data md = { 0 };
+ state_changes_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, state_changes_source,
+ setup_sink_state_changes, check_success_source_state_changes,
+ check_success_sink_state_changes, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_state_changes)
+{
+ state_changes_master_data md = { 0 };
+ state_changes_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, state_changes_source,
+ setup_sink_state_changes, check_success_source_state_changes,
+ check_success_sink_state_changes, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_state_changes)
+{
+ state_changes_master_data md = { 0 };
+ state_changes_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, state_changes_source,
+ setup_sink_state_changes, check_success_source_state_changes,
+ check_success_sink_state_changes, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_state_changes)
+{
+ state_changes_master_data md = { 0 };
+ state_changes_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ state_changes_source, setup_sink_state_changes,
+ check_success_source_state_changes, check_success_sink_state_changes,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** state changes stress test ****/
+
+typedef struct
+{
+ gint n_state_changes;
+} state_changes_stress_input_data;
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+ gboolean async_state_change_completed;
+} state_changes_stress_master_data;
+
+static gboolean
+step_state_changes_stress (gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_stress_input_data *i = td->id;
+ state_changes_stress_master_data *d = td->md;
+ static const GstState states[] =
+ { GST_STATE_NULL, GST_STATE_READY, GST_STATE_PAUSED, GST_STATE_PLAYING };
+ GstState state;
+ GstStateChangeReturn ret;
+
+ /* wait for async state change to complete before continuing */
+ if (!d->async_state_change_completed)
+ return G_SOURCE_CONTINUE;
+
+ if (i->n_state_changes == 0) {
+ ret = gst_element_set_state (td->p, GST_STATE_PLAYING);
+ FAIL_IF (ret == GST_STATE_CHANGE_FAILURE);
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline, td->p);
+ return G_SOURCE_REMOVE;
+ }
+ --i->n_state_changes;
+
+ state = states[rand () % 4];
+ ret = gst_element_set_state (td->p, state);
+ FAIL_IF (ret == GST_STATE_CHANGE_FAILURE);
+
+ if (ret == GST_STATE_CHANGE_ASYNC) {
+ td->state_target = state;
+ d->async_state_change_completed = FALSE;
+ }
+
+ return G_SOURCE_CONTINUE;
+}
+
+static void
+state_changes_stress_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_stress_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing) {
+ d->got_state_changed_to_playing = TRUE;
+ gst_object_ref (td->p);
+ g_timeout_add (50, (GSourceFunc) step_state_changes_stress, td);
+ }
+ d->async_state_change_completed = TRUE;
+}
+
+static void
+state_changes_stress_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = state_changes_stress_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static void
+check_success_source_state_changes_stress (gpointer user_data)
+{
+ test_data *td = user_data;
+ state_changes_stress_input_data *i = td->id;
+ state_changes_stress_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+ FAIL_UNLESS_EQUALS_INT (i->n_state_changes, 0);
+}
+
+GST_START_TEST (test_empty_state_changes_stress)
+{
+ state_changes_stress_input_data id = { 500 };
+ state_changes_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, state_changes_stress_source, NULL,
+ check_success_source_state_changes_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_state_changes_stress)
+{
+ state_changes_stress_input_data id = { 500 };
+ state_changes_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, state_changes_stress_source, NULL,
+ check_success_source_state_changes_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_state_changes_stress)
+{
+ state_changes_stress_input_data id = { 500 };
+ state_changes_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, state_changes_stress_source, NULL,
+ check_success_source_state_changes_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_state_changes_stress)
+{
+ state_changes_stress_input_data id = { 500 };
+ state_changes_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ state_changes_stress_source, NULL,
+ check_success_source_state_changes_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_state_changes_stress)
+{
+ state_changes_stress_input_data id = { 500 };
+ state_changes_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE, state_changes_stress_source, NULL,
+ check_success_source_state_changes_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_state_changes_stress)
+{
+ state_changes_stress_input_data id = { 500 };
+ state_changes_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, state_changes_stress_source, NULL,
+ check_success_source_state_changes_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_state_changes_stress)
+{
+ state_changes_stress_input_data id = { 500 };
+ state_changes_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ state_changes_stress_source, NULL,
+ check_success_source_state_changes_stress, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+/**** serialized query test ****/
+
+typedef struct
+{
+ gboolean sent_query[2];
+ gboolean got_query_reply[2];
+ GstPad *pad[2];
+} serialized_query_master_data;
+
+typedef struct
+{
+ gboolean got_query;
+} serialized_query_slave_data;
+
+static gboolean
+send_drain (gpointer user_data)
+{
+ test_data *td = user_data;
+ serialized_query_master_data *d = td->md;
+ GstQuery *q;
+ gint idx;
+
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ q = gst_query_new_drain ();
+ FAIL_UNLESS (gst_pad_query (d->pad[idx], q));
+ d->got_query_reply[idx] = TRUE;
+ gst_query_unref (q);
+ gst_object_unref (d->pad[idx]);
+ }
+
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline, gst_object_ref (td->p));
+ return G_SOURCE_REMOVE;
+}
+
+static GstPadProbeReturn
+serialized_query_probe_source (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ serialized_query_master_data *d = td->md;
+ GstClockTime ts;
+ gint idx;
+
+ if (GST_IS_BUFFER (info->data)) {
+ ts = GST_BUFFER_TIMESTAMP (info->data);
+ idx = pad2idx (pad, td->two_streams);
+ if (!d->sent_query[idx] && GST_CLOCK_TIME_IS_VALID (ts)
+ && ts > STEP_AT * GST_MSECOND) {
+ g_atomic_int_set (&d->sent_query[idx], TRUE);
+ d->pad[idx] = gst_object_ref (pad);
+ if (!td->two_streams || g_atomic_int_get (&d->sent_query[idx ? 0 : 1]))
+ g_idle_add (send_drain, td);
+ }
+ }
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_serialized_query_probe_source (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, serialized_query_probe_source, user_data);
+}
+
+static void
+serialized_query_source (GstElement * source, gpointer user_data)
+{
+ GstIterator *it;
+ GstStateChangeReturn ret;
+
+ it = gst_bin_iterate_sinks (GST_BIN (source));
+ while (gst_iterator_foreach (it, hook_serialized_query_probe_source,
+ user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC
+ || ret == GST_STATE_CHANGE_SUCCESS);
+}
+
+static GstPadProbeReturn
+serialized_query_probe (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ serialized_query_slave_data *d = td->sd;
+
+ if (GST_IS_QUERY (info->data)) {
+ if (GST_QUERY_TYPE (info->data) == GST_QUERY_DRAIN) {
+ d->got_query = TRUE;
+ }
+ }
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_serialized_query_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, serialized_query_probe, user_data);
+}
+
+static void
+setup_sink_serialized_query (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_serialized_query_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_serialized_query (gpointer user_data)
+{
+ test_data *td = user_data;
+ serialized_query_master_data *d = td->md;
+ gint idx;
+
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ FAIL_UNLESS (d->sent_query[idx]);
+ FAIL_UNLESS (d->got_query_reply[idx]);
+ }
+}
+
+static void
+check_success_sink_serialized_query (gpointer user_data)
+{
+ test_data *td = user_data;
+ serialized_query_slave_data *d = td->sd;
+
+ FAIL_UNLESS (d->got_query);
+}
+
+GST_START_TEST (test_empty_serialized_query)
+{
+ serialized_query_master_data md = { 0 };
+ serialized_query_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, serialized_query_source,
+ setup_sink_serialized_query, check_success_source_serialized_query,
+ check_success_sink_serialized_query, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_serialized_query)
+{
+ serialized_query_master_data md = { 0 };
+ serialized_query_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, serialized_query_source,
+ setup_sink_serialized_query, check_success_source_serialized_query,
+ check_success_sink_serialized_query, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_serialized_query)
+{
+ serialized_query_master_data md = { 0 };
+ serialized_query_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, serialized_query_source,
+ setup_sink_serialized_query, check_success_source_serialized_query,
+ check_success_sink_serialized_query, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_serialized_query)
+{
+ serialized_query_master_data md = { 0 };
+ serialized_query_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ serialized_query_source, setup_sink_serialized_query,
+ check_success_source_serialized_query,
+ check_success_sink_serialized_query, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_serialized_query)
+{
+ serialized_query_master_data md = { 0 };
+ serialized_query_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE, serialized_query_source,
+ setup_sink_serialized_query, check_success_source_serialized_query,
+ check_success_sink_serialized_query, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_serialized_query)
+{
+ serialized_query_master_data md = { 0 };
+ serialized_query_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, serialized_query_source,
+ setup_sink_serialized_query, check_success_source_serialized_query,
+ check_success_sink_serialized_query, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_serialized_query)
+{
+ serialized_query_master_data md = { 0 };
+ serialized_query_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ serialized_query_source, setup_sink_serialized_query,
+ check_success_source_serialized_query,
+ check_success_sink_serialized_query, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** non serialized event test ****/
+
+typedef struct
+{
+ gboolean sent_event[2];
+} non_serialized_event_master_data;
+
+typedef struct
+{
+ gboolean got_event;
+} non_serialized_event_slave_data;
+
+static GstPadProbeReturn
+non_serialized_event_probe_source (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ non_serialized_event_master_data *d = td->md;
+ GstClockTime ts;
+ GstEvent *e;
+ gint idx;
+
+ if (GST_IS_BUFFER (info->data)) {
+ ts = GST_BUFFER_TIMESTAMP (info->data);
+ idx = pad2idx (pad, td->two_streams);
+ if (!g_atomic_int_get (&d->sent_event[idx])
+ && GST_CLOCK_TIME_IS_VALID (ts) && ts > STEP_AT * GST_MSECOND) {
+ e = gst_event_new_custom (GST_EVENT_CUSTOM_DOWNSTREAM_OOB,
+ gst_structure_new ("name", "field", G_TYPE_INT, 42, NULL));
+ FAIL_UNLESS (e);
+ FAIL_UNLESS (gst_pad_send_event (pad, e));
+
+ if (!td->two_streams || g_atomic_int_get (&d->sent_event[idx ? 0 : 1])) {
+ g_timeout_add (STEP_AT, (GSourceFunc) stop_pipeline,
+ gst_object_ref (td->p));
+ }
+ g_atomic_int_set (&d->sent_event[idx], TRUE);
+ }
+ }
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_non_serialized_event_probe_source (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, non_serialized_event_probe_source, user_data);
+}
+
+static void
+non_serialized_event_source (GstElement * source, gpointer user_data)
+{
+ GstIterator *it;
+ GstStateChangeReturn ret;
+
+ it = gst_bin_iterate_sinks (GST_BIN (source));
+ while (gst_iterator_foreach (it, hook_non_serialized_event_probe_source,
+ user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC
+ || ret == GST_STATE_CHANGE_SUCCESS);
+}
+
+static GstPadProbeReturn
+non_serialized_event_probe (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ non_serialized_event_slave_data *d = td->sd;
+ const GstStructure *s;
+ gint val;
+
+ if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_CUSTOM_DOWNSTREAM_OOB) {
+ s = gst_event_get_structure (info->data);
+ FAIL_UNLESS (!strcmp (gst_structure_get_name (s), "name"));
+ FAIL_UNLESS (gst_structure_get_int (s, "field", &val));
+ FAIL_UNLESS (val == 42);
+ d->got_event = TRUE;
+ }
+ }
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_non_serialized_event_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, non_serialized_event_probe, user_data);
+}
+
+static void
+setup_sink_non_serialized_event (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_non_serialized_event_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_non_serialized_event (gpointer user_data)
+{
+ test_data *td = user_data;
+ non_serialized_event_master_data *d = td->md;
+ gint idx;
+
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ FAIL_UNLESS (d->sent_event[idx]);
+ }
+}
+
+static void
+check_success_sink_non_serialized_event (gpointer user_data)
+{
+ test_data *td = user_data;
+ non_serialized_event_slave_data *d = td->sd;
+
+ FAIL_UNLESS (d->got_event);
+}
+
+GST_START_TEST (test_empty_non_serialized_event)
+{
+ non_serialized_event_master_data md = { 0 };
+ non_serialized_event_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, non_serialized_event_source,
+ setup_sink_non_serialized_event,
+ check_success_source_non_serialized_event,
+ check_success_sink_non_serialized_event, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_non_serialized_event)
+{
+ non_serialized_event_master_data md = { 0 };
+ non_serialized_event_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, non_serialized_event_source,
+ setup_sink_non_serialized_event,
+ check_success_source_non_serialized_event,
+ check_success_sink_non_serialized_event, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_non_serialized_event)
+{
+ non_serialized_event_master_data md = { 0 };
+ non_serialized_event_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, non_serialized_event_source,
+ setup_sink_non_serialized_event,
+ check_success_source_non_serialized_event,
+ check_success_sink_non_serialized_event, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_non_serialized_event)
+{
+ non_serialized_event_master_data md = { 0 };
+ non_serialized_event_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ non_serialized_event_source, setup_sink_non_serialized_event,
+ check_success_source_non_serialized_event,
+ check_success_sink_non_serialized_event, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_non_serialized_event)
+{
+ non_serialized_event_master_data md = { 0 };
+ non_serialized_event_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE, non_serialized_event_source,
+ setup_sink_non_serialized_event,
+ check_success_source_non_serialized_event,
+ check_success_sink_non_serialized_event, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_non_serialized_event)
+{
+ non_serialized_event_master_data md = { 0 };
+ non_serialized_event_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, non_serialized_event_source,
+ setup_sink_non_serialized_event,
+ check_success_source_non_serialized_event,
+ check_success_sink_non_serialized_event, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_non_serialized_event)
+{
+ non_serialized_event_master_data md = { 0 };
+ non_serialized_event_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ non_serialized_event_source, setup_sink_non_serialized_event,
+ check_success_source_non_serialized_event,
+ check_success_sink_non_serialized_event, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** meta test ****/
+
+enum
+{
+ TEST_META_PROTECTION = 0,
+ N_TEST_META
+};
+
+typedef struct
+{
+ gboolean meta_sent[N_TEST_META];
+} meta_master_data;
+
+typedef struct
+{
+ gboolean meta_received[N_TEST_META];
+} meta_slave_data;
+
+static GstPadProbeReturn
+meta_probe_source (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ meta_master_data *d = td->md;
+ GstBuffer *buffer;
+ GstProtectionMeta *meta;
+
+ if (GST_IS_BUFFER (info->data)) {
+ buffer = GST_BUFFER (info->data);
+ meta =
+ gst_buffer_add_protection_meta (buffer, gst_structure_new ("name",
+ "somefield", G_TYPE_INT, 42, NULL));
+ FAIL_UNLESS (meta);
+ d->meta_sent[TEST_META_PROTECTION] = TRUE;
+ }
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_meta_probe_source (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, meta_probe_source, user_data);
+}
+
+static void
+meta_source (GstElement * source, gpointer user_data)
+{
+ GstIterator *it;
+ GstStateChangeReturn ret;
+
+ it = gst_bin_iterate_sinks (GST_BIN (source));
+ while (gst_iterator_foreach (it, hook_meta_probe_source, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC
+ || ret == GST_STATE_CHANGE_SUCCESS);
+
+ g_timeout_add (STOP_AT, (GSourceFunc) stop_pipeline, gst_object_ref (source));
+}
+
+static gboolean
+scan_meta (GstBuffer * buffer, GstMeta ** meta, gpointer user_data)
+{
+ test_data *td = user_data;
+ meta_slave_data *d = td->sd;
+ int val;
+ GstStructure *s;
+ GstProtectionMeta *pmeta;
+
+ if ((*meta)->info->api == GST_PROTECTION_META_API_TYPE) {
+ pmeta = (GstProtectionMeta *) * meta;
+ FAIL_UNLESS (GST_IS_STRUCTURE (pmeta->info));
+ s = GST_STRUCTURE (pmeta->info);
+ FAIL_UNLESS (!strcmp (gst_structure_get_name (s), "name"));
+ FAIL_UNLESS (gst_structure_get_int (s, "somefield", &val));
+ FAIL_UNLESS (val == 42);
+ d->meta_received[TEST_META_PROTECTION] = TRUE;
+ }
+ return TRUE;
+}
+
+static GstPadProbeReturn
+meta_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ if (GST_IS_BUFFER (info->data)) {
+ gst_buffer_foreach_meta (info->data, scan_meta, user_data);
+ }
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_meta_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, meta_probe, user_data);
+}
+
+static void
+setup_sink_meta (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_meta_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_meta (gpointer user_data)
+{
+ test_data *td = user_data;
+ meta_master_data *d = td->md;
+ size_t n;
+
+ for (n = 0; n < N_TEST_META; ++n)
+ FAIL_UNLESS (d->meta_sent[n]);
+}
+
+static void
+check_success_sink_meta (gpointer user_data)
+{
+ test_data *td = user_data;
+ meta_slave_data *d = td->sd;
+ size_t n;
+
+ for (n = 0; n < N_TEST_META; ++n)
+ FAIL_UNLESS (d->meta_received[n]);
+}
+
+GST_START_TEST (test_empty_meta)
+{
+ meta_master_data md = { 0 };
+ meta_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE, meta_source, setup_sink_meta,
+ check_success_source_meta, check_success_sink_meta, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_meta)
+{
+ meta_master_data md = { 0 };
+ meta_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, meta_source, setup_sink_meta,
+ check_success_source_meta, check_success_sink_meta, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_meta)
+{
+ meta_master_data md = { 0 };
+ meta_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE, meta_source, setup_sink_meta,
+ check_success_source_meta, check_success_sink_meta, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_meta)
+{
+ meta_master_data md = { 0 };
+ meta_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_SPLIT_SINKS, meta_source,
+ setup_sink_meta, check_success_source_meta, check_success_sink_meta,
+ NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_meta)
+{
+ meta_master_data md = { 0 };
+ meta_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE, meta_source, setup_sink_meta,
+ check_success_source_meta, check_success_sink_meta, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_meta)
+{
+ meta_master_data md = { 0 };
+ meta_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, meta_source, setup_sink_meta,
+ check_success_source_meta, check_success_sink_meta, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_meta)
+{
+ meta_master_data md = { 0 };
+ meta_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ meta_source, setup_sink_meta, check_success_source_meta,
+ check_success_sink_meta, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** source change test ****/
+
+typedef struct
+{
+ void (*switcher) (GstElement *, char *name);
+} source_change_input_data;
+
+typedef struct
+{
+ gboolean source_change_scheduled;
+ gboolean source_changed;
+} source_change_master_data;
+
+typedef struct
+{
+ gboolean got_caps[2][2];
+ gboolean got_buffer[2][2];
+ GstCaps *caps[2];
+} source_change_slave_data;
+
+static gboolean
+stop_source (gpointer user_data)
+{
+ GstElement *source = user_data;
+
+ FAIL_UNLESS (gst_element_set_state (source,
+ GST_STATE_NULL) == GST_STATE_CHANGE_SUCCESS);
+ gst_object_unref (source);
+ return FALSE;
+}
+
+static gboolean
+remove_source (gpointer user_data)
+{
+ GstElement *source = user_data;
+
+ FAIL_UNLESS (gst_element_set_state (source,
+ GST_STATE_NULL) == GST_STATE_CHANGE_SUCCESS);
+ gst_bin_remove (GST_BIN (GST_ELEMENT_PARENT (source)), source);
+ return FALSE;
+}
+
+static void
+switch_to_aiff (GstElement * pipeline, char *name)
+{
+ GstElement *sbin, *filesrc, *ipcpipelinesink;
+ GError *e = NULL;
+
+ sbin =
+ gst_parse_bin_from_description ("pushfilesrc name=filesrc ! aiffparse",
+ TRUE, &e);
+ FAIL_IF (e || !sbin);
+ gst_element_set_name (sbin, name);
+ filesrc = gst_bin_get_by_name (GST_BIN (sbin), "filesrc");
+ FAIL_UNLESS (filesrc);
+ g_object_set (filesrc, "location", "../../tests/files/s16be-id3v2.aiff",
+ NULL);
+ gst_object_unref (filesrc);
+ gst_bin_add (GST_BIN (pipeline), sbin);
+ ipcpipelinesink = gst_bin_get_by_name (GST_BIN (pipeline), "ipcpipelinesink");
+ FAIL_UNLESS (ipcpipelinesink);
+ FAIL_UNLESS (gst_element_link (sbin, ipcpipelinesink));
+ gst_object_unref (ipcpipelinesink);
+ gst_element_sync_state_with_parent (sbin);
+ g_free (name);
+}
+
+static void
+switch_av (GstElement * pipeline, char *name, gboolean live, gboolean Long)
+{
+ GstElement *src, *ipcpipelinesink;
+ gint L = Long ? 10 : 1;
+
+ if (g_str_has_prefix (name, "videotestsrc")) {
+ /* replace video source with audio source */
+ src = gst_element_factory_make ("audiotestsrc", NULL);
+ FAIL_UNLESS (src);
+ g_object_set (src, "is-live", live, "num-buffers", live ? 27 * L : -1,
+ NULL);
+ gst_bin_add (GST_BIN (pipeline), src);
+ ipcpipelinesink =
+ gst_bin_get_by_name (GST_BIN (pipeline), "vipcpipelinesink");
+ FAIL_UNLESS (ipcpipelinesink);
+ FAIL_UNLESS (gst_element_link (src, ipcpipelinesink));
+ gst_object_unref (ipcpipelinesink);
+ gst_element_sync_state_with_parent (src);
+ }
+
+ if (g_str_has_prefix (name, "audiotestsrc")) {
+ /* replace audio source with video source */
+ src = gst_element_factory_make ("videotestsrc", NULL);
+ FAIL_UNLESS (src);
+ g_object_set (src, "is-live", live, "num-buffers", live ? 19 * L : -1,
+ NULL);
+ gst_bin_add (GST_BIN (pipeline), src);
+ ipcpipelinesink =
+ gst_bin_get_by_name (GST_BIN (pipeline), "aipcpipelinesink");
+ FAIL_UNLESS (ipcpipelinesink);
+ FAIL_UNLESS (gst_element_link (src, ipcpipelinesink));
+ gst_object_unref (ipcpipelinesink);
+ gst_element_sync_state_with_parent (src);
+ }
+
+ g_free (name);
+}
+
+static void
+switch_live_av (GstElement * pipeline, char *name)
+{
+ switch_av (pipeline, name, TRUE, FALSE);
+}
+
+static GstPadProbeReturn
+change_source_blocked (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ const source_change_input_data *i = td->id;
+ source_change_master_data *d = td->md;
+ GstElement *source;
+ GstPad *peer;
+
+ peer = gst_pad_get_peer (pad);
+ FAIL_UNLESS (peer);
+ FAIL_UNLESS (gst_pad_unlink (pad, peer));
+ gst_object_unref (peer);
+
+ source = GST_ELEMENT (gst_element_get_parent (pad));
+ FAIL_UNLESS (source);
+ g_object_set_qdata (G_OBJECT (source), to_be_removed_quark (),
+ GINT_TO_POINTER (1));
+
+ gst_bin_remove (GST_BIN (GST_ELEMENT_PARENT (source)), source);
+ (*i->switcher) (td->p, gst_element_get_name (source));
+
+ g_idle_add (stop_source, source);
+
+ d->source_changed = TRUE;
+
+ gst_object_unref (td->p);
+ return GST_PAD_PROBE_REMOVE;
+}
+
+static gboolean
+change_source (gpointer user_data)
+{
+ test_data *td = user_data;
+ GstElement *source;
+ GstPad *pad;
+ static const char *const names[] =
+ { "source", "audiotestsrc", "videotestsrc" };
+ gboolean found = FALSE;
+ size_t n;
+
+ for (n = 0; n < G_N_ELEMENTS (names); ++n) {
+ source = gst_bin_get_by_name (GST_BIN (td->p), names[n]);
+ if (source) {
+ found = TRUE;
+ pad = gst_element_get_static_pad (source, "src");
+ FAIL_UNLESS (pad);
+ gst_object_ref (td->p);
+ gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_IDLE, change_source_blocked,
+ user_data, NULL);
+ gst_object_unref (pad);
+ gst_object_unref (source);
+ }
+ }
+ FAIL_UNLESS (found);
+
+ gst_object_unref (td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static void
+source_change_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ source_change_master_data *d = td->md;
+
+ if (!d->source_change_scheduled) {
+ d->source_change_scheduled = TRUE;
+ gst_object_ref (td->p);
+ g_timeout_add (STEP_AT, change_source, td);
+ }
+}
+
+static void
+source_change_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = source_change_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC
+ || ret == GST_STATE_CHANGE_SUCCESS);
+}
+
+static int
+scppad2idx (GstPad * pad, gboolean two_streams, GstCaps * newcaps)
+{
+ static GQuark scpidx = 0;
+ gpointer p;
+ int idx;
+ GstCaps *caps;
+
+ if (!scpidx)
+ scpidx = g_quark_from_static_string ("scpidx");
+
+ if (!two_streams)
+ return 0;
+
+ p = g_object_get_qdata (G_OBJECT (pad), scpidx);
+ if (p)
+ return GPOINTER_TO_INT (p) - 1;
+
+ caps = gst_pad_get_current_caps (pad);
+ if (!caps)
+ caps = gst_pad_get_pad_template_caps (pad);
+ if ((!caps || gst_caps_is_any (caps)) && newcaps)
+ caps = gst_caps_ref (newcaps);
+ FAIL_UNLESS (caps);
+ idx = caps2idx (caps, two_streams);
+ gst_caps_unref (caps);
+ g_object_set_qdata (G_OBJECT (pad), scpidx, GINT_TO_POINTER (idx + 1));
+ return idx;
+}
+
+static GstPadProbeReturn
+source_change_probe (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ source_change_slave_data *d = td->sd;
+ GstCaps *caps;
+ int idx;
+
+ if (GST_IS_BUFFER (info->data)) {
+ idx = scppad2idx (pad, td->two_streams, NULL);
+ if (d->got_caps[idx][1])
+ d->got_buffer[idx][1] = TRUE;
+ else if (d->got_caps[idx][0])
+ d->got_buffer[idx][0] = TRUE;
+ } else if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_CAPS) {
+ gst_event_parse_caps (info->data, &caps);
+ idx = scppad2idx (pad, td->two_streams, caps);
+ if (!d->got_caps[idx][0]) {
+ FAIL_IF (d->caps[idx]);
+ d->got_caps[idx][0] = TRUE;
+ d->caps[idx] = gst_caps_ref (caps);
+ } else {
+ FAIL_UNLESS (d->caps);
+ if (gst_caps_is_equal (caps, d->caps[idx])) {
+ FAIL ();
+ } else {
+ gst_caps_replace (&d->caps[idx], NULL);
+ d->got_caps[idx][1] = TRUE;
+ }
+ }
+ }
+ }
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_source_change_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, source_change_probe, user_data);
+}
+
+static void
+setup_sink_source_change (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_source_change_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+}
+
+static void
+check_success_source_source_change (gpointer user_data)
+{
+ test_data *td = user_data;
+ source_change_master_data *d = td->md;
+
+ FAIL_UNLESS (d->source_change_scheduled);
+ FAIL_UNLESS (d->source_changed);
+}
+
+static void
+check_success_sink_source_change (gpointer user_data)
+{
+ test_data *td = user_data;
+ source_change_slave_data *d = td->sd;
+ int idx;
+
+ for (idx = 0; idx < (td->two_streams ? 2 : 1); idx++) {
+ FAIL_UNLESS (d->got_caps[idx][0]);
+ FAIL_UNLESS (d->got_buffer[idx][0]);
+ FAIL_UNLESS (d->got_caps[idx][1]);
+ FAIL_UNLESS (d->got_buffer[idx][1]);
+ }
+}
+
+GST_START_TEST (test_non_live_source_change)
+{
+ source_change_input_data id = { switch_to_aiff };
+ source_change_master_data md = { 0 };
+ source_change_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE, source_change_source,
+ setup_sink_source_change, check_success_source_source_change,
+ check_success_sink_source_change, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_source_change)
+{
+ source_change_input_data id = { switch_live_av };
+ source_change_master_data md = { 0 };
+ source_change_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, source_change_source,
+ setup_sink_source_change, check_success_source_source_change,
+ check_success_sink_source_change, &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_source_change)
+{
+ source_change_input_data id = { switch_live_av };
+ source_change_master_data md = { 0 };
+ source_change_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ source_change_source, setup_sink_source_change,
+ check_success_source_source_change, check_success_sink_source_change,
+ &id, &md, &sd);
+}
+
+GST_END_TEST;
+
+/**** dynamic pipeline change stress test ****/
+
+typedef struct
+{
+ guint n_switches_0;
+ void (*switcher0) (test_data *);
+ guint n_switches_1;
+ void (*switcher1) (test_data *);
+} dynamic_pipeline_change_stress_input_data;
+
+typedef struct
+{
+ GMutex mutex;
+ GCond cond;
+ guint n_blocks_left;
+ guint n_blocks_done;
+ gboolean adding_probes;
+ gboolean dynamic_pipeline_change_stress_scheduled;
+} dynamic_pipeline_change_stress_master_data;
+
+static gboolean dynamic_pipeline_change_stress_step (gpointer user_data);
+
+static GstPadProbeReturn
+dynamic_pipeline_change_stress_source_blocked_switch_av (GstPad * pad,
+ GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ dynamic_pipeline_change_stress_master_data *d = td->md;
+ GstElement *source;
+ GstPad *peer;
+
+ /* An idle pad probe could be called directly from the gst_pad_add_probe call
+ if the pad happens to be idle right now. This would deadlock us though, as
+ we need all pads to be blocked at the same time, so we need the iteration
+ over all pads to be done before the pad probes execute. So we keep track of
+ whether we're iterating to add the probes, and pass if so. */
+ if (d->adding_probes) {
+ return GST_PAD_PROBE_PASS;
+ }
+
+ peer = gst_pad_get_peer (pad);
+ FAIL_UNLESS (peer);
+ FAIL_UNLESS (gst_pad_unlink (pad, peer));
+ gst_object_unref (peer);
+
+ source = GST_ELEMENT (gst_element_get_parent (pad));
+ FAIL_UNLESS (source);
+ g_object_set_qdata (G_OBJECT (source), to_be_removed_quark (),
+ GINT_TO_POINTER (1));
+
+ /* we want all pads to be blocked before we proceed */
+ g_mutex_lock (&d->mutex);
+ d->n_blocks_left--;
+ while (d->n_blocks_left > 0)
+ g_cond_wait (&d->cond, &d->mutex);
+ g_mutex_unlock (&d->mutex);
+ g_cond_broadcast (&d->cond);
+
+ g_mutex_lock (&d->mutex);
+ switch_av (td->p, gst_element_get_name (source),
+ ! !(td->features & TEST_FEATURE_LIVE), TRUE);
+ g_mutex_unlock (&d->mutex);
+
+ g_idle_add_full (G_PRIORITY_HIGH, remove_source, source, g_object_unref);
+
+ if (g_atomic_int_dec_and_test (&d->n_blocks_done))
+ g_timeout_add (STEP_AT, dynamic_pipeline_change_stress_step, td);
+
+ return GST_PAD_PROBE_REMOVE;
+}
+
+static void
+change_audio_channel (GstElement * pipeline, char *name,
+ const char *ipcpipelinesink_name, gboolean live)
+{
+ GstElement *src, *ipcpipelinesink;
+
+ /* replace audio source with video source */
+ src = gst_element_factory_make ("audiotestsrc", NULL);
+ FAIL_UNLESS (src);
+ g_object_set (src, "is-live", live, "num-buffers", live ? 190 : -1, NULL);
+
+ gst_bin_add (GST_BIN (pipeline), src);
+ ipcpipelinesink =
+ gst_bin_get_by_name (GST_BIN (pipeline), ipcpipelinesink_name);
+ FAIL_UNLESS (ipcpipelinesink);
+ FAIL_UNLESS (gst_element_link (src, ipcpipelinesink));
+ gst_object_unref (ipcpipelinesink);
+ gst_element_sync_state_with_parent (src);
+
+ g_free (name);
+}
+
+static GstPadProbeReturn
+dynamic_pipeline_change_stress_source_blocked_change_audio_channel (GstPad *
+ pad, GstPadProbeInfo * info, gpointer user_data)
+{
+ test_data *td = user_data;
+ dynamic_pipeline_change_stress_master_data *d = td->md;
+ GstElement *source;
+ GstPad *peer;
+ const char *ipcpipelinesink_name;
+
+ /* An idle pad probe could be called directly from the gst_pad_add_probe call
+ if the pad happens to be idle right now. This would deadlock us though, as
+ we need all pads to be blocked at the same time, so we need the iteration
+ over all pads to be done before the pad probes execute. So we keep track of
+ whether we're iterating to add the probes, and pass if so. */
+ if (d->adding_probes) {
+ return GST_PAD_PROBE_PASS;
+ }
+
+ peer = gst_pad_get_peer (pad);
+ FAIL_UNLESS (peer);
+ ipcpipelinesink_name = GST_ELEMENT_NAME (GST_PAD_PARENT (peer));
+ FAIL_UNLESS (gst_pad_unlink (pad, peer));
+ gst_object_unref (peer);
+
+ source = GST_ELEMENT (gst_element_get_parent (pad));
+ FAIL_UNLESS (source);
+ g_object_set_qdata (G_OBJECT (source), to_be_removed_quark (),
+ GINT_TO_POINTER (1));
+
+ /* we want all pads to be blocked before we proceed */
+ g_mutex_lock (&d->mutex);
+ d->n_blocks_left--;
+ while (d->n_blocks_left > 0)
+ g_cond_wait (&d->cond, &d->mutex);
+ g_cond_broadcast (&d->cond);
+ g_mutex_unlock (&d->mutex);
+
+ g_mutex_lock (&d->mutex);
+ change_audio_channel (td->p, gst_element_get_name (source),
+ ipcpipelinesink_name, ! !(td->features & TEST_FEATURE_LIVE));
+ g_mutex_unlock (&d->mutex);
+
+ g_idle_add_full (G_PRIORITY_HIGH, remove_source, source, g_object_unref);
+
+ if (g_atomic_int_dec_and_test (&d->n_blocks_done))
+ g_timeout_add (STEP_AT, dynamic_pipeline_change_stress_step, td);
+
+ return GST_PAD_PROBE_REMOVE;
+}
+
+typedef struct
+{
+ const char *const *names;
+ size_t n_names;
+ GstPadProbeReturn (*f) (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data);
+ test_data *td;
+} block_if_named_data;
+
+static void
+block_if_named (const GValue * v, gpointer user_data)
+{
+ block_if_named_data *bind = user_data;
+ GstElement *e;
+ GstPad *pad;
+ size_t n;
+
+ e = g_value_get_object (v);
+ FAIL_UNLESS (e);
+ for (n = 0; n < bind->n_names; ++n) {
+ if (g_str_has_prefix (GST_ELEMENT_NAME (e), bind->names[n])) {
+ pad = gst_element_get_static_pad (e, "src");
+ FAIL_UNLESS (pad);
+
+ if (!g_object_get_qdata (G_OBJECT (e), to_be_removed_quark ()))
+ gst_pad_add_probe (pad, GST_PAD_PROBE_TYPE_IDLE, bind->f, bind->td,
+ NULL);
+ gst_object_unref (pad);
+ }
+ }
+}
+
+static void
+count_audio_sources (const GValue * v, gpointer user_data)
+{
+ GstElement *e;
+
+ e = g_value_get_object (v);
+ FAIL_UNLESS (e);
+
+ // we don't want to count the sources that are in the process
+ // of being removed asynchronously
+ if (g_object_get_qdata (G_OBJECT (e), to_be_removed_quark ()))
+ return;
+
+ if (g_str_has_prefix (GST_ELEMENT_NAME (e), "audiotestsrc"))
+ ++ * (guint *) user_data;
+}
+
+static void
+dynamic_pipeline_change_stress_swap_source (test_data * td)
+{
+ dynamic_pipeline_change_stress_master_data *d = td->md;
+ static const char *const names[] =
+ { "source", "audiotestsrc", "videotestsrc" };
+ block_if_named_data bind = { names, sizeof (names) / sizeof (names[0]),
+ dynamic_pipeline_change_stress_source_blocked_switch_av, td
+ };
+ GstIterator *it;
+
+ /* we have two sources, we need to wait for both */
+ d->n_blocks_left = d->n_blocks_done = 2;
+
+ it = gst_bin_iterate_sources (GST_BIN (td->p));
+ d->adding_probes = TRUE;
+ while (gst_iterator_foreach (it, block_if_named, &bind)) {
+ GST_INFO_OBJECT (td->p, "Resync");
+ gst_iterator_resync (it);
+ }
+ d->adding_probes = FALSE;
+ gst_iterator_free (it);
+}
+
+static void
+dynamic_pipeline_change_stress_change_audio_channel (test_data * td)
+{
+ dynamic_pipeline_change_stress_master_data *d = td->md;
+ static const char *const names[] = { "audiotestsrc" };
+ block_if_named_data bind = { names, sizeof (names) / sizeof (names[0]),
+ dynamic_pipeline_change_stress_source_blocked_change_audio_channel, td
+ };
+ GstIterator *it;
+ guint audio_sources;
+
+ /* we have either zero or one audio source */
+ it = gst_bin_iterate_sources (GST_BIN (td->p));
+ audio_sources = 0;
+ while (gst_iterator_foreach (it, count_audio_sources, &audio_sources)) {
+ GST_INFO_OBJECT (td->p, "Resync");
+ gst_iterator_resync (it);
+ }
+ gst_iterator_free (it);
+ d->n_blocks_left = d->n_blocks_done = audio_sources;
+
+ it = gst_bin_iterate_sources (GST_BIN (td->p));
+ d->adding_probes = TRUE;
+ while (gst_iterator_foreach (it, block_if_named, &bind)) {
+ GST_INFO_OBJECT (td->p, "Resync");
+ gst_iterator_resync (it);
+ }
+ d->adding_probes = FALSE;
+ gst_iterator_free (it);
+}
+
+static gboolean
+dynamic_pipeline_change_stress_step (gpointer user_data)
+{
+ test_data *td = user_data;
+ dynamic_pipeline_change_stress_input_data *i = td->id;
+ guint available, idx;
+
+ /* pick a random action among the ones we have left */
+ available = i->n_switches_0 + i->n_switches_1;
+ if (available == 0) {
+ GST_INFO_OBJECT (td->p, "Destroying pipeline");
+ FAIL_UNLESS (gst_element_set_state (td->p,
+ GST_STATE_PLAYING) != GST_STATE_CHANGE_FAILURE);
+ g_timeout_add (STEP_AT, stop_pipeline, td->p);
+ return G_SOURCE_REMOVE;
+ }
+
+ idx = rand () % available;
+ if (idx < i->n_switches_0) {
+ (*i->switcher0) (td);
+ --i->n_switches_0;
+ return G_SOURCE_REMOVE;
+ }
+ idx -= i->n_switches_0;
+
+ if (idx < i->n_switches_1) {
+ (*i->switcher1) (td);
+ --i->n_switches_1;
+ return G_SOURCE_REMOVE;
+ }
+ idx -= i->n_switches_1;
+
+ return G_SOURCE_REMOVE;
+}
+
+static void
+dynamic_pipeline_change_stress_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ dynamic_pipeline_change_stress_master_data *d = td->md;
+
+ if (!d->dynamic_pipeline_change_stress_scheduled) {
+ d->dynamic_pipeline_change_stress_scheduled = TRUE;
+ gst_object_ref (td->p);
+ g_timeout_add (STEP_AT, dynamic_pipeline_change_stress_step, td);
+ }
+}
+
+static void
+dynamic_pipeline_change_stress (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ dynamic_pipeline_change_stress_master_data *d = td->md;
+ GstStateChangeReturn ret;
+
+ g_mutex_init (&d->mutex);
+ g_cond_init (&d->cond);
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = dynamic_pipeline_change_stress_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC
+ || ret == GST_STATE_CHANGE_SUCCESS);
+}
+
+static void
+check_success_source_dynamic_pipeline_change_stress (gpointer user_data)
+{
+ test_data *td = user_data;
+ dynamic_pipeline_change_stress_input_data *i = td->id;
+ dynamic_pipeline_change_stress_master_data *d = td->md;
+
+ FAIL_UNLESS (d->dynamic_pipeline_change_stress_scheduled);
+ FAIL_UNLESS_EQUALS_INT (i->n_switches_0, 0);
+ FAIL_UNLESS_EQUALS_INT (i->n_switches_1, 0);
+
+ g_cond_clear (&d->cond);
+ g_mutex_clear (&d->mutex);
+}
+
+GST_START_TEST (test_non_live_av_dynamic_pipeline_change_stress)
+{
+ dynamic_pipeline_change_stress_input_data id = { 100,
+ dynamic_pipeline_change_stress_swap_source, 100,
+ dynamic_pipeline_change_stress_change_audio_channel
+ };
+ dynamic_pipeline_change_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_HAS_VIDEO,
+ dynamic_pipeline_change_stress, NULL,
+ check_success_source_dynamic_pipeline_change_stress, NULL, &id, &md,
+ NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_non_live_av_2_dynamic_pipeline_change_stress)
+{
+ dynamic_pipeline_change_stress_input_data id = { 100,
+ dynamic_pipeline_change_stress_swap_source, 100,
+ dynamic_pipeline_change_stress_change_audio_channel
+ };
+ dynamic_pipeline_change_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_HAS_VIDEO |
+ TEST_FEATURE_SPLIT_SINKS, dynamic_pipeline_change_stress, NULL,
+ check_success_source_dynamic_pipeline_change_stress, NULL, &id, &md,
+ NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_dynamic_pipeline_change_stress)
+{
+ dynamic_pipeline_change_stress_input_data id = { 100,
+ dynamic_pipeline_change_stress_swap_source, 100,
+ dynamic_pipeline_change_stress_change_audio_channel
+ };
+ dynamic_pipeline_change_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE, dynamic_pipeline_change_stress, NULL,
+ check_success_source_dynamic_pipeline_change_stress, NULL, &id, &md,
+ NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_dynamic_pipeline_change_stress)
+{
+ dynamic_pipeline_change_stress_input_data id = { 100,
+ dynamic_pipeline_change_stress_swap_source, 100,
+ dynamic_pipeline_change_stress_change_audio_channel
+ };
+ dynamic_pipeline_change_stress_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_SPLIT_SINKS,
+ dynamic_pipeline_change_stress, NULL,
+ check_success_source_dynamic_pipeline_change_stress, NULL, &id, &md,
+ NULL);
+}
+
+GST_END_TEST;
+
+/**** error from slave test ****/
+
+typedef struct
+{
+ gboolean crash;
+} error_from_slave_input_data;
+
+typedef struct
+{
+ gboolean second_pass;
+ gboolean got_state_changed_to_playing_on_first_pass;
+ gboolean got_error_on_first_pass;
+ gboolean got_state_changed_to_playing_on_second_pass;
+ gboolean got_error_on_second_pass;
+} error_from_slave_master_data;
+
+static gboolean
+bump_through_NULL (gpointer user_data)
+{
+ test_data *td = user_data;
+ error_from_slave_input_data *i = td->id;
+ error_from_slave_master_data *d = td->md;
+ GstStateChangeReturn ret;
+ GstElement *sink;
+
+ ret = gst_element_set_state (td->p, GST_STATE_NULL);
+ if (!i->crash) {
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_SUCCESS);
+ }
+ FAIL_UNLESS (gst_element_get_state (td->p, NULL, NULL,
+ GST_CLOCK_TIME_NONE) == GST_STATE_CHANGE_SUCCESS);
+
+ d->second_pass = TRUE;
+
+ if (i->crash) {
+ recreate_crashed_slave_process ();
+ /* give the process time to be created in the other process */
+ g_usleep (500 * 1000);
+
+ /* reconnect to to slave process */
+ sink = gst_bin_get_by_name (GST_BIN (td->p), "ipcpipelinesink");
+ FAIL_UNLESS (sink);
+ g_object_set (sink, "fdin", pipesba[0], "fdout", pipesfa[1], NULL);
+ gst_object_unref (sink);
+ }
+
+ ret = gst_element_set_state (td->p, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_SUCCESS
+ || ret == GST_STATE_CHANGE_ASYNC);
+
+ g_timeout_add (STOP_AT, (GSourceFunc) stop_pipeline, td->p);
+ return G_SOURCE_REMOVE;
+}
+
+static void
+disconnect (const GValue * v, gpointer user_data)
+{
+ GstElement *e;
+
+ e = g_value_get_object (v);
+ FAIL_UNLESS (e);
+ g_signal_emit_by_name (G_OBJECT (e), "disconnect", NULL);
+}
+
+static gboolean
+error_from_slave_source_bus_msg (GstBus * bus, GstMessage * message,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ error_from_slave_input_data *i = td->id;
+ error_from_slave_master_data *d = td->md;
+
+ if (GST_MESSAGE_TYPE (message) == GST_MESSAGE_ERROR) {
+ if (!d->second_pass) {
+ if (!d->got_error_on_first_pass) {
+ GstIterator *it;
+
+ d->got_error_on_first_pass = TRUE;
+
+ if (i->crash) {
+ it = gst_bin_iterate_sinks (GST_BIN (td->p));
+ while (gst_iterator_foreach (it, disconnect, NULL))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+ }
+
+ gst_object_ref (td->p);
+ g_timeout_add (STEP_AT, bump_through_NULL, td);
+ }
+
+ /* don't pass the expected error */
+ return TRUE;
+ }
+ } else if (GST_MESSAGE_TYPE (message) == GST_MESSAGE_EOS) {
+ if (!d->second_pass) {
+ /* We'll get an expected EOS as the source reacts to the error */
+ return TRUE;
+ }
+ }
+ return master_bus_msg (bus, message, user_data);
+}
+
+static void
+error_from_slave_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ error_from_slave_master_data *d = td->md;
+
+ if (d->second_pass)
+ d->got_state_changed_to_playing_on_second_pass = TRUE;
+ else
+ d->got_state_changed_to_playing_on_first_pass = TRUE;
+}
+
+static gboolean
+error_from_slave_position_getter (GstElement * element)
+{
+ gint64 pos;
+
+ /* we do not care about the result */
+ gst_element_query_position (element, GST_FORMAT_TIME, &pos);
+
+ return TRUE;
+}
+
+static void
+error_from_slave_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ /* we're on the source, there's already the basic master_bus_msg watch,
+ and gst doesn't want more than one watch, so we remove the watch and
+ call it directly when done in the new watch */
+ gst_bus_remove_watch (GST_ELEMENT_BUS (source));
+ gst_bus_add_watch (GST_ELEMENT_BUS (source), error_from_slave_source_bus_msg,
+ user_data);
+ g_timeout_add (STEP_AT, (GSourceFunc) error_from_slave_position_getter,
+ source);
+
+ td->state_changed_cb = error_from_slave_on_state_changed;
+ td->state_target = GST_STATE_PLAYING;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC);
+}
+
+static gboolean
+error_from_slave_sink_bus_msg (GstBus * bus, GstMessage * message,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ error_from_slave_input_data *i = td->id;
+
+ switch (GST_MESSAGE_TYPE (message)) {
+ case GST_MESSAGE_ERROR:
+ if (!strcmp (GST_ELEMENT_NAME (GST_MESSAGE_SRC (message)),
+ "error-element"))
+ g_object_set (GST_MESSAGE_SRC (message), "error-after", -1, NULL);
+ break;
+ case GST_MESSAGE_ASYNC_DONE:
+ if (GST_IS_PIPELINE (GST_MESSAGE_SRC (message))) {
+ /* We have two identical processes, and only one must crash. They can
+ be distinguished by recovery_pid, however. */
+ if (i->crash && recovery_pid)
+ g_timeout_add (CRASH_AT, (GSourceFunc) crash, NULL);
+ }
+ break;
+ default:
+ break;
+ }
+ return TRUE;
+}
+
+static void
+setup_sink_error_from_slave (GstElement * sink, gpointer user_data)
+{
+ gst_bus_add_watch (GST_ELEMENT_BUS (sink), error_from_slave_sink_bus_msg,
+ user_data);
+}
+
+static void
+check_success_source_error_from_slave (gpointer user_data)
+{
+ test_data *td = user_data;
+ error_from_slave_master_data *d = td->md;
+
+ FAIL_UNLESS (d->second_pass);
+ FAIL_UNLESS (d->got_state_changed_to_playing_on_first_pass);
+ FAIL_UNLESS (d->got_state_changed_to_playing_on_second_pass);
+ FAIL_UNLESS (d->got_error_on_first_pass);
+ FAIL_IF (d->got_error_on_second_pass);
+}
+
+GST_START_TEST (test_empty_error_from_slave)
+{
+ error_from_slave_input_data id = { FALSE };
+ error_from_slave_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_TEST_SOURCE | TEST_FEATURE_ERROR_SINK,
+ error_from_slave_source, setup_sink_error_from_slave,
+ check_success_source_error_from_slave, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_error_from_slave)
+{
+ error_from_slave_input_data id = { FALSE };
+ error_from_slave_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE | TEST_FEATURE_ERROR_SINK,
+ error_from_slave_source, setup_sink_error_from_slave,
+ check_success_source_error_from_slave, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_error_from_slave)
+{
+ error_from_slave_input_data id = { FALSE };
+ error_from_slave_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_ERROR_SINK,
+ error_from_slave_source, setup_sink_error_from_slave,
+ check_success_source_error_from_slave, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_mpegts_2_error_from_slave)
+{
+ error_from_slave_input_data id = { FALSE };
+ error_from_slave_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_MPEGTS_SOURCE | TEST_FEATURE_ERROR_SINK |
+ TEST_FEATURE_SPLIT_SINKS,
+ error_from_slave_source, setup_sink_error_from_slave,
+ check_success_source_error_from_slave, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_a_error_from_slave)
+{
+ error_from_slave_input_data id = { FALSE };
+ error_from_slave_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_A_SOURCE | TEST_FEATURE_ERROR_SINK,
+ error_from_slave_source, setup_sink_error_from_slave,
+ check_success_source_error_from_slave, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_error_from_slave)
+{
+ error_from_slave_input_data id = { FALSE };
+ error_from_slave_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_ERROR_SINK,
+ error_from_slave_source, setup_sink_error_from_slave,
+ check_success_source_error_from_slave, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_live_av_2_error_from_slave)
+{
+ error_from_slave_input_data id = { FALSE };
+ error_from_slave_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_LIVE_AV_SOURCE | TEST_FEATURE_ERROR_SINK |
+ TEST_FEATURE_SPLIT_SINKS,
+ error_from_slave_source, setup_sink_error_from_slave,
+ check_success_source_error_from_slave, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wavparse_slave_process_crash)
+{
+ error_from_slave_input_data id = { TRUE };
+ error_from_slave_master_data md = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE | TEST_FEATURE_RECOVERY_SLAVE_PROCESS,
+ error_from_slave_source, setup_sink_error_from_slave,
+ check_success_source_error_from_slave, NULL, &id, &md, NULL);
+}
+
+GST_END_TEST;
+
+/**** master process crash test ****/
+
+typedef struct
+{
+ gboolean got_state_changed_to_playing;
+} master_process_crash_master_data;
+
+typedef struct
+{
+ gboolean got_error;
+ gboolean got_eos;
+} master_process_crash_slave_data;
+
+static void
+master_process_crash_on_state_changed (gpointer user_data)
+{
+ test_data *td = user_data;
+ master_process_crash_master_data *d = td->md;
+
+ if (!d->got_state_changed_to_playing) {
+ d->got_state_changed_to_playing = TRUE;
+
+ /* We have two identical processes, and only one must crash. They can
+ be distinguished by recovery_pid, however. */
+ if (!recovery_pid)
+ g_timeout_add (CRASH_AT, (GSourceFunc) crash, NULL);
+ }
+}
+
+static void
+master_process_crash_source (GstElement * source, gpointer user_data)
+{
+ test_data *td = user_data;
+ GstStateChangeReturn ret;
+
+ td->state_target = GST_STATE_PLAYING;
+ td->state_changed_cb = master_process_crash_on_state_changed;
+ ret = gst_element_set_state (source, GST_STATE_PLAYING);
+ FAIL_UNLESS (ret == GST_STATE_CHANGE_ASYNC
+ || ret == GST_STATE_CHANGE_SUCCESS);
+}
+
+static GstPadProbeReturn
+master_process_crash_probe (GstPad * pad, GstPadProbeInfo * info,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ master_process_crash_slave_data *d = td->sd;
+
+ if (GST_IS_EVENT (info->data)) {
+ if (GST_EVENT_TYPE (info->data) == GST_EVENT_EOS) {
+ d->got_eos = TRUE;
+ }
+ }
+
+ return GST_PAD_PROBE_OK;
+}
+
+static void
+hook_master_process_crash_probe (const GValue * v, gpointer user_data)
+{
+ hook_probe (v, master_process_crash_probe, user_data);
+}
+
+static gboolean
+go_to_NULL_and_reconnect (gpointer user_data)
+{
+ GstElement *pipeline = user_data;
+ GstStateChangeReturn ret;
+ GstElement *src;
+
+ ret = gst_element_set_state (pipeline, GST_STATE_NULL);
+ FAIL_IF (ret == GST_STATE_CHANGE_FAILURE);
+
+ /* reconnect to to master process */
+ src = gst_bin_get_by_name (GST_BIN (pipeline), "ipcpipelinesrc0");
+ FAIL_UNLESS (src);
+ g_object_set (src, "fdin", pipesfa[0], "fdout", pipesba[1], NULL);
+ gst_object_unref (src);
+
+ gst_object_unref (pipeline);
+ return G_SOURCE_REMOVE;
+}
+
+static gboolean
+master_process_crash_bus_msg (GstBus * bus, GstMessage * message,
+ gpointer user_data)
+{
+ test_data *td = user_data;
+ master_process_crash_slave_data *d = td->sd;
+ GstIterator *it;
+
+ switch (GST_MESSAGE_TYPE (message)) {
+ case GST_MESSAGE_ERROR:
+ if (!d->got_error) {
+ it = gst_bin_iterate_sources (GST_BIN (td->p));
+ while (gst_iterator_foreach (it, disconnect, NULL))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+ g_timeout_add (10, (GSourceFunc) go_to_NULL_and_reconnect,
+ gst_object_ref (td->p));
+ d->got_error = TRUE;
+ }
+ break;
+ default:
+ break;
+ }
+ return TRUE;
+}
+
+static void
+setup_sink_master_process_crash (GstElement * sink, gpointer user_data)
+{
+ GstIterator *it;
+
+ it = gst_bin_iterate_sinks (GST_BIN (sink));
+ while (gst_iterator_foreach (it, hook_master_process_crash_probe, user_data))
+ gst_iterator_resync (it);
+ gst_iterator_free (it);
+
+ gst_bus_add_watch (GST_ELEMENT_BUS (sink), master_process_crash_bus_msg,
+ user_data);
+}
+
+static void
+check_success_source_master_process_crash (gpointer user_data)
+{
+ test_data *td = user_data;
+ master_process_crash_master_data *d = td->md;
+
+ FAIL_UNLESS (d->got_state_changed_to_playing);
+}
+
+static void
+check_success_sink_master_process_crash (gpointer user_data)
+{
+ test_data *td = user_data;
+ master_process_crash_slave_data *d = td->sd;
+
+ FAIL_UNLESS (d->got_error);
+ FAIL_UNLESS (d->got_eos);
+}
+
+GST_START_TEST (test_wavparse_master_process_crash)
+{
+ master_process_crash_master_data md = { 0 };
+ master_process_crash_slave_data sd = { 0 };
+
+ TEST_BASE (TEST_FEATURE_WAV_SOURCE | TEST_FEATURE_RECOVERY_MASTER_PROCESS,
+ master_process_crash_source, setup_sink_master_process_crash,
+ check_success_source_master_process_crash,
+ check_success_sink_master_process_crash, NULL, &md, &sd);
+}
+
+GST_END_TEST;
+
+static Suite *
+ipcpipeline_suite (void)
+{
+ Suite *s = suite_create ("ipcpipeline");
+ TCase *tc_chain = tcase_create ("general");
+
+ setup_lock ();
+
+ suite_add_tcase (s, tc_chain);
+ tcase_set_timeout (tc_chain, 180);
+
+ /* play_pause tests put the pipeline in PLAYING state, then in
+ PAUSED state, then in PLAYING state again. The sink expects
+ async-done messages or state change successes. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_play_pause);
+ tcase_add_test (tc_chain, test_wavparse_play_pause);
+ tcase_add_test (tc_chain, test_mpegts_play_pause);
+ tcase_add_test (tc_chain, test_mpegts_2_play_pause);
+ tcase_add_test (tc_chain, test_live_a_play_pause);
+ tcase_add_test (tc_chain, test_live_av_play_pause);
+ tcase_add_test (tc_chain, test_live_av_2_play_pause);
+ }
+
+ /* flushing_seek tests perform a flushing seek in PLAYING
+ state. The sinks check a buffer with the target timestamp
+ is received after the seek. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_flushing_seek);
+ tcase_add_test (tc_chain, test_wavparse_flushing_seek);
+ tcase_add_test (tc_chain, test_mpegts_flushing_seek);
+ tcase_add_test (tc_chain, test_mpegts_2_flushing_seek);
+ tcase_add_test (tc_chain, test_live_a_flushing_seek);
+ tcase_add_test (tc_chain, test_live_av_flushing_seek);
+ tcase_add_test (tc_chain, test_live_av_2_flushing_seek);
+ }
+
+ /* flushing_seek_in_pause tests perform a flushing seek in
+ PAUSED state. These are disabled for live pipelines since
+ those will not generate data in PAUSED, so we won't get
+ a buffer. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_flushing_seek_in_pause);
+ tcase_add_test (tc_chain, test_wavparse_flushing_seek_in_pause);
+ tcase_add_test (tc_chain, test_mpegts_flushing_seek_in_pause);
+ tcase_add_test (tc_chain, test_mpegts_2_flushing_seek_in_pause);
+ /* live scenarios skipped: live sources do not generate buffers
+ * when paused */
+ }
+
+ /* segment_seek tests perform a segment seek in PLAYING
+ state. The sinks check a buffer with the target timestamp
+ is received after the seek, and that a SEGMENT_DONE is
+ received at the end of the segment. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_segment_seek);
+ tcase_add_test (tc_chain, test_wavparse_segment_seek);
+ /* mpegts skipped: tsdemux does not support segment seeks */
+ tcase_add_test (tc_chain, test_live_a_segment_seek);
+ tcase_add_test (tc_chain, test_live_av_segment_seek);
+ tcase_add_test (tc_chain, test_live_av_2_segment_seek);
+ }
+
+ /* seek_stress tests perform stress testing on seeks, then waits
+ in PLAYING for EOS or segment-done. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_seek_stress);
+ tcase_add_test (tc_chain, test_wavparse_seek_stress);
+ tcase_add_test (tc_chain, test_mpegts_seek_stress);
+ tcase_add_test (tc_chain, test_mpegts_2_seek_stress);
+ tcase_add_test (tc_chain, test_live_a_seek_stress);
+ tcase_add_test (tc_chain, test_live_av_seek_stress);
+ tcase_add_test (tc_chain, test_live_av_2_seek_stress);
+ }
+
+ /* upstream_query tests send position and duration queries, and
+ checks the results are as expected. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_upstream_query);
+ tcase_add_test (tc_chain, test_wavparse_upstream_query);
+ tcase_add_test (tc_chain, test_mpegts_upstream_query);
+ tcase_add_test (tc_chain, test_mpegts_2_upstream_query);
+ tcase_add_test (tc_chain, test_live_a_upstream_query);
+ tcase_add_test (tc_chain, test_live_av_upstream_query);
+ tcase_add_test (tc_chain, test_live_av_2_upstream_query);
+ }
+
+ /* message tests send a sink message downstream, which causes
+ the sinks to reply with the embedded event, which is checked.
+ This is not possible when elements go into pull mode. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_message);
+ tcase_add_test (tc_chain, test_wavparse_message);
+ /* mpegts skipped because it goes into pull mode:
+ https://bugzilla.gnome.org/show_bug.cgi?id=751637 */
+ tcase_add_test (tc_chain, test_live_a_message);
+ tcase_add_test (tc_chain, test_live_av_message);
+ tcase_add_test (tc_chain, test_live_av_2_message);
+ }
+
+ /* end_of_stream tests check the EOS event and message are
+ properly received when the stream reaches its end. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_end_of_stream);
+ tcase_add_test (tc_chain, test_wavparse_end_of_stream);
+ tcase_add_test (tc_chain, test_mpegts_end_of_stream);
+ tcase_add_test (tc_chain, test_mpegts_2_end_of_stream);
+ tcase_add_test (tc_chain, test_live_a_end_of_stream);
+ tcase_add_test (tc_chain, test_live_av_end_of_stream);
+ tcase_add_test (tc_chain, test_live_av_2_end_of_stream);
+ }
+
+ /* reverse_playback tests issue a seek with negative rate,
+ and check buffers timestamp are in decreasing order.
+ This does not work with sources which do not support
+ negative playback rate (live ones, and some demuxers). */
+ if (1) {
+ /* wavparse and tsdemux does not support backward playback */
+ tcase_add_test (tc_chain, test_a_reverse_playback);
+ tcase_add_test (tc_chain, test_av_reverse_playback);
+ tcase_add_test (tc_chain, test_av_2_reverse_playback);
+ }
+
+ /* tags tests check tags are carried to the slave. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_tags);
+ tcase_add_test (tc_chain, test_wavparse_tags);
+ tcase_add_test (tc_chain, test_mpegts_tags);
+ tcase_add_test (tc_chain, test_mpegts_2_tags);
+ tcase_add_test (tc_chain, test_live_a_tags);
+ tcase_add_test (tc_chain, test_live_av_tags);
+ tcase_add_test (tc_chain, test_live_av_2_tags);
+ }
+
+ /* reconfigure tests that pipeline reconfiguration via
+ the reconfigure event works */
+ if (1) {
+ tcase_add_test (tc_chain, test_non_live_a_reconfigure);
+ tcase_add_test (tc_chain, test_non_live_av_reconfigure);
+ tcase_add_test (tc_chain, test_live_a_reconfigure);
+ tcase_add_test (tc_chain, test_live_av_reconfigure);
+ }
+
+ /* state_change tests issue a number of state changes in
+ (hopefully) all interesting configurations, and checks
+ the state changes occured on the slave pipeline. The links
+ are disconnected and reconnected to check it all still
+ works after this. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_state_changes);
+ tcase_add_test (tc_chain, test_wavparse_state_changes);
+ tcase_add_test (tc_chain, test_mpegts_state_changes);
+ tcase_add_test (tc_chain, test_mpegts_2_state_changes);
+ /* live scenarios skipped: live sources will cause no buffer
+ * to flow in PAUSED, so the pipeline will only finish READY->PAUSED
+ * once switching to PLAYING */
+ }
+
+ /* state_changes_stress tests change state randomly and rapidly. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_state_changes_stress);
+ tcase_add_test (tc_chain, test_wavparse_state_changes_stress);
+ tcase_add_test (tc_chain, test_mpegts_state_changes_stress);
+ tcase_add_test (tc_chain, test_mpegts_2_state_changes_stress);
+ tcase_add_test (tc_chain, test_live_a_state_changes_stress);
+ tcase_add_test (tc_chain, test_live_av_state_changes_stress);
+ tcase_add_test (tc_chain, test_live_av_2_state_changes_stress);
+ }
+
+ /* serialized_query tests checks that a serialized query is
+ handled by the slave pipeline. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_serialized_query);
+ tcase_add_test (tc_chain, test_wavparse_serialized_query);
+ tcase_add_test (tc_chain, test_mpegts_serialized_query);
+ tcase_add_test (tc_chain, test_mpegts_2_serialized_query);
+ tcase_add_test (tc_chain, test_live_a_serialized_query);
+ tcase_add_test (tc_chain, test_live_av_serialized_query);
+ tcase_add_test (tc_chain, test_live_av_2_serialized_query);
+ }
+
+ /* non_serialized_event tests checks that a non serialized event
+ is handled by the slave pipeline. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_non_serialized_event);
+ tcase_add_test (tc_chain, test_wavparse_non_serialized_event);
+ tcase_add_test (tc_chain, test_mpegts_non_serialized_event);
+ tcase_add_test (tc_chain, test_mpegts_2_non_serialized_event);
+ tcase_add_test (tc_chain, test_live_a_non_serialized_event);
+ tcase_add_test (tc_chain, test_live_av_non_serialized_event);
+ tcase_add_test (tc_chain, test_live_av_2_non_serialized_event);
+ }
+
+ /* meta tests checks that GstMeta on buffers are correctly
+ received by the slave pipeline. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_meta);
+ tcase_add_test (tc_chain, test_wavparse_meta);
+ tcase_add_test (tc_chain, test_mpegts_meta);
+ tcase_add_test (tc_chain, test_mpegts_2_meta);
+ tcase_add_test (tc_chain, test_live_a_meta);
+ tcase_add_test (tc_chain, test_live_av_meta);
+ tcase_add_test (tc_chain, test_live_av_2_meta);
+ }
+
+ /* source_change tests checks that the pipelines can handle a
+ change of source/caps. */
+ if (1) {
+ tcase_add_test (tc_chain, test_non_live_source_change);
+ tcase_add_test (tc_chain, test_live_av_source_change);
+ tcase_add_test (tc_chain, test_live_av_2_source_change);
+ }
+
+ /* navigation tests checks that navigation events from the slave
+ are received by the master. */
+ if (1) {
+ tcase_add_test (tc_chain, test_non_live_av_navigation);
+ tcase_add_test (tc_chain, test_non_live_av_2_navigation);
+ tcase_add_test (tc_chain, test_live_av_navigation);
+ tcase_add_test (tc_chain, test_live_av_2_navigation);
+ }
+
+ /* dynamic_pipeline_change_stress tests stress tests dynamic
+ pipeline changes. */
+ if (1) {
+ tcase_add_test (tc_chain, test_non_live_av_dynamic_pipeline_change_stress);
+ tcase_add_test (tc_chain,
+ test_non_live_av_2_dynamic_pipeline_change_stress);
+ tcase_add_test (tc_chain, test_live_av_dynamic_pipeline_change_stress);
+ tcase_add_test (tc_chain, test_live_av_2_dynamic_pipeline_change_stress);
+ }
+
+ /* error_from_slave tests checks an error message issued
+ by the slave pipeline is received by the master pipeline. */
+ if (1) {
+ tcase_add_test (tc_chain, test_empty_error_from_slave);
+ tcase_add_test (tc_chain, test_wavparse_error_from_slave);
+ tcase_add_test (tc_chain, test_mpegts_error_from_slave);
+ tcase_add_test (tc_chain, test_mpegts_2_error_from_slave);
+ tcase_add_test (tc_chain, test_live_a_error_from_slave);
+ tcase_add_test (tc_chain, test_live_av_error_from_slave);
+ tcase_add_test (tc_chain, test_live_av_2_error_from_slave);
+ }
+
+ /* slave_process_crash tests test that a crash of the slave
+ process can be recovered from by the master, which can
+ replace the slave process and continue. */
+ tcase_add_test (tc_chain, test_wavparse_slave_process_crash);
+
+ /* master_process_crash tests test that a crash of the master
+ process can be recovered from by the slave. I don't recall
+ how the recovery from that works, but it does! A watchdog
+ process replaces the master process, and the slave will
+ go to NULL and reconnect after it gets a timeout talking
+ with the master pipeline. */
+ tcase_add_test (tc_chain, test_wavparse_master_process_crash);
+
+ return s;
+}
+
+GST_CHECK_MAIN (ipcpipeline);