mirror of
https://git.ffmpeg.org/ffmpeg.git
synced 2025-01-04 14:22:12 +00:00
f685f7d7a8
This adds a new option -hls_flags single_file that creates one .ts file for HLS and adds byteranges to the .m3u8 file, instead of creating one .ts file for each segment. This is helpful at least for storing large number of videos, as the number of files per video is drastically reduced and copying and storing those files takes less requests and inodes. This is based on work by Nicolas Martyanoff, discussed on ffmpeg-devel in July 2014. That patch seems abandoned by the author, and contained unrelated changes. This patch tries to add the minimum amount of code to support the byterange playlists. Signed-off-by: Michael Niedermayer <michaelni@gmx.at>
407 lines
12 KiB
C
407 lines
12 KiB
C
/*
|
|
* Apple HTTP Live Streaming segmenter
|
|
* Copyright (c) 2012, Luca Barbato
|
|
*
|
|
* This file is part of FFmpeg.
|
|
*
|
|
* FFmpeg is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* FFmpeg 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with FFmpeg; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include <float.h>
|
|
#include <stdint.h>
|
|
|
|
#include "libavutil/mathematics.h"
|
|
#include "libavutil/parseutils.h"
|
|
#include "libavutil/avstring.h"
|
|
#include "libavutil/opt.h"
|
|
#include "libavutil/log.h"
|
|
|
|
#include "avformat.h"
|
|
#include "internal.h"
|
|
|
|
typedef struct HLSSegment {
|
|
char filename[1024];
|
|
double duration; /* in seconds */
|
|
int64_t pos;
|
|
int64_t size;
|
|
|
|
struct HLSSegment *next;
|
|
} HLSSegment;
|
|
|
|
typedef enum HLSFlags {
|
|
// Generate a single media file and use byte ranges in the playlist.
|
|
HLS_SINGLE_FILE = (1 << 0),
|
|
} HLSFlags;
|
|
|
|
typedef struct HLSContext {
|
|
const AVClass *class; // Class for private options.
|
|
unsigned number;
|
|
int64_t sequence;
|
|
int64_t start_sequence;
|
|
AVOutputFormat *oformat;
|
|
|
|
AVFormatContext *avf;
|
|
|
|
float time; // Set by a private option.
|
|
int max_nb_segments; // Set by a private option.
|
|
int wrap; // Set by a private option.
|
|
uint32_t flags; // enum HLSFlags
|
|
|
|
int64_t recording_time;
|
|
int has_video;
|
|
int64_t start_pts;
|
|
int64_t end_pts;
|
|
double duration; // last segment duration computed so far, in seconds
|
|
int64_t start_pos; // last segment starting position
|
|
int64_t size; // last segment size
|
|
int nb_entries;
|
|
|
|
HLSSegment *segments;
|
|
HLSSegment *last_segment;
|
|
|
|
char *basename;
|
|
char *baseurl;
|
|
|
|
AVIOContext *pb;
|
|
} HLSContext;
|
|
|
|
static int hls_mux_init(AVFormatContext *s)
|
|
{
|
|
HLSContext *hls = s->priv_data;
|
|
AVFormatContext *oc;
|
|
int i;
|
|
|
|
hls->avf = oc = avformat_alloc_context();
|
|
if (!oc)
|
|
return AVERROR(ENOMEM);
|
|
|
|
oc->oformat = hls->oformat;
|
|
oc->interrupt_callback = s->interrupt_callback;
|
|
av_dict_copy(&oc->metadata, s->metadata, 0);
|
|
|
|
for (i = 0; i < s->nb_streams; i++) {
|
|
AVStream *st;
|
|
if (!(st = avformat_new_stream(oc, NULL)))
|
|
return AVERROR(ENOMEM);
|
|
avcodec_copy_context(st->codec, s->streams[i]->codec);
|
|
st->sample_aspect_ratio = s->streams[i]->sample_aspect_ratio;
|
|
}
|
|
hls->start_pos = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Create a new segment and append it to the segment list */
|
|
static int hls_append_segment(HLSContext *hls, double duration, int64_t pos,
|
|
int64_t size)
|
|
{
|
|
HLSSegment *en = av_malloc(sizeof(*en));
|
|
|
|
if (!en)
|
|
return AVERROR(ENOMEM);
|
|
|
|
av_strlcpy(en->filename, av_basename(hls->avf->filename), sizeof(en->filename));
|
|
|
|
en->duration = duration;
|
|
en->pos = pos;
|
|
en->size = size;
|
|
en->next = NULL;
|
|
|
|
if (!hls->segments)
|
|
hls->segments = en;
|
|
else
|
|
hls->last_segment->next = en;
|
|
|
|
hls->last_segment = en;
|
|
|
|
if (hls->max_nb_segments && hls->nb_entries >= hls->max_nb_segments) {
|
|
en = hls->segments;
|
|
hls->segments = en->next;
|
|
av_free(en);
|
|
} else
|
|
hls->nb_entries++;
|
|
|
|
hls->sequence++;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void hls_free_segments(HLSContext *hls)
|
|
{
|
|
HLSSegment *p = hls->segments, *en;
|
|
|
|
while(p) {
|
|
en = p;
|
|
p = p->next;
|
|
av_free(en);
|
|
}
|
|
}
|
|
|
|
static int hls_window(AVFormatContext *s, int last)
|
|
{
|
|
HLSContext *hls = s->priv_data;
|
|
HLSSegment *en;
|
|
int target_duration = 0;
|
|
int ret = 0;
|
|
int64_t sequence = FFMAX(hls->start_sequence, hls->sequence - hls->nb_entries);
|
|
int version = hls->flags & HLS_SINGLE_FILE ? 4 : 3;
|
|
|
|
if ((ret = avio_open2(&hls->pb, s->filename, AVIO_FLAG_WRITE,
|
|
&s->interrupt_callback, NULL)) < 0)
|
|
goto fail;
|
|
|
|
for (en = hls->segments; en; en = en->next) {
|
|
if (target_duration < en->duration)
|
|
target_duration = ceil(en->duration);
|
|
}
|
|
|
|
avio_printf(hls->pb, "#EXTM3U\n");
|
|
avio_printf(hls->pb, "#EXT-X-VERSION:%d\n", version);
|
|
avio_printf(hls->pb, "#EXT-X-TARGETDURATION:%d\n", target_duration);
|
|
avio_printf(hls->pb, "#EXT-X-MEDIA-SEQUENCE:%"PRId64"\n", sequence);
|
|
|
|
av_log(s, AV_LOG_VERBOSE, "EXT-X-MEDIA-SEQUENCE:%"PRId64"\n",
|
|
sequence);
|
|
|
|
for (en = hls->segments; en; en = en->next) {
|
|
avio_printf(hls->pb, "#EXTINF:%f,\n", en->duration);
|
|
if (hls->flags & HLS_SINGLE_FILE)
|
|
avio_printf(hls->pb, "#EXT-X-BYTERANGE:%"PRIi64"@%"PRIi64"\n",
|
|
en->size, en->pos);
|
|
if (hls->baseurl)
|
|
avio_printf(hls->pb, "%s", hls->baseurl);
|
|
avio_printf(hls->pb, "%s\n", en->filename);
|
|
}
|
|
|
|
if (last)
|
|
avio_printf(hls->pb, "#EXT-X-ENDLIST\n");
|
|
|
|
fail:
|
|
avio_closep(&hls->pb);
|
|
return ret;
|
|
}
|
|
|
|
static int hls_start(AVFormatContext *s)
|
|
{
|
|
HLSContext *c = s->priv_data;
|
|
AVFormatContext *oc = c->avf;
|
|
int err = 0;
|
|
|
|
if (c->flags & HLS_SINGLE_FILE)
|
|
av_strlcpy(oc->filename, c->basename,
|
|
sizeof(oc->filename));
|
|
else
|
|
if (av_get_frame_filename(oc->filename, sizeof(oc->filename),
|
|
c->basename, c->wrap ? c->sequence % c->wrap : c->sequence) < 0) {
|
|
av_log(oc, AV_LOG_ERROR, "Invalid segment filename template '%s'\n", c->basename);
|
|
return AVERROR(EINVAL);
|
|
}
|
|
c->number++;
|
|
|
|
if ((err = avio_open2(&oc->pb, oc->filename, AVIO_FLAG_WRITE,
|
|
&s->interrupt_callback, NULL)) < 0)
|
|
return err;
|
|
|
|
if (oc->oformat->priv_class && oc->priv_data)
|
|
av_opt_set(oc->priv_data, "mpegts_flags", "resend_headers", 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int hls_write_header(AVFormatContext *s)
|
|
{
|
|
HLSContext *hls = s->priv_data;
|
|
int ret, i;
|
|
char *p;
|
|
const char *pattern = "%d.ts";
|
|
int basename_size = strlen(s->filename) + strlen(pattern) + 1;
|
|
|
|
hls->sequence = hls->start_sequence;
|
|
hls->recording_time = hls->time * AV_TIME_BASE;
|
|
hls->start_pts = AV_NOPTS_VALUE;
|
|
|
|
if (hls->flags & HLS_SINGLE_FILE)
|
|
pattern = ".ts";
|
|
|
|
for (i = 0; i < s->nb_streams; i++)
|
|
hls->has_video +=
|
|
s->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO;
|
|
|
|
if (hls->has_video > 1)
|
|
av_log(s, AV_LOG_WARNING,
|
|
"More than a single video stream present, "
|
|
"expect issues decoding it.\n");
|
|
|
|
hls->oformat = av_guess_format("mpegts", NULL, NULL);
|
|
|
|
if (!hls->oformat) {
|
|
ret = AVERROR_MUXER_NOT_FOUND;
|
|
goto fail;
|
|
}
|
|
|
|
hls->basename = av_malloc(basename_size);
|
|
|
|
if (!hls->basename) {
|
|
ret = AVERROR(ENOMEM);
|
|
goto fail;
|
|
}
|
|
|
|
strcpy(hls->basename, s->filename);
|
|
|
|
p = strrchr(hls->basename, '.');
|
|
|
|
if (p)
|
|
*p = '\0';
|
|
|
|
av_strlcat(hls->basename, pattern, basename_size);
|
|
|
|
if ((ret = hls_mux_init(s)) < 0)
|
|
goto fail;
|
|
|
|
if ((ret = hls_start(s)) < 0)
|
|
goto fail;
|
|
|
|
if ((ret = avformat_write_header(hls->avf, NULL)) < 0)
|
|
goto fail;
|
|
|
|
|
|
fail:
|
|
if (ret) {
|
|
av_free(hls->basename);
|
|
if (hls->avf)
|
|
avformat_free_context(hls->avf);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static int hls_write_packet(AVFormatContext *s, AVPacket *pkt)
|
|
{
|
|
HLSContext *hls = s->priv_data;
|
|
AVFormatContext *oc = hls->avf;
|
|
AVStream *st = s->streams[pkt->stream_index];
|
|
int64_t end_pts = hls->recording_time * hls->number;
|
|
int is_ref_pkt = 1;
|
|
int ret, can_split = 1;
|
|
|
|
if (hls->start_pts == AV_NOPTS_VALUE) {
|
|
hls->start_pts = pkt->pts;
|
|
hls->end_pts = pkt->pts;
|
|
}
|
|
|
|
if (hls->has_video) {
|
|
can_split = st->codec->codec_type == AVMEDIA_TYPE_VIDEO &&
|
|
pkt->flags & AV_PKT_FLAG_KEY;
|
|
is_ref_pkt = st->codec->codec_type == AVMEDIA_TYPE_VIDEO;
|
|
}
|
|
if (pkt->pts == AV_NOPTS_VALUE)
|
|
is_ref_pkt = can_split = 0;
|
|
|
|
if (is_ref_pkt)
|
|
hls->duration = (double)(pkt->pts - hls->end_pts)
|
|
* st->time_base.num / st->time_base.den;
|
|
|
|
if (can_split && av_compare_ts(pkt->pts - hls->start_pts, st->time_base,
|
|
end_pts, AV_TIME_BASE_Q) >= 0) {
|
|
int64_t new_start_pos;
|
|
av_write_frame(oc, NULL); /* Flush any buffered data */
|
|
|
|
new_start_pos = avio_tell(hls->avf->pb);
|
|
hls->size = new_start_pos - hls->start_pos;
|
|
ret = hls_append_segment(hls, hls->duration, hls->start_pos, hls->size);
|
|
hls->start_pos = new_start_pos;
|
|
if (ret)
|
|
return ret;
|
|
|
|
hls->end_pts = pkt->pts;
|
|
hls->duration = 0;
|
|
|
|
if (hls->flags & HLS_SINGLE_FILE) {
|
|
if (hls->avf->oformat->priv_class && hls->avf->priv_data)
|
|
av_opt_set(hls->avf->priv_data, "mpegts_flags", "resend_headers", 0);
|
|
hls->number++;
|
|
} else {
|
|
avio_close(oc->pb);
|
|
|
|
ret = hls_start(s);
|
|
}
|
|
|
|
if (ret)
|
|
return ret;
|
|
|
|
oc = hls->avf;
|
|
|
|
if ((ret = hls_window(s, 0)) < 0)
|
|
return ret;
|
|
}
|
|
|
|
ret = ff_write_chained(oc, pkt->stream_index, pkt, s, 0);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int hls_write_trailer(struct AVFormatContext *s)
|
|
{
|
|
HLSContext *hls = s->priv_data;
|
|
AVFormatContext *oc = hls->avf;
|
|
|
|
av_write_trailer(oc);
|
|
hls->size = avio_tell(hls->avf->pb) - hls->start_pos;
|
|
avio_closep(&oc->pb);
|
|
avformat_free_context(oc);
|
|
av_free(hls->basename);
|
|
hls_append_segment(hls, hls->duration, hls->start_pos, hls->size);
|
|
hls_window(s, 1);
|
|
|
|
hls_free_segments(hls);
|
|
avio_close(hls->pb);
|
|
return 0;
|
|
}
|
|
|
|
#define OFFSET(x) offsetof(HLSContext, x)
|
|
#define E AV_OPT_FLAG_ENCODING_PARAM
|
|
static const AVOption options[] = {
|
|
{"start_number", "set first number in the sequence", OFFSET(start_sequence),AV_OPT_TYPE_INT64, {.i64 = 0}, 0, INT64_MAX, E},
|
|
{"hls_time", "set segment length in seconds", OFFSET(time), AV_OPT_TYPE_FLOAT, {.dbl = 2}, 0, FLT_MAX, E},
|
|
{"hls_list_size", "set maximum number of playlist entries", OFFSET(max_nb_segments), AV_OPT_TYPE_INT, {.i64 = 5}, 0, INT_MAX, E},
|
|
{"hls_wrap", "set number after which the index wraps", OFFSET(wrap), AV_OPT_TYPE_INT, {.i64 = 0}, 0, INT_MAX, E},
|
|
{"hls_base_url", "url to prepend to each playlist entry", OFFSET(baseurl), AV_OPT_TYPE_STRING, {.str = NULL}, 0, 0, E},
|
|
{"hls_flags", "set flags affecting HLS playlist and media file generation", OFFSET(flags), AV_OPT_TYPE_FLAGS, {.i64 = 0 }, 0, UINT_MAX, E, "flags"},
|
|
{"single_file", "generate a single media file indexed with byte ranges", 0, AV_OPT_TYPE_CONST, {.i64 = HLS_SINGLE_FILE }, 0, UINT_MAX, E, "flags"},
|
|
|
|
{ NULL },
|
|
};
|
|
|
|
static const AVClass hls_class = {
|
|
.class_name = "hls muxer",
|
|
.item_name = av_default_item_name,
|
|
.option = options,
|
|
.version = LIBAVUTIL_VERSION_INT,
|
|
};
|
|
|
|
|
|
AVOutputFormat ff_hls_muxer = {
|
|
.name = "hls",
|
|
.long_name = NULL_IF_CONFIG_SMALL("Apple HTTP Live Streaming"),
|
|
.extensions = "m3u8",
|
|
.priv_data_size = sizeof(HLSContext),
|
|
.audio_codec = AV_CODEC_ID_AAC,
|
|
.video_codec = AV_CODEC_ID_H264,
|
|
.flags = AVFMT_NOFILE | AVFMT_ALLOW_FLUSH,
|
|
.write_header = hls_write_header,
|
|
.write_packet = hls_write_packet,
|
|
.write_trailer = hls_write_trailer,
|
|
.priv_class = &hls_class,
|
|
};
|