movenc.c 52.4 KB
Newer Older
1 2 3
/*
 * MOV, 3GP, MP4 encoder.
 * Copyright (c) 2003 Thomas Raivio.
4
 * Copyright (c) 2004 Gildas Bazin <gbazin at videolan dot org>.
5 6 7 8 9 10 11 12 13 14 15 16 17
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
18
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19 20
 */
#include "avformat.h"
21
#include "avi.h"
22
#include "avio.h"
23
#include "mov.h"
24

25 26 27
#undef NDEBUG
#include <assert.h>

28 29 30
#define MOV_INDEX_CLUSTER_SIZE 16384
#define globalTimescale 1000

31 32 33
#define MODE_MP4 0
#define MODE_MOV 1
#define MODE_3GP 2
34
#define MODE_PSP 3 // example working PSP command line:
35
// ffmpeg -i testinput.avi  -f psp -r 14.985 -s 320x240 -b 768 -ar 24000 -ab 32 M4V00001.MP4
36
#define MODE_3G2 4
37

38
typedef struct MOVIentry {
39 40
    unsigned int flags, size;
    uint64_t     pos;
41
    unsigned int samplesInChunk;
42
    char         key_frame;
43
    unsigned int entries;
44
    int64_t      cts;
45 46 47
} MOVIentry;

typedef struct MOVIndex {
48
    int         mode;
49 50 51
    int         entry;
    long        timescale;
    long        time;
52
    int64_t     trackDuration;
53 54
    long        sampleCount;
    long        sampleDuration;
55
    long        sampleSize;
56
    int         hasKeyframes;
57
    int         hasBframes;
58
    int         language;
59
    int         trackID;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
60
    int         tag;
61 62 63
    AVCodecContext *enc;

    int         vosLen;
64
    uint8_t     *vosData;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
65
    MOVIentry   *cluster;
66 67
} MOVTrack;

68
typedef struct MOVContext {
69
    int     mode;
70
    int64_t time;
71
    int     nb_streams;
72
    offset_t mdat_pos;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
73
    uint64_t mdat_size;
74 75 76 77
    long    timescale;
    MOVTrack tracks[MAX_STREAMS];
} MOVContext;

78
//FIXME supprt 64bit varaint with wide placeholders
79
static offset_t updateSize (ByteIOContext *pb, offset_t pos)
80
{
81
    offset_t curpos = url_ftell(pb);
82
    url_fseek(pb, pos, SEEK_SET);
83
    put_be32(pb, curpos - pos); /* rewrite size */
84
    url_fseek(pb, curpos, SEEK_SET);
85 86

    return curpos - pos;
87 88
}

89
/* Chunk offset atom */
90
static int mov_write_stco_tag(ByteIOContext *pb, MOVTrack* track)
91 92
{
    int i;
Benjamin Larsson's avatar
Benjamin Larsson committed
93
    int mode64 = 0; //   use 32 bit size variant if possible
94
    offset_t pos = url_ftell(pb);
95
    put_be32(pb, 0); /* size */
96 97 98 99 100
    if (pos > UINT32_MAX) {
        mode64 = 1;
        put_tag(pb, "co64");
    } else
        put_tag(pb, "stco");
101 102 103
    put_be32(pb, 0); /* version & flags */
    put_be32(pb, track->entry); /* entry count */
    for (i=0; i<track->entry; i++) {
104
        if(mode64 == 1)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
105
            put_be64(pb, track->cluster[i].pos);
106
        else
Baptiste Coudurier's avatar
Baptiste Coudurier committed
107
            put_be32(pb, track->cluster[i].pos);
108
    }
109
    return updateSize (pb, pos);
110 111
}

112
/* Sample size atom */
113
static int mov_write_stsz_tag(ByteIOContext *pb, MOVTrack* track)
114
{
115
    int equalChunks = 1;
116
    int i, j, entries = 0, tst = -1, oldtst = -1;
117

118
    offset_t pos = url_ftell(pb);
119
    put_be32(pb, 0); /* size */
120 121 122
    put_tag(pb, "stsz");
    put_be32(pb, 0); /* version & flags */

123
    for (i=0; i<track->entry; i++) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
124
        tst = track->cluster[i].size/track->cluster[i].entries;
125 126
        if(oldtst != -1 && tst != oldtst) {
            equalChunks = 0;
127 128
        }
        oldtst = tst;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
129
        entries += track->cluster[i].entries;
130
    }
131
    if (equalChunks) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
132
        int sSize = track->cluster[0].size/track->cluster[0].entries;
133
        put_be32(pb, sSize); // sample size
134
        put_be32(pb, entries); // sample count
135
    }
136
    else {
137 138
        put_be32(pb, 0); // sample size
        put_be32(pb, entries); // sample count
139
        for (i=0; i<track->entry; i++) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
140 141 142
            for ( j=0; j<track->cluster[i].entries; j++) {
                put_be32(pb, track->cluster[i].size /
                         track->cluster[i].entries);
143
            }
144 145
        }
    }
146
    return updateSize (pb, pos);
147 148
}

149
/* Sample to chunk atom */
150
static int mov_write_stsc_tag(ByteIOContext *pb, MOVTrack* track)
151
{
152 153
    int index = 0, oldval = -1, i;
    offset_t entryPos, curpos;
154

155
    offset_t pos = url_ftell(pb);
156
    put_be32(pb, 0); /* size */
157
    put_tag(pb, "stsc");
158
    put_be32(pb, 0); // version & flags
159
    entryPos = url_ftell(pb);
160
    put_be32(pb, track->entry); // entry count
161
    for (i=0; i<track->entry; i++) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
162
        if(oldval != track->cluster[i].samplesInChunk)
163
        {
164
            put_be32(pb, i+1); // first chunk
Baptiste Coudurier's avatar
Baptiste Coudurier committed
165
            put_be32(pb, track->cluster[i].samplesInChunk); // samples per chunk
166
            put_be32(pb, 0x1); // sample description index
Baptiste Coudurier's avatar
Baptiste Coudurier committed
167
            oldval = track->cluster[i].samplesInChunk;
168
            index++;
169 170
        }
    }
171 172
    curpos = url_ftell(pb);
    url_fseek(pb, entryPos, SEEK_SET);
173
    put_be32(pb, index); // rewrite size
174
    url_fseek(pb, curpos, SEEK_SET);
175

176
    return updateSize (pb, pos);
177 178
}

179
/* Sync sample atom */
180
static int mov_write_stss_tag(ByteIOContext *pb, MOVTrack* track)
181
{
182 183 184
    offset_t curpos, entryPos;
    int i, index = 0;
    offset_t pos = url_ftell(pb);
185
    put_be32(pb, 0); // size
186
    put_tag(pb, "stss");
187
    put_be32(pb, 0); // version & flags
188
    entryPos = url_ftell(pb);
189
    put_be32(pb, track->entry); // entry count
190
    for (i=0; i<track->entry; i++) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
191
        if(track->cluster[i].key_frame == 1) {
192 193 194 195 196 197
            put_be32(pb, i+1);
            index++;
        }
    }
    curpos = url_ftell(pb);
    url_fseek(pb, entryPos, SEEK_SET);
198
    put_be32(pb, index); // rewrite size
199 200
    url_fseek(pb, curpos, SEEK_SET);
    return updateSize (pb, pos);
201 202
}

203
static int mov_write_damr_tag(ByteIOContext *pb)
204 205 206 207 208
{
    put_be32(pb, 0x11); /* size */
    put_tag(pb, "damr");
    put_tag(pb, "FFMP");
    put_byte(pb, 0);
209

210 211 212 213
    //put_be16(pb, 0x80); /* Mode set (all modes for AMR_NB) */
    //put_be16(pb, 0xa); /* Mode change period (no restriction) */
    put_be16(pb, 0x81ff); /* Mode set (all modes for AMR_NB) */
    put_be16(pb, 1); /* Mode change period (no restriction) */
214 215 216
    return 0x11;
}

Baptiste Coudurier's avatar
Baptiste Coudurier committed
217 218 219 220 221 222 223 224 225 226 227 228
static int mov_write_samr_tag(ByteIOContext *pb)
{
    put_be32(pb, 0x11); /* size */
    put_tag(pb, "samr");
    put_tag(pb, "FFMP");
    put_byte(pb, 1);

    put_be16(pb, 0x80); /* Mode set (all modes for AMR_NB) */
    put_be16(pb, 0x5); /* Mode change period (no restriction) */
    return 0x11;
}

229 230 231 232 233 234 235 236
static int mov_write_enda_tag(ByteIOContext *pb)
{
    put_be32(pb, 10);
    put_tag(pb, "enda");
    put_be16(pb, 1); /* little endian */
    return 10;
}

Baptiste Coudurier's avatar
Baptiste Coudurier committed
237 238
static unsigned int descrLength(unsigned int len)
{
Michael Niedermayer's avatar
Michael Niedermayer committed
239 240 241
    int i;
    for(i=1; len>>(7*i); i++);
    return len + 1 + i;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
242 243
}

Michael Niedermayer's avatar
Michael Niedermayer committed
244
static void putDescr(ByteIOContext *pb, int tag, unsigned int size)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
245
{
Michael Niedermayer's avatar
Michael Niedermayer committed
246 247 248 249 250
    int i= descrLength(size) - size - 2;
    put_byte(pb, tag);
    for(; i>0; i--)
        put_byte(pb, (size>>(7*i)) | 0x80);
    put_byte(pb, size & 0x7F);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
251 252 253 254 255
}

static int mov_write_esds_tag(ByteIOContext *pb, MOVTrack* track) // Basic
{
    offset_t pos = url_ftell(pb);
256
    int decoderSpecificInfoLen = track->vosLen ? descrLength(track->vosLen):0;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303

    put_be32(pb, 0);               // size
    put_tag(pb, "esds");
    put_be32(pb, 0);               // Version

    // ES descriptor
    putDescr(pb, 0x03, 3 + descrLength(13 + decoderSpecificInfoLen) +
             descrLength(1));
    put_be16(pb, track->trackID);
    put_byte(pb, 0x00);            // flags (= no flags)

    // DecoderConfig descriptor
    putDescr(pb, 0x04, 13 + decoderSpecificInfoLen);

    // Object type indication
    put_byte(pb, codec_get_tag(ff_mov_obj_type, track->enc->codec_id));

    // the following fields is made of 6 bits to identify the streamtype (4 for video, 5 for audio)
    // plus 1 bit to indicate upstream and 1 bit set to 1 (reserved)
    if(track->enc->codec_type == CODEC_TYPE_AUDIO)
        put_byte(pb, 0x15);            // flags (= Audiostream)
    else
        put_byte(pb, 0x11);            // flags (= Visualstream)

    put_byte(pb,  track->enc->rc_buffer_size>>(3+16));             // Buffersize DB (24 bits)
    put_be16(pb, (track->enc->rc_buffer_size>>3)&0xFFFF);          // Buffersize DB

    put_be32(pb, FFMAX(track->enc->bit_rate, track->enc->rc_max_rate));     // maxbitrate  (FIXME should be max rate in any 1 sec window)
    if(track->enc->rc_max_rate != track->enc->rc_min_rate || track->enc->rc_min_rate==0)
        put_be32(pb, 0);     // vbr
    else
        put_be32(pb, track->enc->rc_max_rate);     // avg bitrate

    if (track->vosLen)
    {
        // DecoderSpecific info descriptor
        putDescr(pb, 0x05, track->vosLen);
        put_buffer(pb, track->vosData, track->vosLen);
    }


    // SL descriptor
    putDescr(pb, 0x06, 1);
    put_byte(pb, 0x02);
    return updateSize (pb, pos);
}

304 305
static int mov_write_wave_tag(ByteIOContext *pb, MOVTrack* track)
{
306
    offset_t pos = url_ftell(pb);
307 308 309 310 311 312

    put_be32(pb, 0);     /* size */
    put_tag(pb, "wave");

    put_be32(pb, 12);    /* size */
    put_tag(pb, "frma");
313
    put_le32(pb, track->tag);
314

315
    if (track->enc->codec_id == CODEC_ID_AAC) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
316 317 318 319
        /* useless atom needed by mplayer, ipod, not needed by quicktime */
        put_be32(pb, 12); /* size */
        put_tag(pb, "mp4a");
        put_be32(pb, 0);
320 321 322 323
        mov_write_esds_tag(pb, track);
    } else if (track->enc->codec_id == CODEC_ID_PCM_S24LE ||
               track->enc->codec_id == CODEC_ID_PCM_S32LE) {
        mov_write_enda_tag(pb);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
324 325
    } else if (track->enc->codec_id == CODEC_ID_AMR_NB) {
        mov_write_samr_tag(pb);
326
    }
327 328 329 330 331 332 333

    put_be32(pb, 8);     /* size */
    put_be32(pb, 0);     /* null tag */

    return updateSize (pb, pos);
}

334
static const CodecTag codec_movaudio_tags[] = {
335 336 337 338 339 340 341
    { CODEC_ID_PCM_MULAW, MKTAG('u', 'l', 'a', 'w') },
    { CODEC_ID_PCM_ALAW, MKTAG('a', 'l', 'a', 'w') },
    { CODEC_ID_ADPCM_IMA_QT, MKTAG('i', 'm', 'a', '4') },
    { CODEC_ID_MACE3, MKTAG('M', 'A', 'C', '3') },
    { CODEC_ID_MACE6, MKTAG('M', 'A', 'C', '6') },
    { CODEC_ID_AAC, MKTAG('m', 'p', '4', 'a') },
    { CODEC_ID_AMR_NB, MKTAG('s', 'a', 'm', 'r') },
342
    { CODEC_ID_AMR_WB, MKTAG('s', 'a', 'w', 'b') },
343 344
    { CODEC_ID_PCM_S16BE, MKTAG('t', 'w', 'o', 's') },
    { CODEC_ID_PCM_S16LE, MKTAG('s', 'o', 'w', 't') },
345 346 347 348
    { CODEC_ID_PCM_S24BE, MKTAG('i', 'n', '2', '4') },
    { CODEC_ID_PCM_S24LE, MKTAG('i', 'n', '2', '4') },
    { CODEC_ID_PCM_S32BE, MKTAG('i', 'n', '3', '2') },
    { CODEC_ID_PCM_S32LE, MKTAG('i', 'n', '3', '2') },
349
    { CODEC_ID_MP3, MKTAG('.', 'm', 'p', '3') },
350
    { CODEC_ID_NONE, 0 },
351 352
};

353
static int mov_write_audio_tag(ByteIOContext *pb, MOVTrack* track)
354
{
355
    offset_t pos = url_ftell(pb);
356 357 358
    int vbr=  track->enc->codec_id == CODEC_ID_AAC ||
              track->enc->codec_id == CODEC_ID_MP3 ||
              track->enc->codec_id == CODEC_ID_AMR_NB;
359
    int version = track->mode == MODE_MOV &&
360
        (vbr ||
361 362
         track->enc->codec_id == CODEC_ID_PCM_S32LE ||
         track->enc->codec_id == CODEC_ID_PCM_S24LE);
363

364
    put_be32(pb, 0); /* size */
Baptiste Coudurier's avatar
Baptiste Coudurier committed
365
    put_le32(pb, track->tag); // store it byteswapped
366 367 368
    put_be32(pb, 0); /* Reserved */
    put_be16(pb, 0); /* Reserved */
    put_be16(pb, 1); /* Data-reference index, XXX  == 1 */
369

370
    /* SoundDescription */
371
    put_be16(pb, version); /* Version */
372
    put_be16(pb, 0); /* Revision level */
373 374
    put_be32(pb, 0); /* Reserved */

375 376
    put_be16(pb, track->enc->channels); /* Number of channels */
    /* TODO: Currently hard-coded to 16-bit, there doesn't seem
377
                 to be a good way to get number of bits of audio */
378
    put_be16(pb, 0x10); /* Reserved */
379

380
    if(vbr) {
381
        put_be16(pb, 0xfffe); /* compression ID (vbr)*/
Baptiste Coudurier's avatar
Baptiste Coudurier committed
382
    } else {
383 384
        put_be16(pb, 0); /* compression ID (= 0) */
    }
385
    put_be16(pb, 0); /* packet size (= 0) */
386 387 388
    put_be16(pb, track->timescale); /* Time scale */
    put_be16(pb, 0); /* Reserved */

389
    if(version == 1) { /* SoundDescription V1 extended info */
390 391
        if (vbr)
            track->sampleSize = 0;
392
        put_be32(pb, track->enc->frame_size); /* Samples per packet */
393
        put_be32(pb, track->sampleSize / track->enc->channels); /* Bytes per packet */
394
        put_be32(pb, track->sampleSize); /* Bytes per frame */
Baptiste Coudurier's avatar
Baptiste Coudurier committed
395
        put_be32(pb, 2); /* Bytes per sample */
396 397
    }

398 399 400 401 402
    if(track->mode == MODE_MOV &&
       (track->enc->codec_id == CODEC_ID_AAC ||
        track->enc->codec_id == CODEC_ID_AMR_NB ||
        track->enc->codec_id == CODEC_ID_PCM_S24LE ||
        track->enc->codec_id == CODEC_ID_PCM_S32LE))
403
        mov_write_wave_tag(pb, track);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
404 405 406 407 408
    else if(track->enc->codec_id == CODEC_ID_AAC)
        mov_write_esds_tag(pb, track);
    else if(track->enc->codec_id == CODEC_ID_AMR_NB)
        mov_write_damr_tag(pb);

409
    return updateSize (pb, pos);
410 411
}

412
static int mov_write_d263_tag(ByteIOContext *pb)
413 414 415 416 417 418 419 420 421
{
    put_be32(pb, 0xf); /* size */
    put_tag(pb, "d263");
    put_tag(pb, "FFMP");
    put_be16(pb, 0x0a);
    put_byte(pb, 0);
    return 0xf;
}

422 423
/* TODO: No idea about these values */
static int mov_write_svq3_tag(ByteIOContext *pb)
424
{
425 426 427 428 429 430
    put_be32(pb, 0x15);
    put_tag(pb, "SMI ");
    put_tag(pb, "SEQH");
    put_be32(pb, 0x5);
    put_be32(pb, 0xe2c0211d);
    put_be32(pb, 0xc0000000);
431
    put_byte(pb, 0);
432
    return 0x15;
433 434
}

435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450
static uint8_t *avc_find_startcode( uint8_t *p, uint8_t *end )
{
    uint8_t *a = p + 4 - ((int)p & 3);

    for( end -= 3; p < a && p < end; p++ ) {
        if( p[0] == 0 && p[1] == 0 && p[2] == 1 )
            return p;
    }

    for( end -= 3; p < end; p += 4 ) {
        uint32_t x = *(uint32_t*)p;
//      if( (x - 0x01000100) & (~x) & 0x80008000 ) // little endian
//      if( (x - 0x00010001) & (~x) & 0x00800080 ) // big endian
        if( (x - 0x01010101) & (~x) & 0x80808080 ) { // generic
            if( p[1] == 0 ) {
                if( p[0] == 0 && p[2] == 1 )
Baptiste Coudurier's avatar
Baptiste Coudurier committed
451
                    return p-1;
452
                if( p[2] == 0 && p[3] == 1 )
Baptiste Coudurier's avatar
Baptiste Coudurier committed
453
                    return p;
454 455 456
            }
            if( p[3] == 0 ) {
                if( p[2] == 0 && p[4] == 1 )
Baptiste Coudurier's avatar
Baptiste Coudurier committed
457
                    return p+1;
458
                if( p[4] == 0 && p[5] == 1 )
Baptiste Coudurier's avatar
Baptiste Coudurier committed
459
                    return p+2;
460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525
            }
        }
    }

    for( end += 3; p < end; p++ ) {
        if( p[0] == 0 && p[1] == 0 && p[2] == 1 )
            return p;
    }

    return end + 3;
}

static void avc_parse_nal_units(uint8_t **buf, int *size)
{
    ByteIOContext pb;
    uint8_t *p = *buf;
    uint8_t *end = p + *size;
    uint8_t *nal_start, *nal_end;

    url_open_dyn_buf(&pb);
    nal_start = avc_find_startcode(p, end);
    while (nal_start < end) {
        while(!*(nal_start++));
        nal_end = avc_find_startcode(nal_start, end);
        put_be32(&pb, nal_end - nal_start);
        put_buffer(&pb, nal_start, nal_end - nal_start);
        nal_start = nal_end;
    }
    av_freep(buf);
    *size = url_close_dyn_buf(&pb, buf);
}

static int mov_write_avcc_tag(ByteIOContext *pb, MOVTrack *track)
{
    offset_t pos = url_ftell(pb);

    put_be32(pb, 0);
    put_tag(pb, "avcC");
    if (track->vosLen > 6) {
        /* check for h264 start code */
        if (BE_32(track->vosData) == 0x00000001) {
            uint8_t *buf, *end;
            uint32_t sps_size=0, pps_size=0;
            uint8_t *sps=0, *pps=0;

            avc_parse_nal_units(&track->vosData, &track->vosLen);
            buf = track->vosData;
            end = track->vosData + track->vosLen;

            /* look for sps and pps */
            while (buf < end) {
                unsigned int size;
                uint8_t nal_type;
                size = BE_32(buf);
                nal_type = buf[4] & 0x1f;
                if (nal_type == 7) { /* SPS */
                    sps = buf + 4;
                    sps_size = size;
                } else if (nal_type == 8) { /* PPS */
                    pps = buf + 4;
                    pps_size = size;
                }
                buf += size + 4;
            }
            assert(sps);
            assert(pps);
526 527 528 529 530 531 532 533

            put_byte(pb, 1); /* version */
            put_byte(pb, sps[1]); /* profile */
            put_byte(pb, sps[2]); /* profile compat */
            put_byte(pb, sps[3]); /* level */
            put_byte(pb, 0xff); /* 6 bits reserved (111111) + 2 bits nal size length - 1 (11) */
            put_byte(pb, 0xe1); /* 3 bits reserved (111) + 5 bits number of sps (00001) */

534 535 536 537 538 539 540 541 542 543 544 545
            put_be16(pb, sps_size);
            put_buffer(pb, sps, sps_size);
            put_byte(pb, 1); /* number of pps */
            put_be16(pb, pps_size);
            put_buffer(pb, pps, pps_size);
        } else {
            put_buffer(pb, track->vosData, track->vosLen);
        }
    }
    return updateSize(pb, pos);
}

546
static const CodecTag codec_movvideo_tags[] = {
547 548 549 550
    { CODEC_ID_SVQ1, MKTAG('S', 'V', 'Q', '1') },
    { CODEC_ID_SVQ3, MKTAG('S', 'V', 'Q', '3') },
    { CODEC_ID_MPEG4, MKTAG('m', 'p', '4', 'v') },
    { CODEC_ID_H263, MKTAG('s', '2', '6', '3') },
551
    { CODEC_ID_H264, MKTAG('a', 'v', 'c', '1') },
552 553 554 555 556 557 558
    /* special handling in mov_find_video_codec_tag */
    { CODEC_ID_DVVIDEO, MKTAG('d', 'v', 'c', ' ') }, /* DV NTSC */
    { CODEC_ID_DVVIDEO, MKTAG('d', 'v', 'c', 'p') }, /* DV PAL */
    { CODEC_ID_DVVIDEO, MKTAG('d', 'v', 'p', 'p') }, /* DVCPRO PAL */
    { CODEC_ID_DVVIDEO, MKTAG('d', 'v', '5', 'n') }, /* DVCPRO50 NTSC */
    { CODEC_ID_DVVIDEO, MKTAG('d', 'v', '5', 'p') }, /* DVCPRO50 PAL */
    { CODEC_ID_NONE, 0 },
559 560
};

Baptiste Coudurier's avatar
Baptiste Coudurier committed
561
static int mov_find_video_codec_tag(AVFormatContext *s, MOVTrack *track)
562
{
Baptiste Coudurier's avatar
Baptiste Coudurier committed
563
    int tag = track->enc->codec_tag;
564 565 566 567 568 569 570 571 572 573 574 575
    if (!tag) {
        if (track->enc->codec_id == CODEC_ID_DVVIDEO) {
            if (track->enc->height == 480) { /* NTSC */
                if (track->enc->pix_fmt == PIX_FMT_YUV422P)
                    tag = MKTAG('d', 'v', '5', 'n');
                else
                    tag = MKTAG('d', 'v', 'c', ' ');
            } else { /* assume PAL */
                if (track->enc->pix_fmt == PIX_FMT_YUV422P)
                    tag = MKTAG('d', 'v', '5', 'p');
                else if (track->enc->pix_fmt == PIX_FMT_YUV420P)
                    tag = MKTAG('d', 'v', 'c', 'p');
Baptiste Coudurier's avatar
Baptiste Coudurier committed
576 577
                else
                    tag = MKTAG('d', 'v', 'p', 'p');
578 579 580 581 582 583
            }
        } else {
            tag = codec_get_tag(codec_movvideo_tags, track->enc->codec_id);
        }
    }
    // if no mac fcc found, try with Microsoft tags
Baptiste Coudurier's avatar
Baptiste Coudurier committed
584
    if (!tag) {
585
        tag = codec_get_tag(codec_bmp_tags, track->enc->codec_id);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607
        if (tag) {
            av_log(s, AV_LOG_INFO, "Warning, using MS style video codec tag, the file may be unplayable!\n");
        }
    }
    assert(tag);
    return tag;
}

static int mov_find_audio_codec_tag(AVFormatContext *s, MOVTrack *track)
{
    int tag = track->enc->codec_tag;
    if (!tag) {
        tag = codec_get_tag(codec_movaudio_tags, track->enc->codec_id);
    }
    // if no mac fcc found, try with Microsoft tags
    if (!tag) {
        int ms_tag = codec_get_tag(codec_wav_tags, track->enc->codec_id);
        if (ms_tag) {
            tag = MKTAG('m', 's', ((ms_tag >> 8) & 0xff), (ms_tag & 0xff));
            av_log(s, AV_LOG_INFO, "Warning, using MS style audio codec tag, the file may be unplayable!\n");
        }
    }
608 609 610 611
    assert(tag);
    return tag;
}

612
static int mov_write_video_tag(ByteIOContext *pb, MOVTrack* track)
613
{
614
    offset_t pos = url_ftell(pb);
615
    char compressor_name[32];
616

617
    put_be32(pb, 0); /* size */
Baptiste Coudurier's avatar
Baptiste Coudurier committed
618
    put_le32(pb, track->tag); // store it byteswapped
619 620 621 622
    put_be32(pb, 0); /* Reserved */
    put_be16(pb, 0); /* Reserved */
    put_be16(pb, 1); /* Data-reference index */

623 624 625 626 627 628 629 630 631 632
    put_be16(pb, 0); /* Codec stream version */
    put_be16(pb, 0); /* Codec stream revision (=0) */
    put_tag(pb, "FFMP"); /* Vendor */
    if(track->enc->codec_id == CODEC_ID_RAWVIDEO) {
        put_be32(pb, 0); /* Temporal Quality */
        put_be32(pb, 0x400); /* Spatial Quality = lossless*/
    } else {
        put_be32(pb, 0x200); /* Temporal Quality = normal */
        put_be32(pb, 0x200); /* Spatial Quality = normal */
    }
633 634
    put_be16(pb, track->enc->width); /* Video width */
    put_be16(pb, track->enc->height); /* Video height */
635 636
    put_be32(pb, 0x00480000); /* Horizontal resolution 72dpi */
    put_be32(pb, 0x00480000); /* Vertical resolution 72dpi */
637 638
    put_be32(pb, 0); /* Data size (= 0) */
    put_be16(pb, 1); /* Frame count (= 1) */
639

640
    memset(compressor_name,0,32);
641
    if (track->enc->codec && track->enc->codec->name)
642
        strncpy(compressor_name,track->enc->codec->name,31);
643
    put_byte(pb, strlen(compressor_name));
644
    put_buffer(pb, compressor_name, 31);
645

646 647 648 649 650 651 652
    put_be16(pb, 0x18); /* Reserved */
    put_be16(pb, 0xffff); /* Reserved */
    if(track->enc->codec_id == CODEC_ID_MPEG4)
        mov_write_esds_tag(pb, track);
    else if(track->enc->codec_id == CODEC_ID_H263)
        mov_write_d263_tag(pb);
    else if(track->enc->codec_id == CODEC_ID_SVQ3)
653
        mov_write_svq3_tag(pb);
654 655
    else if(track->enc->codec_id == CODEC_ID_H264)
        mov_write_avcc_tag(pb, track);
656 657

    return updateSize (pb, pos);
658 659
}

660
static int mov_write_stsd_tag(ByteIOContext *pb, MOVTrack* track)
661
{
662
    offset_t pos = url_ftell(pb);
663 664 665 666
    put_be32(pb, 0); /* size */
    put_tag(pb, "stsd");
    put_be32(pb, 0); /* version & flags */
    put_be32(pb, 1); /* entry count */
667 668 669 670
    if (track->enc->codec_type == CODEC_TYPE_VIDEO)
        mov_write_video_tag(pb, track);
    else if (track->enc->codec_type == CODEC_TYPE_AUDIO)
        mov_write_audio_tag(pb, track);
671
    return updateSize(pb, pos);
672 673
}

674 675 676 677 678 679 680 681 682
static int mov_write_ctts_tag(ByteIOContext *pb, MOVTrack* track)
{
    Time2Sample *ctts_entries;
    uint32_t entries = 0;
    uint32_t atom_size;
    int i;

    ctts_entries = av_malloc((track->entry + 1) * sizeof(*ctts_entries)); /* worst case */
    ctts_entries[0].count = 1;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
683
    ctts_entries[0].duration = track->cluster[0].cts;
684
    for (i=1; i<track->entry; i++) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
685
        if (track->cluster[i].cts == ctts_entries[entries].duration) {
686 687 688
            ctts_entries[entries].count++; /* compress */
        } else {
            entries++;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
689
            ctts_entries[entries].duration = track->cluster[i].cts;
690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706
            ctts_entries[entries].count = 1;
        }
    }
    entries++; /* last one */
    atom_size = 16 + (entries * 8);
    put_be32(pb, atom_size); /* size */
    put_tag(pb, "ctts");
    put_be32(pb, 0); /* version & flags */
    put_be32(pb, entries); /* entry count */
    for (i=0; i<entries; i++) {
        put_be32(pb, ctts_entries[i].count);
        put_be32(pb, ctts_entries[i].duration);
    }
    av_free(ctts_entries);
    return atom_size;
}

Michael Niedermayer's avatar
Michael Niedermayer committed
707
/* TODO: */
708
/* Time to sample atom */
709
static int mov_write_stts_tag(ByteIOContext *pb, MOVTrack* track)
710 711 712 713 714 715
{
    put_be32(pb, 0x18); /* size */
    put_tag(pb, "stts");
    put_be32(pb, 0); /* version & flags */
    put_be32(pb, 1); /* entry count */

716 717
    put_be32(pb, track->sampleCount); /* sample count */
    put_be32(pb, track->sampleDuration); /* sample duration */
718 719 720
    return 0x18;
}

721
static int mov_write_dref_tag(ByteIOContext *pb)
722 723 724 725 726 727 728 729 730 731 732 733 734
{
    put_be32(pb, 28); /* size */
    put_tag(pb, "dref");
    put_be32(pb, 0); /* version & flags */
    put_be32(pb, 1); /* entry count */

    put_be32(pb, 0xc); /* size */
    put_tag(pb, "url ");
    put_be32(pb, 1); /* version & flags */

    return 28;
}

735
static int mov_write_stbl_tag(ByteIOContext *pb, MOVTrack* track)
736
{
737
    offset_t pos = url_ftell(pb);
738 739
    put_be32(pb, 0); /* size */
    put_tag(pb, "stbl");
740 741
    mov_write_stsd_tag(pb, track);
    mov_write_stts_tag(pb, track);
742
    if (track->enc->codec_type == CODEC_TYPE_VIDEO &&
743
        track->hasKeyframes < track->entry)
744
        mov_write_stss_tag(pb, track);
745 746 747
    if (track->enc->codec_type == CODEC_TYPE_VIDEO &&
        track->hasBframes)
        mov_write_ctts_tag(pb, track);
748 749 750 751
    mov_write_stsc_tag(pb, track);
    mov_write_stsz_tag(pb, track);
    mov_write_stco_tag(pb, track);
    return updateSize(pb, pos);
752 753
}

754
static int mov_write_dinf_tag(ByteIOContext *pb)
755
{
756
    offset_t pos = url_ftell(pb);
757 758
    put_be32(pb, 0); /* size */
    put_tag(pb, "dinf");
759 760
    mov_write_dref_tag(pb);
    return updateSize(pb, pos);
761 762
}

763
static int mov_write_smhd_tag(ByteIOContext *pb)
764 765 766 767 768 769 770 771 772
{
    put_be32(pb, 16); /* size */
    put_tag(pb, "smhd");
    put_be32(pb, 0); /* version & flags */
    put_be16(pb, 0); /* reserved (balance, normally = 0) */
    put_be16(pb, 0); /* reserved */
    return 16;
}

773
static int mov_write_vmhd_tag(ByteIOContext *pb)
774 775 776 777 778 779 780 781
{
    put_be32(pb, 0x14); /* size (always 0x14) */
    put_tag(pb, "vmhd");
    put_be32(pb, 0x01); /* version & flags */
    put_be64(pb, 0); /* reserved (graphics mode = copy) */
    return 0x14;
}

782
static int mov_write_hdlr_tag(ByteIOContext *pb, MOVTrack* track)
783
{
784
    const char *descr, *hdlr, *hdlr_type;
785
    offset_t pos = url_ftell(pb);
786

787
    if (!track) { /* no media --> data handler */
788 789 790
        hdlr = "dhlr";
        hdlr_type = "url ";
        descr = "DataHandler";
791
    } else {
792 793 794 795 796 797 798 799
        hdlr = (track->mode == MODE_MOV) ? "mhlr" : "\0\0\0\0";
        if (track->enc->codec_type == CODEC_TYPE_VIDEO) {
            hdlr_type = "vide";
            descr = "VideoHandler";
        } else {
            hdlr_type = "soun";
            descr = "SoundHandler";
        }
800
    }
801

802
    put_be32(pb, 0); /* size */
803 804
    put_tag(pb, "hdlr");
    put_be32(pb, 0); /* Version & flags */
805
    put_buffer(pb, hdlr, 4); /* handler */
806
    put_tag(pb, hdlr_type); /* handler type */
807 808 809
    put_be32(pb ,0); /* reserved */
    put_be32(pb ,0); /* reserved */
    put_be32(pb ,0); /* reserved */
810 811 812 813 814 815 816
    put_byte(pb, strlen(descr)); /* string counter */
    put_buffer(pb, descr, strlen(descr)); /* handler description */
    return updateSize(pb, pos);
}

static int mov_write_minf_tag(ByteIOContext *pb, MOVTrack* track)
{
817
    offset_t pos = url_ftell(pb);
818 819
    put_be32(pb, 0); /* size */
    put_tag(pb, "minf");
820
    if(track->enc->codec_type == CODEC_TYPE_VIDEO)
821
        mov_write_vmhd_tag(pb);
822
    else
823 824 825 826 827
        mov_write_smhd_tag(pb);
    if (track->mode == MODE_MOV) /* FIXME: Why do it for MODE_MOV only ? */
        mov_write_hdlr_tag(pb, NULL);
    mov_write_dinf_tag(pb);
    mov_write_stbl_tag(pb, track);
828
    return updateSize(pb, pos);
829 830
}

831
static int mov_write_mdhd_tag(ByteIOContext *pb, MOVTrack* track)
832
{
833 834 835
    int version = track->trackDuration < INT32_MAX ? 0 : 1;

    (version == 1) ? put_be32(pb, 44) : put_be32(pb, 32); /* size */
836
    put_tag(pb, "mdhd");
837 838 839 840 841 842 843 844 845
    put_byte(pb, version);
    put_be24(pb, 0); /* flags */
    if (version == 1) {
        put_be64(pb, track->time);
        put_be64(pb, track->time);
    } else {
        put_be32(pb, track->time); /* creation time */
        put_be32(pb, track->time); /* modification time */
    }
846
    put_be32(pb, track->timescale); /* time scale (sample rate for audio) */
847
    (version == 1) ? put_be64(pb, track->trackDuration) : put_be32(pb, track->trackDuration); /* duration */
848
    put_be16(pb, track->language); /* language */
849 850 851 852
    put_be16(pb, 0); /* reserved (quality) */
    return 32;
}

853
static int mov_write_mdia_tag(ByteIOContext *pb, MOVTrack* track)
854
{
855
    offset_t pos = url_ftell(pb);
856 857
    put_be32(pb, 0); /* size */
    put_tag(pb, "mdia");
858 859 860 861
    mov_write_mdhd_tag(pb, track);
    mov_write_hdlr_tag(pb, track);
    mov_write_minf_tag(pb, track);
    return updateSize(pb, pos);
862 863
}

864
static int mov_write_tkhd_tag(ByteIOContext *pb, MOVTrack* track