1 /*
2 * GXF demuxer.
3 * Copyright (c) 2006 Reimar Doeffinger
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
28
35 };
36
37 /**
38 * @brief parse gxf timecode and add it to metadata
39 */
41 {
42 char tmp[128];
43 int field = timecode & 0xff;
44 int frame = fields_per_frame ? field / fields_per_frame : field;
45 int second = (timecode >> 8) & 0xff;
46 int minute = (timecode >> 16) & 0xff;
47 int hour = (timecode >> 24) & 0x1f;
48 int drop = (timecode >> 29) & 1;
49 // bit 30: color_frame, unused
50 // ignore invalid time code
51 if (timecode >> 31)
52 return 0;
53 snprintf(tmp,
sizeof(tmp),
"%02d:%02d:%02d%c%02d",
54 hour, minute, second, drop ? ';' : ':', frame);
56 }
57
58 /**
59 * @brief parses a packet header, extracting type and length
60 * @param pb AVIOContext to read header from
61 * @param type detected packet type is stored here
62 * @param length detected packet length, excluding header is stored here
63 * @return 0 if header not found or contains invalid data, 1 otherwise
64 */
67 return 0;
69 return 0;
72 if ((*length >> 24) || *length < 16)
73 return 0;
74 *length -= 16;
76 return 0;
78 return 0;
80 return 0;
81 return 1;
82 }
83
84 /**
85 * @brief check if file starts with a PKT_MAP header
86 */
88 static const uint8_t startcode[] = {0, 0, 0, 0, 1, 0xbc};
// start with map packet
89 static const uint8_t endcode[] = {0, 0, 0, 0, 0xe1, 0xe2};
90 if (!memcmp(p->
buf, startcode,
sizeof(startcode)) &&
91 !memcmp(&p->
buf[16 -
sizeof(endcode)], endcode,
sizeof(endcode)))
93 return 0;
94 }
95
96 /**
97 * @brief gets the stream index for the track with the specified id, creates new
98 * stream if not found
99 * @param id id of stream to find / add
100 * @param format stream format identifier
101 */
103 int i;
106 if (i >= 0)
107 return i;
109 if (!st)
112 switch (format) {
113 case 3:
114 case 4:
117 break;
118 case 13:
119 case 14:
120 case 15:
121 case 16:
122 case 25:
125 break;
126 case 11:
127 case 12:
128 case 20:
132 break;
133 case 22:
134 case 23:
138 break;
139 case 9:
148 break;
149 case 10:
158 break;
159 case 17:
165 break;
166 case 26: /* AVCi50 / AVCi100 (AVC Intra) */
167 case 29: /* AVCHD */
171 break;
172 // timecode tracks:
173 case 7:
174 case 8:
175 case 24:
178 break;
179 case 30:
182 break;
183 default:
186 break;
187 }
189 }
190
191 /**
192 * @brief filters out interesting tags from material information.
193 * @param len length of tag section, will be adjusted to contain remaining bytes
194 * @param si struct to store collected information into
195 */
199 while (*len >= 2) {
202 *len -= 2;
203 if (tlen > *len)
204 return;
205 *len -= tlen;
206 if (tlen == 4) {
212 } else
214 }
215 }
216
218 { 60, 1},
219 {60000, 1001},
220 { 50, 1},
221 { 30, 1},
222 {30000, 1001},
223 { 25, 1},
224 { 24, 1},
225 {24000, 1001},
226 { 0, 0},
227 };
228
229 /**
230 * @brief convert fps tag value to AVRational fps
231 * @param fps fps value from tag
232 * @return fps as AVRational, or 0 / 0 if unknown
233 */
235 if (fps < 1 || fps > 9) fps = 9;
236 return frame_rate_tab[fps - 1];
237 }
238
239 /**
240 * @brief convert UMF attributes flags to AVRational fps
241 * @param flags UMF flags to convert
242 * @return fps as AVRational, or 0 / 0 if unknown
243 */
245 static const AVRational map[] = {{50, 1}, {60000, 1001}, {24, 1},
246 {25, 1}, {30000, 1001}};
247 int idx =
av_log2((flags & 0x7c0) >> 6);
248 return map[idx];
249 }
250
251 /**
252 * @brief filters out interesting tags from track information.
253 * @param len length of tag section, will be adjusted to contain remaining bytes
254 * @param si struct to store collected information into
255 */
260 while (*len >= 2) {
263 *len -= 2;
264 if (tlen > *len)
265 return;
266 *len -= tlen;
267 if (tlen == 4) {
271 else if (tag ==
TRACK_FPF && (value == 1 || value == 2))
273 }
else if (tlen == 8 && tag ==
TRACK_AUX)
275 else
277 }
278 }
279
280 /**
281 * @brief read index from FLT packet into stream 0 av_index
282 */
288 int i;
289 pkt_len -= 8;
292 return;
293 }
295 if (map_cnt > 1000) {
297 map_cnt = 1000;
298 }
299 if (pkt_len < 4 * map_cnt) {
302 return;
303 }
304 pkt_len -= 4 * map_cnt;
306 for (i = 0; i < map_cnt; i++)
308 i * (uint64_t)fields_per_map + 1, 0, 0, 0);
310 }
311
315 int map_len;
319 int i;
322 return 0;
323 }
324 map_len -= 2;
327 return 0;
328 }
329 map_len -= 2;
330 len =
avio_rb16(pb);
// length of material data section
331 if (len > map_len) {
333 return 0;
334 }
338 map_len -= 2;
339 len =
avio_rb16(pb);
// length of track description
340 if (len > map_len) {
342 return 0;
343 }
345 while (len > 0) {
346 int track_type, track_id, track_len;
348 int idx;
349 len -= 4;
353 len -= track_len;
354 if (!(track_type & 0x80)) {
356 continue;
357 }
358 track_type &= 0x7f;
359 if ((track_id & 0xc0) != 0xc0) {
361 continue;
362 }
363 track_id &= 0x3f;
365 // check for timecode tracks
366 if (track_type == 7 || track_type == 8 || track_type == 24) {
370
371 }
373
375 if (idx < 0) continue;
377 if (!main_timebase.
num || !main_timebase.
den) {
380 }
384 }
385 if (len < 0)
387 if (map_len)
391 return -1;
392 }
397 return -1;
398 }
399 }
401 if (len >= 0x39) {
403 len -= 0x39;
405 avio_skip(pb, 0x30);
// payload description
407 if (!main_timebase.
num || !main_timebase.
den) {
409 " This might give wrong results.\n");
410 // this may not always be correct, but simply the best we can get
411 main_timebase.
num = fps.
den;
412 main_timebase.
den = fps.
num * 2;
413 }
414
415 if (len >= 0x18) {
416 len -= 0x18;
422 }
423 } else
425 } else
428 // set a fallback value, 60000/1001 is specified for audio-only files
429 // so use that regardless of why we do not know the video frame rate.
430 if (!main_timebase.
num || !main_timebase.
den)
435 }
436 return 0;
437 }
438
440 { \
441 if (!max_interval-- || url_feof(pb)) \
442 goto out; \
443 tmp = tmp << 8 | avio_r8(pb); \
444 }
445
446 /**
447 * @brief resync the stream on the next media packet with specified properties
448 * @param max_interval how many bytes to search for matching packet at most
449 * @param track track id the media packet must belong to, -1 for any
450 * @param timestamp minimum timestamp (== field number) the packet must have, -1 for any
451 * @return timestamp of packet found
452 */
454 uint32_t tmp;
455 uint64_t last_pos;
456 uint64_t last_found_pos = 0;
457 int cur_track;
464 while (tmp)
467 if (tmp != 1)
473 if (
avio_seek(pb, last_pos, SEEK_SET) < 0)
476 }
481 if ((track >= 0 && track != cur_track) || (timestamp >= 0 && timestamp > cur_timestamp)) {
482 if (
avio_seek(pb, last_pos, SEEK_SET) >= 0)
484 }
486 if (last_found_pos)
488 return cur_timestamp;
489 }
490
494 int pkt_len;
496
499 int track_type, track_id,
ret;
500 int field_nr, field_info, skip = 0;
501 int stream_index;
505 return -1;
506 }
509 continue;
510 }
513 continue;
514 }
515 if (pkt_len < 16) {
517 continue;
518 }
519 pkt_len -= 16;
522 stream_index =
get_sindex(s, track_id, track_type);
523 if (stream_index < 0)
524 return stream_index;
528 avio_rb32(pb);
// "timeline" field number
533 int first = field_info >> 16;
534 int last = field_info & 0xffff; // last is exclusive
536 if (first <= last && last*bps <= pkt_len) {
538 skip = pkt_len - last*
bps;
539 pkt_len = (last-first)*bps;
540 } else
542 }
544 if (skip)
548
549 //set duration manually for DV or else lavf misdetects the frame rate
552
554 }
556 }
557
560 uint64_t pos;
561 uint64_t maxlen = 100 * 1024 * 1024;
564 int64_t found;
565 int idx;
566 if (timestamp < start_time) timestamp =
start_time;
569 if (idx < 0)
570 return -1;
572 if (idx < st->nb_index_entries - 2)
574 maxlen =
FFMAX(maxlen, 200 * 1024);
576 if (res < 0)
579 if (
FFABS(found - timestamp) > 4)
580 return -1;
581 return 0;
582 }
583
585 int64_t *pos, int64_t pos_limit) {
593 }
594
604 };