/* GStreamer * Copyright (C) 2011 Andoni Morales Alastruey * * gstm3u8playlist.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., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ #include #include "gstm3u8playlist.h" #include "gsthlselements.h" #define GST_CAT_DEFAULT hls_debug enum { GST_M3U8_PLAYLIST_TYPE_EVENT, GST_M3U8_PLAYLIST_TYPE_VOD, }; typedef struct _GstM3U8Entry GstM3U8Entry; struct _GstM3U8Entry { gfloat duration; gchar *title; gchar *url; gboolean discontinuous; }; static GstM3U8Entry * gst_m3u8_entry_new (const gchar * url, const gchar * title, gfloat duration, gboolean discontinuous) { GstM3U8Entry *entry; g_return_val_if_fail (url != NULL, NULL); entry = g_new0 (GstM3U8Entry, 1); entry->url = g_strdup (url); entry->title = g_strdup (title); entry->duration = duration; entry->discontinuous = discontinuous; return entry; } static void gst_m3u8_entry_free (GstM3U8Entry * entry) { g_return_if_fail (entry != NULL); g_free (entry->url); g_free (entry->title); g_free (entry); } GstM3U8Playlist * gst_m3u8_playlist_new (guint version, guint window_size) { GstM3U8Playlist *playlist; playlist = g_new0 (GstM3U8Playlist, 1); playlist->version = version; playlist->window_size = window_size; playlist->type = GST_M3U8_PLAYLIST_TYPE_EVENT; playlist->end_list = FALSE; playlist->entries = g_queue_new (); return playlist; } void gst_m3u8_playlist_free (GstM3U8Playlist * playlist) { g_return_if_fail (playlist != NULL); g_queue_foreach (playlist->entries, (GFunc) gst_m3u8_entry_free, NULL); g_queue_free (playlist->entries); g_free (playlist); } gboolean gst_m3u8_playlist_add_entry (GstM3U8Playlist * playlist, const gchar * url, const gchar * title, gfloat duration, guint index, gboolean discontinuous) { GstM3U8Entry *entry; g_return_val_if_fail (playlist != NULL, FALSE); g_return_val_if_fail (url != NULL, FALSE); if (playlist->type == GST_M3U8_PLAYLIST_TYPE_VOD) return FALSE; entry = gst_m3u8_entry_new (url, title, duration, discontinuous); if (playlist->window_size > 0) { /* Delete old entries from the playlist */ while (playlist->entries->length >= playlist->window_size) { GstM3U8Entry *old_entry; old_entry = g_queue_pop_head (playlist->entries); gst_m3u8_entry_free (old_entry); } } playlist->sequence_number = index + 1; g_queue_push_tail (playlist->entries, entry); return TRUE; } static guint gst_m3u8_playlist_target_duration (GstM3U8Playlist * playlist) { guint64 target_duration = 0; GList *l; for (l = playlist->entries->head; l != NULL; l = l->next) { GstM3U8Entry *entry = l->data; if (entry->duration > target_duration) target_duration = entry->duration; } return (guint) ((target_duration + 500 * GST_MSECOND) / GST_SECOND); } gchar * gst_m3u8_playlist_render (GstM3U8Playlist * playlist) { GString *playlist_str; GList *l; g_return_val_if_fail (playlist != NULL, NULL); playlist_str = g_string_new ("#EXTM3U\n"); g_string_append_printf (playlist_str, "#EXT-X-VERSION:%d\n", playlist->version); g_string_append_printf (playlist_str, "#EXT-X-MEDIA-SEQUENCE:%d\n", playlist->sequence_number - playlist->entries->length); g_string_append_printf (playlist_str, "#EXT-X-TARGETDURATION:%u\n", gst_m3u8_playlist_target_duration (playlist)); g_string_append (playlist_str, "\n"); /* Entries */ for (l = playlist->entries->head; l != NULL; l = l->next) { gchar buf[G_ASCII_DTOSTR_BUF_SIZE]; GstM3U8Entry *entry = l->data; if (entry->discontinuous) g_string_append (playlist_str, "#EXT-X-DISCONTINUITY\n"); if (playlist->version < 3) { g_string_append_printf (playlist_str, "#EXTINF:%d,%s\n", (gint) ((entry->duration + 500 * GST_MSECOND) / GST_SECOND), entry->title ? entry->title : ""); } else { g_string_append_printf (playlist_str, "#EXTINF:%s,%s\n", g_ascii_dtostr (buf, sizeof (buf), entry->duration / GST_SECOND), entry->title ? entry->title : ""); } g_string_append_printf (playlist_str, "%s\n", entry->url); } if (playlist->end_list) g_string_append (playlist_str, "#EXT-X-ENDLIST"); return g_string_free (playlist_str, FALSE); }