forked from FFmpeg/FFmpeg
lavf/segment: add support to segment list file entries sliding window listing
In particular, should fix trac ticket #1842.
This commit is contained in:
parent
6b7d9d5153
commit
d4890c1068
3 changed files with 56 additions and 21 deletions
|
@ -545,8 +545,10 @@ the specified @var{segment_time}.
|
||||||
Default value is @code{cache}.
|
Default value is @code{cache}.
|
||||||
|
|
||||||
@item segment_list_size @var{size}
|
@item segment_list_size @var{size}
|
||||||
Overwrite the listfile once it reaches @var{size} entries. If 0
|
Update the list file so that it contains at most the last @var{size}
|
||||||
the listfile is never overwritten. Default value is 0.
|
segments. If 0 the list file will contain all the segments. Default
|
||||||
|
value is 0.
|
||||||
|
|
||||||
@item segment_list type @var{type}
|
@item segment_list type @var{type}
|
||||||
Specify the format for the segment list file.
|
Specify the format for the segment list file.
|
||||||
|
|
||||||
|
|
|
@ -95,6 +95,9 @@ typedef struct {
|
||||||
int reference_stream_index;
|
int reference_stream_index;
|
||||||
|
|
||||||
SegmentListEntry cur_entry;
|
SegmentListEntry cur_entry;
|
||||||
|
SegmentListEntry *segment_list_entries;
|
||||||
|
SegmentListEntry *segment_list_entries_end;
|
||||||
|
|
||||||
int is_first_pkt; ///< tells if it is the first packet in the segment
|
int is_first_pkt; ///< tells if it is the first packet in the segment
|
||||||
} SegmentContext;
|
} SegmentContext;
|
||||||
|
|
||||||
|
@ -211,15 +214,19 @@ static int segment_list_open(AVFormatContext *s)
|
||||||
return ret;
|
return ret;
|
||||||
seg->list_max_segment_time = 0;
|
seg->list_max_segment_time = 0;
|
||||||
|
|
||||||
if (seg->list_type == LIST_TYPE_M3U8) {
|
if (seg->list_type == LIST_TYPE_M3U8 && seg->segment_list_entries) {
|
||||||
|
SegmentListEntry *entry;
|
||||||
|
double max_duration = 0;
|
||||||
|
|
||||||
avio_printf(seg->list_pb, "#EXTM3U\n");
|
avio_printf(seg->list_pb, "#EXTM3U\n");
|
||||||
avio_printf(seg->list_pb, "#EXT-X-VERSION:3\n");
|
avio_printf(seg->list_pb, "#EXT-X-VERSION:3\n");
|
||||||
avio_printf(seg->list_pb, "#EXT-X-MEDIA-SEQUENCE:%d\n", seg->segment_idx);
|
avio_printf(seg->list_pb, "#EXT-X-MEDIA-SEQUENCE:%d\n", seg->segment_list_entries->index);
|
||||||
avio_printf(seg->list_pb, "#EXT-X-ALLOWCACHE:%d\n",
|
avio_printf(seg->list_pb, "#EXT-X-ALLOWCACHE:%d\n",
|
||||||
!!(seg->list_flags & SEGMENT_LIST_FLAG_CACHE));
|
!!(seg->list_flags & SEGMENT_LIST_FLAG_CACHE));
|
||||||
if (seg->list_flags & SEGMENT_LIST_FLAG_LIVE)
|
|
||||||
avio_printf(seg->list_pb,
|
for (entry = seg->segment_list_entries; entry; entry = entry->next)
|
||||||
"#EXT-X-TARGETDURATION:%"PRId64"\n", seg->time / 1000000);
|
max_duration = FFMAX(max_duration, entry->end_time - entry->start_time);
|
||||||
|
avio_printf(seg->list_pb, "#EXT-X-TARGETDURATION:%"PRId64"\n", (int64_t)ceil(max_duration));
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -228,14 +235,6 @@ static int segment_list_open(AVFormatContext *s)
|
||||||
static void segment_list_close(AVFormatContext *s)
|
static void segment_list_close(AVFormatContext *s)
|
||||||
{
|
{
|
||||||
SegmentContext *seg = s->priv_data;
|
SegmentContext *seg = s->priv_data;
|
||||||
|
|
||||||
if (seg->list_type == LIST_TYPE_M3U8) {
|
|
||||||
if (!(seg->list_flags & SEGMENT_LIST_FLAG_LIVE))
|
|
||||||
avio_printf(seg->list_pb, "#EXT-X-TARGETDURATION:%d\n",
|
|
||||||
(int)ceil(seg->list_max_segment_time));
|
|
||||||
avio_printf(seg->list_pb, "#EXT-X-ENDLIST\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
avio_close(seg->list_pb);
|
avio_close(seg->list_pb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -276,15 +275,40 @@ static int segment_end(AVFormatContext *s, int write_trailer)
|
||||||
oc->filename);
|
oc->filename);
|
||||||
|
|
||||||
if (seg->list) {
|
if (seg->list) {
|
||||||
if (seg->list_size && !(seg->segment_count % seg->list_size)) {
|
if (seg->list_size || seg->list_type == LIST_TYPE_M3U8) {
|
||||||
segment_list_close(s);
|
SegmentListEntry *entry = av_mallocz(sizeof(*entry));
|
||||||
if ((ret = segment_list_open(s)) < 0)
|
if (!entry) {
|
||||||
|
ret = AVERROR(ENOMEM);
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* append new element */
|
||||||
|
memcpy(entry, &seg->cur_entry, sizeof(*entry));
|
||||||
|
if (!seg->segment_list_entries)
|
||||||
|
seg->segment_list_entries = seg->segment_list_entries_end = entry;
|
||||||
|
else
|
||||||
|
seg->segment_list_entries_end->next = entry;
|
||||||
|
seg->segment_list_entries_end = entry;
|
||||||
|
|
||||||
|
/* drop first item */
|
||||||
|
if (seg->list_size && seg->segment_count > seg->list_size) {
|
||||||
|
entry = seg->segment_list_entries;
|
||||||
|
seg->segment_list_entries = seg->segment_list_entries->next;
|
||||||
|
av_freep(&entry);
|
||||||
|
}
|
||||||
|
|
||||||
|
segment_list_close(s);
|
||||||
|
if ((ret = segment_list_open(s)) < 0)
|
||||||
|
goto end;
|
||||||
|
for (entry = seg->segment_list_entries; entry; entry = entry->next)
|
||||||
|
segment_list_print_entry(seg->list_pb, seg->list_type, entry);
|
||||||
|
if (seg->list_type == LIST_TYPE_M3U8)
|
||||||
|
avio_printf(seg->list_pb, "#EXT-X-ENDLIST\n");
|
||||||
|
} else {
|
||||||
segment_list_print_entry(seg->list_pb, seg->list_type, &seg->cur_entry);
|
segment_list_print_entry(seg->list_pb, seg->list_type, &seg->cur_entry);
|
||||||
seg->list_max_segment_time =
|
seg->list_max_segment_time =
|
||||||
FFMAX(seg->cur_entry.end_time - seg->cur_entry.start_time, seg->list_max_segment_time);
|
FFMAX(seg->cur_entry.end_time - seg->cur_entry.start_time, seg->list_max_segment_time);
|
||||||
|
}
|
||||||
avio_flush(seg->list_pb);
|
avio_flush(seg->list_pb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -694,6 +718,8 @@ static int seg_write_trailer(struct AVFormatContext *s)
|
||||||
{
|
{
|
||||||
SegmentContext *seg = s->priv_data;
|
SegmentContext *seg = s->priv_data;
|
||||||
AVFormatContext *oc = seg->avf;
|
AVFormatContext *oc = seg->avf;
|
||||||
|
SegmentListEntry *cur, *next;
|
||||||
|
|
||||||
int ret;
|
int ret;
|
||||||
if (!seg->write_header_trailer) {
|
if (!seg->write_header_trailer) {
|
||||||
if ((ret = segment_end(s, 0)) < 0)
|
if ((ret = segment_end(s, 0)) < 0)
|
||||||
|
@ -712,6 +738,13 @@ fail:
|
||||||
av_freep(&seg->times);
|
av_freep(&seg->times);
|
||||||
av_freep(&seg->frames);
|
av_freep(&seg->frames);
|
||||||
|
|
||||||
|
cur = seg->segment_list_entries;
|
||||||
|
while (cur) {
|
||||||
|
next = cur->next;
|
||||||
|
av_free(cur);
|
||||||
|
cur = next;
|
||||||
|
}
|
||||||
|
|
||||||
avformat_free_context(oc);
|
avformat_free_context(oc);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,7 +31,7 @@
|
||||||
|
|
||||||
#define LIBAVFORMAT_VERSION_MAJOR 54
|
#define LIBAVFORMAT_VERSION_MAJOR 54
|
||||||
#define LIBAVFORMAT_VERSION_MINOR 61
|
#define LIBAVFORMAT_VERSION_MINOR 61
|
||||||
#define LIBAVFORMAT_VERSION_MICRO 100
|
#define LIBAVFORMAT_VERSION_MICRO 101
|
||||||
|
|
||||||
#define LIBAVFORMAT_VERSION_INT AV_VERSION_INT(LIBAVFORMAT_VERSION_MAJOR, \
|
#define LIBAVFORMAT_VERSION_INT AV_VERSION_INT(LIBAVFORMAT_VERSION_MAJOR, \
|
||||||
LIBAVFORMAT_VERSION_MINOR, \
|
LIBAVFORMAT_VERSION_MINOR, \
|
||||||
|
|
Loading…
Add table
Reference in a new issue