| Line | Branch | Exec | Source |
|---|---|---|---|
| 1 | /* | ||
| 2 | * WebM DASH Manifest XML muxer | ||
| 3 | * Copyright (c) 2014 Vignesh Venkatasubramanian | ||
| 4 | * | ||
| 5 | * This file is part of FFmpeg. | ||
| 6 | * | ||
| 7 | * FFmpeg is free software; you can redistribute it and/or | ||
| 8 | * modify it under the terms of the GNU Lesser General Public | ||
| 9 | * License as published by the Free Software Foundation; either | ||
| 10 | * version 2.1 of the License, or (at your option) any later version. | ||
| 11 | * | ||
| 12 | * FFmpeg is distributed in the hope that it will be useful, | ||
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 15 | * Lesser General Public License for more details. | ||
| 16 | * | ||
| 17 | * You should have received a copy of the GNU Lesser General Public | ||
| 18 | * License along with FFmpeg; if not, write to the Free Software | ||
| 19 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | ||
| 20 | */ | ||
| 21 | |||
| 22 | /* | ||
| 23 | * WebM DASH Specification: | ||
| 24 | * https://sites.google.com/a/webmproject.org/wiki/adaptive-streaming/webm-dash-specification | ||
| 25 | * ISO DASH Specification: | ||
| 26 | * http://standards.iso.org/ittf/PubliclyAvailableStandards/c065274_ISO_IEC_23009-1_2014.zip | ||
| 27 | */ | ||
| 28 | |||
| 29 | #include <float.h> | ||
| 30 | #include <stdint.h> | ||
| 31 | #include <string.h> | ||
| 32 | #include <time.h> | ||
| 33 | |||
| 34 | #include "avformat.h" | ||
| 35 | #include "matroska.h" | ||
| 36 | #include "mux.h" | ||
| 37 | |||
| 38 | #include "libavutil/avstring.h" | ||
| 39 | #include "libavutil/dict.h" | ||
| 40 | #include "libavutil/mem.h" | ||
| 41 | #include "libavutil/opt.h" | ||
| 42 | #include "libavutil/time_internal.h" | ||
| 43 | |||
| 44 | #include "libavcodec/codec_desc.h" | ||
| 45 | |||
| 46 | typedef struct AdaptationSet { | ||
| 47 | char id[10]; | ||
| 48 | int *streams; | ||
| 49 | int nb_streams; | ||
| 50 | } AdaptationSet; | ||
| 51 | |||
| 52 | typedef struct WebMDashMuxContext { | ||
| 53 | const AVClass *class; | ||
| 54 | char *adaptation_sets; | ||
| 55 | AdaptationSet *as; | ||
| 56 | int nb_as; | ||
| 57 | int representation_id; | ||
| 58 | int is_live; | ||
| 59 | int chunk_start_index; | ||
| 60 | int chunk_duration; | ||
| 61 | char *utc_timing_url; | ||
| 62 | double time_shift_buffer_depth; | ||
| 63 | int minimum_update_period; | ||
| 64 | } WebMDashMuxContext; | ||
| 65 | |||
| 66 | 13 | static const char *get_codec_name(int codec_id) | |
| 67 | { | ||
| 68 | 13 | return avcodec_descriptor_get(codec_id)->name; | |
| 69 | } | ||
| 70 | |||
| 71 | 8 | static double get_duration(AVFormatContext *s) | |
| 72 | { | ||
| 73 | 8 | int i = 0; | |
| 74 | 8 | double max = 0.0; | |
| 75 |
2/2✓ Branch 0 taken 20 times.
✓ Branch 1 taken 8 times.
|
28 | for (i = 0; i < s->nb_streams; i++) { |
| 76 | 20 | AVDictionaryEntry *duration = av_dict_get(s->streams[i]->metadata, | |
| 77 | DURATION, NULL, 0); | ||
| 78 |
2/4✓ Branch 0 taken 20 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 20 times.
|
20 | if (!duration || atof(duration->value) < 0) continue; |
| 79 |
2/2✓ Branch 0 taken 10 times.
✓ Branch 1 taken 10 times.
|
20 | if (atof(duration->value) > max) max = atof(duration->value); |
| 80 | } | ||
| 81 | 8 | return max / 1000; | |
| 82 | } | ||
| 83 | |||
| 84 | 6 | static int write_header(AVFormatContext *s) | |
| 85 | { | ||
| 86 | 6 | WebMDashMuxContext *w = s->priv_data; | |
| 87 | 6 | AVIOContext *pb = s->pb; | |
| 88 | 6 | double min_buffer_time = 1.0; | |
| 89 | 6 | avio_printf(pb, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"); | |
| 90 | 6 | avio_printf(pb, "<MPD\n"); | |
| 91 | 6 | avio_printf(pb, " xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\n"); | |
| 92 | 6 | avio_printf(pb, " xmlns=\"urn:mpeg:DASH:schema:MPD:2011\"\n"); | |
| 93 | 6 | avio_printf(pb, " xsi:schemaLocation=\"urn:mpeg:DASH:schema:MPD:2011\"\n"); | |
| 94 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 4 times.
|
6 | avio_printf(pb, " type=\"%s\"\n", w->is_live ? "dynamic" : "static"); |
| 95 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 2 times.
|
6 | if (!w->is_live) { |
| 96 | 4 | avio_printf(pb, " mediaPresentationDuration=\"PT%gS\"\n", | |
| 97 | get_duration(s)); | ||
| 98 | } | ||
| 99 | 6 | avio_printf(pb, " minBufferTime=\"PT%gS\"\n", min_buffer_time); | |
| 100 | 12 | avio_printf(pb, " profiles=\"%s\"%s", | |
| 101 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 4 times.
|
6 | w->is_live ? "urn:mpeg:dash:profile:isoff-live:2011" : "urn:mpeg:dash:profile:webm-on-demand:2012", |
| 102 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 4 times.
|
6 | w->is_live ? "\n" : ">\n"); |
| 103 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 4 times.
|
6 | if (w->is_live) { |
| 104 | 2 | time_t local_time = time(NULL); | |
| 105 | struct tm gmt_buffer; | ||
| 106 | 2 | struct tm *gmt = gmtime_r(&local_time, &gmt_buffer); | |
| 107 | char gmt_iso[21]; | ||
| 108 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
|
2 | if (!strftime(gmt_iso, 21, "%Y-%m-%dT%H:%M:%SZ", gmt)) { |
| 109 | ✗ | return AVERROR_UNKNOWN; | |
| 110 | } | ||
| 111 |
1/2✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
|
2 | if (s->flags & AVFMT_FLAG_BITEXACT) { |
| 112 | 2 | av_strlcpy(gmt_iso, "", 1); | |
| 113 | } | ||
| 114 | 2 | avio_printf(pb, " availabilityStartTime=\"%s\"\n", gmt_iso); | |
| 115 | 2 | avio_printf(pb, " timeShiftBufferDepth=\"PT%gS\"\n", w->time_shift_buffer_depth); | |
| 116 | 2 | avio_printf(pb, " minimumUpdatePeriod=\"PT%dS\"", w->minimum_update_period); | |
| 117 | 2 | avio_printf(pb, ">\n"); | |
| 118 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
|
2 | if (w->utc_timing_url) { |
| 119 | ✗ | avio_printf(pb, "<UTCTiming\n"); | |
| 120 | ✗ | avio_printf(pb, " schemeIdUri=\"urn:mpeg:dash:utc:http-iso:2014\"\n"); | |
| 121 | ✗ | avio_printf(pb, " value=\"%s\"/>\n", w->utc_timing_url); | |
| 122 | } | ||
| 123 | } | ||
| 124 | 6 | return 0; | |
| 125 | } | ||
| 126 | |||
| 127 | 6 | static void write_footer(AVFormatContext *s) | |
| 128 | { | ||
| 129 | 6 | avio_printf(s->pb, "</MPD>\n"); | |
| 130 | 6 | } | |
| 131 | |||
| 132 | 5 | static int subsegment_alignment(AVFormatContext *s, const AdaptationSet *as) | |
| 133 | { | ||
| 134 | int i; | ||
| 135 | 5 | AVDictionaryEntry *gold = av_dict_get(s->streams[as->streams[0]]->metadata, | |
| 136 | CUE_TIMESTAMPS, NULL, 0); | ||
| 137 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 5 times.
|
5 | if (!gold) return 0; |
| 138 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 1 times.
|
6 | for (i = 1; i < as->nb_streams; i++) { |
| 139 | 5 | AVDictionaryEntry *ts = av_dict_get(s->streams[as->streams[i]]->metadata, | |
| 140 | CUE_TIMESTAMPS, NULL, 0); | ||
| 141 |
3/4✓ Branch 0 taken 5 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 4 times.
✓ Branch 4 taken 1 times.
|
5 | if (!ts || !av_strstart(ts->value, gold->value, NULL)) return 0; |
| 142 | } | ||
| 143 | 1 | return 1; | |
| 144 | } | ||
| 145 | |||
| 146 | 9 | static int bitstream_switching(AVFormatContext *s, const AdaptationSet *as) | |
| 147 | { | ||
| 148 | int i; | ||
| 149 | 9 | const AVStream *gold_st = s->streams[as->streams[0]]; | |
| 150 | 9 | AVDictionaryEntry *gold_track_num = av_dict_get(gold_st->metadata, | |
| 151 | TRACK_NUMBER, NULL, 0); | ||
| 152 | 9 | AVCodecParameters *gold_par = gold_st->codecpar; | |
| 153 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 9 times.
|
9 | if (!gold_track_num) return 0; |
| 154 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 8 times.
|
13 | for (i = 1; i < as->nb_streams; i++) { |
| 155 | 5 | const AVStream *st = s->streams[as->streams[i]]; | |
| 156 | 5 | AVDictionaryEntry *track_num = av_dict_get(st->metadata, | |
| 157 | TRACK_NUMBER, NULL, 0); | ||
| 158 | 5 | AVCodecParameters *par = st->codecpar; | |
| 159 |
3/4✓ Branch 0 taken 5 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 4 times.
✓ Branch 3 taken 1 times.
|
10 | if (!track_num || |
| 160 | 5 | !av_strstart(track_num->value, gold_track_num->value, NULL) || | |
| 161 |
1/2✓ Branch 0 taken 4 times.
✗ Branch 1 not taken.
|
4 | gold_par->codec_id != par->codec_id || |
| 162 |
1/2✓ Branch 0 taken 4 times.
✗ Branch 1 not taken.
|
4 | gold_par->extradata_size != par->extradata_size || |
| 163 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 3 times.
|
4 | (par->extradata_size > 0 && |
| 164 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | memcmp(gold_par->extradata, par->extradata, par->extradata_size))) { |
| 165 | 1 | return 0; | |
| 166 | } | ||
| 167 | } | ||
| 168 | 8 | return 1; | |
| 169 | } | ||
| 170 | |||
| 171 | /* | ||
| 172 | * Writes a Representation within an Adaptation Set. Returns 0 on success and | ||
| 173 | * < 0 on failure. | ||
| 174 | */ | ||
| 175 | 14 | static int write_representation(AVFormatContext *s, AVStream *st, char *id, | |
| 176 | int output_width, int output_height, | ||
| 177 | int output_sample_rate) | ||
| 178 | { | ||
| 179 | 14 | WebMDashMuxContext *w = s->priv_data; | |
| 180 | 14 | AVIOContext *pb = s->pb; | |
| 181 | 14 | const AVCodecParameters *par = st->codecpar; | |
| 182 | 14 | AVDictionaryEntry *bandwidth = av_dict_get(st->metadata, BANDWIDTH, NULL, 0); | |
| 183 | const char *bandwidth_str; | ||
| 184 | 14 | avio_printf(pb, "<Representation id=\"%s\"", id); | |
| 185 |
2/2✓ Branch 0 taken 12 times.
✓ Branch 1 taken 2 times.
|
14 | if (bandwidth) { |
| 186 | 12 | bandwidth_str = bandwidth->value; | |
| 187 |
1/2✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
|
2 | } else if (w->is_live) { |
| 188 | // if bandwidth for live was not provided, use a default | ||
| 189 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 1 times.
|
2 | bandwidth_str = (par->codec_type == AVMEDIA_TYPE_AUDIO) ? "128000" : "1000000"; |
| 190 | } else { | ||
| 191 | ✗ | return AVERROR(EINVAL); | |
| 192 | } | ||
| 193 | 14 | avio_printf(pb, " bandwidth=\"%s\"", bandwidth_str); | |
| 194 |
4/4✓ Branch 0 taken 8 times.
✓ Branch 1 taken 6 times.
✓ Branch 2 taken 4 times.
✓ Branch 3 taken 4 times.
|
14 | if (par->codec_type == AVMEDIA_TYPE_VIDEO && output_width) |
| 195 | 4 | avio_printf(pb, " width=\"%d\"", par->width); | |
| 196 |
4/4✓ Branch 0 taken 8 times.
✓ Branch 1 taken 6 times.
✓ Branch 2 taken 4 times.
✓ Branch 3 taken 4 times.
|
14 | if (par->codec_type == AVMEDIA_TYPE_VIDEO && output_height) |
| 197 | 4 | avio_printf(pb, " height=\"%d\"", par->height); | |
| 198 |
4/4✓ Branch 0 taken 6 times.
✓ Branch 1 taken 8 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 4 times.
|
14 | if (par->codec_type == AVMEDIA_TYPE_AUDIO && output_sample_rate) |
| 199 | 2 | avio_printf(pb, " audioSamplingRate=\"%d\"", par->sample_rate); | |
| 200 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 10 times.
|
14 | if (w->is_live) { |
| 201 | // For live streams, Codec and Mime Type always go in the Representation tag. | ||
| 202 | 4 | avio_printf(pb, " codecs=\"%s\"", get_codec_name(par->codec_id)); | |
| 203 | 4 | avio_printf(pb, " mimeType=\"%s/webm\"", | |
| 204 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 2 times.
|
4 | par->codec_type == AVMEDIA_TYPE_VIDEO ? "video" : "audio"); |
| 205 | // For live streams, subsegments always start with key frames. So this | ||
| 206 | // is always 1. | ||
| 207 | 4 | avio_printf(pb, " startsWithSAP=\"1\""); | |
| 208 | 4 | avio_printf(pb, ">"); | |
| 209 | } else { | ||
| 210 | 10 | AVDictionaryEntry *irange = av_dict_get(st->metadata, INITIALIZATION_RANGE, NULL, 0); | |
| 211 | 10 | AVDictionaryEntry *cues_start = av_dict_get(st->metadata, CUES_START, NULL, 0); | |
| 212 | 10 | AVDictionaryEntry *cues_end = av_dict_get(st->metadata, CUES_END, NULL, 0); | |
| 213 | 10 | AVDictionaryEntry *filename = av_dict_get(st->metadata, FILENAME, NULL, 0); | |
| 214 |
4/8✓ Branch 0 taken 10 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 10 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 10 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 10 times.
|
10 | if (!irange || !cues_start || !cues_end || !filename) |
| 215 | ✗ | return AVERROR(EINVAL); | |
| 216 | |||
| 217 | 10 | avio_printf(pb, ">\n"); | |
| 218 | 10 | avio_printf(pb, "<BaseURL>%s</BaseURL>\n", filename->value); | |
| 219 | 10 | avio_printf(pb, "<SegmentBase\n"); | |
| 220 | 10 | avio_printf(pb, " indexRange=\"%s-%s\">\n", cues_start->value, cues_end->value); | |
| 221 | 10 | avio_printf(pb, "<Initialization\n"); | |
| 222 | 10 | avio_printf(pb, " range=\"0-%s\" />\n", irange->value); | |
| 223 | 10 | avio_printf(pb, "</SegmentBase>\n"); | |
| 224 | } | ||
| 225 | 14 | avio_printf(pb, "</Representation>\n"); | |
| 226 | 14 | return 0; | |
| 227 | } | ||
| 228 | |||
| 229 | /* | ||
| 230 | * Checks if width of all streams are the same. Returns 1 if true, 0 otherwise. | ||
| 231 | */ | ||
| 232 | 3 | static int check_matching_width(AVFormatContext *s, const AdaptationSet *as) | |
| 233 | { | ||
| 234 | int first_width, i; | ||
| 235 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 3 times.
|
3 | if (as->nb_streams < 2) return 1; |
| 236 | 3 | first_width = s->streams[as->streams[0]]->codecpar->width; | |
| 237 |
2/2✓ Branch 0 taken 3 times.
✓ Branch 1 taken 2 times.
|
5 | for (i = 1; i < as->nb_streams; i++) |
| 238 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 2 times.
|
3 | if (first_width != s->streams[as->streams[i]]->codecpar->width) |
| 239 | 1 | return 0; | |
| 240 | 2 | return 1; | |
| 241 | } | ||
| 242 | |||
| 243 | /* | ||
| 244 | * Checks if height of all streams are the same. Returns 1 if true, 0 otherwise. | ||
| 245 | */ | ||
| 246 | 3 | static int check_matching_height(AVFormatContext *s, const AdaptationSet *as) | |
| 247 | { | ||
| 248 | int first_height, i; | ||
| 249 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 3 times.
|
3 | if (as->nb_streams < 2) return 1; |
| 250 | 3 | first_height = s->streams[as->streams[0]]->codecpar->height; | |
| 251 |
2/2✓ Branch 0 taken 3 times.
✓ Branch 1 taken 2 times.
|
5 | for (i = 1; i < as->nb_streams; i++) |
| 252 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 2 times.
|
3 | if (first_height != s->streams[as->streams[i]]->codecpar->height) |
| 253 | 1 | return 0; | |
| 254 | 2 | return 1; | |
| 255 | } | ||
| 256 | |||
| 257 | /* | ||
| 258 | * Checks if sample rate of all streams are the same. Returns 1 if true, 0 otherwise. | ||
| 259 | */ | ||
| 260 | 2 | static int check_matching_sample_rate(AVFormatContext *s, const AdaptationSet *as) | |
| 261 | { | ||
| 262 | int first_sample_rate, i; | ||
| 263 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
|
2 | if (as->nb_streams < 2) return 1; |
| 264 | 2 | first_sample_rate = s->streams[as->streams[0]]->codecpar->sample_rate; | |
| 265 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 2 times.
|
4 | for (i = 1; i < as->nb_streams; i++) |
| 266 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
|
2 | if (first_sample_rate != s->streams[as->streams[i]]->codecpar->sample_rate) |
| 267 | ✗ | return 0; | |
| 268 | 2 | return 1; | |
| 269 | } | ||
| 270 | |||
| 271 | 6 | static void free_adaptation_sets(AVFormatContext *s) | |
| 272 | { | ||
| 273 | 6 | WebMDashMuxContext *w = s->priv_data; | |
| 274 | int i; | ||
| 275 |
2/2✓ Branch 0 taken 9 times.
✓ Branch 1 taken 6 times.
|
15 | for (i = 0; i < w->nb_as; i++) { |
| 276 | 9 | av_freep(&w->as[i].streams); | |
| 277 | } | ||
| 278 | 6 | av_freep(&w->as); | |
| 279 | 6 | w->nb_as = 0; | |
| 280 | 6 | } | |
| 281 | |||
| 282 | /* | ||
| 283 | * Parses a live header filename and returns the position of the '_' and '.' | ||
| 284 | * delimiting <file_description> and <representation_id>. | ||
| 285 | * | ||
| 286 | * Name of the header file should conform to the following pattern: | ||
| 287 | * <file_description>_<representation_id>.hdr where <file_description> can be | ||
| 288 | * anything. The chunks should be named according to the following pattern: | ||
| 289 | * <file_description>_<representation_id>_<chunk_number>.chk | ||
| 290 | */ | ||
| 291 | 8 | static int split_filename(char *filename, char **underscore_pos, | |
| 292 | char **period_pos) | ||
| 293 | { | ||
| 294 | 8 | *underscore_pos = strrchr(filename, '_'); | |
| 295 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 8 times.
|
8 | if (!*underscore_pos) |
| 296 | ✗ | return AVERROR(EINVAL); | |
| 297 | 8 | *period_pos = strchr(*underscore_pos, '.'); | |
| 298 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 8 times.
|
8 | if (!*period_pos) |
| 299 | ✗ | return AVERROR(EINVAL); | |
| 300 | 8 | return 0; | |
| 301 | } | ||
| 302 | |||
| 303 | /* | ||
| 304 | * Writes an Adaptation Set. Returns 0 on success and < 0 on failure. | ||
| 305 | */ | ||
| 306 | 9 | static int write_adaptation_set(AVFormatContext *s, int as_index) | |
| 307 | { | ||
| 308 | 9 | WebMDashMuxContext *w = s->priv_data; | |
| 309 | 9 | AdaptationSet *as = &w->as[as_index]; | |
| 310 | 9 | const AVStream *st = s->streams[as->streams[0]]; | |
| 311 | 9 | AVCodecParameters *par = st->codecpar; | |
| 312 | AVDictionaryEntry *lang; | ||
| 313 | 9 | AVIOContext *pb = s->pb; | |
| 314 | int i; | ||
| 315 | static const char boolean[2][6] = { "false", "true" }; | ||
| 316 | 9 | int subsegmentStartsWithSAP = 1; | |
| 317 | |||
| 318 | // Width, Height and Sample Rate will go in the AdaptationSet tag if they | ||
| 319 | // are the same for all contained Representations. otherwise, they will go | ||
| 320 | // on their respective Representation tag. For live streams, they always go | ||
| 321 | // in the Representation tag. | ||
| 322 | 9 | int width_in_as = 1, height_in_as = 1, sample_rate_in_as = 1; | |
| 323 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 4 times.
|
9 | if (par->codec_type == AVMEDIA_TYPE_VIDEO) { |
| 324 |
4/4✓ Branch 0 taken 3 times.
✓ Branch 1 taken 2 times.
✓ Branch 3 taken 2 times.
✓ Branch 4 taken 1 times.
|
5 | width_in_as = !w->is_live && check_matching_width (s, as); |
| 325 |
4/4✓ Branch 0 taken 3 times.
✓ Branch 1 taken 2 times.
✓ Branch 3 taken 2 times.
✓ Branch 4 taken 1 times.
|
5 | height_in_as = !w->is_live && check_matching_height(s, as); |
| 326 | } else { | ||
| 327 |
3/4✓ Branch 0 taken 2 times.
✓ Branch 1 taken 2 times.
✓ Branch 3 taken 2 times.
✗ Branch 4 not taken.
|
4 | sample_rate_in_as = !w->is_live && check_matching_sample_rate(s, as); |
| 328 | } | ||
| 329 | |||
| 330 | 9 | avio_printf(pb, "<AdaptationSet id=\"%s\"", as->id); | |
| 331 | 9 | avio_printf(pb, " mimeType=\"%s/webm\"", | |
| 332 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 4 times.
|
9 | par->codec_type == AVMEDIA_TYPE_VIDEO ? "video" : "audio"); |
| 333 | 9 | avio_printf(pb, " codecs=\"%s\"", get_codec_name(par->codec_id)); | |
| 334 | |||
| 335 | 9 | lang = av_dict_get(st->metadata, "language", NULL, 0); | |
| 336 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 4 times.
|
9 | if (lang) |
| 337 | 5 | avio_printf(pb, " lang=\"%s\"", lang->value); | |
| 338 | |||
| 339 |
4/4✓ Branch 0 taken 5 times.
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 3 times.
|
9 | if (par->codec_type == AVMEDIA_TYPE_VIDEO && width_in_as) |
| 340 | 2 | avio_printf(pb, " width=\"%d\"", par->width); | |
| 341 |
4/4✓ Branch 0 taken 5 times.
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 3 times.
|
9 | if (par->codec_type == AVMEDIA_TYPE_VIDEO && height_in_as) |
| 342 | 2 | avio_printf(pb, " height=\"%d\"", par->height); | |
| 343 |
4/4✓ Branch 0 taken 4 times.
✓ Branch 1 taken 5 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 2 times.
|
9 | if (par->codec_type == AVMEDIA_TYPE_AUDIO && sample_rate_in_as) |
| 344 | 2 | avio_printf(pb, " audioSamplingRate=\"%d\"", par->sample_rate); | |
| 345 | |||
| 346 | 9 | avio_printf(pb, " bitstreamSwitching=\"%s\"", | |
| 347 | 9 | boolean[bitstream_switching(s, as)]); | |
| 348 | 9 | avio_printf(pb, " subsegmentAlignment=\"%s\"", | |
| 349 |
4/4✓ Branch 0 taken 5 times.
✓ Branch 1 taken 4 times.
✓ Branch 3 taken 1 times.
✓ Branch 4 taken 4 times.
|
9 | boolean[w->is_live || subsegment_alignment(s, as)]); |
| 350 | |||
| 351 |
2/2✓ Branch 0 taken 14 times.
✓ Branch 1 taken 9 times.
|
23 | for (i = 0; i < as->nb_streams; i++) { |
| 352 | 14 | AVDictionaryEntry *kf = av_dict_get(s->streams[as->streams[i]]->metadata, | |
| 353 | CLUSTER_KEYFRAME, NULL, 0); | ||
| 354 |
5/6✓ Branch 0 taken 10 times.
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 10 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 1 times.
✓ Branch 5 taken 9 times.
|
14 | if (!w->is_live && (!kf || !strncmp(kf->value, "0", 1))) subsegmentStartsWithSAP = 0; |
| 355 | } | ||
| 356 | 9 | avio_printf(pb, " subsegmentStartsWithSAP=\"%d\"", subsegmentStartsWithSAP); | |
| 357 | 9 | avio_printf(pb, ">\n"); | |
| 358 | |||
| 359 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 5 times.
|
9 | if (w->is_live) { |
| 360 | AVDictionaryEntry *filename = | ||
| 361 | 4 | av_dict_get(st->metadata, FILENAME, NULL, 0); | |
| 362 | char *underscore_pos, *period_pos; | ||
| 363 | int ret; | ||
| 364 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 4 times.
|
4 | if (!filename) |
| 365 | ✗ | return AVERROR(EINVAL); | |
| 366 | 4 | ret = split_filename(filename->value, &underscore_pos, &period_pos); | |
| 367 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 4 times.
|
4 | if (ret) return ret; |
| 368 | 4 | *underscore_pos = '\0'; | |
| 369 | 4 | avio_printf(pb, "<ContentComponent id=\"1\" type=\"%s\"/>\n", | |
| 370 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 2 times.
|
4 | par->codec_type == AVMEDIA_TYPE_VIDEO ? "video" : "audio"); |
| 371 | 4 | avio_printf(pb, "<SegmentTemplate"); | |
| 372 | 4 | avio_printf(pb, " timescale=\"1000\""); | |
| 373 | 4 | avio_printf(pb, " duration=\"%d\"", w->chunk_duration); | |
| 374 | 4 | avio_printf(pb, " media=\"%s_$RepresentationID$_$Number$.chk\"", | |
| 375 | filename->value); | ||
| 376 | 4 | avio_printf(pb, " startNumber=\"%d\"", w->chunk_start_index); | |
| 377 | 4 | avio_printf(pb, " initialization=\"%s_$RepresentationID$.hdr\"", | |
| 378 | filename->value); | ||
| 379 | 4 | avio_printf(pb, "/>\n"); | |
| 380 | 4 | *underscore_pos = '_'; | |
| 381 | } | ||
| 382 | |||
| 383 |
2/2✓ Branch 0 taken 14 times.
✓ Branch 1 taken 9 times.
|
23 | for (i = 0; i < as->nb_streams; i++) { |
| 384 | 14 | char buf[25], *representation_id = buf, *underscore_pos, *period_pos; | |
| 385 | 14 | AVStream *st = s->streams[as->streams[i]]; | |
| 386 | int ret; | ||
| 387 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 10 times.
|
14 | if (w->is_live) { |
| 388 | AVDictionaryEntry *filename = | ||
| 389 | 4 | av_dict_get(st->metadata, FILENAME, NULL, 0); | |
| 390 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 4 times.
|
4 | if (!filename) |
| 391 | ✗ | return AVERROR(EINVAL); | |
| 392 | 4 | ret = split_filename(filename->value, &underscore_pos, &period_pos); | |
| 393 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 4 times.
|
4 | if (ret < 0) |
| 394 | ✗ | return ret; | |
| 395 | 4 | representation_id = underscore_pos + 1; | |
| 396 | 4 | *period_pos = '\0'; | |
| 397 | } else { | ||
| 398 | 10 | snprintf(buf, sizeof(buf), "%d", w->representation_id++); | |
| 399 | } | ||
| 400 | 14 | ret = write_representation(s, st, representation_id, !width_in_as, | |
| 401 | !height_in_as, !sample_rate_in_as); | ||
| 402 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 14 times.
|
14 | if (ret) return ret; |
| 403 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 10 times.
|
14 | if (w->is_live) |
| 404 | 4 | *period_pos = '.'; | |
| 405 | } | ||
| 406 | 9 | avio_printf(s->pb, "</AdaptationSet>\n"); | |
| 407 | 9 | return 0; | |
| 408 | } | ||
| 409 | |||
| 410 | 6 | static int parse_adaptation_sets(AVFormatContext *s) | |
| 411 | { | ||
| 412 | 6 | WebMDashMuxContext *w = s->priv_data; | |
| 413 | 6 | char *p = w->adaptation_sets; | |
| 414 | char *q; | ||
| 415 | enum { new_set, parsed_id, parsing_streams } state; | ||
| 416 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
|
6 | if (!w->adaptation_sets) { |
| 417 | ✗ | av_log(s, AV_LOG_ERROR, "The 'adaptation_sets' option must be set.\n"); | |
| 418 | ✗ | return AVERROR(EINVAL); | |
| 419 | } | ||
| 420 | // syntax id=0,streams=0,1,2 id=1,streams=3,4 and so on | ||
| 421 | 6 | state = new_set; | |
| 422 | while (1) { | ||
| 423 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 32 times.
|
32 | if (*p == '\0') { |
| 424 | ✗ | if (state == new_set) | |
| 425 | ✗ | break; | |
| 426 | else | ||
| 427 | ✗ | return AVERROR(EINVAL); | |
| 428 |
3/4✓ Branch 0 taken 9 times.
✓ Branch 1 taken 23 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 9 times.
|
32 | } else if (state == new_set && *p == ' ') { |
| 429 | ✗ | p++; | |
| 430 | ✗ | continue; | |
| 431 |
3/4✓ Branch 0 taken 9 times.
✓ Branch 1 taken 23 times.
✓ Branch 2 taken 9 times.
✗ Branch 3 not taken.
|
32 | } else if (state == new_set && !strncmp(p, "id=", 3)) { |
| 432 | 9 | void *mem = av_realloc(w->as, sizeof(*w->as) * (w->nb_as + 1)); | |
| 433 | const char *comma; | ||
| 434 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 9 times.
|
9 | if (mem == NULL) |
| 435 | ✗ | return AVERROR(ENOMEM); | |
| 436 | 9 | w->as = mem; | |
| 437 | 9 | ++w->nb_as; | |
| 438 | 9 | w->as[w->nb_as - 1].nb_streams = 0; | |
| 439 | 9 | w->as[w->nb_as - 1].streams = NULL; | |
| 440 | 9 | p += 3; // consume "id=" | |
| 441 | 9 | q = w->as[w->nb_as - 1].id; | |
| 442 | 9 | comma = strchr(p, ','); | |
| 443 |
2/4✓ Branch 0 taken 9 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 9 times.
|
9 | if (!comma || comma - p >= sizeof(w->as[w->nb_as - 1].id)) { |
| 444 | ✗ | av_log(s, AV_LOG_ERROR, "'id' in 'adaptation_sets' is malformed.\n"); | |
| 445 | ✗ | return AVERROR(EINVAL); | |
| 446 | } | ||
| 447 |
2/2✓ Branch 0 taken 9 times.
✓ Branch 1 taken 9 times.
|
18 | while (*p != ',') *q++ = *p++; |
| 448 | 9 | *q = 0; | |
| 449 | 9 | p++; | |
| 450 | 9 | state = parsed_id; | |
| 451 |
3/4✓ Branch 0 taken 9 times.
✓ Branch 1 taken 14 times.
✓ Branch 2 taken 9 times.
✗ Branch 3 not taken.
|
23 | } else if (state == parsed_id && !strncmp(p, "streams=", 8)) { |
| 452 | 9 | p += 8; // consume "streams=" | |
| 453 | 9 | state = parsing_streams; | |
| 454 |
1/2✓ Branch 0 taken 14 times.
✗ Branch 1 not taken.
|
14 | } else if (state == parsing_streams) { |
| 455 | 14 | struct AdaptationSet *as = &w->as[w->nb_as - 1]; | |
| 456 | int64_t num; | ||
| 457 | 14 | int ret = av_reallocp_array(&as->streams, ++as->nb_streams, | |
| 458 | sizeof(*as->streams)); | ||
| 459 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 14 times.
|
14 | if (ret < 0) |
| 460 | ✗ | return ret; | |
| 461 | 14 | num = strtoll(p, &q, 10); | |
| 462 |
7/10✓ Branch 0 taken 14 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 11 times.
✓ Branch 3 taken 3 times.
✓ Branch 4 taken 5 times.
✓ Branch 5 taken 6 times.
✓ Branch 6 taken 5 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 14 times.
✗ Branch 9 not taken.
|
14 | if (!av_isdigit(*p) || (*q != ' ' && *q != '\0' && *q != ',') || |
| 463 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 14 times.
|
14 | num < 0 || num >= s->nb_streams) { |
| 464 | ✗ | av_log(s, AV_LOG_ERROR, "Invalid value for 'streams' in adapation_sets.\n"); | |
| 465 | ✗ | return AVERROR(EINVAL); | |
| 466 | } | ||
| 467 | 14 | as->streams[as->nb_streams - 1] = num; | |
| 468 |
2/2✓ Branch 0 taken 6 times.
✓ Branch 1 taken 8 times.
|
14 | if (*q == '\0') break; |
| 469 |
2/2✓ Branch 0 taken 3 times.
✓ Branch 1 taken 5 times.
|
8 | if (*q == ' ') state = new_set; |
| 470 | 8 | p = ++q; | |
| 471 | } else { | ||
| 472 | ✗ | return -1; | |
| 473 | } | ||
| 474 | } | ||
| 475 | 6 | return 0; | |
| 476 | } | ||
| 477 | |||
| 478 | 6 | static int webm_dash_manifest_write_header(AVFormatContext *s) | |
| 479 | { | ||
| 480 | int i; | ||
| 481 | 6 | double start = 0.0; | |
| 482 | int ret; | ||
| 483 | 6 | WebMDashMuxContext *w = s->priv_data; | |
| 484 | |||
| 485 |
2/2✓ Branch 0 taken 14 times.
✓ Branch 1 taken 6 times.
|
20 | for (unsigned i = 0; i < s->nb_streams; i++) { |
| 486 | 14 | enum AVCodecID codec_id = s->streams[i]->codecpar->codec_id; | |
| 487 |
5/6✓ Branch 0 taken 8 times.
✓ Branch 1 taken 6 times.
✓ Branch 2 taken 6 times.
✓ Branch 3 taken 2 times.
✓ Branch 4 taken 6 times.
✗ Branch 5 not taken.
|
14 | if (codec_id != AV_CODEC_ID_VP8 && codec_id != AV_CODEC_ID_VP9 && |
| 488 |
1/4✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
|
6 | codec_id != AV_CODEC_ID_AV1 && codec_id != AV_CODEC_ID_VORBIS && |
| 489 | codec_id != AV_CODEC_ID_OPUS) | ||
| 490 | ✗ | return AVERROR(EINVAL); | |
| 491 | } | ||
| 492 | |||
| 493 | 6 | ret = parse_adaptation_sets(s); | |
| 494 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
|
6 | if (ret < 0) { |
| 495 | ✗ | goto fail; | |
| 496 | } | ||
| 497 | 6 | ret = write_header(s); | |
| 498 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
|
6 | if (ret < 0) { |
| 499 | ✗ | goto fail; | |
| 500 | } | ||
| 501 | 6 | avio_printf(s->pb, "<Period id=\"0\""); | |
| 502 | 6 | avio_printf(s->pb, " start=\"PT%gS\"", start); | |
| 503 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 2 times.
|
6 | if (!w->is_live) { |
| 504 | 4 | avio_printf(s->pb, " duration=\"PT%gS\"", get_duration(s)); | |
| 505 | } | ||
| 506 | 6 | avio_printf(s->pb, " >\n"); | |
| 507 | |||
| 508 |
2/2✓ Branch 0 taken 9 times.
✓ Branch 1 taken 6 times.
|
15 | for (i = 0; i < w->nb_as; i++) { |
| 509 | 9 | ret = write_adaptation_set(s, i); | |
| 510 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 9 times.
|
9 | if (ret < 0) { |
| 511 | ✗ | goto fail; | |
| 512 | } | ||
| 513 | } | ||
| 514 | |||
| 515 | 6 | avio_printf(s->pb, "</Period>\n"); | |
| 516 | 6 | write_footer(s); | |
| 517 | 6 | fail: | |
| 518 | 6 | free_adaptation_sets(s); | |
| 519 | 6 | return ret < 0 ? ret : 0; | |
| 520 | } | ||
| 521 | |||
| 522 | ✗ | static int webm_dash_manifest_write_packet(AVFormatContext *s, AVPacket *pkt) | |
| 523 | { | ||
| 524 | ✗ | return AVERROR_EOF; | |
| 525 | } | ||
| 526 | |||
| 527 | #define OFFSET(x) offsetof(WebMDashMuxContext, x) | ||
| 528 | static const AVOption options[] = { | ||
| 529 | { "adaptation_sets", "Adaptation sets. Syntax: id=0,streams=0,1,2 id=1,streams=3,4 and so on", OFFSET(adaptation_sets), AV_OPT_TYPE_STRING, { 0 }, 0, 0, AV_OPT_FLAG_ENCODING_PARAM }, | ||
| 530 | { "live", "create a live stream manifest", OFFSET(is_live), AV_OPT_TYPE_BOOL, {.i64 = 0}, 0, 1, AV_OPT_FLAG_ENCODING_PARAM }, | ||
| 531 | { "chunk_start_index", "start index of the chunk", OFFSET(chunk_start_index), AV_OPT_TYPE_INT, {.i64 = 0}, 0, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM }, | ||
| 532 | { "chunk_duration_ms", "duration of each chunk (in milliseconds)", OFFSET(chunk_duration), AV_OPT_TYPE_INT, {.i64 = 1000}, 0, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM }, | ||
| 533 | { "utc_timing_url", "URL of the page that will return the UTC timestamp in ISO format", OFFSET(utc_timing_url), AV_OPT_TYPE_STRING, { 0 }, 0, 0, AV_OPT_FLAG_ENCODING_PARAM }, | ||
| 534 | { "time_shift_buffer_depth", "Smallest time (in seconds) shifting buffer for which any Representation is guaranteed to be available.", OFFSET(time_shift_buffer_depth), AV_OPT_TYPE_DOUBLE, { .dbl = 60.0 }, 1.0, DBL_MAX, AV_OPT_FLAG_ENCODING_PARAM }, | ||
| 535 | { "minimum_update_period", "Minimum Update Period (in seconds) of the manifest.", OFFSET(minimum_update_period), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, AV_OPT_FLAG_ENCODING_PARAM }, | ||
| 536 | { NULL }, | ||
| 537 | }; | ||
| 538 | |||
| 539 | static const AVClass webm_dash_class = { | ||
| 540 | .class_name = "WebM DASH Manifest muxer", | ||
| 541 | .item_name = av_default_item_name, | ||
| 542 | .option = options, | ||
| 543 | .version = LIBAVUTIL_VERSION_INT, | ||
| 544 | }; | ||
| 545 | |||
| 546 | const FFOutputFormat ff_webm_dash_manifest_muxer = { | ||
| 547 | .p.name = "webm_dash_manifest", | ||
| 548 | .p.long_name = NULL_IF_CONFIG_SMALL("WebM DASH Manifest"), | ||
| 549 | .p.mime_type = "application/xml", | ||
| 550 | .p.extensions = "xml", | ||
| 551 | .priv_data_size = sizeof(WebMDashMuxContext), | ||
| 552 | .write_header = webm_dash_manifest_write_header, | ||
| 553 | .write_packet = webm_dash_manifest_write_packet, | ||
| 554 | .p.priv_class = &webm_dash_class, | ||
| 555 | }; | ||
| 556 |