oggenc.c 23.9 KB
Newer Older
Baptiste Coudurier's avatar
Baptiste Coudurier committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
/*
 * Ogg muxer
 * Copyright (c) 2007 Baptiste Coudurier <baptiste dot coudurier at free dot fr>
 *
 * This file is part of FFmpeg.
 *
 * FFmpeg 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.1 of the License, or (at your option) any later version.
 *
 * FFmpeg 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 FFmpeg; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */

22 23
#include <stdint.h>

24
#include "libavutil/crc.h"
25
#include "libavutil/mathematics.h"
26
#include "libavutil/opt.h"
27
#include "libavutil/random_seed.h"
28 29
#include "libavcodec/xiph.h"
#include "libavcodec/bytestream.h"
30
#include "libavcodec/flac.h"
Baptiste Coudurier's avatar
Baptiste Coudurier committed
31
#include "avformat.h"
32
#include "avio_internal.h"
33
#include "internal.h"
34
#include "vorbiscomment.h"
Baptiste Coudurier's avatar
Baptiste Coudurier committed
35

36 37
#define MAX_PAGE_SIZE 65025

38
typedef struct OGGPage {
39
    int64_t start_granule;
40 41 42 43 44 45 46 47 48
    int64_t granule;
    int stream_index;
    uint8_t flags;
    uint8_t segments_count;
    uint8_t segments[255];
    uint8_t data[MAX_PAGE_SIZE];
    uint16_t size;
} OGGPage;

49
typedef struct OGGStreamContext {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
50 51 52 53 54 55 56
    unsigned page_counter;
    uint8_t *header[3];
    int header_len[3];
    /** for theora granule */
    int kfgshift;
    int64_t last_kf_pts;
    int vrev;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
57
    int eos;
58 59
    unsigned page_count; ///< number of page buffered
    OGGPage page; ///< current page
60
    unsigned serial_num; ///< serial number
61
    int64_t last_granule; ///< last packet granule
Baptiste Coudurier's avatar
Baptiste Coudurier committed
62 63
} OGGStreamContext;

64 65 66 67 68
typedef struct OGGPageList {
    OGGPage page;
    struct OGGPageList *next;
} OGGPageList;

69
typedef struct OGGContext {
70
    const AVClass *class;
71
    OGGPageList *page_list;
72
    int pref_size; ///< preferred page size (0 => fill all segments)
73
    int64_t pref_duration;      ///< preferred page duration (0 => fill all segments)
74
    int serial_offset;
75 76
} OGGContext;

77 78
#define OFFSET(x) offsetof(OGGContext, x)
#define PARAM AV_OPT_FLAG_ENCODING_PARAM
79 80

static const AVOption options[] = {
81 82
    { "serial_offset", "serial number offset",
        OFFSET(serial_offset), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, PARAM },
83
    { "oggpagesize", "Set preferred Ogg page size.",
84
      OFFSET(pref_size), AV_OPT_TYPE_INT, {.i64 = 0}, 0, MAX_PAGE_SIZE, PARAM},
85
    { "pagesize", "preferred page size in bytes (deprecated)",
86
        OFFSET(pref_size), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, MAX_PAGE_SIZE, PARAM },
87
    { "page_duration", "preferred page duration, in microseconds",
88
        OFFSET(pref_duration), AV_OPT_TYPE_INT64, { .i64 = 1000000 }, 0, INT64_MAX, PARAM },
89 90 91
    { NULL },
};

92
#define OGG_CLASS(flavor, name)\
93
static const AVClass flavor ## _muxer_class = {\
94
    .class_name = #name " muxer",\
95 96 97
    .item_name  = av_default_item_name,\
    .option     = options,\
    .version    = LIBAVUTIL_VERSION_INT,\
98 99
};

100
static void ogg_update_checksum(AVFormatContext *s, AVIOContext *pb, int64_t crc_offset)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
101
{
102
    int64_t pos = avio_tell(pb);
103
    uint32_t checksum = ffio_get_checksum(pb);
104
    avio_seek(pb, crc_offset, SEEK_SET);
105
    avio_wb32(pb, checksum);
106
    avio_seek(pb, pos, SEEK_SET);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
107 108
}

109
static int ogg_write_page(AVFormatContext *s, OGGPage *page, int extra_flags)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
110
{
111
    OGGStreamContext *oggstream = s->streams[page->stream_index]->priv_data;
112
    AVIOContext *pb;
113
    int64_t crc_offset;
114 115
    int ret, size;
    uint8_t *buf;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
116

117
    ret = avio_open_dyn_buf(&pb);
118 119
    if (ret < 0)
        return ret;
120
    ffio_init_checksum(pb, ff_crc04C11DB7_update, 0);
121
    ffio_wfourcc(pb, "OggS");
122 123 124 125 126
    avio_w8(pb, 0);
    avio_w8(pb, page->flags | extra_flags);
    avio_wl64(pb, page->granule);
    avio_wl32(pb, oggstream->serial_num);
    avio_wl32(pb, oggstream->page_counter++);
127
    crc_offset = avio_tell(pb);
128 129 130 131
    avio_wl32(pb, 0); // crc
    avio_w8(pb, page->segments_count);
    avio_write(pb, page->segments, page->segments_count);
    avio_write(pb, page->data, page->size);
132 133

    ogg_update_checksum(s, pb, crc_offset);
134
    avio_flush(pb);
135

136
    size = avio_close_dyn_buf(pb, &buf);
137 138 139
    if (size < 0)
        return size;

140
    avio_write(s->pb, buf, size);
141
    avio_flush(s->pb);
142
    av_free(buf);
143
    oggstream->page_count--;
144
    return 0;
145 146
}

147 148 149 150 151
static int ogg_key_granule(OGGStreamContext *oggstream, int64_t granule)
{
    return oggstream->kfgshift && !(granule & ((1<<oggstream->kfgshift)-1));
}

152
static int64_t ogg_granule_to_timestamp(OGGStreamContext *oggstream, int64_t granule)
153 154
{
    if (oggstream->kfgshift)
155 156
        return (granule>>oggstream->kfgshift) +
            (granule & ((1<<oggstream->kfgshift)-1));
157
    else
158
        return granule;
159 160 161 162 163 164 165 166 167 168 169
}

static int ogg_compare_granule(AVFormatContext *s, OGGPage *next, OGGPage *page)
{
    AVStream *st2 = s->streams[next->stream_index];
    AVStream *st  = s->streams[page->stream_index];
    int64_t next_granule, cur_granule;

    if (next->granule == -1 || page->granule == -1)
        return 0;

170
    next_granule = av_rescale_q(ogg_granule_to_timestamp(st2->priv_data, next->granule),
171
                                st2->time_base, AV_TIME_BASE_Q);
172
    cur_granule  = av_rescale_q(ogg_granule_to_timestamp(st->priv_data, page->granule),
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195
                                st ->time_base, AV_TIME_BASE_Q);
    return next_granule > cur_granule;
}

static int ogg_reset_cur_page(OGGStreamContext *oggstream)
{
    oggstream->page.granule = -1;
    oggstream->page.flags = 0;
    oggstream->page.segments_count = 0;
    oggstream->page.size = 0;
    return 0;
}

static int ogg_buffer_page(AVFormatContext *s, OGGStreamContext *oggstream)
{
    OGGContext *ogg = s->priv_data;
    OGGPageList **p = &ogg->page_list;
    OGGPageList *l = av_mallocz(sizeof(*l));

    if (!l)
        return AVERROR(ENOMEM);
    l->page = oggstream->page;

196
    oggstream->page.start_granule = oggstream->page.granule;
197 198 199 200 201 202 203 204 205 206 207 208 209 210 211
    oggstream->page_count++;
    ogg_reset_cur_page(oggstream);

    while (*p) {
        if (ogg_compare_granule(s, &(*p)->page, &l->page))
            break;
        p = &(*p)->next;
    }
    l->next = *p;
    *p = l;

    return 0;
}

static int ogg_buffer_data(AVFormatContext *s, AVStream *st,
212 213
                           uint8_t *data, unsigned size, int64_t granule,
                           int header)
214 215
{
    OGGStreamContext *oggstream = st->priv_data;
216
    OGGContext *ogg = s->priv_data;
217 218
    int total_segments = size / 255 + 1;
    uint8_t *p = data;
219 220 221
    int i, segments, len, flush = 0;

    // Handles VFR by flushing page because this frame needs to have a timestamp
222 223 224
    // For theora, keyframes also need to have a timestamp to correctly mark
    // them as such, otherwise seeking will not work correctly at the very
    // least with old libogg versions.
225
    // Do not try to flush header packets though, that will create broken files.
226
    if (st->codec->codec_id == AV_CODEC_ID_THEORA && !header &&
227 228 229
        (ogg_granule_to_timestamp(oggstream, granule) >
         ogg_granule_to_timestamp(oggstream, oggstream->last_granule) + 1 ||
         ogg_key_granule(oggstream, granule))) {
230 231 232 233
        if (oggstream->page.granule != -1)
            ogg_buffer_page(s, oggstream);
        flush = 1;
    }
234

235 236 237 238 239 240
    // avoid a continued page
    if (!header && oggstream->page.size > 0 &&
        MAX_PAGE_SIZE - oggstream->page.size < size) {
        ogg_buffer_page(s, oggstream);
    }

241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262
    for (i = 0; i < total_segments; ) {
        OGGPage *page = &oggstream->page;

        segments = FFMIN(total_segments - i, 255 - page->segments_count);

        if (i && !page->segments_count)
            page->flags |= 1; // continued packet

        memset(page->segments+page->segments_count, 255, segments - 1);
        page->segments_count += segments - 1;

        len = FFMIN(size, segments*255);
        page->segments[page->segments_count++] = len - (segments-1)*255;
        memcpy(page->data+page->size, p, len);
        p += len;
        size -= len;
        i += segments;
        page->size += len;

        if (i == total_segments)
            page->granule = granule;

263
        {
264 265 266 267 268 269 270
            AVStream *st = s->streams[page->stream_index];

            int64_t start = av_rescale_q(page->start_granule, st->time_base,
                                         AV_TIME_BASE_Q);
            int64_t next  = av_rescale_q(page->granule, st->time_base,
                                         AV_TIME_BASE_Q);

271
            if (page->segments_count == 255) {
272
                ogg_buffer_page(s, oggstream);
273 274 275 276 277
            } else if (!header) {
                if ((ogg->pref_size     > 0 && page->size   >= ogg->pref_size) ||
                    (ogg->pref_duration > 0 && next - start >= ogg->pref_duration)) {
                    ogg_buffer_page(s, oggstream);
                }
278
            }
279 280
        }
    }
281 282 283 284

    if (flush && oggstream->page.granule != -1)
        ogg_buffer_page(s, oggstream);

285
    return 0;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
286 287
}

288
static uint8_t *ogg_write_vorbiscomment(int64_t offset, int bitexact,
289
                                        int *header_len, AVDictionary **m, int framing_bit)
290 291
{
    const char *vendor = bitexact ? "ffmpeg" : LIBAVFORMAT_IDENT;
292
    int64_t size;
293 294
    uint8_t *p, *p0;

295 296
    ff_metadata_conv(m, ff_vorbiscomment_metadata_conv, NULL);

297
    size = offset + ff_vorbiscomment_length(*m, vendor) + framing_bit;
298 299
    if (size > INT_MAX)
        return NULL;
300 301 302 303 304 305
    p = av_mallocz(size);
    if (!p)
        return NULL;
    p0 = p;

    p += offset;
306
    ff_vorbiscomment_write(&p, m, vendor);
307 308
    if (framing_bit)
        bytestream_put_byte(&p, 1);
309 310 311 312 313

    *header_len = size;
    return p0;
}

314
static int ogg_build_flac_headers(AVCodecContext *avctx,
315
                                  OGGStreamContext *oggstream, int bitexact,
316
                                  AVDictionary **m)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
317 318
{
    uint8_t *p;
319

320 321
    if (avctx->extradata_size < FLAC_STREAMINFO_SIZE)
        return AVERROR(EINVAL);
322 323

    // first packet: STREAMINFO
324 325
    oggstream->header_len[0] = 51;
    oggstream->header[0] = av_mallocz(51); // per ogg flac specs
Baptiste Coudurier's avatar
Baptiste Coudurier committed
326
    p = oggstream->header[0];
327
    if (!p)
328
        return AVERROR(ENOMEM);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
329 330 331 332 333 334 335 336
    bytestream_put_byte(&p, 0x7F);
    bytestream_put_buffer(&p, "FLAC", 4);
    bytestream_put_byte(&p, 1); // major version
    bytestream_put_byte(&p, 0); // minor version
    bytestream_put_be16(&p, 1); // headers packets without this one
    bytestream_put_buffer(&p, "fLaC", 4);
    bytestream_put_byte(&p, 0x00); // streaminfo
    bytestream_put_be24(&p, 34);
337
    bytestream_put_buffer(&p, avctx->extradata, FLAC_STREAMINFO_SIZE);
338 339

    // second packet: VorbisComment
340
    p = ogg_write_vorbiscomment(4, bitexact, &oggstream->header_len[1], m, 0);
341
    if (!p)
342
        return AVERROR(ENOMEM);
343
    oggstream->header[1] = p;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
344 345
    bytestream_put_byte(&p, 0x84); // last metadata block and vorbis comment
    bytestream_put_be24(&p, oggstream->header_len[1] - 4);
346 347 348 349 350 351 352

    return 0;
}

#define SPEEX_HEADER_SIZE 80

static int ogg_build_speex_headers(AVCodecContext *avctx,
353
                                   OGGStreamContext *oggstream, int bitexact,
354
                                   AVDictionary **m)
355 356 357 358
{
    uint8_t *p;

    if (avctx->extradata_size < SPEEX_HEADER_SIZE)
359
        return AVERROR_INVALIDDATA;
360 361 362 363

    // first packet: Speex header
    p = av_mallocz(SPEEX_HEADER_SIZE);
    if (!p)
364
        return AVERROR(ENOMEM);
365 366 367 368 369 370
    oggstream->header[0] = p;
    oggstream->header_len[0] = SPEEX_HEADER_SIZE;
    bytestream_put_buffer(&p, avctx->extradata, SPEEX_HEADER_SIZE);
    AV_WL32(&oggstream->header[0][68], 0);  // set extra_headers to 0

    // second packet: VorbisComment
371
    p = ogg_write_vorbiscomment(0, bitexact, &oggstream->header_len[1], m, 0);
372
    if (!p)
373
        return AVERROR(ENOMEM);
374
    oggstream->header[1] = p;
375

Baptiste Coudurier's avatar
Baptiste Coudurier committed
376 377 378
    return 0;
}

379 380 381 382 383 384 385 386 387
#define OPUS_HEADER_SIZE 19

static int ogg_build_opus_headers(AVCodecContext *avctx,
                                  OGGStreamContext *oggstream, int bitexact,
                                  AVDictionary **m)
{
    uint8_t *p;

    if (avctx->extradata_size < OPUS_HEADER_SIZE)
388
        return AVERROR_INVALIDDATA;
389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407

    /* first packet: Opus header */
    p = av_mallocz(avctx->extradata_size);
    if (!p)
        return AVERROR(ENOMEM);
    oggstream->header[0] = p;
    oggstream->header_len[0] = avctx->extradata_size;
    bytestream_put_buffer(&p, avctx->extradata, avctx->extradata_size);

    /* second packet: VorbisComment */
    p = ogg_write_vorbiscomment(8, bitexact, &oggstream->header_len[1], m, 0);
    if (!p)
        return AVERROR(ENOMEM);
    oggstream->header[1] = p;
    bytestream_put_buffer(&p, "OpusTags", 8);

    return 0;
}

408 409 410 411 412 413 414 415 416 417 418 419 420 421
static void ogg_write_pages(AVFormatContext *s, int flush)
{
    OGGContext *ogg = s->priv_data;
    OGGPageList *next, *p;

    if (!ogg->page_list)
        return;

    for (p = ogg->page_list; p; ) {
        OGGStreamContext *oggstream =
            s->streams[p->page.stream_index]->priv_data;
        if (oggstream->page_count < 2 && !flush)
            break;
        ogg_write_page(s, &p->page,
422
                       flush == 1 && oggstream->page_count == 1 ? 4 : 0); // eos
423 424 425 426 427 428 429
        next = p->next;
        av_freep(&p);
        p = next;
    }
    ogg->page_list = p;
}

Baptiste Coudurier's avatar
Baptiste Coudurier committed
430 431
static int ogg_write_header(AVFormatContext *s)
{
432
    OGGContext *ogg = s->priv_data;
433
    OGGStreamContext *oggstream = NULL;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
434
    int i, j;
435

436 437 438
    if (ogg->pref_size)
        av_log(s, AV_LOG_WARNING, "The pagesize option is deprecated\n");

Baptiste Coudurier's avatar
Baptiste Coudurier committed
439 440
    for (i = 0; i < s->nb_streams; i++) {
        AVStream *st = s->streams[i];
441
        unsigned serial_num = i + ogg->serial_offset;
442

443
        if (st->codec->codec_type == AVMEDIA_TYPE_AUDIO) {
444 445 446 447 448
            if (st->codec->codec_id == AV_CODEC_ID_OPUS)
                /* Opus requires a fixed 48kHz clock */
                avpriv_set_pts_info(st, 64, 1, 48000);
            else
                avpriv_set_pts_info(st, 64, 1, st->codec->sample_rate);
449
        }
450

451 452 453
        if (st->codec->codec_id != AV_CODEC_ID_VORBIS &&
            st->codec->codec_id != AV_CODEC_ID_THEORA &&
            st->codec->codec_id != AV_CODEC_ID_SPEEX  &&
454 455
            st->codec->codec_id != AV_CODEC_ID_FLAC   &&
            st->codec->codec_id != AV_CODEC_ID_OPUS) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
456
            av_log(s, AV_LOG_ERROR, "Unsupported codec id in stream %d\n", i);
457
            return AVERROR(EINVAL);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
458 459 460 461
        }

        if (!st->codec->extradata || !st->codec->extradata_size) {
            av_log(s, AV_LOG_ERROR, "No extradata present\n");
462
            return AVERROR_INVALIDDATA;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
463 464
        }
        oggstream = av_mallocz(sizeof(*oggstream));
465 466 467
        if (!oggstream)
            return AVERROR(ENOMEM);

468
        oggstream->page.stream_index = i;
469

470
        if (!(s->flags & AVFMT_FLAG_BITEXACT))
471 472 473 474 475 476 477 478 479 480
            do {
                serial_num = av_get_random_seed();
                for (j = 0; j < i; j++) {
                    OGGStreamContext *sc = s->streams[j]->priv_data;
                    if (serial_num == sc->serial_num)
                        break;
                }
            } while (j < i);
        oggstream->serial_num = serial_num;

481 482
        av_dict_copy(&st->metadata, s->metadata, AV_DICT_DONT_OVERWRITE);

Baptiste Coudurier's avatar
Baptiste Coudurier committed
483
        st->priv_data = oggstream;
484
        if (st->codec->codec_id == AV_CODEC_ID_FLAC) {
485
            int err = ogg_build_flac_headers(st->codec, oggstream,
486
                                             s->flags & AVFMT_FLAG_BITEXACT,
487
                                             &st->metadata);
488
            if (err) {
489
                av_log(s, AV_LOG_ERROR, "Error writing FLAC headers\n");
Baptiste Coudurier's avatar
Baptiste Coudurier committed
490
                av_freep(&st->priv_data);
491
                return err;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
492
            }
493
        } else if (st->codec->codec_id == AV_CODEC_ID_SPEEX) {
494
            int err = ogg_build_speex_headers(st->codec, oggstream,
495
                                              s->flags & AVFMT_FLAG_BITEXACT,
496
                                              &st->metadata);
497 498 499 500 501
            if (err) {
                av_log(s, AV_LOG_ERROR, "Error writing Speex headers\n");
                av_freep(&st->priv_data);
                return err;
            }
502 503
        } else if (st->codec->codec_id == AV_CODEC_ID_OPUS) {
            int err = ogg_build_opus_headers(st->codec, oggstream,
504
                                             s->flags & AVFMT_FLAG_BITEXACT,
505
                                             &st->metadata);
506 507 508 509 510
            if (err) {
                av_log(s, AV_LOG_ERROR, "Error writing Opus headers\n");
                av_freep(&st->priv_data);
                return err;
            }
Baptiste Coudurier's avatar
Baptiste Coudurier committed
511
        } else {
512
            uint8_t *p;
513 514 515
            const char *cstr = st->codec->codec_id == AV_CODEC_ID_VORBIS ? "vorbis" : "theora";
            int header_type = st->codec->codec_id == AV_CODEC_ID_VORBIS ? 3 : 0x81;
            int framing_bit = st->codec->codec_id == AV_CODEC_ID_VORBIS ? 1 : 0;
516

517
            if (avpriv_split_xiph_headers(st->codec->extradata, st->codec->extradata_size,
518
                                      st->codec->codec_id == AV_CODEC_ID_VORBIS ? 30 : 42,
519
                                      (const uint8_t**)oggstream->header, oggstream->header_len) < 0) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
520 521
                av_log(s, AV_LOG_ERROR, "Extradata corrupted\n");
                av_freep(&st->priv_data);
522
                return AVERROR_INVALIDDATA;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
523
            }
524

525
            p = ogg_write_vorbiscomment(7, s->flags & AVFMT_FLAG_BITEXACT,
526
                                        &oggstream->header_len[1], &st->metadata,
527
                                        framing_bit);
528
            oggstream->header[1] = p;
529 530 531 532 533 534
            if (!p)
                return AVERROR(ENOMEM);

            bytestream_put_byte(&p, header_type);
            bytestream_put_buffer(&p, cstr, 6);

535
            if (st->codec->codec_id == AV_CODEC_ID_THEORA) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
536 537 538 539 540 541 542 543 544
                /** KFGSHIFT is the width of the less significant section of the granule position
                    The less significant section is the frame count since the last keyframe */
                oggstream->kfgshift = ((oggstream->header[0][40]&3)<<3)|(oggstream->header[0][41]>>5);
                oggstream->vrev = oggstream->header[0][9];
                av_log(s, AV_LOG_DEBUG, "theora kfgshift %d, vrev %d\n",
                       oggstream->kfgshift, oggstream->vrev);
            }
        }
    }
545 546 547 548

    for (j = 0; j < s->nb_streams; j++) {
        OGGStreamContext *oggstream = s->streams[j]->priv_data;
        ogg_buffer_data(s, s->streams[j], oggstream->header[0],
549
                        oggstream->header_len[0], 0, 1);
550 551 552 553 554 555 556
        oggstream->page.flags |= 2; // bos
        ogg_buffer_page(s, oggstream);
    }
    for (j = 0; j < s->nb_streams; j++) {
        AVStream *st = s->streams[j];
        OGGStreamContext *oggstream = st->priv_data;
        for (i = 1; i < 3; i++) {
557
            if (oggstream->header_len[i])
558
                ogg_buffer_data(s, st, oggstream->header[i],
559
                                oggstream->header_len[i], 0, 1);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
560
        }
561
        ogg_buffer_page(s, oggstream);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
562
    }
563 564 565

    oggstream->page.start_granule = AV_NOPTS_VALUE;

566
    ogg_write_pages(s, 2);
567

Baptiste Coudurier's avatar
Baptiste Coudurier committed
568 569 570
    return 0;
}

571
static int ogg_write_packet_internal(AVFormatContext *s, AVPacket *pkt)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
572 573 574
{
    AVStream *st = s->streams[pkt->stream_index];
    OGGStreamContext *oggstream = st->priv_data;
575
    int ret;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
576 577
    int64_t granule;

578
    if (st->codec->codec_id == AV_CODEC_ID_THEORA) {
Baptiste Coudurier's avatar
Baptiste Coudurier committed
579 580
        int64_t pts = oggstream->vrev < 1 ? pkt->pts : pkt->pts + pkt->duration;
        int pframe_count;
581
        if (pkt->flags & AV_PKT_FLAG_KEY)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
582 583 584 585 586 587 588 589
            oggstream->last_kf_pts = pts;
        pframe_count = pts - oggstream->last_kf_pts;
        // prevent frame count from overflow if key frame flag is not set
        if (pframe_count >= (1<<oggstream->kfgshift)) {
            oggstream->last_kf_pts += pframe_count;
            pframe_count = 0;
        }
        granule = (oggstream->last_kf_pts<<oggstream->kfgshift) | pframe_count;
590
    } else if (st->codec->codec_id == AV_CODEC_ID_OPUS)
591 592 593 594
        granule = pkt->pts + pkt->duration +
                  av_rescale_q(st->codec->initial_padding,
                               (AVRational){ 1, st->codec->sample_rate },
                               st->time_base);
595
    else
Baptiste Coudurier's avatar
Baptiste Coudurier committed
596 597
        granule = pkt->pts + pkt->duration;

598 599 600
    if (oggstream->page.start_granule == AV_NOPTS_VALUE)
        oggstream->page.start_granule = pkt->pts;

601
    ret = ogg_buffer_data(s, st, pkt->data, pkt->size, granule, 0);
602 603
    if (ret < 0)
        return ret;
Baptiste Coudurier's avatar
Baptiste Coudurier committed
604

605
    ogg_write_pages(s, 0);
606

607 608
    oggstream->last_granule = granule;

609
    return 0;
610 611
}

612 613 614 615 616 617 618 619 620 621 622 623 624 625
static int ogg_write_packet(AVFormatContext *s, AVPacket *pkt)
{
    int i;

    if (pkt)
        return ogg_write_packet_internal(s, pkt);

    for (i = 0; i < s->nb_streams; i++) {
        OGGStreamContext *oggstream = s->streams[i]->priv_data;
        if (oggstream->page.segments_count)
            ogg_buffer_page(s, oggstream);
    }

    ogg_write_pages(s, 2);
626
    return 1;
627 628
}

629
static int ogg_write_trailer(AVFormatContext *s)
Baptiste Coudurier's avatar
Baptiste Coudurier committed
630
{
631
    int i;
632

633 634 635 636 637 638 639
    /* flush current page if needed */
    for (i = 0; i < s->nb_streams; i++) {
        OGGStreamContext *oggstream = s->streams[i]->priv_data;

        if (oggstream->page.size > 0)
            ogg_buffer_page(s, oggstream);
    }
640

641
    ogg_write_pages(s, 1);
642

Baptiste Coudurier's avatar
Baptiste Coudurier committed
643 644 645
    for (i = 0; i < s->nb_streams; i++) {
        AVStream *st = s->streams[i];
        OGGStreamContext *oggstream = st->priv_data;
646
        if (st->codec->codec_id == AV_CODEC_ID_FLAC ||
647 648
            st->codec->codec_id == AV_CODEC_ID_SPEEX ||
            st->codec->codec_id == AV_CODEC_ID_OPUS) {
649
            av_freep(&oggstream->header[0]);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
650
        }
651
        av_freep(&oggstream->header[1]);
Baptiste Coudurier's avatar
Baptiste Coudurier committed
652 653 654 655 656
        av_freep(&st->priv_data);
    }
    return 0;
}

657
#if CONFIG_OGG_MUXER
658
OGG_CLASS(ogg, Ogg)
659
AVOutputFormat ff_ogg_muxer = {
660 661 662
    .name              = "ogg",
    .long_name         = NULL_IF_CONFIG_SMALL("Ogg"),
    .mime_type         = "application/ogg",
663
    .extensions        = "ogg,ogv"
664
#if !CONFIG_SPX_MUXER
665 666 667 668 669 670
                         ",spx"
#endif
#if !CONFIG_OPUS_MUXER
                         ",opus"
#endif
                         ,
671
    .priv_data_size    = sizeof(OGGContext),
672 673
    .audio_codec       = CONFIG_LIBVORBIS_ENCODER ?
                         AV_CODEC_ID_VORBIS : AV_CODEC_ID_FLAC,
674
    .video_codec       = AV_CODEC_ID_THEORA,
675 676 677
    .write_header      = ogg_write_header,
    .write_packet      = ogg_write_packet,
    .write_trailer     = ogg_write_trailer,
678
    .flags             = AVFMT_TS_NEGATIVE | AVFMT_ALLOW_FLUSH,
679
    .priv_class        = &ogg_muxer_class,
Baptiste Coudurier's avatar
Baptiste Coudurier committed
680
};
681 682
#endif

Carl Eugen Hoyos's avatar
Carl Eugen Hoyos committed
683
#if CONFIG_OGA_MUXER
684
OGG_CLASS(oga, Ogg audio)
Carl Eugen Hoyos's avatar
Carl Eugen Hoyos committed
685 686
AVOutputFormat ff_oga_muxer = {
    .name              = "oga",
687
    .long_name         = NULL_IF_CONFIG_SMALL("Ogg Audio"),
Carl Eugen Hoyos's avatar
Carl Eugen Hoyos committed
688 689 690
    .mime_type         = "audio/ogg",
    .extensions        = "oga",
    .priv_data_size    = sizeof(OGGContext),
691 692
    .audio_codec       = CONFIG_LIBVORBIS_ENCODER ?
                         AV_CODEC_ID_VORBIS : AV_CODEC_ID_FLAC,
Carl Eugen Hoyos's avatar
Carl Eugen Hoyos committed
693 694 695
    .write_header      = ogg_write_header,
    .write_packet      = ogg_write_packet,
    .write_trailer     = ogg_write_trailer,
696
    .flags             = AVFMT_TS_NEGATIVE | AVFMT_ALLOW_FLUSH,
Carl Eugen Hoyos's avatar
Carl Eugen Hoyos committed
697 698 699 700
    .priv_class        = &oga_muxer_class,
};
#endif

701
#if CONFIG_SPX_MUXER
702
OGG_CLASS(spx, Ogg Speex)
703 704 705
AVOutputFormat ff_spx_muxer = {
    .name              = "spx",
    .long_name         = NULL_IF_CONFIG_SMALL("Ogg Speex"),
706 707 708 709 710 711 712
    .mime_type         = "audio/ogg",
    .extensions        = "spx",
    .priv_data_size    = sizeof(OGGContext),
    .audio_codec       = AV_CODEC_ID_SPEEX,
    .write_header      = ogg_write_header,
    .write_packet      = ogg_write_packet,
    .write_trailer     = ogg_write_trailer,
713
    .flags             = AVFMT_TS_NEGATIVE | AVFMT_ALLOW_FLUSH,
714
    .priv_class        = &spx_muxer_class,
715 716 717 718
};
#endif

#if CONFIG_OPUS_MUXER
719
OGG_CLASS(opus, Ogg Opus)
720 721
AVOutputFormat ff_opus_muxer = {
    .name              = "opus",
722
    .long_name         = NULL_IF_CONFIG_SMALL("Ogg Opus"),
723 724 725 726 727 728 729
    .mime_type         = "audio/ogg",
    .extensions        = "opus",
    .priv_data_size    = sizeof(OGGContext),
    .audio_codec       = AV_CODEC_ID_OPUS,
    .write_header      = ogg_write_header,
    .write_packet      = ogg_write_packet,
    .write_trailer     = ogg_write_trailer,
730
    .flags             = AVFMT_TS_NEGATIVE | AVFMT_ALLOW_FLUSH,
731 732 733
    .priv_class        = &opus_muxer_class,
};
#endif