2023-01-24 16:03:21 +00:00
|
|
|
/* GStreamer
|
|
|
|
*
|
|
|
|
* Copyright (C) 2014 Samsung Electronics. All rights reserved.
|
|
|
|
* Author: Thiago Santos <thiagoss@osg.samsung.com>
|
|
|
|
*
|
|
|
|
* Copyright (C) 2021-2022 Centricular Ltd
|
|
|
|
* Author: Edward Hervey <edward@centricular.com>
|
|
|
|
* Author: 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 "gsthlsdemux.h"
|
|
|
|
#include "gsthlsdemux-playlist-loader.h"
|
|
|
|
#include "m3u8.h"
|
|
|
|
|
|
|
|
GST_DEBUG_CATEGORY_EXTERN (gst_hls_demux2_debug);
|
|
|
|
#define GST_CAT_DEFAULT gst_hls_demux2_debug
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
#define MAX_DOWNLOAD_ERROR_COUNT 3
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
typedef enum _PlaylistLoaderState PlaylistLoaderState;
|
|
|
|
|
|
|
|
enum _PlaylistLoaderState
|
|
|
|
{
|
|
|
|
PLAYLIST_LOADER_STATE_STOPPED = 0,
|
|
|
|
PLAYLIST_LOADER_STATE_STARTING,
|
|
|
|
PLAYLIST_LOADER_STATE_LOADING,
|
|
|
|
PLAYLIST_LOADER_STATE_WAITING,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _GstHLSDemuxPlaylistLoaderPrivate
|
|
|
|
{
|
|
|
|
GstAdaptiveDemux *demux;
|
|
|
|
|
|
|
|
GstHLSDemuxPlaylistLoaderSuccessCallback success_cb;
|
|
|
|
GstHLSDemuxPlaylistLoaderErrorCallback error_cb;
|
|
|
|
gpointer userdata;
|
|
|
|
|
|
|
|
GstAdaptiveDemuxLoop *scheduler_task;
|
|
|
|
DownloadHelper *download_helper;
|
|
|
|
DownloadRequest *download_request;
|
|
|
|
|
|
|
|
PlaylistLoaderState state;
|
|
|
|
guint pending_cb_id;
|
|
|
|
|
|
|
|
gchar *base_uri;
|
|
|
|
gchar *target_playlist_uri;
|
|
|
|
|
|
|
|
gchar *loading_playlist_uri;
|
|
|
|
|
|
|
|
gboolean delta_merge_failed;
|
|
|
|
gchar *current_playlist_uri;
|
|
|
|
GstHLSMediaPlaylist *current_playlist;
|
2022-12-27 15:01:57 +00:00
|
|
|
|
|
|
|
gchar *current_playlist_redirect_uri;
|
|
|
|
|
|
|
|
guint download_error_count;
|
2023-01-24 16:03:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#define gst_hls_demux_playlist_loader_parent_class parent_class
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (GstHLSDemuxPlaylistLoader,
|
|
|
|
gst_hls_demux_playlist_loader, GST_TYPE_OBJECT);
|
|
|
|
|
|
|
|
static void gst_hls_demux_playlist_loader_finalize (GObject * object);
|
|
|
|
static gboolean gst_hls_demux_playlist_loader_update (GstHLSDemuxPlaylistLoader
|
|
|
|
* pl);
|
|
|
|
static void start_playlist_download (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate * priv);
|
|
|
|
|
|
|
|
/* Takes ownership of the loop ref */
|
|
|
|
GstHLSDemuxPlaylistLoader *
|
|
|
|
gst_hls_demux_playlist_loader_new (GstAdaptiveDemux * demux,
|
2023-01-25 10:07:43 +00:00
|
|
|
DownloadHelper * download_helper)
|
2023-01-24 16:03:21 +00:00
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoader *pl =
|
|
|
|
g_object_new (GST_TYPE_HLS_DEMUX_PLAYLIST_LOADER, NULL);
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
|
|
|
priv->demux = demux;
|
|
|
|
priv->scheduler_task = gst_adaptive_demux_get_loop (demux);
|
|
|
|
priv->download_helper = download_helper;
|
|
|
|
|
|
|
|
return pl;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gst_hls_demux_playlist_loader_set_callbacks (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
GstHLSDemuxPlaylistLoaderSuccessCallback success_cb,
|
|
|
|
GstHLSDemuxPlaylistLoaderErrorCallback error_cb, gpointer userdata)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
|
|
|
priv->success_cb = success_cb;
|
|
|
|
priv->error_cb = error_cb;
|
|
|
|
priv->userdata = userdata;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_playlist_loader_class_init (GstHLSDemuxPlaylistLoaderClass *
|
|
|
|
klass)
|
|
|
|
{
|
|
|
|
GObjectClass *gobject_class = (GObjectClass *) klass;
|
|
|
|
|
|
|
|
gobject_class->finalize = gst_hls_demux_playlist_loader_finalize;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_playlist_loader_init (GstHLSDemuxPlaylistLoader * pl)
|
|
|
|
{
|
|
|
|
pl->priv = gst_hls_demux_playlist_loader_get_instance_private (pl);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_hls_demux_playlist_loader_finalize (GObject * object)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoader *pl = GST_HLS_DEMUX_PLAYLIST_LOADER (object);
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
|
|
|
if (priv->pending_cb_id != 0) {
|
|
|
|
gst_adaptive_demux_loop_cancel_call (priv->scheduler_task,
|
|
|
|
priv->pending_cb_id);
|
|
|
|
priv->pending_cb_id = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (priv->download_request) {
|
|
|
|
downloadhelper_cancel_request (priv->download_helper,
|
|
|
|
priv->download_request);
|
|
|
|
download_request_unref (priv->download_request);
|
|
|
|
priv->download_request = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (priv->scheduler_task)
|
|
|
|
gst_adaptive_demux_loop_unref (priv->scheduler_task);
|
|
|
|
|
|
|
|
g_free (priv->base_uri);
|
|
|
|
g_free (priv->target_playlist_uri);
|
|
|
|
g_free (priv->loading_playlist_uri);
|
|
|
|
|
|
|
|
if (priv->current_playlist)
|
|
|
|
gst_hls_media_playlist_unref (priv->current_playlist);
|
|
|
|
g_free (priv->current_playlist_uri);
|
2022-12-27 15:01:57 +00:00
|
|
|
g_free (priv->current_playlist_redirect_uri);
|
2023-01-24 16:03:21 +00:00
|
|
|
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
schedule_state_update (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate * priv)
|
|
|
|
{
|
|
|
|
g_assert (priv->pending_cb_id == 0);
|
|
|
|
priv->pending_cb_id =
|
|
|
|
gst_adaptive_demux_loop_call (priv->scheduler_task,
|
|
|
|
(GSourceFunc) gst_hls_demux_playlist_loader_update,
|
|
|
|
gst_object_ref (pl), (GDestroyNotify) gst_object_unref);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
schedule_next_playlist_load (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate * priv, GstClockTime next_load_interval)
|
|
|
|
{
|
2023-01-04 10:49:14 +00:00
|
|
|
/* If we have a valid request time, compute a more accurate download time for
|
|
|
|
* the playlist.
|
|
|
|
*
|
|
|
|
* This better takes into account the time it took to actually get and process
|
|
|
|
* the current playlist.
|
|
|
|
*/
|
|
|
|
if (priv->current_playlist
|
|
|
|
&& GST_CLOCK_TIME_IS_VALID (priv->current_playlist->request_time)) {
|
|
|
|
GstClockTime now = gst_adaptive_demux2_get_monotonic_time (priv->demux);
|
|
|
|
GstClockTimeDiff load_diff = GST_CLOCK_DIFF (now,
|
|
|
|
priv->current_playlist->request_time + next_load_interval);
|
|
|
|
GST_LOG_OBJECT (pl,
|
|
|
|
"now %" GST_TIME_FORMAT " request_time %" GST_TIME_FORMAT
|
|
|
|
" next_load_interval %" GST_TIME_FORMAT, GST_TIME_ARGS (now),
|
|
|
|
GST_TIME_ARGS (priv->current_playlist->request_time),
|
|
|
|
GST_TIME_ARGS (next_load_interval));
|
|
|
|
if (load_diff < 0) {
|
|
|
|
GST_LOG_OBJECT (pl, "Playlist update already late by %" GST_STIME_FORMAT,
|
|
|
|
GST_STIME_ARGS (load_diff));
|
|
|
|
};
|
|
|
|
next_load_interval = MAX (0, load_diff);
|
|
|
|
}
|
|
|
|
|
|
|
|
GST_LOG_OBJECT (pl, "Scheduling next playlist reload in %" GST_TIME_FORMAT,
|
|
|
|
GST_TIME_ARGS (next_load_interval));
|
2023-01-24 16:03:21 +00:00
|
|
|
g_assert (priv->pending_cb_id == 0);
|
|
|
|
priv->state = PLAYLIST_LOADER_STATE_WAITING;
|
|
|
|
priv->pending_cb_id =
|
|
|
|
gst_adaptive_demux_loop_call_delayed (priv->scheduler_task,
|
|
|
|
next_load_interval,
|
|
|
|
(GSourceFunc) gst_hls_demux_playlist_loader_update,
|
|
|
|
gst_object_ref (pl), (GDestroyNotify) gst_object_unref);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gst_hls_demux_playlist_loader_start (GstHLSDemuxPlaylistLoader * pl)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
if (priv->state != PLAYLIST_LOADER_STATE_STOPPED) {
|
|
|
|
GST_LOG_OBJECT (pl, "Already started - state %d", priv->state);
|
2023-01-24 16:03:21 +00:00
|
|
|
return; /* Already active */
|
2022-12-27 15:01:57 +00:00
|
|
|
}
|
2023-01-24 16:03:21 +00:00
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (pl, "Starting playlist loading");
|
|
|
|
priv->state = PLAYLIST_LOADER_STATE_STARTING;
|
|
|
|
schedule_state_update (pl, priv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gst_hls_demux_playlist_loader_stop (GstHLSDemuxPlaylistLoader * pl)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
|
|
|
if (priv->state == PLAYLIST_LOADER_STATE_STOPPED)
|
|
|
|
return; /* Not runnning */
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (pl, "Stopping playlist loading");
|
|
|
|
|
|
|
|
if (priv->pending_cb_id != 0) {
|
|
|
|
gst_adaptive_demux_loop_cancel_call (priv->scheduler_task,
|
|
|
|
priv->pending_cb_id);
|
|
|
|
priv->pending_cb_id = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (priv->download_request) {
|
|
|
|
downloadhelper_cancel_request (priv->download_helper,
|
|
|
|
priv->download_request);
|
|
|
|
download_request_unref (priv->download_request);
|
|
|
|
priv->download_request = NULL;
|
|
|
|
}
|
2022-12-27 15:01:57 +00:00
|
|
|
|
|
|
|
priv->state = PLAYLIST_LOADER_STATE_STOPPED;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gst_hls_demux_playlist_loader_set_playlist_uri (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
const gchar * base_uri, const gchar * new_playlist_uri)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
|
|
|
gboolean playlist_uri_change = (priv->target_playlist_uri == NULL
|
|
|
|
|| g_strcmp0 (new_playlist_uri, priv->target_playlist_uri) != 0);
|
|
|
|
|
|
|
|
if (!playlist_uri_change)
|
|
|
|
return;
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (pl, "Setting target playlist URI to %s", new_playlist_uri);
|
|
|
|
|
|
|
|
g_free (priv->base_uri);
|
|
|
|
g_free (priv->target_playlist_uri);
|
|
|
|
|
|
|
|
priv->base_uri = g_strdup (base_uri);
|
|
|
|
priv->target_playlist_uri = g_strdup (new_playlist_uri);
|
|
|
|
priv->delta_merge_failed = FALSE;
|
|
|
|
|
|
|
|
switch (priv->state) {
|
|
|
|
case PLAYLIST_LOADER_STATE_STOPPED:
|
|
|
|
return; /* Not runnning */
|
|
|
|
case PLAYLIST_LOADER_STATE_STARTING:
|
|
|
|
case PLAYLIST_LOADER_STATE_LOADING:
|
|
|
|
/* If there's no pending state check, trigger one */
|
|
|
|
if (priv->pending_cb_id == 0) {
|
|
|
|
GST_LOG_OBJECT (pl, "Scheduling state update from state %d",
|
|
|
|
priv->state);
|
|
|
|
schedule_state_update (pl, priv);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case PLAYLIST_LOADER_STATE_WAITING:
|
2023-01-02 11:32:13 +00:00
|
|
|
/* Waiting for the next time to load a live playlist, but the playlist has changed so
|
|
|
|
* cancel that and trigger a new one */
|
|
|
|
g_assert (priv->pending_cb_id != 0);
|
|
|
|
gst_adaptive_demux_loop_cancel_call (priv->scheduler_task,
|
|
|
|
priv->pending_cb_id);
|
|
|
|
priv->pending_cb_id = 0;
|
|
|
|
schedule_state_update (pl, priv);
|
|
|
|
break;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check that the current playlist matches the target URI, and return
|
2022-12-26 16:04:36 +00:00
|
|
|
* TRUE if so */
|
|
|
|
gboolean
|
|
|
|
gst_hls_demux_playlist_loader_has_current_uri (GstHLSDemuxPlaylistLoader * pl,
|
2023-01-24 16:03:21 +00:00
|
|
|
const gchar * target_playlist_uri)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
2022-12-26 16:04:36 +00:00
|
|
|
if (target_playlist_uri == NULL)
|
|
|
|
target_playlist_uri = priv->target_playlist_uri;
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
if (priv->current_playlist == NULL
|
|
|
|
|| !g_str_equal (target_playlist_uri, priv->current_playlist_uri))
|
2022-12-26 16:04:36 +00:00
|
|
|
return FALSE;
|
2023-01-24 16:03:21 +00:00
|
|
|
|
2022-12-26 16:04:36 +00:00
|
|
|
return TRUE;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
enum PlaylistDownloadParamFlags
|
|
|
|
{
|
|
|
|
PLAYLIST_DOWNLOAD_FLAG_SKIP_V1 = (1 << 0),
|
|
|
|
PLAYLIST_DOWNLOAD_FLAG_SKIP_V2 = (1 << 1), /* V2 also skips date-ranges */
|
|
|
|
PLAYLIST_DOWNLOAD_FLAG_BLOCKING_REQUEST = (1 << 2),
|
|
|
|
};
|
|
|
|
|
|
|
|
struct PlaylistDownloadParams
|
|
|
|
{
|
|
|
|
enum PlaylistDownloadParamFlags flags;
|
|
|
|
gint64 next_msn, next_part;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define HLS_SKIP_QUERY_KEY "_HLS_skip"
|
|
|
|
#define HLS_MSN_QUERY_KEY "_HLS_msn"
|
|
|
|
#define HLS_PART_QUERY_KEY "_HLS_part"
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
static gchar *
|
|
|
|
remove_HLS_directives_from_uri (const gchar * playlist_uri)
|
|
|
|
{
|
|
|
|
|
|
|
|
/* Catch the simple case and keep NULL as NULL */
|
|
|
|
if (playlist_uri == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
GstUri *uri = gst_uri_from_string (playlist_uri);
|
|
|
|
gst_uri_remove_query_key (uri, HLS_SKIP_QUERY_KEY);
|
|
|
|
gst_uri_remove_query_key (uri, HLS_MSN_QUERY_KEY);
|
|
|
|
gst_uri_remove_query_key (uri, HLS_PART_QUERY_KEY);
|
|
|
|
|
|
|
|
GList *keys = gst_uri_get_query_keys (uri);
|
|
|
|
if (keys)
|
|
|
|
keys = g_list_sort (keys, (GCompareFunc) g_strcmp0);
|
|
|
|
gchar *out_uri = gst_uri_to_string_with_keys (uri, keys);
|
|
|
|
gst_uri_unref (uri);
|
|
|
|
|
|
|
|
return out_uri;
|
|
|
|
}
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
static gchar *
|
|
|
|
apply_directives_to_uri (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
const gchar * playlist_uri, struct PlaylistDownloadParams *dl_params)
|
|
|
|
{
|
2023-04-04 13:58:36 +00:00
|
|
|
/* Short-circuit URI parsing if nothing will change */
|
|
|
|
if (dl_params->flags == 0)
|
|
|
|
return g_strdup (playlist_uri);
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
GstUri *uri = gst_uri_from_string (playlist_uri);
|
|
|
|
|
|
|
|
if (dl_params->flags & PLAYLIST_DOWNLOAD_FLAG_SKIP_V1) {
|
|
|
|
GST_LOG_OBJECT (pl, "Doing HLS skip (v1) request");
|
|
|
|
gst_uri_set_query_value (uri, HLS_SKIP_QUERY_KEY, "YES");
|
|
|
|
} else if (dl_params->flags & PLAYLIST_DOWNLOAD_FLAG_SKIP_V2) {
|
|
|
|
GST_LOG_OBJECT (pl, "Doing HLS skip (v2) request");
|
|
|
|
gst_uri_set_query_value (uri, HLS_SKIP_QUERY_KEY, "v2");
|
|
|
|
} else {
|
|
|
|
gst_uri_remove_query_key (uri, HLS_SKIP_QUERY_KEY);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dl_params->flags & PLAYLIST_DOWNLOAD_FLAG_BLOCKING_REQUEST
|
|
|
|
&& dl_params->next_msn != -1) {
|
|
|
|
GST_LOG_OBJECT (pl,
|
|
|
|
"Doing HLS blocking request for URI %s with MSN %" G_GINT64_FORMAT
|
|
|
|
" part %" G_GINT64_FORMAT, playlist_uri, dl_params->next_msn,
|
|
|
|
dl_params->next_part);
|
|
|
|
|
|
|
|
gchar *next_msn_str =
|
|
|
|
g_strdup_printf ("%" G_GINT64_FORMAT, dl_params->next_msn);
|
|
|
|
gst_uri_set_query_value (uri, HLS_MSN_QUERY_KEY, next_msn_str);
|
|
|
|
g_free (next_msn_str);
|
|
|
|
|
|
|
|
if (dl_params->next_part != -1) {
|
|
|
|
gchar *next_part_str =
|
|
|
|
g_strdup_printf ("%" G_GINT64_FORMAT, dl_params->next_part);
|
|
|
|
gst_uri_set_query_value (uri, HLS_PART_QUERY_KEY, next_part_str);
|
|
|
|
g_free (next_part_str);
|
|
|
|
} else {
|
|
|
|
gst_uri_remove_query_key (uri, HLS_PART_QUERY_KEY);
|
|
|
|
}
|
2022-12-27 15:01:57 +00:00
|
|
|
} else {
|
|
|
|
gst_uri_remove_query_key (uri, HLS_MSN_QUERY_KEY);
|
|
|
|
gst_uri_remove_query_key (uri, HLS_PART_QUERY_KEY);
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Produce the resulting URI with query arguments in UTF-8 order
|
|
|
|
* as required by the HLS spec:
|
|
|
|
* `Clients using Delivery Directives (Section 6.2.5) MUST ensure that
|
|
|
|
* all query parameters appear in UTF-8 order within the URI.`
|
|
|
|
*/
|
|
|
|
GList *keys = gst_uri_get_query_keys (uri);
|
|
|
|
if (keys)
|
|
|
|
keys = g_list_sort (keys, (GCompareFunc) g_strcmp0);
|
|
|
|
gchar *out_uri = gst_uri_to_string_with_keys (uri, keys);
|
|
|
|
gst_uri_unref (uri);
|
|
|
|
|
|
|
|
return out_uri;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GstClockTime
|
|
|
|
get_playlist_reload_interval (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate * priv, GstHLSMediaPlaylist * playlist)
|
|
|
|
{
|
|
|
|
if (playlist == NULL)
|
|
|
|
return GST_CLOCK_TIME_NONE; /* No playlist yet */
|
|
|
|
|
|
|
|
/* Use the most recent segment (or part segment) duration, as per
|
|
|
|
* https://datatracker.ietf.org/doc/html/draft-pantos-hls-rfc8216bis-11#section-6.3.4
|
|
|
|
*/
|
|
|
|
GstClockTime target_duration = GST_CLOCK_TIME_NONE;
|
|
|
|
GstClockTime min_reload_interval = playlist->targetduration / 2;
|
|
|
|
|
|
|
|
if (playlist->segments->len) {
|
|
|
|
GstM3U8MediaSegment *last_seg =
|
|
|
|
g_ptr_array_index (playlist->segments, playlist->segments->len - 1);
|
|
|
|
|
2023-01-25 10:07:43 +00:00
|
|
|
if (last_seg->partial_segments) {
|
2023-01-24 16:03:21 +00:00
|
|
|
GstM3U8PartialSegment *last_part =
|
|
|
|
g_ptr_array_index (last_seg->partial_segments,
|
|
|
|
last_seg->partial_segments->len - 1);
|
|
|
|
|
|
|
|
target_duration = last_part->duration;
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (playlist->partial_targetduration)) {
|
|
|
|
min_reload_interval = playlist->partial_targetduration / 2;
|
|
|
|
} else {
|
|
|
|
min_reload_interval = target_duration / 2;
|
|
|
|
}
|
2023-01-04 10:45:30 +00:00
|
|
|
} else {
|
|
|
|
target_duration = last_seg->duration;
|
|
|
|
min_reload_interval = target_duration / 2;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
2023-01-25 10:07:43 +00:00
|
|
|
} else if (GST_CLOCK_TIME_IS_VALID (playlist->partial_targetduration)) {
|
2023-01-24 16:03:21 +00:00
|
|
|
target_duration = playlist->partial_targetduration;
|
|
|
|
min_reload_interval = target_duration / 2;
|
|
|
|
} else if (playlist->version > 5) {
|
|
|
|
target_duration = playlist->targetduration;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (playlist->reloaded && target_duration > min_reload_interval) {
|
|
|
|
GST_DEBUG_OBJECT (pl,
|
2023-01-04 10:45:30 +00:00
|
|
|
"Playlist didn't change previously, returning lower update interval (%"
|
|
|
|
GST_TIME_FORMAT " -> %" GST_TIME_FORMAT ")",
|
|
|
|
GST_TIME_ARGS (target_duration), GST_TIME_ARGS (min_reload_interval));
|
2023-01-24 16:03:21 +00:00
|
|
|
target_duration = min_reload_interval;
|
|
|
|
}
|
|
|
|
|
2023-01-04 10:45:30 +00:00
|
|
|
GST_DEBUG_OBJECT (pl, "Returning target duration %" GST_TIME_FORMAT,
|
|
|
|
GST_TIME_ARGS (target_duration));
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
return target_duration;
|
|
|
|
}
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
static void
|
|
|
|
handle_download_error (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate * priv)
|
|
|
|
{
|
|
|
|
if (++priv->download_error_count > MAX_DOWNLOAD_ERROR_COUNT) {
|
|
|
|
GST_DEBUG_OBJECT (pl,
|
|
|
|
"Reached %d download failures on URI %s. Reporting the failure",
|
|
|
|
priv->download_error_count, priv->loading_playlist_uri);
|
|
|
|
if (priv->error_cb)
|
|
|
|
priv->error_cb (pl, priv->loading_playlist_uri, priv->userdata);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The error callback may have provided a new playlist to load, which
|
|
|
|
* will have scheduled a state update immediately. In that case,
|
|
|
|
* don't trigger our own delayed retry */
|
|
|
|
if (priv->pending_cb_id == 0)
|
|
|
|
schedule_next_playlist_load (pl, priv, 100 * GST_MSECOND);
|
|
|
|
}
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
static void
|
|
|
|
on_download_complete (DownloadRequest * download, DownloadRequestState state,
|
|
|
|
GstHLSDemuxPlaylistLoader * pl)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
|
|
|
if (priv->state != PLAYLIST_LOADER_STATE_LOADING) {
|
|
|
|
GST_DEBUG_OBJECT (pl, "Loader state changed to %d. Aborting", priv->state);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!g_str_equal (priv->target_playlist_uri, priv->loading_playlist_uri)) {
|
|
|
|
/* This callback happened just as the playlist URI was updated. There should be
|
|
|
|
* a pending state update scheduled, but we can just kick off the new download
|
|
|
|
* immediately */
|
|
|
|
GST_DEBUG_OBJECT (pl,
|
|
|
|
"Target playlist URI changed from %s to %s. Discarding download",
|
|
|
|
priv->loading_playlist_uri, priv->target_playlist_uri);
|
|
|
|
start_playlist_download (pl, priv);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (pl, "Handling completed playlist download for URI %s",
|
|
|
|
download->uri);
|
|
|
|
|
|
|
|
/* If we got a permanent redirect, use that as the new
|
|
|
|
* playlist URI, otherwise set the base URI of the playlist
|
|
|
|
* to the redirect target if any (NULL if there was no redirect) */
|
|
|
|
GstHLSMediaPlaylist *playlist = NULL;
|
|
|
|
gchar *base_uri, *uri;
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
if (download->redirect_uri) {
|
|
|
|
/* Strip HLS request params from the playlist and redirect URI */
|
|
|
|
uri = remove_HLS_directives_from_uri (download->redirect_uri);
|
2023-01-24 16:03:21 +00:00
|
|
|
base_uri = NULL;
|
2022-12-27 15:01:57 +00:00
|
|
|
|
|
|
|
if (download->redirect_permanent) {
|
|
|
|
/* Store this redirect as the future request URI for this playlist */
|
|
|
|
g_free (priv->current_playlist_redirect_uri);
|
|
|
|
priv->current_playlist_redirect_uri = g_strdup (uri);
|
|
|
|
}
|
2023-01-24 16:03:21 +00:00
|
|
|
} else {
|
2022-12-27 15:01:57 +00:00
|
|
|
/* Strip HLS request params from the playlist and redirect URI */
|
|
|
|
uri = remove_HLS_directives_from_uri (download->uri);
|
|
|
|
base_uri = remove_HLS_directives_from_uri (download->redirect_uri);
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Calculate the newest time we know this playlist was valid to store on the HLS Media Playlist */
|
|
|
|
GstClockTime playlist_ts =
|
|
|
|
MAX (0, GST_CLOCK_DIFF (download_request_get_age (download),
|
|
|
|
download->download_start_time));
|
|
|
|
|
|
|
|
GstBuffer *buf = download_request_take_buffer (download);
|
|
|
|
|
|
|
|
/* there should be a buf if there wasn't an error (handled above) */
|
|
|
|
g_assert (buf);
|
|
|
|
|
|
|
|
gchar *playlist_data = gst_hls_buf_to_utf8_text (buf);
|
|
|
|
gst_buffer_unref (buf);
|
|
|
|
|
|
|
|
if (playlist_data == NULL) {
|
|
|
|
GST_WARNING_OBJECT (pl, "Couldn't validate playlist encoding");
|
2022-12-27 15:01:57 +00:00
|
|
|
goto error_retry_out;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GstHLSMediaPlaylist *current_playlist = priv->current_playlist;
|
|
|
|
gboolean playlist_uri_change = (current_playlist == NULL
|
|
|
|
|| g_strcmp0 (priv->loading_playlist_uri,
|
|
|
|
priv->current_playlist_uri) != 0);
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
gboolean always_reload = FALSE;
|
|
|
|
#if 0
|
|
|
|
/* Test code the reports a playlist load error if we load
|
|
|
|
the same playlist 2 times in a row and the URI contains "video.m3u8"
|
|
|
|
https://playertest.longtailvideo.com/adaptive/elephants_dream_v4/redundant.m3u8
|
|
|
|
works as a test URL
|
|
|
|
*/
|
|
|
|
static gint playlist_load_counter = 0;
|
|
|
|
if (playlist_uri_change)
|
|
|
|
playlist_load_counter = 0;
|
|
|
|
else if (strstr (priv->loading_playlist_uri, "video.m3u8") != NULL) {
|
|
|
|
playlist_load_counter++;
|
|
|
|
if (playlist_load_counter > 1) {
|
|
|
|
g_print ("Triggering playlist failure for %s\n",
|
|
|
|
priv->loading_playlist_uri);
|
|
|
|
goto error_retry_out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
always_reload = TRUE;
|
|
|
|
#endif
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
if (!playlist_uri_change && current_playlist
|
|
|
|
&& gst_hls_media_playlist_has_same_data (current_playlist,
|
|
|
|
playlist_data)) {
|
|
|
|
GST_DEBUG_OBJECT (pl, "playlist data was unchanged");
|
|
|
|
playlist = gst_hls_media_playlist_ref (current_playlist);
|
|
|
|
playlist->reloaded = TRUE;
|
2023-01-04 10:49:14 +00:00
|
|
|
playlist->request_time = GST_CLOCK_TIME_NONE;
|
2023-01-24 16:03:21 +00:00
|
|
|
g_free (playlist_data);
|
|
|
|
} else {
|
|
|
|
playlist =
|
|
|
|
gst_hls_media_playlist_parse (playlist_data, playlist_ts, uri,
|
|
|
|
base_uri);
|
|
|
|
if (!playlist) {
|
|
|
|
GST_WARNING_OBJECT (pl, "Couldn't parse playlist");
|
2022-12-27 15:01:57 +00:00
|
|
|
goto error_retry_out;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
2023-01-04 10:49:14 +00:00
|
|
|
playlist->request_time = download->download_request_time;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Transfer over any skipped segments from the current playlist if
|
|
|
|
* we did a delta playlist update */
|
|
|
|
if (!playlist_uri_change && current_playlist && playlist
|
|
|
|
&& playlist->skipped_segments > 0) {
|
|
|
|
if (!gst_hls_media_playlist_sync_skipped_segments (playlist,
|
|
|
|
current_playlist)) {
|
|
|
|
GST_DEBUG_OBJECT (pl,
|
|
|
|
"Could not merge delta update to playlist. Retrying with full request");
|
|
|
|
|
|
|
|
gst_hls_media_playlist_unref (playlist);
|
|
|
|
|
|
|
|
/* Delta playlist update failed. Load a full playlist */
|
|
|
|
priv->delta_merge_failed = TRUE;
|
|
|
|
start_playlist_download (pl, priv);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (priv->current_playlist_uri);
|
|
|
|
if (priv->current_playlist)
|
|
|
|
gst_hls_media_playlist_unref (priv->current_playlist);
|
|
|
|
|
|
|
|
priv->current_playlist_uri = g_strdup (priv->loading_playlist_uri);
|
|
|
|
priv->current_playlist = playlist;
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
/* Successfully loaded the playlist. Forget any prior failures */
|
|
|
|
priv->download_error_count = 0;
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
if (priv->success_cb)
|
|
|
|
priv->success_cb (pl, priv->current_playlist_uri, priv->current_playlist,
|
|
|
|
priv->userdata);
|
|
|
|
|
|
|
|
g_free (priv->loading_playlist_uri);
|
|
|
|
priv->loading_playlist_uri = NULL;
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
if (gst_hls_media_playlist_is_live (playlist) || always_reload) {
|
2023-01-24 16:03:21 +00:00
|
|
|
/* Schedule the next playlist load. If we can do a blocking load,
|
|
|
|
* do it immediately, otherwise delayed */
|
|
|
|
if (playlist->can_block_reload) {
|
|
|
|
start_playlist_download (pl, priv);
|
|
|
|
} else {
|
|
|
|
GstClockTime delay = get_playlist_reload_interval (pl, priv, playlist);
|
|
|
|
schedule_next_playlist_load (pl, priv, delay);
|
|
|
|
}
|
2022-12-26 16:04:36 +00:00
|
|
|
} else {
|
|
|
|
GST_LOG_OBJECT (pl, "Playlist is not live. Not scheduling a reload");
|
|
|
|
/* Go back to the starting state until/if the playlist uri is updated */
|
|
|
|
priv->state = PLAYLIST_LOADER_STATE_STARTING;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
g_free (uri);
|
|
|
|
g_free (base_uri);
|
|
|
|
return;
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
error_retry_out:
|
|
|
|
/* Got invalid playlist data, retry soon or error out */
|
|
|
|
handle_download_error (pl, priv);
|
2023-01-24 16:03:21 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_download_error (DownloadRequest * download, DownloadRequestState state,
|
|
|
|
GstHLSDemuxPlaylistLoader * pl)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
|
|
|
if (priv->state != PLAYLIST_LOADER_STATE_LOADING) {
|
|
|
|
GST_DEBUG_OBJECT (pl, "Loader state changed to %d. Aborting", priv->state);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
GST_WARNING_OBJECT (pl,
|
|
|
|
"Couldn't retrieve playlist, got HTTP status code %d",
|
|
|
|
download->status_code);
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
handle_download_error (pl, priv);
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
start_playlist_download (GstHLSDemuxPlaylistLoader * pl,
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate * priv)
|
|
|
|
{
|
|
|
|
gboolean allow_skip = !priv->delta_merge_failed;
|
|
|
|
const gchar *orig_uri = priv->target_playlist_uri;
|
|
|
|
|
|
|
|
/* Can't download yet */
|
|
|
|
if (orig_uri == NULL)
|
|
|
|
return;
|
|
|
|
|
2022-12-26 16:04:36 +00:00
|
|
|
struct PlaylistDownloadParams dl_params;
|
|
|
|
memset (&dl_params, 0, sizeof (struct PlaylistDownloadParams));
|
2023-01-24 16:03:21 +00:00
|
|
|
|
|
|
|
GstHLSMediaPlaylist *current_playlist = priv->current_playlist;
|
|
|
|
|
|
|
|
/* If there's no previous playlist, or the URI changed this
|
|
|
|
* is not a refresh/update but a switch to a new playlist */
|
|
|
|
gboolean playlist_uri_change = (current_playlist == NULL
|
|
|
|
|| g_strcmp0 (orig_uri, priv->current_playlist_uri) != 0);
|
|
|
|
|
|
|
|
if (!playlist_uri_change) {
|
|
|
|
GST_LOG_OBJECT (pl, "Updating the playlist");
|
|
|
|
|
2022-12-27 15:01:57 +00:00
|
|
|
/* If we have a redirect stored for this playlist URI, use that instead */
|
|
|
|
if (priv->current_playlist_redirect_uri) {
|
|
|
|
orig_uri = priv->current_playlist_redirect_uri;
|
|
|
|
GST_LOG_OBJECT (pl, "Using redirected playlist URI %s", orig_uri);
|
|
|
|
}
|
|
|
|
|
2023-01-24 16:03:21 +00:00
|
|
|
/* See if we can do a delta playlist update (if the playlist age is less than
|
|
|
|
* one half of the Skip Boundary */
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (current_playlist->skip_boundary) && allow_skip) {
|
|
|
|
GstClockTime now = gst_adaptive_demux2_get_monotonic_time (priv->demux);
|
|
|
|
GstClockTimeDiff playlist_age =
|
|
|
|
GST_CLOCK_DIFF (current_playlist->playlist_ts, now);
|
|
|
|
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (current_playlist->playlist_ts) &&
|
|
|
|
playlist_age <= current_playlist->skip_boundary / 2) {
|
|
|
|
if (current_playlist->can_skip_dateranges) {
|
|
|
|
dl_params.flags |= PLAYLIST_DOWNLOAD_FLAG_SKIP_V2;
|
|
|
|
} else {
|
|
|
|
dl_params.flags |= PLAYLIST_DOWNLOAD_FLAG_SKIP_V1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (GST_CLOCK_TIME_IS_VALID (current_playlist->skip_boundary)) {
|
|
|
|
GST_DEBUG_OBJECT (pl,
|
|
|
|
"Doing full playlist update after failed delta request");
|
|
|
|
}
|
2022-12-27 15:01:57 +00:00
|
|
|
} else {
|
|
|
|
/* This is the first time loading this playlist URI, clear the error counter
|
|
|
|
* and redirect URI */
|
2024-03-18 13:57:43 +00:00
|
|
|
if (!priv->loading_playlist_uri
|
|
|
|
|| g_strcmp0 (orig_uri, priv->loading_playlist_uri))
|
|
|
|
priv->download_error_count = 0;
|
2022-12-27 15:01:57 +00:00
|
|
|
g_free (priv->current_playlist_redirect_uri);
|
|
|
|
priv->current_playlist_redirect_uri = NULL;
|
2023-01-24 16:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Blocking playlist reload check */
|
|
|
|
if (current_playlist != NULL && current_playlist->can_block_reload) {
|
|
|
|
if (playlist_uri_change) {
|
|
|
|
/* FIXME: We're changing playlist, but if there's a EXT-X-RENDITION-REPORT
|
|
|
|
* for the new playlist we might be able to use it to do a blocking request */
|
|
|
|
} else {
|
|
|
|
/* Get the next MSN (and/or possibly part number) for the request params */
|
|
|
|
gst_hls_media_playlist_get_next_msn_and_part (current_playlist,
|
2023-01-25 10:07:43 +00:00
|
|
|
&dl_params.next_msn, &dl_params.next_part);
|
2023-01-24 16:03:21 +00:00
|
|
|
dl_params.flags |= PLAYLIST_DOWNLOAD_FLAG_BLOCKING_REQUEST;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gchar *target_uri = apply_directives_to_uri (pl, orig_uri, &dl_params);
|
|
|
|
|
|
|
|
if (priv->download_request == NULL) {
|
|
|
|
priv->download_request = download_request_new_uri (target_uri);
|
|
|
|
|
|
|
|
download_request_set_callbacks (priv->download_request,
|
|
|
|
(DownloadRequestEventCallback) on_download_complete,
|
|
|
|
(DownloadRequestEventCallback) on_download_error,
|
|
|
|
(DownloadRequestEventCallback) NULL,
|
|
|
|
(DownloadRequestEventCallback) NULL, pl);
|
|
|
|
} else {
|
|
|
|
download_request_set_uri (priv->download_request, target_uri, 0, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (pl, "Submitting playlist download request for URI %s",
|
|
|
|
target_uri);
|
|
|
|
g_free (target_uri);
|
|
|
|
|
|
|
|
g_free (priv->loading_playlist_uri);
|
|
|
|
priv->loading_playlist_uri = g_strdup (orig_uri);
|
|
|
|
priv->state = PLAYLIST_LOADER_STATE_LOADING;
|
|
|
|
|
|
|
|
if (!downloadhelper_submit_request (priv->download_helper,
|
2023-02-15 16:32:39 +00:00
|
|
|
NULL, DOWNLOAD_FLAG_COMPRESS | DOWNLOAD_FLAG_FORCE_REFRESH,
|
2023-01-24 16:03:21 +00:00
|
|
|
priv->download_request, NULL)) {
|
|
|
|
/* Failed to submit the download - could be invalid URI, but
|
|
|
|
* could just mean the download helper was stopped */
|
|
|
|
priv->state = PLAYLIST_LOADER_STATE_STOPPED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gst_hls_demux_playlist_loader_update (GstHLSDemuxPlaylistLoader * pl)
|
|
|
|
{
|
|
|
|
GstHLSDemuxPlaylistLoaderPrivate *priv = pl->priv;
|
|
|
|
|
|
|
|
GST_LOG_OBJECT (pl, "Updating at state %d", priv->state);
|
|
|
|
priv->pending_cb_id = 0;
|
|
|
|
|
|
|
|
switch (priv->state) {
|
|
|
|
case PLAYLIST_LOADER_STATE_STOPPED:
|
|
|
|
break;
|
|
|
|
case PLAYLIST_LOADER_STATE_STARTING:
|
|
|
|
if (priv->target_playlist_uri)
|
|
|
|
start_playlist_download (pl, priv);
|
|
|
|
break;
|
|
|
|
case PLAYLIST_LOADER_STATE_LOADING:
|
|
|
|
/* A download is in progress, but if we reach here it's
|
|
|
|
* because the target playlist URI got updated, so check
|
|
|
|
* for cancelling the current download. */
|
|
|
|
if (g_str_equal (priv->target_playlist_uri, priv->current_playlist_uri))
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* A download is in progress. Cancel it and trigger a new one */
|
|
|
|
if (priv->download_request) {
|
|
|
|
GST_DEBUG_OBJECT (pl,
|
|
|
|
"Playlist URI changed from %s to %s. Cancelling current download",
|
|
|
|
priv->target_playlist_uri, priv->current_playlist_uri);
|
|
|
|
downloadhelper_cancel_request (priv->download_helper,
|
|
|
|
priv->download_request);
|
|
|
|
download_request_unref (priv->download_request);
|
|
|
|
priv->download_request = NULL;
|
|
|
|
}
|
|
|
|
start_playlist_download (pl, priv);
|
|
|
|
break;
|
|
|
|
case PLAYLIST_LOADER_STATE_WAITING:
|
|
|
|
/* We were waiting until time to load a playlist. Load it now */
|
|
|
|
start_playlist_download (pl, priv);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
}
|