avienc.c 14.2 KB
Newer Older
F
Fabrice Bellard 已提交
1 2
/*
 * AVI encoder.
F
Fabrice Bellard 已提交
3
 * Copyright (c) 2000 Fabrice Bellard.
F
Fabrice Bellard 已提交
4
 *
F
Fabrice Bellard 已提交
5 6 7 8
 * 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.
F
Fabrice Bellard 已提交
9
 *
F
Fabrice Bellard 已提交
10
 * This library is distributed in the hope that it will be useful,
F
Fabrice Bellard 已提交
11
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
F
Fabrice Bellard 已提交
12 13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
F
Fabrice Bellard 已提交
14
 *
F
Fabrice Bellard 已提交
15 16 17
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
F
Fabrice Bellard 已提交
18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
 */
#include "avformat.h"
#include "avi.h"

/*
 * TODO: 
 *  - fill all fields if non streamed (nb_frames for example)
 */

typedef struct AVIIndex {
    unsigned char tag[4];
    unsigned int flags, pos, len;
    struct AVIIndex *next;
} AVIIndex;

typedef struct {
34 35
    offset_t movi_list, frames_hdr_all, frames_hdr_strm[MAX_STREAMS];
    int audio_strm_length[MAX_STREAMS];
F
Fabrice Bellard 已提交
36 37 38
    AVIIndex *first, *last;
} AVIContext;

Z
Zdenek Kabelac 已提交
39
offset_t start_tag(ByteIOContext *pb, const char *tag)
F
Fabrice Bellard 已提交
40 41 42 43 44 45 46 47 48 49 50 51
{
    put_tag(pb, tag);
    put_le32(pb, 0);
    return url_ftell(pb);
}

void end_tag(ByteIOContext *pb, offset_t start)
{
    offset_t pos;

    pos = url_ftell(pb);
    url_fseek(pb, start - 4, SEEK_SET);
52
    put_le32(pb, (uint32_t)(pos - start));
F
Fabrice Bellard 已提交
53 54 55 56 57
    url_fseek(pb, pos, SEEK_SET);
}

/* Note: when encoding, the first matching tag is used, so order is
   important if multiple tags possible for a given codec. */
Z
Zdenek Kabelac 已提交
58
const CodecTag codec_bmp_tags[] = {
J
Juanjo 已提交
59 60
    { CODEC_ID_H263, MKTAG('H', '2', '6', '3') },
    { CODEC_ID_H263P, MKTAG('H', '2', '6', '3') },
F
Fabrice Bellard 已提交
61 62
    { CODEC_ID_H263I, MKTAG('I', '2', '6', '3') }, /* intel h263 */
    { CODEC_ID_MJPEG, MKTAG('M', 'J', 'P', 'G') },
63 64 65 66 67 68
    { CODEC_ID_MPEG4, MKTAG('D', 'I', 'V', 'X'), .invalid_asf = 1 },
    { CODEC_ID_MPEG4, MKTAG('d', 'i', 'v', 'x'), .invalid_asf = 1 },
    { CODEC_ID_MPEG4, MKTAG('D', 'X', '5', '0'), .invalid_asf = 1 },
    { CODEC_ID_MPEG4, MKTAG('X', 'V', 'I', 'D'), .invalid_asf = 1 },
    { CODEC_ID_MPEG4, MKTAG('x', 'v', 'i', 'd'), .invalid_asf = 1 },
    { CODEC_ID_MPEG4, MKTAG('m', 'p', '4', 's'), .invalid_asf = 1 },
M
Michael Niedermayer 已提交
69 70 71
    { CODEC_ID_MPEG4, MKTAG('M', 'P', '4', 'S') },
    { CODEC_ID_MPEG4, MKTAG('M', '4', 'S', '2') },
    { CODEC_ID_MPEG4, MKTAG('m', '4', 's', '2') },
F
Fabrice Bellard 已提交
72
    { CODEC_ID_MPEG4, MKTAG(0x04, 0, 0, 0) }, /* some broken avi use this */
73
    { CODEC_ID_MSMPEG4V3, MKTAG('D', 'I', 'V', '3'), .invalid_asf = 1 }, /* default signature when using MSMPEG4 */
M
Michael Niedermayer 已提交
74
    { CODEC_ID_MSMPEG4V3, MKTAG('d', 'i', 'v', '3'), .invalid_asf = 1 },
75 76
    { CODEC_ID_MSMPEG4V3, MKTAG('M', 'P', '4', '3') }, 
    { CODEC_ID_MSMPEG4V2, MKTAG('M', 'P', '4', '2') }, 
M
wmv1  
Michael Niedermayer 已提交
77 78
    { CODEC_ID_MSMPEG4V1, MKTAG('M', 'P', 'G', '4') }, 
    { CODEC_ID_WMV1, MKTAG('W', 'M', 'V', '1') }, 
79
    { CODEC_ID_WMV2, MKTAG('W', 'M', 'V', '2') }, 
F
Fabrice Bellard 已提交
80 81
    { CODEC_ID_DVVIDEO, MKTAG('d', 'v', 's', 'l') }, 
    { CODEC_ID_DVVIDEO, MKTAG('d', 'v', 's', 'd') }, 
82
    { CODEC_ID_DVVIDEO, MKTAG('D', 'V', 'S', 'D') }, 
F
Fabrice Bellard 已提交
83
    { CODEC_ID_DVVIDEO, MKTAG('d', 'v', 'h', 'd') }, 
F
Fabrice Bellard 已提交
84 85 86
    { CODEC_ID_MPEG1VIDEO, MKTAG('m', 'p', 'g', '1') }, 
    { CODEC_ID_MPEG1VIDEO, MKTAG('m', 'p', 'g', '2') }, 
    { CODEC_ID_MPEG1VIDEO, MKTAG('P', 'I', 'M', '1') }, 
A
Alex Beregszaszi 已提交
87
    { CODEC_ID_MJPEG, MKTAG('M', 'J', 'P', 'G') },
M
Michael Niedermayer 已提交
88 89
    { CODEC_ID_HUFFYUV, MKTAG('H', 'F', 'Y', 'U') },
    { CODEC_ID_HUFFYUV, MKTAG('h', 'f', 'y', 'u') },
M
Mike Melanson 已提交
90 91
    { CODEC_ID_CYUV, MKTAG('C', 'Y', 'U', 'V') },
    { CODEC_ID_CYUV, MKTAG('c', 'y', 'u', 'v') },
92
    { CODEC_ID_RAWVIDEO, MKTAG('Y', '4', '2', '2') },
F
Fabrice Bellard 已提交
93 94 95
    { 0, 0 },
};

Z
Zdenek Kabelac 已提交
96
unsigned int codec_get_tag(const CodecTag *tags, int id)
F
Fabrice Bellard 已提交
97 98 99 100 101 102 103 104 105
{
    while (tags->id != 0) {
        if (tags->id == id)
            return tags->tag;
        tags++;
    }
    return 0;
}

106 107 108 109 110 111 112 113 114 115
static unsigned int codec_get_asf_tag(const CodecTag *tags, int id)
{
    while (tags->id != 0) {
        if (!tags->invalid_asf && tags->id == id)
            return tags->tag;
        tags++;
    }
    return 0;
}

Z
Zdenek Kabelac 已提交
116
int codec_get_id(const CodecTag *tags, unsigned int tag)
F
Fabrice Bellard 已提交
117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
{
    while (tags->id != 0) {
        if (tags->tag == tag)
            return tags->id;
        tags++;
    }
    return 0;
}

unsigned int codec_get_bmp_tag(int id)
{
    return codec_get_tag(codec_bmp_tags, id);
}

/* BITMAPINFOHEADER header */
Z
Zdenek Kabelac 已提交
132
void put_bmp_header(ByteIOContext *pb, AVCodecContext *enc, const CodecTag *tags, int for_asf)
F
Fabrice Bellard 已提交
133
{
M
Michael Niedermayer 已提交
134
    put_le32(pb, 40 + enc->extradata_size); /* size */
F
Fabrice Bellard 已提交
135 136 137
    put_le32(pb, enc->width);
    put_le32(pb, enc->height);
    put_le16(pb, 1); /* planes */
M
Michael Niedermayer 已提交
138 139
    
    put_le16(pb, enc->bits_per_sample ? enc->bits_per_sample : 24); /* depth */
F
Fabrice Bellard 已提交
140
    /* compression type */
141
    put_le32(pb, for_asf ? codec_get_asf_tag(tags, enc->codec_id) : codec_get_tag(tags, enc->codec_id));
F
Fabrice Bellard 已提交
142 143 144 145 146
    put_le32(pb, enc->width * enc->height * 3);
    put_le32(pb, 0);
    put_le32(pb, 0);
    put_le32(pb, 0);
    put_le32(pb, 0);
M
Michael Niedermayer 已提交
147 148 149 150 151
    
    put_buffer(pb, enc->extradata, enc->extradata_size);

    if (enc->extradata_size & 1)
        put_byte(pb, 0);
F
Fabrice Bellard 已提交
152 153
}

Z
Zdenek Kabelac 已提交
154
static void parse_specific_params(AVCodecContext *stream, int *au_byterate, int *au_ssize, int *au_scale)
155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170
{
    switch(stream->codec_id) {
    case CODEC_ID_PCM_S16LE:
       *au_scale = *au_ssize = 2*stream->channels;
       *au_byterate = *au_ssize * stream->sample_rate;
        break;
    case CODEC_ID_PCM_U8:
    case CODEC_ID_PCM_ALAW:
    case CODEC_ID_PCM_MULAW:
        *au_scale = *au_ssize = stream->channels;
        *au_byterate = *au_ssize * stream->sample_rate;
        break;
    case CODEC_ID_MP2:
        *au_ssize = 1;
        *au_scale = 1;
        *au_byterate = stream->bit_rate / 8;
171 172 173 174
    case CODEC_ID_MP3LAME:
        *au_ssize = 1;
        *au_scale = 1;
        *au_byterate = stream->bit_rate / 8;    
175 176 177 178 179 180 181 182
    default:
        *au_ssize = 1;
        *au_scale = 1; 
        *au_byterate = stream->bit_rate / 8;
        break;
    }
}

F
Fabrice Bellard 已提交
183 184
static int avi_write_header(AVFormatContext *s)
{
F
Fabrice Bellard 已提交
185
    AVIContext *avi = s->priv_data;
F
Fabrice Bellard 已提交
186
    ByteIOContext *pb = &s->pb;
187
    int bitrate, n, i, nb_frames, au_byterate, au_ssize, au_scale;
F
Fabrice Bellard 已提交
188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211
    AVCodecContext *stream, *video_enc;
    offset_t list1, list2, strh, strf;

    put_tag(pb, "RIFF");
    put_le32(pb, 0); /* file length */
    put_tag(pb, "AVI ");

    /* header list */
    list1 = start_tag(pb, "LIST");
    put_tag(pb, "hdrl");

    /* avi header */
    put_tag(pb, "avih");
    put_le32(pb, 14 * 4);
    bitrate = 0;

    video_enc = NULL;
    for(n=0;n<s->nb_streams;n++) {
        stream = &s->streams[n]->codec;
        bitrate += stream->bit_rate;
        if (stream->codec_type == CODEC_TYPE_VIDEO)
            video_enc = stream;
    }
    
212 213
/*	allowing audio-only AVI file 
    
F
Fabrice Bellard 已提交
214
    if (!video_enc) {
215
        av_free(avi);
F
Fabrice Bellard 已提交
216 217
        return -1;
    }
218
*/    
F
Fabrice Bellard 已提交
219 220
    nb_frames = 0;

221
    if(video_enc){
222
        put_le32(pb, (uint32_t)(int64_t_C(1000000) * video_enc->frame_rate_base / video_enc->frame_rate));
223 224 225
    } else {
	put_le32(pb, 0);
    }
F
Fabrice Bellard 已提交
226 227 228
    put_le32(pb, bitrate / 8); /* XXX: not quite exact */
    put_le32(pb, 0); /* padding */
    put_le32(pb, AVIF_TRUSTCKTYPE | AVIF_HASINDEX | AVIF_ISINTERLEAVED); /* flags */
229
    avi->frames_hdr_all = url_ftell(pb); /* remember this offset to fill later */
F
Fabrice Bellard 已提交
230 231 232 233
    put_le32(pb, nb_frames); /* nb frames, filled later */
    put_le32(pb, 0); /* initial frame */
    put_le32(pb, s->nb_streams); /* nb streams */
    put_le32(pb, 1024 * 1024); /* suggested buffer size */
234
    if(video_enc){    
F
Fabrice Bellard 已提交
235 236
    put_le32(pb, video_enc->width);
    put_le32(pb, video_enc->height);
237 238 239 240
    } else {
	put_le32(pb, 0);
	put_le32(pb, 0);
    }	
F
Fabrice Bellard 已提交
241 242 243 244 245 246 247 248 249 250 251 252
    put_le32(pb, 0); /* reserved */
    put_le32(pb, 0); /* reserved */
    put_le32(pb, 0); /* reserved */
    put_le32(pb, 0); /* reserved */
    
    /* stream list */
    for(i=0;i<n;i++) {
        list2 = start_tag(pb, "LIST");
        put_tag(pb, "strl");
    
        stream = &s->streams[i]->codec;

253 254 255 256 257
        /* FourCC should really be set by the codec itself */
        if (! stream->codec_tag) {
            stream->codec_tag = codec_get_bmp_tag(stream->codec_id);
        }

F
Fabrice Bellard 已提交
258 259 260 261 262
        /* stream generic header */
        strh = start_tag(pb, "strh");
        switch(stream->codec_type) {
        case CODEC_TYPE_VIDEO:
            put_tag(pb, "vids");
263
            put_le32(pb, stream->codec_tag);
F
Fabrice Bellard 已提交
264 265 266 267
            put_le32(pb, 0); /* flags */
            put_le16(pb, 0); /* priority */
            put_le16(pb, 0); /* language */
            put_le32(pb, 0); /* initial frame */
268
            
269 270
            put_le32(pb, stream->frame_rate_base); /* scale */
            put_le32(pb, stream->frame_rate); /* rate */
271

F
Fabrice Bellard 已提交
272
            put_le32(pb, 0); /* start */
273
            avi->frames_hdr_strm[i] = url_ftell(pb); /* remember this offset to fill later */
F
Fabrice Bellard 已提交
274 275
            put_le32(pb, nb_frames); /* length, XXX: fill later */
            put_le32(pb, 1024 * 1024); /* suggested buffer size */
276
            put_le32(pb, -1); /* quality */
F
Fabrice Bellard 已提交
277 278 279 280 281 282 283 284
            put_le32(pb, stream->width * stream->height * 3); /* sample size */
            put_le16(pb, 0);
            put_le16(pb, 0);
            put_le16(pb, stream->width);
            put_le16(pb, stream->height);
            break;
        case CODEC_TYPE_AUDIO:
            put_tag(pb, "auds");
285
            put_le32(pb, 1); /* tag */
F
Fabrice Bellard 已提交
286 287 288 289
            put_le32(pb, 0); /* flags */
            put_le16(pb, 0); /* priority */
            put_le16(pb, 0); /* language */
            put_le32(pb, 0); /* initial frame */
290 291 292
            parse_specific_params(stream, &au_byterate, &au_ssize, &au_scale);
            put_le32(pb, au_scale); /* scale */
            put_le32(pb, au_byterate); /* rate */
F
Fabrice Bellard 已提交
293
            put_le32(pb, 0); /* start */
294
            avi->frames_hdr_strm[i] = url_ftell(pb); /* remember this offset to fill later */
F
Fabrice Bellard 已提交
295 296 297
            put_le32(pb, 0); /* length, XXX: filled later */
            put_le32(pb, 12 * 1024); /* suggested buffer size */
            put_le32(pb, -1); /* quality */
298
            put_le32(pb, au_ssize); /* sample size */
F
Fabrice Bellard 已提交
299 300 301
            put_le32(pb, 0);
            put_le32(pb, 0);
            break;
302
        default:
303
            av_abort();
F
Fabrice Bellard 已提交
304 305 306 307 308 309
        }
        end_tag(pb, strh);

        strf = start_tag(pb, "strf");
        switch(stream->codec_type) {
        case CODEC_TYPE_VIDEO:
310
	    put_bmp_header(pb, stream, codec_bmp_tags, 0);
F
Fabrice Bellard 已提交
311 312
            break;
        case CODEC_TYPE_AUDIO:
313
            if (put_wav_header(pb, stream) < 0) {
314
                av_free(avi);
315 316
                return -1;
            }
F
Fabrice Bellard 已提交
317
            break;
318
        default:
319
            av_abort();
F
Fabrice Bellard 已提交
320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337
        }
        end_tag(pb, strf);
        end_tag(pb, list2);
    }

    end_tag(pb, list1);
    
    avi->movi_list = start_tag(pb, "LIST");
    avi->first = NULL;
    avi->last = NULL;
    put_tag(pb, "movi");

    put_flush_packet(pb);

    return 0;
}

static int avi_write_packet(AVFormatContext *s, int stream_index,
338
                            uint8_t *buf, int size, int force_pts)
F
Fabrice Bellard 已提交
339 340 341 342 343 344 345 346 347 348 349 350 351 352 353
{
    AVIContext *avi = s->priv_data;
    ByteIOContext *pb = &s->pb;
    AVIIndex *idx;
    unsigned char tag[5];
    unsigned int flags;
    AVCodecContext *enc;
    
    enc = &s->streams[stream_index]->codec;

    tag[0] = '0';
    tag[1] = '0' + stream_index;
    if (enc->codec_type == CODEC_TYPE_VIDEO) {
        tag[2] = 'd';
        tag[3] = 'c';
M
Michael Niedermayer 已提交
354
        flags = enc->coded_frame->key_frame ? 0x10 : 0x00;
F
Fabrice Bellard 已提交
355 356 357 358 359
    } else {
        tag[2] = 'w';
        tag[3] = 'b';
        flags = 0x10;
    }
360 361
    if (enc->codec_type == CODEC_TYPE_AUDIO) 
       avi->audio_strm_length[stream_index] += size;
F
Fabrice Bellard 已提交
362 363

    if (!url_is_streamed(&s->pb)) {
364
        idx = av_malloc(sizeof(AVIIndex));
F
Fabrice Bellard 已提交
365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391
        memcpy(idx->tag, tag, 4);
        idx->flags = flags;
        idx->pos = url_ftell(pb) - avi->movi_list;
        idx->len = size;
        idx->next = NULL;
        if (!avi->last)
            avi->first = idx;
        else
            avi->last->next = idx;
        avi->last = idx;
    }
    
    put_buffer(pb, tag, 4);
    put_le32(pb, size);
    put_buffer(pb, buf, size);
    if (size & 1)
        put_byte(pb, 0);

    put_flush_packet(pb);
    return 0;
}

static int avi_write_trailer(AVFormatContext *s)
{
    ByteIOContext *pb = &s->pb;
    AVIContext *avi = s->priv_data;
    offset_t file_size, idx_chunk;
392 393
    int n, nb_frames, au_byterate, au_ssize, au_scale;
    AVCodecContext *stream;
F
Fabrice Bellard 已提交
394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412
    AVIIndex *idx;

    if (!url_is_streamed(&s->pb)) {
        end_tag(pb, avi->movi_list);

        idx_chunk = start_tag(pb, "idx1");
        idx = avi->first;
        while (idx != NULL) {
            put_buffer(pb, idx->tag, 4);
            put_le32(pb, idx->flags);
            put_le32(pb, idx->pos);
            put_le32(pb, idx->len);
            idx = idx->next;
        }
        end_tag(pb, idx_chunk);
        
        /* update file size */
        file_size = url_ftell(pb);
        url_fseek(pb, 4, SEEK_SET);
413
        put_le32(pb, (uint32_t)(file_size - 8));
414 415 416 417 418 419 420 421 422 423 424 425

        /* Fill in frame/sample counters */
        nb_frames = 0;
        for(n=0;n<s->nb_streams;n++) {
            if (avi->frames_hdr_strm[n] != 0) {
                stream = &s->streams[n]->codec;
                url_fseek(pb, avi->frames_hdr_strm[n], SEEK_SET);
                if (stream->codec_type == CODEC_TYPE_VIDEO) {
                    put_le32(pb, stream->frame_number); 
                    if (nb_frames < stream->frame_number)
                        nb_frames = stream->frame_number;
                } else {
426
                    if (stream->codec_id == CODEC_ID_MP2 || stream->codec_id == CODEC_ID_MP3LAME) {
427 428 429 430 431 432 433 434 435 436 437 438 439
                        put_le32(pb, stream->frame_number);
                        nb_frames += stream->frame_number;
                    } else {
                        parse_specific_params(stream, &au_byterate, &au_ssize, &au_scale);
                        put_le32(pb, avi->audio_strm_length[n] / au_ssize);
                    }
                }
            }
       }
       if (avi->frames_hdr_all != 0) {
           url_fseek(pb, avi->frames_hdr_all, SEEK_SET);
           put_le32(pb, nb_frames); 
       }
F
Fabrice Bellard 已提交
440 441 442 443 444 445
        url_fseek(pb, file_size, SEEK_SET);
    }
    put_flush_packet(pb);
    return 0;
}

F
Fabrice Bellard 已提交
446
static AVOutputFormat avi_oformat = {
F
Fabrice Bellard 已提交
447 448 449 450
    "avi",
    "avi format",
    "video/x-msvideo",
    "avi",
F
Fabrice Bellard 已提交
451
    sizeof(AVIContext),
F
Fabrice Bellard 已提交
452
    CODEC_ID_MP2,
453
    CODEC_ID_MSMPEG4V3,
F
Fabrice Bellard 已提交
454 455 456 457
    avi_write_header,
    avi_write_packet,
    avi_write_trailer,
};
F
Fabrice Bellard 已提交
458 459 460 461 462 463

int avienc_init(void)
{
    av_register_output_format(&avi_oformat);
    return 0;
}