movenc.c 51.8 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
 * This file is part of FFmpeg.
 *
 * FFmpeg is free software; you can redistribute it and/or
9 10
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
11
 * version 2.1 of the License, or (at your option) any later version.
12
 *
13
 * FFmpeg is distributed in the hope that it will be useful,
14 15 16 17 18
 * 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
19
 * License along with FFmpeg; if not, write to the Free Software
20
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 22
 */
#include "avformat.h"
23
#include "riff.h"
24
#include "avio.h"
25
#include "isom.h"
26

27 28 29
#undef NDEBUG
#include <assert.h>

30 31 32
#define MOV_INDEX_CLUSTER_SIZE 16384
#define globalTimescale 1000

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

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

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

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

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

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

    return curpos - pos;
89 90
}

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

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

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

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

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

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

178
    return updateSize (pb, pos);
179 180
}

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

205
static int mov_write_amr_tag(ByteIOContext *pb, MOVTrack *track)
206 207
{
    put_be32(pb, 0x11); /* size */
208 209
    if (track->mode == MODE_MOV) put_tag(pb, "samr");
    else                         put_tag(pb, "damr");
210
    put_tag(pb, "FFMP");
211
    put_byte(pb, 0); /* decoder version */
Baptiste Coudurier's avatar
Baptiste Coudurier committed
212

Baptiste Coudurier's avatar
Baptiste Coudurier committed
213
    put_be16(pb, 0x81FF); /* Mode set (all modes for AMR_NB) */
214 215
    put_byte(pb, 0x00); /* Mode change period (no restriction) */
    put_byte(pb, 0x01); /* Frames per sample */
Baptiste Coudurier's avatar
Baptiste Coudurier committed
216 217 218
    return 0x11;
}

219 220 221 222 223 224 225 226
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
227 228
static unsigned int descrLength(unsigned int len)
{
Michael Niedermayer's avatar
Michael Niedermayer committed
229 230 231
    int i;
    for(i=1; len>>(7*i); i++);
    return len + 1 + i;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
232 233
}

Michael Niedermayer's avatar
Michael Niedermayer committed
234
static void putDescr(ByteIOContext *pb, int tag, unsigned int size)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
235
{
Michael Niedermayer's avatar
Michael Niedermayer committed
236 237 238 239 240
    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
241 242 243 244 245
}

static int mov_write_esds_tag(ByteIOContext *pb, MOVTrack* track) // Basic
{
    offset_t pos = url_ftell(pb);
246
    int decoderSpecificInfoLen = track->vosLen ? descrLength(track->vosLen):0;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
247 248 249 250 251 252 253 254 255 256 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

    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);
}

294 295
static int mov_write_wave_tag(ByteIOContext *pb, MOVTrack* track)
{
296
    offset_t pos = url_ftell(pb);
297 298 299 300 301 302

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

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

305
    if (track->enc->codec_id == CODEC_ID_AAC) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
306 307 308 309
        /* useless atom needed by mplayer, ipod, not needed by quicktime */
        put_be32(pb, 12); /* size */
        put_tag(pb, "mp4a");
        put_be32(pb, 0);
310 311 312 313
        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
314
    } else if (track->enc->codec_id == CODEC_ID_AMR_NB) {
315
        mov_write_amr_tag(pb, track);
316
    }
317 318 319 320 321 322 323

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

    return updateSize (pb, pos);
}

324
static const CodecTag codec_movaudio_tags[] = {
325 326 327 328 329 330 331
    { 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') },
332
    { CODEC_ID_AMR_WB, MKTAG('s', 'a', 'w', 'b') },
333 334
    { CODEC_ID_PCM_S16BE, MKTAG('t', 'w', 'o', 's') },
    { CODEC_ID_PCM_S16LE, MKTAG('s', 'o', 'w', 't') },
335 336 337 338
    { 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') },
339
    { CODEC_ID_MP3, MKTAG('.', 'm', 'p', '3') },
340
    { CODEC_ID_NONE, 0 },
341 342
};

343
static int mov_write_audio_tag(ByteIOContext *pb, MOVTrack* track)
344
{
345
    offset_t pos = url_ftell(pb);
346 347 348 349 350 351 352
    int vbr=  track->mode == MODE_MOV &&
        (track->enc->codec_id == CODEC_ID_AAC ||
         track->enc->codec_id == CODEC_ID_MP3 ||
         track->enc->codec_id == CODEC_ID_AMR_NB);
    int version = vbr ||
        track->enc->codec_id == CODEC_ID_PCM_S32LE ||
        track->enc->codec_id == CODEC_ID_PCM_S24LE;
353

354
    put_be32(pb, 0); /* size */
Baptiste Coudurier's avatar
Baptiste Coudurier committed
355
    put_le32(pb, track->tag); // store it byteswapped
356 357 358
    put_be32(pb, 0); /* Reserved */
    put_be16(pb, 0); /* Reserved */
    put_be16(pb, 1); /* Data-reference index, XXX  == 1 */
359

360
    /* SoundDescription */
361
    put_be16(pb, version); /* Version */
362
    put_be16(pb, 0); /* Revision level */
363 364
    put_be32(pb, 0); /* Reserved */

365
    put_be16(pb, track->mode == MODE_MOV ? track->enc->channels : 2); /* Number of channels */
366 367
    /* FIXME 8 bit for 'raw ' in mov */
    put_be16(pb, 16); /* Reserved */
368

Baptiste Coudurier's avatar
Baptiste Coudurier committed
369
    put_be16(pb, vbr ? 0xfffe : 0); /* compression ID */
370
    put_be16(pb, 0); /* packet size (= 0) */
371 372 373
    put_be16(pb, track->timescale); /* Time scale */
    put_be16(pb, 0); /* Reserved */

374
    if(version == 1) { /* SoundDescription V1 extended info */
375 376
        if (vbr)
            track->sampleSize = 0;
377
        put_be32(pb, track->enc->frame_size); /* Samples per packet */
378
        put_be32(pb, track->sampleSize / track->enc->channels); /* Bytes per packet */
379
        put_be32(pb, track->sampleSize); /* Bytes per frame */
Baptiste Coudurier's avatar
Baptiste Coudurier committed
380
        put_be32(pb, 2); /* Bytes per sample */
381 382
    }

383 384 385 386 387
    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))
388
        mov_write_wave_tag(pb, track);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
389 390 391
    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)
392
        mov_write_amr_tag(pb, track);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
393

394
    return updateSize (pb, pos);
395 396
}

397
static int mov_write_d263_tag(ByteIOContext *pb)
398 399 400 401
{
    put_be32(pb, 0xf); /* size */
    put_tag(pb, "d263");
    put_tag(pb, "FFMP");
402 403 404 405
    put_byte(pb, 0); /* decoder version */
    /* FIXME use AVCodecContext level/profile, when encoder will set values */
    put_byte(pb, 0xa); /* level */
    put_byte(pb, 0); /* profile */
406 407 408
    return 0xf;
}

409 410
/* TODO: No idea about these values */
static int mov_write_svq3_tag(ByteIOContext *pb)
411
{
412 413 414 415 416 417
    put_be32(pb, 0x15);
    put_tag(pb, "SMI ");
    put_tag(pb, "SEQH");
    put_be32(pb, 0x5);
    put_be32(pb, 0xe2c0211d);
    put_be32(pb, 0xc0000000);
418
    put_byte(pb, 0);
419
    return 0x15;
420 421
}

422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437
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
438
                    return p-1;
439
                if( p[2] == 0 && p[3] == 1 )
Baptiste Coudurier's avatar
Baptiste Coudurier committed
440
                    return p;
441 442 443
            }
            if( p[3] == 0 ) {
                if( p[2] == 0 && p[4] == 1 )
Baptiste Coudurier's avatar
Baptiste Coudurier committed
444
                    return p+1;
445
                if( p[4] == 0 && p[5] == 1 )
Baptiste Coudurier's avatar
Baptiste Coudurier committed
446
                    return p+2;
447 448 449 450 451 452 453 454 455 456 457 458 459 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
            }
        }
    }

    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);
513 514 515 516 517 518 519 520

            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) */

521 522 523 524 525 526 527 528 529 530 531 532
            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);
}

533
static const CodecTag codec_movvideo_tags[] = {
534 535 536
    { CODEC_ID_SVQ1, MKTAG('S', 'V', 'Q', '1') },
    { CODEC_ID_SVQ3, MKTAG('S', 'V', 'Q', '3') },
    { CODEC_ID_MPEG4, MKTAG('m', 'p', '4', 'v') },
537
    { CODEC_ID_H263, MKTAG('h', '2', '6', '3') },
538
    { CODEC_ID_H263, MKTAG('s', '2', '6', '3') },
539
    { CODEC_ID_H264, MKTAG('a', 'v', 'c', '1') },
540 541 542 543 544 545 546
    /* 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 },
547 548
};

Baptiste Coudurier's avatar
Baptiste Coudurier committed
549
static int mov_find_video_codec_tag(AVFormatContext *s, MOVTrack *track)
550
{
Baptiste Coudurier's avatar
Baptiste Coudurier committed
551
    int tag = track->enc->codec_tag;
552 553 554 555 556 557 558 559 560 561 562 563
    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
564 565
                else
                    tag = MKTAG('d', 'v', 'p', 'p');
566
            }
567 568 569 570 571
        } else if (track->enc->codec_id == CODEC_ID_H263) {
            if (track->mode == MODE_MOV)
                tag = MKTAG('h', '2', '6', '3');
            else
                tag = MKTAG('s', '2', '6', '3');
572 573 574 575 576
        } 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
577
    if (!tag) {
578
        tag = codec_get_tag(codec_bmp_tags, track->enc->codec_id);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600
        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");
        }
    }
601 602 603 604
    assert(tag);
    return tag;
}

605
static int mov_write_video_tag(ByteIOContext *pb, MOVTrack* track)
606
{
607
    offset_t pos = url_ftell(pb);
608
    char compressor_name[32];
609

610
    put_be32(pb, 0); /* size */
Baptiste Coudurier's avatar
Baptiste Coudurier committed
611
    put_le32(pb, track->tag); // store it byteswapped
612 613 614 615
    put_be32(pb, 0); /* Reserved */
    put_be16(pb, 0); /* Reserved */
    put_be16(pb, 1); /* Data-reference index */

616 617
    put_be16(pb, 0); /* Codec stream version */
    put_be16(pb, 0); /* Codec stream revision (=0) */
618
    if (track->mode == MODE_MOV) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
619 620 621 622 623 624 625 626
        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 */
        }
627 628 629 630 631
    } else {
        put_be32(pb, 0); /* Reserved */
        put_be32(pb, 0); /* Reserved */
        put_be32(pb, 0); /* Reserved */
    }
632 633
    put_be16(pb, track->enc->width); /* Video width */
    put_be16(pb, track->enc->height); /* Video height */
634 635
    put_be32(pb, 0x00480000); /* Horizontal resolution 72dpi */
    put_be32(pb, 0x00480000); /* Vertical resolution 72dpi */
636 637
    put_be32(pb, 0); /* Data size (= 0) */
    put_be16(pb, 1); /* Frame count (= 1) */
638

639
    memset(compressor_name,0,32);
640 641
    /* FIXME not sure, ISO 14496-1 draft where it shall be set to 0 */
    if (track->mode == MODE_MOV && 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
<