/* GStreamer unit test for splitmuxsink elements * * Copyright (C) 2007 David A. Schleef * Copyright (C) 2015 Jan Schmidt * Copyright (C) 2025 Stephane Cerveau * * 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 #include #include #include 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, "dashsink-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, guint num_segments_expected, const GstClockTime * segment_durations) { GstBus *bus = gst_element_get_bus (GST_ELEMENT (pipeline)); GstMessage *msg; guint segments_seen = 0; gst_element_set_state (pipeline, GST_STATE_PLAYING); do { msg = gst_bus_poll (bus, GST_MESSAGE_EOS | GST_MESSAGE_ERROR | GST_MESSAGE_ELEMENT, -1); if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_EOS || GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR) { break; } if (num_segments_expected != 0) { // Handle dashsink element message const GstStructure *s = gst_message_get_structure (msg); if (gst_structure_has_name (s, "dashsink-new-segment")) { GstClockTime segment_duration; guint segment_id; fail_unless (gst_structure_get_uint (s, "segment-id", &segment_id)); fail_unless (segment_id < num_segments_expected); fail_unless (gst_structure_get_clock_time (s, "duration", &segment_duration)); if (segment_durations != NULL) { fail_unless (segment_durations[segment_id] == segment_duration, "Expected duration %" GST_TIME_FORMAT " for fragment %u. Got duration %" GST_TIME_FORMAT, GST_TIME_ARGS (segment_durations[segment_id]), segment_id, GST_TIME_ARGS (segment_duration)); } segments_seen++; } } gst_message_unref (msg); } while (TRUE); gst_element_set_state (pipeline, GST_STATE_NULL); gst_object_unref (bus); if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_ERROR) dump_error (msg); else if (num_segments_expected != 0) { // Success. Check we got the expected number of fragment messages fail_unless (segments_seen == num_segments_expected); } 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) { 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); if (user_data) { guint *num_frame = (guint *) user_data; *num_frame = *num_frame + 1; } return GST_FLOW_OK; } static void test_playback (const gchar * filename, GstClockTime exp_first_time, GstClockTime exp_last_time, guint num_segments_expected, const GstClockTime * segment_durations) { GstMessage *msg; GstElement *pipeline; GstElement *appsink; GstElement *fakesink; GstAppSinkCallbacks callbacks = { NULL }; gchar *uri; GST_DEBUG ("Playing back file %s", filename); pipeline = gst_element_factory_make ("playbin", NULL); fail_if (pipeline == NULL); appsink = gst_element_factory_make ("appsink", NULL); fail_if (appsink == NULL); /*Full speed playback */ g_object_set (G_OBJECT (appsink), "sync", FALSE, NULL); g_object_set (G_OBJECT (pipeline), "video-sink", appsink, NULL); fakesink = gst_element_factory_make ("fakesink", NULL); fail_if (fakesink == NULL); g_object_set (G_OBJECT (pipeline), "audio-sink", fakesink, NULL); uri = g_strdup_printf ("file://%s", filename); 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, 0, NULL); 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_dashsink_video_ts) { GstMessage *msg; GstElement *pipeline; GstElement *sink; guint count; gchar *filename; /* This pipeline has a small time cutoff - it should start a new file * every GOP, ie 1 second */ pipeline = gst_parse_launch ("dashsink name=dashsink videotestsrc num-buffers=15 ! video/x-raw,width=80,height=64,framerate=5/1 ! openh264enc ! dashsink.video_0", NULL); fail_if (pipeline == NULL); sink = gst_bin_get_by_name (GST_BIN (pipeline), "dashsink"); fail_if (sink == NULL); g_object_set (G_OBJECT (sink), "mpd-root-path", tmpdir, NULL); g_object_set (G_OBJECT (sink), "target-duration", 1, NULL); g_object_set (G_OBJECT (sink), "use-segment-list", TRUE, NULL); g_object_unref (sink); GstClockTime durations[] = { GST_SECOND, GST_SECOND, GST_SECOND }; msg = run_pipeline (pipeline, 3, durations); 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 == 4, "Expected 4 output files, got %d", count); filename = g_build_filename (tmpdir, "dash.mpd", NULL); // mpegtsmux generates a first PTS at 0.125 second and does not end at 3 seconds exactly. test_playback (filename, 0.125 * GST_SECOND, 2.925 * GST_SECOND, 3, durations); g_free (filename); } GST_END_TEST; static Suite * dashsink_suite (void) { Suite *s = suite_create ("dashsink"); TCase *tc_chain = tcase_create ("general"); gboolean have_h264; /* we assume that if encoder/muxer are there, decoder/demuxer will be a well */ have_h264 = gst_registry_check_feature_version (gst_registry_get (), "openh264enc", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0); suite_add_tcase (s, tc_chain); if (have_h264) { tcase_add_checked_fixture (tc_chain, tempdir_setup, tempdir_cleanup); tcase_add_test (tc_chain, test_dashsink_video_ts); } else { GST_INFO ("Skipping tests, missing plugins: openh264enc"); } return s; } GST_CHECK_MAIN (dashsink);