tests: Split splitmux test case
authorSeungha Yang <seungha@centricular.com>
Fri, 3 Apr 2020 07:49:25 +0000 (16:49 +0900)
committerSeungha Yang <seungha@centricular.com>
Fri, 3 Apr 2020 08:08:51 +0000 (17:08 +0900)
Since we are adding more and more tests into splitmux,
we need to split it to avoid CI timeout.

tests/check/elements/splitmux.c [deleted file]
tests/check/elements/splitmuxsink.c [new file with mode: 0644]
tests/check/elements/splitmuxsrc.c [new file with mode: 0644]
tests/check/meson.build

diff --git a/tests/check/elements/splitmux.c b/tests/check/elements/splitmux.c
deleted file mode 100644 (file)
index be8c5c9..0000000
+++ /dev/null
@@ -1,1293 +0,0 @@
-/* GStreamer unit test for splitmuxsrc/sink elements
- *
- * Copyright (C) 2007 David A. Schleef <ds@schleef.org>
- * Copyright (C) 2015 Jan Schmidt <jan@centricular.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.
- */
-
-#ifdef HAVE_CONFIG_H
-#  include "config.h"
-#endif
-
-#include <glib/gstdio.h>
-
-#include <gst/check/gstcheck.h>
-#include <gst/app/app.h>
-#include <gst/video/video.h>
-#include <stdlib.h>
-
-gchar *tmpdir = NULL;
-GstClockTime first_ts;
-GstClockTime last_ts;
-gdouble current_rate;
-
-static void
-tempdir_setup (void)
-{
-  const gchar *systmp = g_get_tmp_dir ();
-  tmpdir = g_build_filename (systmp, "splitmux-test-XXXXXX", NULL);
-  /* Rewrites tmpdir template input: */
-  tmpdir = g_mkdtemp (tmpdir);
-}
-
-static void
-tempdir_cleanup (void)
-{
-  GDir *d;
-  const gchar *f;
-
-  fail_if (tmpdir == NULL);
-
-  d = g_dir_open (tmpdir, 0, NULL);
-  fail_if (d == NULL);
-
-  while ((f = g_dir_read_name (d)) != NULL) {
-    gchar *fname = g_build_filename (tmpdir, f, NULL);
-    fail_if (g_remove (fname) != 0, "Failed to remove tmp file %s", fname);
-    g_free (fname);
-  }
-  g_dir_close (d);
-
-  fail_if (g_remove (tmpdir) != 0, "Failed to delete tmpdir %s", tmpdir);
-
-  g_free (tmpdir);
-  tmpdir = NULL;
-}
-
-static guint
-count_files (const gchar * target)
-{
-  GDir *d;
-  const gchar *f;
-  guint ret = 0;
-
-  d = g_dir_open (target, 0, NULL);
-  fail_if (d == NULL);
-
-  while ((f = g_dir_read_name (d)) != NULL)
-    ret++;
-  g_dir_close (d);
-
-  return ret;
-}
-
-static void
-dump_error (GstMessage * msg)
-{
-  GError *err = NULL;
-  gchar *dbg_info;
-
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR);
-
-  gst_message_parse_error (msg, &err, &dbg_info);
-
-  g_printerr ("ERROR from element %s: %s\n",
-      GST_OBJECT_NAME (msg->src), err->message);
-  g_printerr ("Debugging info: %s\n", (dbg_info) ? dbg_info : "none");
-  g_error_free (err);
-  g_free (dbg_info);
-}
-
-static GstMessage *
-run_pipeline (GstElement * pipeline)
-{
-  GstBus *bus = gst_element_get_bus (GST_ELEMENT (pipeline));
-  GstMessage *msg;
-
-  gst_element_set_state (pipeline, GST_STATE_PLAYING);
-  msg = gst_bus_poll (bus, GST_MESSAGE_EOS | GST_MESSAGE_ERROR, -1);
-  gst_element_set_state (pipeline, GST_STATE_NULL);
-
-  gst_object_unref (bus);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-
-  return msg;
-}
-
-static void
-seek_pipeline (GstElement * pipeline, gdouble rate, GstClockTime start,
-    GstClockTime end)
-{
-  /* Pause the pipeline, seek to the desired range / rate, wait for PAUSED again, then
-   * clear the tracking vars for start_ts / end_ts */
-  gst_element_set_state (pipeline, GST_STATE_PAUSED);
-  gst_element_get_state (pipeline, NULL, NULL, GST_CLOCK_TIME_NONE);
-
-  /* specific end time not implemented: */
-  fail_unless (end == GST_CLOCK_TIME_NONE);
-
-  gst_element_seek (pipeline, rate, GST_FORMAT_TIME,
-      GST_SEEK_FLAG_FLUSH | GST_SEEK_FLAG_ACCURATE, GST_SEEK_TYPE_SET, start,
-      GST_SEEK_TYPE_END, 0);
-
-  /* Wait for the pipeline to preroll again */
-  gst_element_get_state (pipeline, NULL, NULL, GST_CLOCK_TIME_NONE);
-
-  GST_LOG ("Seeked pipeline. Rate %f time range %" GST_TIME_FORMAT " to %"
-      GST_TIME_FORMAT, rate, GST_TIME_ARGS (start), GST_TIME_ARGS (end));
-
-  /* Clear tracking variables now that the seek is complete */
-  first_ts = last_ts = GST_CLOCK_TIME_NONE;
-  current_rate = rate;
-};
-
-static GstFlowReturn
-receive_sample (GstAppSink * appsink, gpointer user_data G_GNUC_UNUSED)
-{
-  GstSample *sample;
-  GstSegment *seg;
-  GstBuffer *buf;
-  GstClockTime start;
-  GstClockTime end;
-
-  g_signal_emit_by_name (appsink, "pull-sample", &sample);
-  fail_unless (sample != NULL);
-
-  seg = gst_sample_get_segment (sample);
-  fail_unless (seg != NULL);
-
-  buf = gst_sample_get_buffer (sample);
-  fail_unless (buf != NULL);
-
-  GST_LOG ("Got buffer %" GST_PTR_FORMAT, buf);
-
-  start = GST_BUFFER_PTS (buf);
-  end = start;
-
-  if (GST_CLOCK_TIME_IS_VALID (start))
-    start = gst_segment_to_stream_time (seg, GST_FORMAT_TIME, start);
-
-  if (GST_CLOCK_TIME_IS_VALID (end)) {
-    if (GST_BUFFER_DURATION_IS_VALID (buf))
-      end += GST_BUFFER_DURATION (buf);
-
-    end = gst_segment_to_stream_time (seg, GST_FORMAT_TIME, end);
-  }
-
-  GST_DEBUG ("Got buffer stream time %" GST_TIME_FORMAT " to %" GST_TIME_FORMAT,
-      GST_TIME_ARGS (start), GST_TIME_ARGS (end));
-
-  /* Check time is moving in the right direction */
-  if (current_rate > 0) {
-    if (GST_CLOCK_TIME_IS_VALID (first_ts))
-      fail_unless (start >= first_ts,
-          "Timestamps went backward during forward play, %" GST_TIME_FORMAT
-          " < %" GST_TIME_FORMAT, GST_TIME_ARGS (start),
-          GST_TIME_ARGS (first_ts));
-    if (GST_CLOCK_TIME_IS_VALID (last_ts))
-      fail_unless (end >= last_ts,
-          "Timestamps went backward during forward play, %" GST_TIME_FORMAT
-          " < %" GST_TIME_FORMAT, GST_TIME_ARGS (end), GST_TIME_ARGS (last_ts));
-  } else {
-    fail_unless (start <= first_ts,
-        "Timestamps went forward during reverse play, %" GST_TIME_FORMAT " > %"
-        GST_TIME_FORMAT, GST_TIME_ARGS (start), GST_TIME_ARGS (first_ts));
-    fail_unless (end <= last_ts,
-        "Timestamps went forward during reverse play, %" GST_TIME_FORMAT " > %"
-        GST_TIME_FORMAT, GST_TIME_ARGS (end), GST_TIME_ARGS (last_ts));
-  }
-
-  /* update the range of timestamps we've encountered */
-  if (!GST_CLOCK_TIME_IS_VALID (first_ts) || start < first_ts)
-    first_ts = start;
-  if (!GST_CLOCK_TIME_IS_VALID (last_ts) || end > last_ts)
-    last_ts = end;
-
-  gst_sample_unref (sample);
-
-  return GST_FLOW_OK;
-}
-
-static void
-test_playback (const gchar * in_pattern, GstClockTime exp_first_time,
-    GstClockTime exp_last_time, gboolean test_reverse)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *appsink;
-  GstElement *fakesink2;
-  GstAppSinkCallbacks callbacks = { NULL };
-  gchar *uri;
-
-  GST_DEBUG ("Playing back files matching %s", in_pattern);
-
-  pipeline = gst_element_factory_make ("playbin", NULL);
-  fail_if (pipeline == NULL);
-
-  appsink = gst_element_factory_make ("appsink", NULL);
-  fail_if (appsink == NULL);
-  g_object_set (G_OBJECT (appsink), "sync", FALSE, NULL);
-
-  g_object_set (G_OBJECT (pipeline), "video-sink", appsink, NULL);
-  fakesink2 = gst_element_factory_make ("fakesink", NULL);
-  fail_if (fakesink2 == NULL);
-  g_object_set (G_OBJECT (pipeline), "audio-sink", fakesink2, NULL);
-
-  uri = g_strdup_printf ("splitmux://%s", in_pattern);
-
-  g_object_set (G_OBJECT (pipeline), "uri", uri, NULL);
-  g_free (uri);
-
-  callbacks.new_sample = receive_sample;
-  gst_app_sink_set_callbacks (GST_APP_SINK (appsink), &callbacks, NULL, NULL);
-
-  /* test forwards */
-  seek_pipeline (pipeline, 1.0, 0, -1);
-  fail_unless (first_ts == GST_CLOCK_TIME_NONE);
-  msg = run_pipeline (pipeline);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-
-  /* Check we saw the entire range of values */
-  fail_unless (first_ts == exp_first_time,
-      "Expected start of playback range %" GST_TIME_FORMAT ", got %"
-      GST_TIME_FORMAT, GST_TIME_ARGS (exp_first_time),
-      GST_TIME_ARGS (first_ts));
-  fail_unless (last_ts == exp_last_time,
-      "Expected end of playback range %" GST_TIME_FORMAT ", got %"
-      GST_TIME_FORMAT, GST_TIME_ARGS (exp_last_time), GST_TIME_ARGS (last_ts));
-
-  if (test_reverse) {
-    /* Test backwards */
-    seek_pipeline (pipeline, -1.0, 0, -1);
-    msg = run_pipeline (pipeline);
-    fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-    gst_message_unref (msg);
-    /* Check we saw the entire range of values */
-    fail_unless (first_ts == exp_first_time,
-        "Expected start of playback range %" GST_TIME_FORMAT
-        ", got %" GST_TIME_FORMAT, GST_TIME_ARGS (exp_first_time),
-        GST_TIME_ARGS (first_ts));
-    fail_unless (last_ts == exp_last_time,
-        "Expected end of playback range %" GST_TIME_FORMAT
-        ", got %" GST_TIME_FORMAT, GST_TIME_ARGS (exp_last_time),
-        GST_TIME_ARGS (last_ts));
-  }
-
-  gst_object_unref (pipeline);
-}
-
-GST_START_TEST (test_splitmuxsrc)
-{
-  gchar *in_pattern =
-      g_build_filename (GST_TEST_FILES_PATH, "splitvideo*.ogg", NULL);
-  test_playback (in_pattern, 0, 3 * GST_SECOND, TRUE);
-  g_free (in_pattern);
-}
-
-GST_END_TEST;
-
-static gchar **
-src_format_location_cb (GstElement * splitmuxsrc, gpointer user_data)
-{
-  gchar **result = g_malloc0_n (4, sizeof (gchar *));
-  result[0] = g_build_filename (GST_TEST_FILES_PATH, "splitvideo00.ogg", NULL);
-  result[1] = g_build_filename (GST_TEST_FILES_PATH, "splitvideo01.ogg", NULL);
-  result[2] = g_build_filename (GST_TEST_FILES_PATH, "splitvideo02.ogg", NULL);
-  return result;
-}
-
-GST_START_TEST (test_splitmuxsrc_format_location)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *src;
-  GError *error = NULL;
-
-  pipeline = gst_parse_launch ("splitmuxsrc name=splitsrc ! decodebin "
-      "! fakesink", &error);
-  g_assert_no_error (error);
-  fail_if (pipeline == NULL);
-
-  src = gst_bin_get_by_name (GST_BIN (pipeline), "splitsrc");
-  g_signal_connect (src, "format-location",
-      (GCallback) src_format_location_cb, NULL);
-  g_object_unref (src);
-
-  msg = run_pipeline (pipeline);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-  gst_object_unref (pipeline);
-}
-
-GST_END_TEST;
-
-static gchar *
-check_format_location (GstElement * object,
-    guint fragment_id, GstSample * first_sample)
-{
-  GstBuffer *buf = gst_sample_get_buffer (first_sample);
-
-  /* Must have a buffer */
-  fail_if (buf == NULL);
-  GST_LOG ("New file - first buffer %" GST_TIME_FORMAT,
-      GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)));
-
-  return NULL;
-}
-
-GST_START_TEST (test_splitmuxsink)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *sink;
-  GstPad *splitmux_sink_pad;
-  GstPad *enc_src_pad;
-  gchar *dest_pattern;
-  guint count;
-  gchar *in_pattern;
-
-  /* This pipeline has a small time cutoff - it should start a new file
-   * every GOP, ie 1 second */
-  pipeline =
-      gst_parse_launch
-      ("videotestsrc num-buffers=15 ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
-      " queue ! theoraenc keyframe-force=5 ! splitmuxsink name=splitsink "
-      " max-size-time=1000000 max-size-bytes=1000000 muxer=oggmux", NULL);
-  fail_if (pipeline == NULL);
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  g_signal_connect (sink, "format-location-full",
-      (GCallback) check_format_location, NULL);
-  dest_pattern = g_build_filename (tmpdir, "out%05d.ogg", NULL);
-  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
-  g_free (dest_pattern);
-  g_object_unref (sink);
-
-  msg = run_pipeline (pipeline);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-
-  /* unlink manually and release request pad to ensure that we *can* do that
-   * - https://bugzilla.gnome.org/show_bug.cgi?id=753622 */
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  splitmux_sink_pad = gst_element_get_static_pad (sink, "video");
-  fail_if (splitmux_sink_pad == NULL);
-  enc_src_pad = gst_pad_get_peer (splitmux_sink_pad);
-  fail_if (enc_src_pad == NULL);
-  fail_unless (gst_pad_unlink (enc_src_pad, splitmux_sink_pad));
-  gst_object_unref (enc_src_pad);
-  gst_element_release_request_pad (sink, splitmux_sink_pad);
-  gst_object_unref (splitmux_sink_pad);
-  /* at this point the pad must be released - try to find it again to verify */
-  splitmux_sink_pad = gst_element_get_static_pad (sink, "video");
-  fail_if (splitmux_sink_pad != NULL);
-  g_object_unref (sink);
-
-  gst_object_unref (pipeline);
-
-  count = count_files (tmpdir);
-  fail_unless (count == 3, "Expected 3 output files, got %d", count);
-
-  in_pattern = g_build_filename (tmpdir, "out*.ogg", NULL);
-  test_playback (in_pattern, 0, 3 * GST_SECOND, TRUE);
-  g_free (in_pattern);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsink_multivid)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *sink;
-  gchar *dest_pattern;
-  guint count;
-  gchar *in_pattern;
-
-  /* This pipeline should start a new file every GOP, ie 1 second,
-   * driven by the primary video stream and with 2 auxiliary video streams */
-  pipeline =
-      gst_parse_launch
-      ("splitmuxsink name=splitsink "
-      " max-size-time=1000000 max-size-bytes=1000000 muxer=qtmux "
-      "videotestsrc num-buffers=15 ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
-      " queue ! vp8enc keyframe-max-dist=5 ! splitsink.video "
-      "videotestsrc num-buffers=15 pattern=snow ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
-      " queue ! vp8enc keyframe-max-dist=6 ! splitsink.video_aux_0 "
-      "videotestsrc num-buffers=15 pattern=ball ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
-      " queue ! vp8enc keyframe-max-dist=8 ! splitsink.video_aux_1 ", NULL);
-  fail_if (pipeline == NULL);
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  g_signal_connect (sink, "format-location-full",
-      (GCallback) check_format_location, NULL);
-  dest_pattern = g_build_filename (tmpdir, "out%05d.m4v", NULL);
-  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
-  g_free (dest_pattern);
-  g_object_unref (sink);
-
-  msg = run_pipeline (pipeline);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-
-  gst_object_unref (pipeline);
-
-  count = count_files (tmpdir);
-  fail_unless (count == 3, "Expected 3 output files, got %d", count);
-
-  in_pattern = g_build_filename (tmpdir, "out*.m4v", NULL);
-  /* FIXME: Reverse playback works poorly with multiple video streams
-   * in qtdemux (at least, maybe other demuxers) at the time this was
-   * written, and causes test failures like buffers being output
-   * multiple times by qtdemux as it loops through GOPs. Disable that
-   * for now */
-  test_playback (in_pattern, 0, 3 * GST_SECOND, FALSE);
-  g_free (in_pattern);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsink_async)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *sink;
-  GstPad *splitmux_sink_pad;
-  GstPad *enc_src_pad;
-  gchar *dest_pattern;
-  guint count;
-  gchar *in_pattern;
-
-  pipeline =
-      gst_parse_launch
-      ("videotestsrc num-buffers=15 ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
-      " queue ! theoraenc keyframe-force=5 ! splitmuxsink name=splitsink "
-      " max-size-time=1000000000 async-finalize=true "
-      " muxer-factory=matroskamux audiotestsrc num-buffers=15 samplesperbuffer=9600 ! "
-      " audio/x-raw,rate=48000 ! splitsink.audio_%u", NULL);
-  fail_if (pipeline == NULL);
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  g_signal_connect (sink, "format-location-full",
-      (GCallback) check_format_location, NULL);
-  dest_pattern = g_build_filename (tmpdir, "matroska%05d.mkv", NULL);
-  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
-  g_free (dest_pattern);
-  g_object_unref (sink);
-
-  msg = run_pipeline (pipeline);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-
-  /* unlink manually and release request pad to ensure that we *can* do that
-   * - https://bugzilla.gnome.org/show_bug.cgi?id=753622 */
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  splitmux_sink_pad = gst_element_get_static_pad (sink, "video");
-  fail_if (splitmux_sink_pad == NULL);
-  enc_src_pad = gst_pad_get_peer (splitmux_sink_pad);
-  fail_if (enc_src_pad == NULL);
-  fail_unless (gst_pad_unlink (enc_src_pad, splitmux_sink_pad));
-  gst_object_unref (enc_src_pad);
-  gst_element_release_request_pad (sink, splitmux_sink_pad);
-  gst_object_unref (splitmux_sink_pad);
-  /* at this point the pad must be released - try to find it again to verify */
-  splitmux_sink_pad = gst_element_get_static_pad (sink, "video");
-  fail_if (splitmux_sink_pad != NULL);
-  g_object_unref (sink);
-
-  gst_object_unref (pipeline);
-
-  count = count_files (tmpdir);
-  fail_unless (count == 3, "Expected 3 output files, got %d", count);
-
-  in_pattern = g_build_filename (tmpdir, "matroska*.mkv", NULL);
-  test_playback (in_pattern, 0, 3 * GST_SECOND, TRUE);
-  g_free (in_pattern);
-}
-
-GST_END_TEST;
-
-static GstPadProbeReturn
-intercept_stream_start (GstPad * pad, GstPadProbeInfo * info,
-    gpointer user_data)
-{
-  GstEvent *event = gst_pad_probe_info_get_event (info);
-
-  if (GST_EVENT_TYPE (event) == GST_EVENT_STREAM_START) {
-    GstStreamFlags flags;
-    event = gst_event_make_writable (event);
-    gst_event_parse_stream_flags (event, &flags);
-    gst_event_set_stream_flags (event, flags | GST_STREAM_FLAG_SPARSE);
-    GST_PAD_PROBE_INFO_DATA (info) = event;
-  }
-
-  return GST_PAD_PROBE_OK;
-}
-
-static GstFlowReturn
-new_sample_verify_continuous_timestamps (GstAppSink * appsink,
-    gpointer user_data)
-{
-  GstSample *sample;
-  GstBuffer *buffer;
-  GstClockTime *prev_ts = user_data;
-  GstClockTime new_ts;
-
-  sample = gst_app_sink_pull_sample (appsink);
-  buffer = gst_sample_get_buffer (sample);
-
-  new_ts = GST_BUFFER_PTS (buffer);
-  if (GST_CLOCK_TIME_IS_VALID (*prev_ts)) {
-    fail_unless (*prev_ts < new_ts,
-        "%s: prev_ts (%" GST_TIME_FORMAT ") >= new_ts (%" GST_TIME_FORMAT ")",
-        GST_OBJECT_NAME (appsink), GST_TIME_ARGS (*prev_ts),
-        GST_TIME_ARGS (new_ts));
-  }
-
-  *prev_ts = new_ts;
-  gst_sample_unref (sample);
-  return GST_FLOW_OK;
-}
-
-static GstFlowReturn
-new_sample_verify_1sec_offset (GstAppSink * appsink, gpointer user_data)
-{
-  GstSample *sample;
-  GstBuffer *buffer;
-  GstClockTime *prev_ts = user_data;
-  GstClockTime new_ts;
-
-  sample = gst_app_sink_pull_sample (appsink);
-  buffer = gst_sample_get_buffer (sample);
-
-  new_ts = GST_BUFFER_PTS (buffer);
-  if (GST_CLOCK_TIME_IS_VALID (*prev_ts)) {
-    fail_unless (new_ts > (*prev_ts + 900 * GST_MSECOND),
-        "%s: prev_ts (%" GST_TIME_FORMAT ") + 0.9s >= new_ts (%"
-        GST_TIME_FORMAT ")", GST_OBJECT_NAME (appsink),
-        GST_TIME_ARGS (*prev_ts), GST_TIME_ARGS (new_ts));
-  }
-
-  *prev_ts = new_ts;
-  gst_sample_unref (sample);
-  return GST_FLOW_OK;
-}
-
-/* https://bugzilla.gnome.org/show_bug.cgi?id=761086 */
-GST_START_TEST (test_splitmuxsrc_sparse_streams)
-{
-  GstElement *pipeline;
-  GstElement *element;
-  gchar *dest_pattern;
-  GstElement *appsrc;
-  GstPad *appsrc_src;
-  GstBus *bus;
-  GstMessage *msg;
-  gint i;
-
-  /* generate files */
-
-  /* in this test, we have 5sec of data with files split at 1sec intervals */
-  pipeline =
-      gst_parse_launch
-      ("videotestsrc num-buffers=75 !"
-      "  video/x-raw,width=80,height=64,framerate=15/1 !"
-      "  theoraenc keyframe-force=5 ! splitmuxsink name=splitsink"
-      "    max-size-time=1000000000 muxer=matroskamux"
-      " audiotestsrc num-buffers=100 samplesperbuffer=1024 !"
-      "  audio/x-raw,rate=20000 ! vorbisenc ! splitsink.audio_%u"
-      " appsrc name=appsrc format=time caps=text/x-raw,format=utf8 !"
-      "  splitsink.subtitle_%u", NULL);
-  fail_if (pipeline == NULL);
-
-  element = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (element == NULL);
-  dest_pattern = g_build_filename (tmpdir, "out%05d.ogg", NULL);
-  g_object_set (G_OBJECT (element), "location", dest_pattern, NULL);
-  g_clear_pointer (&dest_pattern, g_free);
-  g_clear_object (&element);
-
-  appsrc = gst_bin_get_by_name (GST_BIN (pipeline), "appsrc");
-  fail_if (appsrc == NULL);
-
-  /* add the SPARSE flag on the stream-start event of the subtitle stream */
-  appsrc_src = gst_element_get_static_pad (appsrc, "src");
-  fail_if (appsrc_src == NULL);
-  gst_pad_add_probe (appsrc_src, GST_PAD_PROBE_TYPE_EVENT_DOWNSTREAM,
-      intercept_stream_start, NULL, NULL);
-  g_clear_object (&appsrc_src);
-
-  bus = gst_pipeline_get_bus (GST_PIPELINE (pipeline));
-
-  gst_element_set_state (pipeline, GST_STATE_PLAYING);
-
-  /* push subtitles, one per second, starting from t=100ms */
-  for (i = 0; i < 5; i++) {
-    GstBuffer *buffer = gst_buffer_new_allocate (NULL, 5, NULL);
-    GstMapInfo info;
-
-    gst_buffer_map (buffer, &info, GST_MAP_WRITE);
-    strcpy ((char *) info.data, "test");
-    gst_buffer_unmap (buffer, &info);
-
-    GST_BUFFER_PTS (buffer) = i * GST_SECOND + 100 * GST_MSECOND;
-    GST_BUFFER_DTS (buffer) = GST_BUFFER_PTS (buffer);
-
-    fail_if (gst_app_src_push_buffer (GST_APP_SRC (appsrc), buffer)
-        != GST_FLOW_OK);
-  }
-  fail_if (gst_app_src_end_of_stream (GST_APP_SRC (appsrc)) != GST_FLOW_OK);
-
-  msg = gst_bus_timed_pop_filtered (bus, 5 * GST_SECOND, GST_MESSAGE_EOS);
-  g_clear_pointer (&msg, gst_message_unref);
-
-  gst_element_set_state (pipeline, GST_STATE_NULL);
-
-  g_clear_object (&appsrc);
-  g_clear_object (&bus);
-  g_clear_object (&pipeline);
-
-  /* read and verify */
-
-  pipeline =
-      gst_parse_launch
-      ("splitmuxsrc name=splitsrc"
-      " splitsrc. ! theoradec ! appsink name=vsink sync=false emit-signals=true"
-      " splitsrc. ! vorbisdec ! appsink name=asink sync=false emit-signals=true"
-      " splitsrc. ! text/x-raw ! appsink name=tsink sync=false emit-signals=true",
-      NULL);
-  fail_if (pipeline == NULL);
-
-  element = gst_bin_get_by_name (GST_BIN (pipeline), "splitsrc");
-  fail_if (element == NULL);
-  dest_pattern = g_build_filename (tmpdir, "out*.ogg", NULL);
-  g_object_set (G_OBJECT (element), "location", dest_pattern, NULL);
-  g_clear_pointer (&dest_pattern, g_free);
-  g_clear_object (&element);
-
-  {
-    GstClockTime vsink_prev_ts = GST_CLOCK_TIME_NONE;
-    GstClockTime asink_prev_ts = GST_CLOCK_TIME_NONE;
-    GstClockTime tsink_prev_ts = GST_CLOCK_TIME_NONE;
-
-    /* verify that timestamps are continuously increasing for audio + video.
-     * if we hit bug 761086, timestamps will jump about -900ms after switching
-     * to a new part, because this is the difference between the last subtitle
-     * pts and the last audio/video pts */
-    element = gst_bin_get_by_name (GST_BIN (pipeline), "vsink");
-    g_signal_connect (element, "new-sample",
-        (GCallback) new_sample_verify_continuous_timestamps, &vsink_prev_ts);
-    g_clear_object (&element);
-
-    element = gst_bin_get_by_name (GST_BIN (pipeline), "asink");
-    g_signal_connect (element, "new-sample",
-        (GCallback) new_sample_verify_continuous_timestamps, &asink_prev_ts);
-    g_clear_object (&element);
-
-    /* also verify that subtitle timestamps are increasing by about 1s.
-     * if we hit bug 761086, timestamps will increase by exactly 100ms instead,
-     * because this is the relative difference between a part's start time
-     * (remember a new part starts every 1sec) and the subtitle's pts in that
-     * part, which will be added to the max_ts of the previous part, which
-     * equals the last subtitle's pts (and should not!) */
-    element = gst_bin_get_by_name (GST_BIN (pipeline), "tsink");
-    g_signal_connect (element, "new-sample",
-        (GCallback) new_sample_verify_1sec_offset, &tsink_prev_ts);
-    g_clear_object (&element);
-
-    msg = run_pipeline (pipeline);
-  }
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-
-  g_clear_pointer (&msg, gst_message_unref);
-  g_clear_object (&pipeline);
-}
-
-GST_END_TEST;
-
-struct CapsChangeData
-{
-  guint count;
-  GstElement *cf;
-};
-
-static GstPadProbeReturn
-switch_caps (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
-{
-  struct CapsChangeData *data = (struct CapsChangeData *) (user_data);
-
-  if (data->count == 4) {
-    GST_INFO ("Saw 5 buffers to the encoder. Switching caps");
-    gst_util_set_object_arg (G_OBJECT (data->cf), "caps",
-        "video/x-raw,width=160,height=128,framerate=10/1");
-  }
-  data->count++;
-  return GST_PAD_PROBE_OK;
-}
-
-GST_START_TEST (test_splitmuxsrc_caps_change)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *sink;
-  GstElement *cf;
-  GstPad *sinkpad;
-  gchar *dest_pattern;
-  guint count;
-  gchar *in_pattern;
-  struct CapsChangeData data;
-
-  /* This test creates a new file only by changing the caps, which
-   * qtmux will reject (for now - if qtmux starts supporting caps
-   * changes, this test will break and need fixing/disabling */
-  pipeline =
-      gst_parse_launch
-      ("videotestsrc num-buffers=10 !"
-      "  capsfilter name=c caps=video/x-raw,width=80,height=64,framerate=10/1 !"
-      "  jpegenc ! splitmuxsink name=splitsink muxer=qtmux", NULL);
-  fail_if (pipeline == NULL);
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  g_signal_connect (sink, "format-location-full",
-      (GCallback) check_format_location, NULL);
-  dest_pattern = g_build_filename (tmpdir, "out%05d.mp4", NULL);
-  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
-  g_free (dest_pattern);
-  g_object_unref (sink);
-
-  cf = gst_bin_get_by_name (GST_BIN (pipeline), "c");
-  sinkpad = gst_element_get_static_pad (cf, "sink");
-
-  data.cf = cf;
-  data.count = 0;
-
-  gst_pad_add_probe (sinkpad, GST_PAD_PROBE_TYPE_BUFFER,
-      switch_caps, &data, NULL);
-
-  gst_object_unref (sinkpad);
-  gst_object_unref (cf);
-
-  msg = run_pipeline (pipeline);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-
-  gst_object_unref (pipeline);
-
-  count = count_files (tmpdir);
-  fail_unless (count == 2, "Expected 2 output files, got %d", count);
-
-  in_pattern = g_build_filename (tmpdir, "out*.mp4", NULL);
-  test_playback (in_pattern, 0, GST_SECOND, TRUE);
-  g_free (in_pattern);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsrc_robust_mux)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *sink;
-  gchar *dest_pattern;
-  gchar *in_pattern;
-
-  /* This test creates a new file only by changing the caps, which
-   * qtmux will reject (for now - if qtmux starts supporting caps
-   * changes, this test will break and need fixing/disabling */
-  pipeline =
-      gst_parse_launch
-      ("videotestsrc num-buffers=10 !"
-      "  video/x-raw,width=80,height=64,framerate=10/1 !"
-      "  jpegenc ! splitmuxsink name=splitsink muxer=\"qtmux reserved-bytes-per-sec=200 reserved-moov-update-period=100000000 \" max-size-time=500000000 use-robust-muxing=true",
-      NULL);
-  fail_if (pipeline == NULL);
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  g_signal_connect (sink, "format-location-full",
-      (GCallback) check_format_location, NULL);
-  dest_pattern = g_build_filename (tmpdir, "out%05d.mp4", NULL);
-  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
-  g_free (dest_pattern);
-  g_object_unref (sink);
-
-  msg = run_pipeline (pipeline);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-
-  gst_object_unref (pipeline);
-
-  /* Unlike other tests, we don't check an explicit file size, because the overflow detection
-   * can be racy (depends on exactly when buffers get handed to the muxer and when it updates the
-   * reserved duration property. All we care about is that the muxing didn't fail because space ran out */
-
-  in_pattern = g_build_filename (tmpdir, "out*.mp4", NULL);
-  test_playback (in_pattern, 0, GST_SECOND, TRUE);
-  g_free (in_pattern);
-}
-
-GST_END_TEST;
-
-/* For verifying bug https://bugzilla.gnome.org/show_bug.cgi?id=762893 */
-GST_START_TEST (test_splitmuxsink_reuse_simple)
-{
-  GstElement *sink;
-  GstPad *pad;
-
-  sink = gst_element_factory_make ("splitmuxsink", NULL);
-  pad = gst_element_get_request_pad (sink, "video");
-  fail_unless (pad != NULL);
-  g_object_set (sink, "location", "/dev/null", NULL);
-
-  fail_unless (gst_element_set_state (sink,
-          GST_STATE_PLAYING) == GST_STATE_CHANGE_ASYNC);
-  fail_unless (gst_element_set_state (sink,
-          GST_STATE_NULL) == GST_STATE_CHANGE_SUCCESS);
-  fail_unless (gst_element_set_state (sink,
-          GST_STATE_PLAYING) == GST_STATE_CHANGE_ASYNC);
-  fail_unless (gst_element_set_state (sink,
-          GST_STATE_NULL) == GST_STATE_CHANGE_SUCCESS);
-
-  gst_element_release_request_pad (sink, pad);
-  gst_object_unref (pad);
-  gst_object_unref (sink);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsink_muxer_pad_map)
-{
-  GstElement *sink, *muxer;
-  GstPad *muxpad;
-  GstPad *pad1 = NULL, *pad2 = NULL;
-  GstStructure *pad_map;
-
-  pad_map = gst_structure_new ("x-pad-map",
-      "video", G_TYPE_STRING, "video_100",
-      "audio_0", G_TYPE_STRING, "audio_101", NULL);
-
-  muxer = gst_element_factory_make ("qtmux", NULL);
-  fail_if (muxer == NULL);
-  sink = gst_element_factory_make ("splitmuxsink", NULL);
-  fail_if (sink == NULL);
-
-  g_object_set (sink, "muxer", muxer, "muxer-pad-map", pad_map, NULL);
-  gst_structure_free (pad_map);
-
-  pad1 = gst_element_get_request_pad (sink, "video");
-  fail_unless (g_str_equal ("video", GST_PAD_NAME (pad1)));
-  muxpad = gst_element_get_static_pad (muxer, "video_100");
-  fail_unless (muxpad != NULL);
-  gst_object_unref (muxpad);
-
-  pad2 = gst_element_get_request_pad (sink, "audio_0");
-  fail_unless (g_str_equal ("audio_0", GST_PAD_NAME (pad2)));
-  muxpad = gst_element_get_static_pad (muxer, "audio_101");
-  fail_unless (muxpad != NULL);
-  gst_object_unref (muxpad);
-
-  g_object_set (sink, "location", "/dev/null", NULL);
-
-  fail_unless (gst_element_set_state (sink,
-          GST_STATE_PLAYING) == GST_STATE_CHANGE_ASYNC);
-  fail_unless (gst_element_set_state (sink,
-          GST_STATE_NULL) == GST_STATE_CHANGE_SUCCESS);
-
-  gst_element_release_request_pad (sink, pad1);
-  gst_object_unref (pad1);
-  gst_element_release_request_pad (sink, pad2);
-  gst_object_unref (pad2);
-  gst_object_unref (sink);
-}
-
-GST_END_TEST;
-
-static GstPadProbeReturn
-count_upstrea_fku (GstPad * pad, GstPadProbeInfo * info,
-    guint * upstream_fku_count)
-{
-  GstEvent *event = GST_PAD_PROBE_INFO_EVENT (info);
-
-  switch (GST_EVENT_TYPE (event)) {
-    case GST_EVENT_CUSTOM_UPSTREAM:
-      if (gst_video_event_is_force_key_unit (event))
-        *upstream_fku_count += 1;
-      break;
-    default:
-      break;
-  }
-
-  return GST_PAD_PROBE_OK;
-}
-
-static void
-splitmuxsink_split_by_keyframe (gboolean send_keyframe_request,
-    guint max_size_time_sec, guint encoder_key_interval_sec)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *sink;
-  GstElement *enc;
-  GstPad *srcpad;
-  gchar *pipeline_str;
-  gchar *dest_pattern;
-  guint count;
-  guint expected_count;
-  gchar *in_pattern;
-  guint upstream_fku_count = 0;
-  guint expected_fku_count;
-
-  pipeline_str = g_strdup_printf ("splitmuxsink name=splitsink "
-      "max-size-time=%" G_GUINT64_FORMAT
-      " send-keyframe-requests=%s muxer=qtmux "
-      "videotestsrc num-buffers=30 ! video/x-raw,width=80,height=64,framerate=5/1 "
-      "! videoconvert ! queue ! vp8enc name=enc keyframe-max-dist=%d ! splitsink.video ",
-      max_size_time_sec * GST_SECOND, send_keyframe_request ? "true" : "false",
-      encoder_key_interval_sec * 5);
-
-  pipeline = gst_parse_launch (pipeline_str, NULL);
-  g_free (pipeline_str);
-
-  fail_if (pipeline == NULL);
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  g_signal_connect (sink, "format-location-full",
-      (GCallback) check_format_location, NULL);
-  dest_pattern = g_build_filename (tmpdir, "out%05d.m4v", NULL);
-  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
-  g_free (dest_pattern);
-  g_object_unref (sink);
-
-  enc = gst_bin_get_by_name (GST_BIN (pipeline), "enc");
-  fail_if (enc == NULL);
-  srcpad = gst_element_get_static_pad (enc, "src");
-  fail_if (srcpad == NULL);
-
-  gst_pad_add_probe (srcpad, GST_PAD_PROBE_TYPE_EVENT_UPSTREAM,
-      (GstPadProbeCallback) count_upstrea_fku, &upstream_fku_count, NULL);
-  gst_object_unref (srcpad);
-  gst_object_unref (enc);
-
-  msg = run_pipeline (pipeline);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-
-  gst_object_unref (pipeline);
-
-  count = count_files (tmpdir);
-  expected_count = 6 / max_size_time_sec;
-  fail_unless (count == expected_count,
-      "Expected %d output files, got %d", expected_count, count);
-
-  if (!send_keyframe_request) {
-    expected_fku_count = 0;
-  } else {
-    expected_fku_count = count;
-  }
-
-  GST_INFO ("Upstream force keyunit event count %d", upstream_fku_count);
-
-  fail_unless (upstream_fku_count == expected_fku_count,
-      "Expected upstream force keyunit event count %d, got %d",
-      expected_fku_count, upstream_fku_count);
-
-  in_pattern = g_build_filename (tmpdir, "out*.m4v", NULL);
-  /* FIXME: Reverse playback works poorly with multiple video streams
-   * in qtdemux (at least, maybe other demuxers) at the time this was
-   * written, and causes test failures like buffers being output
-   * multiple times by qtdemux as it loops through GOPs. Disable that
-   * for now */
-  test_playback (in_pattern, 0, 6 * GST_SECOND, FALSE);
-  g_free (in_pattern);
-}
-
-GST_START_TEST (test_splitmuxsink_without_keyframe_request)
-{
-  /* This encoding option is intended to produce keyframe per 1 seconds
-   * but splitmuxsink will split file per 2 second without keyframe request */
-  splitmuxsink_split_by_keyframe (FALSE, 2, 1);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsink_keyframe_request)
-{
-  /* This encoding option is intended to produce keyframe per 2 seconds
-   * and splitmuxsink will request keyframe per 2 seconds as well.
-   * This should produce 2 seconds long files */
-  splitmuxsink_split_by_keyframe (TRUE, 2, 2);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsink_keyframe_request_more)
-{
-  /* This encoding option is intended to produce keyframe per 2 seconds
-   * but splitmuxsink will request keyframe per 1 second. This should produce
-   * 1 second long files */
-  splitmuxsink_split_by_keyframe (TRUE, 1, 2);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsink_keyframe_request_less)
-{
-  /* This encoding option is intended to produce keyframe per 1 second
-   * but splitmuxsink will request keyframe per 2 seconds. This should produce
-   * 2 seconds long files */
-  splitmuxsink_split_by_keyframe (TRUE, 2, 1);
-}
-
-GST_END_TEST;
-
-static void
-splitmuxsink_split_by_keyframe_timecode (gboolean send_keyframe_request,
-    const gchar * maxsize_timecode_string, guint maxsize_timecode_in_sec,
-    guint encoder_key_interval_sec)
-{
-  GstMessage *msg;
-  GstElement *pipeline;
-  GstElement *sink;
-  GstElement *enc;
-  GstPad *srcpad;
-  gchar *pipeline_str;
-  gchar *dest_pattern;
-  guint count;
-  guint expected_count;
-  gchar *in_pattern;
-  guint upstream_fku_count = 0;
-  guint expected_fku_count;
-
-  pipeline_str = g_strdup_printf ("splitmuxsink name=splitsink "
-      "max-size-timecode=%s"
-      " send-keyframe-requests=%s muxer=qtmux "
-      "videotestsrc num-buffers=30 ! video/x-raw,width=80,height=64,framerate=5/1 "
-      "! videoconvert ! timecodestamper ! queue ! vp8enc name=enc keyframe-max-dist=%d ! splitsink.video ",
-      maxsize_timecode_string, send_keyframe_request ? "true" : "false",
-      encoder_key_interval_sec ? encoder_key_interval_sec * 5 : 1);
-
-  pipeline = gst_parse_launch (pipeline_str, NULL);
-  g_free (pipeline_str);
-
-  fail_if (pipeline == NULL);
-  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
-  fail_if (sink == NULL);
-  g_signal_connect (sink, "format-location-full",
-      (GCallback) check_format_location, NULL);
-  dest_pattern = g_build_filename (tmpdir, "out%05d.m4v", NULL);
-  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
-  g_free (dest_pattern);
-  g_object_unref (sink);
-
-  enc = gst_bin_get_by_name (GST_BIN (pipeline), "enc");
-  fail_if (enc == NULL);
-  srcpad = gst_element_get_static_pad (enc, "src");
-  fail_if (srcpad == NULL);
-
-  gst_pad_add_probe (srcpad, GST_PAD_PROBE_TYPE_EVENT_UPSTREAM,
-      (GstPadProbeCallback) count_upstrea_fku, &upstream_fku_count, NULL);
-  gst_object_unref (srcpad);
-  gst_object_unref (enc);
-
-  msg = run_pipeline (pipeline);
-
-  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
-    dump_error (msg);
-  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
-  gst_message_unref (msg);
-
-  gst_object_unref (pipeline);
-
-  count = count_files (tmpdir);
-  expected_count = (6 / maxsize_timecode_in_sec) +
-      (6 % maxsize_timecode_in_sec ? 1 : 0);
-  fail_unless (count == expected_count,
-      "Expected %d output files, got %d", expected_count, count);
-
-  if (!send_keyframe_request) {
-    expected_fku_count = 0;
-  } else {
-    expected_fku_count = count;
-  }
-
-  GST_INFO ("Upstream force keyunit event count %d", upstream_fku_count);
-
-  fail_unless (upstream_fku_count == expected_fku_count,
-      "Expected upstream force keyunit event count %d, got %d",
-      expected_fku_count, upstream_fku_count);
-
-  in_pattern = g_build_filename (tmpdir, "out*.m4v", NULL);
-  /* FIXME: Reverse playback works poorly with multiple video streams
-   * in qtdemux (at least, maybe other demuxers) at the time this was
-   * written, and causes test failures like buffers being output
-   * multiple times by qtdemux as it loops through GOPs. Disable that
-   * for now */
-  test_playback (in_pattern, 0, 6 * GST_SECOND, FALSE);
-  g_free (in_pattern);
-}
-
-GST_START_TEST (test_splitmuxsink_without_keyframe_request_timecode)
-{
-  /* This encoding option is intended to produce keyframe per 1 second
-   * but splitmuxsink will split file per 2 second without keyframe request */
-  splitmuxsink_split_by_keyframe_timecode (FALSE, "00:00:02:00", 2, 1);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsink_keyframe_request_timecode)
-{
-  /* This encoding option is intended to produce keyframe per 1 second
-   * but splitmuxsink will request keyframe per 2 seconds. This should produce
-   * 2 seconds long files */
-  splitmuxsink_split_by_keyframe_timecode (TRUE, "00:00:02:00", 2, 1);
-}
-
-GST_END_TEST;
-
-GST_START_TEST
-    (test_splitmuxsink_keyframe_request_timecode_trailing_small_segment) {
-  /* This encoding option is intended to produce keyframe per 1 second
-   * but splitmuxsink will request keyframe per 4 seconds. This should produce
-   * 4 seconds long files */
-  splitmuxsink_split_by_keyframe_timecode (TRUE, "00:00:04:00", 4, 1);
-}
-
-GST_END_TEST;
-
-GST_START_TEST (test_splitmuxsink_keyframe_request_timecode_all_intra)
-{
-  /* This encoding option is intended to produce keyframe for every frame.
-   * This should produce 1 second long files */
-  splitmuxsink_split_by_keyframe_timecode (TRUE, "00:00:01:00", 1, 0);
-}
-
-GST_END_TEST;
-
-static Suite *
-splitmux_suite (void)
-{
-  Suite *s = suite_create ("splitmux");
-  TCase *tc_chain = tcase_create ("general");
-  TCase *tc_chain_basic = tcase_create ("basic");
-  TCase *tc_chain_complex = tcase_create ("complex");
-  TCase *tc_chain_mp4_jpeg = tcase_create ("caps_change");
-  gboolean have_theora, have_ogg, have_vorbis, have_matroska, have_qtmux,
-      have_jpeg, have_vp8, have_timecodestamper;
-
-  /* we assume that if encoder/muxer are there, decoder/demuxer will be a well */
-  have_theora = gst_registry_check_feature_version (gst_registry_get (),
-      "theoraenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
-  have_ogg = gst_registry_check_feature_version (gst_registry_get (),
-      "oggmux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
-  have_vorbis = gst_registry_check_feature_version (gst_registry_get (),
-      "vorbisenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
-  have_matroska = gst_registry_check_feature_version (gst_registry_get (),
-      "matroskamux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
-  have_qtmux = gst_registry_check_feature_version (gst_registry_get (),
-      "qtmux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
-  have_jpeg = gst_registry_check_feature_version (gst_registry_get (),
-      "jpegenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
-  have_vp8 = gst_registry_check_feature_version (gst_registry_get (),
-      "vp8enc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
-  have_timecodestamper =
-      gst_registry_check_feature_version (gst_registry_get (),
-      "timecodestamper", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
-
-  suite_add_tcase (s, tc_chain);
-  suite_add_tcase (s, tc_chain_basic);
-  suite_add_tcase (s, tc_chain_complex);
-  suite_add_tcase (s, tc_chain_mp4_jpeg);
-
-  tcase_add_test (tc_chain_basic, test_splitmuxsink_reuse_simple);
-
-  if (have_theora && have_ogg) {
-    tcase_add_checked_fixture (tc_chain, tempdir_setup, tempdir_cleanup);
-
-    tcase_add_test (tc_chain, test_splitmuxsrc);
-    tcase_add_test (tc_chain, test_splitmuxsrc_format_location);
-    tcase_add_test (tc_chain, test_splitmuxsink);
-
-    if (have_matroska && have_vorbis) {
-      tcase_add_checked_fixture (tc_chain_complex, tempdir_setup,
-          tempdir_cleanup);
-
-      tcase_add_test (tc_chain_complex, test_splitmuxsrc_sparse_streams);
-      tcase_add_test (tc_chain, test_splitmuxsink_async);
-    } else {
-      GST_INFO ("Skipping tests, missing plugins: matroska and/or vorbis");
-    }
-  } else {
-    GST_INFO ("Skipping tests, missing plugins: theora and/or ogg");
-  }
-
-
-  if (have_qtmux && have_jpeg) {
-    tcase_add_checked_fixture (tc_chain_mp4_jpeg, tempdir_setup,
-        tempdir_cleanup);
-    tcase_add_test (tc_chain_mp4_jpeg, test_splitmuxsrc_caps_change);
-    tcase_add_test (tc_chain_mp4_jpeg, test_splitmuxsrc_robust_mux);
-    tcase_add_test (tc_chain_mp4_jpeg, test_splitmuxsink_muxer_pad_map);
-  } else {
-    GST_INFO ("Skipping tests, missing plugins: jpegenc or mp4mux");
-  }
-
-  if (have_qtmux && have_vp8) {
-    tcase_add_test (tc_chain, test_splitmuxsink_multivid);
-    tcase_add_test (tc_chain, test_splitmuxsink_without_keyframe_request);
-    tcase_add_test (tc_chain, test_splitmuxsink_keyframe_request);
-    tcase_add_test (tc_chain, test_splitmuxsink_keyframe_request_more);
-    tcase_add_test (tc_chain, test_splitmuxsink_keyframe_request_less);
-  } else {
-    GST_INFO ("Skipping tests, missing plugins: vp8enc or mp4mux");
-  }
-
-  if (have_qtmux && have_vp8 && have_timecodestamper) {
-    tcase_add_test (tc_chain,
-        test_splitmuxsink_without_keyframe_request_timecode);
-    tcase_add_test (tc_chain, test_splitmuxsink_keyframe_request_timecode);
-    tcase_add_test (tc_chain,
-        test_splitmuxsink_keyframe_request_timecode_trailing_small_segment);
-    tcase_add_test (tc_chain,
-        test_splitmuxsink_keyframe_request_timecode_all_intra);
-  } else {
-    GST_INFO
-        ("Skipping tests, missing plugins: vp8enc, mp4mux, or timecodestamper");
-  }
-
-  return s;
-}
-
-GST_CHECK_MAIN (splitmux);
diff --git a/tests/check/elements/splitmuxsink.c b/tests/check/elements/splitmuxsink.c
new file mode 100644 (file)
index 0000000..5ed5b5b
--- /dev/null
@@ -0,0 +1,911 @@
+/* GStreamer unit test for splitmuxsink elements
+ *
+ * Copyright (C) 2007 David A. Schleef <ds@schleef.org>
+ * Copyright (C) 2015 Jan Schmidt <jan@centricular.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.
+ */
+
+#ifdef HAVE_CONFIG_H
+#  include "config.h"
+#endif
+
+#include <glib/gstdio.h>
+
+#include <gst/check/gstcheck.h>
+#include <gst/app/app.h>
+#include <gst/video/video.h>
+#include <stdlib.h>
+
+gchar *tmpdir = NULL;
+GstClockTime first_ts;
+GstClockTime last_ts;
+gdouble current_rate;
+
+static void
+tempdir_setup (void)
+{
+  const gchar *systmp = g_get_tmp_dir ();
+  tmpdir = g_build_filename (systmp, "splitmux-test-XXXXXX", NULL);
+  /* Rewrites tmpdir template input: */
+  tmpdir = g_mkdtemp (tmpdir);
+}
+
+static void
+tempdir_cleanup (void)
+{
+  GDir *d;
+  const gchar *f;
+
+  fail_if (tmpdir == NULL);
+
+  d = g_dir_open (tmpdir, 0, NULL);
+  fail_if (d == NULL);
+
+  while ((f = g_dir_read_name (d)) != NULL) {
+    gchar *fname = g_build_filename (tmpdir, f, NULL);
+    fail_if (g_remove (fname) != 0, "Failed to remove tmp file %s", fname);
+    g_free (fname);
+  }
+  g_dir_close (d);
+
+  fail_if (g_remove (tmpdir) != 0, "Failed to delete tmpdir %s", tmpdir);
+
+  g_free (tmpdir);
+  tmpdir = NULL;
+}
+
+static guint
+count_files (const gchar * target)
+{
+  GDir *d;
+  const gchar *f;
+  guint ret = 0;
+
+  d = g_dir_open (target, 0, NULL);
+  fail_if (d == NULL);
+
+  while ((f = g_dir_read_name (d)) != NULL)
+    ret++;
+  g_dir_close (d);
+
+  return ret;
+}
+
+static void
+dump_error (GstMessage * msg)
+{
+  GError *err = NULL;
+  gchar *dbg_info;
+
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR);
+
+  gst_message_parse_error (msg, &err, &dbg_info);
+
+  g_printerr ("ERROR from element %s: %s\n",
+      GST_OBJECT_NAME (msg->src), err->message);
+  g_printerr ("Debugging info: %s\n", (dbg_info) ? dbg_info : "none");
+  g_error_free (err);
+  g_free (dbg_info);
+}
+
+static GstMessage *
+run_pipeline (GstElement * pipeline)
+{
+  GstBus *bus = gst_element_get_bus (GST_ELEMENT (pipeline));
+  GstMessage *msg;
+
+  gst_element_set_state (pipeline, GST_STATE_PLAYING);
+  msg = gst_bus_poll (bus, GST_MESSAGE_EOS | GST_MESSAGE_ERROR, -1);
+  gst_element_set_state (pipeline, GST_STATE_NULL);
+
+  gst_object_unref (bus);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+
+  return msg;
+}
+
+static void
+seek_pipeline (GstElement * pipeline, gdouble rate, GstClockTime start,
+    GstClockTime end)
+{
+  /* Pause the pipeline, seek to the desired range / rate, wait for PAUSED again, then
+   * clear the tracking vars for start_ts / end_ts */
+  gst_element_set_state (pipeline, GST_STATE_PAUSED);
+  gst_element_get_state (pipeline, NULL, NULL, GST_CLOCK_TIME_NONE);
+
+  /* specific end time not implemented: */
+  fail_unless (end == GST_CLOCK_TIME_NONE);
+
+  gst_element_seek (pipeline, rate, GST_FORMAT_TIME,
+      GST_SEEK_FLAG_FLUSH | GST_SEEK_FLAG_ACCURATE, GST_SEEK_TYPE_SET, start,
+      GST_SEEK_TYPE_END, 0);
+
+  /* Wait for the pipeline to preroll again */
+  gst_element_get_state (pipeline, NULL, NULL, GST_CLOCK_TIME_NONE);
+
+  GST_LOG ("Seeked pipeline. Rate %f time range %" GST_TIME_FORMAT " to %"
+      GST_TIME_FORMAT, rate, GST_TIME_ARGS (start), GST_TIME_ARGS (end));
+
+  /* Clear tracking variables now that the seek is complete */
+  first_ts = last_ts = GST_CLOCK_TIME_NONE;
+  current_rate = rate;
+};
+
+static GstFlowReturn
+receive_sample (GstAppSink * appsink, gpointer user_data G_GNUC_UNUSED)
+{
+  GstSample *sample;
+  GstSegment *seg;
+  GstBuffer *buf;
+  GstClockTime start;
+  GstClockTime end;
+
+  g_signal_emit_by_name (appsink, "pull-sample", &sample);
+  fail_unless (sample != NULL);
+
+  seg = gst_sample_get_segment (sample);
+  fail_unless (seg != NULL);
+
+  buf = gst_sample_get_buffer (sample);
+  fail_unless (buf != NULL);
+
+  GST_LOG ("Got buffer %" GST_PTR_FORMAT, buf);
+
+  start = GST_BUFFER_PTS (buf);
+  end = start;
+
+  if (GST_CLOCK_TIME_IS_VALID (start))
+    start = gst_segment_to_stream_time (seg, GST_FORMAT_TIME, start);
+
+  if (GST_CLOCK_TIME_IS_VALID (end)) {
+    if (GST_BUFFER_DURATION_IS_VALID (buf))
+      end += GST_BUFFER_DURATION (buf);
+
+    end = gst_segment_to_stream_time (seg, GST_FORMAT_TIME, end);
+  }
+
+  GST_DEBUG ("Got buffer stream time %" GST_TIME_FORMAT " to %" GST_TIME_FORMAT,
+      GST_TIME_ARGS (start), GST_TIME_ARGS (end));
+
+  /* Check time is moving in the right direction */
+  if (current_rate > 0) {
+    if (GST_CLOCK_TIME_IS_VALID (first_ts))
+      fail_unless (start >= first_ts,
+          "Timestamps went backward during forward play, %" GST_TIME_FORMAT
+          " < %" GST_TIME_FORMAT, GST_TIME_ARGS (start),
+          GST_TIME_ARGS (first_ts));
+    if (GST_CLOCK_TIME_IS_VALID (last_ts))
+      fail_unless (end >= last_ts,
+          "Timestamps went backward during forward play, %" GST_TIME_FORMAT
+          " < %" GST_TIME_FORMAT, GST_TIME_ARGS (end), GST_TIME_ARGS (last_ts));
+  } else {
+    fail_unless (start <= first_ts,
+        "Timestamps went forward during reverse play, %" GST_TIME_FORMAT " > %"
+        GST_TIME_FORMAT, GST_TIME_ARGS (start), GST_TIME_ARGS (first_ts));
+    fail_unless (end <= last_ts,
+        "Timestamps went forward during reverse play, %" GST_TIME_FORMAT " > %"
+        GST_TIME_FORMAT, GST_TIME_ARGS (end), GST_TIME_ARGS (last_ts));
+  }
+
+  /* update the range of timestamps we've encountered */
+  if (!GST_CLOCK_TIME_IS_VALID (first_ts) || start < first_ts)
+    first_ts = start;
+  if (!GST_CLOCK_TIME_IS_VALID (last_ts) || end > last_ts)
+    last_ts = end;
+
+  gst_sample_unref (sample);
+
+  return GST_FLOW_OK;
+}
+
+static void
+test_playback (const gchar * in_pattern, GstClockTime exp_first_time,
+    GstClockTime exp_last_time, gboolean test_reverse)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *appsink;
+  GstElement *fakesink2;
+  GstAppSinkCallbacks callbacks = { NULL };
+  gchar *uri;
+
+  GST_DEBUG ("Playing back files matching %s", in_pattern);
+
+  pipeline = gst_element_factory_make ("playbin", NULL);
+  fail_if (pipeline == NULL);
+
+  appsink = gst_element_factory_make ("appsink", NULL);
+  fail_if (appsink == NULL);
+  g_object_set (G_OBJECT (appsink), "sync", FALSE, NULL);
+
+  g_object_set (G_OBJECT (pipeline), "video-sink", appsink, NULL);
+  fakesink2 = gst_element_factory_make ("fakesink", NULL);
+  fail_if (fakesink2 == NULL);
+  g_object_set (G_OBJECT (pipeline), "audio-sink", fakesink2, NULL);
+
+  uri = g_strdup_printf ("splitmux://%s", in_pattern);
+
+  g_object_set (G_OBJECT (pipeline), "uri", uri, NULL);
+  g_free (uri);
+
+  callbacks.new_sample = receive_sample;
+  gst_app_sink_set_callbacks (GST_APP_SINK (appsink), &callbacks, NULL, NULL);
+
+  /* test forwards */
+  seek_pipeline (pipeline, 1.0, 0, -1);
+  fail_unless (first_ts == GST_CLOCK_TIME_NONE);
+  msg = run_pipeline (pipeline);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  /* Check we saw the entire range of values */
+  fail_unless (first_ts == exp_first_time,
+      "Expected start of playback range %" GST_TIME_FORMAT ", got %"
+      GST_TIME_FORMAT, GST_TIME_ARGS (exp_first_time),
+      GST_TIME_ARGS (first_ts));
+  fail_unless (last_ts == exp_last_time,
+      "Expected end of playback range %" GST_TIME_FORMAT ", got %"
+      GST_TIME_FORMAT, GST_TIME_ARGS (exp_last_time), GST_TIME_ARGS (last_ts));
+
+  if (test_reverse) {
+    /* Test backwards */
+    seek_pipeline (pipeline, -1.0, 0, -1);
+    msg = run_pipeline (pipeline);
+    fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+    gst_message_unref (msg);
+    /* Check we saw the entire range of values */
+    fail_unless (first_ts == exp_first_time,
+        "Expected start of playback range %" GST_TIME_FORMAT
+        ", got %" GST_TIME_FORMAT, GST_TIME_ARGS (exp_first_time),
+        GST_TIME_ARGS (first_ts));
+    fail_unless (last_ts == exp_last_time,
+        "Expected end of playback range %" GST_TIME_FORMAT
+        ", got %" GST_TIME_FORMAT, GST_TIME_ARGS (exp_last_time),
+        GST_TIME_ARGS (last_ts));
+  }
+
+  gst_object_unref (pipeline);
+}
+
+static gchar *
+check_format_location (GstElement * object,
+    guint fragment_id, GstSample * first_sample)
+{
+  GstBuffer *buf = gst_sample_get_buffer (first_sample);
+
+  /* Must have a buffer */
+  fail_if (buf == NULL);
+  GST_LOG ("New file - first buffer %" GST_TIME_FORMAT,
+      GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)));
+
+  return NULL;
+}
+
+GST_START_TEST (test_splitmuxsink)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *sink;
+  GstPad *splitmux_sink_pad;
+  GstPad *enc_src_pad;
+  gchar *dest_pattern;
+  guint count;
+  gchar *in_pattern;
+
+  /* This pipeline has a small time cutoff - it should start a new file
+   * every GOP, ie 1 second */
+  pipeline =
+      gst_parse_launch
+      ("videotestsrc num-buffers=15 ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
+      " queue ! theoraenc keyframe-force=5 ! splitmuxsink name=splitsink "
+      " max-size-time=1000000 max-size-bytes=1000000 muxer=oggmux", NULL);
+  fail_if (pipeline == NULL);
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  g_signal_connect (sink, "format-location-full",
+      (GCallback) check_format_location, NULL);
+  dest_pattern = g_build_filename (tmpdir, "out%05d.ogg", NULL);
+  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
+  g_free (dest_pattern);
+  g_object_unref (sink);
+
+  msg = run_pipeline (pipeline);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  /* unlink manually and release request pad to ensure that we *can* do that
+   * - https://bugzilla.gnome.org/show_bug.cgi?id=753622 */
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  splitmux_sink_pad = gst_element_get_static_pad (sink, "video");
+  fail_if (splitmux_sink_pad == NULL);
+  enc_src_pad = gst_pad_get_peer (splitmux_sink_pad);
+  fail_if (enc_src_pad == NULL);
+  fail_unless (gst_pad_unlink (enc_src_pad, splitmux_sink_pad));
+  gst_object_unref (enc_src_pad);
+  gst_element_release_request_pad (sink, splitmux_sink_pad);
+  gst_object_unref (splitmux_sink_pad);
+  /* at this point the pad must be released - try to find it again to verify */
+  splitmux_sink_pad = gst_element_get_static_pad (sink, "video");
+  fail_if (splitmux_sink_pad != NULL);
+  g_object_unref (sink);
+
+  gst_object_unref (pipeline);
+
+  count = count_files (tmpdir);
+  fail_unless (count == 3, "Expected 3 output files, got %d", count);
+
+  in_pattern = g_build_filename (tmpdir, "out*.ogg", NULL);
+  test_playback (in_pattern, 0, 3 * GST_SECOND, TRUE);
+  g_free (in_pattern);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsink_multivid)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *sink;
+  gchar *dest_pattern;
+  guint count;
+  gchar *in_pattern;
+
+  /* This pipeline should start a new file every GOP, ie 1 second,
+   * driven by the primary video stream and with 2 auxiliary video streams */
+  pipeline =
+      gst_parse_launch
+      ("splitmuxsink name=splitsink "
+      " max-size-time=1000000 max-size-bytes=1000000 muxer=qtmux "
+      "videotestsrc num-buffers=15 ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
+      " queue ! vp8enc keyframe-max-dist=5 ! splitsink.video "
+      "videotestsrc num-buffers=15 pattern=snow ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
+      " queue ! vp8enc keyframe-max-dist=6 ! splitsink.video_aux_0 "
+      "videotestsrc num-buffers=15 pattern=ball ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
+      " queue ! vp8enc keyframe-max-dist=8 ! splitsink.video_aux_1 ", NULL);
+  fail_if (pipeline == NULL);
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  g_signal_connect (sink, "format-location-full",
+      (GCallback) check_format_location, NULL);
+  dest_pattern = g_build_filename (tmpdir, "out%05d.m4v", NULL);
+  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
+  g_free (dest_pattern);
+  g_object_unref (sink);
+
+  msg = run_pipeline (pipeline);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  gst_object_unref (pipeline);
+
+  count = count_files (tmpdir);
+  fail_unless (count == 3, "Expected 3 output files, got %d", count);
+
+  in_pattern = g_build_filename (tmpdir, "out*.m4v", NULL);
+  /* FIXME: Reverse playback works poorly with multiple video streams
+   * in qtdemux (at least, maybe other demuxers) at the time this was
+   * written, and causes test failures like buffers being output
+   * multiple times by qtdemux as it loops through GOPs. Disable that
+   * for now */
+  test_playback (in_pattern, 0, 3 * GST_SECOND, FALSE);
+  g_free (in_pattern);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsink_async)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *sink;
+  GstPad *splitmux_sink_pad;
+  GstPad *enc_src_pad;
+  gchar *dest_pattern;
+  guint count;
+  gchar *in_pattern;
+
+  pipeline =
+      gst_parse_launch
+      ("videotestsrc num-buffers=15 ! video/x-raw,width=80,height=64,framerate=5/1 ! videoconvert !"
+      " queue ! theoraenc keyframe-force=5 ! splitmuxsink name=splitsink "
+      " max-size-time=1000000000 async-finalize=true "
+      " muxer-factory=matroskamux audiotestsrc num-buffers=15 samplesperbuffer=9600 ! "
+      " audio/x-raw,rate=48000 ! splitsink.audio_%u", NULL);
+  fail_if (pipeline == NULL);
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  g_signal_connect (sink, "format-location-full",
+      (GCallback) check_format_location, NULL);
+  dest_pattern = g_build_filename (tmpdir, "matroska%05d.mkv", NULL);
+  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
+  g_free (dest_pattern);
+  g_object_unref (sink);
+
+  msg = run_pipeline (pipeline);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  /* unlink manually and release request pad to ensure that we *can* do that
+   * - https://bugzilla.gnome.org/show_bug.cgi?id=753622 */
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  splitmux_sink_pad = gst_element_get_static_pad (sink, "video");
+  fail_if (splitmux_sink_pad == NULL);
+  enc_src_pad = gst_pad_get_peer (splitmux_sink_pad);
+  fail_if (enc_src_pad == NULL);
+  fail_unless (gst_pad_unlink (enc_src_pad, splitmux_sink_pad));
+  gst_object_unref (enc_src_pad);
+  gst_element_release_request_pad (sink, splitmux_sink_pad);
+  gst_object_unref (splitmux_sink_pad);
+  /* at this point the pad must be released - try to find it again to verify */
+  splitmux_sink_pad = gst_element_get_static_pad (sink, "video");
+  fail_if (splitmux_sink_pad != NULL);
+  g_object_unref (sink);
+
+  gst_object_unref (pipeline);
+
+  count = count_files (tmpdir);
+  fail_unless (count == 3, "Expected 3 output files, got %d", count);
+
+  in_pattern = g_build_filename (tmpdir, "matroska*.mkv", NULL);
+  test_playback (in_pattern, 0, 3 * GST_SECOND, TRUE);
+  g_free (in_pattern);
+}
+
+GST_END_TEST;
+
+/* For verifying bug https://bugzilla.gnome.org/show_bug.cgi?id=762893 */
+GST_START_TEST (test_splitmuxsink_reuse_simple)
+{
+  GstElement *sink;
+  GstPad *pad;
+
+  sink = gst_element_factory_make ("splitmuxsink", NULL);
+  pad = gst_element_get_request_pad (sink, "video");
+  fail_unless (pad != NULL);
+  g_object_set (sink, "location", "/dev/null", NULL);
+
+  fail_unless (gst_element_set_state (sink,
+          GST_STATE_PLAYING) == GST_STATE_CHANGE_ASYNC);
+  fail_unless (gst_element_set_state (sink,
+          GST_STATE_NULL) == GST_STATE_CHANGE_SUCCESS);
+  fail_unless (gst_element_set_state (sink,
+          GST_STATE_PLAYING) == GST_STATE_CHANGE_ASYNC);
+  fail_unless (gst_element_set_state (sink,
+          GST_STATE_NULL) == GST_STATE_CHANGE_SUCCESS);
+
+  gst_element_release_request_pad (sink, pad);
+  gst_object_unref (pad);
+  gst_object_unref (sink);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsink_muxer_pad_map)
+{
+  GstElement *sink, *muxer;
+  GstPad *muxpad;
+  GstPad *pad1 = NULL, *pad2 = NULL;
+  GstStructure *pad_map;
+
+  pad_map = gst_structure_new ("x-pad-map",
+      "video", G_TYPE_STRING, "video_100",
+      "audio_0", G_TYPE_STRING, "audio_101", NULL);
+
+  muxer = gst_element_factory_make ("qtmux", NULL);
+  fail_if (muxer == NULL);
+  sink = gst_element_factory_make ("splitmuxsink", NULL);
+  fail_if (sink == NULL);
+
+  g_object_set (sink, "muxer", muxer, "muxer-pad-map", pad_map, NULL);
+  gst_structure_free (pad_map);
+
+  pad1 = gst_element_get_request_pad (sink, "video");
+  fail_unless (g_str_equal ("video", GST_PAD_NAME (pad1)));
+  muxpad = gst_element_get_static_pad (muxer, "video_100");
+  fail_unless (muxpad != NULL);
+  gst_object_unref (muxpad);
+
+  pad2 = gst_element_get_request_pad (sink, "audio_0");
+  fail_unless (g_str_equal ("audio_0", GST_PAD_NAME (pad2)));
+  muxpad = gst_element_get_static_pad (muxer, "audio_101");
+  fail_unless (muxpad != NULL);
+  gst_object_unref (muxpad);
+
+  g_object_set (sink, "location", "/dev/null", NULL);
+
+  fail_unless (gst_element_set_state (sink,
+          GST_STATE_PLAYING) == GST_STATE_CHANGE_ASYNC);
+  fail_unless (gst_element_set_state (sink,
+          GST_STATE_NULL) == GST_STATE_CHANGE_SUCCESS);
+
+  gst_element_release_request_pad (sink, pad1);
+  gst_object_unref (pad1);
+  gst_element_release_request_pad (sink, pad2);
+  gst_object_unref (pad2);
+  gst_object_unref (sink);
+}
+
+GST_END_TEST;
+
+static GstPadProbeReturn
+count_upstrea_fku (GstPad * pad, GstPadProbeInfo * info,
+    guint * upstream_fku_count)
+{
+  GstEvent *event = GST_PAD_PROBE_INFO_EVENT (info);
+
+  switch (GST_EVENT_TYPE (event)) {
+    case GST_EVENT_CUSTOM_UPSTREAM:
+      if (gst_video_event_is_force_key_unit (event))
+        *upstream_fku_count += 1;
+      break;
+    default:
+      break;
+  }
+
+  return GST_PAD_PROBE_OK;
+}
+
+static void
+splitmuxsink_split_by_keyframe (gboolean send_keyframe_request,
+    guint max_size_time_sec, guint encoder_key_interval_sec)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *sink;
+  GstElement *enc;
+  GstPad *srcpad;
+  gchar *pipeline_str;
+  gchar *dest_pattern;
+  guint count;
+  guint expected_count;
+  gchar *in_pattern;
+  guint upstream_fku_count = 0;
+  guint expected_fku_count;
+
+  pipeline_str = g_strdup_printf ("splitmuxsink name=splitsink "
+      "max-size-time=%" G_GUINT64_FORMAT
+      " send-keyframe-requests=%s muxer=qtmux "
+      "videotestsrc num-buffers=30 ! video/x-raw,width=80,height=64,framerate=5/1 "
+      "! videoconvert ! queue ! vp8enc name=enc keyframe-max-dist=%d ! splitsink.video ",
+      max_size_time_sec * GST_SECOND, send_keyframe_request ? "true" : "false",
+      encoder_key_interval_sec * 5);
+
+  pipeline = gst_parse_launch (pipeline_str, NULL);
+  g_free (pipeline_str);
+
+  fail_if (pipeline == NULL);
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  g_signal_connect (sink, "format-location-full",
+      (GCallback) check_format_location, NULL);
+  dest_pattern = g_build_filename (tmpdir, "out%05d.m4v", NULL);
+  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
+  g_free (dest_pattern);
+  g_object_unref (sink);
+
+  enc = gst_bin_get_by_name (GST_BIN (pipeline), "enc");
+  fail_if (enc == NULL);
+  srcpad = gst_element_get_static_pad (enc, "src");
+  fail_if (srcpad == NULL);
+
+  gst_pad_add_probe (srcpad, GST_PAD_PROBE_TYPE_EVENT_UPSTREAM,
+      (GstPadProbeCallback) count_upstrea_fku, &upstream_fku_count, NULL);
+  gst_object_unref (srcpad);
+  gst_object_unref (enc);
+
+  msg = run_pipeline (pipeline);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  gst_object_unref (pipeline);
+
+  count = count_files (tmpdir);
+  expected_count = 6 / max_size_time_sec;
+  fail_unless (count == expected_count,
+      "Expected %d output files, got %d", expected_count, count);
+
+  if (!send_keyframe_request) {
+    expected_fku_count = 0;
+  } else {
+    expected_fku_count = count;
+  }
+
+  GST_INFO ("Upstream force keyunit event count %d", upstream_fku_count);
+
+  fail_unless (upstream_fku_count == expected_fku_count,
+      "Expected upstream force keyunit event count %d, got %d",
+      expected_fku_count, upstream_fku_count);
+
+  in_pattern = g_build_filename (tmpdir, "out*.m4v", NULL);
+  /* FIXME: Reverse playback works poorly with multiple video streams
+   * in qtdemux (at least, maybe other demuxers) at the time this was
+   * written, and causes test failures like buffers being output
+   * multiple times by qtdemux as it loops through GOPs. Disable that
+   * for now */
+  test_playback (in_pattern, 0, 6 * GST_SECOND, FALSE);
+  g_free (in_pattern);
+}
+
+GST_START_TEST (test_splitmuxsink_without_keyframe_request)
+{
+  /* This encoding option is intended to produce keyframe per 1 seconds
+   * but splitmuxsink will split file per 2 second without keyframe request */
+  splitmuxsink_split_by_keyframe (FALSE, 2, 1);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsink_keyframe_request)
+{
+  /* This encoding option is intended to produce keyframe per 2 seconds
+   * and splitmuxsink will request keyframe per 2 seconds as well.
+   * This should produce 2 seconds long files */
+  splitmuxsink_split_by_keyframe (TRUE, 2, 2);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsink_keyframe_request_more)
+{
+  /* This encoding option is intended to produce keyframe per 2 seconds
+   * but splitmuxsink will request keyframe per 1 second. This should produce
+   * 1 second long files */
+  splitmuxsink_split_by_keyframe (TRUE, 1, 2);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsink_keyframe_request_less)
+{
+  /* This encoding option is intended to produce keyframe per 1 second
+   * but splitmuxsink will request keyframe per 2 seconds. This should produce
+   * 2 seconds long files */
+  splitmuxsink_split_by_keyframe (TRUE, 2, 1);
+}
+
+GST_END_TEST;
+
+static void
+splitmuxsink_split_by_keyframe_timecode (gboolean send_keyframe_request,
+    const gchar * maxsize_timecode_string, guint maxsize_timecode_in_sec,
+    guint encoder_key_interval_sec)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *sink;
+  GstElement *enc;
+  GstPad *srcpad;
+  gchar *pipeline_str;
+  gchar *dest_pattern;
+  guint count;
+  guint expected_count;
+  gchar *in_pattern;
+  guint upstream_fku_count = 0;
+  guint expected_fku_count;
+
+  pipeline_str = g_strdup_printf ("splitmuxsink name=splitsink "
+      "max-size-timecode=%s"
+      " send-keyframe-requests=%s muxer=qtmux "
+      "videotestsrc num-buffers=30 ! video/x-raw,width=80,height=64,framerate=5/1 "
+      "! videoconvert ! timecodestamper ! queue ! vp8enc name=enc keyframe-max-dist=%d ! splitsink.video ",
+      maxsize_timecode_string, send_keyframe_request ? "true" : "false",
+      encoder_key_interval_sec ? encoder_key_interval_sec * 5 : 1);
+
+  pipeline = gst_parse_launch (pipeline_str, NULL);
+  g_free (pipeline_str);
+
+  fail_if (pipeline == NULL);
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  g_signal_connect (sink, "format-location-full",
+      (GCallback) check_format_location, NULL);
+  dest_pattern = g_build_filename (tmpdir, "out%05d.m4v", NULL);
+  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
+  g_free (dest_pattern);
+  g_object_unref (sink);
+
+  enc = gst_bin_get_by_name (GST_BIN (pipeline), "enc");
+  fail_if (enc == NULL);
+  srcpad = gst_element_get_static_pad (enc, "src");
+  fail_if (srcpad == NULL);
+
+  gst_pad_add_probe (srcpad, GST_PAD_PROBE_TYPE_EVENT_UPSTREAM,
+      (GstPadProbeCallback) count_upstrea_fku, &upstream_fku_count, NULL);
+  gst_object_unref (srcpad);
+  gst_object_unref (enc);
+
+  msg = run_pipeline (pipeline);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  gst_object_unref (pipeline);
+
+  count = count_files (tmpdir);
+  expected_count = (6 / maxsize_timecode_in_sec) +
+      (6 % maxsize_timecode_in_sec ? 1 : 0);
+  fail_unless (count == expected_count,
+      "Expected %d output files, got %d", expected_count, count);
+
+  if (!send_keyframe_request) {
+    expected_fku_count = 0;
+  } else {
+    expected_fku_count = count;
+  }
+
+  GST_INFO ("Upstream force keyunit event count %d", upstream_fku_count);
+
+  fail_unless (upstream_fku_count == expected_fku_count,
+      "Expected upstream force keyunit event count %d, got %d",
+      expected_fku_count, upstream_fku_count);
+
+  in_pattern = g_build_filename (tmpdir, "out*.m4v", NULL);
+  /* FIXME: Reverse playback works poorly with multiple video streams
+   * in qtdemux (at least, maybe other demuxers) at the time this was
+   * written, and causes test failures like buffers being output
+   * multiple times by qtdemux as it loops through GOPs. Disable that
+   * for now */
+  test_playback (in_pattern, 0, 6 * GST_SECOND, FALSE);
+  g_free (in_pattern);
+}
+
+GST_START_TEST (test_splitmuxsink_without_keyframe_request_timecode)
+{
+  /* This encoding option is intended to produce keyframe per 1 second
+   * but splitmuxsink will split file per 2 second without keyframe request */
+  splitmuxsink_split_by_keyframe_timecode (FALSE, "00:00:02:00", 2, 1);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsink_keyframe_request_timecode)
+{
+  /* This encoding option is intended to produce keyframe per 1 second
+   * but splitmuxsink will request keyframe per 2 seconds. This should produce
+   * 2 seconds long files */
+  splitmuxsink_split_by_keyframe_timecode (TRUE, "00:00:02:00", 2, 1);
+}
+
+GST_END_TEST;
+
+GST_START_TEST
+    (test_splitmuxsink_keyframe_request_timecode_trailing_small_segment) {
+  /* This encoding option is intended to produce keyframe per 1 second
+   * but splitmuxsink will request keyframe per 4 seconds. This should produce
+   * 4 seconds long files */
+  splitmuxsink_split_by_keyframe_timecode (TRUE, "00:00:04:00", 4, 1);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsink_keyframe_request_timecode_all_intra)
+{
+  /* This encoding option is intended to produce keyframe for every frame.
+   * This should produce 1 second long files */
+  splitmuxsink_split_by_keyframe_timecode (TRUE, "00:00:01:00", 1, 0);
+}
+
+GST_END_TEST;
+
+static Suite *
+splitmuxsink_suite (void)
+{
+  Suite *s = suite_create ("splitmuxsink");
+  TCase *tc_chain = tcase_create ("general");
+  TCase *tc_chain_basic = tcase_create ("basic");
+  TCase *tc_chain_complex = tcase_create ("complex");
+  TCase *tc_chain_mp4_jpeg = tcase_create ("caps_change");
+  gboolean have_theora, have_ogg, have_vorbis, have_matroska, have_qtmux,
+      have_jpeg, have_vp8, have_timecodestamper;
+
+  /* we assume that if encoder/muxer are there, decoder/demuxer will be a well */
+  have_theora = gst_registry_check_feature_version (gst_registry_get (),
+      "theoraenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_ogg = gst_registry_check_feature_version (gst_registry_get (),
+      "oggmux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_vorbis = gst_registry_check_feature_version (gst_registry_get (),
+      "vorbisenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_matroska = gst_registry_check_feature_version (gst_registry_get (),
+      "matroskamux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_qtmux = gst_registry_check_feature_version (gst_registry_get (),
+      "qtmux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_jpeg = gst_registry_check_feature_version (gst_registry_get (),
+      "jpegenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_vp8 = gst_registry_check_feature_version (gst_registry_get (),
+      "vp8enc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_timecodestamper =
+      gst_registry_check_feature_version (gst_registry_get (),
+      "timecodestamper", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+
+  suite_add_tcase (s, tc_chain);
+  suite_add_tcase (s, tc_chain_basic);
+  suite_add_tcase (s, tc_chain_complex);
+  suite_add_tcase (s, tc_chain_mp4_jpeg);
+
+  tcase_add_test (tc_chain_basic, test_splitmuxsink_reuse_simple);
+
+  if (have_theora && have_ogg) {
+    tcase_add_checked_fixture (tc_chain, tempdir_setup, tempdir_cleanup);
+
+    tcase_add_test (tc_chain, test_splitmuxsink);
+
+    if (have_matroska && have_vorbis) {
+      tcase_add_checked_fixture (tc_chain_complex, tempdir_setup,
+          tempdir_cleanup);
+
+      tcase_add_test (tc_chain, test_splitmuxsink_async);
+    } else {
+      GST_INFO ("Skipping tests, missing plugins: matroska and/or vorbis");
+    }
+  } else {
+    GST_INFO ("Skipping tests, missing plugins: theora and/or ogg");
+  }
+
+
+  if (have_qtmux && have_jpeg) {
+    tcase_add_checked_fixture (tc_chain_mp4_jpeg, tempdir_setup,
+        tempdir_cleanup);
+    tcase_add_test (tc_chain_mp4_jpeg, test_splitmuxsink_muxer_pad_map);
+  } else {
+    GST_INFO ("Skipping tests, missing plugins: jpegenc or mp4mux");
+  }
+
+  if (have_qtmux && have_vp8) {
+    tcase_add_test (tc_chain, test_splitmuxsink_multivid);
+    tcase_add_test (tc_chain, test_splitmuxsink_without_keyframe_request);
+    tcase_add_test (tc_chain, test_splitmuxsink_keyframe_request);
+    tcase_add_test (tc_chain, test_splitmuxsink_keyframe_request_more);
+    tcase_add_test (tc_chain, test_splitmuxsink_keyframe_request_less);
+  } else {
+    GST_INFO ("Skipping tests, missing plugins: vp8enc or mp4mux");
+  }
+
+  if (have_qtmux && have_vp8 && have_timecodestamper) {
+    tcase_add_test (tc_chain,
+        test_splitmuxsink_without_keyframe_request_timecode);
+    tcase_add_test (tc_chain, test_splitmuxsink_keyframe_request_timecode);
+    tcase_add_test (tc_chain,
+        test_splitmuxsink_keyframe_request_timecode_trailing_small_segment);
+    tcase_add_test (tc_chain,
+        test_splitmuxsink_keyframe_request_timecode_all_intra);
+  } else {
+    GST_INFO
+        ("Skipping tests, missing plugins: vp8enc, mp4mux, or timecodestamper");
+  }
+
+  return s;
+}
+
+GST_CHECK_MAIN (splitmuxsink);
diff --git a/tests/check/elements/splitmuxsrc.c b/tests/check/elements/splitmuxsrc.c
new file mode 100644 (file)
index 0000000..e689a83
--- /dev/null
@@ -0,0 +1,736 @@
+/* GStreamer unit test for splitmuxsrc elements
+ *
+ * Copyright (C) 2007 David A. Schleef <ds@schleef.org>
+ * Copyright (C) 2015 Jan Schmidt <jan@centricular.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.
+ */
+
+#ifdef HAVE_CONFIG_H
+#  include "config.h"
+#endif
+
+#include <glib/gstdio.h>
+
+#include <gst/check/gstcheck.h>
+#include <gst/app/app.h>
+#include <gst/video/video.h>
+#include <stdlib.h>
+
+gchar *tmpdir = NULL;
+GstClockTime first_ts;
+GstClockTime last_ts;
+gdouble current_rate;
+
+static void
+tempdir_setup (void)
+{
+  const gchar *systmp = g_get_tmp_dir ();
+  tmpdir = g_build_filename (systmp, "splitmux-test-XXXXXX", NULL);
+  /* Rewrites tmpdir template input: */
+  tmpdir = g_mkdtemp (tmpdir);
+}
+
+static void
+tempdir_cleanup (void)
+{
+  GDir *d;
+  const gchar *f;
+
+  fail_if (tmpdir == NULL);
+
+  d = g_dir_open (tmpdir, 0, NULL);
+  fail_if (d == NULL);
+
+  while ((f = g_dir_read_name (d)) != NULL) {
+    gchar *fname = g_build_filename (tmpdir, f, NULL);
+    fail_if (g_remove (fname) != 0, "Failed to remove tmp file %s", fname);
+    g_free (fname);
+  }
+  g_dir_close (d);
+
+  fail_if (g_remove (tmpdir) != 0, "Failed to delete tmpdir %s", tmpdir);
+
+  g_free (tmpdir);
+  tmpdir = NULL;
+}
+
+static guint
+count_files (const gchar * target)
+{
+  GDir *d;
+  const gchar *f;
+  guint ret = 0;
+
+  d = g_dir_open (target, 0, NULL);
+  fail_if (d == NULL);
+
+  while ((f = g_dir_read_name (d)) != NULL)
+    ret++;
+  g_dir_close (d);
+
+  return ret;
+}
+
+static void
+dump_error (GstMessage * msg)
+{
+  GError *err = NULL;
+  gchar *dbg_info;
+
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR);
+
+  gst_message_parse_error (msg, &err, &dbg_info);
+
+  g_printerr ("ERROR from element %s: %s\n",
+      GST_OBJECT_NAME (msg->src), err->message);
+  g_printerr ("Debugging info: %s\n", (dbg_info) ? dbg_info : "none");
+  g_error_free (err);
+  g_free (dbg_info);
+}
+
+static GstMessage *
+run_pipeline (GstElement * pipeline)
+{
+  GstBus *bus = gst_element_get_bus (GST_ELEMENT (pipeline));
+  GstMessage *msg;
+
+  gst_element_set_state (pipeline, GST_STATE_PLAYING);
+  msg = gst_bus_poll (bus, GST_MESSAGE_EOS | GST_MESSAGE_ERROR, -1);
+  gst_element_set_state (pipeline, GST_STATE_NULL);
+
+  gst_object_unref (bus);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+
+  return msg;
+}
+
+static void
+seek_pipeline (GstElement * pipeline, gdouble rate, GstClockTime start,
+    GstClockTime end)
+{
+  /* Pause the pipeline, seek to the desired range / rate, wait for PAUSED again, then
+   * clear the tracking vars for start_ts / end_ts */
+  gst_element_set_state (pipeline, GST_STATE_PAUSED);
+  gst_element_get_state (pipeline, NULL, NULL, GST_CLOCK_TIME_NONE);
+
+  /* specific end time not implemented: */
+  fail_unless (end == GST_CLOCK_TIME_NONE);
+
+  gst_element_seek (pipeline, rate, GST_FORMAT_TIME,
+      GST_SEEK_FLAG_FLUSH | GST_SEEK_FLAG_ACCURATE, GST_SEEK_TYPE_SET, start,
+      GST_SEEK_TYPE_END, 0);
+
+  /* Wait for the pipeline to preroll again */
+  gst_element_get_state (pipeline, NULL, NULL, GST_CLOCK_TIME_NONE);
+
+  GST_LOG ("Seeked pipeline. Rate %f time range %" GST_TIME_FORMAT " to %"
+      GST_TIME_FORMAT, rate, GST_TIME_ARGS (start), GST_TIME_ARGS (end));
+
+  /* Clear tracking variables now that the seek is complete */
+  first_ts = last_ts = GST_CLOCK_TIME_NONE;
+  current_rate = rate;
+};
+
+static GstFlowReturn
+receive_sample (GstAppSink * appsink, gpointer user_data G_GNUC_UNUSED)
+{
+  GstSample *sample;
+  GstSegment *seg;
+  GstBuffer *buf;
+  GstClockTime start;
+  GstClockTime end;
+
+  g_signal_emit_by_name (appsink, "pull-sample", &sample);
+  fail_unless (sample != NULL);
+
+  seg = gst_sample_get_segment (sample);
+  fail_unless (seg != NULL);
+
+  buf = gst_sample_get_buffer (sample);
+  fail_unless (buf != NULL);
+
+  GST_LOG ("Got buffer %" GST_PTR_FORMAT, buf);
+
+  start = GST_BUFFER_PTS (buf);
+  end = start;
+
+  if (GST_CLOCK_TIME_IS_VALID (start))
+    start = gst_segment_to_stream_time (seg, GST_FORMAT_TIME, start);
+
+  if (GST_CLOCK_TIME_IS_VALID (end)) {
+    if (GST_BUFFER_DURATION_IS_VALID (buf))
+      end += GST_BUFFER_DURATION (buf);
+
+    end = gst_segment_to_stream_time (seg, GST_FORMAT_TIME, end);
+  }
+
+  GST_DEBUG ("Got buffer stream time %" GST_TIME_FORMAT " to %" GST_TIME_FORMAT,
+      GST_TIME_ARGS (start), GST_TIME_ARGS (end));
+
+  /* Check time is moving in the right direction */
+  if (current_rate > 0) {
+    if (GST_CLOCK_TIME_IS_VALID (first_ts))
+      fail_unless (start >= first_ts,
+          "Timestamps went backward during forward play, %" GST_TIME_FORMAT
+          " < %" GST_TIME_FORMAT, GST_TIME_ARGS (start),
+          GST_TIME_ARGS (first_ts));
+    if (GST_CLOCK_TIME_IS_VALID (last_ts))
+      fail_unless (end >= last_ts,
+          "Timestamps went backward during forward play, %" GST_TIME_FORMAT
+          " < %" GST_TIME_FORMAT, GST_TIME_ARGS (end), GST_TIME_ARGS (last_ts));
+  } else {
+    fail_unless (start <= first_ts,
+        "Timestamps went forward during reverse play, %" GST_TIME_FORMAT " > %"
+        GST_TIME_FORMAT, GST_TIME_ARGS (start), GST_TIME_ARGS (first_ts));
+    fail_unless (end <= last_ts,
+        "Timestamps went forward during reverse play, %" GST_TIME_FORMAT " > %"
+        GST_TIME_FORMAT, GST_TIME_ARGS (end), GST_TIME_ARGS (last_ts));
+  }
+
+  /* update the range of timestamps we've encountered */
+  if (!GST_CLOCK_TIME_IS_VALID (first_ts) || start < first_ts)
+    first_ts = start;
+  if (!GST_CLOCK_TIME_IS_VALID (last_ts) || end > last_ts)
+    last_ts = end;
+
+  gst_sample_unref (sample);
+
+  return GST_FLOW_OK;
+}
+
+static void
+test_playback (const gchar * in_pattern, GstClockTime exp_first_time,
+    GstClockTime exp_last_time, gboolean test_reverse)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *appsink;
+  GstElement *fakesink2;
+  GstAppSinkCallbacks callbacks = { NULL };
+  gchar *uri;
+
+  GST_DEBUG ("Playing back files matching %s", in_pattern);
+
+  pipeline = gst_element_factory_make ("playbin", NULL);
+  fail_if (pipeline == NULL);
+
+  appsink = gst_element_factory_make ("appsink", NULL);
+  fail_if (appsink == NULL);
+  g_object_set (G_OBJECT (appsink), "sync", FALSE, NULL);
+
+  g_object_set (G_OBJECT (pipeline), "video-sink", appsink, NULL);
+  fakesink2 = gst_element_factory_make ("fakesink", NULL);
+  fail_if (fakesink2 == NULL);
+  g_object_set (G_OBJECT (pipeline), "audio-sink", fakesink2, NULL);
+
+  uri = g_strdup_printf ("splitmux://%s", in_pattern);
+
+  g_object_set (G_OBJECT (pipeline), "uri", uri, NULL);
+  g_free (uri);
+
+  callbacks.new_sample = receive_sample;
+  gst_app_sink_set_callbacks (GST_APP_SINK (appsink), &callbacks, NULL, NULL);
+
+  /* test forwards */
+  seek_pipeline (pipeline, 1.0, 0, -1);
+  fail_unless (first_ts == GST_CLOCK_TIME_NONE);
+  msg = run_pipeline (pipeline);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  /* Check we saw the entire range of values */
+  fail_unless (first_ts == exp_first_time,
+      "Expected start of playback range %" GST_TIME_FORMAT ", got %"
+      GST_TIME_FORMAT, GST_TIME_ARGS (exp_first_time),
+      GST_TIME_ARGS (first_ts));
+  fail_unless (last_ts == exp_last_time,
+      "Expected end of playback range %" GST_TIME_FORMAT ", got %"
+      GST_TIME_FORMAT, GST_TIME_ARGS (exp_last_time), GST_TIME_ARGS (last_ts));
+
+  if (test_reverse) {
+    /* Test backwards */
+    seek_pipeline (pipeline, -1.0, 0, -1);
+    msg = run_pipeline (pipeline);
+    fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+    gst_message_unref (msg);
+    /* Check we saw the entire range of values */
+    fail_unless (first_ts == exp_first_time,
+        "Expected start of playback range %" GST_TIME_FORMAT
+        ", got %" GST_TIME_FORMAT, GST_TIME_ARGS (exp_first_time),
+        GST_TIME_ARGS (first_ts));
+    fail_unless (last_ts == exp_last_time,
+        "Expected end of playback range %" GST_TIME_FORMAT
+        ", got %" GST_TIME_FORMAT, GST_TIME_ARGS (exp_last_time),
+        GST_TIME_ARGS (last_ts));
+  }
+
+  gst_object_unref (pipeline);
+}
+
+GST_START_TEST (test_splitmuxsrc)
+{
+  gchar *in_pattern =
+      g_build_filename (GST_TEST_FILES_PATH, "splitvideo*.ogg", NULL);
+  test_playback (in_pattern, 0, 3 * GST_SECOND, TRUE);
+  g_free (in_pattern);
+}
+
+GST_END_TEST;
+
+static gchar **
+src_format_location_cb (GstElement * splitmuxsrc, gpointer user_data)
+{
+  gchar **result = g_malloc0_n (4, sizeof (gchar *));
+  result[0] = g_build_filename (GST_TEST_FILES_PATH, "splitvideo00.ogg", NULL);
+  result[1] = g_build_filename (GST_TEST_FILES_PATH, "splitvideo01.ogg", NULL);
+  result[2] = g_build_filename (GST_TEST_FILES_PATH, "splitvideo02.ogg", NULL);
+  return result;
+}
+
+GST_START_TEST (test_splitmuxsrc_format_location)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *src;
+  GError *error = NULL;
+
+  pipeline = gst_parse_launch ("splitmuxsrc name=splitsrc ! decodebin "
+      "! fakesink", &error);
+  g_assert_no_error (error);
+  fail_if (pipeline == NULL);
+
+  src = gst_bin_get_by_name (GST_BIN (pipeline), "splitsrc");
+  g_signal_connect (src, "format-location",
+      (GCallback) src_format_location_cb, NULL);
+  g_object_unref (src);
+
+  msg = run_pipeline (pipeline);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+  gst_object_unref (pipeline);
+}
+
+GST_END_TEST;
+
+static gchar *
+check_format_location (GstElement * object,
+    guint fragment_id, GstSample * first_sample)
+{
+  GstBuffer *buf = gst_sample_get_buffer (first_sample);
+
+  /* Must have a buffer */
+  fail_if (buf == NULL);
+  GST_LOG ("New file - first buffer %" GST_TIME_FORMAT,
+      GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)));
+
+  return NULL;
+}
+
+static GstPadProbeReturn
+intercept_stream_start (GstPad * pad, GstPadProbeInfo * info,
+    gpointer user_data)
+{
+  GstEvent *event = gst_pad_probe_info_get_event (info);
+
+  if (GST_EVENT_TYPE (event) == GST_EVENT_STREAM_START) {
+    GstStreamFlags flags;
+    event = gst_event_make_writable (event);
+    gst_event_parse_stream_flags (event, &flags);
+    gst_event_set_stream_flags (event, flags | GST_STREAM_FLAG_SPARSE);
+    GST_PAD_PROBE_INFO_DATA (info) = event;
+  }
+
+  return GST_PAD_PROBE_OK;
+}
+
+static GstFlowReturn
+new_sample_verify_continuous_timestamps (GstAppSink * appsink,
+    gpointer user_data)
+{
+  GstSample *sample;
+  GstBuffer *buffer;
+  GstClockTime *prev_ts = user_data;
+  GstClockTime new_ts;
+
+  sample = gst_app_sink_pull_sample (appsink);
+  buffer = gst_sample_get_buffer (sample);
+
+  new_ts = GST_BUFFER_PTS (buffer);
+  if (GST_CLOCK_TIME_IS_VALID (*prev_ts)) {
+    fail_unless (*prev_ts < new_ts,
+        "%s: prev_ts (%" GST_TIME_FORMAT ") >= new_ts (%" GST_TIME_FORMAT ")",
+        GST_OBJECT_NAME (appsink), GST_TIME_ARGS (*prev_ts),
+        GST_TIME_ARGS (new_ts));
+  }
+
+  *prev_ts = new_ts;
+  gst_sample_unref (sample);
+  return GST_FLOW_OK;
+}
+
+static GstFlowReturn
+new_sample_verify_1sec_offset (GstAppSink * appsink, gpointer user_data)
+{
+  GstSample *sample;
+  GstBuffer *buffer;
+  GstClockTime *prev_ts = user_data;
+  GstClockTime new_ts;
+
+  sample = gst_app_sink_pull_sample (appsink);
+  buffer = gst_sample_get_buffer (sample);
+
+  new_ts = GST_BUFFER_PTS (buffer);
+  if (GST_CLOCK_TIME_IS_VALID (*prev_ts)) {
+    fail_unless (new_ts > (*prev_ts + 900 * GST_MSECOND),
+        "%s: prev_ts (%" GST_TIME_FORMAT ") + 0.9s >= new_ts (%"
+        GST_TIME_FORMAT ")", GST_OBJECT_NAME (appsink),
+        GST_TIME_ARGS (*prev_ts), GST_TIME_ARGS (new_ts));
+  }
+
+  *prev_ts = new_ts;
+  gst_sample_unref (sample);
+  return GST_FLOW_OK;
+}
+
+/* https://bugzilla.gnome.org/show_bug.cgi?id=761086 */
+GST_START_TEST (test_splitmuxsrc_sparse_streams)
+{
+  GstElement *pipeline;
+  GstElement *element;
+  gchar *dest_pattern;
+  GstElement *appsrc;
+  GstPad *appsrc_src;
+  GstBus *bus;
+  GstMessage *msg;
+  gint i;
+
+  /* generate files */
+
+  /* in this test, we have 5sec of data with files split at 1sec intervals */
+  pipeline =
+      gst_parse_launch
+      ("videotestsrc num-buffers=75 !"
+      "  video/x-raw,width=80,height=64,framerate=15/1 !"
+      "  theoraenc keyframe-force=5 ! splitmuxsink name=splitsink"
+      "    max-size-time=1000000000 muxer=matroskamux"
+      " audiotestsrc num-buffers=100 samplesperbuffer=1024 !"
+      "  audio/x-raw,rate=20000 ! vorbisenc ! splitsink.audio_%u"
+      " appsrc name=appsrc format=time caps=text/x-raw,format=utf8 !"
+      "  splitsink.subtitle_%u", NULL);
+  fail_if (pipeline == NULL);
+
+  element = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (element == NULL);
+  dest_pattern = g_build_filename (tmpdir, "out%05d.ogg", NULL);
+  g_object_set (G_OBJECT (element), "location", dest_pattern, NULL);
+  g_clear_pointer (&dest_pattern, g_free);
+  g_clear_object (&element);
+
+  appsrc = gst_bin_get_by_name (GST_BIN (pipeline), "appsrc");
+  fail_if (appsrc == NULL);
+
+  /* add the SPARSE flag on the stream-start event of the subtitle stream */
+  appsrc_src = gst_element_get_static_pad (appsrc, "src");
+  fail_if (appsrc_src == NULL);
+  gst_pad_add_probe (appsrc_src, GST_PAD_PROBE_TYPE_EVENT_DOWNSTREAM,
+      intercept_stream_start, NULL, NULL);
+  g_clear_object (&appsrc_src);
+
+  bus = gst_pipeline_get_bus (GST_PIPELINE (pipeline));
+
+  gst_element_set_state (pipeline, GST_STATE_PLAYING);
+
+  /* push subtitles, one per second, starting from t=100ms */
+  for (i = 0; i < 5; i++) {
+    GstBuffer *buffer = gst_buffer_new_allocate (NULL, 5, NULL);
+    GstMapInfo info;
+
+    gst_buffer_map (buffer, &info, GST_MAP_WRITE);
+    strcpy ((char *) info.data, "test");
+    gst_buffer_unmap (buffer, &info);
+
+    GST_BUFFER_PTS (buffer) = i * GST_SECOND + 100 * GST_MSECOND;
+    GST_BUFFER_DTS (buffer) = GST_BUFFER_PTS (buffer);
+
+    fail_if (gst_app_src_push_buffer (GST_APP_SRC (appsrc), buffer)
+        != GST_FLOW_OK);
+  }
+  fail_if (gst_app_src_end_of_stream (GST_APP_SRC (appsrc)) != GST_FLOW_OK);
+
+  msg = gst_bus_timed_pop_filtered (bus, 5 * GST_SECOND, GST_MESSAGE_EOS);
+  g_clear_pointer (&msg, gst_message_unref);
+
+  gst_element_set_state (pipeline, GST_STATE_NULL);
+
+  g_clear_object (&appsrc);
+  g_clear_object (&bus);
+  g_clear_object (&pipeline);
+
+  /* read and verify */
+
+  pipeline =
+      gst_parse_launch
+      ("splitmuxsrc name=splitsrc"
+      " splitsrc. ! theoradec ! appsink name=vsink sync=false emit-signals=true"
+      " splitsrc. ! vorbisdec ! appsink name=asink sync=false emit-signals=true"
+      " splitsrc. ! text/x-raw ! appsink name=tsink sync=false emit-signals=true",
+      NULL);
+  fail_if (pipeline == NULL);
+
+  element = gst_bin_get_by_name (GST_BIN (pipeline), "splitsrc");
+  fail_if (element == NULL);
+  dest_pattern = g_build_filename (tmpdir, "out*.ogg", NULL);
+  g_object_set (G_OBJECT (element), "location", dest_pattern, NULL);
+  g_clear_pointer (&dest_pattern, g_free);
+  g_clear_object (&element);
+
+  {
+    GstClockTime vsink_prev_ts = GST_CLOCK_TIME_NONE;
+    GstClockTime asink_prev_ts = GST_CLOCK_TIME_NONE;
+    GstClockTime tsink_prev_ts = GST_CLOCK_TIME_NONE;
+
+    /* verify that timestamps are continuously increasing for audio + video.
+     * if we hit bug 761086, timestamps will jump about -900ms after switching
+     * to a new part, because this is the difference between the last subtitle
+     * pts and the last audio/video pts */
+    element = gst_bin_get_by_name (GST_BIN (pipeline), "vsink");
+    g_signal_connect (element, "new-sample",
+        (GCallback) new_sample_verify_continuous_timestamps, &vsink_prev_ts);
+    g_clear_object (&element);
+
+    element = gst_bin_get_by_name (GST_BIN (pipeline), "asink");
+    g_signal_connect (element, "new-sample",
+        (GCallback) new_sample_verify_continuous_timestamps, &asink_prev_ts);
+    g_clear_object (&element);
+
+    /* also verify that subtitle timestamps are increasing by about 1s.
+     * if we hit bug 761086, timestamps will increase by exactly 100ms instead,
+     * because this is the relative difference between a part's start time
+     * (remember a new part starts every 1sec) and the subtitle's pts in that
+     * part, which will be added to the max_ts of the previous part, which
+     * equals the last subtitle's pts (and should not!) */
+    element = gst_bin_get_by_name (GST_BIN (pipeline), "tsink");
+    g_signal_connect (element, "new-sample",
+        (GCallback) new_sample_verify_1sec_offset, &tsink_prev_ts);
+    g_clear_object (&element);
+
+    msg = run_pipeline (pipeline);
+  }
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+
+  g_clear_pointer (&msg, gst_message_unref);
+  g_clear_object (&pipeline);
+}
+
+GST_END_TEST;
+
+struct CapsChangeData
+{
+  guint count;
+  GstElement *cf;
+};
+
+static GstPadProbeReturn
+switch_caps (GstPad * pad, GstPadProbeInfo * info, gpointer user_data)
+{
+  struct CapsChangeData *data = (struct CapsChangeData *) (user_data);
+
+  if (data->count == 4) {
+    GST_INFO ("Saw 5 buffers to the encoder. Switching caps");
+    gst_util_set_object_arg (G_OBJECT (data->cf), "caps",
+        "video/x-raw,width=160,height=128,framerate=10/1");
+  }
+  data->count++;
+  return GST_PAD_PROBE_OK;
+}
+
+GST_START_TEST (test_splitmuxsrc_caps_change)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *sink;
+  GstElement *cf;
+  GstPad *sinkpad;
+  gchar *dest_pattern;
+  guint count;
+  gchar *in_pattern;
+  struct CapsChangeData data;
+
+  /* This test creates a new file only by changing the caps, which
+   * qtmux will reject (for now - if qtmux starts supporting caps
+   * changes, this test will break and need fixing/disabling */
+  pipeline =
+      gst_parse_launch
+      ("videotestsrc num-buffers=10 !"
+      "  capsfilter name=c caps=video/x-raw,width=80,height=64,framerate=10/1 !"
+      "  jpegenc ! splitmuxsink name=splitsink muxer=qtmux", NULL);
+  fail_if (pipeline == NULL);
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  g_signal_connect (sink, "format-location-full",
+      (GCallback) check_format_location, NULL);
+  dest_pattern = g_build_filename (tmpdir, "out%05d.mp4", NULL);
+  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
+  g_free (dest_pattern);
+  g_object_unref (sink);
+
+  cf = gst_bin_get_by_name (GST_BIN (pipeline), "c");
+  sinkpad = gst_element_get_static_pad (cf, "sink");
+
+  data.cf = cf;
+  data.count = 0;
+
+  gst_pad_add_probe (sinkpad, GST_PAD_PROBE_TYPE_BUFFER,
+      switch_caps, &data, NULL);
+
+  gst_object_unref (sinkpad);
+  gst_object_unref (cf);
+
+  msg = run_pipeline (pipeline);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  gst_object_unref (pipeline);
+
+  count = count_files (tmpdir);
+  fail_unless (count == 2, "Expected 2 output files, got %d", count);
+
+  in_pattern = g_build_filename (tmpdir, "out*.mp4", NULL);
+  test_playback (in_pattern, 0, GST_SECOND, TRUE);
+  g_free (in_pattern);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_splitmuxsrc_robust_mux)
+{
+  GstMessage *msg;
+  GstElement *pipeline;
+  GstElement *sink;
+  gchar *dest_pattern;
+  gchar *in_pattern;
+
+  /* This test creates a new file only by changing the caps, which
+   * qtmux will reject (for now - if qtmux starts supporting caps
+   * changes, this test will break and need fixing/disabling */
+  pipeline =
+      gst_parse_launch
+      ("videotestsrc num-buffers=10 !"
+      "  video/x-raw,width=80,height=64,framerate=10/1 !"
+      "  jpegenc ! splitmuxsink name=splitsink muxer=\"qtmux reserved-bytes-per-sec=200 reserved-moov-update-period=100000000 \" max-size-time=500000000 use-robust-muxing=true",
+      NULL);
+  fail_if (pipeline == NULL);
+  sink = gst_bin_get_by_name (GST_BIN (pipeline), "splitsink");
+  fail_if (sink == NULL);
+  g_signal_connect (sink, "format-location-full",
+      (GCallback) check_format_location, NULL);
+  dest_pattern = g_build_filename (tmpdir, "out%05d.mp4", NULL);
+  g_object_set (G_OBJECT (sink), "location", dest_pattern, NULL);
+  g_free (dest_pattern);
+  g_object_unref (sink);
+
+  msg = run_pipeline (pipeline);
+
+  if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR)
+    dump_error (msg);
+  fail_unless (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS);
+  gst_message_unref (msg);
+
+  gst_object_unref (pipeline);
+
+  /* Unlike other tests, we don't check an explicit file size, because the overflow detection
+   * can be racy (depends on exactly when buffers get handed to the muxer and when it updates the
+   * reserved duration property. All we care about is that the muxing didn't fail because space ran out */
+
+  in_pattern = g_build_filename (tmpdir, "out*.mp4", NULL);
+  test_playback (in_pattern, 0, GST_SECOND, TRUE);
+  g_free (in_pattern);
+}
+
+GST_END_TEST;
+
+static Suite *
+splitmuxsrc_suite (void)
+{
+  Suite *s = suite_create ("splitmuxsrc");
+  TCase *tc_chain = tcase_create ("general");
+  TCase *tc_chain_complex = tcase_create ("complex");
+  TCase *tc_chain_mp4_jpeg = tcase_create ("caps_change");
+  gboolean have_theora, have_ogg, have_vorbis, have_matroska, have_qtmux,
+      have_jpeg;
+
+  /* we assume that if encoder/muxer are there, decoder/demuxer will be a well */
+  have_theora = gst_registry_check_feature_version (gst_registry_get (),
+      "theoraenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_ogg = gst_registry_check_feature_version (gst_registry_get (),
+      "oggmux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_vorbis = gst_registry_check_feature_version (gst_registry_get (),
+      "vorbisenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_matroska = gst_registry_check_feature_version (gst_registry_get (),
+      "matroskamux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_qtmux = gst_registry_check_feature_version (gst_registry_get (),
+      "qtmux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+  have_jpeg = gst_registry_check_feature_version (gst_registry_get (),
+      "jpegenc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0);
+
+  suite_add_tcase (s, tc_chain);
+  suite_add_tcase (s, tc_chain_complex);
+  suite_add_tcase (s, tc_chain_mp4_jpeg);
+
+  if (have_theora && have_ogg) {
+    tcase_add_checked_fixture (tc_chain, tempdir_setup, tempdir_cleanup);
+
+    tcase_add_test (tc_chain, test_splitmuxsrc);
+    tcase_add_test (tc_chain, test_splitmuxsrc_format_location);
+
+    if (have_matroska && have_vorbis) {
+      tcase_add_checked_fixture (tc_chain_complex, tempdir_setup,
+          tempdir_cleanup);
+
+      tcase_add_test (tc_chain_complex, test_splitmuxsrc_sparse_streams);
+    } else {
+      GST_INFO ("Skipping tests, missing plugins: matroska and/or vorbis");
+    }
+  } else {
+    GST_INFO ("Skipping tests, missing plugins: theora and/or ogg");
+  }
+
+
+  if (have_qtmux && have_jpeg) {
+    tcase_add_checked_fixture (tc_chain_mp4_jpeg, tempdir_setup,
+        tempdir_cleanup);
+    tcase_add_test (tc_chain_mp4_jpeg, test_splitmuxsrc_caps_change);
+    tcase_add_test (tc_chain_mp4_jpeg, test_splitmuxsrc_robust_mux);
+  } else {
+    GST_INFO ("Skipping tests, missing plugins: jpegenc or mp4mux");
+  }
+
+  return s;
+}
+
+GST_CHECK_MAIN (splitmuxsrc);
index e533d1559e13608bfd79cd64c6ff2bfbf9735b6e..fd5a931c7752bc5639612fafc73ab3c179cf08df 100644 (file)
@@ -53,7 +53,8 @@ good_tests = [
   [ 'elements/matroskamux', false, [gstriff_dep] ],
   [ 'elements/matroskaparse', false, [gstriff_dep] ],
   [ 'elements/multifile' ],
-  [ 'elements/splitmux', ],
+  [ 'elements/splitmuxsink', ],
+  [ 'elements/splitmuxsrc', ],
   [ 'elements/qtmux', false, [gstriff_dep, zlib_dep] ],
   [ 'elements/qtdemux', false, [gstriff_dep, zlib_dep] ],
   [ 'elements/rganalysis' ],