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 15:
122 break;
123 case 14:
124 case 16:
127 break;
128 case 11:
129 case 12:
130 case 20:
134 break;
135 case 22:
136 case 23:
140 break;
141 case 9:
150 break;
151 case 10:
160 break;
161 case 17:
167 break;
168 // timecode tracks:
169 case 7:
170 case 8:
171 case 24:
174 break;
175 default:
178 break;
179 }
181 }
182
183 /**
184 * @brief filters out interesting tags from material information.
185 * @param len length of tag section, will be adjusted to contain remaining bytes
186 * @param si struct to store collected information into
187 */
191 while (*len >= 2) {
194 *len -= 2;
195 if (tlen > *len)
196 return;
197 *len -= tlen;
198 if (tlen == 4) {
204 } else
206 }
207 }
208
210 { 60, 1},
211 {60000, 1001},
212 { 50, 1},
213 { 30, 1},
214 {30000, 1001},
215 { 25, 1},
216 { 24, 1},
217 {24000, 1001},
218 { 0, 0},
219 };
220
221 /**
222 * @brief convert fps tag value to AVRational fps
223 * @param fps fps value from tag
224 * @return fps as AVRational, or 0 / 0 if unknown
225 */
227 if (fps < 1 || fps > 9) fps = 9;
228 return frame_rate_tab[fps - 1];
229 }
230
231 /**
232 * @brief convert UMF attributes flags to AVRational fps
233 * @param flags UMF flags to convert
234 * @return fps as AVRational, or 0 / 0 if unknown
235 */
237 static const AVRational map[] = {{50, 1}, {60000, 1001}, {24, 1},
238 {25, 1}, {30000, 1001}};
239 int idx =
av_log2((flags & 0x7c0) >> 6);
240 return map[idx];
241 }
242
243 /**
244 * @brief filters out interesting tags from track information.
245 * @param len length of tag section, will be adjusted to contain remaining bytes
246 * @param si struct to store collected information into
247 */
252 while (*len >= 2) {
255 *len -= 2;
256 if (tlen > *len)
257 return;
258 *len -= tlen;
259 if (tlen == 4) {
263 else if (tag ==
TRACK_FPF && (value == 1 || value == 2))
265 }
else if (tlen == 8 && tag ==
TRACK_AUX)
267 else
269 }
270 }
271
272 /**
273 * @brief read index from FLT packet into stream 0 av_index
274 */
280 int i;
281 pkt_len -= 8;
284 return;
285 }
287 if (map_cnt > 1000) {
289 map_cnt = 1000;
290 }
291 if (pkt_len < 4 * map_cnt) {
294 return;
295 }
296 pkt_len -= 4 * map_cnt;
298 for (i = 0; i < map_cnt; i++)
300 i * (uint64_t)fields_per_map + 1, 0, 0, 0);
302 }
303
307 int map_len;
311 int i;
314 return 0;
315 }
316 map_len -= 2;
319 return 0;
320 }
321 map_len -= 2;
322 len =
avio_rb16(pb);
// length of material data section
323 if (len > map_len) {
325 return 0;
326 }
330 map_len -= 2;
331 len =
avio_rb16(pb);
// length of track description
332 if (len > map_len) {
334 return 0;
335 }
337 while (len > 0) {
338 int track_type, track_id, track_len;
340 int idx;
341 len -= 4;
345 len -= track_len;
346 if (!(track_type & 0x80)) {
348 continue;
349 }
350 track_type &= 0x7f;
351 if ((track_id & 0xc0) != 0xc0) {
353 continue;
354 }
355 track_id &= 0x3f;
357 // check for timecode tracks
358 if (track_type == 7 || track_type == 8 || track_type == 24) {
362
363 }
365
367 if (idx < 0) continue;
369 if (!main_timebase.
num || !main_timebase.
den) {
372 }
376 }
377 if (len < 0)
379 if (map_len)
383 return -1;
384 }
389 return -1;
390 }
391 }
393 if (len >= 0x39) {
395 len -= 0x39;
397 avio_skip(pb, 0x30);
// payload description
399 if (!main_timebase.
num || !main_timebase.
den) {
401 " This might give wrong results.\n");
402 // this may not always be correct, but simply the best we can get
403 main_timebase.
num = fps.
den;
404 main_timebase.
den = fps.
num * 2;
405 }
406
407 if (len >= 0x18) {
408 len -= 0x18;
414 }
415 } else
417 } else
420 // set a fallback value, 60000/1001 is specified for audio-only files
421 // so use that regardless of why we do not know the video frame rate.
422 if (!main_timebase.
num || !main_timebase.
den)
427 }
428 return 0;
429 }
430
432 { \
433 if (!max_interval-- || url_feof(pb)) \
434 goto out; \
435 tmp = tmp << 8 | avio_r8(pb); \
436 }
437
438 /**
439 * @brief resync the stream on the next media packet with specified properties
440 * @param max_interval how many bytes to search for matching packet at most
441 * @param track track id the media packet must belong to, -1 for any
442 * @param timestamp minimum timestamp (== field number) the packet must have, -1 for any
443 * @return timestamp of packet found
444 */
446 uint32_t tmp;
447 uint64_t last_pos;
448 uint64_t last_found_pos = 0;
449 int cur_track;
455 start:
456 while (tmp)
459 if (tmp != 1)
460 goto start;
463 goto out;
465 if (
avio_seek(pb, last_pos, SEEK_SET) < 0)
466 goto out;
467 goto start;
468 }
473 if ((track >= 0 && track != cur_track) || (timestamp >= 0 && timestamp > cur_timestamp)) {
474 if (
avio_seek(pb, last_pos, SEEK_SET) >= 0)
475 goto start;
476 }
477 out:
478 if (last_found_pos)
480 return cur_timestamp;
481 }
482
486 int pkt_len;
488
491 int track_type, track_id, ret;
492 int field_nr, field_info, skip = 0;
493 int stream_index;
497 return -1;
498 }
501 continue;
502 }
505 continue;
506 }
507 if (pkt_len < 16) {
509 continue;
510 }
511 pkt_len -= 16;
514 stream_index =
get_sindex(s, track_id, track_type);
515 if (stream_index < 0)
516 return stream_index;
520 avio_rb32(pb);
// "timeline" field number
525 int first = field_info >> 16;
526 int last = field_info & 0xffff; // last is exclusive
528 if (first <= last && last*bps <= pkt_len) {
530 skip = pkt_len - last*
bps;
531 pkt_len = (last-first)*bps;
532 } else
534 }
536 if (skip)
540
541 //set duration manually for DV or else lavf misdetects the frame rate
544
545 return ret;
546 }
548 }
549
551 int res = 0;
552 uint64_t pos;
553 uint64_t maxlen = 100 * 1024 * 1024;
556 int64_t found;
557 int idx;
558 if (timestamp < start_time) timestamp =
start_time;
561 if (idx < 0)
562 return -1;
564 if (idx < st->nb_index_entries - 2)
566 maxlen =
FFMAX(maxlen, 200 * 1024);
568 if (res < 0)
569 return res;
571 if (
FFABS(found - timestamp) > 4)
572 return -1;
573 return 0;
574 }
575
577 int64_t *pos, int64_t pos_limit) {
579 int64_t res;
584 return res;
585 }
586
596 };