int64_t start_pos;
int range_length, index_length;
int64_t time;
+ double prog_date_time;
int64_t duration;
int n;
} Segment;
int64_t last_duration;
int64_t total_duration;
char availability_start_time[100];
+ time_t start_time_s;
char dirname[1024];
const char *single_file_name; /* file names as specified in options */
const char *init_seg_name;
SegmentType segment_type_option; /* segment type as specified in options */
int ignore_io_errors;
int lhls;
+ int master_publish_rate;
} DASHContext;
static struct codec_string {
const char *proto = avio_find_protocol_name(c->dirname);
int use_rename = proto && !strcmp(proto, "file");
int i, start_index, start_number;
+ double prog_date_time = 0;
get_start_index_number(os, c, &start_index, &start_number);
for (i = start_index; i < os->nb_segments; i++) {
Segment *seg = os->segments[i];
+
+ if (prog_date_time == 0) {
+ if (os->nb_segments == 1)
+ prog_date_time = c->start_time_s;
+ else
+ prog_date_time = seg->prog_date_time;
+ }
+ seg->prog_date_time = prog_date_time;
+
ret = ff_hls_write_file_entry(c->m3u8_out, 0, c->single_file,
(double) seg->duration / timescale, 0,
seg->range_length, seg->start_pos, NULL,
c->single_file ? os->initfile : seg->file,
- NULL);
+ &prog_date_time);
if (ret < 0) {
av_log(os->ctx, AV_LOG_WARNING, "ff_hls_write_file_entry get error\n");
}
return ret;
}
- if (c->hls_playlist && !c->master_playlist_created) {
+ if (c->hls_playlist) {
char filename_hls[1024];
const char *audio_group = "A1";
char audio_codec_str[128] = "\0";
int is_default = 1;
int max_audio_bitrate = 0;
+ // Publish master playlist only the configured rate
+ if (c->master_playlist_created && (!c->master_publish_rate ||
+ c->streams[0].segment_index % c->master_publish_rate))
+ return 0;
+
if (*c->dirname)
snprintf(filename_hls, sizeof(filename_hls), "%smaster.m3u8", c->dirname);
else
if (os->segment_type == SEGMENT_TYPE_MP4) {
if (c->streaming)
+ // frag_every_frame : Allows lower latency streaming
+ // skip_sidx : Reduce bitrate overhead
+ // skip_trailer : Avoids growing memory usage with time
av_dict_set(&opts, "movflags", "frag_every_frame+dash+delay_moov+skip_sidx+skip_trailer", 0);
else
av_dict_set(&opts, "movflags", "frag_custom+dash+delay_moov", 0);
os->first_pts = pkt->pts;
os->last_pts = pkt->pts;
- if (!c->availability_start_time[0])
+ if (!c->availability_start_time[0]) {
+ int64_t start_time_us = av_gettime();
+ c->start_time_s = start_time_us / 1000000;
format_date_now(c->availability_start_time,
sizeof(c->availability_start_time));
+ }
if (!os->availability_time_offset && pkt->duration) {
int64_t frame_duration = av_rescale_q(pkt->duration, st->time_base,
{ "webm", "make segment file in WebM format", 0, AV_OPT_TYPE_CONST, {.i64 = SEGMENT_TYPE_WEBM }, 0, UINT_MAX, E, "segment_type"},
{ "ignore_io_errors", "Ignore IO errors during open and write. Useful for long-duration runs with network output", OFFSET(ignore_io_errors), AV_OPT_TYPE_BOOL, { .i64 = 0 }, 0, 1, E },
{ "lhls", "Enable Low-latency HLS(Experimental). Adds #EXT-X-PREFETCH tag with current segment's URI", OFFSET(lhls), AV_OPT_TYPE_BOOL, { .i64 = 0 }, 0, 1, E },
+ { "master_m3u8_publish_rate", "Publish master playlist every after this many segment intervals", OFFSET(master_publish_rate), AV_OPT_TYPE_INT, {.i64 = 0}, 0, UINT_MAX, E},
{ NULL },
};