2011-02-14 17:51:32 +00:00
|
|
|
/* GStreamer
|
|
|
|
* Copyright (C) 2010 Marc-Andre Lureau <marcandre.lureau@gmail.com>
|
|
|
|
* Copyright (C) 2010 Andoni Morales Alastruey <ylatuya@gmail.com>
|
2011-09-08 22:13:19 +00:00
|
|
|
* Copyright (C) 2011, Hewlett-Packard Development Company, L.P.
|
|
|
|
* Author: Youness Alaoui <youness.alaoui@collabora.co.uk>, Collabora Ltd.
|
|
|
|
* Author: Sebastian Dröge <sebastian.droege@collabora.co.uk>, Collabora Ltd.
|
2011-02-14 17:51:32 +00:00
|
|
|
*
|
|
|
|
* Gsthlsdemux.c:
|
|
|
|
*
|
|
|
|
* 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., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* SECTION:element-hlsdemux
|
|
|
|
*
|
2011-03-30 01:34:39 +00:00
|
|
|
* HTTP Live Streaming demuxer element.
|
2011-02-14 17:51:32 +00:00
|
|
|
*
|
|
|
|
* <refsect2>
|
|
|
|
* <title>Example launch line</title>
|
|
|
|
* |[
|
2011-04-13 20:25:57 +00:00
|
|
|
* gst-launch souphttpsrc location=http://devimages.apple.com/iphone/samples/bipbop/gear4/prog_index.m3u8 ! hlsdemux ! decodebin2 ! ffmpegcolorspace ! videoscale ! autovideosink
|
2011-02-14 17:51:32 +00:00
|
|
|
* ]|
|
|
|
|
* </refsect2>
|
|
|
|
*
|
|
|
|
* Last reviewed on 2010-10-07
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2012-01-27 14:49:58 +00:00
|
|
|
/* FIXME 0.11: suppress warnings for deprecated API such as GStaticRecMutex
|
|
|
|
* with newer GLib versions (>= 2.31.0) */
|
|
|
|
#define GLIB_DISABLE_DEPRECATION_WARNINGS
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
#include <string.h>
|
2011-02-15 20:49:20 +00:00
|
|
|
#include <gst/base/gsttypefindhelper.h>
|
2012-01-27 14:49:58 +00:00
|
|
|
#include <gst/glib-compat-private.h>
|
2011-02-14 17:51:32 +00:00
|
|
|
#include "gsthlsdemux.h"
|
|
|
|
|
2011-11-04 15:34:11 +00:00
|
|
|
static GstStaticPadTemplate srctemplate = GST_STATIC_PAD_TEMPLATE ("src_%u",
|
2011-02-14 17:51:32 +00:00
|
|
|
GST_PAD_SRC,
|
2011-07-26 15:02:05 +00:00
|
|
|
GST_PAD_SOMETIMES,
|
2011-02-15 20:49:20 +00:00
|
|
|
GST_STATIC_CAPS_ANY);
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
|
|
|
|
GST_PAD_SINK,
|
|
|
|
GST_PAD_ALWAYS,
|
2011-03-30 14:53:12 +00:00
|
|
|
GST_STATIC_CAPS ("application/x-hls"));
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (gst_hls_demux_debug);
|
|
|
|
#define GST_CAT_DEFAULT gst_hls_demux_debug
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
|
|
|
|
PROP_FRAGMENTS_CACHE,
|
|
|
|
PROP_BITRATE_SWITCH_TOLERANCE,
|
|
|
|
PROP_LAST
|
|
|
|
};
|
|
|
|
|
|
|
|
static const float update_interval_factor[] = { 1, 0.5, 1.5, 3 };
|
|
|
|
|
|
|
|
#define DEFAULT_FRAGMENTS_CACHE 3
|
|
|
|
#define DEFAULT_FAILED_COUNT 3
|
|
|
|
#define DEFAULT_BITRATE_SWITCH_TOLERANCE 0.4
|
|
|
|
|
|
|
|
/* GObject */
|
|
|
|
static void gst_hls_demux_set_property (GObject * object, guint prop_id,
|
|
|
|
const GValue * value, GParamSpec * pspec);
|
|
|
|
static void gst_hls_demux_get_property (GObject * object, guint prop_id,
|
|
|
|
GValue * value, GParamSpec * pspec);
|
|
|
|
static void gst_hls_demux_dispose (GObject * obj);
|
|
|
|
|
|
|
|
/* GstElement */
|
|
|
|
static GstStateChangeReturn
|
|
|
|
gst_hls_demux_change_state (GstElement * element, GstStateChange transition);
|
|
|
|
|
|
|
|
/* GstHLSDemux */
|
|
|
|
static GstFlowReturn gst_hls_demux_chain (GstPad * pad, GstBuffer * buf);
|
|
|
|
static gboolean gst_hls_demux_sink_event (GstPad * pad, GstEvent * event);
|
2011-04-13 21:06:18 +00:00
|
|
|
static gboolean gst_hls_demux_src_event (GstPad * pad, GstEvent * event);
|
2011-03-12 11:50:25 +00:00
|
|
|
static gboolean gst_hls_demux_src_query (GstPad * pad, GstQuery * query);
|
2012-03-15 22:21:58 +00:00
|
|
|
static void gst_hls_demux_stream_loop (GstHLSDemux * demux);
|
|
|
|
static void gst_hls_demux_updates_loop (GstHLSDemux * demux);
|
2011-02-14 17:51:32 +00:00
|
|
|
static void gst_hls_demux_stop (GstHLSDemux * demux);
|
|
|
|
static gboolean gst_hls_demux_cache_fragments (GstHLSDemux * demux);
|
|
|
|
static gboolean gst_hls_demux_schedule (GstHLSDemux * demux);
|
|
|
|
static gboolean gst_hls_demux_switch_playlist (GstHLSDemux * demux);
|
2012-03-15 18:42:44 +00:00
|
|
|
static gboolean gst_hls_demux_get_next_fragment (GstHLSDemux * demux,
|
|
|
|
gboolean caching);
|
2011-09-01 22:46:19 +00:00
|
|
|
static gboolean gst_hls_demux_update_playlist (GstHLSDemux * demux);
|
2011-02-15 20:55:26 +00:00
|
|
|
static void gst_hls_demux_reset (GstHLSDemux * demux, gboolean dispose);
|
2011-03-30 01:34:39 +00:00
|
|
|
static gboolean gst_hls_demux_set_location (GstHLSDemux * demux,
|
|
|
|
const gchar * uri);
|
2012-03-15 18:42:44 +00:00
|
|
|
static gchar *gst_hls_src_buf_to_utf8_playlist (GstBuffer * buf);
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
_do_init (GType type)
|
|
|
|
{
|
2011-03-30 01:34:39 +00:00
|
|
|
GST_DEBUG_CATEGORY_INIT (gst_hls_demux_debug, "hlsdemux", 0,
|
|
|
|
"hlsdemux element");
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GST_BOILERPLATE_FULL (GstHLSDemux, gst_hls_demux, GstElement,
|
|
|
|
GST_TYPE_ELEMENT, _do_init);
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_base_init (gpointer g_class)
|
|
|
|
{
|
|
|
|
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
|
|
|
|
|
|
|
|
gst_element_class_add_pad_template (element_class,
|
|
|
|
gst_static_pad_template_get (&srctemplate));
|
|
|
|
|
|
|
|
gst_element_class_add_pad_template (element_class,
|
|
|
|
gst_static_pad_template_get (&sinktemplate));
|
|
|
|
|
|
|
|
gst_element_class_set_details_simple (element_class,
|
2011-03-30 01:34:39 +00:00
|
|
|
"HLS Demuxer",
|
2011-02-15 20:49:20 +00:00
|
|
|
"Demuxer/URIList",
|
2011-03-30 01:34:39 +00:00
|
|
|
"HTTP Live Streaming demuxer",
|
2011-02-14 17:51:32 +00:00
|
|
|
"Marc-Andre Lureau <marcandre.lureau@gmail.com>\n"
|
|
|
|
"Andoni Morales Alastruey <ylatuya@gmail.com>");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_dispose (GObject * obj)
|
|
|
|
{
|
|
|
|
GstHLSDemux *demux = GST_HLS_DEMUX (obj);
|
|
|
|
|
2012-03-15 22:21:58 +00:00
|
|
|
if (demux->stream_task) {
|
|
|
|
if (GST_TASK_STATE (demux->stream_task) != GST_TASK_STOPPED) {
|
|
|
|
GST_DEBUG_OBJECT (demux, "Leaving streaming task");
|
|
|
|
gst_task_stop (demux->stream_task);
|
|
|
|
gst_task_join (demux->stream_task);
|
|
|
|
}
|
|
|
|
gst_object_unref (demux->stream_task);
|
|
|
|
g_static_rec_mutex_free (&demux->stream_lock);
|
|
|
|
demux->stream_task = NULL;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2012-03-15 22:21:58 +00:00
|
|
|
if (demux->updates_task) {
|
|
|
|
if (GST_TASK_STATE (demux->updates_task) != GST_TASK_STOPPED) {
|
|
|
|
GST_DEBUG_OBJECT (demux, "Leaving updates task");
|
|
|
|
gst_task_stop (demux->updates_task);
|
|
|
|
gst_task_join (demux->updates_task);
|
|
|
|
}
|
|
|
|
gst_object_unref (demux->updates_task);
|
|
|
|
g_mutex_free (demux->updates_timed_lock);
|
|
|
|
g_static_rec_mutex_free (&demux->updates_lock);
|
|
|
|
demux->updates_task = NULL;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
if (demux->downloader != NULL) {
|
|
|
|
g_object_unref (demux->downloader);
|
|
|
|
demux->downloader = NULL;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-02-15 20:55:26 +00:00
|
|
|
gst_hls_demux_reset (demux, TRUE);
|
|
|
|
|
2012-01-12 15:58:36 +00:00
|
|
|
g_queue_free (demux->queue);
|
2011-04-01 23:08:02 +00:00
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_class_init (GstHLSDemuxClass * klass)
|
|
|
|
{
|
|
|
|
GObjectClass *gobject_class;
|
|
|
|
GstElementClass *gstelement_class;
|
|
|
|
|
2011-03-30 08:11:24 +00:00
|
|
|
gobject_class = (GObjectClass *) klass;
|
2011-02-14 17:51:32 +00:00
|
|
|
gstelement_class = (GstElementClass *) klass;
|
|
|
|
|
|
|
|
gobject_class->set_property = gst_hls_demux_set_property;
|
|
|
|
gobject_class->get_property = gst_hls_demux_get_property;
|
|
|
|
gobject_class->dispose = gst_hls_demux_dispose;
|
|
|
|
|
|
|
|
g_object_class_install_property (gobject_class, PROP_FRAGMENTS_CACHE,
|
|
|
|
g_param_spec_uint ("fragments-cache", "Fragments cache",
|
|
|
|
"Number of fragments needed to be cached to start playing",
|
2011-03-30 08:11:24 +00:00
|
|
|
2, G_MAXUINT, DEFAULT_FRAGMENTS_CACHE,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
g_object_class_install_property (gobject_class, PROP_BITRATE_SWITCH_TOLERANCE,
|
|
|
|
g_param_spec_float ("bitrate-switch-tolerance",
|
|
|
|
"Bitrate switch tolerance",
|
|
|
|
"Tolerance with respect of the fragment duration to switch to "
|
|
|
|
"a different bitrate if the client is too slow/fast.",
|
2011-03-30 08:11:24 +00:00
|
|
|
0, 1, DEFAULT_BITRATE_SWITCH_TOLERANCE,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-03-30 08:11:24 +00:00
|
|
|
gstelement_class->change_state =
|
|
|
|
GST_DEBUG_FUNCPTR (gst_hls_demux_change_state);
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_init (GstHLSDemux * demux, GstHLSDemuxClass * klass)
|
|
|
|
{
|
|
|
|
/* sink pad */
|
|
|
|
demux->sinkpad = gst_pad_new_from_static_template (&sinktemplate, "sink");
|
|
|
|
gst_pad_set_chain_function (demux->sinkpad,
|
|
|
|
GST_DEBUG_FUNCPTR (gst_hls_demux_chain));
|
|
|
|
gst_pad_set_event_function (demux->sinkpad,
|
|
|
|
GST_DEBUG_FUNCPTR (gst_hls_demux_sink_event));
|
|
|
|
gst_element_add_pad (GST_ELEMENT (demux), demux->sinkpad);
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
/* Downloader */
|
|
|
|
demux->downloader = gst_uri_downloader_new ();
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-07-26 15:02:05 +00:00
|
|
|
demux->do_typefind = TRUE;
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
/* Properties */
|
|
|
|
demux->fragments_cache = DEFAULT_FRAGMENTS_CACHE;
|
|
|
|
demux->bitrate_switch_tol = DEFAULT_BITRATE_SWITCH_TOLERANCE;
|
|
|
|
|
|
|
|
demux->queue = g_queue_new ();
|
2012-03-15 22:21:58 +00:00
|
|
|
|
|
|
|
/* Updates task */
|
|
|
|
g_static_rec_mutex_init (&demux->updates_lock);
|
|
|
|
demux->updates_task =
|
|
|
|
gst_task_create ((GstTaskFunction) gst_hls_demux_updates_loop, demux);
|
|
|
|
gst_task_set_lock (demux->updates_task, &demux->updates_lock);
|
|
|
|
demux->updates_timed_lock = g_mutex_new ();
|
|
|
|
|
|
|
|
/* Streaming task */
|
|
|
|
g_static_rec_mutex_init (&demux->stream_lock);
|
|
|
|
demux->stream_task =
|
|
|
|
gst_task_create ((GstTaskFunction) gst_hls_demux_stream_loop, demux);
|
|
|
|
gst_task_set_lock (demux->stream_task, &demux->stream_lock);
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-03-30 01:34:39 +00:00
|
|
|
gst_hls_demux_set_property (GObject * object, guint prop_id,
|
|
|
|
const GValue * value, GParamSpec * pspec)
|
2011-02-14 17:51:32 +00:00
|
|
|
{
|
|
|
|
GstHLSDemux *demux = GST_HLS_DEMUX (object);
|
|
|
|
|
|
|
|
switch (prop_id) {
|
|
|
|
case PROP_FRAGMENTS_CACHE:
|
|
|
|
demux->fragments_cache = g_value_get_uint (value);
|
|
|
|
break;
|
|
|
|
case PROP_BITRATE_SWITCH_TOLERANCE:
|
|
|
|
demux->bitrate_switch_tol = g_value_get_float (value);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_get_property (GObject * object, guint prop_id, GValue * value,
|
|
|
|
GParamSpec * pspec)
|
|
|
|
{
|
|
|
|
GstHLSDemux *demux = GST_HLS_DEMUX (object);
|
|
|
|
|
|
|
|
switch (prop_id) {
|
|
|
|
case PROP_FRAGMENTS_CACHE:
|
|
|
|
g_value_set_uint (value, demux->fragments_cache);
|
|
|
|
break;
|
|
|
|
case PROP_BITRATE_SWITCH_TOLERANCE:
|
|
|
|
g_value_set_float (value, demux->bitrate_switch_tol);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static GstStateChangeReturn
|
|
|
|
gst_hls_demux_change_state (GstElement * element, GstStateChange transition)
|
|
|
|
{
|
|
|
|
GstStateChangeReturn ret;
|
|
|
|
GstHLSDemux *demux = GST_HLS_DEMUX (element);
|
|
|
|
|
|
|
|
switch (transition) {
|
2011-09-02 12:55:45 +00:00
|
|
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
2011-02-15 20:55:26 +00:00
|
|
|
gst_hls_demux_reset (demux, FALSE);
|
2011-02-14 17:51:32 +00:00
|
|
|
break;
|
2011-08-31 03:07:48 +00:00
|
|
|
case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
|
|
|
|
/* Start the streaming loop in paused only if we already received
|
|
|
|
the main playlist. It might have been stopped if we were in PAUSED
|
|
|
|
state and we filled our queue with enough cached fragments
|
|
|
|
*/
|
|
|
|
if (gst_m3u8_client_get_uri (demux->client)[0] != '\0')
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_start (demux->updates_task);
|
2011-08-31 03:07:48 +00:00
|
|
|
break;
|
2011-02-14 17:51:32 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
2011-02-15 02:41:43 +00:00
|
|
|
|
|
|
|
switch (transition) {
|
2011-08-31 03:07:48 +00:00
|
|
|
case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_stop (demux->updates_task);
|
2011-08-31 03:07:48 +00:00
|
|
|
break;
|
2011-02-15 02:41:43 +00:00
|
|
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
2011-02-15 21:40:21 +00:00
|
|
|
demux->cancelled = TRUE;
|
2011-08-23 20:49:33 +00:00
|
|
|
gst_hls_demux_stop (demux);
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_join (demux->stream_task);
|
2011-09-02 17:44:31 +00:00
|
|
|
gst_hls_demux_reset (demux, FALSE);
|
2011-02-15 02:41:43 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-04-13 21:06:18 +00:00
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_src_event (GstPad * pad, GstEvent * event)
|
|
|
|
{
|
2011-08-16 19:46:49 +00:00
|
|
|
GstHLSDemux *demux;
|
|
|
|
|
|
|
|
demux = GST_HLS_DEMUX (gst_pad_get_element_private (pad));
|
|
|
|
|
2011-04-13 21:06:18 +00:00
|
|
|
switch (event->type) {
|
|
|
|
case GST_EVENT_SEEK:
|
2011-08-16 19:46:49 +00:00
|
|
|
{
|
|
|
|
gdouble rate;
|
|
|
|
GstFormat format;
|
|
|
|
GstSeekFlags flags;
|
|
|
|
GstSeekType start_type, stop_type;
|
|
|
|
gint64 start, stop;
|
|
|
|
GList *walk;
|
2012-02-28 15:40:31 +00:00
|
|
|
GstClockTime current_pos, target_pos;
|
2011-08-16 19:46:49 +00:00
|
|
|
gint current_sequence;
|
|
|
|
GstM3U8MediaFile *file;
|
|
|
|
|
|
|
|
GST_INFO_OBJECT (demux, "Received GST_EVENT_SEEK");
|
2011-08-22 23:41:39 +00:00
|
|
|
|
|
|
|
if (gst_m3u8_client_is_live (demux->client)) {
|
|
|
|
GST_WARNING_OBJECT (demux, "Received seek event for live stream");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2011-08-16 19:46:49 +00:00
|
|
|
gst_event_parse_seek (event, &rate, &format, &flags, &start_type, &start,
|
|
|
|
&stop_type, &stop);
|
|
|
|
|
|
|
|
if (format != GST_FORMAT_TIME)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (demux, "seek event, rate: %f start: %" GST_TIME_FORMAT
|
|
|
|
" stop: %" GST_TIME_FORMAT, rate, GST_TIME_ARGS (start),
|
|
|
|
GST_TIME_ARGS (stop));
|
|
|
|
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_LOCK (demux->client);
|
2011-08-16 19:46:49 +00:00
|
|
|
file = GST_M3U8_MEDIA_FILE (demux->client->current->files->data);
|
|
|
|
current_sequence = file->sequence;
|
|
|
|
current_pos = 0;
|
2012-02-28 15:40:31 +00:00
|
|
|
target_pos = (GstClockTime) start;
|
2011-08-16 19:46:49 +00:00
|
|
|
for (walk = demux->client->current->files; walk; walk = walk->next) {
|
|
|
|
file = walk->data;
|
|
|
|
|
|
|
|
current_sequence = file->sequence;
|
2012-02-28 15:40:31 +00:00
|
|
|
if (current_pos <= target_pos
|
|
|
|
&& target_pos < current_pos + file->duration) {
|
2011-08-16 19:46:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
current_pos += file->duration;
|
|
|
|
}
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
2011-08-16 19:46:49 +00:00
|
|
|
|
|
|
|
if (walk == NULL) {
|
|
|
|
GST_WARNING_OBJECT (demux, "Could not find seeked fragment");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & GST_SEEK_FLAG_FLUSH) {
|
|
|
|
GST_DEBUG_OBJECT (demux, "sending flush start");
|
|
|
|
gst_pad_push_event (demux->srcpad, gst_event_new_flush_start ());
|
|
|
|
}
|
|
|
|
|
2011-08-23 20:49:33 +00:00
|
|
|
demux->cancelled = TRUE;
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_pause (demux->stream_task);
|
2012-03-15 18:42:44 +00:00
|
|
|
gst_uri_downloader_cancel (demux->downloader);
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_stop (demux->updates_task);
|
|
|
|
gst_task_pause (demux->stream_task);
|
2011-08-16 19:46:49 +00:00
|
|
|
|
|
|
|
/* wait for streaming to finish */
|
2012-03-15 22:21:58 +00:00
|
|
|
g_static_rec_mutex_lock (&demux->stream_lock);
|
2011-08-16 19:46:49 +00:00
|
|
|
|
|
|
|
demux->need_cache = TRUE;
|
|
|
|
while (!g_queue_is_empty (demux->queue)) {
|
2012-03-15 18:42:44 +00:00
|
|
|
GstBufferList *buf_list = g_queue_pop_head (demux->queue);
|
|
|
|
gst_buffer_list_unref (buf_list);
|
2011-08-16 19:46:49 +00:00
|
|
|
}
|
2011-08-26 22:47:35 +00:00
|
|
|
g_queue_clear (demux->queue);
|
2011-08-16 19:46:49 +00:00
|
|
|
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_LOCK (demux->client);
|
2011-08-16 19:46:49 +00:00
|
|
|
GST_DEBUG_OBJECT (demux, "seeking to sequence %d", current_sequence);
|
|
|
|
demux->client->sequence = current_sequence;
|
2011-09-08 23:56:33 +00:00
|
|
|
gst_m3u8_client_get_current_position (demux->client, &demux->position);
|
|
|
|
demux->position_shift = start - demux->position;
|
2011-08-16 19:46:49 +00:00
|
|
|
demux->need_segment = TRUE;
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
|
|
|
|
2011-08-16 19:46:49 +00:00
|
|
|
|
|
|
|
if (flags & GST_SEEK_FLAG_FLUSH) {
|
|
|
|
GST_DEBUG_OBJECT (demux, "sending flush stop");
|
|
|
|
gst_pad_push_event (demux->srcpad, gst_event_new_flush_stop ());
|
|
|
|
}
|
|
|
|
|
2011-08-23 20:49:33 +00:00
|
|
|
demux->cancelled = FALSE;
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_start (demux->stream_task);
|
|
|
|
g_static_rec_mutex_unlock (&demux->stream_lock);
|
2011-08-16 19:46:49 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
2011-04-13 21:06:18 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return gst_pad_event_default (pad, event);
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_sink_event (GstPad * pad, GstEvent * event)
|
|
|
|
{
|
|
|
|
GstHLSDemux *demux = GST_HLS_DEMUX (gst_pad_get_parent (pad));
|
|
|
|
GstQuery *query;
|
2011-02-15 01:13:56 +00:00
|
|
|
gboolean ret;
|
2011-02-14 17:51:32 +00:00
|
|
|
gchar *uri;
|
|
|
|
|
|
|
|
|
|
|
|
switch (event->type) {
|
|
|
|
case GST_EVENT_EOS:{
|
|
|
|
gchar *playlist;
|
|
|
|
|
|
|
|
if (demux->playlist == NULL) {
|
|
|
|
GST_WARNING_OBJECT (demux, "Received EOS without a playlist.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-03-30 01:34:39 +00:00
|
|
|
GST_DEBUG_OBJECT (demux,
|
|
|
|
"Got EOS on the sink pad: main playlist fetched");
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-02-15 01:13:56 +00:00
|
|
|
query = gst_query_new_uri ();
|
|
|
|
ret = gst_pad_peer_query (demux->sinkpad, query);
|
|
|
|
if (ret) {
|
|
|
|
gst_query_parse_uri (query, &uri);
|
|
|
|
gst_hls_demux_set_location (demux, uri);
|
|
|
|
g_free (uri);
|
|
|
|
}
|
2011-03-29 21:18:24 +00:00
|
|
|
gst_query_unref (query);
|
2011-02-15 01:13:56 +00:00
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
playlist = gst_hls_src_buf_to_utf8_playlist (demux->playlist);
|
2011-09-08 20:29:42 +00:00
|
|
|
demux->playlist = NULL;
|
2011-04-01 23:21:34 +00:00
|
|
|
if (playlist == NULL) {
|
|
|
|
GST_WARNING_OBJECT (demux, "Error validating first playlist.");
|
|
|
|
} else if (!gst_m3u8_client_update (demux->client, playlist)) {
|
2011-02-15 03:39:34 +00:00
|
|
|
/* In most cases, this will happen if we set a wrong url in the
|
2011-02-15 02:42:29 +00:00
|
|
|
* source element and we have received the 404 HTML response instead of
|
|
|
|
* the playlist */
|
2011-07-27 10:02:41 +00:00
|
|
|
GST_ELEMENT_ERROR (demux, STREAM, DECODE, ("Invalid playlist."),
|
|
|
|
(NULL));
|
2011-02-15 02:42:29 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-02-15 01:13:56 +00:00
|
|
|
if (!ret && gst_m3u8_client_is_live (demux->client)) {
|
2011-02-14 17:51:32 +00:00
|
|
|
GST_ELEMENT_ERROR (demux, RESOURCE, NOT_FOUND,
|
|
|
|
("Failed querying the playlist uri, "
|
2011-07-27 10:02:41 +00:00
|
|
|
"required for live sources."), (NULL));
|
2011-02-14 17:51:32 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_start (demux->stream_task);
|
2011-02-14 17:51:32 +00:00
|
|
|
gst_event_unref (event);
|
|
|
|
return TRUE;
|
|
|
|
}
|
2011-08-01 16:48:03 +00:00
|
|
|
case GST_EVENT_NEWSEGMENT:
|
|
|
|
/* Swallow newsegments, we'll push our own */
|
|
|
|
gst_event_unref (event);
|
|
|
|
return TRUE;
|
2011-02-14 17:51:32 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return gst_pad_event_default (pad, event);
|
|
|
|
}
|
|
|
|
|
2011-03-12 11:50:25 +00:00
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_src_query (GstPad * pad, GstQuery * query)
|
|
|
|
{
|
|
|
|
GstHLSDemux *hlsdemux;
|
|
|
|
gboolean ret = FALSE;
|
|
|
|
|
|
|
|
if (query == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
hlsdemux = GST_HLS_DEMUX (gst_pad_get_element_private (pad));
|
|
|
|
|
|
|
|
switch (query->type) {
|
|
|
|
case GST_QUERY_DURATION:{
|
2011-08-23 01:54:03 +00:00
|
|
|
GstClockTime duration = -1;
|
2011-05-03 11:01:25 +00:00
|
|
|
GstFormat fmt;
|
2011-03-12 11:50:25 +00:00
|
|
|
|
2011-05-03 11:01:25 +00:00
|
|
|
gst_query_parse_duration (query, &fmt, NULL);
|
|
|
|
if (fmt == GST_FORMAT_TIME) {
|
|
|
|
duration = gst_m3u8_client_get_duration (hlsdemux->client);
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (duration) && duration > 0) {
|
|
|
|
gst_query_set_duration (query, GST_FORMAT_TIME, duration);
|
|
|
|
ret = TRUE;
|
|
|
|
}
|
2011-03-12 11:50:25 +00:00
|
|
|
}
|
2011-08-23 01:54:03 +00:00
|
|
|
GST_INFO_OBJECT (hlsdemux, "GST_QUERY_DURATION returns %s with duration %"
|
|
|
|
GST_TIME_FORMAT, ret ? "TRUE" : "FALSE", GST_TIME_ARGS (duration));
|
2011-03-12 11:50:25 +00:00
|
|
|
break;
|
|
|
|
}
|
2011-03-12 12:00:06 +00:00
|
|
|
case GST_QUERY_URI:
|
|
|
|
if (hlsdemux->client) {
|
|
|
|
/* FIXME: Do we answer with the variant playlist, with the current
|
|
|
|
* playlist or the the uri of the least downlowaded fragment? */
|
2011-09-02 23:49:38 +00:00
|
|
|
gst_query_set_uri (query, gst_m3u8_client_get_uri (hlsdemux->client));
|
2011-03-12 12:00:06 +00:00
|
|
|
ret = TRUE;
|
|
|
|
}
|
|
|
|
break;
|
2011-05-03 10:51:44 +00:00
|
|
|
case GST_QUERY_SEEKING:{
|
|
|
|
GstFormat fmt;
|
2011-08-23 01:54:26 +00:00
|
|
|
gint64 stop = -1;
|
2011-05-03 10:51:44 +00:00
|
|
|
|
|
|
|
gst_query_parse_seeking (query, &fmt, NULL, NULL, NULL);
|
2011-08-23 01:54:26 +00:00
|
|
|
GST_INFO_OBJECT (hlsdemux, "Received GST_QUERY_SEEKING with format %d",
|
|
|
|
fmt);
|
2011-05-03 10:51:44 +00:00
|
|
|
if (fmt == GST_FORMAT_TIME) {
|
|
|
|
GstClockTime duration;
|
|
|
|
|
|
|
|
duration = gst_m3u8_client_get_duration (hlsdemux->client);
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (duration) && duration > 0)
|
|
|
|
stop = duration;
|
2011-08-23 01:54:26 +00:00
|
|
|
|
|
|
|
gst_query_set_seeking (query, fmt,
|
|
|
|
!gst_m3u8_client_is_live (hlsdemux->client), 0, stop);
|
|
|
|
ret = TRUE;
|
|
|
|
GST_INFO_OBJECT (hlsdemux, "GST_QUERY_SEEKING returning with stop : %"
|
|
|
|
GST_TIME_FORMAT, GST_TIME_ARGS (stop));
|
2011-05-03 10:51:44 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2011-03-12 11:50:25 +00:00
|
|
|
default:
|
|
|
|
/* Don't fordward queries upstream because of the special nature of this
|
2011-05-03 10:51:44 +00:00
|
|
|
* "demuxer", which relies on the upstream element only to be fed with the
|
2011-03-12 11:50:25 +00:00
|
|
|
* first playlist */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
static GstFlowReturn
|
|
|
|
gst_hls_demux_chain (GstPad * pad, GstBuffer * buf)
|
|
|
|
{
|
|
|
|
GstHLSDemux *demux = GST_HLS_DEMUX (gst_pad_get_parent (pad));
|
|
|
|
|
2011-02-16 00:19:45 +00:00
|
|
|
if (demux->playlist == NULL)
|
2011-02-14 17:51:32 +00:00
|
|
|
demux->playlist = buf;
|
2011-02-16 00:19:45 +00:00
|
|
|
else
|
2012-03-28 10:49:54 +00:00
|
|
|
demux->playlist = gst_buffer_append (demux->playlist, buf);
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
gst_object_unref (demux);
|
|
|
|
|
|
|
|
return GST_FLOW_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_stop (GstHLSDemux * demux)
|
|
|
|
{
|
2012-03-15 18:42:44 +00:00
|
|
|
gst_uri_downloader_cancel (demux->downloader);
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2012-03-15 22:21:58 +00:00
|
|
|
if (GST_TASK_STATE (demux->updates_task) != GST_TASK_STOPPED) {
|
|
|
|
demux->stop_stream_task = TRUE;
|
|
|
|
gst_task_stop (demux->updates_task);
|
|
|
|
GST_TASK_SIGNAL (demux->updates_task);
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
2011-02-15 03:39:34 +00:00
|
|
|
|
2012-03-15 22:21:58 +00:00
|
|
|
if (GST_TASK_STATE (demux->stream_task) != GST_TASK_STOPPED)
|
|
|
|
gst_task_stop (demux->stream_task);
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
2011-07-26 15:02:05 +00:00
|
|
|
static void
|
|
|
|
switch_pads (GstHLSDemux * demux, GstCaps * newcaps)
|
|
|
|
{
|
|
|
|
GstPad *oldpad = demux->srcpad;
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
GST_DEBUG ("Switching pads (oldpad:%p) with caps: %" GST_PTR_FORMAT, oldpad,
|
|
|
|
newcaps);
|
2011-07-26 15:02:05 +00:00
|
|
|
|
2011-08-31 01:08:39 +00:00
|
|
|
/* FIXME: This is a workaround for a bug in playsink.
|
|
|
|
* If we're switching from an audio-only or video-only fragment
|
|
|
|
* to an audio-video segment, the new sink doesn't know about
|
|
|
|
* the current running time and audio/video will go out of sync.
|
|
|
|
*
|
|
|
|
* This should be fixed in playsink by distributing the
|
|
|
|
* current running time to newly created sinks and is
|
|
|
|
* fixed in 0.11 with the new segments.
|
|
|
|
*/
|
|
|
|
if (demux->srcpad)
|
|
|
|
gst_pad_push_event (demux->srcpad, gst_event_new_flush_stop ());
|
|
|
|
|
2011-07-26 15:02:05 +00:00
|
|
|
/* First create and activate new pad */
|
|
|
|
demux->srcpad = gst_pad_new_from_static_template (&srctemplate, NULL);
|
|
|
|
gst_pad_set_event_function (demux->srcpad,
|
|
|
|
GST_DEBUG_FUNCPTR (gst_hls_demux_src_event));
|
|
|
|
gst_pad_set_query_function (demux->srcpad,
|
|
|
|
GST_DEBUG_FUNCPTR (gst_hls_demux_src_query));
|
|
|
|
gst_pad_set_element_private (demux->srcpad, demux);
|
|
|
|
gst_pad_set_active (demux->srcpad, TRUE);
|
|
|
|
gst_pad_set_caps (demux->srcpad, newcaps);
|
2011-08-02 19:21:18 +00:00
|
|
|
gst_element_add_pad (GST_ELEMENT (demux), demux->srcpad);
|
2011-07-26 15:02:05 +00:00
|
|
|
|
|
|
|
gst_element_no_more_pads (GST_ELEMENT (demux));
|
|
|
|
|
|
|
|
if (oldpad) {
|
|
|
|
/* Push out EOS */
|
|
|
|
gst_pad_push_event (oldpad, gst_event_new_eos ());
|
|
|
|
gst_pad_set_active (oldpad, FALSE);
|
|
|
|
gst_element_remove_pad (GST_ELEMENT (demux), oldpad);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
static void
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_hls_demux_stream_loop (GstHLSDemux * demux)
|
2011-02-14 17:51:32 +00:00
|
|
|
{
|
2012-03-15 18:42:44 +00:00
|
|
|
GstBufferList *buffer_list;
|
2011-02-14 17:51:32 +00:00
|
|
|
GstBuffer *buf;
|
|
|
|
GstFlowReturn ret;
|
|
|
|
|
2011-02-15 03:39:34 +00:00
|
|
|
/* Loop for the source pad task. The task is started when we have
|
|
|
|
* received the main playlist from the source element. It tries first to
|
|
|
|
* cache the first fragments and then it waits until it has more data in the
|
|
|
|
* queue. This task is woken up when we push a new fragment to the queue or
|
|
|
|
* when we reached the end of the playlist */
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
if (G_UNLIKELY (demux->need_cache)) {
|
2011-02-15 23:55:30 +00:00
|
|
|
if (!gst_hls_demux_cache_fragments (demux))
|
2011-02-14 17:51:32 +00:00
|
|
|
goto cache_error;
|
2011-02-15 23:55:30 +00:00
|
|
|
|
2011-08-31 03:07:48 +00:00
|
|
|
/* we can start now the updates thread (only if on playing) */
|
|
|
|
if (GST_STATE (demux) == GST_STATE_PLAYING)
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_start (demux->updates_task);
|
2011-02-14 17:51:32 +00:00
|
|
|
GST_INFO_OBJECT (demux, "First fragments cached successfully");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (g_queue_is_empty (demux->queue)) {
|
2011-04-13 21:35:50 +00:00
|
|
|
if (demux->end_of_playlist)
|
2011-02-14 17:51:32 +00:00
|
|
|
goto end_of_playlist;
|
|
|
|
|
2011-08-31 03:07:48 +00:00
|
|
|
goto pause_task;
|
2011-03-29 21:06:14 +00:00
|
|
|
}
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
buffer_list = g_queue_pop_head (demux->queue);
|
|
|
|
/* Work with the first buffer of the list */
|
|
|
|
buf = gst_buffer_list_get (buffer_list, 0, 0);
|
2011-07-26 15:02:05 +00:00
|
|
|
/* Figure out if we need to create/switch pads */
|
|
|
|
if (G_UNLIKELY (!demux->srcpad
|
2011-08-16 19:53:31 +00:00
|
|
|
|| GST_BUFFER_CAPS (buf) != GST_PAD_CAPS (demux->srcpad)
|
|
|
|
|| demux->need_segment)) {
|
2011-07-26 15:02:05 +00:00
|
|
|
switch_pads (demux, GST_BUFFER_CAPS (buf));
|
2011-08-16 19:53:31 +00:00
|
|
|
demux->need_segment = TRUE;
|
|
|
|
}
|
|
|
|
if (demux->need_segment) {
|
2011-09-08 23:56:33 +00:00
|
|
|
GstClockTime start = demux->position + demux->position_shift;
|
2011-08-01 16:48:03 +00:00
|
|
|
/* And send a newsegment */
|
2011-09-08 23:56:33 +00:00
|
|
|
GST_DEBUG_OBJECT (demux, "Sending new-segment. segment start:%"
|
|
|
|
GST_TIME_FORMAT, GST_TIME_ARGS (start));
|
2011-08-01 16:48:03 +00:00
|
|
|
gst_pad_push_event (demux->srcpad,
|
2011-09-08 23:56:33 +00:00
|
|
|
gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_TIME,
|
|
|
|
start, GST_CLOCK_TIME_NONE, start));
|
2011-08-16 19:53:31 +00:00
|
|
|
demux->need_segment = FALSE;
|
2011-09-08 23:56:33 +00:00
|
|
|
demux->position_shift = 0;
|
2011-07-26 15:02:05 +00:00
|
|
|
}
|
|
|
|
|
2011-08-01 16:48:03 +00:00
|
|
|
if (GST_CLOCK_TIME_IS_VALID (GST_BUFFER_DURATION (buf)))
|
|
|
|
demux->position += GST_BUFFER_DURATION (buf);
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
ret = gst_pad_push_list (demux->srcpad, buffer_list);
|
2011-02-14 17:51:32 +00:00
|
|
|
if (ret != GST_FLOW_OK)
|
2012-03-15 18:42:44 +00:00
|
|
|
goto error_pushing;
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
end_of_playlist:
|
|
|
|
{
|
|
|
|
GST_DEBUG_OBJECT (demux, "Reached end of playlist, sending EOS");
|
|
|
|
gst_pad_push_event (demux->srcpad, gst_event_new_eos ());
|
|
|
|
gst_hls_demux_stop (demux);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
cache_error:
|
|
|
|
{
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_pause (demux->stream_task);
|
2011-08-25 23:37:25 +00:00
|
|
|
if (!demux->cancelled) {
|
|
|
|
GST_ELEMENT_ERROR (demux, RESOURCE, NOT_FOUND,
|
|
|
|
("Could not cache the first fragments"), (NULL));
|
|
|
|
gst_hls_demux_stop (demux);
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
error_pushing:
|
2011-02-14 17:51:32 +00:00
|
|
|
{
|
|
|
|
/* FIXME: handle error */
|
2012-03-15 18:42:44 +00:00
|
|
|
GST_DEBUG_OBJECT (demux, "Error pushing buffer: %s... stopping task",
|
|
|
|
gst_flow_get_name (ret));
|
2011-02-14 17:51:32 +00:00
|
|
|
gst_hls_demux_stop (demux);
|
|
|
|
return;
|
|
|
|
}
|
2011-08-16 19:41:08 +00:00
|
|
|
|
2011-08-31 03:07:48 +00:00
|
|
|
pause_task:
|
2011-08-16 19:41:08 +00:00
|
|
|
{
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_pause (demux->stream_task);
|
2011-08-16 19:41:08 +00:00
|
|
|
return;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-02-15 20:55:26 +00:00
|
|
|
gst_hls_demux_reset (GstHLSDemux * demux, gboolean dispose)
|
2011-02-14 17:51:32 +00:00
|
|
|
{
|
|
|
|
demux->need_cache = TRUE;
|
|
|
|
demux->accumulated_delay = 0;
|
|
|
|
demux->end_of_playlist = FALSE;
|
2011-02-15 21:40:21 +00:00
|
|
|
demux->cancelled = FALSE;
|
2011-07-26 15:02:05 +00:00
|
|
|
demux->do_typefind = TRUE;
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-02-15 20:49:20 +00:00
|
|
|
if (demux->input_caps) {
|
|
|
|
gst_caps_unref (demux->input_caps);
|
|
|
|
demux->input_caps = NULL;
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
if (demux->playlist) {
|
|
|
|
gst_buffer_unref (demux->playlist);
|
|
|
|
demux->playlist = NULL;
|
|
|
|
}
|
|
|
|
|
2011-09-02 12:54:56 +00:00
|
|
|
if (demux->client) {
|
2011-02-14 17:51:32 +00:00
|
|
|
gst_m3u8_client_free (demux->client);
|
2011-09-02 12:54:56 +00:00
|
|
|
demux->client = NULL;
|
|
|
|
}
|
2011-02-15 20:55:26 +00:00
|
|
|
|
|
|
|
if (!dispose) {
|
|
|
|
demux->client = gst_m3u8_client_new ("");
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
while (!g_queue_is_empty (demux->queue)) {
|
2012-03-15 18:42:44 +00:00
|
|
|
GstBufferList *buffer_list = g_queue_pop_head (demux->queue);
|
|
|
|
gst_buffer_list_unref (buffer_list);
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
g_queue_clear (demux->queue);
|
2011-08-16 19:46:49 +00:00
|
|
|
|
|
|
|
demux->position = 0;
|
2011-09-08 23:56:33 +00:00
|
|
|
demux->position_shift = 0;
|
2011-08-16 19:46:49 +00:00
|
|
|
demux->need_segment = TRUE;
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_set_location (GstHLSDemux * demux, const gchar * uri)
|
|
|
|
{
|
|
|
|
if (demux->client)
|
|
|
|
gst_m3u8_client_free (demux->client);
|
|
|
|
demux->client = gst_m3u8_client_new (uri);
|
|
|
|
GST_INFO_OBJECT (demux, "Changed location: %s", uri);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2012-03-15 22:21:58 +00:00
|
|
|
void
|
|
|
|
gst_hls_demux_updates_loop (GstHLSDemux * demux)
|
2011-02-14 17:51:32 +00:00
|
|
|
{
|
2011-02-15 03:39:34 +00:00
|
|
|
/* Loop for the updates. It's started when the first fragments are cached and
|
|
|
|
* schedules the next update of the playlist (for lives sources) and the next
|
|
|
|
* update of fragments. When a new fragment is downloaded, it compares the
|
|
|
|
* download time with the next scheduled update to check if we can or should
|
|
|
|
* switch to a different bitrate */
|
|
|
|
|
2012-03-15 22:21:58 +00:00
|
|
|
/* block until the next scheduled update or the signal to quit this thread */
|
|
|
|
g_mutex_lock (demux->updates_timed_lock);
|
|
|
|
GST_DEBUG_OBJECT (demux, "Started updates task");
|
2011-02-14 17:51:32 +00:00
|
|
|
while (TRUE) {
|
2012-03-15 22:21:58 +00:00
|
|
|
if (g_cond_timed_wait (GST_TASK_GET_COND (demux->updates_task),
|
|
|
|
demux->updates_timed_lock, &demux->next_update)) {
|
2011-02-14 17:51:32 +00:00
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
/* update the playlist for live sources */
|
|
|
|
if (gst_m3u8_client_is_live (demux->client)) {
|
2011-09-01 22:46:19 +00:00
|
|
|
if (!gst_hls_demux_update_playlist (demux)) {
|
2011-09-01 23:18:51 +00:00
|
|
|
demux->client->update_failed_count++;
|
|
|
|
if (demux->client->update_failed_count < DEFAULT_FAILED_COUNT) {
|
|
|
|
GST_WARNING_OBJECT (demux, "Could not update the playlist");
|
|
|
|
gst_hls_demux_schedule (demux);
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
GST_ELEMENT_ERROR (demux, RESOURCE, NOT_FOUND,
|
|
|
|
("Could not update the playlist"), (NULL));
|
|
|
|
goto quit;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* schedule the next update */
|
|
|
|
gst_hls_demux_schedule (demux);
|
|
|
|
|
|
|
|
/* if it's a live source and the playlist couldn't be updated, there aren't
|
2011-02-15 03:39:34 +00:00
|
|
|
* more fragments in the playlist, so we just wait for the next schedulled
|
2011-02-14 17:51:32 +00:00
|
|
|
* update */
|
|
|
|
if (gst_m3u8_client_is_live (demux->client) &&
|
|
|
|
demux->client->update_failed_count > 0) {
|
|
|
|
GST_WARNING_OBJECT (demux,
|
|
|
|
"The playlist hasn't been updated, failed count is %d",
|
|
|
|
demux->client->update_failed_count);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fetch the next fragment */
|
2011-08-31 03:07:48 +00:00
|
|
|
if (g_queue_is_empty (demux->queue)) {
|
2012-03-15 18:42:44 +00:00
|
|
|
if (!gst_hls_demux_get_next_fragment (demux, FALSE)) {
|
2011-09-01 23:18:51 +00:00
|
|
|
if (!demux->end_of_playlist && !demux->cancelled) {
|
|
|
|
demux->client->update_failed_count++;
|
|
|
|
if (demux->client->update_failed_count < DEFAULT_FAILED_COUNT) {
|
|
|
|
GST_WARNING_OBJECT (demux, "Could not fetch the next fragment");
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
GST_ELEMENT_ERROR (demux, RESOURCE, NOT_FOUND,
|
|
|
|
("Could not fetch the next fragment"), (NULL));
|
|
|
|
goto quit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
demux->client->update_failed_count = 0;
|
2011-08-31 03:07:48 +00:00
|
|
|
}
|
2011-09-01 23:18:51 +00:00
|
|
|
|
2011-08-31 03:07:48 +00:00
|
|
|
/* try to switch to another bitrate if needed */
|
|
|
|
gst_hls_demux_switch_playlist (demux);
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
quit:
|
|
|
|
{
|
2012-03-15 18:42:44 +00:00
|
|
|
GST_DEBUG_OBJECT (demux, "Stopped updates task");
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_hls_demux_stop (demux);
|
|
|
|
g_mutex_unlock (demux->updates_timed_lock);
|
2011-08-31 03:01:58 +00:00
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_cache_fragments (GstHLSDemux * demux)
|
|
|
|
{
|
|
|
|
gint i;
|
|
|
|
|
2011-02-15 03:39:34 +00:00
|
|
|
/* If this playlist is a variant playlist, select the first one
|
2011-02-14 17:51:32 +00:00
|
|
|
* and update it */
|
|
|
|
if (gst_m3u8_client_has_variant_playlist (demux->client)) {
|
2011-09-02 23:49:38 +00:00
|
|
|
GstM3U8 *child = NULL;
|
|
|
|
|
|
|
|
GST_M3U8_CLIENT_LOCK (demux->client);
|
|
|
|
child = demux->client->main->current_variant->data;
|
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
2011-02-14 17:51:32 +00:00
|
|
|
gst_m3u8_client_set_current (demux->client, child);
|
2011-09-01 22:46:19 +00:00
|
|
|
if (!gst_hls_demux_update_playlist (demux)) {
|
2011-02-14 17:51:32 +00:00
|
|
|
GST_ERROR_OBJECT (demux, "Could not fetch the child playlist %s",
|
|
|
|
child->uri);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If it's a live source, set the sequence number to the end of the list
|
|
|
|
* and substract the 'fragmets_cache' to start from the last fragment*/
|
|
|
|
if (gst_m3u8_client_is_live (demux->client)) {
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_LOCK (demux->client);
|
2011-02-14 17:51:32 +00:00
|
|
|
demux->client->sequence += g_list_length (demux->client->current->files);
|
|
|
|
if (demux->client->sequence >= demux->fragments_cache)
|
|
|
|
demux->client->sequence -= demux->fragments_cache;
|
|
|
|
else
|
|
|
|
demux->client->sequence = 0;
|
2011-09-08 23:56:33 +00:00
|
|
|
gst_m3u8_client_get_current_position (demux->client, &demux->position);
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
2011-08-31 02:05:08 +00:00
|
|
|
} else {
|
|
|
|
GstClockTime duration = gst_m3u8_client_get_duration (demux->client);
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (demux, "Sending duration message : %" GST_TIME_FORMAT,
|
|
|
|
GST_TIME_ARGS (duration));
|
|
|
|
if (duration != GST_CLOCK_TIME_NONE)
|
|
|
|
gst_element_post_message (GST_ELEMENT (demux),
|
|
|
|
gst_message_new_duration (GST_OBJECT (demux),
|
|
|
|
GST_FORMAT_TIME, duration));
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Cache the first fragments */
|
2011-08-23 22:38:05 +00:00
|
|
|
for (i = 0; i < demux->fragments_cache; i++) {
|
2011-08-23 22:38:26 +00:00
|
|
|
gst_element_post_message (GST_ELEMENT (demux),
|
|
|
|
gst_message_new_buffering (GST_OBJECT (demux),
|
|
|
|
100 * i / demux->fragments_cache));
|
2011-08-02 19:21:47 +00:00
|
|
|
g_get_current_time (&demux->next_update);
|
|
|
|
g_time_val_add (&demux->next_update,
|
2011-09-02 23:49:38 +00:00
|
|
|
gst_m3u8_client_get_target_duration (demux->client)
|
|
|
|
/ GST_SECOND * G_USEC_PER_SEC);
|
2012-03-15 18:42:44 +00:00
|
|
|
if (!gst_hls_demux_get_next_fragment (demux, TRUE)) {
|
2012-02-24 08:54:35 +00:00
|
|
|
if (demux->end_of_playlist)
|
|
|
|
break;
|
2011-02-15 21:40:21 +00:00
|
|
|
if (!demux->cancelled)
|
|
|
|
GST_ERROR_OBJECT (demux, "Error caching the first fragments");
|
2011-02-14 17:51:32 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
2011-02-15 21:40:21 +00:00
|
|
|
/* make sure we stop caching fragments if something cancelled it */
|
|
|
|
if (demux->cancelled)
|
|
|
|
return FALSE;
|
2011-08-02 19:21:47 +00:00
|
|
|
gst_hls_demux_switch_playlist (demux);
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
2011-08-23 22:38:26 +00:00
|
|
|
gst_element_post_message (GST_ELEMENT (demux),
|
|
|
|
gst_message_new_buffering (GST_OBJECT (demux), 100));
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
g_get_current_time (&demux->next_update);
|
|
|
|
|
|
|
|
demux->need_cache = FALSE;
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-04-01 23:21:34 +00:00
|
|
|
static gchar *
|
2012-03-15 18:42:44 +00:00
|
|
|
gst_hls_src_buf_to_utf8_playlist (GstBuffer * buf)
|
2011-04-01 23:21:34 +00:00
|
|
|
{
|
2012-03-15 18:42:44 +00:00
|
|
|
gint size;
|
|
|
|
gchar *data;
|
2011-04-01 23:21:34 +00:00
|
|
|
gchar *playlist;
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
data = (gchar *) GST_BUFFER_DATA (buf);
|
|
|
|
size = GST_BUFFER_SIZE (buf);
|
|
|
|
|
2011-04-01 23:21:34 +00:00
|
|
|
if (!g_utf8_validate (data, size, NULL))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* alloc size + 1 to end with a null character */
|
|
|
|
playlist = g_malloc0 (size + 1);
|
|
|
|
memcpy (playlist, data, size + 1);
|
2012-03-15 18:42:44 +00:00
|
|
|
|
|
|
|
gst_buffer_unref (buf);
|
2011-04-01 23:21:34 +00:00
|
|
|
return playlist;
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
static gboolean
|
2011-09-01 22:46:19 +00:00
|
|
|
gst_hls_demux_update_playlist (GstHLSDemux * demux)
|
2011-02-14 17:51:32 +00:00
|
|
|
{
|
2012-03-15 18:42:44 +00:00
|
|
|
GstFragment *download;
|
|
|
|
GstBufferListIterator *it;
|
|
|
|
GstBuffer *buf;
|
2011-02-14 17:51:32 +00:00
|
|
|
gchar *playlist;
|
2012-03-15 18:42:44 +00:00
|
|
|
|
2011-09-02 23:49:38 +00:00
|
|
|
const gchar *uri = gst_m3u8_client_get_current_uri (demux->client);
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
download = gst_uri_downloader_fetch_uri (demux->downloader, uri);
|
|
|
|
|
|
|
|
if (download == NULL)
|
2011-02-14 17:51:32 +00:00
|
|
|
return FALSE;
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
/* Merge all the buffers in the list to build a unique buffer with the
|
|
|
|
* playlist */
|
|
|
|
it = gst_buffer_list_iterate (gst_fragment_get_buffer_list (download));
|
|
|
|
|
|
|
|
/* skip the first group, which contains the headers, which are not set in the
|
|
|
|
* demuxer*/
|
|
|
|
gst_buffer_list_iterator_next_group (it);
|
|
|
|
buf = gst_buffer_list_iterator_merge_group (it);
|
|
|
|
|
|
|
|
playlist = gst_hls_src_buf_to_utf8_playlist (buf);
|
|
|
|
gst_buffer_list_iterator_free (it);
|
|
|
|
g_object_unref (download);
|
|
|
|
|
2011-04-01 23:21:34 +00:00
|
|
|
if (playlist == NULL) {
|
|
|
|
GST_WARNING_OBJECT (demux, "Couldn't not validate playlist encoding");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
gst_m3u8_client_update (demux->client, playlist);
|
2011-02-14 17:51:32 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_change_playlist (GstHLSDemux * demux, gboolean is_fast)
|
|
|
|
{
|
2011-03-12 12:15:52 +00:00
|
|
|
GList *list;
|
2011-03-12 12:32:57 +00:00
|
|
|
GstStructure *s;
|
2011-09-02 23:49:38 +00:00
|
|
|
gint new_bandwidth;
|
2011-03-12 12:15:52 +00:00
|
|
|
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_LOCK (demux->client);
|
2011-03-12 12:15:52 +00:00
|
|
|
if (is_fast)
|
2011-08-18 23:54:59 +00:00
|
|
|
list = g_list_next (demux->client->main->current_variant);
|
2011-03-12 12:15:52 +00:00
|
|
|
else
|
2011-08-18 23:54:59 +00:00
|
|
|
list = g_list_previous (demux->client->main->current_variant);
|
2011-03-12 12:15:52 +00:00
|
|
|
|
|
|
|
/* Don't do anything else if the playlist is the same */
|
2011-09-02 23:49:38 +00:00
|
|
|
if (!list || list->data == demux->client->current) {
|
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
2011-03-12 12:15:52 +00:00
|
|
|
return TRUE;
|
2011-09-02 23:49:38 +00:00
|
|
|
}
|
2011-03-12 12:15:52 +00:00
|
|
|
|
2011-08-18 23:54:59 +00:00
|
|
|
demux->client->main->current_variant = list;
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-08-18 23:54:59 +00:00
|
|
|
gst_m3u8_client_set_current (demux->client, list->data);
|
2011-09-02 23:49:38 +00:00
|
|
|
|
|
|
|
GST_M3U8_CLIENT_LOCK (demux->client);
|
|
|
|
new_bandwidth = demux->client->current->bandwidth;
|
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
|
|
|
|
2011-09-01 22:46:19 +00:00
|
|
|
gst_hls_demux_update_playlist (demux);
|
2011-02-14 17:51:32 +00:00
|
|
|
GST_INFO_OBJECT (demux, "Client is %s, switching to bitrate %d",
|
2011-09-02 23:49:38 +00:00
|
|
|
is_fast ? "fast" : "slow", new_bandwidth);
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-03-12 12:32:57 +00:00
|
|
|
s = gst_structure_new ("playlist",
|
2011-09-02 23:49:38 +00:00
|
|
|
"uri", G_TYPE_STRING, gst_m3u8_client_get_current_uri (demux->client),
|
|
|
|
"bitrate", G_TYPE_INT, new_bandwidth, NULL);
|
2011-03-12 12:32:57 +00:00
|
|
|
gst_element_post_message (GST_ELEMENT_CAST (demux),
|
|
|
|
gst_message_new_element (GST_OBJECT_CAST (demux), s));
|
|
|
|
|
2011-07-26 15:02:05 +00:00
|
|
|
/* Force typefinding since we might have changed media type */
|
|
|
|
demux->do_typefind = TRUE;
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_schedule (GstHLSDemux * demux)
|
|
|
|
{
|
|
|
|
gfloat update_factor;
|
|
|
|
gint count;
|
|
|
|
|
|
|
|
/* As defined in §6.3.4. Reloading the Playlist file:
|
|
|
|
* "If the client reloads a Playlist file and finds that it has not
|
|
|
|
* changed then it MUST wait for a period of time before retrying. The
|
|
|
|
* minimum delay is a multiple of the target duration. This multiple is
|
|
|
|
* 0.5 for the first attempt, 1.5 for the second, and 3.0 thereafter."
|
|
|
|
*/
|
|
|
|
count = demux->client->update_failed_count;
|
|
|
|
if (count < 3)
|
|
|
|
update_factor = update_interval_factor[count];
|
|
|
|
else
|
|
|
|
update_factor = update_interval_factor[3];
|
|
|
|
|
|
|
|
/* schedule the next update using the target duration field of the
|
|
|
|
* playlist */
|
|
|
|
g_time_val_add (&demux->next_update,
|
2011-09-02 23:49:38 +00:00
|
|
|
gst_m3u8_client_get_target_duration (demux->client)
|
|
|
|
/ GST_SECOND * G_USEC_PER_SEC * update_factor);
|
2011-02-14 17:51:32 +00:00
|
|
|
GST_DEBUG_OBJECT (demux, "Next update scheduled at %s",
|
|
|
|
g_time_val_to_iso8601 (&demux->next_update));
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_switch_playlist (GstHLSDemux * demux)
|
|
|
|
{
|
|
|
|
GTimeVal now;
|
|
|
|
gint64 diff, limit;
|
|
|
|
|
|
|
|
g_get_current_time (&now);
|
2011-09-02 23:49:38 +00:00
|
|
|
GST_M3U8_CLIENT_LOCK (demux->client);
|
|
|
|
if (!demux->client->main->lists) {
|
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
2011-02-14 17:51:32 +00:00
|
|
|
return TRUE;
|
2011-09-02 23:49:38 +00:00
|
|
|
}
|
|
|
|
GST_M3U8_CLIENT_UNLOCK (demux->client);
|
2011-02-14 17:51:32 +00:00
|
|
|
|
|
|
|
/* compare the time when the fragment was downloaded with the time when it was
|
|
|
|
* scheduled */
|
|
|
|
diff = (GST_TIMEVAL_TO_TIME (demux->next_update) - GST_TIMEVAL_TO_TIME (now));
|
2011-09-02 23:49:38 +00:00
|
|
|
limit = gst_m3u8_client_get_target_duration (demux->client)
|
|
|
|
* demux->bitrate_switch_tol;
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2011-07-26 15:02:05 +00:00
|
|
|
GST_DEBUG ("diff:%s%" GST_TIME_FORMAT ", limit:%" GST_TIME_FORMAT,
|
|
|
|
diff < 0 ? "-" : " ", GST_TIME_ARGS (ABS (diff)), GST_TIME_ARGS (limit));
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
/* if we are on time switch to a higher bitrate */
|
|
|
|
if (diff > limit) {
|
2011-08-31 03:47:52 +00:00
|
|
|
while (diff > limit) {
|
|
|
|
gst_hls_demux_change_playlist (demux, TRUE);
|
|
|
|
diff -= limit;
|
|
|
|
}
|
|
|
|
demux->accumulated_delay = 0;
|
2011-02-14 17:51:32 +00:00
|
|
|
} else if (diff < 0) {
|
2011-02-15 03:39:34 +00:00
|
|
|
/* if the client is too slow wait until it has accumulated a certain delay to
|
2011-02-14 17:51:32 +00:00
|
|
|
* switch to a lower bitrate */
|
|
|
|
demux->accumulated_delay -= diff;
|
|
|
|
if (demux->accumulated_delay >= limit) {
|
2011-08-31 03:47:52 +00:00
|
|
|
while (demux->accumulated_delay >= limit) {
|
|
|
|
gst_hls_demux_change_playlist (demux, FALSE);
|
|
|
|
demux->accumulated_delay -= limit;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
demux->accumulated_delay = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2012-03-15 18:42:44 +00:00
|
|
|
gst_hls_demux_get_next_fragment (GstHLSDemux * demux, gboolean caching)
|
2011-02-14 17:51:32 +00:00
|
|
|
{
|
2012-03-15 18:42:44 +00:00
|
|
|
GstFragment *download;
|
2011-02-14 17:51:32 +00:00
|
|
|
const gchar *next_fragment_uri;
|
2011-04-14 11:34:53 +00:00
|
|
|
GstClockTime duration;
|
2011-08-16 19:43:08 +00:00
|
|
|
GstClockTime timestamp;
|
2012-03-15 18:42:44 +00:00
|
|
|
GstBufferList *buffer_list;
|
|
|
|
GstBuffer *buf;
|
2011-02-14 17:51:32 +00:00
|
|
|
gboolean discont;
|
|
|
|
|
2011-04-14 11:34:53 +00:00
|
|
|
if (!gst_m3u8_client_get_next_fragment (demux->client, &discont,
|
2011-08-16 19:43:08 +00:00
|
|
|
&next_fragment_uri, &duration, ×tamp)) {
|
2011-02-14 17:51:32 +00:00
|
|
|
GST_INFO_OBJECT (demux, "This playlist doesn't contain more fragments");
|
|
|
|
demux->end_of_playlist = TRUE;
|
2012-03-15 22:21:58 +00:00
|
|
|
gst_task_start (demux->stream_task);
|
2011-02-14 17:51:32 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
GST_INFO_OBJECT (demux, "Fetching next fragment %s", next_fragment_uri);
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
download = gst_uri_downloader_fetch_uri (demux->downloader,
|
|
|
|
next_fragment_uri);
|
|
|
|
|
|
|
|
if (download == NULL)
|
|
|
|
goto error;
|
2011-02-14 17:51:32 +00:00
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
buffer_list = gst_fragment_get_buffer_list (download);
|
|
|
|
buf = gst_buffer_list_get (buffer_list, 0, 0);
|
2011-04-14 11:34:53 +00:00
|
|
|
GST_BUFFER_DURATION (buf) = duration;
|
2011-08-16 19:43:08 +00:00
|
|
|
GST_BUFFER_TIMESTAMP (buf) = timestamp;
|
2011-02-15 20:49:20 +00:00
|
|
|
|
2011-07-26 15:02:05 +00:00
|
|
|
/* We actually need to do this every time we switch bitrate */
|
|
|
|
if (G_UNLIKELY (demux->do_typefind)) {
|
|
|
|
GstCaps *caps = gst_type_find_helper_for_buffer (NULL, buf, NULL);
|
|
|
|
|
|
|
|
if (!demux->input_caps || !gst_caps_is_equal (caps, demux->input_caps)) {
|
|
|
|
gst_caps_replace (&demux->input_caps, caps);
|
|
|
|
/* gst_pad_set_caps (demux->srcpad, demux->input_caps); */
|
2011-02-15 20:49:20 +00:00
|
|
|
GST_INFO_OBJECT (demux, "Input source caps: %" GST_PTR_FORMAT,
|
|
|
|
demux->input_caps);
|
2011-07-26 15:02:05 +00:00
|
|
|
demux->do_typefind = FALSE;
|
|
|
|
} else
|
|
|
|
gst_caps_unref (caps);
|
2011-02-15 20:49:20 +00:00
|
|
|
}
|
2011-07-26 15:02:05 +00:00
|
|
|
gst_buffer_set_caps (buf, demux->input_caps);
|
2011-02-15 20:49:20 +00:00
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
if (discont) {
|
2011-02-15 03:39:34 +00:00
|
|
|
GST_DEBUG_OBJECT (demux, "Marking fragment as discontinuous");
|
2011-02-15 20:49:20 +00:00
|
|
|
GST_BUFFER_FLAG_SET (buf, GST_BUFFER_FLAG_DISCONT);
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|
|
|
|
|
2012-03-15 18:42:44 +00:00
|
|
|
g_queue_push_tail (demux->queue, buffer_list);
|
|
|
|
g_object_unref (download);
|
|
|
|
if (!caching) {
|
|
|
|
GST_TASK_SIGNAL (demux->updates_task);
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:51:32 +00:00
|
|
|
return TRUE;
|
2012-03-15 18:42:44 +00:00
|
|
|
|
|
|
|
error:
|
|
|
|
{
|
|
|
|
gst_hls_demux_stop (demux);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2011-02-14 17:51:32 +00:00
|
|
|
}
|