2002-04-08 15:32:01 +03:00
|
|
|
/*
|
2006-10-23 11:57:54 +03:00
|
|
|
* MOV demuxer
|
2009-01-19 17:46:40 +02:00
|
|
|
* Copyright (c) 2001 Fabrice Bellard
|
2009-03-15 12:53:12 +02:00
|
|
|
* Copyright (c) 2009 Baptiste Coudurier <baptiste dot coudurier at gmail dot com>
|
2002-04-08 15:32:01 +03:00
|
|
|
*
|
2013-04-10 20:26:49 +03:00
|
|
|
* first version by Francois Revol <revol@free.fr>
|
|
|
|
* seek function by Gael Chardon <gael.dev@4now.net>
|
|
|
|
*
|
2011-03-18 19:35:10 +02:00
|
|
|
* This file is part of Libav.
|
2006-10-07 18:30:46 +03:00
|
|
|
*
|
2011-03-18 19:35:10 +02:00
|
|
|
* Libav is free software; you can redistribute it and/or
|
2002-05-26 01:34:32 +03:00
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
2006-10-07 18:30:46 +03:00
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2002-04-08 15:32:01 +03:00
|
|
|
*
|
2011-03-18 19:35:10 +02:00
|
|
|
* Libav is distributed in the hope that it will be useful,
|
2002-04-08 15:32:01 +03:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2002-05-26 01:34:32 +03:00
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
2002-04-08 15:32:01 +03:00
|
|
|
*
|
2002-05-26 01:34:32 +03:00
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2011-03-18 19:35:10 +02:00
|
|
|
* License along with Libav; if not, write to the Free Software
|
2006-01-13 00:43:26 +02:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
2002-04-08 15:32:01 +03:00
|
|
|
*/
|
2004-03-24 21:03:20 +02:00
|
|
|
|
2014-03-10 17:35:59 +03:00
|
|
|
#include <inttypes.h>
|
2004-03-24 21:03:20 +02:00
|
|
|
#include <limits.h>
|
2013-11-23 23:32:55 +03:00
|
|
|
#include <stdint.h>
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2012-04-19 15:48:16 +03:00
|
|
|
#include "libavutil/attributes.h"
|
2012-11-10 18:00:00 +03:00
|
|
|
#include "libavutil/channel_layout.h"
|
2009-01-12 00:19:48 +02:00
|
|
|
#include "libavutil/intreadwrite.h"
|
2011-11-27 17:04:16 +03:00
|
|
|
#include "libavutil/intfloat.h"
|
2011-06-04 14:58:23 +03:00
|
|
|
#include "libavutil/mathematics.h"
|
2014-10-24 10:46:36 +03:00
|
|
|
#include "libavutil/time_internal.h"
|
2009-02-15 04:28:54 +02:00
|
|
|
#include "libavutil/avstring.h"
|
2011-05-22 13:46:29 +03:00
|
|
|
#include "libavutil/dict.h"
|
2014-11-29 18:51:13 +02:00
|
|
|
#include "libavutil/opt.h"
|
2016-10-19 00:21:31 +02:00
|
|
|
#include "libavutil/pixdesc.h"
|
2016-11-02 17:29:15 +02:00
|
|
|
#include "libavutil/spherical.h"
|
|
|
|
#include "libavutil/stereo3d.h"
|
2016-04-16 12:17:24 +02:00
|
|
|
|
2012-02-12 23:06:58 +03:00
|
|
|
#include "libavcodec/ac3tab.h"
|
2016-04-16 12:17:24 +02:00
|
|
|
#include "libavcodec/bitstream.h"
|
|
|
|
|
2002-04-08 15:32:01 +03:00
|
|
|
#include "avformat.h"
|
2011-11-29 21:28:15 +03:00
|
|
|
#include "internal.h"
|
2011-02-20 12:04:13 +02:00
|
|
|
#include "avio_internal.h"
|
2006-07-12 03:09:34 +03:00
|
|
|
#include "riff.h"
|
2006-08-01 17:58:15 +03:00
|
|
|
#include "isom.h"
|
2011-10-21 16:51:33 +03:00
|
|
|
#include "id3v1.h"
|
2011-12-03 21:32:45 +03:00
|
|
|
#include "mov_chan.h"
|
2014-04-02 00:22:21 +03:00
|
|
|
#include "replaygain.h"
|
2002-04-08 15:32:01 +03:00
|
|
|
|
2009-01-14 01:44:16 +02:00
|
|
|
#if CONFIG_ZLIB
|
2002-07-20 23:05:50 +03:00
|
|
|
#include <zlib.h>
|
|
|
|
#endif
|
|
|
|
|
2003-11-10 05:17:32 +02:00
|
|
|
#include "qtpalette.h"
|
|
|
|
|
2004-07-06 05:35:39 +03:00
|
|
|
|
2002-04-08 15:32:01 +03:00
|
|
|
/* those functions parse an atom */
|
|
|
|
/* links atom IDs to parse functions */
|
|
|
|
typedef struct MOVParseTableEntry {
|
2003-02-11 18:35:48 +02:00
|
|
|
uint32_t type;
|
2011-02-20 12:04:12 +02:00
|
|
|
int (*parse)(MOVContext *ctx, AVIOContext *pb, MOVAtom atom);
|
2002-04-08 15:32:01 +03:00
|
|
|
} MOVParseTableEntry;
|
|
|
|
|
2012-07-04 06:49:49 +03:00
|
|
|
static int mov_read_default(MOVContext *c, AVIOContext *pb, MOVAtom atom);
|
2007-12-16 20:20:27 +02:00
|
|
|
|
2011-10-11 10:53:39 +03:00
|
|
|
static int mov_metadata_track_or_disc_number(MOVContext *c, AVIOContext *pb,
|
|
|
|
unsigned len, const char *key)
|
2009-03-16 08:43:38 +02:00
|
|
|
{
|
|
|
|
char buf[16];
|
|
|
|
|
2011-11-17 16:12:34 +03:00
|
|
|
short current, total = 0;
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb16(pb); // unknown
|
2011-10-11 10:53:39 +03:00
|
|
|
current = avio_rb16(pb);
|
2011-11-17 16:12:34 +03:00
|
|
|
if (len >= 6)
|
|
|
|
total = avio_rb16(pb);
|
2011-10-11 10:53:39 +03:00
|
|
|
if (!total)
|
|
|
|
snprintf(buf, sizeof(buf), "%d", current);
|
|
|
|
else
|
|
|
|
snprintf(buf, sizeof(buf), "%d/%d", current, total);
|
2014-08-13 00:03:54 +03:00
|
|
|
c->fc->event_flags |= AVFMT_EVENT_FLAG_METADATA_UPDATED;
|
2011-10-11 10:53:39 +03:00
|
|
|
av_dict_set(&c->fc->metadata, key, buf, 0);
|
2009-03-16 08:43:38 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-10-21 16:04:13 +03:00
|
|
|
static int mov_metadata_int8_bypass_padding(MOVContext *c, AVIOContext *pb,
|
|
|
|
unsigned len, const char *key)
|
2011-10-14 20:14:18 +03:00
|
|
|
{
|
2011-10-21 16:04:13 +03:00
|
|
|
char buf[16];
|
2011-10-14 20:14:18 +03:00
|
|
|
|
2011-10-21 16:04:13 +03:00
|
|
|
/* bypass padding bytes */
|
|
|
|
avio_r8(pb);
|
|
|
|
avio_r8(pb);
|
|
|
|
avio_r8(pb);
|
2011-10-14 20:14:18 +03:00
|
|
|
|
2011-11-27 06:41:54 +03:00
|
|
|
snprintf(buf, sizeof(buf), "%d", avio_r8(pb));
|
2014-08-13 00:03:54 +03:00
|
|
|
c->fc->event_flags |= AVFMT_EVENT_FLAG_METADATA_UPDATED;
|
2011-10-21 16:04:13 +03:00
|
|
|
av_dict_set(&c->fc->metadata, key, buf, 0);
|
2011-10-14 20:14:18 +03:00
|
|
|
|
2011-10-21 16:04:13 +03:00
|
|
|
return 0;
|
2011-10-14 20:14:18 +03:00
|
|
|
}
|
|
|
|
|
2011-10-21 16:04:14 +03:00
|
|
|
static int mov_metadata_int8_no_padding(MOVContext *c, AVIOContext *pb,
|
|
|
|
unsigned len, const char *key)
|
2011-10-14 20:14:18 +03:00
|
|
|
{
|
2011-10-21 16:04:14 +03:00
|
|
|
char buf[16];
|
2011-10-14 20:14:18 +03:00
|
|
|
|
2011-11-27 06:41:54 +03:00
|
|
|
snprintf(buf, sizeof(buf), "%d", avio_r8(pb));
|
2014-08-13 00:03:54 +03:00
|
|
|
c->fc->event_flags |= AVFMT_EVENT_FLAG_METADATA_UPDATED;
|
2011-10-21 16:04:14 +03:00
|
|
|
av_dict_set(&c->fc->metadata, key, buf, 0);
|
2011-10-14 20:14:18 +03:00
|
|
|
|
2011-10-21 16:04:14 +03:00
|
|
|
return 0;
|
2011-10-14 20:14:18 +03:00
|
|
|
}
|
|
|
|
|
2011-10-21 16:51:33 +03:00
|
|
|
static int mov_metadata_gnre(MOVContext *c, AVIOContext *pb,
|
|
|
|
unsigned len, const char *key)
|
|
|
|
{
|
|
|
|
short genre;
|
|
|
|
char buf[20];
|
|
|
|
|
|
|
|
avio_r8(pb); // unknown
|
|
|
|
|
|
|
|
genre = avio_r8(pb);
|
|
|
|
if (genre < 1 || genre > ID3v1_GENRE_MAX)
|
|
|
|
return 0;
|
|
|
|
snprintf(buf, sizeof(buf), "%s", ff_id3v1_genre_str[genre-1]);
|
2014-08-13 00:03:54 +03:00
|
|
|
c->fc->event_flags |= AVFMT_EVENT_FLAG_METADATA_UPDATED;
|
2011-10-21 16:51:33 +03:00
|
|
|
av_dict_set(&c->fc->metadata, key, buf, 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-03-09 03:03:42 +02:00
|
|
|
static const uint32_t mac_to_unicode[128] = {
|
|
|
|
0x00C4,0x00C5,0x00C7,0x00C9,0x00D1,0x00D6,0x00DC,0x00E1,
|
|
|
|
0x00E0,0x00E2,0x00E4,0x00E3,0x00E5,0x00E7,0x00E9,0x00E8,
|
|
|
|
0x00EA,0x00EB,0x00ED,0x00EC,0x00EE,0x00EF,0x00F1,0x00F3,
|
|
|
|
0x00F2,0x00F4,0x00F6,0x00F5,0x00FA,0x00F9,0x00FB,0x00FC,
|
|
|
|
0x2020,0x00B0,0x00A2,0x00A3,0x00A7,0x2022,0x00B6,0x00DF,
|
|
|
|
0x00AE,0x00A9,0x2122,0x00B4,0x00A8,0x2260,0x00C6,0x00D8,
|
|
|
|
0x221E,0x00B1,0x2264,0x2265,0x00A5,0x00B5,0x2202,0x2211,
|
|
|
|
0x220F,0x03C0,0x222B,0x00AA,0x00BA,0x03A9,0x00E6,0x00F8,
|
|
|
|
0x00BF,0x00A1,0x00AC,0x221A,0x0192,0x2248,0x2206,0x00AB,
|
|
|
|
0x00BB,0x2026,0x00A0,0x00C0,0x00C3,0x00D5,0x0152,0x0153,
|
|
|
|
0x2013,0x2014,0x201C,0x201D,0x2018,0x2019,0x00F7,0x25CA,
|
|
|
|
0x00FF,0x0178,0x2044,0x20AC,0x2039,0x203A,0xFB01,0xFB02,
|
|
|
|
0x2021,0x00B7,0x201A,0x201E,0x2030,0x00C2,0x00CA,0x00C1,
|
|
|
|
0x00CB,0x00C8,0x00CD,0x00CE,0x00CF,0x00CC,0x00D3,0x00D4,
|
|
|
|
0xF8FF,0x00D2,0x00DA,0x00DB,0x00D9,0x0131,0x02C6,0x02DC,
|
|
|
|
0x00AF,0x02D8,0x02D9,0x02DA,0x00B8,0x02DD,0x02DB,0x02C7,
|
|
|
|
};
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_mac_string(MOVContext *c, AVIOContext *pb, int len,
|
2010-03-09 03:03:42 +02:00
|
|
|
char *dst, int dstlen)
|
|
|
|
{
|
|
|
|
char *p = dst;
|
|
|
|
char *end = dst+dstlen-1;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
2011-02-21 17:43:01 +02:00
|
|
|
uint8_t t, c = avio_r8(pb);
|
2016-12-17 15:17:20 +02:00
|
|
|
|
|
|
|
if (p >= end)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (c < 0x80)
|
2010-03-09 03:03:42 +02:00
|
|
|
*p++ = c;
|
|
|
|
else
|
|
|
|
PUT_UTF8(mac_to_unicode[c-0x80], t, if (p < end) *p++ = t;);
|
|
|
|
}
|
|
|
|
*p = 0;
|
|
|
|
return p - dst;
|
|
|
|
}
|
|
|
|
|
2012-06-21 19:24:27 +03:00
|
|
|
static int mov_read_covr(MOVContext *c, AVIOContext *pb, int type, int len)
|
|
|
|
{
|
|
|
|
AVPacket pkt;
|
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2012-08-05 12:11:04 +03:00
|
|
|
enum AVCodecID id;
|
2012-06-21 19:24:27 +03:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
switch (type) {
|
2012-08-05 12:11:04 +03:00
|
|
|
case 0xd: id = AV_CODEC_ID_MJPEG; break;
|
|
|
|
case 0xe: id = AV_CODEC_ID_PNG; break;
|
|
|
|
case 0x1b: id = AV_CODEC_ID_BMP; break;
|
2012-06-21 19:24:27 +03:00
|
|
|
default:
|
|
|
|
av_log(c->fc, AV_LOG_WARNING, "Unknown cover type: 0x%x.\n", type);
|
|
|
|
avio_skip(pb, len);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
st = avformat_new_stream(c->fc, NULL);
|
|
|
|
if (!st)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
sc = av_mallocz(sizeof(*sc));
|
|
|
|
if (!sc)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
st->priv_data = sc;
|
|
|
|
|
|
|
|
ret = av_get_packet(pb, &pkt, len);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2015-10-31 15:36:38 +02:00
|
|
|
if (pkt.size >= 8 && id != AV_CODEC_ID_BMP) {
|
|
|
|
if (AV_RB64(pkt.data) == 0x89504e470d0a1a0a) {
|
|
|
|
id = AV_CODEC_ID_PNG;
|
|
|
|
} else {
|
|
|
|
id = AV_CODEC_ID_MJPEG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-21 19:24:27 +03:00
|
|
|
st->disposition |= AV_DISPOSITION_ATTACHED_PIC;
|
|
|
|
|
|
|
|
st->attached_pic = pkt;
|
|
|
|
st->attached_pic.stream_index = st->index;
|
|
|
|
st->attached_pic.flags |= AV_PKT_FLAG_KEY;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_VIDEO;
|
|
|
|
st->codecpar->codec_id = id;
|
2012-06-21 19:24:27 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-05-18 12:33:55 +02:00
|
|
|
// 3GPP TS 26.244
|
2014-06-03 14:48:19 +03:00
|
|
|
static int mov_metadata_loci(MOVContext *c, AVIOContext *pb, unsigned len)
|
|
|
|
{
|
|
|
|
char language[4] = { 0 };
|
2015-05-30 15:23:19 +02:00
|
|
|
char buf[200], place[100];
|
2014-06-03 14:48:19 +03:00
|
|
|
uint16_t langcode = 0;
|
2015-05-30 15:28:31 +02:00
|
|
|
double longitude, latitude, altitude;
|
2014-06-03 14:48:19 +03:00
|
|
|
const char *key = "location";
|
|
|
|
|
2015-05-16 13:50:38 +02:00
|
|
|
if (len < 4 + 2 + 1 + 1 + 4 + 4 + 4) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "loci too short\n");
|
2014-06-03 14:48:19 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2015-05-16 13:50:38 +02:00
|
|
|
}
|
2014-06-03 14:48:19 +03:00
|
|
|
|
|
|
|
avio_skip(pb, 4); // version+flags
|
|
|
|
langcode = avio_rb16(pb);
|
|
|
|
ff_mov_lang_to_iso639(langcode, language);
|
|
|
|
len -= 6;
|
|
|
|
|
2015-05-30 15:23:19 +02:00
|
|
|
len -= avio_get_str(pb, len, place, sizeof(place));
|
2015-05-16 13:50:38 +02:00
|
|
|
if (len < 1) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "place name too long\n");
|
2014-06-03 14:48:19 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2015-05-16 13:50:38 +02:00
|
|
|
}
|
2014-06-03 14:48:19 +03:00
|
|
|
avio_skip(pb, 1); // role
|
|
|
|
len -= 1;
|
|
|
|
|
2015-05-16 13:51:18 +02:00
|
|
|
if (len < 12) {
|
2015-05-16 13:50:38 +02:00
|
|
|
av_log(c->fc, AV_LOG_ERROR,
|
2015-05-16 13:51:18 +02:00
|
|
|
"loci too short (%u bytes left, need at least %d)\n", len, 12);
|
2014-06-03 14:48:19 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2015-05-16 13:50:38 +02:00
|
|
|
}
|
2014-06-03 14:48:19 +03:00
|
|
|
longitude = ((int32_t) avio_rb32(pb)) / (float) (1 << 16);
|
|
|
|
latitude = ((int32_t) avio_rb32(pb)) / (float) (1 << 16);
|
2015-05-30 15:28:31 +02:00
|
|
|
altitude = ((int32_t) avio_rb32(pb)) / (float) (1 << 16);
|
2014-06-03 14:48:19 +03:00
|
|
|
|
|
|
|
// Try to output in the same format as the ?xyz field
|
2015-05-30 15:28:31 +02:00
|
|
|
snprintf(buf, sizeof(buf), "%+08.4f%+09.4f", latitude, longitude);
|
|
|
|
if (altitude)
|
|
|
|
av_strlcatf(buf, sizeof(buf), "%+f", altitude);
|
|
|
|
av_strlcatf(buf, sizeof(buf), "/%s", place);
|
|
|
|
|
2014-06-03 14:48:19 +03:00
|
|
|
if (*language && strcmp(language, "und")) {
|
|
|
|
char key2[16];
|
|
|
|
snprintf(key2, sizeof(key2), "%s-%s", key, language);
|
|
|
|
av_dict_set(&c->fc->metadata, key2, buf, 0);
|
|
|
|
}
|
2014-08-13 00:03:54 +03:00
|
|
|
c->fc->event_flags |= AVFMT_EVENT_FLAG_METADATA_UPDATED;
|
2014-06-03 14:48:19 +03:00
|
|
|
return av_dict_set(&c->fc->metadata, key, buf, 0);
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_udta_string(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2009-03-16 08:06:05 +02:00
|
|
|
{
|
|
|
|
char tmp_key[5];
|
2014-11-29 18:51:14 +02:00
|
|
|
char *str, key2[32], language[4] = {0};
|
2009-03-16 08:06:05 +02:00
|
|
|
const char *key = NULL;
|
2012-06-21 19:13:34 +03:00
|
|
|
uint16_t langcode = 0;
|
2014-11-29 18:51:14 +02:00
|
|
|
uint32_t data_type = 0, str_size, str_size_alloc;
|
2011-10-11 10:53:39 +03:00
|
|
|
int (*parse)(MOVContext*, AVIOContext*, unsigned, const char*) = NULL;
|
2014-12-03 04:42:29 +02:00
|
|
|
int raw = 0;
|
2009-03-16 08:06:05 +02:00
|
|
|
|
|
|
|
switch (atom.type) {
|
2014-12-03 04:42:29 +02:00
|
|
|
case MKTAG( '@','P','R','M'): key = "premiere_version"; raw = 1; break;
|
|
|
|
case MKTAG( '@','P','R','Q'): key = "quicktime_version"; raw = 1; break;
|
2014-12-03 04:43:24 +02:00
|
|
|
case MKTAG( 'X','M','P','_'):
|
|
|
|
if (c->export_xmp) { key = "xmp"; raw = 1; } break;
|
2011-10-11 10:53:40 +03:00
|
|
|
case MKTAG( 'a','A','R','T'): key = "album_artist"; break;
|
2014-11-29 01:25:20 +02:00
|
|
|
case MKTAG( 'a','k','I','D'): key = "account_type";
|
|
|
|
parse = mov_metadata_int8_no_padding; break;
|
|
|
|
case MKTAG( 'a','p','I','D'): key = "account_id"; break;
|
|
|
|
case MKTAG( 'c','a','t','g'): key = "category"; break;
|
|
|
|
case MKTAG( 'c','p','i','l'): key = "compilation";
|
|
|
|
parse = mov_metadata_int8_no_padding; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG( 'c','p','r','t'): key = "copyright"; break;
|
|
|
|
case MKTAG( 'd','e','s','c'): key = "description"; break;
|
|
|
|
case MKTAG( 'd','i','s','k'): key = "disc";
|
|
|
|
parse = mov_metadata_track_or_disc_number; break;
|
2014-11-29 01:25:20 +02:00
|
|
|
case MKTAG( 'e','g','i','d'): key = "episode_uid";
|
|
|
|
parse = mov_metadata_int8_no_padding; break;
|
2011-10-21 16:51:33 +03:00
|
|
|
case MKTAG( 'g','n','r','e'): key = "genre";
|
|
|
|
parse = mov_metadata_gnre; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG( 'h','d','v','d'): key = "hd_video";
|
|
|
|
parse = mov_metadata_int8_no_padding; break;
|
2014-11-29 01:25:20 +02:00
|
|
|
case MKTAG( 'k','e','y','w'): key = "keywords"; break;
|
2009-11-23 10:47:44 +02:00
|
|
|
case MKTAG( 'l','d','e','s'): key = "synopsis"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG( 'l','o','c','i'):
|
|
|
|
return mov_metadata_loci(c, pb, atom.size);
|
2014-11-29 01:25:20 +02:00
|
|
|
case MKTAG( 'p','c','s','t'): key = "podcast";
|
|
|
|
parse = mov_metadata_int8_no_padding; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG( 'p','g','a','p'): key = "gapless_playback";
|
|
|
|
parse = mov_metadata_int8_no_padding; break;
|
2014-11-29 01:25:20 +02:00
|
|
|
case MKTAG( 'p','u','r','d'): key = "purchase_date"; break;
|
|
|
|
case MKTAG( 'r','t','n','g'): key = "rating";
|
|
|
|
parse = mov_metadata_int8_no_padding; break;
|
|
|
|
case MKTAG( 's','o','a','a'): key = "sort_album_artist"; break;
|
|
|
|
case MKTAG( 's','o','a','l'): key = "sort_album"; break;
|
|
|
|
case MKTAG( 's','o','a','r'): key = "sort_artist"; break;
|
|
|
|
case MKTAG( 's','o','c','o'): key = "sort_composer"; break;
|
|
|
|
case MKTAG( 's','o','n','m'): key = "sort_name"; break;
|
|
|
|
case MKTAG( 's','o','s','n'): key = "sort_show"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG( 's','t','i','k'): key = "media_type";
|
|
|
|
parse = mov_metadata_int8_no_padding; break;
|
2009-03-16 08:43:38 +02:00
|
|
|
case MKTAG( 't','r','k','n'): key = "track";
|
2011-10-11 10:53:39 +03:00
|
|
|
parse = mov_metadata_track_or_disc_number; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG( 't','v','e','n'): key = "episode_id"; break;
|
2011-10-14 20:14:18 +03:00
|
|
|
case MKTAG( 't','v','e','s'): key = "episode_sort";
|
2011-10-21 16:04:13 +03:00
|
|
|
parse = mov_metadata_int8_bypass_padding; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG( 't','v','n','n'): key = "network"; break;
|
|
|
|
case MKTAG( 't','v','s','h'): key = "show"; break;
|
2011-10-14 20:14:18 +03:00
|
|
|
case MKTAG( 't','v','s','n'): key = "season_number";
|
2011-10-21 16:04:13 +03:00
|
|
|
parse = mov_metadata_int8_bypass_padding; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'A','R','T'): key = "artist"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'P','R','D'): key = "producer"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'a','l','b'): key = "album"; break;
|
|
|
|
case MKTAG(0xa9,'a','u','t'): key = "artist"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'c','h','p'): key = "chapter"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'c','m','t'): key = "comment"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'c','o','m'): key = "composer"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'c','p','y'): key = "copyright"; break;
|
|
|
|
case MKTAG(0xa9,'d','a','y'): key = "date"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'d','i','r'): key = "director"; break;
|
|
|
|
case MKTAG(0xa9,'d','i','s'): key = "disclaimer"; break;
|
|
|
|
case MKTAG(0xa9,'e','d','1'): key = "edit_date"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'e','n','c'): key = "encoder"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'f','m','t'): key = "original_format"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'g','e','n'): key = "genre"; break;
|
2014-11-29 01:25:20 +02:00
|
|
|
case MKTAG(0xa9,'g','r','p'): key = "grouping"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'h','s','t'): key = "host_computer"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'i','n','f'): key = "comment"; break;
|
2014-11-29 01:25:20 +02:00
|
|
|
case MKTAG(0xa9,'l','y','r'): key = "lyrics"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'m','a','k'): key = "make"; break;
|
|
|
|
case MKTAG(0xa9,'m','o','d'): key = "model"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'n','a','m'): key = "title"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'o','p','e'): key = "original_artist"; break;
|
|
|
|
case MKTAG(0xa9,'p','r','d'): key = "producer"; break;
|
|
|
|
case MKTAG(0xa9,'p','r','f'): key = "performers"; break;
|
|
|
|
case MKTAG(0xa9,'r','e','q'): key = "playback_requirements"; break;
|
|
|
|
case MKTAG(0xa9,'s','r','c'): key = "original_source"; break;
|
2014-11-29 01:25:20 +02:00
|
|
|
case MKTAG(0xa9,'s','t','3'): key = "subtitle"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'s','w','r'): key = "encoder"; break;
|
|
|
|
case MKTAG(0xa9,'t','o','o'): key = "encoder"; break;
|
2014-11-29 01:44:06 +02:00
|
|
|
case MKTAG(0xa9,'t','r','k'): key = "track"; break;
|
|
|
|
case MKTAG(0xa9,'u','r','l'): key = "URL"; break;
|
|
|
|
case MKTAG(0xa9,'w','r','n'): key = "warning"; break;
|
2014-12-03 04:41:43 +02:00
|
|
|
case MKTAG(0xa9,'w','r','t'): key = "composer"; break;
|
|
|
|
case MKTAG(0xa9,'x','y','z'): key = "location"; break;
|
2009-03-16 08:06:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (c->itunes_metadata && atom.size > 8) {
|
2011-02-21 17:43:01 +02:00
|
|
|
int data_size = avio_rb32(pb);
|
|
|
|
int tag = avio_rl32(pb);
|
2009-03-16 08:06:05 +02:00
|
|
|
if (tag == MKTAG('d','a','t','a')) {
|
2011-02-21 17:43:01 +02:00
|
|
|
data_type = avio_rb32(pb); // type
|
|
|
|
avio_rb32(pb); // unknown
|
2009-03-16 08:06:05 +02:00
|
|
|
str_size = data_size - 16;
|
|
|
|
atom.size -= 16;
|
2012-06-21 19:24:27 +03:00
|
|
|
|
|
|
|
if (atom.type == MKTAG('c', 'o', 'v', 'r')) {
|
|
|
|
int ret = mov_read_covr(c, pb, data_type, str_size);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Error parsing cover art.\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
2009-03-16 08:06:05 +02:00
|
|
|
} else return 0;
|
2014-12-03 04:42:29 +02:00
|
|
|
} else if (atom.size > 4 && key && !c->itunes_metadata && !raw) {
|
2011-02-21 17:43:01 +02:00
|
|
|
str_size = avio_rb16(pb); // string length
|
|
|
|
langcode = avio_rb16(pb);
|
2010-03-09 03:03:42 +02:00
|
|
|
ff_mov_lang_to_iso639(langcode, language);
|
2009-03-16 08:06:05 +02:00
|
|
|
atom.size -= 4;
|
|
|
|
} else
|
|
|
|
str_size = atom.size;
|
|
|
|
|
2014-11-29 18:51:13 +02:00
|
|
|
if (c->export_all && !key) {
|
2009-03-16 08:06:05 +02:00
|
|
|
snprintf(tmp_key, 5, "%.4s", (char*)&atom.type);
|
|
|
|
key = tmp_key;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!key)
|
|
|
|
return 0;
|
|
|
|
if (atom.size < 0)
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-03-16 08:06:05 +02:00
|
|
|
|
2014-11-29 18:51:14 +02:00
|
|
|
// allocate twice as much as worst-case
|
2014-12-15 12:09:10 +02:00
|
|
|
str_size_alloc = (raw ? str_size : str_size * 2) + 1;
|
2014-11-29 18:51:14 +02:00
|
|
|
str = av_malloc(str_size_alloc);
|
|
|
|
if (!str)
|
|
|
|
return AVERROR(ENOMEM);
|
2009-03-16 08:43:38 +02:00
|
|
|
|
|
|
|
if (parse)
|
2011-10-11 10:53:39 +03:00
|
|
|
parse(c, pb, str_size, key);
|
2009-03-16 08:43:38 +02:00
|
|
|
else {
|
2014-12-03 04:42:29 +02:00
|
|
|
if (!raw && (data_type == 3 || (data_type == 0 && (langcode < 0x400 || langcode == 0x7fff)))) { // MAC Encoded
|
2014-11-29 18:51:14 +02:00
|
|
|
mov_read_mac_string(c, pb, str_size, str, str_size_alloc);
|
2010-03-09 03:03:42 +02:00
|
|
|
} else {
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret = ffio_read_size(pb, str, str_size);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_free(str);
|
|
|
|
return ret;
|
|
|
|
}
|
2010-03-09 03:03:42 +02:00
|
|
|
str[str_size] = 0;
|
|
|
|
}
|
2014-08-13 00:03:54 +03:00
|
|
|
c->fc->event_flags |= AVFMT_EVENT_FLAG_METADATA_UPDATED;
|
2011-05-22 13:46:29 +03:00
|
|
|
av_dict_set(&c->fc->metadata, key, str, 0);
|
2009-03-16 08:44:16 +02:00
|
|
|
if (*language && strcmp(language, "und")) {
|
|
|
|
snprintf(key2, sizeof(key2), "%s-%s", key, language);
|
2011-05-22 13:46:29 +03:00
|
|
|
av_dict_set(&c->fc->metadata, key2, str, 0);
|
2009-03-16 08:44:16 +02:00
|
|
|
}
|
2009-03-16 08:43:38 +02:00
|
|
|
}
|
2009-03-16 08:06:05 +02:00
|
|
|
|
2014-11-29 18:51:14 +02:00
|
|
|
av_freep(&str);
|
2009-03-16 08:06:05 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2009-03-16 08:03:29 +02:00
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_chpl(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2010-05-05 11:41:01 +03:00
|
|
|
{
|
|
|
|
int64_t start;
|
2010-07-04 07:59:12 +03:00
|
|
|
int i, nb_chapters, str_len, version;
|
2010-05-05 11:41:01 +03:00
|
|
|
char str[256+1];
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2010-05-05 11:41:01 +03:00
|
|
|
|
|
|
|
if ((atom.size -= 5) < 0)
|
|
|
|
return 0;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
version = avio_r8(pb);
|
|
|
|
avio_rb24(pb);
|
2010-07-04 07:59:12 +03:00
|
|
|
if (version)
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); // ???
|
|
|
|
nb_chapters = avio_r8(pb);
|
2010-05-05 11:41:01 +03:00
|
|
|
|
|
|
|
for (i = 0; i < nb_chapters; i++) {
|
|
|
|
if (atom.size < 9)
|
|
|
|
return 0;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
start = avio_rb64(pb);
|
|
|
|
str_len = avio_r8(pb);
|
2010-05-05 11:41:01 +03:00
|
|
|
|
|
|
|
if ((atom.size -= 9+str_len) < 0)
|
|
|
|
return 0;
|
|
|
|
|
2015-05-26 15:24:39 +02:00
|
|
|
ret = ffio_read_size(pb, str, str_len);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2010-05-05 11:41:01 +03:00
|
|
|
str[str_len] = 0;
|
2011-10-17 09:58:50 +03:00
|
|
|
avpriv_new_chapter(c->fc, i, (AVRational){1,10000000}, start, AV_NOPTS_VALUE, str);
|
2010-05-05 11:41:01 +03:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-27 00:18:31 +03:00
|
|
|
#define MIN_DATA_ENTRY_BOX_SIZE 12
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_dref(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2008-02-26 20:21:58 +02:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2008-02-26 20:21:58 +02:00
|
|
|
int entries, i, j;
|
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); // version + flags
|
|
|
|
entries = avio_rb32(pb);
|
2016-03-08 12:57:16 +02:00
|
|
|
if (!entries ||
|
|
|
|
entries > (atom.size - 1) / MIN_DATA_ENTRY_BOX_SIZE + 1 ||
|
2012-11-27 00:18:31 +03:00
|
|
|
entries >= UINT_MAX / sizeof(*sc->drefs))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2016-03-08 12:57:16 +02:00
|
|
|
sc->drefs_count = 0;
|
2012-04-13 03:56:57 +03:00
|
|
|
av_free(sc->drefs);
|
2008-02-26 20:21:58 +02:00
|
|
|
sc->drefs = av_mallocz(entries * sizeof(*sc->drefs));
|
2009-02-22 02:20:07 +02:00
|
|
|
if (!sc->drefs)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
sc->drefs_count = entries;
|
2008-02-26 20:21:58 +02:00
|
|
|
|
2015-11-09 16:14:43 +02:00
|
|
|
for (i = 0; i < entries; i++) {
|
2008-12-11 22:06:56 +02:00
|
|
|
MOVDref *dref = &sc->drefs[i];
|
2011-02-21 17:43:01 +02:00
|
|
|
uint32_t size = avio_rb32(pb);
|
2011-03-03 21:11:45 +02:00
|
|
|
int64_t next = avio_tell(pb) + size - 4;
|
2008-02-26 20:21:58 +02:00
|
|
|
|
2010-09-08 23:20:24 +03:00
|
|
|
if (size < 12)
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2010-09-08 23:20:24 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
dref->type = avio_rl32(pb);
|
|
|
|
avio_rb32(pb); // version + flags
|
2008-02-26 20:21:58 +02:00
|
|
|
|
|
|
|
if (dref->type == MKTAG('a','l','i','s') && size > 150) {
|
|
|
|
/* macintosh alias record */
|
|
|
|
uint16_t volume_len, len;
|
|
|
|
int16_t type;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2008-02-26 20:21:58 +02:00
|
|
|
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, 10);
|
2008-02-26 20:21:58 +02:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
volume_len = avio_r8(pb);
|
2008-02-26 20:21:58 +02:00
|
|
|
volume_len = FFMIN(volume_len, 27);
|
2015-05-26 15:24:39 +02:00
|
|
|
ret = ffio_read_size(pb, dref->volume, 27);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2009-11-15 05:26:47 +02:00
|
|
|
dref->volume[volume_len] = 0;
|
2015-12-11 18:29:11 +02:00
|
|
|
av_log(c->fc, AV_LOG_DEBUG, "volume %s, len %"PRIu16"\n",
|
|
|
|
dref->volume, volume_len);
|
2008-02-26 20:21:58 +02:00
|
|
|
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, 12);
|
2009-11-15 05:26:47 +02:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
len = avio_r8(pb);
|
2009-11-15 05:26:47 +02:00
|
|
|
len = FFMIN(len, 63);
|
2015-05-26 15:24:39 +02:00
|
|
|
ret = ffio_read_size(pb, dref->filename, 63);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2009-11-15 05:26:47 +02:00
|
|
|
dref->filename[len] = 0;
|
2015-12-11 18:29:11 +02:00
|
|
|
av_log(c->fc, AV_LOG_DEBUG, "filename %s, len %"PRIu16"\n",
|
|
|
|
dref->filename, len);
|
2009-11-15 05:26:47 +02:00
|
|
|
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, 16);
|
2009-11-15 05:26:47 +02:00
|
|
|
|
|
|
|
/* read next level up_from_alias/down_to_target */
|
2011-02-21 17:43:01 +02:00
|
|
|
dref->nlvl_from = avio_rb16(pb);
|
|
|
|
dref->nlvl_to = avio_rb16(pb);
|
2015-12-11 18:29:11 +02:00
|
|
|
av_log(c->fc, AV_LOG_DEBUG, "nlvl from %"PRId16", nlvl to %"PRId16"\n",
|
2009-11-15 05:26:47 +02:00
|
|
|
dref->nlvl_from, dref->nlvl_to);
|
|
|
|
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, 16);
|
2008-02-26 20:21:58 +02:00
|
|
|
|
2011-03-03 21:11:45 +02:00
|
|
|
for (type = 0; type != -1 && avio_tell(pb) < next; ) {
|
2011-12-18 23:37:54 +03:00
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
2011-02-21 17:43:01 +02:00
|
|
|
type = avio_rb16(pb);
|
|
|
|
len = avio_rb16(pb);
|
2015-12-11 18:29:11 +02:00
|
|
|
av_log(c->fc, AV_LOG_DEBUG, "type %"PRId16", len %"PRIu16"\n",
|
|
|
|
type, len);
|
2008-02-26 20:21:58 +02:00
|
|
|
if (len&1)
|
|
|
|
len += 1;
|
2015-11-09 16:14:42 +02:00
|
|
|
if (type == 2) { // absolute path
|
2008-02-27 11:31:41 +02:00
|
|
|
av_free(dref->path);
|
2008-02-26 20:21:58 +02:00
|
|
|
dref->path = av_mallocz(len+1);
|
2008-02-27 11:30:56 +02:00
|
|
|
if (!dref->path)
|
|
|
|
return AVERROR(ENOMEM);
|
2015-05-26 15:24:39 +02:00
|
|
|
|
|
|
|
ret = ffio_read_size(pb, dref->path, len);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_freep(&dref->path);
|
|
|
|
return ret;
|
|
|
|
}
|
2009-11-15 05:26:47 +02:00
|
|
|
if (len > volume_len && !strncmp(dref->path, dref->volume, volume_len)) {
|
2008-02-26 20:21:58 +02:00
|
|
|
len -= volume_len;
|
|
|
|
memmove(dref->path, dref->path+volume_len, len);
|
|
|
|
dref->path[len] = 0;
|
|
|
|
}
|
2016-02-16 04:44:07 +02:00
|
|
|
// trim string of any ending zeros
|
|
|
|
for (j = len - 1; j >= 0; j--) {
|
|
|
|
if (dref->path[j] == 0)
|
|
|
|
len--;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
2008-02-26 20:21:58 +02:00
|
|
|
for (j = 0; j < len; j++)
|
2015-11-09 16:14:42 +02:00
|
|
|
if (dref->path[j] == ':' || dref->path[j] == 0)
|
2008-02-26 20:21:58 +02:00
|
|
|
dref->path[j] = '/';
|
|
|
|
av_log(c->fc, AV_LOG_DEBUG, "path %s\n", dref->path);
|
2009-11-15 05:26:47 +02:00
|
|
|
} else if (type == 0) { // directory name
|
|
|
|
av_free(dref->dir);
|
|
|
|
dref->dir = av_malloc(len+1);
|
|
|
|
if (!dref->dir)
|
|
|
|
return AVERROR(ENOMEM);
|
2015-05-26 15:24:39 +02:00
|
|
|
|
|
|
|
ret = ffio_read_size(pb, dref->dir, len);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_freep(&dref->dir);
|
|
|
|
return ret;
|
|
|
|
}
|
2009-11-15 05:26:47 +02:00
|
|
|
dref->dir[len] = 0;
|
|
|
|
for (j = 0; j < len; j++)
|
|
|
|
if (dref->dir[j] == ':')
|
|
|
|
dref->dir[j] = '/';
|
|
|
|
av_log(c->fc, AV_LOG_DEBUG, "dir %s\n", dref->dir);
|
2008-02-26 20:21:58 +02:00
|
|
|
} else
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, len);
|
2008-02-26 20:21:58 +02:00
|
|
|
}
|
2015-11-09 16:14:43 +02:00
|
|
|
} else {
|
2015-12-11 18:29:11 +02:00
|
|
|
av_log(c->fc, AV_LOG_DEBUG, "Unknown dref type 0x08%"PRIx32" size %"PRIu32"\n",
|
2015-11-09 16:14:43 +02:00
|
|
|
dref->type, size);
|
|
|
|
entries--;
|
|
|
|
i--;
|
2008-02-26 20:21:58 +02:00
|
|
|
}
|
2011-02-28 15:57:54 +02:00
|
|
|
avio_seek(pb, next, SEEK_SET);
|
2008-02-26 20:21:58 +02:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_hdlr(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2002-04-08 15:32:01 +03:00
|
|
|
{
|
2009-03-03 20:19:44 +02:00
|
|
|
AVStream *st;
|
2003-02-11 18:35:48 +02:00
|
|
|
uint32_t type;
|
2011-06-03 14:52:29 +03:00
|
|
|
uint32_t av_unused ctype;
|
2015-02-03 16:58:50 +02:00
|
|
|
int64_t title_size;
|
|
|
|
char *title_str;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2003-04-23 11:58:42 +03:00
|
|
|
|
2009-03-03 20:19:44 +02:00
|
|
|
if (c->fc->nb_streams < 1) // meta before first trak
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
2002-04-08 15:32:01 +03:00
|
|
|
|
|
|
|
/* component type */
|
2011-02-21 17:43:01 +02:00
|
|
|
ctype = avio_rl32(pb);
|
|
|
|
type = avio_rl32(pb); /* component subtype */
|
2002-04-08 15:32:01 +03:00
|
|
|
|
2016-12-21 12:25:34 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "ctype= %.4s (0x%08"PRIx32")\n", (char *)&ctype, ctype);
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "stype= %.4s\n", (char*)&type);
|
2009-03-16 07:13:55 +02:00
|
|
|
|
2008-04-25 15:02:01 +03:00
|
|
|
if (type == MKTAG('v','i','d','e'))
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_VIDEO;
|
2011-10-11 11:12:38 +03:00
|
|
|
else if (type == MKTAG('s','o','u','n'))
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
|
2011-10-11 11:12:38 +03:00
|
|
|
else if (type == MKTAG('m','1','a',' '))
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_MP2;
|
2011-10-11 11:12:38 +03:00
|
|
|
else if ((type == MKTAG('s','u','b','p')) || (type == MKTAG('c','l','c','p')))
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_SUBTITLE;
|
2009-03-16 08:46:57 +02:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); /* component manufacture */
|
|
|
|
avio_rb32(pb); /* component flags */
|
|
|
|
avio_rb32(pb); /* component flags mask */
|
2002-04-08 15:32:01 +03:00
|
|
|
|
2015-02-03 16:58:50 +02:00
|
|
|
title_size = atom.size - 24;
|
|
|
|
if (title_size > 0) {
|
|
|
|
title_str = av_malloc(title_size + 1); /* Add null terminator */
|
|
|
|
if (!title_str)
|
|
|
|
return AVERROR(ENOMEM);
|
2015-05-26 15:24:39 +02:00
|
|
|
|
|
|
|
ret = ffio_read_size(pb, title_str, title_size);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_freep(&title_str);
|
|
|
|
return ret;
|
|
|
|
}
|
2015-02-03 16:58:50 +02:00
|
|
|
title_str[title_size] = 0;
|
|
|
|
if (title_str[0]) {
|
|
|
|
int off = (!c->isom && title_str[0] == title_size - 1);
|
|
|
|
av_dict_set(&st->metadata, "handler_name", title_str + off, 0);
|
|
|
|
}
|
|
|
|
av_freep(&title_str);
|
|
|
|
}
|
|
|
|
|
2002-04-08 15:32:01 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-08-15 23:13:03 +03:00
|
|
|
int ff_mov_read_esds(AVFormatContext *fc, AVIOContext *pb)
|
2003-04-22 11:36:39 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
2011-06-01 19:26:27 +03:00
|
|
|
int tag;
|
2003-04-23 11:58:42 +03:00
|
|
|
|
2009-09-16 14:35:01 +03:00
|
|
|
if (fc->nb_streams < 1)
|
2009-06-24 06:38:47 +03:00
|
|
|
return 0;
|
2009-09-16 14:35:01 +03:00
|
|
|
st = fc->streams[fc->nb_streams-1];
|
2009-06-24 06:38:47 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); /* version + flags */
|
2011-06-01 19:26:27 +03:00
|
|
|
ff_mp4_read_descr(fc, pb, &tag);
|
2003-04-22 11:36:39 +03:00
|
|
|
if (tag == MP4ESDescrTag) {
|
2011-08-25 02:59:30 +03:00
|
|
|
ff_mp4_parse_es_descr(pb, NULL);
|
2003-04-22 11:36:39 +03:00
|
|
|
} else
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb16(pb); /* ID */
|
2003-04-22 11:36:39 +03:00
|
|
|
|
2011-06-01 19:26:27 +03:00
|
|
|
ff_mp4_read_descr(fc, pb, &tag);
|
2010-11-23 02:51:12 +02:00
|
|
|
if (tag == MP4DecConfigDescrTag)
|
|
|
|
ff_mp4_read_dec_config_descr(fc, st, pb);
|
2003-04-22 11:36:39 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_esds(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2009-09-16 14:35:01 +03:00
|
|
|
{
|
2014-08-15 23:13:03 +03:00
|
|
|
return ff_mov_read_esds(c->fc, pb);
|
2009-09-16 14:35:01 +03:00
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_dac3(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2010-09-07 00:52:11 +03:00
|
|
|
{
|
|
|
|
AVStream *st;
|
2014-11-14 13:54:52 +02:00
|
|
|
enum AVAudioServiceType *ast;
|
2011-03-24 18:42:05 +02:00
|
|
|
int ac3info, acmod, lfeon, bsmod;
|
2010-09-07 00:52:11 +03:00
|
|
|
|
2010-09-28 03:33:21 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
2010-09-07 00:52:11 +03:00
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
|
2015-11-05 19:49:02 +02:00
|
|
|
ast = (enum AVAudioServiceType*)av_stream_new_side_data(st, AV_PKT_DATA_AUDIO_SERVICE_TYPE,
|
2014-11-14 13:54:52 +02:00
|
|
|
sizeof(*ast));
|
|
|
|
if (!ast)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
ac3info = avio_rb24(pb);
|
2011-03-24 18:42:05 +02:00
|
|
|
bsmod = (ac3info >> 14) & 0x7;
|
2010-09-07 00:52:11 +03:00
|
|
|
acmod = (ac3info >> 11) & 0x7;
|
|
|
|
lfeon = (ac3info >> 10) & 0x1;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->channels = ((int[]){2,1,2,3,3,4,4,5})[acmod] + lfeon;
|
|
|
|
st->codecpar->channel_layout = avpriv_ac3_channel_layout_tab[acmod];
|
2012-02-12 23:06:58 +03:00
|
|
|
if (lfeon)
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->channel_layout |= AV_CH_LOW_FREQUENCY;
|
2014-11-14 13:54:52 +02:00
|
|
|
*ast = bsmod;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->channels > 1 && bsmod == 0x7)
|
2014-11-14 13:54:52 +02:00
|
|
|
*ast = AV_AUDIO_SERVICE_TYPE_KARAOKE;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
#if FF_API_LAVF_AVCTX
|
|
|
|
FF_DISABLE_DEPRECATION_WARNINGS
|
2014-11-14 13:54:52 +02:00
|
|
|
st->codec->audio_service_type = *ast;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
FF_ENABLE_DEPRECATION_WARNINGS
|
|
|
|
#endif
|
2010-09-07 00:52:11 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-05-31 02:17:13 +03:00
|
|
|
static int mov_read_dec3(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
2014-11-14 13:54:52 +02:00
|
|
|
enum AVAudioServiceType *ast;
|
2011-05-31 02:17:13 +03:00
|
|
|
int eac3info, acmod, lfeon, bsmod;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
|
2015-11-05 19:49:02 +02:00
|
|
|
ast = (enum AVAudioServiceType*)av_stream_new_side_data(st, AV_PKT_DATA_AUDIO_SERVICE_TYPE,
|
2014-11-14 13:54:52 +02:00
|
|
|
sizeof(*ast));
|
|
|
|
if (!ast)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
2011-05-31 02:17:13 +03:00
|
|
|
/* No need to parse fields for additional independent substreams and its
|
|
|
|
* associated dependent substreams since libavcodec's E-AC-3 decoder
|
|
|
|
* does not support them yet. */
|
|
|
|
avio_rb16(pb); /* data_rate and num_ind_sub */
|
|
|
|
eac3info = avio_rb24(pb);
|
|
|
|
bsmod = (eac3info >> 12) & 0x1f;
|
|
|
|
acmod = (eac3info >> 9) & 0x7;
|
|
|
|
lfeon = (eac3info >> 8) & 0x1;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->channel_layout = avpriv_ac3_channel_layout_tab[acmod];
|
2011-05-31 02:17:13 +03:00
|
|
|
if (lfeon)
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->channel_layout |= AV_CH_LOW_FREQUENCY;
|
|
|
|
st->codecpar->channels = av_get_channel_layout_nb_channels(st->codecpar->channel_layout);
|
2014-11-14 13:54:52 +02:00
|
|
|
*ast = bsmod;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->channels > 1 && bsmod == 0x7)
|
2014-11-14 13:54:52 +02:00
|
|
|
*ast = AV_AUDIO_SERVICE_TYPE_KARAOKE;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
#if FF_API_LAVF_AVCTX
|
|
|
|
FF_DISABLE_DEPRECATION_WARNINGS
|
2014-11-14 13:54:52 +02:00
|
|
|
st->codec->audio_service_type = *ast;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
FF_ENABLE_DEPRECATION_WARNINGS
|
|
|
|
#endif
|
2011-05-31 02:17:13 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-12-03 21:32:45 +03:00
|
|
|
static int mov_read_chan(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
|
|
|
|
if (atom.size < 16)
|
|
|
|
return 0;
|
|
|
|
|
2014-12-04 21:13:39 +02:00
|
|
|
/* skip version and flags */
|
|
|
|
avio_skip(pb, 4);
|
|
|
|
|
2012-09-06 13:46:37 +03:00
|
|
|
ff_mov_read_chan(c->fc, pb, st, atom.size - 4);
|
2011-12-03 21:32:45 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-06-06 20:59:46 +03:00
|
|
|
static int mov_read_wfex(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
return ff_get_wav_header(c->fc, pb, st->codecpar, atom.size);
|
2011-06-06 20:59:46 +03:00
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_pasp(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2008-12-12 00:24:56 +02:00
|
|
|
{
|
2011-02-21 17:43:01 +02:00
|
|
|
const int num = avio_rb32(pb);
|
|
|
|
const int den = avio_rb32(pb);
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
|
2010-09-08 23:18:14 +03:00
|
|
|
if ((st->sample_aspect_ratio.den != 1 || st->sample_aspect_ratio.num) && // default
|
|
|
|
(den != st->sample_aspect_ratio.den || num != st->sample_aspect_ratio.num)) {
|
|
|
|
av_log(c->fc, AV_LOG_WARNING,
|
|
|
|
"sample aspect ratio already set to %d:%d, ignoring 'pasp' atom (%d:%d)\n",
|
|
|
|
st->sample_aspect_ratio.num, st->sample_aspect_ratio.den,
|
|
|
|
num, den);
|
|
|
|
} else if (den != 0) {
|
2008-12-12 00:24:56 +02:00
|
|
|
st->sample_aspect_ratio.num = num;
|
|
|
|
st->sample_aspect_ratio.den = den;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2003-05-12 13:59:18 +03:00
|
|
|
/* this atom contains actual media data */
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_mdat(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2002-04-08 15:32:01 +03:00
|
|
|
{
|
2011-10-11 11:12:38 +03:00
|
|
|
if (atom.size == 0) /* wrong one (MP4) */
|
2003-05-12 13:59:18 +03:00
|
|
|
return 0;
|
|
|
|
c->found_mdat=1;
|
|
|
|
return 0; /* now go for moov */
|
|
|
|
}
|
|
|
|
|
2009-09-26 23:51:38 +03:00
|
|
|
/* read major brand, minor version and compatible brands and store them as metadata */
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_ftyp(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2006-02-18 11:03:22 +02:00
|
|
|
{
|
2009-09-26 23:51:38 +03:00
|
|
|
uint32_t minor_ver;
|
|
|
|
int comp_brand_size;
|
|
|
|
char minor_ver_str[11]; /* 32 bit integer -> 10 digits + null */
|
|
|
|
char* comp_brands_str;
|
|
|
|
uint8_t type[5] = {0};
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret = ffio_read_size(pb, type, 4);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2009-09-26 23:51:38 +03:00
|
|
|
|
2009-09-27 00:06:18 +03:00
|
|
|
if (strcmp(type, "qt "))
|
2006-11-03 17:02:51 +02:00
|
|
|
c->isom = 1;
|
2006-11-01 21:16:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_DEBUG, "ISO: File Type Major Brand: %.4s\n",(char *)&type);
|
2011-05-22 13:46:29 +03:00
|
|
|
av_dict_set(&c->fc->metadata, "major_brand", type, 0);
|
2011-02-21 17:43:01 +02:00
|
|
|
minor_ver = avio_rb32(pb); /* minor version */
|
2014-03-10 17:35:59 +03:00
|
|
|
snprintf(minor_ver_str, sizeof(minor_ver_str), "%"PRIu32"", minor_ver);
|
2011-05-22 13:46:29 +03:00
|
|
|
av_dict_set(&c->fc->metadata, "minor_version", minor_ver_str, 0);
|
2009-09-26 23:51:38 +03:00
|
|
|
|
|
|
|
comp_brand_size = atom.size - 8;
|
|
|
|
if (comp_brand_size < 0)
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-09-26 23:51:38 +03:00
|
|
|
comp_brands_str = av_malloc(comp_brand_size + 1); /* Add null terminator */
|
|
|
|
if (!comp_brands_str)
|
|
|
|
return AVERROR(ENOMEM);
|
2015-05-26 15:24:39 +02:00
|
|
|
|
|
|
|
ret = ffio_read_size(pb, comp_brands_str, comp_brand_size);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_freep(&comp_brands_str);
|
|
|
|
return ret;
|
|
|
|
}
|
2009-09-26 23:51:38 +03:00
|
|
|
comp_brands_str[comp_brand_size] = 0;
|
2011-05-22 13:46:29 +03:00
|
|
|
av_dict_set(&c->fc->metadata, "compatible_brands", comp_brands_str, 0);
|
2009-09-26 23:51:38 +03:00
|
|
|
av_freep(&comp_brands_str);
|
|
|
|
|
2006-02-18 11:03:22 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2003-05-12 13:59:18 +03:00
|
|
|
/* this atom should contain all header atoms */
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_moov(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2003-05-12 13:59:18 +03:00
|
|
|
{
|
2012-01-15 00:25:12 +03:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
if ((ret = mov_read_default(c, pb, atom)) < 0)
|
|
|
|
return ret;
|
2003-05-12 13:59:18 +03:00
|
|
|
/* we parsed the 'moov' atom, we can terminate the parsing as soon as we find the 'mdat' */
|
|
|
|
/* so we don't parse the whole file if over a network */
|
|
|
|
c->found_moov=1;
|
|
|
|
return 0; /* now go for mdat */
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_moof(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2008-03-21 15:40:23 +02:00
|
|
|
{
|
2014-07-01 08:17:56 +03:00
|
|
|
c->fragment.moof_offset = c->fragment.implicit_offset = avio_tell(pb) - 8;
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "moof offset %"PRIx64"\n", c->fragment.moof_offset);
|
2008-03-21 15:40:23 +02:00
|
|
|
return mov_read_default(c, pb, atom);
|
|
|
|
}
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-05-22 13:46:29 +03:00
|
|
|
static void mov_metadata_creation_time(AVDictionary **metadata, time_t time)
|
2010-11-21 21:02:52 +02:00
|
|
|
{
|
|
|
|
char buffer[32];
|
|
|
|
if (time) {
|
2014-10-24 10:46:36 +03:00
|
|
|
struct tm *ptm, tmbuf;
|
2010-11-21 21:02:52 +02:00
|
|
|
time -= 2082844800; /* seconds between 1904-01-01 and Epoch */
|
2014-10-24 10:46:36 +03:00
|
|
|
ptm = gmtime_r(&time, &tmbuf);
|
2011-01-05 21:21:04 +02:00
|
|
|
if (!ptm) return;
|
2014-10-24 10:43:20 +03:00
|
|
|
if (strftime(buffer, sizeof(buffer), "%Y-%m-%d %H:%M:%S", ptm))
|
|
|
|
av_dict_set(metadata, "creation_time", buffer, 0);
|
2010-11-21 21:02:52 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_mdhd(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2003-05-12 13:59:18 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
int version;
|
2009-02-15 04:28:54 +02:00
|
|
|
char language[4] = {0};
|
2009-01-22 00:32:17 +02:00
|
|
|
unsigned lang;
|
2010-11-21 21:02:52 +02:00
|
|
|
time_t creation_time;
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2013-07-15 17:13:54 +03:00
|
|
|
if (sc->time_scale) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Multiple mdhd?\n");
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
}
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
version = avio_r8(pb);
|
2012-01-15 00:25:12 +03:00
|
|
|
if (version > 1) {
|
2013-02-26 01:54:28 +03:00
|
|
|
avpriv_request_sample(c->fc, "Version %d", version);
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_PATCHWELCOME;
|
|
|
|
}
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb24(pb); /* flags */
|
2006-03-22 12:56:51 +02:00
|
|
|
if (version == 1) {
|
2011-02-21 17:43:01 +02:00
|
|
|
creation_time = avio_rb64(pb);
|
|
|
|
avio_rb64(pb);
|
2006-03-22 12:56:51 +02:00
|
|
|
} else {
|
2011-02-21 17:43:01 +02:00
|
|
|
creation_time = avio_rb32(pb);
|
|
|
|
avio_rb32(pb); /* modification time */
|
2006-03-22 12:56:51 +02:00
|
|
|
}
|
2010-11-21 21:02:52 +02:00
|
|
|
mov_metadata_creation_time(&st->metadata, creation_time);
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
sc->time_scale = avio_rb32(pb);
|
|
|
|
st->duration = (version == 1) ? avio_rb64(pb) : avio_rb32(pb); /* duration */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
lang = avio_rb16(pb); /* language */
|
2009-02-15 04:28:54 +02:00
|
|
|
if (ff_mov_lang_to_iso639(lang, language))
|
2011-05-22 13:46:29 +03:00
|
|
|
av_dict_set(&st->metadata, "language", language, 0);
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb16(pb); /* quality */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_mvhd(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2003-05-12 13:59:18 +03:00
|
|
|
{
|
2016-10-24 23:46:47 +02:00
|
|
|
int i;
|
2010-11-21 21:02:52 +02:00
|
|
|
time_t creation_time;
|
2011-02-21 17:43:01 +02:00
|
|
|
int version = avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2006-03-22 12:45:04 +02:00
|
|
|
if (version == 1) {
|
2011-02-21 17:43:01 +02:00
|
|
|
creation_time = avio_rb64(pb);
|
|
|
|
avio_rb64(pb);
|
2006-03-22 12:45:04 +02:00
|
|
|
} else {
|
2011-02-21 17:43:01 +02:00
|
|
|
creation_time = avio_rb32(pb);
|
|
|
|
avio_rb32(pb); /* modification time */
|
2006-03-22 12:45:04 +02:00
|
|
|
}
|
2010-11-21 21:02:52 +02:00
|
|
|
mov_metadata_creation_time(&c->fc->metadata, creation_time);
|
2011-02-21 17:43:01 +02:00
|
|
|
c->time_scale = avio_rb32(pb); /* time scale */
|
2007-04-26 14:24:10 +03:00
|
|
|
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "time scale = %i\n", c->time_scale);
|
2007-04-26 14:24:10 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
c->duration = (version == 1) ? avio_rb64(pb) : avio_rb32(pb); /* duration */
|
|
|
|
avio_rb32(pb); /* preferred scale */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb16(pb); /* preferred volume */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, 10); /* reserved */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2016-10-24 23:46:47 +02:00
|
|
|
/* movie display matrix, store it in main context and use it later on */
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
c->movie_display_matrix[i][0] = avio_rb32(pb); // 16.16 fixed point
|
|
|
|
c->movie_display_matrix[i][1] = avio_rb32(pb); // 16.16 fixed point
|
|
|
|
c->movie_display_matrix[i][2] = avio_rb32(pb); // 2.30 fixed point
|
|
|
|
}
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); /* preview time */
|
|
|
|
avio_rb32(pb); /* preview duration */
|
|
|
|
avio_rb32(pb); /* poster time */
|
|
|
|
avio_rb32(pb); /* selection time */
|
|
|
|
avio_rb32(pb); /* selection duration */
|
|
|
|
avio_rb32(pb); /* current time */
|
|
|
|
avio_rb32(pb); /* next track ID */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_smi(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2003-05-12 18:39:38 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2009-06-24 06:38:47 +03:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
2003-05-12 18:39:38 +03:00
|
|
|
|
2011-10-11 11:12:38 +03:00
|
|
|
if ((uint64_t)atom.size > (1<<30))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2003-05-12 18:39:38 +03:00
|
|
|
// currently SVQ3 decoder expect full STSD header - so let's fake it
|
|
|
|
// this should be fixed and just SMI header should be passed
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
av_free(st->codecpar->extradata);
|
|
|
|
st->codecpar->extradata = av_mallocz(atom.size + 0x5a + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!st->codecpar->extradata)
|
2008-02-25 14:42:32 +02:00
|
|
|
return AVERROR(ENOMEM);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->extradata_size = 0x5a + atom.size;
|
|
|
|
memcpy(st->codecpar->extradata, "SVQ3", 4); // fake
|
2015-05-26 15:24:39 +02:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
ret = ffio_read_size(pb, st->codecpar->extradata + 0x5a, atom.size);
|
2015-05-26 15:24:39 +02:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "Reading SMI %"PRId64" %s\n", atom.size, st->codecpar->extradata + 0x5a);
|
2003-05-12 18:39:38 +03:00
|
|
|
return 0;
|
|
|
|
}
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_enda(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2006-02-19 22:00:00 +02:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
int little_endian;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
2006-02-19 22:00:00 +02:00
|
|
|
|
2015-03-13 21:45:14 +02:00
|
|
|
little_endian = !!avio_rb16(pb);
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "enda %d\n", little_endian);
|
2009-01-10 00:13:37 +02:00
|
|
|
if (little_endian == 1) {
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
switch (st->codecpar->codec_id) {
|
2012-08-05 12:11:04 +03:00
|
|
|
case AV_CODEC_ID_PCM_S24BE:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_PCM_S24LE;
|
2006-02-19 22:00:00 +02:00
|
|
|
break;
|
2012-08-05 12:11:04 +03:00
|
|
|
case AV_CODEC_ID_PCM_S32BE:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_PCM_S32LE;
|
2006-02-19 22:00:00 +02:00
|
|
|
break;
|
2012-08-05 12:11:04 +03:00
|
|
|
case AV_CODEC_ID_PCM_F32BE:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_PCM_F32LE;
|
2008-08-19 14:00:40 +03:00
|
|
|
break;
|
2012-08-05 12:11:04 +03:00
|
|
|
case AV_CODEC_ID_PCM_F64BE:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_PCM_F64LE;
|
2008-08-19 14:00:40 +03:00
|
|
|
break;
|
2006-02-19 22:00:00 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-10-02 16:28:35 +03:00
|
|
|
static int mov_read_colr(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
char color_parameter_type[5] = { 0 };
|
2015-06-30 15:38:47 +02:00
|
|
|
uint16_t color_primaries, color_trc, color_matrix;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2014-10-02 16:28:35 +03:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams - 1];
|
|
|
|
|
2015-05-26 15:24:39 +02:00
|
|
|
ret = ffio_read_size(pb, color_parameter_type, 4);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2014-10-02 16:28:35 +03:00
|
|
|
if (strncmp(color_parameter_type, "nclx", 4) &&
|
|
|
|
strncmp(color_parameter_type, "nclc", 4)) {
|
|
|
|
av_log(c->fc, AV_LOG_WARNING, "unsupported color_parameter_type %s\n",
|
|
|
|
color_parameter_type);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
color_primaries = avio_rb16(pb);
|
|
|
|
color_trc = avio_rb16(pb);
|
|
|
|
color_matrix = avio_rb16(pb);
|
|
|
|
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE,
|
|
|
|
"%s: pri %"PRIu16" trc %"PRIu16" matrix %"PRIu16"",
|
|
|
|
color_parameter_type, color_primaries, color_trc, color_matrix);
|
2014-10-02 16:28:35 +03:00
|
|
|
|
2015-03-23 21:43:27 +02:00
|
|
|
if (!strncmp(color_parameter_type, "nclx", 4)) {
|
2014-10-02 16:28:35 +03:00
|
|
|
uint8_t color_range = avio_r8(pb) >> 7;
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, " full %"PRIu8"", color_range);
|
2014-10-02 16:28:35 +03:00
|
|
|
if (color_range)
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->color_range = AVCOL_RANGE_JPEG;
|
2014-10-02 16:28:35 +03:00
|
|
|
else
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->color_range = AVCOL_RANGE_MPEG;
|
2016-10-19 00:21:31 +02:00
|
|
|
|
|
|
|
if (!av_color_primaries_name(color_primaries))
|
2014-10-02 16:28:35 +03:00
|
|
|
color_primaries = AVCOL_PRI_UNSPECIFIED;
|
2016-10-19 00:21:31 +02:00
|
|
|
if (!av_color_transfer_name(color_trc))
|
2014-10-02 16:28:35 +03:00
|
|
|
color_trc = AVCOL_TRC_UNSPECIFIED;
|
2016-10-19 00:21:31 +02:00
|
|
|
if (!av_color_space_name(color_matrix))
|
2014-10-02 16:28:35 +03:00
|
|
|
color_matrix = AVCOL_SPC_UNSPECIFIED;
|
2016-10-19 00:21:31 +02:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->color_primaries = color_primaries;
|
|
|
|
st->codecpar->color_trc = color_trc;
|
|
|
|
st->codecpar->color_space = color_matrix;
|
2015-03-23 21:43:27 +02:00
|
|
|
} else if (!strncmp(color_parameter_type, "nclc", 4)) {
|
2014-10-02 16:28:35 +03:00
|
|
|
/* color primaries, Table 4-4 */
|
|
|
|
switch (color_primaries) {
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
case 1: st->codecpar->color_primaries = AVCOL_PRI_BT709; break;
|
|
|
|
case 5: st->codecpar->color_primaries = AVCOL_PRI_SMPTE170M; break;
|
|
|
|
case 6: st->codecpar->color_primaries = AVCOL_PRI_SMPTE240M; break;
|
2016-10-19 00:21:31 +02:00
|
|
|
case 9: st->codecpar->color_primaries = AVCOL_PRI_BT2020; break;
|
|
|
|
case 10: st->codecpar->color_primaries = AVCOL_PRI_SMPTE431; break;
|
|
|
|
case 11: st->codecpar->color_primaries = AVCOL_PRI_SMPTE432; break;
|
2014-10-02 16:28:35 +03:00
|
|
|
}
|
|
|
|
/* color transfer, Table 4-5 */
|
|
|
|
switch (color_trc) {
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
case 1: st->codecpar->color_trc = AVCOL_TRC_BT709; break;
|
|
|
|
case 7: st->codecpar->color_trc = AVCOL_TRC_SMPTE240M; break;
|
2016-10-19 00:21:31 +02:00
|
|
|
case 17: st->codecpar->color_trc = AVCOL_TRC_SMPTE428; break;
|
2014-10-02 16:28:35 +03:00
|
|
|
}
|
|
|
|
/* color matrix, Table 4-6 */
|
|
|
|
switch (color_matrix) {
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
case 1: st->codecpar->color_space = AVCOL_SPC_BT709; break;
|
|
|
|
case 6: st->codecpar->color_space = AVCOL_SPC_BT470BG; break;
|
|
|
|
case 7: st->codecpar->color_space = AVCOL_SPC_SMPTE240M; break;
|
2016-10-19 00:21:31 +02:00
|
|
|
case 9: st->codecpar->color_space = AVCOL_SPC_BT2020_NCL; break;
|
2014-10-02 16:28:35 +03:00
|
|
|
}
|
|
|
|
}
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "\n");
|
2014-10-02 16:28:35 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-12-14 05:49:06 +03:00
|
|
|
static int mov_read_fiel(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
unsigned mov_field_order;
|
|
|
|
enum AVFieldOrder decoded_field_order = AV_FIELD_UNKNOWN;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1) // will happen with jp2 files
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
if (atom.size < 2)
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
mov_field_order = avio_rb16(pb);
|
|
|
|
if ((mov_field_order & 0xFF00) == 0x0100)
|
|
|
|
decoded_field_order = AV_FIELD_PROGRESSIVE;
|
|
|
|
else if ((mov_field_order & 0xFF00) == 0x0200) {
|
|
|
|
switch (mov_field_order & 0xFF) {
|
|
|
|
case 0x01: decoded_field_order = AV_FIELD_TT;
|
|
|
|
break;
|
|
|
|
case 0x06: decoded_field_order = AV_FIELD_BB;
|
|
|
|
break;
|
|
|
|
case 0x09: decoded_field_order = AV_FIELD_TB;
|
|
|
|
break;
|
|
|
|
case 0x0E: decoded_field_order = AV_FIELD_BT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (decoded_field_order == AV_FIELD_UNKNOWN && mov_field_order) {
|
|
|
|
av_log(NULL, AV_LOG_ERROR, "Unknown MOV field order 0x%04x\n", mov_field_order);
|
|
|
|
}
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->field_order = decoded_field_order;
|
2011-12-14 05:49:06 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-04-27 19:45:23 +02:00
|
|
|
/* FIXME modify QDM2/SVQ3/H.264 decoders to take full atom as extradata */
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_extradata(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2006-02-19 22:00:00 +02:00
|
|
|
{
|
2008-12-16 10:47:28 +02:00
|
|
|
AVStream *st;
|
|
|
|
uint64_t size;
|
2007-10-08 16:06:48 +03:00
|
|
|
uint8_t *buf;
|
2013-09-18 19:12:36 +03:00
|
|
|
int err;
|
2008-12-16 10:47:28 +02:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1) // will happen with jp2 files
|
|
|
|
return 0;
|
|
|
|
st= c->fc->streams[c->fc->nb_streams-1];
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
size= (uint64_t)st->codecpar->extradata_size + atom.size + 8 + AV_INPUT_BUFFER_PADDING_SIZE;
|
2011-10-11 11:12:38 +03:00
|
|
|
if (size > INT_MAX || (uint64_t)atom.size > INT_MAX)
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if ((err = av_reallocp(&st->codecpar->extradata, size)) < 0) {
|
|
|
|
st->codecpar->extradata_size = 0;
|
2013-09-18 19:12:36 +03:00
|
|
|
return err;
|
2013-09-26 16:37:02 +03:00
|
|
|
}
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
buf = st->codecpar->extradata + st->codecpar->extradata_size;
|
|
|
|
st->codecpar->extradata_size= size - AV_INPUT_BUFFER_PADDING_SIZE;
|
2007-10-08 16:06:48 +03:00
|
|
|
AV_WB32( buf , atom.size + 8);
|
|
|
|
AV_WL32( buf + 4, atom.type);
|
2015-05-26 15:24:39 +02:00
|
|
|
|
|
|
|
err = ffio_read_size(pb, buf + 8, atom.size);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
2006-02-19 22:00:00 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_wave(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2005-10-18 23:16:43 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2009-06-24 06:38:47 +03:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
2005-10-18 23:16:43 +03:00
|
|
|
|
2011-10-11 11:12:38 +03:00
|
|
|
if ((uint64_t)atom.size > (1<<30))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2005-12-17 20:14:38 +02:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_id == AV_CODEC_ID_QDM2 || st->codecpar->codec_id == AV_CODEC_ID_QDMC) {
|
2011-02-06 14:54:03 +02:00
|
|
|
// pass all frma atom to codec, needed at least for QDMC and QDM2
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
av_free(st->codecpar->extradata);
|
|
|
|
st->codecpar->extradata = av_mallocz(atom.size + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!st->codecpar->extradata)
|
2008-02-25 14:42:32 +02:00
|
|
|
return AVERROR(ENOMEM);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->extradata_size = atom.size;
|
2015-05-26 15:24:39 +02:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
ret = ffio_read_size(pb, st->codecpar->extradata, atom.size);
|
2015-05-26 15:24:39 +02:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2006-02-11 23:43:21 +02:00
|
|
|
} else if (atom.size > 8) { /* to read frma, esds atoms */
|
2012-01-15 00:25:12 +03:00
|
|
|
if ((ret = mov_read_default(c, pb, atom)) < 0)
|
|
|
|
return ret;
|
2006-04-19 15:51:58 +03:00
|
|
|
} else
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, atom.size);
|
2005-10-18 23:16:43 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-12-19 17:39:10 +02:00
|
|
|
/**
|
|
|
|
* This function reads atom content and puts data in extradata without tag
|
|
|
|
* nor size unlike mov_read_extradata.
|
|
|
|
*/
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_glbl(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2004-12-09 21:44:55 +02:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2009-06-24 06:38:47 +03:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
2004-12-09 21:44:55 +02:00
|
|
|
|
2011-10-11 11:12:38 +03:00
|
|
|
if ((uint64_t)atom.size > (1<<30))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2005-01-08 16:21:33 +02:00
|
|
|
|
2011-12-14 05:49:06 +03:00
|
|
|
if (atom.size >= 10) {
|
2012-02-23 13:34:28 +03:00
|
|
|
// Broken files created by legacy versions of libavformat will
|
2011-12-14 05:49:06 +03:00
|
|
|
// wrap a whole fiel atom inside of a glbl atom.
|
|
|
|
unsigned size = avio_rb32(pb);
|
|
|
|
unsigned type = avio_rl32(pb);
|
|
|
|
avio_seek(pb, -8, SEEK_CUR);
|
|
|
|
if (type == MKTAG('f','i','e','l') && size == atom.size)
|
|
|
|
return mov_read_default(c, pb, atom);
|
|
|
|
}
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
av_free(st->codecpar->extradata);
|
|
|
|
st->codecpar->extradata = av_mallocz(atom.size + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!st->codecpar->extradata)
|
2008-02-25 14:42:32 +02:00
|
|
|
return AVERROR(ENOMEM);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->extradata_size = atom.size;
|
2015-05-26 15:24:39 +02:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
ret = ffio_read_size(pb, st->codecpar->extradata, atom.size);
|
2015-05-26 15:24:39 +02:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2004-12-09 21:44:55 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-22 02:32:16 +03:00
|
|
|
static int mov_read_dvc1(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
uint8_t profile_level;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2012-01-22 02:32:16 +03:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
|
|
|
|
if (atom.size >= (1<<28) || atom.size < 7)
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
|
|
|
|
profile_level = avio_r8(pb);
|
2012-06-18 22:06:35 +03:00
|
|
|
if ((profile_level & 0xf0) != 0xc0)
|
2012-01-22 02:32:16 +03:00
|
|
|
return 0;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
av_free(st->codecpar->extradata);
|
|
|
|
st->codecpar->extradata = av_mallocz(atom.size - 7 + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!st->codecpar->extradata)
|
2012-01-22 02:32:16 +03:00
|
|
|
return AVERROR(ENOMEM);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->extradata_size = atom.size - 7;
|
2012-01-22 02:32:16 +03:00
|
|
|
avio_seek(pb, 6, SEEK_CUR);
|
2015-05-26 15:24:39 +02:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
ret = ffio_read_size(pb, st->codecpar->extradata, st->codecpar->extradata_size);
|
2015-05-26 15:24:39 +02:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2012-01-22 02:32:16 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-16 23:36:24 +03:00
|
|
|
/**
|
|
|
|
* An strf atom is a BITMAPINFOHEADER struct. This struct is 40 bytes itself,
|
|
|
|
* but can have extradata appended at the end after the 40 bytes belonging
|
|
|
|
* to the struct.
|
|
|
|
*/
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_strf(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2010-04-16 23:36:24 +03:00
|
|
|
{
|
|
|
|
AVStream *st;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2010-04-16 23:36:24 +03:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
if (atom.size <= 40)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
|
2011-10-11 11:12:38 +03:00
|
|
|
if ((uint64_t)atom.size > (1<<30))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2010-04-16 23:36:24 +03:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
av_free(st->codecpar->extradata);
|
|
|
|
st->codecpar->extradata = av_mallocz(atom.size - 40 + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!st->codecpar->extradata)
|
2010-04-16 23:36:24 +03:00
|
|
|
return AVERROR(ENOMEM);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->extradata_size = atom.size - 40;
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, 40);
|
2015-05-26 15:24:39 +02:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
ret = ffio_read_size(pb, st->codecpar->extradata, atom.size - 40);
|
2015-05-26 15:24:39 +02:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2010-04-16 23:36:24 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_stco(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2003-05-12 13:59:18 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2005-01-08 16:21:33 +02:00
|
|
|
unsigned int i, entries;
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
entries = avio_rb32(pb);
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2011-12-05 03:38:05 +03:00
|
|
|
if (!entries)
|
|
|
|
return 0;
|
2011-10-11 11:12:38 +03:00
|
|
|
if (entries >= UINT_MAX/sizeof(int64_t))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2006-09-29 13:30:52 +03:00
|
|
|
sc->chunk_offsets = av_malloc(entries * sizeof(int64_t));
|
2003-05-12 13:59:18 +03:00
|
|
|
if (!sc->chunk_offsets)
|
2009-02-22 02:20:07 +02:00
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
sc->chunk_count = entries;
|
|
|
|
|
2008-04-25 15:02:01 +03:00
|
|
|
if (atom.type == MKTAG('s','t','c','o'))
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++)
|
2011-02-21 17:43:01 +02:00
|
|
|
sc->chunk_offsets[i] = avio_rb32(pb);
|
2008-04-25 15:02:01 +03:00
|
|
|
else if (atom.type == MKTAG('c','o','6','4'))
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++)
|
2011-02-21 17:43:01 +02:00
|
|
|
sc->chunk_offsets[i] = avio_rb64(pb);
|
2008-04-25 15:02:01 +03:00
|
|
|
else
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
sc->chunk_count = i;
|
|
|
|
|
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2003-05-12 13:59:18 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-08-03 21:37:35 +03:00
|
|
|
/**
|
|
|
|
* Compute codec id for 'lpcm' tag.
|
|
|
|
* See CoreAudioTypes and AudioStreamBasicDescription at Apple.
|
|
|
|
*/
|
2012-08-05 12:11:04 +03:00
|
|
|
enum AVCodecID ff_mov_get_lpcm_codec_id(int bps, int flags)
|
2008-08-03 21:37:35 +03:00
|
|
|
{
|
2012-11-27 22:52:38 +03:00
|
|
|
/* lpcm flags:
|
|
|
|
* 0x1 = float
|
|
|
|
* 0x2 = big-endian
|
|
|
|
* 0x4 = signed
|
|
|
|
*/
|
|
|
|
return ff_get_pcm_codec_id(bps, flags & 1, flags & 2, flags & 4 ? -1 : 0);
|
2008-08-03 21:37:35 +03:00
|
|
|
}
|
|
|
|
|
2013-08-22 16:00:46 +03:00
|
|
|
static int mov_codec_id(AVStream *st, uint32_t format)
|
|
|
|
{
|
|
|
|
int id = ff_codec_get_id(ff_codec_movaudio_tags, format);
|
|
|
|
|
|
|
|
if (id <= 0 &&
|
|
|
|
((format & 0xFFFF) == 'm' + ('s' << 8) ||
|
|
|
|
(format & 0xFFFF) == 'T' + ('S' << 8)))
|
|
|
|
id = ff_codec_get_id(ff_codec_wav_tags, av_bswap32(format) & 0xFFFF);
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_type != AVMEDIA_TYPE_VIDEO && id > 0) {
|
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
|
|
|
|
} else if (st->codecpar->codec_type != AVMEDIA_TYPE_AUDIO &&
|
2016-04-27 19:45:23 +02:00
|
|
|
/* skip old ASF MPEG-4 tag */
|
2013-08-22 16:00:46 +03:00
|
|
|
format && format != MKTAG('m','p','4','s')) {
|
|
|
|
id = ff_codec_get_id(ff_codec_movvideo_tags, format);
|
|
|
|
if (id <= 0)
|
|
|
|
id = ff_codec_get_id(ff_codec_bmp_tags, format);
|
|
|
|
if (id > 0)
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_VIDEO;
|
|
|
|
else if (st->codecpar->codec_type == AVMEDIA_TYPE_DATA) {
|
2013-08-22 16:00:46 +03:00
|
|
|
id = ff_codec_get_id(ff_codec_movsubtitle_tags, format);
|
|
|
|
if (id > 0)
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_SUBTITLE;
|
2013-08-22 16:00:46 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_tag = format;
|
2013-08-22 16:00:46 +03:00
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2013-08-22 16:05:27 +03:00
|
|
|
static void mov_parse_stsd_video(MOVContext *c, AVIOContext *pb,
|
|
|
|
AVStream *st, MOVStreamContext *sc)
|
|
|
|
{
|
2014-04-29 16:37:52 +03:00
|
|
|
uint8_t codec_name[32];
|
2013-08-22 16:05:27 +03:00
|
|
|
unsigned int color_depth, len, j;
|
|
|
|
int color_greyscale;
|
|
|
|
int color_table_id;
|
|
|
|
|
|
|
|
avio_rb16(pb); /* version */
|
|
|
|
avio_rb16(pb); /* revision level */
|
|
|
|
avio_rb32(pb); /* vendor */
|
|
|
|
avio_rb32(pb); /* temporal quality */
|
|
|
|
avio_rb32(pb); /* spatial quality */
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->width = avio_rb16(pb); /* width */
|
|
|
|
st->codecpar->height = avio_rb16(pb); /* height */
|
2013-08-22 16:05:27 +03:00
|
|
|
|
|
|
|
avio_rb32(pb); /* horiz resolution */
|
|
|
|
avio_rb32(pb); /* vert resolution */
|
|
|
|
avio_rb32(pb); /* data size, always 0 */
|
|
|
|
avio_rb16(pb); /* frames per samples */
|
|
|
|
|
|
|
|
len = avio_r8(pb); /* codec name, pascal string */
|
|
|
|
if (len > 31)
|
|
|
|
len = 31;
|
2014-04-29 16:37:52 +03:00
|
|
|
mov_read_mac_string(c, pb, len, codec_name, sizeof(codec_name));
|
2013-08-22 16:05:27 +03:00
|
|
|
if (len < 31)
|
|
|
|
avio_skip(pb, 31 - len);
|
2014-04-29 16:37:52 +03:00
|
|
|
|
|
|
|
if (codec_name[0])
|
|
|
|
av_dict_set(&st->metadata, "encoder", codec_name, 0);
|
|
|
|
|
2013-08-22 16:05:27 +03:00
|
|
|
/* codec_tag YV12 triggers an UV swap in rawdec.c */
|
2017-01-29 21:45:59 +02:00
|
|
|
if (!strncmp(codec_name, "Planar Y'CbCr 8-bit 4:2:0", 25))
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_tag = MKTAG('I', '4', '2', '0');
|
2016-04-27 19:45:23 +02:00
|
|
|
/* Flash Media Server uses tag H.263 with Sorenson Spark */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_tag == MKTAG('H','2','6','3') &&
|
2017-01-29 21:45:59 +02:00
|
|
|
!strncmp(codec_name, "Sorenson H263", 13))
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_FLV1;
|
2013-08-22 16:05:27 +03:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->bits_per_coded_sample = avio_rb16(pb); /* depth */
|
2013-08-22 16:05:27 +03:00
|
|
|
color_table_id = avio_rb16(pb); /* colortable id */
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "depth %d, ctab id %d\n",
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->bits_per_coded_sample, color_table_id);
|
2013-08-22 16:05:27 +03:00
|
|
|
/* figure out the palette situation */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
color_depth = st->codecpar->bits_per_coded_sample & 0x1F;
|
|
|
|
color_greyscale = st->codecpar->bits_per_coded_sample & 0x20;
|
2013-08-22 16:05:27 +03:00
|
|
|
|
|
|
|
/* if the depth is 2, 4, or 8 bpp, file is palettized */
|
|
|
|
if ((color_depth == 2) || (color_depth == 4) || (color_depth == 8)) {
|
|
|
|
/* for palette traversal */
|
|
|
|
unsigned int color_start, color_count, color_end;
|
|
|
|
unsigned char r, g, b;
|
|
|
|
|
|
|
|
if (color_greyscale) {
|
|
|
|
int color_index, color_dec;
|
|
|
|
/* compute the greyscale palette */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->bits_per_coded_sample = color_depth;
|
2013-08-22 16:05:27 +03:00
|
|
|
color_count = 1 << color_depth;
|
|
|
|
color_index = 255;
|
|
|
|
color_dec = 256 / (color_count - 1);
|
|
|
|
for (j = 0; j < color_count; j++) {
|
|
|
|
r = g = b = color_index;
|
|
|
|
sc->palette[j] = (r << 16) | (g << 8) | (b);
|
|
|
|
color_index -= color_dec;
|
|
|
|
if (color_index < 0)
|
|
|
|
color_index = 0;
|
|
|
|
}
|
|
|
|
} else if (color_table_id) {
|
|
|
|
const uint8_t *color_table;
|
|
|
|
/* if flag bit 3 is set, use the default palette */
|
|
|
|
color_count = 1 << color_depth;
|
|
|
|
if (color_depth == 2)
|
|
|
|
color_table = ff_qt_default_palette_4;
|
|
|
|
else if (color_depth == 4)
|
|
|
|
color_table = ff_qt_default_palette_16;
|
|
|
|
else
|
|
|
|
color_table = ff_qt_default_palette_256;
|
|
|
|
|
|
|
|
for (j = 0; j < color_count; j++) {
|
|
|
|
r = color_table[j * 3 + 0];
|
|
|
|
g = color_table[j * 3 + 1];
|
|
|
|
b = color_table[j * 3 + 2];
|
|
|
|
sc->palette[j] = (r << 16) | (g << 8) | (b);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* load the palette from the file */
|
|
|
|
color_start = avio_rb32(pb);
|
|
|
|
color_count = avio_rb16(pb);
|
|
|
|
color_end = avio_rb16(pb);
|
|
|
|
if ((color_start <= 255) && (color_end <= 255)) {
|
|
|
|
for (j = color_start; j <= color_end; j++) {
|
|
|
|
/* each R, G, or B component is 16 bits;
|
|
|
|
* only use the top 8 bits; skip alpha bytes
|
|
|
|
* up front */
|
|
|
|
avio_r8(pb);
|
|
|
|
avio_r8(pb);
|
|
|
|
r = avio_r8(pb);
|
|
|
|
avio_r8(pb);
|
|
|
|
g = avio_r8(pb);
|
|
|
|
avio_r8(pb);
|
|
|
|
b = avio_r8(pb);
|
|
|
|
avio_r8(pb);
|
|
|
|
sc->palette[j] = (r << 16) | (g << 8) | (b);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sc->has_palette = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-22 16:05:27 +03:00
|
|
|
static void mov_parse_stsd_audio(MOVContext *c, AVIOContext *pb,
|
|
|
|
AVStream *st, MOVStreamContext *sc)
|
|
|
|
{
|
|
|
|
int bits_per_sample, flags;
|
|
|
|
uint16_t version = avio_rb16(pb);
|
|
|
|
|
|
|
|
avio_rb16(pb); /* revision level */
|
|
|
|
avio_rb32(pb); /* vendor */
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->channels = avio_rb16(pb); /* channel count */
|
|
|
|
st->codecpar->bits_per_coded_sample = avio_rb16(pb); /* sample size */
|
|
|
|
av_log(c->fc, AV_LOG_TRACE, "audio channels %d\n", st->codecpar->channels);
|
2013-08-22 16:05:27 +03:00
|
|
|
|
|
|
|
sc->audio_cid = avio_rb16(pb);
|
|
|
|
avio_rb16(pb); /* packet size = 0 */
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->sample_rate = ((avio_rb32(pb) >> 16));
|
2013-08-22 16:05:27 +03:00
|
|
|
|
|
|
|
// Read QT version 1 fields. In version 0 these do not exist.
|
2015-12-11 18:29:11 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "version =%"PRIu16", isom =%d\n", version, c->isom);
|
2013-08-22 16:05:27 +03:00
|
|
|
if (!c->isom) {
|
|
|
|
if (version == 1) {
|
|
|
|
sc->samples_per_frame = avio_rb32(pb);
|
|
|
|
avio_rb32(pb); /* bytes per packet */
|
|
|
|
sc->bytes_per_frame = avio_rb32(pb);
|
|
|
|
avio_rb32(pb); /* bytes per sample */
|
|
|
|
} else if (version == 2) {
|
|
|
|
avio_rb32(pb); /* sizeof struct only */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->sample_rate = av_int2double(avio_rb64(pb));
|
|
|
|
st->codecpar->channels = avio_rb32(pb);
|
2013-08-22 16:05:27 +03:00
|
|
|
avio_rb32(pb); /* always 0x7F000000 */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->bits_per_coded_sample = avio_rb32(pb);
|
2013-08-22 16:05:27 +03:00
|
|
|
|
|
|
|
flags = avio_rb32(pb); /* lpcm format specific flag */
|
|
|
|
sc->bytes_per_frame = avio_rb32(pb);
|
|
|
|
sc->samples_per_frame = avio_rb32(pb);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_tag == MKTAG('l','p','c','m'))
|
|
|
|
st->codecpar->codec_id =
|
|
|
|
ff_mov_get_lpcm_codec_id(st->codecpar->bits_per_coded_sample,
|
2013-08-22 16:05:27 +03:00
|
|
|
flags);
|
|
|
|
}
|
2014-05-09 17:05:42 +03:00
|
|
|
if (version == 0 || (version == 1 && sc->audio_cid != -2)) {
|
|
|
|
/* can't correctly handle variable sized packet as audio unit */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
switch (st->codecpar->codec_id) {
|
2014-05-09 17:05:42 +03:00
|
|
|
case AV_CODEC_ID_MP2:
|
|
|
|
case AV_CODEC_ID_MP3:
|
|
|
|
st->need_parsing = AVSTREAM_PARSE_FULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-08-22 16:05:27 +03:00
|
|
|
}
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
switch (st->codecpar->codec_id) {
|
2013-08-22 16:05:27 +03:00
|
|
|
case AV_CODEC_ID_PCM_S8:
|
|
|
|
case AV_CODEC_ID_PCM_U8:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->bits_per_coded_sample == 16)
|
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_PCM_S16BE;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_PCM_S16LE:
|
|
|
|
case AV_CODEC_ID_PCM_S16BE:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->bits_per_coded_sample == 8)
|
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_PCM_S8;
|
|
|
|
else if (st->codecpar->bits_per_coded_sample == 24)
|
|
|
|
st->codecpar->codec_id =
|
|
|
|
st->codecpar->codec_id == AV_CODEC_ID_PCM_S16BE ?
|
2013-08-22 16:05:27 +03:00
|
|
|
AV_CODEC_ID_PCM_S24BE : AV_CODEC_ID_PCM_S24LE;
|
|
|
|
break;
|
|
|
|
/* set values for old format before stsd version 1 appeared */
|
|
|
|
case AV_CODEC_ID_MACE3:
|
|
|
|
sc->samples_per_frame = 6;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
sc->bytes_per_frame = 2 * st->codecpar->channels;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_MACE6:
|
|
|
|
sc->samples_per_frame = 6;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
sc->bytes_per_frame = 1 * st->codecpar->channels;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_ADPCM_IMA_QT:
|
|
|
|
sc->samples_per_frame = 64;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
sc->bytes_per_frame = 34 * st->codecpar->channels;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_GSM:
|
|
|
|
sc->samples_per_frame = 160;
|
|
|
|
sc->bytes_per_frame = 33;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
bits_per_sample = av_get_bits_per_sample(st->codecpar->codec_id);
|
2013-08-22 16:05:27 +03:00
|
|
|
if (bits_per_sample) {
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->bits_per_coded_sample = bits_per_sample;
|
|
|
|
sc->sample_size = (bits_per_sample >> 3) * st->codecpar->channels;
|
2013-08-22 16:05:27 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-22 16:05:27 +03:00
|
|
|
static void mov_parse_stsd_subtitle(MOVContext *c, AVIOContext *pb,
|
|
|
|
AVStream *st, MOVStreamContext *sc,
|
2016-02-10 00:07:18 +02:00
|
|
|
int64_t size)
|
2013-08-22 16:05:27 +03:00
|
|
|
{
|
|
|
|
// ttxt stsd contains display flags, justification, background
|
|
|
|
// color, fonts, and default styles, so fake an atom to read it
|
|
|
|
MOVAtom fake_atom = { .size = size };
|
|
|
|
// mp4s contains a regular esds atom
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_tag != AV_RL32("mp4s"))
|
2013-08-22 16:05:27 +03:00
|
|
|
mov_read_glbl(c, pb, fake_atom);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->width = sc->width;
|
|
|
|
st->codecpar->height = sc->height;
|
2013-08-22 16:05:27 +03:00
|
|
|
}
|
|
|
|
|
2014-03-20 10:05:37 +03:00
|
|
|
static uint32_t yuv_to_rgba(uint32_t ycbcr)
|
|
|
|
{
|
|
|
|
uint8_t r, g, b;
|
|
|
|
int y, cb, cr;
|
|
|
|
|
|
|
|
y = (ycbcr >> 16) & 0xFF;
|
|
|
|
cr = (ycbcr >> 8) & 0xFF;
|
|
|
|
cb = ycbcr & 0xFF;
|
|
|
|
|
|
|
|
b = av_clip_uint8(1.164 * (y - 16) + 2.018 * (cb - 128));
|
|
|
|
g = av_clip_uint8(1.164 * (y - 16) - 0.813 * (cr - 128) - 0.391 * (cb - 128));
|
|
|
|
r = av_clip_uint8(1.164 * (y - 16) + 1.596 * (cr - 128));
|
|
|
|
|
|
|
|
return (r << 16) | (g << 8) | b;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mov_rewrite_dvd_sub_extradata(AVStream *st)
|
|
|
|
{
|
|
|
|
char buf[256] = {0};
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
uint8_t *src = st->codecpar->extradata;
|
2014-03-20 10:05:37 +03:00
|
|
|
int i;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->extradata_size != 64)
|
2014-03-20 10:05:37 +03:00
|
|
|
return 0;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->width > 0 && st->codecpar->height > 0)
|
2014-03-20 10:05:37 +03:00
|
|
|
snprintf(buf, sizeof(buf), "size: %dx%d\n",
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->width, st->codecpar->height);
|
2014-03-20 10:05:37 +03:00
|
|
|
av_strlcat(buf, "palette: ", sizeof(buf));
|
|
|
|
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
|
|
uint32_t yuv = AV_RB32(src + i * 4);
|
|
|
|
uint32_t rgba = yuv_to_rgba(yuv);
|
|
|
|
|
2014-04-01 21:19:59 +03:00
|
|
|
av_strlcatf(buf, sizeof(buf), "%06"PRIx32"%s", rgba, i != 15 ? ", " : "");
|
2014-03-20 10:05:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (av_strlcat(buf, "\n", sizeof(buf)) >= sizeof(buf))
|
|
|
|
return 0;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
av_freep(&st->codecpar->extradata);
|
|
|
|
st->codecpar->extradata_size = 0;
|
|
|
|
st->codecpar->extradata = av_mallocz(strlen(buf) + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!st->codecpar->extradata)
|
2014-03-20 10:05:37 +03:00
|
|
|
return AVERROR(ENOMEM);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->extradata_size = strlen(buf);
|
|
|
|
memcpy(st->codecpar->extradata, buf, st->codecpar->extradata_size);
|
2014-03-20 10:05:37 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-25 14:02:07 +03:00
|
|
|
static int mov_parse_stsd_data(MOVContext *c, AVIOContext *pb,
|
|
|
|
AVStream *st, MOVStreamContext *sc,
|
2016-02-10 00:07:18 +02:00
|
|
|
int64_t size)
|
2013-08-25 14:02:07 +03:00
|
|
|
{
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_tag == MKTAG('t','m','c','d')) {
|
|
|
|
st->codecpar->extradata_size = size;
|
|
|
|
st->codecpar->extradata = av_malloc(size + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!st->codecpar->extradata)
|
2013-08-25 14:02:07 +03:00
|
|
|
return AVERROR(ENOMEM);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
ret = ffio_read_size(pb, st->codecpar->extradata, size);
|
2015-05-26 15:24:39 +02:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2013-08-25 14:02:07 +03:00
|
|
|
} else {
|
|
|
|
/* other codec type, just skip (rtp, mp4s ...) */
|
|
|
|
avio_skip(pb, size);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-22 16:05:27 +03:00
|
|
|
static int mov_finalize_stsd_codec(MOVContext *c, AVIOContext *pb,
|
|
|
|
AVStream *st, MOVStreamContext *sc)
|
|
|
|
{
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO &&
|
|
|
|
!st->codecpar->sample_rate && sc->time_scale > 1)
|
|
|
|
st->codecpar->sample_rate = sc->time_scale;
|
2013-08-22 16:05:27 +03:00
|
|
|
|
|
|
|
/* special codec parameters handling */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
switch (st->codecpar->codec_id) {
|
2013-08-22 16:05:27 +03:00
|
|
|
#if CONFIG_DV_DEMUXER
|
|
|
|
case AV_CODEC_ID_DVAUDIO:
|
2015-06-03 15:16:49 +02:00
|
|
|
c->dv_fctx = avformat_alloc_context();
|
|
|
|
if (!c->dv_fctx) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "dv demux context alloc error\n");
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
}
|
2013-08-22 16:05:27 +03:00
|
|
|
c->dv_demux = avpriv_dv_init_demux(c->dv_fctx);
|
|
|
|
if (!c->dv_demux) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "dv demux context init error\n");
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
}
|
|
|
|
sc->dv_audio_container = 1;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_id = AV_CODEC_ID_PCM_S16LE;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
/* no ifdef since parameters are always those */
|
|
|
|
case AV_CODEC_ID_QCELP:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->channels = 1;
|
2013-08-22 16:05:27 +03:00
|
|
|
// force sample rate for qcelp when not stored in mov
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_tag != MKTAG('Q','c','l','p'))
|
|
|
|
st->codecpar->sample_rate = 8000;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_AMR_NB:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->channels = 1;
|
2013-08-22 16:05:27 +03:00
|
|
|
/* force sample rate for amr, stsd in 3gp does not store sample rate */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->sample_rate = 8000;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_AMR_WB:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->channels = 1;
|
|
|
|
st->codecpar->sample_rate = 16000;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_MP2:
|
|
|
|
case AV_CODEC_ID_MP3:
|
|
|
|
/* force type after stsd for m1a hdlr */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_GSM:
|
|
|
|
case AV_CODEC_ID_ADPCM_MS:
|
|
|
|
case AV_CODEC_ID_ADPCM_IMA_WAV:
|
|
|
|
case AV_CODEC_ID_ILBC:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->block_align = sc->bytes_per_frame;
|
2013-08-22 16:05:27 +03:00
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_ALAC:
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->extradata_size == 36) {
|
|
|
|
st->codecpar->channels = AV_RB8 (st->codecpar->extradata + 21);
|
|
|
|
st->codecpar->sample_rate = AV_RB32(st->codecpar->extradata + 32);
|
2013-08-22 16:05:27 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case AV_CODEC_ID_VC1:
|
|
|
|
st->need_parsing = AVSTREAM_PARSE_FULL;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-22 16:05:27 +03:00
|
|
|
static int mov_skip_multiple_stsd(MOVContext *c, AVIOContext *pb,
|
|
|
|
int codec_tag, int format,
|
2016-02-10 00:07:18 +02:00
|
|
|
int64_t size)
|
2013-08-22 16:05:27 +03:00
|
|
|
{
|
|
|
|
int video_codec_id = ff_codec_get_id(ff_codec_movvideo_tags, format);
|
|
|
|
|
|
|
|
if (codec_tag &&
|
2016-11-02 18:03:18 +02:00
|
|
|
(codec_tag != format &&
|
|
|
|
// prores is allowed to have differing data format and codec tag
|
|
|
|
codec_tag != AV_RL32("apcn") && codec_tag != AV_RL32("apch") &&
|
2016-11-08 05:30:22 +02:00
|
|
|
// so is dv (sigh)
|
|
|
|
codec_tag != AV_RL32("dvpp") && codec_tag != AV_RL32("dvcp") &&
|
2016-11-02 18:03:18 +02:00
|
|
|
(c->fc->video_codec_id ? video_codec_id != c->fc->video_codec_id
|
|
|
|
: codec_tag != MKTAG('j','p','e','g')))) {
|
2013-08-22 16:05:27 +03:00
|
|
|
/* Multiple fourcc, we skip JPEG. This is not correct, we should
|
|
|
|
* export it as a separate AVStream but this needs a few changes
|
|
|
|
* in the MOV demuxer, patch welcome. */
|
|
|
|
|
|
|
|
av_log(c->fc, AV_LOG_WARNING, "multiple fourcc not supported\n");
|
|
|
|
avio_skip(pb, size);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
int ff_mov_read_stsd_entries(MOVContext *c, AVIOContext *pb, int entries)
|
2003-05-12 13:59:18 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2013-08-22 16:05:27 +03:00
|
|
|
int pseudo_stream_id;
|
2003-11-10 05:17:32 +02:00
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
for (pseudo_stream_id = 0;
|
|
|
|
pseudo_stream_id < entries && !pb->eof_reached;
|
|
|
|
pseudo_stream_id++) {
|
2008-03-21 13:16:16 +02:00
|
|
|
//Parsing Sample description table
|
2012-08-05 12:11:04 +03:00
|
|
|
enum AVCodecID id;
|
2013-08-25 14:02:07 +03:00
|
|
|
int ret, dref_id = 1;
|
2010-06-25 04:19:03 +03:00
|
|
|
MOVAtom a = { AV_RL32("stsd") };
|
2011-03-03 21:11:45 +02:00
|
|
|
int64_t start_pos = avio_tell(pb);
|
2016-02-10 00:07:18 +02:00
|
|
|
int64_t size = avio_rb32(pb); /* size */
|
2011-02-21 17:43:01 +02:00
|
|
|
uint32_t format = avio_rl32(pb); /* data format */
|
2003-04-22 11:36:39 +03:00
|
|
|
|
2009-06-04 02:13:53 +03:00
|
|
|
if (size >= 16) {
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); /* reserved */
|
|
|
|
avio_rb16(pb); /* reserved */
|
|
|
|
dref_id = avio_rb16(pb);
|
2011-12-15 22:51:00 +03:00
|
|
|
} else {
|
2016-02-16 04:46:16 +02:00
|
|
|
av_log(c->fc, AV_LOG_ERROR,
|
|
|
|
"invalid size %"PRId64" in stsd\n", size);
|
2011-12-15 22:51:00 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-06-04 02:13:53 +03:00
|
|
|
}
|
2002-10-14 20:00:34 +03:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (mov_skip_multiple_stsd(c, pb, st->codecpar->codec_tag, format,
|
2013-08-22 16:05:27 +03:00
|
|
|
size - (avio_tell(pb) - start_pos)))
|
2006-07-07 11:56:05 +03:00
|
|
|
continue;
|
2013-08-22 16:05:27 +03:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
sc->pseudo_stream_id = st->codecpar->codec_tag ? -1 : pseudo_stream_id;
|
2008-02-26 20:21:58 +02:00
|
|
|
sc->dref_id= dref_id;
|
2006-07-07 11:56:05 +03:00
|
|
|
|
2013-08-22 16:00:46 +03:00
|
|
|
id = mov_codec_id(st, format);
|
2006-03-24 18:52:56 +02:00
|
|
|
|
2015-06-30 15:38:48 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE,
|
2015-12-11 18:29:11 +02:00
|
|
|
"size=%"PRId64" format=0x%08"PRIx32" codec_type=%d\n",
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
size, format, st->codecpar->codec_type);
|
2006-05-01 18:52:57 +03:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_type==AVMEDIA_TYPE_VIDEO) {
|
|
|
|
st->codecpar->codec_id = id;
|
2013-08-22 16:05:27 +03:00
|
|
|
mov_parse_stsd_video(c, pb, st, sc);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
} else if (st->codecpar->codec_type==AVMEDIA_TYPE_AUDIO) {
|
|
|
|
st->codecpar->codec_id = id;
|
2013-08-22 16:05:27 +03:00
|
|
|
mov_parse_stsd_audio(c, pb, st, sc);
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
} else if (st->codecpar->codec_type==AVMEDIA_TYPE_SUBTITLE){
|
|
|
|
st->codecpar->codec_id = id;
|
2013-08-22 16:05:27 +03:00
|
|
|
mov_parse_stsd_subtitle(c, pb, st, sc,
|
|
|
|
size - (avio_tell(pb) - start_pos));
|
2006-02-19 22:00:00 +02:00
|
|
|
} else {
|
2013-08-25 14:02:07 +03:00
|
|
|
ret = mov_parse_stsd_data(c, pb, st, sc,
|
|
|
|
size - (avio_tell(pb) - start_pos));
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2002-04-08 15:32:01 +03:00
|
|
|
}
|
2013-10-03 08:03:53 +03:00
|
|
|
/* this will read extra atoms at the end (wave, alac, damr, avcC, hvcC, SMI ...) */
|
2011-03-03 21:11:45 +02:00
|
|
|
a.size = size - (avio_tell(pb) - start_pos);
|
2007-12-04 00:22:50 +02:00
|
|
|
if (a.size > 8) {
|
2012-01-15 00:25:12 +03:00
|
|
|
if ((ret = mov_read_default(c, pb, a)) < 0)
|
|
|
|
return ret;
|
2007-12-04 00:22:50 +02:00
|
|
|
} else if (a.size > 0)
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, a.size);
|
2016-06-15 21:24:30 +02:00
|
|
|
|
|
|
|
if (sc->extradata) {
|
|
|
|
int extra_size = st->codecpar->extradata_size;
|
|
|
|
|
|
|
|
/* Move the current stream extradata to the stream context one. */
|
|
|
|
sc->extradata_size[pseudo_stream_id] = extra_size;
|
|
|
|
sc->extradata[pseudo_stream_id] = av_malloc(extra_size + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!sc->extradata[pseudo_stream_id])
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
memcpy(sc->extradata[pseudo_stream_id], st->codecpar->extradata, extra_size);
|
|
|
|
av_freep(&st->codecpar->extradata);
|
|
|
|
st->codecpar->extradata_size = 0;
|
|
|
|
}
|
2002-04-08 15:32:01 +03:00
|
|
|
}
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2013-08-22 16:05:27 +03:00
|
|
|
return mov_finalize_stsd_codec(c, pb, st, sc);
|
2002-04-08 15:32:01 +03:00
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_stsd(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2010-10-06 00:35:44 +03:00
|
|
|
{
|
2016-06-15 21:24:30 +02:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2016-06-29 18:03:59 +02:00
|
|
|
int ret, entries;
|
2016-06-15 21:24:30 +02:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams - 1];
|
|
|
|
sc = st->priv_data;
|
2010-10-06 00:35:44 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
2016-06-29 18:03:59 +02:00
|
|
|
entries = avio_rb32(pb);
|
2016-06-15 21:24:30 +02:00
|
|
|
|
2017-02-11 23:44:08 +02:00
|
|
|
if (sc->extradata) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR,
|
|
|
|
"Duplicate stsd found in this track.\n");
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
}
|
|
|
|
|
2016-06-15 21:24:30 +02:00
|
|
|
/* Prepare space for hosting multiple extradata. */
|
2016-06-29 18:03:59 +02:00
|
|
|
sc->extradata = av_mallocz_array(entries, sizeof(*sc->extradata));
|
2016-06-15 21:24:30 +02:00
|
|
|
if (!sc->extradata)
|
|
|
|
return AVERROR(ENOMEM);
|
2010-10-06 00:35:44 +03:00
|
|
|
|
2016-06-29 18:03:59 +02:00
|
|
|
sc->stsd_count = entries;
|
2016-07-15 10:00:34 +02:00
|
|
|
sc->extradata_size = av_mallocz_array(sc->stsd_count, sizeof(*sc->extradata_size));
|
2016-06-15 21:24:30 +02:00
|
|
|
if (!sc->extradata_size)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
|
|
|
ret = ff_mov_read_stsd_entries(c, pb, sc->stsd_count);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* Restore back the primary extradata. */
|
|
|
|
av_free(st->codecpar->extradata);
|
|
|
|
st->codecpar->extradata_size = sc->extradata_size[0];
|
|
|
|
st->codecpar->extradata = av_mallocz(sc->extradata_size[0] + AV_INPUT_BUFFER_PADDING_SIZE);
|
|
|
|
if (!st->codecpar->extradata)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
memcpy(st->codecpar->extradata, sc->extradata[0], sc->extradata_size[0]);
|
|
|
|
|
|
|
|
return 0;
|
2010-10-06 00:35:44 +03:00
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_stsc(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2002-04-08 15:32:01 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2005-01-08 16:21:33 +02:00
|
|
|
unsigned int i, entries;
|
2003-04-23 11:58:42 +03:00
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
2002-04-08 15:32:01 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
entries = avio_rb32(pb);
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "track[%u].stsc.entries = %u\n", c->fc->nb_streams - 1, entries);
|
2007-04-26 14:24:10 +03:00
|
|
|
|
2011-12-05 03:38:05 +03:00
|
|
|
if (!entries)
|
|
|
|
return 0;
|
2011-10-11 11:12:38 +03:00
|
|
|
if (entries >= UINT_MAX / sizeof(*sc->stsc_data))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-02-08 11:59:31 +02:00
|
|
|
sc->stsc_data = av_malloc(entries * sizeof(*sc->stsc_data));
|
|
|
|
if (!sc->stsc_data)
|
2009-02-22 02:20:07 +02:00
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++) {
|
2011-02-21 17:43:01 +02:00
|
|
|
sc->stsc_data[i].first = avio_rb32(pb);
|
|
|
|
sc->stsc_data[i].count = avio_rb32(pb);
|
|
|
|
sc->stsc_data[i].id = avio_rb32(pb);
|
2016-07-26 11:42:11 +02:00
|
|
|
if (sc->stsc_data[i].id <= 0 || sc->stsc_data[i].id > sc->stsd_count) {
|
|
|
|
sc->stsc_data[i].id = 1;
|
2016-07-13 21:02:41 +02:00
|
|
|
if (c->fc->error_recognition & AV_EF_EXPLODE) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Invalid stsc index.\n");
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
}
|
|
|
|
}
|
2002-04-08 15:32:01 +03:00
|
|
|
}
|
2012-09-16 21:32:12 +03:00
|
|
|
|
|
|
|
sc->stsc_count = i;
|
|
|
|
|
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2002-04-08 15:32:01 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-03 14:05:27 +02:00
|
|
|
static inline int mov_stsc_index_valid(unsigned int index, unsigned int count)
|
2016-07-20 18:48:09 +02:00
|
|
|
{
|
2017-02-03 14:05:27 +02:00
|
|
|
return index < count - 1;
|
2016-07-20 18:48:09 +02:00
|
|
|
}
|
|
|
|
|
2016-06-15 21:24:30 +02:00
|
|
|
/* Compute the samples value for the stsc entry at the given index. */
|
2017-02-03 14:05:27 +02:00
|
|
|
static inline int mov_get_stsc_samples(MOVStreamContext *sc, unsigned int index)
|
2016-06-15 21:24:30 +02:00
|
|
|
{
|
|
|
|
int chunk_count;
|
|
|
|
|
2016-07-20 18:48:09 +02:00
|
|
|
if (mov_stsc_index_valid(index, sc->stsc_count))
|
2016-06-15 21:24:30 +02:00
|
|
|
chunk_count = sc->stsc_data[index + 1].first - sc->stsc_data[index].first;
|
|
|
|
else
|
|
|
|
chunk_count = sc->chunk_count - (sc->stsc_data[index].first - 1);
|
|
|
|
|
|
|
|
return sc->stsc_data[index].count * chunk_count;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_stps(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2009-05-13 11:31:22 +03:00
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
unsigned i, entries;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); // version + flags
|
2009-05-13 11:31:22 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
entries = avio_rb32(pb);
|
2009-05-13 11:31:22 +03:00
|
|
|
if (entries >= UINT_MAX / sizeof(*sc->stps_data))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-05-13 11:31:22 +03:00
|
|
|
sc->stps_data = av_malloc(entries * sizeof(*sc->stps_data));
|
|
|
|
if (!sc->stps_data)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++) {
|
2011-02-21 17:43:01 +02:00
|
|
|
sc->stps_data[i] = avio_rb32(pb);
|
2009-05-13 11:31:22 +03:00
|
|
|
}
|
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
sc->stps_count = i;
|
|
|
|
|
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2009-05-13 11:31:22 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_stss(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2004-03-13 23:02:26 +02:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2005-01-08 16:21:33 +02:00
|
|
|
unsigned int i, entries;
|
2004-03-13 23:02:26 +02:00
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
2004-03-13 23:02:26 +02:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
entries = avio_rb32(pb);
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "keyframe_count = %u\n", entries);
|
2009-02-22 02:20:07 +02:00
|
|
|
|
2012-02-03 21:43:21 +03:00
|
|
|
if (!entries)
|
2012-04-13 12:59:14 +03:00
|
|
|
{
|
|
|
|
sc->keyframe_absent = 1;
|
2012-02-03 21:43:21 +03:00
|
|
|
return 0;
|
2012-04-13 12:59:14 +03:00
|
|
|
}
|
2011-10-11 11:12:38 +03:00
|
|
|
if (entries >= UINT_MAX / sizeof(int))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2014-08-12 17:39:10 +03:00
|
|
|
av_freep(&sc->keyframes);
|
2007-04-21 21:37:10 +03:00
|
|
|
sc->keyframes = av_malloc(entries * sizeof(int));
|
2004-03-13 23:02:26 +02:00
|
|
|
if (!sc->keyframes)
|
2009-02-22 02:20:07 +02:00
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++) {
|
2011-02-21 17:43:01 +02:00
|
|
|
sc->keyframes[i] = avio_rb32(pb);
|
2004-03-13 23:02:26 +02:00
|
|
|
}
|
2012-09-16 21:32:12 +03:00
|
|
|
|
|
|
|
sc->keyframe_count = i;
|
|
|
|
|
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2004-03-13 23:02:26 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_stsz(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2002-04-08 15:32:01 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2009-03-16 18:14:36 +02:00
|
|
|
unsigned int i, entries, sample_size, field_size, num_bytes;
|
2016-04-16 12:17:24 +02:00
|
|
|
BitstreamContext bc;
|
2009-03-16 18:14:36 +02:00
|
|
|
unsigned char* buf;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2003-04-23 11:58:42 +03:00
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
2003-04-22 11:36:39 +03:00
|
|
|
|
2009-03-16 18:14:36 +02:00
|
|
|
if (atom.type == MKTAG('s','t','s','z')) {
|
2011-02-21 17:43:01 +02:00
|
|
|
sample_size = avio_rb32(pb);
|
2009-03-16 18:24:38 +02:00
|
|
|
if (!sc->sample_size) /* do not overwrite value computed in stsd */
|
|
|
|
sc->sample_size = sample_size;
|
|
|
|
field_size = 32;
|
2009-03-16 18:14:36 +02:00
|
|
|
} else {
|
|
|
|
sample_size = 0;
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb24(pb); /* reserved */
|
|
|
|
field_size = avio_r8(pb);
|
2009-03-16 18:14:36 +02:00
|
|
|
}
|
2011-02-21 17:43:01 +02:00
|
|
|
entries = avio_rb32(pb);
|
2009-02-22 02:20:07 +02:00
|
|
|
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "sample_size = %u sample_count = %u\n", sc->sample_size, entries);
|
2005-01-08 16:21:33 +02:00
|
|
|
|
2002-04-08 15:32:01 +03:00
|
|
|
sc->sample_count = entries;
|
2006-07-08 15:14:41 +03:00
|
|
|
if (sample_size)
|
|
|
|
return 0;
|
|
|
|
|
2009-03-16 18:14:36 +02:00
|
|
|
if (field_size != 4 && field_size != 8 && field_size != 16 && field_size != 32) {
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Invalid sample field size %u\n", field_size);
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-03-16 18:14:36 +02:00
|
|
|
}
|
|
|
|
|
2011-12-05 03:38:05 +03:00
|
|
|
if (!entries)
|
|
|
|
return 0;
|
2009-09-08 01:42:51 +03:00
|
|
|
if (entries >= UINT_MAX / sizeof(int) || entries >= (UINT_MAX - 4) / field_size)
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2007-04-21 21:37:10 +03:00
|
|
|
sc->sample_sizes = av_malloc(entries * sizeof(int));
|
2003-04-23 11:58:42 +03:00
|
|
|
if (!sc->sample_sizes)
|
2009-02-22 02:20:07 +02:00
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
2009-03-16 18:14:36 +02:00
|
|
|
num_bytes = (entries*field_size+4)>>3;
|
|
|
|
|
2015-06-29 23:48:34 +02:00
|
|
|
buf = av_malloc(num_bytes+AV_INPUT_BUFFER_PADDING_SIZE);
|
2009-03-16 18:14:36 +02:00
|
|
|
if (!buf) {
|
|
|
|
av_freep(&sc->sample_sizes);
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
}
|
|
|
|
|
2015-05-26 15:24:39 +02:00
|
|
|
ret = ffio_read_size(pb, buf, num_bytes);
|
|
|
|
if (ret < 0) {
|
2009-03-16 18:14:36 +02:00
|
|
|
av_freep(&sc->sample_sizes);
|
|
|
|
av_free(buf);
|
2015-05-26 15:24:39 +02:00
|
|
|
return ret;
|
2009-03-16 18:14:36 +02:00
|
|
|
}
|
|
|
|
|
2016-06-06 13:20:17 +02:00
|
|
|
bitstream_init8(&bc, buf, num_bytes);
|
2009-03-16 18:14:36 +02:00
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++) {
|
2016-04-16 12:17:24 +02:00
|
|
|
sc->sample_sizes[i] = bitstream_read(&bc, field_size);
|
2012-01-13 15:40:24 +03:00
|
|
|
sc->data_size += sc->sample_sizes[i];
|
|
|
|
}
|
2009-03-16 18:14:36 +02:00
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
sc->sample_count = i;
|
|
|
|
|
2015-05-26 15:24:36 +02:00
|
|
|
av_free(buf);
|
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2002-04-08 15:32:01 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_stts(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2002-07-20 23:05:50 +03:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2005-01-08 16:21:33 +02:00
|
|
|
unsigned int i, entries;
|
2004-03-24 21:03:20 +02:00
|
|
|
int64_t duration=0;
|
|
|
|
int64_t total_sample_count=0;
|
2003-04-23 11:58:42 +03:00
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
|
|
|
entries = avio_rb32(pb);
|
2009-02-22 02:20:07 +02:00
|
|
|
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "track[%u].stts.entries = %u\n",
|
2011-10-11 11:12:38 +03:00
|
|
|
c->fc->nb_streams-1, entries);
|
2009-02-22 02:20:07 +02:00
|
|
|
|
2011-12-03 04:00:11 +03:00
|
|
|
if (!entries)
|
|
|
|
return 0;
|
|
|
|
if (entries >= UINT_MAX / sizeof(*sc->stts_data))
|
2011-10-11 11:14:06 +03:00
|
|
|
return AVERROR(EINVAL);
|
2011-10-11 11:12:38 +03:00
|
|
|
|
2014-01-13 15:46:07 +03:00
|
|
|
av_free(sc->stts_data);
|
2008-12-11 22:06:56 +02:00
|
|
|
sc->stts_data = av_malloc(entries * sizeof(*sc->stts_data));
|
2007-12-04 00:26:07 +02:00
|
|
|
if (!sc->stts_data)
|
2009-02-22 02:20:07 +02:00
|
|
|
return AVERROR(ENOMEM);
|
2011-10-11 11:12:38 +03:00
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++) {
|
2005-02-07 15:50:39 +02:00
|
|
|
int sample_duration;
|
|
|
|
int sample_count;
|
2002-07-20 23:05:50 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
sample_count=avio_rb32(pb);
|
|
|
|
sample_duration = avio_rb32(pb);
|
2013-09-28 23:57:36 +03:00
|
|
|
if (sample_count < 0) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Invalid sample_count=%d\n", sample_count);
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
}
|
2006-02-02 22:56:35 +02:00
|
|
|
sc->stts_data[i].count= sample_count;
|
|
|
|
sc->stts_data[i].duration= sample_duration;
|
|
|
|
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "sample_count=%d, sample_duration=%d\n",
|
2011-10-11 11:12:38 +03:00
|
|
|
sample_count, sample_duration);
|
2006-02-12 12:24:43 +02:00
|
|
|
|
2006-03-23 14:11:51 +02:00
|
|
|
duration+=(int64_t)sample_duration*sample_count;
|
2003-05-14 14:40:16 +03:00
|
|
|
total_sample_count+=sample_count;
|
|
|
|
}
|
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
sc->stts_count = i;
|
|
|
|
|
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2005-06-24 14:43:08 +03:00
|
|
|
st->nb_frames= total_sample_count;
|
2011-10-11 11:12:38 +03:00
|
|
|
if (duration)
|
2005-06-24 14:43:08 +03:00
|
|
|
st->duration= duration;
|
2012-02-02 18:30:27 +03:00
|
|
|
sc->track_end = duration;
|
2002-07-20 23:05:50 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_ctts(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2005-02-07 18:58:15 +02:00
|
|
|
{
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2005-02-07 18:58:15 +02:00
|
|
|
unsigned int i, entries;
|
|
|
|
|
2009-06-24 06:38:47 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
|
|
|
entries = avio_rb32(pb);
|
2009-02-22 02:20:07 +02:00
|
|
|
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "track[%u].ctts.entries = %u\n", c->fc->nb_streams - 1, entries);
|
2009-02-22 02:20:07 +02:00
|
|
|
|
2015-05-26 15:24:36 +02:00
|
|
|
av_freep(&sc->ctts_data);
|
|
|
|
|
2011-12-05 03:38:05 +03:00
|
|
|
if (!entries)
|
|
|
|
return 0;
|
2011-10-11 11:12:38 +03:00
|
|
|
if (entries >= UINT_MAX / sizeof(*sc->ctts_data))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2015-05-31 12:28:33 +02:00
|
|
|
sc->ctts_data = av_realloc(NULL, entries * sizeof(*sc->ctts_data));
|
2007-12-04 00:26:07 +02:00
|
|
|
if (!sc->ctts_data)
|
2009-02-22 02:20:07 +02:00
|
|
|
return AVERROR(ENOMEM);
|
2006-02-12 12:24:43 +02:00
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++) {
|
2011-02-21 17:43:01 +02:00
|
|
|
int count =avio_rb32(pb);
|
|
|
|
int duration =avio_rb32(pb);
|
2006-03-11 13:30:07 +02:00
|
|
|
|
|
|
|
sc->ctts_data[i].count = count;
|
|
|
|
sc->ctts_data[i].duration= duration;
|
2009-11-05 09:30:53 +02:00
|
|
|
if (duration < 0)
|
|
|
|
sc->dts_shift = FFMAX(sc->dts_shift, -duration);
|
2005-02-07 18:58:15 +02:00
|
|
|
}
|
2009-11-05 09:30:53 +02:00
|
|
|
|
2012-09-16 21:32:12 +03:00
|
|
|
sc->ctts_count = i;
|
|
|
|
|
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "dts shift %d\n", sc->dts_shift);
|
2009-11-05 09:30:53 +02:00
|
|
|
|
2005-02-07 18:58:15 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-04-24 02:51:29 +03:00
|
|
|
static int mov_read_sbgp(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
unsigned int i, entries;
|
|
|
|
uint8_t version;
|
|
|
|
uint32_t grouping_type;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
|
|
|
version = avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
|
|
|
grouping_type = avio_rl32(pb);
|
|
|
|
if (grouping_type != MKTAG( 'r','a','p',' '))
|
|
|
|
return 0; /* only support 'rap ' grouping */
|
|
|
|
if (version == 1)
|
|
|
|
avio_rb32(pb); /* grouping_type_parameter */
|
|
|
|
|
|
|
|
entries = avio_rb32(pb);
|
|
|
|
if (!entries)
|
|
|
|
return 0;
|
|
|
|
if (entries >= UINT_MAX / sizeof(*sc->rap_group))
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
sc->rap_group = av_malloc(entries * sizeof(*sc->rap_group));
|
|
|
|
if (!sc->rap_group)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++) {
|
|
|
|
sc->rap_group[i].count = avio_rb32(pb); /* sample_count */
|
|
|
|
sc->rap_group[i].index = avio_rb32(pb); /* group_description_index */
|
|
|
|
}
|
|
|
|
|
|
|
|
sc->rap_group_count = i;
|
|
|
|
|
|
|
|
return pb->eof_reached ? AVERROR_EOF : 0;
|
|
|
|
}
|
|
|
|
|
2008-03-20 17:57:11 +02:00
|
|
|
static void mov_build_index(MOVContext *mov, AVStream *st)
|
|
|
|
{
|
|
|
|
MOVStreamContext *sc = st->priv_data;
|
2008-10-03 13:16:29 +03:00
|
|
|
int64_t current_offset;
|
2008-03-20 17:57:11 +02:00
|
|
|
int64_t current_dts = 0;
|
|
|
|
unsigned int stts_index = 0;
|
|
|
|
unsigned int stsc_index = 0;
|
|
|
|
unsigned int stss_index = 0;
|
2009-05-13 11:31:22 +03:00
|
|
|
unsigned int stps_index = 0;
|
2008-03-20 17:57:11 +02:00
|
|
|
unsigned int i, j;
|
2009-10-24 23:40:00 +03:00
|
|
|
uint64_t stream_size = 0;
|
2008-03-20 17:57:11 +02:00
|
|
|
|
2009-02-08 06:27:07 +02:00
|
|
|
/* adjust first dts according to edit list */
|
2011-01-11 23:06:37 +02:00
|
|
|
if (sc->time_offset && mov->time_scale > 0) {
|
2011-05-13 17:46:48 +03:00
|
|
|
if (sc->time_offset < 0)
|
|
|
|
sc->time_offset = av_rescale(sc->time_offset, sc->time_scale, mov->time_scale);
|
|
|
|
current_dts = -sc->time_offset;
|
2009-02-08 06:27:07 +02:00
|
|
|
}
|
|
|
|
|
2008-03-27 14:35:22 +02:00
|
|
|
/* only use old uncompressed audio chunk demuxing when stts specifies it */
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (!(st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO &&
|
2008-03-27 14:35:22 +02:00
|
|
|
sc->stts_count == 1 && sc->stts_data[0].duration == 1)) {
|
2008-03-20 17:57:11 +02:00
|
|
|
unsigned int current_sample = 0;
|
|
|
|
unsigned int stts_sample = 0;
|
2009-05-13 11:31:22 +03:00
|
|
|
unsigned int sample_size;
|
2008-03-20 17:57:11 +02:00
|
|
|
unsigned int distance = 0;
|
2012-04-24 02:51:29 +03:00
|
|
|
unsigned int rap_group_index = 0;
|
|
|
|
unsigned int rap_group_sample = 0;
|
|
|
|
int rap_group_present = sc->rap_group_count && sc->rap_group;
|
2012-04-16 17:27:45 +03:00
|
|
|
int key_off = (sc->keyframes && sc->keyframes[0] > 0) || (sc->stps_data && sc->stps_data[0] > 0);
|
2008-03-20 17:57:11 +02:00
|
|
|
|
2015-02-28 23:51:16 +02:00
|
|
|
current_dts -= sc->dts_shift;
|
|
|
|
|
2011-12-05 03:38:05 +03:00
|
|
|
if (!sc->sample_count)
|
|
|
|
return;
|
2012-03-28 22:56:07 +03:00
|
|
|
if (sc->sample_count >= UINT_MAX / sizeof(*st->index_entries) - st->nb_index_entries)
|
2010-03-09 02:46:26 +02:00
|
|
|
return;
|
2013-09-10 12:57:35 +03:00
|
|
|
if (av_reallocp_array(&st->index_entries,
|
|
|
|
st->nb_index_entries + sc->sample_count,
|
|
|
|
sizeof(*st->index_entries)) < 0) {
|
|
|
|
st->nb_index_entries = 0;
|
2010-03-09 02:46:26 +02:00
|
|
|
return;
|
2013-09-10 12:57:35 +03:00
|
|
|
}
|
2012-03-28 22:56:07 +03:00
|
|
|
st->index_entries_allocated_size = (st->nb_index_entries + sc->sample_count) * sizeof(*st->index_entries);
|
2010-03-09 02:46:26 +02:00
|
|
|
|
2008-03-20 17:57:11 +02:00
|
|
|
for (i = 0; i < sc->chunk_count; i++) {
|
|
|
|
current_offset = sc->chunk_offsets[i];
|
2016-07-20 18:48:09 +02:00
|
|
|
while (mov_stsc_index_valid(stsc_index, sc->stsc_count) &&
|
2009-02-08 11:59:31 +02:00
|
|
|
i + 1 == sc->stsc_data[stsc_index + 1].first)
|
2008-03-20 17:57:11 +02:00
|
|
|
stsc_index++;
|
2009-02-08 11:59:31 +02:00
|
|
|
for (j = 0; j < sc->stsc_data[stsc_index].count; j++) {
|
2009-05-13 11:31:22 +03:00
|
|
|
int keyframe = 0;
|
2008-03-20 17:57:11 +02:00
|
|
|
if (current_sample >= sc->sample_count) {
|
|
|
|
av_log(mov->fc, AV_LOG_ERROR, "wrong sample count\n");
|
2009-03-15 09:21:22 +02:00
|
|
|
return;
|
2008-03-20 17:57:11 +02:00
|
|
|
}
|
2009-05-13 11:31:22 +03:00
|
|
|
|
2012-04-13 12:59:14 +03:00
|
|
|
if (!sc->keyframe_absent && (!sc->keyframe_count || current_sample+key_off == sc->keyframes[stss_index])) {
|
2009-05-13 11:31:22 +03:00
|
|
|
keyframe = 1;
|
2008-03-20 17:57:11 +02:00
|
|
|
if (stss_index + 1 < sc->keyframe_count)
|
|
|
|
stss_index++;
|
2009-05-13 11:31:22 +03:00
|
|
|
} else if (sc->stps_count && current_sample+key_off == sc->stps_data[stps_index]) {
|
|
|
|
keyframe = 1;
|
|
|
|
if (stps_index + 1 < sc->stps_count)
|
|
|
|
stps_index++;
|
2008-03-20 17:57:11 +02:00
|
|
|
}
|
2012-04-24 02:51:29 +03:00
|
|
|
if (rap_group_present && rap_group_index < sc->rap_group_count) {
|
|
|
|
if (sc->rap_group[rap_group_index].index > 0)
|
|
|
|
keyframe = 1;
|
|
|
|
if (++rap_group_sample == sc->rap_group[rap_group_index].count) {
|
|
|
|
rap_group_sample = 0;
|
|
|
|
rap_group_index++;
|
|
|
|
}
|
|
|
|
}
|
2009-05-13 11:31:22 +03:00
|
|
|
if (keyframe)
|
|
|
|
distance = 0;
|
2008-03-20 17:57:11 +02:00
|
|
|
sample_size = sc->sample_size > 0 ? sc->sample_size : sc->sample_sizes[current_sample];
|
2011-10-11 11:12:38 +03:00
|
|
|
if (sc->pseudo_stream_id == -1 ||
|
2009-02-08 11:59:31 +02:00
|
|
|
sc->stsc_data[stsc_index].id - 1 == sc->pseudo_stream_id) {
|
2010-03-09 02:46:26 +02:00
|
|
|
AVIndexEntry *e = &st->index_entries[st->nb_index_entries++];
|
|
|
|
e->pos = current_offset;
|
2015-02-28 23:51:16 +02:00
|
|
|
e->timestamp = current_dts;
|
2010-03-09 02:46:26 +02:00
|
|
|
e->size = sample_size;
|
|
|
|
e->min_distance = distance;
|
|
|
|
e->flags = keyframe ? AVINDEX_KEYFRAME : 0;
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(mov->fc, AV_LOG_TRACE, "AVIndex stream %d, sample %u, offset %"PRIx64", dts %"PRId64", "
|
|
|
|
"size %u, distance %u, keyframe %d\n", st->index, current_sample,
|
|
|
|
current_offset, current_dts, sample_size, distance, keyframe);
|
2008-05-20 05:24:02 +03:00
|
|
|
}
|
2009-03-15 11:06:36 +02:00
|
|
|
|
2008-03-20 17:57:11 +02:00
|
|
|
current_offset += sample_size;
|
2009-10-24 23:40:00 +03:00
|
|
|
stream_size += sample_size;
|
2009-06-11 08:13:23 +03:00
|
|
|
current_dts += sc->stts_data[stts_index].duration;
|
2008-03-20 17:57:11 +02:00
|
|
|
distance++;
|
|
|
|
stts_sample++;
|
|
|
|
current_sample++;
|
|
|
|
if (stts_index + 1 < sc->stts_count && stts_sample == sc->stts_data[stts_index].count) {
|
|
|
|
stts_sample = 0;
|
|
|
|
stts_index++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-11-05 09:17:10 +02:00
|
|
|
if (st->duration > 0)
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->bit_rate = stream_size*8*sc->time_scale/st->duration;
|
2009-05-31 07:36:26 +03:00
|
|
|
} else {
|
2010-03-09 02:46:26 +02:00
|
|
|
unsigned chunk_samples, total = 0;
|
2009-05-31 07:36:26 +03:00
|
|
|
|
2010-03-09 02:46:26 +02:00
|
|
|
// compute total chunk count
|
|
|
|
for (i = 0; i < sc->stsc_count; i++) {
|
|
|
|
unsigned count, chunk_count;
|
2009-05-31 07:36:26 +03:00
|
|
|
|
2010-03-09 02:46:26 +02:00
|
|
|
chunk_samples = sc->stsc_data[i].count;
|
2011-12-03 14:29:41 +03:00
|
|
|
if (i != sc->stsc_count - 1 &&
|
|
|
|
sc->samples_per_frame && chunk_samples % sc->samples_per_frame) {
|
2009-05-31 07:36:26 +03:00
|
|
|
av_log(mov->fc, AV_LOG_ERROR, "error unaligned chunk\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-03-09 02:46:26 +02:00
|
|
|
if (sc->samples_per_frame >= 160) { // gsm
|
|
|
|
count = chunk_samples / sc->samples_per_frame;
|
|
|
|
} else if (sc->samples_per_frame > 1) {
|
|
|
|
unsigned samples = (1024/sc->samples_per_frame)*sc->samples_per_frame;
|
|
|
|
count = (chunk_samples+samples-1) / samples;
|
|
|
|
} else {
|
|
|
|
count = (chunk_samples+1023) / 1024;
|
|
|
|
}
|
|
|
|
|
2016-07-20 18:48:09 +02:00
|
|
|
if (mov_stsc_index_valid(i, sc->stsc_count))
|
2010-03-09 02:46:26 +02:00
|
|
|
chunk_count = sc->stsc_data[i+1].first - sc->stsc_data[i].first;
|
|
|
|
else
|
|
|
|
chunk_count = sc->chunk_count - (sc->stsc_data[i].first - 1);
|
|
|
|
total += chunk_count * count;
|
|
|
|
}
|
|
|
|
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(mov->fc, AV_LOG_TRACE, "chunk count %u\n", total);
|
2012-03-28 22:56:07 +03:00
|
|
|
if (total >= UINT_MAX / sizeof(*st->index_entries) - st->nb_index_entries)
|
2010-03-09 02:46:26 +02:00
|
|
|
return;
|
2013-09-10 12:57:35 +03:00
|
|
|
if (av_reallocp_array(&st->index_entries,
|
|
|
|
st->nb_index_entries + total,
|
|
|
|
sizeof(*st->index_entries)) < 0) {
|
|
|
|
st->nb_index_entries = 0;
|
2010-03-09 02:46:26 +02:00
|
|
|
return;
|
2013-09-10 12:57:35 +03:00
|
|
|
}
|
2012-03-28 22:56:07 +03:00
|
|
|
st->index_entries_allocated_size = (st->nb_index_entries + total) * sizeof(*st->index_entries);
|
2010-03-09 02:46:26 +02:00
|
|
|
|
|
|
|
// populate index
|
|
|
|
for (i = 0; i < sc->chunk_count; i++) {
|
|
|
|
current_offset = sc->chunk_offsets[i];
|
2016-07-20 18:48:09 +02:00
|
|
|
if (mov_stsc_index_valid(stsc_index, sc->stsc_count) &&
|
2010-03-09 02:46:26 +02:00
|
|
|
i + 1 == sc->stsc_data[stsc_index + 1].first)
|
|
|
|
stsc_index++;
|
|
|
|
chunk_samples = sc->stsc_data[stsc_index].count;
|
|
|
|
|
2009-05-31 07:36:26 +03:00
|
|
|
while (chunk_samples > 0) {
|
2010-03-09 02:46:26 +02:00
|
|
|
AVIndexEntry *e;
|
2009-05-31 07:36:26 +03:00
|
|
|
unsigned size, samples;
|
|
|
|
|
2015-05-26 15:24:37 +02:00
|
|
|
if (sc->samples_per_frame > 1 && !sc->bytes_per_frame) {
|
|
|
|
avpriv_request_sample(mov->fc,
|
|
|
|
"Zero bytes per frame, but %d samples per frame",
|
|
|
|
sc->samples_per_frame);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-05-31 07:36:26 +03:00
|
|
|
if (sc->samples_per_frame >= 160) { // gsm
|
|
|
|
samples = sc->samples_per_frame;
|
|
|
|
size = sc->bytes_per_frame;
|
|
|
|
} else {
|
|
|
|
if (sc->samples_per_frame > 1) {
|
|
|
|
samples = FFMIN((1024 / sc->samples_per_frame)*
|
|
|
|
sc->samples_per_frame, chunk_samples);
|
|
|
|
size = (samples / sc->samples_per_frame) * sc->bytes_per_frame;
|
2008-03-20 17:57:11 +02:00
|
|
|
} else {
|
2009-05-31 07:36:26 +03:00
|
|
|
samples = FFMIN(1024, chunk_samples);
|
|
|
|
size = samples * sc->sample_size;
|
2008-03-20 17:57:11 +02:00
|
|
|
}
|
|
|
|
}
|
2009-05-31 07:36:26 +03:00
|
|
|
|
2010-03-09 02:46:26 +02:00
|
|
|
if (st->nb_index_entries >= total) {
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(mov->fc, AV_LOG_ERROR, "wrong chunk count %u\n", total);
|
2010-03-09 02:46:26 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
e = &st->index_entries[st->nb_index_entries++];
|
|
|
|
e->pos = current_offset;
|
|
|
|
e->timestamp = current_dts;
|
|
|
|
e->size = size;
|
|
|
|
e->min_distance = 0;
|
|
|
|
e->flags = AVINDEX_KEYFRAME;
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(mov->fc, AV_LOG_TRACE, "AVIndex stream %d, chunk %u, offset %"PRIx64", dts %"PRId64", "
|
|
|
|
"size %u, duration %u\n", st->index, i, current_offset, current_dts,
|
|
|
|
size, samples);
|
2009-05-31 07:36:26 +03:00
|
|
|
|
|
|
|
current_offset += size;
|
2009-06-11 08:13:23 +03:00
|
|
|
current_dts += samples;
|
2009-05-31 07:36:26 +03:00
|
|
|
chunk_samples -= samples;
|
2008-03-20 17:57:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-03-20 17:55:50 +02:00
|
|
|
|
2016-01-16 18:53:43 +02:00
|
|
|
static int mov_open_dref(AVFormatContext *s, AVIOContext **pb, char *src,
|
|
|
|
MOVDref *ref)
|
2009-11-15 05:26:47 +02:00
|
|
|
{
|
2010-02-21 00:48:09 +02:00
|
|
|
/* try relative path, we do not try the absolute because it can leak information about our
|
|
|
|
system to an attacker */
|
2015-07-10 19:46:19 +02:00
|
|
|
if (ref->nlvl_to > 0 && ref->nlvl_from > 0) {
|
2009-11-15 05:26:47 +02:00
|
|
|
char filename[1024];
|
|
|
|
char *src_path;
|
|
|
|
int i, l;
|
|
|
|
|
|
|
|
/* find a source dir */
|
|
|
|
src_path = strrchr(src, '/');
|
|
|
|
if (src_path)
|
|
|
|
src_path++;
|
|
|
|
else
|
|
|
|
src_path = src;
|
|
|
|
|
|
|
|
/* find a next level down to target */
|
|
|
|
for (i = 0, l = strlen(ref->path) - 1; l >= 0; l--)
|
|
|
|
if (ref->path[l] == '/') {
|
|
|
|
if (i == ref->nlvl_to - 1)
|
|
|
|
break;
|
|
|
|
else
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* compose filename if next level down to target was found */
|
2010-02-21 00:57:23 +02:00
|
|
|
if (i == ref->nlvl_to - 1 && src_path - src < sizeof(filename)) {
|
2009-11-15 05:26:47 +02:00
|
|
|
memcpy(filename, src, src_path - src);
|
|
|
|
filename[src_path - src] = 0;
|
|
|
|
|
|
|
|
for (i = 1; i < ref->nlvl_from; i++)
|
|
|
|
av_strlcat(filename, "../", 1024);
|
|
|
|
|
|
|
|
av_strlcat(filename, ref->path + l + 1, 1024);
|
|
|
|
|
2016-01-16 18:53:43 +02:00
|
|
|
if (!s->io_open(s, pb, filename, AVIO_FLAG_READ, NULL))
|
2009-11-15 05:26:47 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return AVERROR(ENOENT);
|
2011-02-10 13:42:21 +02:00
|
|
|
}
|
2009-11-15 05:26:47 +02:00
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_trak(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2003-05-12 13:59:18 +03:00
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
2008-03-20 17:55:50 +02:00
|
|
|
int ret;
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-06-18 12:43:24 +03:00
|
|
|
st = avformat_new_stream(c->fc, NULL);
|
2008-03-21 12:48:22 +02:00
|
|
|
if (!st) return AVERROR(ENOMEM);
|
2011-06-18 12:43:24 +03:00
|
|
|
st->id = c->fc->nb_streams;
|
2006-09-29 13:30:52 +03:00
|
|
|
sc = av_mallocz(sizeof(MOVStreamContext));
|
2008-03-21 12:47:14 +02:00
|
|
|
if (!sc) return AVERROR(ENOMEM);
|
2003-05-12 13:59:18 +03:00
|
|
|
|
|
|
|
st->priv_data = sc;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->codec_type = AVMEDIA_TYPE_DATA;
|
2009-01-11 23:04:25 +02:00
|
|
|
sc->ffindex = st->index;
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2008-03-20 17:55:50 +02:00
|
|
|
if ((ret = mov_read_default(c, pb, atom)) < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* sanity checks */
|
2009-03-15 11:06:36 +02:00
|
|
|
if (sc->chunk_count && (!sc->stts_count || !sc->stsc_count ||
|
|
|
|
(!sc->sample_size && !sc->sample_count))) {
|
2008-03-27 15:19:19 +02:00
|
|
|
av_log(c->fc, AV_LOG_ERROR, "stream %d, missing mandatory atoms, broken header\n",
|
|
|
|
st->index);
|
2008-03-20 17:55:50 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2009-03-15 11:06:36 +02:00
|
|
|
|
2011-01-11 23:06:37 +02:00
|
|
|
if (sc->time_scale <= 0) {
|
2009-11-05 09:11:54 +02:00
|
|
|
av_log(c->fc, AV_LOG_WARNING, "stream %d, timescale not set\n", st->index);
|
2009-03-15 11:06:36 +02:00
|
|
|
sc->time_scale = c->time_scale;
|
2011-01-11 23:06:37 +02:00
|
|
|
if (sc->time_scale <= 0)
|
2009-11-05 09:11:54 +02:00
|
|
|
sc->time_scale = 1;
|
|
|
|
}
|
2009-03-15 11:06:36 +02:00
|
|
|
|
2011-11-29 21:28:15 +03:00
|
|
|
avpriv_set_pts_info(st, 64, 1, sc->time_scale);
|
2008-03-20 17:55:50 +02:00
|
|
|
|
|
|
|
mov_build_index(c, st);
|
|
|
|
|
|
|
|
if (sc->dref_id-1 < sc->drefs_count && sc->drefs[sc->dref_id-1].path) {
|
2009-11-15 05:26:47 +02:00
|
|
|
MOVDref *dref = &sc->drefs[sc->dref_id - 1];
|
2016-01-20 19:27:09 +02:00
|
|
|
if (c->enable_drefs) {
|
2016-01-16 18:53:43 +02:00
|
|
|
if (mov_open_dref(c->fc, &sc->pb, c->fc->filename, dref) < 0)
|
2016-01-20 19:27:09 +02:00
|
|
|
av_log(c->fc, AV_LOG_ERROR,
|
|
|
|
"stream %d, error opening alias: path='%s', dir='%s', "
|
2015-12-11 18:29:11 +02:00
|
|
|
"filename='%s', volume='%s', nlvl_from=%"PRId16", nlvl_to=%"PRId16"\n",
|
2016-01-20 19:27:09 +02:00
|
|
|
st->index, dref->path, dref->dir, dref->filename,
|
|
|
|
dref->volume, dref->nlvl_from, dref->nlvl_to);
|
|
|
|
} else {
|
|
|
|
av_log(c->fc, AV_LOG_WARNING,
|
|
|
|
"Skipped opening external track: "
|
|
|
|
"stream %d, alias: path='%s', dir='%s', "
|
2015-12-11 18:29:11 +02:00
|
|
|
"filename='%s', volume='%s', nlvl_from=%"PRId16", nlvl_to=%"PRId16"."
|
2016-01-20 19:27:09 +02:00
|
|
|
"Set enable_drefs to allow this.\n",
|
2009-11-15 05:26:47 +02:00
|
|
|
st->index, dref->path, dref->dir, dref->filename,
|
|
|
|
dref->volume, dref->nlvl_from, dref->nlvl_to);
|
2016-01-20 19:27:09 +02:00
|
|
|
}
|
2008-03-20 17:55:50 +02:00
|
|
|
} else
|
|
|
|
sc->pb = c->fc->pb;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) {
|
2010-07-02 02:18:27 +03:00
|
|
|
if (!st->sample_aspect_ratio.num &&
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
(st->codecpar->width != sc->width || st->codecpar->height != sc->height)) {
|
|
|
|
st->sample_aspect_ratio = av_d2q(((double)st->codecpar->height * sc->width) /
|
|
|
|
((double)st->codecpar->width * sc->height), INT_MAX);
|
2010-01-19 01:56:04 +02:00
|
|
|
}
|
2010-01-18 23:38:39 +02:00
|
|
|
}
|
|
|
|
|
2012-09-26 13:13:53 +03:00
|
|
|
// done for ai5q, ai52, ai55, ai1q, ai12 and ai15.
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (!st->codecpar->extradata_size && st->codecpar->codec_id == AV_CODEC_ID_H264 &&
|
|
|
|
TAG_IS_AVCI(st->codecpar->codec_tag)) {
|
2012-09-26 13:13:53 +03:00
|
|
|
ret = ff_generate_avci_extradata(st);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
switch (st->codecpar->codec_id) {
|
2009-01-14 01:44:16 +02:00
|
|
|
#if CONFIG_H261_DECODER
|
2012-08-05 12:11:04 +03:00
|
|
|
case AV_CODEC_ID_H261:
|
2008-03-20 17:55:50 +02:00
|
|
|
#endif
|
2009-01-14 01:44:16 +02:00
|
|
|
#if CONFIG_H263_DECODER
|
2012-08-05 12:11:04 +03:00
|
|
|
case AV_CODEC_ID_H263:
|
2008-03-20 17:55:50 +02:00
|
|
|
#endif
|
2009-01-14 01:44:16 +02:00
|
|
|
#if CONFIG_MPEG4_DECODER
|
2012-08-05 12:11:04 +03:00
|
|
|
case AV_CODEC_ID_MPEG4:
|
2008-03-20 17:55:50 +02:00
|
|
|
#endif
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->width = 0; /* let decoder init width/height */
|
|
|
|
st->codecpar->height= 0;
|
2008-03-20 17:55:50 +02:00
|
|
|
break;
|
2016-02-06 14:15:36 +02:00
|
|
|
case AV_CODEC_ID_MP3:
|
|
|
|
st->need_parsing = AVSTREAM_PARSE_FULL;
|
|
|
|
break;
|
2008-03-20 17:55:50 +02:00
|
|
|
}
|
2008-03-20 18:00:45 +02:00
|
|
|
|
|
|
|
/* Do not need those anymore. */
|
|
|
|
av_freep(&sc->chunk_offsets);
|
|
|
|
av_freep(&sc->sample_sizes);
|
|
|
|
av_freep(&sc->keyframes);
|
|
|
|
av_freep(&sc->stts_data);
|
2009-05-13 11:31:22 +03:00
|
|
|
av_freep(&sc->stps_data);
|
2012-04-24 02:51:29 +03:00
|
|
|
av_freep(&sc->rap_group);
|
2008-03-20 18:00:45 +02:00
|
|
|
|
2008-03-20 17:55:50 +02:00
|
|
|
return 0;
|
2003-05-12 13:59:18 +03:00
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_ilst(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2008-11-16 03:59:35 +02:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
c->itunes_metadata = 1;
|
|
|
|
ret = mov_read_default(c, pb, atom);
|
|
|
|
c->itunes_metadata = 0;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-02-10 00:07:18 +02:00
|
|
|
static int mov_read_replaygain(MOVContext *c, AVIOContext *pb, int64_t size)
|
2014-04-02 00:22:21 +03:00
|
|
|
{
|
|
|
|
int64_t end = avio_tell(pb) + size;
|
|
|
|
uint8_t *key = NULL, *val = NULL;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
uint8_t **p;
|
|
|
|
uint32_t len, tag;
|
2015-05-26 15:24:39 +02:00
|
|
|
int ret;
|
2014-04-02 00:22:21 +03:00
|
|
|
|
|
|
|
if (end - avio_tell(pb) <= 12)
|
|
|
|
break;
|
|
|
|
|
|
|
|
len = avio_rb32(pb);
|
|
|
|
tag = avio_rl32(pb);
|
|
|
|
avio_skip(pb, 4); // flags
|
|
|
|
|
|
|
|
if (len < 12 || len - 12 > end - avio_tell(pb))
|
|
|
|
break;
|
|
|
|
len -= 12;
|
|
|
|
|
|
|
|
if (tag == MKTAG('n', 'a', 'm', 'e'))
|
|
|
|
p = &key;
|
|
|
|
else if (tag == MKTAG('d', 'a', 't', 'a') && len > 4) {
|
|
|
|
avio_skip(pb, 4);
|
|
|
|
len -= 4;
|
|
|
|
p = &val;
|
|
|
|
} else
|
|
|
|
break;
|
|
|
|
|
|
|
|
*p = av_malloc(len + 1);
|
|
|
|
if (!*p)
|
|
|
|
break;
|
2015-05-26 15:24:39 +02:00
|
|
|
ret = ffio_read_size(pb, *p, len);
|
|
|
|
if (ret < 0) {
|
|
|
|
av_freep(p);
|
|
|
|
return ret;
|
|
|
|
}
|
2014-04-02 00:22:21 +03:00
|
|
|
(*p)[len] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (key && val) {
|
|
|
|
av_dict_set(&c->fc->metadata, key, val,
|
|
|
|
AV_DICT_DONT_STRDUP_KEY | AV_DICT_DONT_STRDUP_VAL);
|
|
|
|
key = val = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
avio_seek(pb, end, SEEK_SET);
|
|
|
|
av_freep(&key);
|
|
|
|
av_freep(&val);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mov_read_custom(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
int64_t end = avio_tell(pb) + atom.size;
|
|
|
|
uint32_t tag, len;
|
|
|
|
|
|
|
|
if (atom.size < 8)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
len = avio_rb32(pb);
|
|
|
|
tag = avio_rl32(pb);
|
|
|
|
|
|
|
|
if (len > atom.size)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
if (tag == MKTAG('m', 'e', 'a', 'n') && len > 12) {
|
|
|
|
uint8_t domain[128];
|
|
|
|
int domain_len;
|
|
|
|
|
|
|
|
avio_skip(pb, 4); // flags
|
|
|
|
len -= 12;
|
|
|
|
|
|
|
|
domain_len = avio_get_str(pb, len, domain, sizeof(domain));
|
|
|
|
avio_skip(pb, len - domain_len);
|
|
|
|
if (!strcmp(domain, "org.hydrogenaudio.replaygain"))
|
|
|
|
return mov_read_replaygain(c, pb, end - avio_tell(pb));
|
|
|
|
}
|
|
|
|
|
|
|
|
fail:
|
|
|
|
av_log(c->fc, AV_LOG_VERBOSE,
|
|
|
|
"Unhandled or malformed custom metadata of size %"PRId64"\n", atom.size);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_meta(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2008-11-16 03:59:35 +02:00
|
|
|
{
|
2009-03-16 01:54:53 +02:00
|
|
|
while (atom.size > 8) {
|
2011-02-21 17:43:01 +02:00
|
|
|
uint32_t tag = avio_rl32(pb);
|
2009-03-16 01:54:53 +02:00
|
|
|
atom.size -= 4;
|
|
|
|
if (tag == MKTAG('h','d','l','r')) {
|
2011-02-28 15:57:54 +02:00
|
|
|
avio_seek(pb, -8, SEEK_CUR);
|
2009-03-16 01:54:53 +02:00
|
|
|
atom.size += 8;
|
|
|
|
return mov_read_default(c, pb, atom);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
2008-11-16 03:59:35 +02:00
|
|
|
}
|
|
|
|
|
2016-10-24 23:46:47 +02:00
|
|
|
// return 1 when matrix is identity, 0 otherwise
|
|
|
|
#define IS_MATRIX_IDENT(matrix) \
|
|
|
|
( (matrix)[0][0] == (1 << 16) && \
|
|
|
|
(matrix)[1][1] == (1 << 16) && \
|
|
|
|
(matrix)[2][2] == (1 << 30) && \
|
2016-11-15 17:14:30 +02:00
|
|
|
!(matrix)[0][1] && !(matrix)[0][2] && \
|
|
|
|
!(matrix)[1][0] && !(matrix)[1][2] && \
|
2016-10-24 23:46:47 +02:00
|
|
|
!(matrix)[2][0] && !(matrix)[2][1])
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_tkhd(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2003-05-12 13:59:18 +03:00
|
|
|
{
|
2016-10-24 23:46:47 +02:00
|
|
|
int i, j, e;
|
2008-07-24 10:55:11 +03:00
|
|
|
int width;
|
|
|
|
int height;
|
|
|
|
int64_t disp_transform[2];
|
2014-05-16 05:06:50 +03:00
|
|
|
int display_matrix[3][3];
|
2016-10-24 23:46:47 +02:00
|
|
|
int res_display_matrix[3][3] = { { 0 } };
|
2009-06-24 06:38:47 +03:00
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
int version;
|
2013-08-22 17:25:17 +03:00
|
|
|
int flags;
|
2009-06-24 06:38:47 +03:00
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams-1];
|
|
|
|
sc = st->priv_data;
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
version = avio_r8(pb);
|
2013-08-22 17:25:17 +03:00
|
|
|
flags = avio_rb24(pb);
|
|
|
|
st->disposition |= (flags & MOV_TKHD_FLAG_ENABLED) ? AV_DISPOSITION_DEFAULT : 0;
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2006-03-22 12:45:04 +02:00
|
|
|
if (version == 1) {
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb64(pb);
|
|
|
|
avio_rb64(pb);
|
2006-03-22 12:45:04 +02:00
|
|
|
} else {
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); /* creation time */
|
|
|
|
avio_rb32(pb); /* modification time */
|
2006-03-22 12:45:04 +02:00
|
|
|
}
|
2011-02-21 17:43:01 +02:00
|
|
|
st->id = (int)avio_rb32(pb); /* track id (NOT 0 !)*/
|
|
|
|
avio_rb32(pb); /* reserved */
|
2009-02-11 22:19:35 +02:00
|
|
|
|
2008-03-21 13:16:16 +02:00
|
|
|
/* highlevel (considering edits) duration in movie timebase */
|
2011-02-21 17:43:01 +02:00
|
|
|
(version == 1) ? avio_rb64(pb) : avio_rb32(pb);
|
|
|
|
avio_rb32(pb); /* reserved */
|
|
|
|
avio_rb32(pb); /* reserved */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb16(pb); /* layer */
|
|
|
|
avio_rb16(pb); /* alternate group */
|
|
|
|
avio_rb16(pb); /* volume */
|
|
|
|
avio_rb16(pb); /* reserved */
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2008-07-24 10:55:11 +03:00
|
|
|
//read in the display matrix (outlined in ISO 14496-12, Section 6.2.2)
|
|
|
|
// they're kept in fixed point format through all calculations
|
2014-05-16 05:06:50 +03:00
|
|
|
// save u,v,z to store the whole matrix in the AV_PKT_DATA_DISPLAYMATRIX
|
|
|
|
// side data, but the scale factor is not needed to calculate aspect ratio
|
2008-07-24 10:55:11 +03:00
|
|
|
for (i = 0; i < 3; i++) {
|
2011-02-21 17:43:01 +02:00
|
|
|
display_matrix[i][0] = avio_rb32(pb); // 16.16 fixed point
|
|
|
|
display_matrix[i][1] = avio_rb32(pb); // 16.16 fixed point
|
2014-05-16 05:06:50 +03:00
|
|
|
display_matrix[i][2] = avio_rb32(pb); // 2.30 fixed point
|
2008-07-24 10:55:11 +03:00
|
|
|
}
|
2003-05-12 13:59:18 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
width = avio_rb32(pb); // 16.16 fixed point track width
|
|
|
|
height = avio_rb32(pb); // 16.16 fixed point track height
|
2009-01-12 01:24:28 +02:00
|
|
|
sc->width = width >> 16;
|
|
|
|
sc->height = height >> 16;
|
2008-07-24 10:55:11 +03:00
|
|
|
|
2016-10-24 23:46:47 +02:00
|
|
|
// apply the moov display matrix (after the tkhd one)
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
const int sh[3] = { 16, 16, 30 };
|
|
|
|
for (j = 0; j < 3; j++) {
|
|
|
|
for (e = 0; e < 3; e++) {
|
|
|
|
res_display_matrix[i][j] +=
|
|
|
|
((int64_t) display_matrix[i][e] *
|
|
|
|
c->movie_display_matrix[e][j]) >> sh[e];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-05-16 05:06:50 +03:00
|
|
|
|
2016-10-24 23:46:47 +02:00
|
|
|
// save the matrix when it is not the default identity
|
|
|
|
if (!IS_MATRIX_IDENT(res_display_matrix)) {
|
2014-05-16 05:06:50 +03:00
|
|
|
av_freep(&sc->display_matrix);
|
|
|
|
sc->display_matrix = av_malloc(sizeof(int32_t) * 9);
|
|
|
|
if (!sc->display_matrix)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
for (j = 0; j < 3; j++)
|
2016-10-24 23:46:47 +02:00
|
|
|
sc->display_matrix[i * 3 + j] = res_display_matrix[i][j];
|
2014-05-16 05:06:50 +03:00
|
|
|
}
|
|
|
|
|
2009-06-25 03:17:47 +03:00
|
|
|
// transform the display width/height according to the matrix
|
2015-01-22 21:10:27 +02:00
|
|
|
// skip this when the display matrix is the identity one
|
2008-07-24 10:55:11 +03:00
|
|
|
// to keep the same scale, use [width height 1<<16]
|
2015-01-22 21:10:27 +02:00
|
|
|
if (width && height && sc->display_matrix) {
|
2008-07-24 10:55:11 +03:00
|
|
|
for (i = 0; i < 2; i++)
|
|
|
|
disp_transform[i] =
|
2016-10-24 23:46:47 +02:00
|
|
|
(int64_t) width * sc->display_matrix[0 + i] +
|
|
|
|
(int64_t) height * sc->display_matrix[3 + i] +
|
|
|
|
((int64_t) sc->display_matrix[6 + i] << 16);
|
2008-07-24 10:55:11 +03:00
|
|
|
|
|
|
|
//sample aspect ratio is new width/height divided by old width/height
|
2015-01-22 21:10:27 +02:00
|
|
|
if (disp_transform[0] > 0 && disp_transform[1] > 0)
|
|
|
|
st->sample_aspect_ratio = av_d2q(
|
|
|
|
((double) disp_transform[0] * height) /
|
|
|
|
((double) disp_transform[1] * width), INT_MAX);
|
2008-07-24 10:55:11 +03:00
|
|
|
}
|
2003-05-12 13:59:18 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_tfhd(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2008-03-21 15:40:23 +02:00
|
|
|
{
|
|
|
|
MOVFragment *frag = &c->fragment;
|
|
|
|
MOVTrackExt *trex = NULL;
|
|
|
|
int flags, track_id, i;
|
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
flags = avio_rb24(pb);
|
2008-03-21 15:40:23 +02:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
track_id = avio_rb32(pb);
|
2009-03-04 09:57:09 +02:00
|
|
|
if (!track_id)
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2008-03-21 15:40:23 +02:00
|
|
|
frag->track_id = track_id;
|
|
|
|
for (i = 0; i < c->trex_count; i++)
|
|
|
|
if (c->trex_data[i].track_id == frag->track_id) {
|
|
|
|
trex = &c->trex_data[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!trex) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "could not find corresponding trex\n");
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2008-03-21 15:40:23 +02:00
|
|
|
}
|
|
|
|
|
2012-02-16 22:48:30 +03:00
|
|
|
frag->base_data_offset = flags & MOV_TFHD_BASE_DATA_OFFSET ?
|
2014-07-01 08:17:56 +03:00
|
|
|
avio_rb64(pb) : flags & MOV_TFHD_DEFAULT_BASE_IS_MOOF ?
|
|
|
|
frag->moof_offset : frag->implicit_offset;
|
2012-02-16 22:48:30 +03:00
|
|
|
frag->stsd_id = flags & MOV_TFHD_STSD_ID ? avio_rb32(pb) : trex->stsd_id;
|
|
|
|
|
|
|
|
frag->duration = flags & MOV_TFHD_DEFAULT_DURATION ?
|
|
|
|
avio_rb32(pb) : trex->duration;
|
|
|
|
frag->size = flags & MOV_TFHD_DEFAULT_SIZE ?
|
|
|
|
avio_rb32(pb) : trex->size;
|
|
|
|
frag->flags = flags & MOV_TFHD_DEFAULT_FLAGS ?
|
|
|
|
avio_rb32(pb) : trex->flags;
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "frag flags 0x%x\n", frag->flags);
|
2008-03-21 15:40:23 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_chap(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2010-04-21 09:36:09 +03:00
|
|
|
{
|
2011-02-21 17:43:01 +02:00
|
|
|
c->chapter_track = avio_rb32(pb);
|
2010-04-21 09:36:09 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_trex(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2008-03-21 15:40:23 +02:00
|
|
|
{
|
|
|
|
MOVTrackExt *trex;
|
2013-09-10 12:57:35 +03:00
|
|
|
int err;
|
2008-03-21 15:40:23 +02:00
|
|
|
|
|
|
|
if ((uint64_t)c->trex_count+1 >= UINT_MAX / sizeof(*c->trex_data))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2013-09-10 12:57:35 +03:00
|
|
|
if ((err = av_reallocp_array(&c->trex_data, c->trex_count + 1,
|
|
|
|
sizeof(*c->trex_data))) < 0) {
|
|
|
|
c->trex_count = 0;
|
|
|
|
return err;
|
|
|
|
}
|
2008-03-21 15:40:23 +02:00
|
|
|
trex = &c->trex_data[c->trex_count++];
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
avio_rb24(pb); /* flags */
|
|
|
|
trex->track_id = avio_rb32(pb);
|
|
|
|
trex->stsd_id = avio_rb32(pb);
|
|
|
|
trex->duration = avio_rb32(pb);
|
|
|
|
trex->size = avio_rb32(pb);
|
|
|
|
trex->flags = avio_rb32(pb);
|
2008-03-21 15:40:23 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-10-12 01:24:26 +03:00
|
|
|
static int mov_read_tfdt(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
MOVFragment *frag = &c->fragment;
|
|
|
|
AVStream *st = NULL;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
int version, i;
|
|
|
|
|
|
|
|
for (i = 0; i < c->fc->nb_streams; i++) {
|
|
|
|
if (c->fc->streams[i]->id == frag->track_id) {
|
|
|
|
st = c->fc->streams[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!st) {
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_ERROR, "could not find corresponding track id %u\n", frag->track_id);
|
2014-10-12 01:24:26 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
}
|
|
|
|
sc = st->priv_data;
|
|
|
|
if (sc->pseudo_stream_id + 1 != frag->stsd_id)
|
|
|
|
return 0;
|
|
|
|
version = avio_r8(pb);
|
|
|
|
avio_rb24(pb); /* flags */
|
|
|
|
if (version) {
|
|
|
|
sc->track_end = avio_rb64(pb);
|
|
|
|
} else {
|
|
|
|
sc->track_end = avio_rb32(pb);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_trun(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2008-03-21 15:40:23 +02:00
|
|
|
{
|
|
|
|
MOVFragment *frag = &c->fragment;
|
2009-03-04 09:57:09 +02:00
|
|
|
AVStream *st = NULL;
|
2008-03-30 23:49:29 +03:00
|
|
|
MOVStreamContext *sc;
|
2011-04-29 17:39:29 +03:00
|
|
|
MOVStts *ctts_data;
|
2008-03-21 15:40:23 +02:00
|
|
|
uint64_t offset;
|
|
|
|
int64_t dts;
|
|
|
|
int data_offset = 0;
|
|
|
|
unsigned entries, first_sample_flags = frag->flags;
|
2015-09-02 12:05:45 +02:00
|
|
|
int flags, distance, i, err;
|
2008-03-21 15:40:23 +02:00
|
|
|
|
2009-03-04 09:57:09 +02:00
|
|
|
for (i = 0; i < c->fc->nb_streams; i++) {
|
|
|
|
if (c->fc->streams[i]->id == frag->track_id) {
|
|
|
|
st = c->fc->streams[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!st) {
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_ERROR, "could not find corresponding track id %u\n", frag->track_id);
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-03-04 09:57:09 +02:00
|
|
|
}
|
2008-03-30 23:49:29 +03:00
|
|
|
sc = st->priv_data;
|
2008-03-21 15:40:23 +02:00
|
|
|
if (sc->pseudo_stream_id+1 != frag->stsd_id)
|
|
|
|
return 0;
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_r8(pb); /* version */
|
|
|
|
flags = avio_rb24(pb);
|
|
|
|
entries = avio_rb32(pb);
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "flags 0x%x entries %u\n", flags, entries);
|
2011-04-29 17:39:29 +03:00
|
|
|
|
|
|
|
/* Always assume the presence of composition time offsets.
|
|
|
|
* Without this assumption, for instance, we cannot deal with a track in fragmented movies that meet the following.
|
|
|
|
* 1) in the initial movie, there are no samples.
|
|
|
|
* 2) in the first movie fragment, there is only one sample without composition time offset.
|
|
|
|
* 3) in the subsequent movie fragments, there are samples with composition time offset. */
|
|
|
|
if (!sc->ctts_count && sc->sample_count)
|
|
|
|
{
|
|
|
|
/* Complement ctts table if moov atom doesn't have ctts atom. */
|
2013-10-04 09:52:02 +03:00
|
|
|
ctts_data = av_realloc(NULL, sizeof(*sc->ctts_data));
|
2009-02-22 02:20:07 +02:00
|
|
|
if (!ctts_data)
|
2008-03-21 15:40:23 +02:00
|
|
|
return AVERROR(ENOMEM);
|
2009-02-22 02:20:07 +02:00
|
|
|
sc->ctts_data = ctts_data;
|
2011-04-29 17:39:29 +03:00
|
|
|
sc->ctts_data[sc->ctts_count].count = sc->sample_count;
|
|
|
|
sc->ctts_data[sc->ctts_count].duration = 0;
|
|
|
|
sc->ctts_count++;
|
2008-03-21 15:40:23 +02:00
|
|
|
}
|
2011-04-29 17:39:29 +03:00
|
|
|
if ((uint64_t)entries+sc->ctts_count >= UINT_MAX/sizeof(*sc->ctts_data))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2013-09-10 12:57:35 +03:00
|
|
|
if ((err = av_reallocp_array(&sc->ctts_data, entries + sc->ctts_count,
|
|
|
|
sizeof(*sc->ctts_data))) < 0) {
|
|
|
|
sc->ctts_count = 0;
|
|
|
|
return err;
|
|
|
|
}
|
2012-02-16 22:58:09 +03:00
|
|
|
if (flags & MOV_TRUN_DATA_OFFSET) data_offset = avio_rb32(pb);
|
|
|
|
if (flags & MOV_TRUN_FIRST_SAMPLE_FLAGS) first_sample_flags = avio_rb32(pb);
|
2012-02-02 18:30:27 +03:00
|
|
|
dts = sc->track_end - sc->time_offset;
|
2008-03-21 15:40:23 +02:00
|
|
|
offset = frag->base_data_offset + data_offset;
|
|
|
|
distance = 0;
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "first sample flags 0x%x\n", first_sample_flags);
|
2012-09-16 21:32:12 +03:00
|
|
|
for (i = 0; i < entries && !pb->eof_reached; i++) {
|
2008-03-21 15:40:23 +02:00
|
|
|
unsigned sample_size = frag->size;
|
|
|
|
int sample_flags = i ? frag->flags : first_sample_flags;
|
|
|
|
unsigned sample_duration = frag->duration;
|
2012-01-31 13:45:02 +03:00
|
|
|
int keyframe = 0;
|
2008-03-21 15:40:23 +02:00
|
|
|
|
2012-02-16 22:58:09 +03:00
|
|
|
if (flags & MOV_TRUN_SAMPLE_DURATION) sample_duration = avio_rb32(pb);
|
|
|
|
if (flags & MOV_TRUN_SAMPLE_SIZE) sample_size = avio_rb32(pb);
|
|
|
|
if (flags & MOV_TRUN_SAMPLE_FLAGS) sample_flags = avio_rb32(pb);
|
2015-02-28 23:51:16 +02:00
|
|
|
sc->ctts_data[sc->ctts_count].count = 1;
|
|
|
|
sc->ctts_data[sc->ctts_count].duration = (flags & MOV_TRUN_SAMPLE_CTS) ?
|
|
|
|
avio_rb32(pb) : 0;
|
2011-04-29 17:39:29 +03:00
|
|
|
sc->ctts_count++;
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO)
|
2012-01-31 13:45:02 +03:00
|
|
|
keyframe = 1;
|
2015-09-02 12:05:45 +02:00
|
|
|
else
|
|
|
|
keyframe =
|
2012-01-31 13:45:02 +03:00
|
|
|
!(sample_flags & (MOV_FRAG_SAMPLE_FLAG_IS_NON_SYNC |
|
|
|
|
MOV_FRAG_SAMPLE_FLAG_DEPENDS_YES));
|
|
|
|
if (keyframe)
|
2008-03-21 15:40:23 +02:00
|
|
|
distance = 0;
|
2015-02-28 23:51:16 +02:00
|
|
|
av_add_index_entry(st, offset, dts, sample_size, distance,
|
2008-03-21 15:40:23 +02:00
|
|
|
keyframe ? AVINDEX_KEYFRAME : 0);
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "AVIndex stream %d, sample %u, offset %"PRIx64", dts %"PRId64", "
|
|
|
|
"size %u, distance %d, keyframe %d\n", st->index, sc->sample_count+i,
|
|
|
|
offset, dts, sample_size, distance, keyframe);
|
2008-03-21 15:40:23 +02:00
|
|
|
distance++;
|
2009-06-11 08:13:23 +03:00
|
|
|
dts += sample_duration;
|
2008-03-21 15:40:23 +02:00
|
|
|
offset += sample_size;
|
2012-01-13 15:40:24 +03:00
|
|
|
sc->data_size += sample_size;
|
2008-03-21 15:40:23 +02:00
|
|
|
}
|
2012-09-16 21:32:12 +03:00
|
|
|
|
|
|
|
if (pb->eof_reached)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
|
2014-07-01 08:17:56 +03:00
|
|
|
frag->implicit_offset = offset;
|
2012-02-02 18:30:27 +03:00
|
|
|
st->duration = sc->track_end = dts + sc->time_offset;
|
2008-03-21 15:40:23 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2003-05-12 13:59:18 +03:00
|
|
|
/* this atom should be null (from specs), but some buggy files put the 'moov' atom inside it... */
|
|
|
|
/* like the files created with Adobe Premiere 5.0, for samples see */
|
|
|
|
/* http://graphics.tudelft.nl/~wouter/publications/soundtests/ */
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_wide(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2003-05-12 13:59:18 +03:00
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if (atom.size < 8)
|
|
|
|
return 0; /* continue */
|
2011-02-21 17:43:01 +02:00
|
|
|
if (avio_rb32(pb) != 0) { /* 0 sized mdat atom... use the 'wide' atom size */
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, atom.size - 4);
|
2003-05-12 13:59:18 +03:00
|
|
|
return 0;
|
|
|
|
}
|
2011-02-21 17:43:01 +02:00
|
|
|
atom.type = avio_rl32(pb);
|
2003-05-12 13:59:18 +03:00
|
|
|
atom.size -= 8;
|
2008-04-25 14:57:39 +03:00
|
|
|
if (atom.type != MKTAG('m','d','a','t')) {
|
2011-03-15 10:14:38 +02:00
|
|
|
avio_skip(pb, atom.size);
|
2003-05-12 13:59:18 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
err = mov_read_mdat(c, pb, atom);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_cmov(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2002-07-20 23:05:50 +03:00
|
|
|
{
|
2009-01-14 01:44:16 +02:00
|
|
|
#if CONFIG_ZLIB
|
2011-02-20 12:04:12 +02:00
|
|
|
AVIOContext ctx;
|
2003-04-23 11:58:42 +03:00
|
|
|
uint8_t *cmov_data;
|
|
|
|
uint8_t *moov_data; /* uncompressed data */
|
2002-07-20 23:05:50 +03:00
|
|
|
long cmov_len, moov_len;
|
2008-05-22 20:16:12 +03:00
|
|
|
int ret = -1;
|
2003-04-23 11:58:42 +03:00
|
|
|
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); /* dcom atom */
|
|
|
|
if (avio_rl32(pb) != MKTAG('d','c','o','m'))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2011-02-21 17:43:01 +02:00
|
|
|
if (avio_rl32(pb) != MKTAG('z','l','i','b')) {
|
2009-02-27 10:31:20 +02:00
|
|
|
av_log(c->fc, AV_LOG_ERROR, "unknown compression for cmov atom !");
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2002-07-20 23:05:50 +03:00
|
|
|
}
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); /* cmvd atom */
|
|
|
|
if (avio_rl32(pb) != MKTAG('c','m','v','d'))
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2011-02-21 17:43:01 +02:00
|
|
|
moov_len = avio_rb32(pb); /* uncompressed size */
|
2003-05-12 13:59:18 +03:00
|
|
|
cmov_len = atom.size - 6 * 4;
|
2003-04-22 11:36:39 +03:00
|
|
|
|
2006-09-29 13:30:52 +03:00
|
|
|
cmov_data = av_malloc(cmov_len);
|
2002-07-20 23:05:50 +03:00
|
|
|
if (!cmov_data)
|
2009-02-22 02:20:07 +02:00
|
|
|
return AVERROR(ENOMEM);
|
2006-09-29 13:30:52 +03:00
|
|
|
moov_data = av_malloc(moov_len);
|
2002-07-20 23:05:50 +03:00
|
|
|
if (!moov_data) {
|
|
|
|
av_free(cmov_data);
|
2009-02-22 02:20:07 +02:00
|
|
|
return AVERROR(ENOMEM);
|
2002-07-20 23:05:50 +03:00
|
|
|
}
|
2015-05-26 15:24:39 +02:00
|
|
|
ret = ffio_read_size(pb, cmov_data, cmov_len);
|
|
|
|
if (ret < 0)
|
|
|
|
goto free_and_return;
|
|
|
|
|
2011-10-11 11:12:38 +03:00
|
|
|
if (uncompress (moov_data, (uLongf *) &moov_len, (const Bytef *)cmov_data, cmov_len) != Z_OK)
|
2008-05-22 20:16:12 +03:00
|
|
|
goto free_and_return;
|
2011-10-11 11:12:38 +03:00
|
|
|
if (ffio_init_context(&ctx, moov_data, moov_len, 0, NULL, NULL, NULL, NULL) != 0)
|
2008-05-22 20:16:12 +03:00
|
|
|
goto free_and_return;
|
2008-04-25 14:57:39 +03:00
|
|
|
atom.type = MKTAG('m','o','o','v');
|
2003-05-12 13:59:18 +03:00
|
|
|
atom.size = moov_len;
|
|
|
|
ret = mov_read_default(c, &ctx, atom);
|
2008-05-22 20:16:12 +03:00
|
|
|
free_and_return:
|
2002-07-20 23:05:50 +03:00
|
|
|
av_free(moov_data);
|
|
|
|
av_free(cmov_data);
|
|
|
|
return ret;
|
2006-10-20 16:06:50 +03:00
|
|
|
#else
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "this file requires zlib support compiled in\n");
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR(ENOSYS);
|
2002-07-20 23:05:50 +03:00
|
|
|
#endif
|
2006-10-20 16:06:50 +03:00
|
|
|
}
|
2002-07-20 23:05:50 +03:00
|
|
|
|
2004-07-06 05:35:39 +03:00
|
|
|
/* edit list atom */
|
2011-02-20 12:04:12 +02:00
|
|
|
static int mov_read_elst(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
2004-07-06 05:35:39 +03:00
|
|
|
{
|
2009-09-08 01:36:33 +03:00
|
|
|
MOVStreamContext *sc;
|
2011-05-11 23:28:27 +03:00
|
|
|
int i, edit_count, version;
|
2006-04-19 15:53:47 +03:00
|
|
|
|
2009-09-08 01:36:33 +03:00
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
sc = c->fc->streams[c->fc->nb_streams-1]->priv_data;
|
|
|
|
|
2011-05-11 23:28:27 +03:00
|
|
|
version = avio_r8(pb); /* version */
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb24(pb); /* flags */
|
|
|
|
edit_count = avio_rb32(pb); /* entries */
|
2006-04-19 15:53:47 +03:00
|
|
|
|
2011-10-11 11:12:38 +03:00
|
|
|
if ((uint64_t)edit_count*12+8 > atom.size)
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-06-15 05:07:41 +03:00
|
|
|
|
2011-10-11 11:12:38 +03:00
|
|
|
for (i=0; i<edit_count; i++){
|
2011-05-11 23:28:27 +03:00
|
|
|
int64_t time;
|
|
|
|
int64_t duration;
|
|
|
|
if (version == 1) {
|
|
|
|
duration = avio_rb64(pb);
|
|
|
|
time = avio_rb64(pb);
|
|
|
|
} else {
|
|
|
|
duration = avio_rb32(pb); /* segment duration */
|
2011-06-04 19:28:43 +03:00
|
|
|
time = (int32_t)avio_rb32(pb); /* media time */
|
2011-05-11 23:28:27 +03:00
|
|
|
}
|
2011-02-21 17:43:01 +02:00
|
|
|
avio_rb32(pb); /* Media rate */
|
2009-03-16 22:49:52 +02:00
|
|
|
if (i == 0 && time >= -1) {
|
|
|
|
sc->time_offset = time != -1 ? time : -duration;
|
2009-02-11 04:32:39 +02:00
|
|
|
}
|
2006-04-19 15:53:47 +03:00
|
|
|
}
|
2009-02-08 06:27:07 +02:00
|
|
|
|
2011-10-11 11:12:38 +03:00
|
|
|
if (edit_count > 1)
|
2009-02-08 06:27:07 +02:00
|
|
|
av_log(c->fc, AV_LOG_WARNING, "multiple edit list entries, "
|
|
|
|
"a/v desync might occur, patch welcome\n");
|
|
|
|
|
2015-12-15 11:50:18 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "track[%u].edit_count = %i\n", c->fc->nb_streams - 1, edit_count);
|
2006-04-19 15:53:47 +03:00
|
|
|
return 0;
|
2004-07-06 05:35:39 +03:00
|
|
|
}
|
|
|
|
|
2016-11-02 17:29:15 +02:00
|
|
|
static int mov_read_st3d(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
enum AVStereo3DType type;
|
|
|
|
int mode;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
st = c->fc->streams[c->fc->nb_streams - 1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
|
|
|
if (atom.size < 5) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Empty stereoscopic video box\n");
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
}
|
|
|
|
avio_skip(pb, 4); /* version + flags */
|
|
|
|
|
|
|
|
mode = avio_r8(pb);
|
|
|
|
switch (mode) {
|
|
|
|
case 0:
|
|
|
|
type = AV_STEREO3D_2D;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
type = AV_STEREO3D_TOPBOTTOM;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
type = AV_STEREO3D_SIDEBYSIDE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
av_log(c->fc, AV_LOG_WARNING, "Unknown st3d mode value %d\n", mode);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
sc->stereo3d = av_stereo3d_alloc();
|
|
|
|
if (!sc->stereo3d)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
|
|
|
sc->stereo3d->type = type;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mov_read_sv3d(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
int size;
|
|
|
|
int32_t yaw, pitch, roll;
|
|
|
|
uint32_t tag;
|
|
|
|
enum AVSphericalProjection projection;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
st = c->fc->streams[c->fc->nb_streams - 1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
|
|
|
if (atom.size < 8) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Empty spherical video box\n");
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
}
|
|
|
|
|
|
|
|
size = avio_rb32(pb);
|
|
|
|
if (size > atom.size)
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
|
|
|
|
tag = avio_rl32(pb);
|
|
|
|
if (tag != MKTAG('s','v','h','d')) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Missing spherical video header\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
avio_skip(pb, 4); /* version + flags */
|
|
|
|
avio_skip(pb, avio_r8(pb)); /* metadata_source */
|
|
|
|
|
|
|
|
size = avio_rb32(pb);
|
|
|
|
if (size > atom.size)
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
|
|
|
|
tag = avio_rl32(pb);
|
|
|
|
if (tag != MKTAG('p','r','o','j')) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Missing projection box\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
size = avio_rb32(pb);
|
|
|
|
if (size > atom.size)
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
|
|
|
|
tag = avio_rl32(pb);
|
|
|
|
if (tag != MKTAG('p','r','h','d')) {
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Missing projection header box\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
avio_skip(pb, 4); /* version + flags */
|
|
|
|
|
|
|
|
/* 16.16 fixed point */
|
|
|
|
yaw = avio_rb32(pb);
|
|
|
|
pitch = avio_rb32(pb);
|
|
|
|
roll = avio_rb32(pb);
|
|
|
|
|
|
|
|
size = avio_rb32(pb);
|
|
|
|
if (size > atom.size)
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
|
|
|
|
tag = avio_rl32(pb);
|
|
|
|
avio_skip(pb, 4); /* version + flags */
|
|
|
|
switch (tag) {
|
|
|
|
case MKTAG('c','b','m','p'):
|
|
|
|
projection = AV_SPHERICAL_CUBEMAP;
|
|
|
|
break;
|
|
|
|
case MKTAG('e','q','u','i'):
|
|
|
|
projection = AV_SPHERICAL_EQUIRECTANGULAR;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
av_log(c->fc, AV_LOG_ERROR, "Unknown projection type\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
sc->spherical = av_spherical_alloc(&sc->spherical_size);
|
|
|
|
if (!sc->spherical)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
|
|
|
|
sc->spherical->projection = projection;
|
|
|
|
|
|
|
|
sc->spherical->yaw = yaw;
|
|
|
|
sc->spherical->pitch = pitch;
|
|
|
|
sc->spherical->roll = roll;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mov_parse_uuid_spherical(MOVStreamContext *sc, AVIOContext *pb, size_t len)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
uint8_t *buffer = av_malloc(len + 1);
|
|
|
|
const char *val;
|
|
|
|
|
|
|
|
if (!buffer)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
buffer[len] = '\0';
|
|
|
|
|
|
|
|
ret = ffio_read_size(pb, buffer, len);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/* Check for mandatory keys and values, try to support XML as best-effort */
|
|
|
|
if (av_stristr(buffer, "<GSpherical:StitchingSoftware>") &&
|
|
|
|
(val = av_stristr(buffer, "<GSpherical:Spherical>")) &&
|
|
|
|
av_stristr(val, "true") &&
|
|
|
|
(val = av_stristr(buffer, "<GSpherical:Stitched>")) &&
|
|
|
|
av_stristr(val, "true") &&
|
|
|
|
(val = av_stristr(buffer, "<GSpherical:ProjectionType>")) &&
|
|
|
|
av_stristr(val, "equirectangular")) {
|
|
|
|
sc->spherical = av_spherical_alloc(&sc->spherical_size);
|
|
|
|
if (!sc->spherical)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
sc->spherical->projection = AV_SPHERICAL_EQUIRECTANGULAR;
|
|
|
|
|
|
|
|
if (av_stristr(buffer, "<GSpherical:StereoMode>")) {
|
|
|
|
enum AVStereo3DType mode;
|
|
|
|
|
|
|
|
if (av_stristr(buffer, "left-right"))
|
|
|
|
mode = AV_STEREO3D_SIDEBYSIDE;
|
|
|
|
else if (av_stristr(buffer, "top-bottom"))
|
|
|
|
mode = AV_STEREO3D_TOPBOTTOM;
|
|
|
|
else
|
|
|
|
mode = AV_STEREO3D_2D;
|
|
|
|
|
|
|
|
sc->stereo3d = av_stereo3d_alloc();
|
|
|
|
if (!sc->stereo3d)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
sc->stereo3d->type = mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* orientation */
|
|
|
|
val = av_stristr(buffer, "<GSpherical:InitialViewHeadingDegrees>");
|
|
|
|
if (val)
|
|
|
|
sc->spherical->yaw = strtol(val, NULL, 10) * (1 << 16);
|
|
|
|
val = av_stristr(buffer, "<GSpherical:InitialViewPitchDegrees>");
|
|
|
|
if (val)
|
|
|
|
sc->spherical->pitch = strtol(val, NULL, 10) * (1 << 16);
|
|
|
|
val = av_stristr(buffer, "<GSpherical:InitialViewRollDegrees>");
|
|
|
|
if (val)
|
|
|
|
sc->spherical->roll = strtol(val, NULL, 10) * (1 << 16);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
av_free(buffer);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mov_read_uuid(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
AVStream *st;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
uint8_t uuid[16];
|
|
|
|
static const uint8_t uuid_spherical[] = {
|
|
|
|
0xff, 0xcc, 0x82, 0x63, 0xf8, 0x55, 0x4a, 0x93,
|
|
|
|
0x88, 0x14, 0x58, 0x7a, 0x02, 0x52, 0x1f, 0xdd,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (atom.size < sizeof(uuid) || atom.size == INT64_MAX)
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
|
|
|
|
if (c->fc->nb_streams < 1)
|
|
|
|
return 0;
|
|
|
|
st = c->fc->streams[c->fc->nb_streams - 1];
|
|
|
|
sc = st->priv_data;
|
|
|
|
|
|
|
|
ret = ffio_read_size(pb, uuid, sizeof(uuid));
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (!memcmp(uuid, uuid_spherical, sizeof(uuid)) && !sc->spherical) {
|
|
|
|
size_t len = atom.size - sizeof(uuid);
|
|
|
|
ret = mov_parse_uuid_spherical(sc, pb, len);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
if (!sc->spherical)
|
|
|
|
av_log(c->fc, AV_LOG_WARNING, "Invalid spherical metadata found\n");
|
|
|
|
} else {
|
|
|
|
int i;
|
|
|
|
av_log(c->fc, AV_LOG_VERBOSE, "Unknown UUID found: 0x");
|
|
|
|
for (i = 0; i < sizeof(uuid); i++)
|
|
|
|
av_log(c->fc, AV_LOG_WARNING, "%02x", uuid[i]);
|
|
|
|
av_log(c->fc, AV_LOG_WARNING, "\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-04-08 15:32:01 +03:00
|
|
|
static const MOVParseTableEntry mov_default_parse_table[] = {
|
2008-06-23 02:52:57 +03:00
|
|
|
{ MKTAG('a','v','s','s'), mov_read_extradata },
|
2010-05-05 11:41:01 +03:00
|
|
|
{ MKTAG('c','h','p','l'), mov_read_chpl },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('c','o','6','4'), mov_read_stco },
|
2014-10-02 16:28:35 +03:00
|
|
|
{ MKTAG('c','o','l','r'), mov_read_colr },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('c','t','t','s'), mov_read_ctts }, /* composition time to sample */
|
|
|
|
{ MKTAG('d','i','n','f'), mov_read_default },
|
|
|
|
{ MKTAG('d','r','e','f'), mov_read_dref },
|
|
|
|
{ MKTAG('e','d','t','s'), mov_read_default },
|
|
|
|
{ MKTAG('e','l','s','t'), mov_read_elst },
|
|
|
|
{ MKTAG('e','n','d','a'), mov_read_enda },
|
2011-12-14 05:49:06 +03:00
|
|
|
{ MKTAG('f','i','e','l'), mov_read_fiel },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('f','t','y','p'), mov_read_ftyp },
|
|
|
|
{ MKTAG('g','l','b','l'), mov_read_glbl },
|
|
|
|
{ MKTAG('h','d','l','r'), mov_read_hdlr },
|
2008-11-16 03:59:35 +02:00
|
|
|
{ MKTAG('i','l','s','t'), mov_read_ilst },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('j','p','2','h'), mov_read_extradata },
|
|
|
|
{ MKTAG('m','d','a','t'), mov_read_mdat },
|
|
|
|
{ MKTAG('m','d','h','d'), mov_read_mdhd },
|
|
|
|
{ MKTAG('m','d','i','a'), mov_read_default },
|
2008-11-16 03:59:35 +02:00
|
|
|
{ MKTAG('m','e','t','a'), mov_read_meta },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('m','i','n','f'), mov_read_default },
|
|
|
|
{ MKTAG('m','o','o','f'), mov_read_moof },
|
|
|
|
{ MKTAG('m','o','o','v'), mov_read_moov },
|
|
|
|
{ MKTAG('m','v','e','x'), mov_read_default },
|
|
|
|
{ MKTAG('m','v','h','d'), mov_read_mvhd },
|
|
|
|
{ MKTAG('S','M','I',' '), mov_read_smi }, /* Sorenson extension ??? */
|
|
|
|
{ MKTAG('a','l','a','c'), mov_read_extradata }, /* alac specific atom */
|
|
|
|
{ MKTAG('a','v','c','C'), mov_read_glbl },
|
2008-12-12 00:24:56 +02:00
|
|
|
{ MKTAG('p','a','s','p'), mov_read_pasp },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('s','t','b','l'), mov_read_default },
|
|
|
|
{ MKTAG('s','t','c','o'), mov_read_stco },
|
2009-05-13 11:31:22 +03:00
|
|
|
{ MKTAG('s','t','p','s'), mov_read_stps },
|
2010-04-16 23:36:24 +03:00
|
|
|
{ MKTAG('s','t','r','f'), mov_read_strf },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('s','t','s','c'), mov_read_stsc },
|
|
|
|
{ MKTAG('s','t','s','d'), mov_read_stsd }, /* sample description */
|
|
|
|
{ MKTAG('s','t','s','s'), mov_read_stss }, /* sync sample */
|
|
|
|
{ MKTAG('s','t','s','z'), mov_read_stsz }, /* sample size */
|
|
|
|
{ MKTAG('s','t','t','s'), mov_read_stts },
|
2009-03-16 18:14:36 +02:00
|
|
|
{ MKTAG('s','t','z','2'), mov_read_stsz }, /* compact sample size */
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('t','k','h','d'), mov_read_tkhd }, /* track header */
|
2014-10-12 01:24:26 +03:00
|
|
|
{ MKTAG('t','f','d','t'), mov_read_tfdt },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('t','f','h','d'), mov_read_tfhd }, /* track fragment header */
|
|
|
|
{ MKTAG('t','r','a','k'), mov_read_trak },
|
|
|
|
{ MKTAG('t','r','a','f'), mov_read_default },
|
2010-04-21 09:36:09 +03:00
|
|
|
{ MKTAG('t','r','e','f'), mov_read_default },
|
|
|
|
{ MKTAG('c','h','a','p'), mov_read_chap },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('t','r','e','x'), mov_read_trex },
|
|
|
|
{ MKTAG('t','r','u','n'), mov_read_trun },
|
2008-11-15 22:17:19 +02:00
|
|
|
{ MKTAG('u','d','t','a'), mov_read_default },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('w','a','v','e'), mov_read_wave },
|
|
|
|
{ MKTAG('e','s','d','s'), mov_read_esds },
|
2010-09-07 00:52:11 +03:00
|
|
|
{ MKTAG('d','a','c','3'), mov_read_dac3 }, /* AC-3 info */
|
2011-05-31 02:17:13 +03:00
|
|
|
{ MKTAG('d','e','c','3'), mov_read_dec3 }, /* EAC-3 info */
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('w','i','d','e'), mov_read_wide }, /* place holder */
|
2011-06-06 20:59:46 +03:00
|
|
|
{ MKTAG('w','f','e','x'), mov_read_wfex },
|
2008-04-25 14:57:39 +03:00
|
|
|
{ MKTAG('c','m','o','v'), mov_read_cmov },
|
2011-12-03 21:32:45 +03:00
|
|
|
{ MKTAG('c','h','a','n'), mov_read_chan }, /* channel layout */
|
2012-01-22 02:32:16 +03:00
|
|
|
{ MKTAG('d','v','c','1'), mov_read_dvc1 },
|
2012-04-24 02:51:29 +03:00
|
|
|
{ MKTAG('s','b','g','p'), mov_read_sbgp },
|
2013-10-03 08:03:53 +03:00
|
|
|
{ MKTAG('h','v','c','C'), mov_read_glbl },
|
2016-11-02 17:29:15 +02:00
|
|
|
{ MKTAG('s','t','3','d'), mov_read_st3d }, /* stereoscopic 3D video box */
|
|
|
|
{ MKTAG('s','v','3','d'), mov_read_sv3d }, /* spherical video box */
|
|
|
|
{ MKTAG('u','u','i','d'), mov_read_uuid }, /* universal unique identifier */
|
2014-04-02 00:22:21 +03:00
|
|
|
{ MKTAG('-','-','-','-'), mov_read_custom },
|
2007-12-16 20:51:00 +02:00
|
|
|
{ 0, NULL }
|
2002-04-08 15:32:01 +03:00
|
|
|
};
|
|
|
|
|
2012-07-04 06:49:49 +03:00
|
|
|
static int mov_read_default(MOVContext *c, AVIOContext *pb, MOVAtom atom)
|
|
|
|
{
|
|
|
|
int64_t total_size = 0;
|
|
|
|
MOVAtom a;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (atom.size < 0)
|
|
|
|
atom.size = INT64_MAX;
|
|
|
|
while (total_size + 8 < atom.size && !pb->eof_reached) {
|
|
|
|
int (*parse)(MOVContext*, AVIOContext*, MOVAtom) = NULL;
|
|
|
|
a.size = atom.size;
|
|
|
|
a.type=0;
|
|
|
|
if (atom.size >= 8) {
|
|
|
|
a.size = avio_rb32(pb);
|
|
|
|
a.type = avio_rl32(pb);
|
|
|
|
}
|
2015-12-11 18:29:11 +02:00
|
|
|
av_log(c->fc, AV_LOG_TRACE, "type: %08"PRIx32" '%.4s' parent:'%.4s' sz: %"PRId64" %"PRId64" %"PRId64"\n",
|
|
|
|
a.type, (char*)&a.type, (char*)&atom.type, a.size, total_size, atom.size);
|
2012-07-04 06:49:49 +03:00
|
|
|
total_size += 8;
|
|
|
|
if (a.size == 1) { /* 64 bit extended size */
|
|
|
|
a.size = avio_rb64(pb) - 8;
|
|
|
|
total_size += 8;
|
|
|
|
}
|
|
|
|
if (a.size == 0) {
|
|
|
|
a.size = atom.size - total_size;
|
|
|
|
if (a.size <= 8)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
a.size -= 8;
|
|
|
|
if (a.size < 0)
|
|
|
|
break;
|
|
|
|
a.size = FFMIN(a.size, atom.size - total_size);
|
|
|
|
|
|
|
|
for (i = 0; mov_default_parse_table[i].type; i++)
|
|
|
|
if (mov_default_parse_table[i].type == a.type) {
|
|
|
|
parse = mov_default_parse_table[i].parse;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// container is user data
|
|
|
|
if (!parse && (atom.type == MKTAG('u','d','t','a') ||
|
|
|
|
atom.type == MKTAG('i','l','s','t')))
|
|
|
|
parse = mov_read_udta_string;
|
|
|
|
|
|
|
|
if (!parse) { /* skip leaf atoms data */
|
|
|
|
avio_skip(pb, a.size);
|
|
|
|
} else {
|
|
|
|
int64_t start_pos = avio_tell(pb);
|
|
|
|
int64_t left;
|
|
|
|
int err = parse(c, pb, a);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
if (c->found_moov && c->found_mdat &&
|
2016-09-27 16:26:37 +02:00
|
|
|
((!(pb->seekable & AVIO_SEEKABLE_NORMAL) || c->fc->flags & AVFMT_FLAG_IGNIDX) ||
|
2012-07-04 06:49:49 +03:00
|
|
|
start_pos + a.size == avio_size(pb))) {
|
2016-09-27 16:26:37 +02:00
|
|
|
if (!(pb->seekable & AVIO_SEEKABLE_NORMAL) || c->fc->flags & AVFMT_FLAG_IGNIDX)
|
2012-07-04 06:49:49 +03:00
|
|
|
c->next_root_atom = start_pos + a.size;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
left = a.size - avio_tell(pb) + start_pos;
|
|
|
|
if (left > 0) /* skip garbage at atom end */
|
|
|
|
avio_skip(pb, left);
|
2013-07-15 15:59:50 +03:00
|
|
|
else if (left < 0) {
|
|
|
|
av_log(c->fc, AV_LOG_WARNING,
|
|
|
|
"overread end of atom '%.4s' by %"PRId64" bytes\n",
|
|
|
|
(char*)&a.type, -left);
|
|
|
|
avio_seek(pb, left, SEEK_CUR);
|
|
|
|
}
|
2012-07-04 06:49:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
total_size += a.size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (total_size < atom.size && atom.size < 0x7ffff)
|
|
|
|
avio_skip(pb, atom.size - total_size);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-05-20 19:31:13 +03:00
|
|
|
static int mov_probe(AVProbeData *p)
|
|
|
|
{
|
2002-10-14 20:00:34 +03:00
|
|
|
unsigned int offset;
|
|
|
|
uint32_t tag;
|
2005-01-27 08:53:09 +02:00
|
|
|
int score = 0;
|
2002-07-24 01:05:35 +03:00
|
|
|
|
2002-05-20 19:31:13 +03:00
|
|
|
/* check file header */
|
2002-10-14 20:00:34 +03:00
|
|
|
offset = 0;
|
2011-10-11 11:12:38 +03:00
|
|
|
for (;;) {
|
2002-10-14 20:00:34 +03:00
|
|
|
/* ignore invalid offset */
|
|
|
|
if ((offset + 8) > (unsigned int)p->buf_size)
|
2005-01-27 08:53:09 +02:00
|
|
|
return score;
|
2007-01-20 00:12:59 +02:00
|
|
|
tag = AV_RL32(p->buf + offset + 4);
|
2002-10-14 20:00:34 +03:00
|
|
|
switch(tag) {
|
2005-01-27 08:53:09 +02:00
|
|
|
/* check for obvious tags */
|
2008-04-25 14:57:39 +03:00
|
|
|
case MKTAG('j','P',' ',' '): /* jpeg 2000 signature */
|
|
|
|
case MKTAG('m','o','o','v'):
|
|
|
|
case MKTAG('m','d','a','t'):
|
|
|
|
case MKTAG('p','n','o','t'): /* detect movs with preview pics like ew.mov and april.mov */
|
|
|
|
case MKTAG('u','d','t','a'): /* Packet Video PVAuthor adds this and a lot of more junk */
|
2008-07-18 03:23:37 +03:00
|
|
|
case MKTAG('f','t','y','p'):
|
2002-07-24 01:05:35 +03:00
|
|
|
return AVPROBE_SCORE_MAX;
|
2005-01-27 08:53:09 +02:00
|
|
|
/* those are more common words, so rate then a bit less */
|
2008-04-25 14:57:39 +03:00
|
|
|
case MKTAG('e','d','i','w'): /* xdcam files have reverted first tags */
|
|
|
|
case MKTAG('w','i','d','e'):
|
|
|
|
case MKTAG('f','r','e','e'):
|
|
|
|
case MKTAG('j','u','n','k'):
|
|
|
|
case MKTAG('p','i','c','t'):
|
2005-01-27 08:53:09 +02:00
|
|
|
return AVPROBE_SCORE_MAX - 5;
|
2008-07-18 03:24:31 +03:00
|
|
|
case MKTAG(0x82,0x82,0x7f,0x7d):
|
2008-04-25 14:57:39 +03:00
|
|
|
case MKTAG('s','k','i','p'):
|
|
|
|
case MKTAG('u','u','i','d'):
|
|
|
|
case MKTAG('p','r','f','l'):
|
2007-01-20 00:12:59 +02:00
|
|
|
offset = AV_RB32(p->buf+offset) + offset;
|
2005-01-27 08:53:09 +02:00
|
|
|
/* if we only find those cause probedata is too small at least rate them */
|
2013-03-25 18:12:51 +03:00
|
|
|
score = AVPROBE_SCORE_EXTENSION;
|
2002-10-14 20:00:34 +03:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* unrecognized tag */
|
2005-01-27 08:53:09 +02:00
|
|
|
return score;
|
2002-10-14 20:00:34 +03:00
|
|
|
}
|
2002-07-24 01:05:35 +03:00
|
|
|
}
|
2002-05-20 19:31:13 +03:00
|
|
|
}
|
|
|
|
|
2010-04-21 09:36:09 +03:00
|
|
|
// must be done after parsing all trak because there's no order requirement
|
|
|
|
static void mov_read_chapters(AVFormatContext *s)
|
|
|
|
{
|
|
|
|
MOVContext *mov = s->priv_data;
|
|
|
|
AVStream *st = NULL;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
int64_t cur_pos;
|
2011-01-23 13:35:23 +02:00
|
|
|
int i;
|
2010-04-21 09:36:09 +03:00
|
|
|
|
|
|
|
for (i = 0; i < s->nb_streams; i++)
|
|
|
|
if (s->streams[i]->id == mov->chapter_track) {
|
|
|
|
st = s->streams[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!st) {
|
|
|
|
av_log(s, AV_LOG_ERROR, "Referenced QT chapter track not found\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
st->discard = AVDISCARD_ALL;
|
|
|
|
sc = st->priv_data;
|
2011-03-03 21:11:45 +02:00
|
|
|
cur_pos = avio_tell(sc->pb);
|
2010-04-21 09:36:09 +03:00
|
|
|
|
|
|
|
for (i = 0; i < st->nb_index_entries; i++) {
|
|
|
|
AVIndexEntry *sample = &st->index_entries[i];
|
|
|
|
int64_t end = i+1 < st->nb_index_entries ? st->index_entries[i+1].timestamp : st->duration;
|
2011-01-23 13:35:23 +02:00
|
|
|
uint8_t *title;
|
|
|
|
uint16_t ch;
|
|
|
|
int len, title_len;
|
2010-04-21 09:36:09 +03:00
|
|
|
|
2011-02-28 15:57:54 +02:00
|
|
|
if (avio_seek(sc->pb, sample->pos, SEEK_SET) != sample->pos) {
|
2010-04-21 09:36:09 +03:00
|
|
|
av_log(s, AV_LOG_ERROR, "Chapter %d not found in file\n", i);
|
|
|
|
goto finish;
|
|
|
|
}
|
|
|
|
|
|
|
|
// the first two bytes are the length of the title
|
2011-02-21 17:43:01 +02:00
|
|
|
len = avio_rb16(sc->pb);
|
2010-04-21 09:36:09 +03:00
|
|
|
if (len > sample->size-2)
|
|
|
|
continue;
|
2011-01-23 13:35:23 +02:00
|
|
|
title_len = 2*len + 1;
|
|
|
|
if (!(title = av_mallocz(title_len)))
|
|
|
|
goto finish;
|
2010-04-21 09:36:09 +03:00
|
|
|
|
|
|
|
// The samples could theoretically be in any encoding if there's an encd
|
|
|
|
// atom following, but in practice are only utf-8 or utf-16, distinguished
|
|
|
|
// instead by the presence of a BOM
|
2011-10-14 00:47:06 +03:00
|
|
|
if (!len) {
|
|
|
|
title[0] = 0;
|
|
|
|
} else {
|
|
|
|
ch = avio_rb16(sc->pb);
|
|
|
|
if (ch == 0xfeff)
|
|
|
|
avio_get_str16be(sc->pb, len, title, title_len);
|
|
|
|
else if (ch == 0xfffe)
|
|
|
|
avio_get_str16le(sc->pb, len, title, title_len);
|
|
|
|
else {
|
|
|
|
AV_WB16(title, ch);
|
|
|
|
if (len == 1 || len == 2)
|
2011-10-14 20:38:42 +03:00
|
|
|
title[len] = 0;
|
2011-10-14 00:47:06 +03:00
|
|
|
else
|
|
|
|
avio_get_str(sc->pb, len - 2, title + 2, title_len - 2);
|
|
|
|
}
|
2010-04-21 09:36:09 +03:00
|
|
|
}
|
|
|
|
|
2011-10-17 09:58:50 +03:00
|
|
|
avpriv_new_chapter(s, i, st->time_base, sample->timestamp, end, title);
|
2011-01-26 23:54:42 +02:00
|
|
|
av_freep(&title);
|
2010-04-21 09:36:09 +03:00
|
|
|
}
|
|
|
|
finish:
|
2011-02-28 15:57:54 +02:00
|
|
|
avio_seek(sc->pb, cur_pos, SEEK_SET);
|
2010-04-21 09:36:09 +03:00
|
|
|
}
|
|
|
|
|
2012-04-15 03:48:42 +03:00
|
|
|
static int mov_read_close(AVFormatContext *s)
|
|
|
|
{
|
|
|
|
MOVContext *mov = s->priv_data;
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < s->nb_streams; i++) {
|
|
|
|
AVStream *st = s->streams[i];
|
|
|
|
MOVStreamContext *sc = st->priv_data;
|
|
|
|
|
|
|
|
av_freep(&sc->ctts_data);
|
|
|
|
for (j = 0; j < sc->drefs_count; j++) {
|
|
|
|
av_freep(&sc->drefs[j].path);
|
|
|
|
av_freep(&sc->drefs[j].dir);
|
|
|
|
}
|
|
|
|
av_freep(&sc->drefs);
|
|
|
|
if (sc->pb && sc->pb != s->pb)
|
2016-01-16 18:53:43 +02:00
|
|
|
ff_format_io_close(s, &sc->pb);
|
2014-01-13 15:43:23 +03:00
|
|
|
|
|
|
|
av_freep(&sc->chunk_offsets);
|
|
|
|
av_freep(&sc->stsc_data);
|
|
|
|
av_freep(&sc->sample_sizes);
|
|
|
|
av_freep(&sc->keyframes);
|
|
|
|
av_freep(&sc->stts_data);
|
|
|
|
av_freep(&sc->stps_data);
|
|
|
|
av_freep(&sc->rap_group);
|
2014-05-16 05:06:50 +03:00
|
|
|
av_freep(&sc->display_matrix);
|
2016-06-15 21:24:30 +02:00
|
|
|
|
|
|
|
for (j = 0; j < sc->stsd_count; j++)
|
|
|
|
av_free(sc->extradata[j]);
|
|
|
|
av_freep(&sc->extradata);
|
|
|
|
av_freep(&sc->extradata_size);
|
2016-11-02 17:29:15 +02:00
|
|
|
|
|
|
|
av_freep(&sc->stereo3d);
|
|
|
|
av_freep(&sc->spherical);
|
2012-04-15 03:48:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mov->dv_demux) {
|
2014-06-19 09:56:24 +03:00
|
|
|
avformat_free_context(mov->dv_fctx);
|
|
|
|
mov->dv_fctx = NULL;
|
2012-04-15 03:48:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
av_freep(&mov->trex_data);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-12 15:20:36 +03:00
|
|
|
static int mov_read_header(AVFormatContext *s)
|
2002-04-08 15:32:01 +03:00
|
|
|
{
|
2007-04-08 23:24:16 +03:00
|
|
|
MOVContext *mov = s->priv_data;
|
2011-02-20 12:04:12 +02:00
|
|
|
AVIOContext *pb = s->pb;
|
2008-03-20 18:00:45 +02:00
|
|
|
int err;
|
2010-06-25 04:19:03 +03:00
|
|
|
MOVAtom atom = { AV_RL32("root") };
|
2014-03-20 10:05:36 +03:00
|
|
|
int i;
|
2002-04-08 15:32:01 +03:00
|
|
|
|
|
|
|
mov->fc = s;
|
2008-03-21 13:16:16 +02:00
|
|
|
/* .mov and .mp4 aren't streamable anyway (only progressive download if moov is before mdat) */
|
2016-09-27 16:26:37 +02:00
|
|
|
if (pb->seekable & AVIO_SEEKABLE_NORMAL)
|
2011-03-04 20:57:36 +02:00
|
|
|
atom.size = avio_size(pb);
|
2002-04-08 15:32:01 +03:00
|
|
|
else
|
2007-04-21 21:01:47 +03:00
|
|
|
atom.size = INT64_MAX;
|
2002-04-08 15:32:01 +03:00
|
|
|
|
|
|
|
/* check MOV header */
|
2008-04-05 15:46:59 +03:00
|
|
|
if ((err = mov_read_default(mov, pb, atom)) < 0) {
|
|
|
|
av_log(s, AV_LOG_ERROR, "error reading header: %d\n", err);
|
2012-04-15 03:48:42 +03:00
|
|
|
mov_read_close(s);
|
2008-04-05 15:46:59 +03:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
if (!mov->found_moov) {
|
|
|
|
av_log(s, AV_LOG_ERROR, "moov atom not found\n");
|
2012-04-15 03:48:42 +03:00
|
|
|
mov_read_close(s);
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2002-04-08 15:32:01 +03:00
|
|
|
}
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(mov->fc, AV_LOG_TRACE, "on_parse_exit_offset=%"PRId64"\n", avio_tell(pb));
|
2006-02-12 12:24:43 +02:00
|
|
|
|
2016-09-27 16:26:37 +02:00
|
|
|
if ((pb->seekable & AVIO_SEEKABLE_NORMAL) && mov->chapter_track > 0)
|
2010-04-21 09:36:09 +03:00
|
|
|
mov_read_chapters(s);
|
|
|
|
|
2014-03-20 10:05:36 +03:00
|
|
|
for (i = 0; i < s->nb_streams; i++) {
|
|
|
|
AVStream *st = s->streams[i];
|
|
|
|
MOVStreamContext *sc = st->priv_data;
|
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_type == AVMEDIA_TYPE_SUBTITLE) {
|
|
|
|
if (st->codecpar->width <= 0 || st->codecpar->height <= 0) {
|
|
|
|
st->codecpar->width = sc->width;
|
|
|
|
st->codecpar->height = sc->height;
|
2014-03-20 10:05:36 +03:00
|
|
|
}
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
if (st->codecpar->codec_id == AV_CODEC_ID_DVD_SUBTITLE) {
|
2014-03-20 10:05:37 +03:00
|
|
|
if ((err = mov_rewrite_dvd_sub_extradata(st)) < 0)
|
|
|
|
return err;
|
|
|
|
}
|
2014-03-20 10:05:36 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-13 15:40:24 +03:00
|
|
|
if (mov->trex_data) {
|
|
|
|
for (i = 0; i < s->nb_streams; i++) {
|
|
|
|
AVStream *st = s->streams[i];
|
|
|
|
MOVStreamContext *sc = st->priv_data;
|
2013-09-16 20:17:47 +03:00
|
|
|
if (st->duration > 0)
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
st->codecpar->bit_rate = sc->data_size * 8 * sc->time_scale / st->duration;
|
2012-01-13 15:40:24 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-02 00:22:21 +03:00
|
|
|
for (i = 0; i < s->nb_streams; i++) {
|
|
|
|
AVStream *st = s->streams[i];
|
2014-05-16 05:06:50 +03:00
|
|
|
MOVStreamContext *sc = st->priv_data;
|
2014-04-02 00:22:21 +03:00
|
|
|
|
lavf: replace AVStream.codec with AVStream.codecpar
Currently, AVStream contains an embedded AVCodecContext instance, which
is used by demuxers to export stream parameters to the caller and by
muxers to receive stream parameters from the caller. It is also used
internally as the codec context that is passed to parsers.
In addition, it is also widely used by the callers as the decoding (when
demuxer) or encoding (when muxing) context, though this has been
officially discouraged since Libav 11.
There are multiple important problems with this approach:
- the fields in AVCodecContext are in general one of
* stream parameters
* codec options
* codec state
However, it's not clear which ones are which. It is consequently
unclear which fields are a demuxer allowed to set or a muxer allowed to
read. This leads to erratic behaviour depending on whether decoding or
encoding is being performed or not (and whether it uses the AVStream
embedded codec context).
- various synchronization issues arising from the fact that the same
context is used by several different APIs (muxers/demuxers,
parsers, bitstream filters and encoders/decoders) simultaneously, with
there being no clear rules for who can modify what and the different
processes being typically delayed with respect to each other.
- avformat_find_stream_info() making it necessary to support opening
and closing a single codec context multiple times, thus
complicating the semantics of freeing various allocated objects in the
codec context.
Those problems are resolved by replacing the AVStream embedded codec
context with a newly added AVCodecParameters instance, which stores only
the stream parameters exported by the demuxers or read by the muxers.
2014-06-18 21:42:52 +03:00
|
|
|
switch (st->codecpar->codec_type) {
|
2014-05-16 05:06:50 +03:00
|
|
|
case AVMEDIA_TYPE_AUDIO:
|
|
|
|
err = ff_replaygain_export(st, s->metadata);
|
|
|
|
if (err < 0) {
|
|
|
|
mov_read_close(s);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case AVMEDIA_TYPE_VIDEO:
|
|
|
|
if (sc->display_matrix) {
|
2016-11-22 01:54:29 +02:00
|
|
|
err = av_stream_add_side_data(st, AV_PKT_DATA_DISPLAYMATRIX,
|
|
|
|
(uint8_t *)sc->display_matrix,
|
|
|
|
sizeof(int32_t) * 9);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
2014-05-16 05:06:50 +03:00
|
|
|
|
|
|
|
sc->display_matrix = NULL;
|
|
|
|
}
|
2016-11-02 17:29:15 +02:00
|
|
|
if (sc->stereo3d) {
|
|
|
|
err = av_stream_add_side_data(st, AV_PKT_DATA_STEREO3D,
|
|
|
|
(uint8_t *)sc->stereo3d,
|
|
|
|
sizeof(*sc->stereo3d));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
sc->stereo3d = NULL;
|
|
|
|
}
|
|
|
|
if (sc->spherical) {
|
|
|
|
err = av_stream_add_side_data(st, AV_PKT_DATA_SPHERICAL,
|
|
|
|
(uint8_t *)sc->spherical,
|
|
|
|
sc->spherical_size);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
sc->spherical = NULL;
|
|
|
|
}
|
2014-05-16 05:06:50 +03:00
|
|
|
break;
|
2014-04-02 00:22:21 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-04-08 15:32:01 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-24 11:57:53 +03:00
|
|
|
static AVIndexEntry *mov_find_next_sample(AVFormatContext *s, AVStream **st)
|
2002-04-08 15:32:01 +03:00
|
|
|
{
|
2009-06-24 11:57:53 +03:00
|
|
|
AVIndexEntry *sample = NULL;
|
2005-05-21 12:50:25 +03:00
|
|
|
int64_t best_dts = INT64_MAX;
|
2009-06-24 11:57:53 +03:00
|
|
|
int i;
|
2007-12-16 20:30:45 +02:00
|
|
|
for (i = 0; i < s->nb_streams; i++) {
|
2009-06-24 11:23:40 +03:00
|
|
|
AVStream *avst = s->streams[i];
|
|
|
|
MOVStreamContext *msc = avst->priv_data;
|
2009-06-24 12:38:42 +03:00
|
|
|
if (msc->pb && msc->current_sample < avst->nb_index_entries) {
|
2009-06-24 11:23:40 +03:00
|
|
|
AVIndexEntry *current_sample = &avst->index_entries[msc->current_sample];
|
2015-02-28 23:51:16 +02:00
|
|
|
int64_t dts = av_rescale(current_sample->timestamp, AV_TIME_BASE, msc->time_scale);
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(s, AV_LOG_TRACE, "stream %d, sample %d, dts %"PRId64"\n", i, msc->current_sample, dts);
|
2016-09-27 16:26:37 +02:00
|
|
|
if (!sample || (!(s->pb->seekable & AVIO_SEEKABLE_NORMAL) && current_sample->pos < sample->pos) ||
|
|
|
|
((s->pb->seekable & AVIO_SEEKABLE_NORMAL) &&
|
2008-02-26 20:21:58 +02:00
|
|
|
((msc->pb != s->pb && dts < best_dts) || (msc->pb == s->pb &&
|
2007-12-16 20:08:06 +02:00
|
|
|
((FFABS(best_dts - dts) <= AV_TIME_BASE && current_sample->pos < sample->pos) ||
|
2008-02-26 20:21:58 +02:00
|
|
|
(FFABS(best_dts - dts) > AV_TIME_BASE && dts < best_dts)))))) {
|
2006-07-08 15:14:41 +03:00
|
|
|
sample = current_sample;
|
|
|
|
best_dts = dts;
|
2009-06-24 11:57:53 +03:00
|
|
|
*st = avst;
|
2005-05-21 12:50:25 +03:00
|
|
|
}
|
2002-04-08 15:32:01 +03:00
|
|
|
}
|
|
|
|
}
|
2009-06-24 11:57:53 +03:00
|
|
|
return sample;
|
|
|
|
}
|
|
|
|
|
2016-06-15 21:24:30 +02:00
|
|
|
static int mov_change_extradata(MOVStreamContext *sc, AVPacket *pkt)
|
|
|
|
{
|
|
|
|
uint8_t *side, *extradata;
|
|
|
|
int extradata_size;
|
|
|
|
|
|
|
|
/* Save the current index. */
|
|
|
|
sc->last_stsd_index = sc->stsc_data[sc->stsc_index].id - 1;
|
|
|
|
|
|
|
|
/* Notify the decoder that extradata changed. */
|
|
|
|
extradata_size = sc->extradata_size[sc->last_stsd_index];
|
|
|
|
extradata = sc->extradata[sc->last_stsd_index];
|
|
|
|
if (extradata_size > 0 && extradata) {
|
|
|
|
side = av_packet_new_side_data(pkt,
|
|
|
|
AV_PKT_DATA_NEW_EXTRADATA,
|
|
|
|
extradata_size);
|
|
|
|
if (!side)
|
|
|
|
return AVERROR(ENOMEM);
|
|
|
|
memcpy(side, extradata, extradata_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-24 11:57:53 +03:00
|
|
|
static int mov_read_packet(AVFormatContext *s, AVPacket *pkt)
|
|
|
|
{
|
|
|
|
MOVContext *mov = s->priv_data;
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
AVIndexEntry *sample;
|
|
|
|
AVStream *st = NULL;
|
|
|
|
int ret;
|
|
|
|
retry:
|
|
|
|
sample = mov_find_next_sample(s, &st);
|
2008-03-21 15:43:19 +02:00
|
|
|
if (!sample) {
|
|
|
|
mov->found_mdat = 0;
|
2012-02-21 13:03:56 +03:00
|
|
|
if (!mov->next_root_atom)
|
|
|
|
return AVERROR_EOF;
|
|
|
|
avio_seek(s->pb, mov->next_root_atom, SEEK_SET);
|
|
|
|
mov->next_root_atom = 0;
|
|
|
|
if (mov_read_default(mov, s->pb, (MOVAtom){ AV_RL32("root"), INT64_MAX }) < 0 ||
|
2011-03-07 22:50:25 +02:00
|
|
|
s->pb->eof_reached)
|
2009-05-30 09:15:27 +03:00
|
|
|
return AVERROR_EOF;
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(s, AV_LOG_TRACE, "read fragments, offset 0x%"PRIx64"\n", avio_tell(s->pb));
|
2008-03-21 15:43:19 +02:00
|
|
|
goto retry;
|
|
|
|
}
|
2009-06-24 11:23:40 +03:00
|
|
|
sc = st->priv_data;
|
2006-07-08 15:14:41 +03:00
|
|
|
/* must be done just before reading, to avoid infinite loop on sample */
|
|
|
|
sc->current_sample++;
|
2009-06-24 12:38:42 +03:00
|
|
|
|
|
|
|
if (st->discard != AVDISCARD_ALL) {
|
2011-02-28 15:57:54 +02:00
|
|
|
if (avio_seek(sc->pb, sample->pos, SEEK_SET) != sample->pos) {
|
2009-06-24 12:40:00 +03:00
|
|
|
av_log(mov->fc, AV_LOG_ERROR, "stream %d, offset 0x%"PRIx64": partial file\n",
|
|
|
|
sc->ffindex, sample->pos);
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-06-24 12:40:00 +03:00
|
|
|
}
|
|
|
|
ret = av_get_packet(sc->pb, pkt, sample->size);
|
2009-06-24 11:50:46 +03:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2011-04-09 16:49:51 +03:00
|
|
|
if (sc->has_palette) {
|
|
|
|
uint8_t *pal;
|
|
|
|
|
|
|
|
pal = av_packet_new_side_data(pkt, AV_PKT_DATA_PALETTE, AVPALETTE_SIZE);
|
|
|
|
if (!pal) {
|
|
|
|
av_log(mov->fc, AV_LOG_ERROR, "Cannot append palette to packet\n");
|
|
|
|
} else {
|
|
|
|
memcpy(pal, sc->palette, AVPALETTE_SIZE);
|
|
|
|
sc->has_palette = 0;
|
|
|
|
}
|
|
|
|
}
|
2009-06-24 12:40:00 +03:00
|
|
|
#if CONFIG_DV_DEMUXER
|
|
|
|
if (mov->dv_demux && sc->dv_audio_container) {
|
2011-10-17 09:58:50 +03:00
|
|
|
avpriv_dv_produce_packet(mov->dv_demux, pkt, pkt->data, pkt->size);
|
2009-06-24 12:40:00 +03:00
|
|
|
av_free(pkt->data);
|
|
|
|
pkt->size = 0;
|
2011-10-17 09:58:50 +03:00
|
|
|
ret = avpriv_dv_get_packet(mov->dv_demux, pkt);
|
2009-06-24 12:40:00 +03:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
2006-08-24 11:37:51 +03:00
|
|
|
#endif
|
2009-06-24 12:38:42 +03:00
|
|
|
}
|
|
|
|
|
2006-07-08 15:14:41 +03:00
|
|
|
pkt->stream_index = sc->ffindex;
|
2015-02-28 23:51:16 +02:00
|
|
|
pkt->dts = sample->timestamp;
|
2012-03-21 21:24:10 +03:00
|
|
|
if (sc->ctts_data && sc->ctts_index < sc->ctts_count) {
|
2015-02-28 23:51:16 +02:00
|
|
|
pkt->pts = pkt->dts + sc->dts_shift + sc->ctts_data[sc->ctts_index].duration;
|
2006-07-08 15:14:41 +03:00
|
|
|
/* update ctts context */
|
2009-02-08 11:57:28 +02:00
|
|
|
sc->ctts_sample++;
|
|
|
|
if (sc->ctts_index < sc->ctts_count &&
|
|
|
|
sc->ctts_data[sc->ctts_index].count == sc->ctts_sample) {
|
|
|
|
sc->ctts_index++;
|
|
|
|
sc->ctts_sample = 0;
|
2006-05-12 00:29:37 +03:00
|
|
|
}
|
2006-07-08 15:14:41 +03:00
|
|
|
} else {
|
2009-03-15 09:21:22 +02:00
|
|
|
int64_t next_dts = (sc->current_sample < st->nb_index_entries) ?
|
2008-04-26 20:51:02 +03:00
|
|
|
st->index_entries[sc->current_sample].timestamp : st->duration;
|
|
|
|
pkt->duration = next_dts - pkt->dts;
|
2015-02-28 23:51:16 +02:00
|
|
|
pkt->pts = pkt->dts;
|
2006-05-12 00:29:37 +03:00
|
|
|
}
|
2009-06-24 12:38:42 +03:00
|
|
|
if (st->discard == AVDISCARD_ALL)
|
|
|
|
goto retry;
|
2010-03-31 15:29:58 +03:00
|
|
|
pkt->flags |= sample->flags & AVINDEX_KEYFRAME ? AV_PKT_FLAG_KEY : 0;
|
2006-07-08 15:14:41 +03:00
|
|
|
pkt->pos = sample->pos;
|
2015-09-26 18:13:55 +02:00
|
|
|
av_log(s, AV_LOG_TRACE, "stream %d, pts %"PRId64", dts %"PRId64", pos 0x%"PRIx64", duration %"PRId64"\n",
|
2008-01-06 16:18:13 +02:00
|
|
|
pkt->stream_index, pkt->pts, pkt->dts, pkt->pos, pkt->duration);
|
2016-06-15 21:24:30 +02:00
|
|
|
|
|
|
|
/* Multiple stsd handling. */
|
|
|
|
if (sc->stsc_data) {
|
|
|
|
/* Keep track of the stsc index for the given sample, then check
|
|
|
|
* if the stsd index is different from the last used one. */
|
|
|
|
sc->stsc_sample++;
|
2016-07-20 18:48:09 +02:00
|
|
|
if (mov_stsc_index_valid(sc->stsc_index, sc->stsc_count) &&
|
2016-06-15 21:24:30 +02:00
|
|
|
mov_get_stsc_samples(sc, sc->stsc_index) == sc->stsc_sample) {
|
|
|
|
sc->stsc_index++;
|
|
|
|
sc->stsc_sample = 0;
|
|
|
|
/* Do not check indexes after a switch. */
|
|
|
|
} else if (sc->stsc_data[sc->stsc_index].id - 1 != sc->last_stsd_index) {
|
|
|
|
ret = mov_change_extradata(sc, pkt);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-07-08 15:14:41 +03:00
|
|
|
return 0;
|
|
|
|
}
|
2002-07-24 01:05:35 +03:00
|
|
|
|
2009-11-19 23:03:25 +02:00
|
|
|
static int mov_seek_stream(AVFormatContext *s, AVStream *st, int64_t timestamp, int flags)
|
2006-07-08 15:14:41 +03:00
|
|
|
{
|
|
|
|
MOVStreamContext *sc = st->priv_data;
|
|
|
|
int sample, time_sample;
|
2017-02-03 14:05:27 +02:00
|
|
|
unsigned int i;
|
2005-12-17 20:14:38 +02:00
|
|
|
|
2006-07-08 15:14:41 +03:00
|
|
|
sample = av_index_search_timestamp(st, timestamp, flags);
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(s, AV_LOG_TRACE, "stream %d, timestamp %"PRId64", sample %d\n", st->index, timestamp, sample);
|
2010-07-02 11:24:37 +03:00
|
|
|
if (sample < 0 && st->nb_index_entries && timestamp < st->index_entries[0].timestamp)
|
|
|
|
sample = 0;
|
2006-07-08 15:14:41 +03:00
|
|
|
if (sample < 0) /* not sure what to do */
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2006-07-08 15:14:41 +03:00
|
|
|
sc->current_sample = sample;
|
2015-03-16 10:57:35 +02:00
|
|
|
av_log(s, AV_LOG_TRACE, "stream %d, found sample %d\n", st->index, sc->current_sample);
|
2006-07-08 15:14:41 +03:00
|
|
|
/* adjust ctts index */
|
|
|
|
if (sc->ctts_data) {
|
|
|
|
time_sample = 0;
|
|
|
|
for (i = 0; i < sc->ctts_count; i++) {
|
2007-05-13 01:02:27 +03:00
|
|
|
int next = time_sample + sc->ctts_data[i].count;
|
|
|
|
if (next > sc->current_sample) {
|
2009-02-08 11:57:28 +02:00
|
|
|
sc->ctts_index = i;
|
|
|
|
sc->ctts_sample = sc->current_sample - time_sample;
|
2006-07-08 15:14:41 +03:00
|
|
|
break;
|
2005-12-17 20:14:38 +02:00
|
|
|
}
|
2007-05-13 01:02:27 +03:00
|
|
|
time_sample = next;
|
2004-03-13 23:02:26 +02:00
|
|
|
}
|
|
|
|
}
|
2016-06-15 21:24:30 +02:00
|
|
|
|
|
|
|
/* adjust stsd index */
|
|
|
|
time_sample = 0;
|
|
|
|
for (i = 0; i < sc->stsc_count; i++) {
|
|
|
|
int next = time_sample + mov_get_stsc_samples(sc, i);
|
|
|
|
if (next > sc->current_sample) {
|
|
|
|
sc->stsc_index = i;
|
|
|
|
sc->stsc_sample = sc->current_sample - time_sample;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
time_sample = next;
|
|
|
|
}
|
|
|
|
|
2006-07-08 15:14:41 +03:00
|
|
|
return sample;
|
2002-04-08 15:32:01 +03:00
|
|
|
}
|
|
|
|
|
2005-06-24 14:43:08 +03:00
|
|
|
static int mov_read_seek(AVFormatContext *s, int stream_index, int64_t sample_time, int flags)
|
2004-07-06 05:35:39 +03:00
|
|
|
{
|
2015-03-16 22:08:35 +02:00
|
|
|
MOVContext *mc = s->priv_data;
|
2006-07-08 15:14:41 +03:00
|
|
|
AVStream *st;
|
|
|
|
int sample;
|
|
|
|
int i;
|
2004-07-06 05:35:39 +03:00
|
|
|
|
2006-07-08 15:14:41 +03:00
|
|
|
if (stream_index >= s->nb_streams)
|
2012-01-15 00:25:12 +03:00
|
|
|
return AVERROR_INVALIDDATA;
|
2009-01-27 06:29:33 +02:00
|
|
|
if (sample_time < 0)
|
|
|
|
sample_time = 0;
|
2004-07-06 05:35:39 +03:00
|
|
|
|
2006-07-08 15:14:41 +03:00
|
|
|
st = s->streams[stream_index];
|
2009-11-19 23:03:25 +02:00
|
|
|
sample = mov_seek_stream(s, st, sample_time, flags);
|
2006-07-08 15:14:41 +03:00
|
|
|
if (sample < 0)
|
2012-01-15 00:25:12 +03:00
|
|
|
return sample;
|
2004-07-06 05:35:39 +03:00
|
|
|
|
2015-03-16 22:08:35 +02:00
|
|
|
if (mc->seek_individually) {
|
|
|
|
/* adjust seek timestamp to found sample timestamp */
|
|
|
|
int64_t seek_timestamp = st->index_entries[sample].timestamp;
|
2004-07-06 05:35:39 +03:00
|
|
|
|
2015-03-16 22:08:35 +02:00
|
|
|
for (i = 0; i < s->nb_streams; i++) {
|
|
|
|
int64_t timestamp;
|
|
|
|
st = s->streams[i];
|
|
|
|
if (stream_index == i)
|
|
|
|
continue;
|
2004-07-06 05:35:39 +03:00
|
|
|
|
2015-03-16 22:08:35 +02:00
|
|
|
timestamp = av_rescale_q(seek_timestamp, s->streams[stream_index]->time_base, st->time_base);
|
|
|
|
mov_seek_stream(s, st, timestamp, flags);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (i = 0; i < s->nb_streams; i++) {
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
st = s->streams[i];
|
|
|
|
sc = st->priv_data;
|
|
|
|
sc->current_sample = 0;
|
|
|
|
}
|
|
|
|
while (1) {
|
|
|
|
MOVStreamContext *sc;
|
|
|
|
AVIndexEntry *entry = mov_find_next_sample(s, &st);
|
|
|
|
if (!entry)
|
|
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
sc = st->priv_data;
|
|
|
|
if (sc->ffindex == stream_index && sc->current_sample == sample)
|
|
|
|
break;
|
|
|
|
sc->current_sample++;
|
|
|
|
}
|
2005-12-17 20:14:38 +02:00
|
|
|
}
|
2004-07-06 05:35:39 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-11-29 18:51:13 +02:00
|
|
|
#define OFFSET(x) offsetof(MOVContext, x)
|
|
|
|
#define FLAGS AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_DECODING_PARAM
|
|
|
|
static const AVOption mov_options[] = {
|
2015-03-16 22:08:35 +02:00
|
|
|
{"seek_streams_individually",
|
|
|
|
"Seek each stream individually to the to the closest point",
|
|
|
|
OFFSET(seek_individually), AV_OPT_TYPE_INT, { .i64 = 1 },
|
|
|
|
0, 1, FLAGS},
|
2014-11-29 18:51:13 +02:00
|
|
|
{ "export_all", "Export unrecognized metadata entries", OFFSET(export_all),
|
|
|
|
AV_OPT_TYPE_INT, { .i64 = 0 }, 0, 1, .flags = FLAGS },
|
2014-12-03 04:43:24 +02:00
|
|
|
{ "export_xmp", "Export full XMP metadata", OFFSET(export_xmp),
|
|
|
|
AV_OPT_TYPE_INT, { .i64 = 0 }, 0, 1, .flags = FLAGS },
|
2016-01-20 19:27:09 +02:00
|
|
|
{ "enable_drefs", "Enable external track support.", OFFSET(enable_drefs),
|
|
|
|
AV_OPT_TYPE_INT, { .i64 = 0 }, 0, 1, .flags = FLAGS },
|
2014-11-29 18:51:13 +02:00
|
|
|
{ NULL },
|
|
|
|
};
|
|
|
|
|
|
|
|
static const AVClass mov_class = {
|
|
|
|
.class_name = "mov,mp4,m4a,3gp,3g2,mj2",
|
|
|
|
.item_name = av_default_item_name,
|
|
|
|
.option = mov_options,
|
|
|
|
.version = LIBAVUTIL_VERSION_INT,
|
|
|
|
};
|
|
|
|
|
2011-01-26 00:03:28 +02:00
|
|
|
AVInputFormat ff_mov_demuxer = {
|
2011-07-16 23:18:12 +03:00
|
|
|
.name = "mov,mp4,m4a,3gp,3g2,mj2",
|
2012-07-24 04:23:48 +03:00
|
|
|
.long_name = NULL_IF_CONFIG_SMALL("QuickTime / MOV"),
|
2014-11-29 18:51:13 +02:00
|
|
|
.priv_class = &mov_class,
|
2011-07-16 23:18:12 +03:00
|
|
|
.priv_data_size = sizeof(MOVContext),
|
2014-04-01 20:07:50 +03:00
|
|
|
.extensions = "mov,mp4,m4a,3gp,3g2,mj2",
|
2011-07-16 23:18:12 +03:00
|
|
|
.read_probe = mov_probe,
|
|
|
|
.read_header = mov_read_header,
|
|
|
|
.read_packet = mov_read_packet,
|
|
|
|
.read_close = mov_read_close,
|
|
|
|
.read_seek = mov_read_seek,
|
2002-04-08 15:32:01 +03:00
|
|
|
};
|