/* GStreamer interactive test for accurate segment seeking * Copyright (C) 2014 Tim-Philipp Müller <tim 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. * */ /* Plays the provided file one second at a time using segment seeks. * Theoretically this should be just as smooth as if we played the * file from start to stop in one go, certainly without hickups. */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #include <gst/gst.h> #define SEGMENT_DURATION (1 * GST_SECOND) int main (int argc, char **argv) { GstElement *playbin; GstMessage *msg; gchar *uri; gint64 dur, start, stop; gboolean prerolled = FALSE; if (argc < 2) { g_printerr ("Usage: %s FILENAME\n", argv[0]); return -1; } gst_init (&argc, &argv); if (gst_uri_is_valid (argv[1])) uri = g_strdup (argv[1]); else uri = gst_filename_to_uri (argv[1], NULL); g_print ("uri: %s\n", uri); playbin = gst_element_factory_make ("playbin", NULL); g_object_set (playbin, "uri", uri, NULL); #if 0 { GstElement *src; playbin = gst_parse_launch ("uridecodebin name=d ! queue ! " "filesink location=/tmp/raw1.data", NULL); src = gst_bin_get_by_name (GST_BIN (playbin), "d"); g_object_set (src, "uri", uri, NULL); gst_object_unref (src); } #endif gst_element_set_state (playbin, GST_STATE_PAUSED); /* wait for preroll */ msg = gst_bus_timed_pop_filtered (GST_ELEMENT_BUS (playbin), GST_CLOCK_TIME_NONE, GST_MESSAGE_ASYNC_DONE | GST_MESSAGE_ERROR); g_assert (GST_MESSAGE_TYPE (msg) != GST_MESSAGE_ERROR); prerolled = TRUE; gst_message_unref (msg); if (!gst_element_query_duration (playbin, GST_FORMAT_TIME, &dur)) g_error ("Failed to query duration!\n"); g_print ("Duration: %" GST_TIME_FORMAT "\n", GST_TIME_ARGS (dur)); start = 0; do { GstSeekFlags seek_flags; gboolean ret; gboolean segment_done = FALSE; seek_flags = GST_SEEK_FLAG_ACCURATE | GST_SEEK_FLAG_SEGMENT; if (start == 0) { prerolled = FALSE; seek_flags |= GST_SEEK_FLAG_FLUSH; } stop = start + SEGMENT_DURATION; g_print ("Segment: %" GST_TIME_FORMAT "-%" GST_TIME_FORMAT "\n", GST_TIME_ARGS (start), GST_TIME_ARGS (stop)); ret = gst_element_seek (playbin, 1.0, GST_FORMAT_TIME, seek_flags, GST_SEEK_TYPE_SET, start, GST_SEEK_TYPE_SET, stop); g_assert (ret); if (!prerolled) { while (!prerolled) { /* wait for preroll again */ msg = gst_bus_timed_pop_filtered (GST_ELEMENT_BUS (playbin), GST_CLOCK_TIME_NONE, GST_MESSAGE_SEGMENT_DONE | GST_MESSAGE_ASYNC_DONE | GST_MESSAGE_ERROR); g_assert (GST_MESSAGE_TYPE (msg) != GST_MESSAGE_ERROR); if (GST_MESSAGE_TYPE (msg) == GST_MESSAGE_SEGMENT_DONE) { segment_done = TRUE; } else { prerolled = TRUE; } gst_message_unref (msg); } gst_element_set_state (playbin, GST_STATE_PLAYING); } /* wait for end of segment if we didn't get it above already */ if (!segment_done) { msg = gst_bus_timed_pop_filtered (GST_ELEMENT_BUS (playbin), GST_CLOCK_TIME_NONE, GST_MESSAGE_SEGMENT_DONE | GST_MESSAGE_ERROR); g_assert (GST_MESSAGE_TYPE (msg) != GST_MESSAGE_ERROR); gst_message_unref (msg); } start = stop; } while (start < dur); gst_element_set_state (playbin, GST_STATE_NULL); gst_object_unref (playbin); return 0; }