Merge remote-tracking branch 'qatar/master'
authorMichael Niedermayer <michaelni@gmx.at>
Thu, 5 Sep 2013 11:34:37 +0000 (13:34 +0200)
committerMichael Niedermayer <michaelni@gmx.at>
Thu, 5 Sep 2013 11:34:37 +0000 (13:34 +0200)
* qatar/master:
  matroskaenc: Allow chapters to be written in trailer

Conflicts:
libavformat/matroskaenc.c

Merged-by: Michael Niedermayer <michaelni@gmx.at>
1  2 
libavformat/avformat.h
libavformat/matroskaenc.c

diff --combined libavformat/avformat.h
@@@ -1,20 -1,20 +1,20 @@@
  /*
   * copyright (c) 2001 Fabrice Bellard
   *
 - * This file is part of Libav.
 + * This file is part of FFmpeg.
   *
 - * Libav is free software; you can redistribute it and/or
 + * 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.
   *
 - * Libav is distributed in the hope that it will be useful,
 + * 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 Libav; if not, write to the Free Software
 + * License along with FFmpeg; if not, write to the Free Software
   * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
   */
  
@@@ -50,7 -50,7 +50,7 @@@
   *
   * Main lavf structure used for both muxing and demuxing is AVFormatContext,
   * which exports all information about the file being read or written. As with
 - * most Libav structures, its size is not part of public ABI, so it cannot be
 + * most Libavformat structures, its size is not part of public ABI, so it cannot be
   * allocated on stack or directly with av_malloc(). To create an
   * AVFormatContext, use avformat_alloc_context() (some functions, like
   * avformat_open_input() might do that for you).
@@@ -220,7 -220,7 +220,7 @@@ struct AVFormatContext
   *
   * Metadata is exported or set as pairs of key/value strings in the 'metadata'
   * fields of the AVFormatContext, AVStream, AVChapter and AVProgram structs
 - * using the @ref lavu_dict "AVDictionary" API. Like all strings in Libav,
 + * using the @ref lavu_dict "AVDictionary" API. Like all strings in FFmpeg,
   * metadata is assumed to be UTF-8 encoded Unicode. Note that metadata
   * exported by demuxers isn't checked to be valid UTF-8 in most cases.
   *
@@@ -337,7 -337,6 +337,7 @@@ typedef struct AVProbeData 
      int buf_size;       /**< Size of buf except extra allocated bytes */
  } AVProbeData;
  
 +#define AVPROBE_SCORE_RETRY (AVPROBE_SCORE_MAX/4)
  #define AVPROBE_SCORE_EXTENSION  50 ///< score for file extension
  #define AVPROBE_SCORE_MAX       100 ///< maximum score
  
  #define AVFMT_NOGENSEARCH   0x4000 /**< Format does not allow to fall back on generic search */
  #define AVFMT_NO_BYTE_SEEK  0x8000 /**< Format does not allow seeking by bytes */
  #define AVFMT_ALLOW_FLUSH  0x10000 /**< Format allows flushing. If not set, the muxer will not receive a NULL packet in the write_packet function. */
 -#define AVFMT_TS_NONSTRICT 0x20000 /**< Format does not require strictly
 +#if LIBAVFORMAT_VERSION_MAJOR <= 54
 +#define AVFMT_TS_NONSTRICT 0x8020000 //we try to be compatible to the ABIs of ffmpeg and major forks
 +#else
 +#define AVFMT_TS_NONSTRICT 0x20000
 +#endif
 +                                   /**< Format does not require strictly
                                          increasing timestamps, but they must
                                          still be monotonic */
  #define AVFMT_TS_NEGATIVE  0x40000 /**< Format allows muxing negative
                                          timestamps. If not set the timestamp
                                          will be shifted in av_write_frame and
                                          av_interleaved_write_frame so they
 -                                        start from 0. */
 +                                        start from 0.
 +                                        The user or muxer can override this through
 +                                        AVFormatContext.avoid_negative_ts
 +                                        */
 +
 +#define AVFMT_SEEK_TO_PTS   0x4000000 /**< Seeking is based on PTS */
  
  /**
   * @addtogroup lavf_encoding
@@@ -447,12 -436,8 +447,12 @@@ typedef struct AVOutputFormat 
       *
       * @return 1 if the codec is supported, 0 if it is not.
       *         A negative number if unknown.
 +     *         MKTAG('A', 'P', 'I', 'C') if the codec is only supported as AV_DISPOSITION_ATTACHED_PIC
       */
      int (*query_codec)(enum AVCodecID id, int std_compliance);
 +
 +    void (*get_output_timestamp)(struct AVFormatContext *s, int stream,
 +                                 int64_t *dts, int64_t *wall);
  } AVOutputFormat;
  /**
   * @}
@@@ -479,7 -464,7 +479,7 @@@ typedef struct AVInputFormat 
      /**
       * Can use flags: AVFMT_NOFILE, AVFMT_NEEDNUMBER, AVFMT_SHOW_IDS,
       * AVFMT_GENERIC_INDEX, AVFMT_TS_DISCONT, AVFMT_NOBINSEARCH,
 -     * AVFMT_NOGENSEARCH, AVFMT_NO_BYTE_SEEK.
 +     * AVFMT_NOGENSEARCH, AVFMT_NO_BYTE_SEEK, AVFMT_SEEK_TO_PTS.
       */
      int flags;
  
@@@ -592,19 -577,11 +592,19 @@@ enum AVStreamParseType 
      AVSTREAM_PARSE_HEADERS,    /**< Only parse headers, do not repack. */
      AVSTREAM_PARSE_TIMESTAMPS, /**< full parsing and interpolation of timestamps for frames not starting on a packet boundary */
      AVSTREAM_PARSE_FULL_ONCE,  /**< full parsing and repack of the first frame only, only implemented for H.264 currently */
 +    AVSTREAM_PARSE_FULL_RAW=MKTAG(0,'R','A','W'),       /**< full parsing and repack with timestamp and position generation by parser for raw
 +                                                             this assumes that each packet in the file contains no demuxer level headers and
 +                                                             just codec level data, otherwise position generation would fail */
  };
  
  typedef struct AVIndexEntry {
      int64_t pos;
 -    int64_t timestamp;
 +    int64_t timestamp;        /**<
 +                               * Timestamp in AVStream.time_base units, preferably the time from which on correctly decoded frames are available
 +                               * when seeking to this entry. That means preferable PTS on keyframe based formats.
 +                               * But demuxers can choose to store a different timestamp, if it is more convenient for the implementation or nothing better
 +                               * is known
 +                               */
  #define AVINDEX_KEYFRAME 0x0001
      int flags:2;
      int size:30; //Yeah, trying to keep the size of this small to reduce memory requirements (it is 24 vs. 32 bytes due to possible 8-byte alignment).
  #define AV_DISPOSITION_ATTACHED_PIC      0x0400
  
  /**
 + * To specify text track kind (different from subtitles default).
 + */
 +#define AV_DISPOSITION_CAPTIONS     0x10000
 +#define AV_DISPOSITION_DESCRIPTIONS 0x20000
 +#define AV_DISPOSITION_METADATA     0x40000
 +
 +/**
 + * Options for behavior on timestamp wrap detection.
 + */
 +#define AV_PTS_WRAP_IGNORE      0   ///< ignore the wrap
 +#define AV_PTS_WRAP_ADD_OFFSET  1   ///< add the format specific offset on wrap detection
 +#define AV_PTS_WRAP_SUB_OFFSET  -1  ///< subtract the format specific offset on wrap detection
 +
 +/**
   * Stream structure.
   * New fields can be added to the end with minor version bumps.
   * Removal, reordering and changes to existing fields require a major
@@@ -695,12 -658,10 +695,12 @@@ typedef struct AVStream 
      AVRational time_base;
  
      /**
 -     * Decoding: pts of the first frame of the stream, in stream time base.
 +     * Decoding: pts of the first frame of the stream in presentation order, in stream time base.
       * Only set this if you are absolutely 100% sure that the value you set
       * it to really is the pts of the first frame.
       * This may be undefined (AV_NOPTS_VALUE).
 +     * @note The ASF header does NOT contain a correct start_time the ASF
 +     * demuxer must NOT set this.
       */
      int64_t start_time;
  
      /**
       * Stream information used internally by av_find_stream_info()
       */
 -#define MAX_STD_TIMEBASES (60*12+5)
 +#define MAX_STD_TIMEBASES (60*12+6)
      struct {
 -        int nb_decoded_frames;
 +        int64_t last_dts;
 +        int64_t duration_gcd;
 +        int duration_count;
 +        double (*duration_error)[2][MAX_STD_TIMEBASES];
 +        int64_t codec_info_duration;
 +        int64_t codec_info_duration_fields;
          int found_decoder;
  
 +        int64_t last_duration;
 +
          /**
           * Those are used for average framerate estimation.
           */
                                      support seeking natively. */
      int nb_index_entries;
      unsigned int index_entries_allocated_size;
 +
 +    /**
 +     * Real base framerate of the stream.
 +     * This is the lowest framerate with which all timestamps can be
 +     * represented accurately (it is the least common multiple of all
 +     * framerates in the stream). Note, this value is just a guess!
 +     * For example, if the time base is 1/90000 and all frames have either
 +     * approximately 3600 or 1800 timer ticks, then r_frame_rate will be 50/1.
 +     *
 +     * Code outside avformat should access this field using:
 +     * av_stream_get/set_r_frame_rate(stream)
 +     */
 +    AVRational r_frame_rate;
 +
 +    /**
 +     * Stream Identifier
 +     * This is the MPEG-TS stream identifier +1
 +     * 0 means unknown
 +     */
 +    int stream_identifier;
 +
 +    int64_t interleaver_chunk_size;
 +    int64_t interleaver_chunk_duration;
 +
 +    /**
 +     * stream probing state
 +     * -1   -> probing finished
 +     *  0   -> no probing requested
 +     * rest -> perform probing with request_probe being the minimum score to accept.
 +     * NOT PART OF PUBLIC API
 +     */
 +    int request_probe;
 +    /**
 +     * Indicates that everything up to the next keyframe
 +     * should be discarded.
 +     */
 +    int skip_to_keyframe;
 +
 +    /**
 +     * Number of samples to skip at the start of the frame decoded from the next packet.
 +     */
 +    int skip_samples;
 +
 +    /**
 +     * Number of internally decoded frames, used internally in libavformat, do not access
 +     * its lifetime differs from info which is why it is not in that structure.
 +     */
 +    int nb_decoded_frames;
 +
 +    /**
 +     * Timestamp offset added to timestamps before muxing
 +     * NOT PART OF PUBLIC API
 +     */
 +    int64_t mux_ts_offset;
 +
 +    /**
 +     * Internal data to check for wrapping of the time stamp
 +     */
 +    int64_t pts_wrap_reference;
 +
 +    /**
 +     * Options for behavior, when a wrap is detected.
 +     *
 +     * Defined by AV_PTS_WRAP_ values.
 +     *
 +     * If correction is enabled, there are two possibilities:
 +     * If the first time stamp is near the wrap point, the wrap offset
 +     * will be subtracted, which will create negative time stamps.
 +     * Otherwise the offset will be added.
 +     */
 +    int pts_wrap_behavior;
 +
  } AVStream;
  
 +AVRational av_stream_get_r_frame_rate(const AVStream *s);
 +void       av_stream_set_r_frame_rate(AVStream *s, AVRational r);
 +
  #define AV_PROGRAM_RUNNING 1
  
  /**
@@@ -908,23 -787,6 +908,23 @@@ typedef struct AVProgram 
      unsigned int   *stream_index;
      unsigned int   nb_stream_indexes;
      AVDictionary *metadata;
 +
 +    int program_num;
 +    int pmt_pid;
 +    int pcr_pid;
 +
 +    /*****************************************************************
 +     * All fields below this line are not part of the public API. They
 +     * may not be used outside of libavformat and can be changed and
 +     * removed at will.
 +     * New public fields should be added right above.
 +     *****************************************************************
 +     */
 +    int64_t start_time;
 +    int64_t end_time;
 +
 +    int64_t pts_wrap_reference;    ///< reference dts for wrap detection
 +    int pts_wrap_behavior;         ///< behavior on wrap detection
  } AVProgram;
  
  #define AVFMTCTX_NOHEADER      0x0001 /**< signal that no header is present
@@@ -937,17 -799,6 +937,17 @@@ typedef struct AVChapter 
      AVDictionary *metadata;
  } AVChapter;
  
 +
 +/**
 + * The duration of a video can be estimated through various ways, and this enum can be used
 + * to know how the duration was estimated.
 + */
 +enum AVDurationEstimationMethod {
 +    AVFMT_DURATION_FROM_PTS,    ///< Duration accurately estimated from PTSes
 +    AVFMT_DURATION_FROM_STREAM, ///< Duration estimated from a stream with a known duration
 +    AVFMT_DURATION_FROM_BITRATE ///< Duration estimated from bitrate (less accurate)
 +};
 +
  /**
   * Format I/O context.
   * New fields can be added to the end with minor version bumps.
@@@ -1026,7 -877,7 +1026,7 @@@ typedef struct AVFormatContext 
      /**
       * Decoding: total stream bitrate in bit/s, 0 if not
       * available. Never set it directly if the file_size and the
 -     * duration are known as Libav can compute it automatically.
 +     * duration are known as FFmpeg can compute it automatically.
       */
      int bit_rate;
  
  #define AVFMT_FLAG_NOBUFFER     0x0040 ///< Do not buffer frames when possible
  #define AVFMT_FLAG_CUSTOM_IO    0x0080 ///< The caller has supplied a custom AVIOContext, don't avio_close() it.
  #define AVFMT_FLAG_DISCARD_CORRUPT  0x0100 ///< Discard frames marked corrupted
 +#define AVFMT_FLAG_MP4A_LATM    0x8000 ///< Enable RTP MP4A-LATM payload
 +#define AVFMT_FLAG_SORT_DTS    0x10000 ///< try to interleave outputted packets by dts (using this flag can slow demuxing down)
 +#define AVFMT_FLAG_PRIV_OPT    0x20000 ///< Enable use of private options by delaying codec open (this could be made default once all code is converted)
 +#define AVFMT_FLAG_KEEP_SIDE_DATA 0x40000 ///< Don't merge side data but keep it separate.
  
      /**
       * decoding: size of data to probe; encoding: unused.
       */
      unsigned int max_picture_buffer;
  
+     /**
+      * Number of chapters in AVChapter array.
+      * When muxing, chapters are normally written in the file header,
+      * so nb_chapters should normally be initialized before write_header
+      * is called. Some muxers (e.g. mov and mkv) can also write chapters
+      * in the trailer.  To write chapters in the trailer, nb_chapters
+      * must be zero when write_header is called and non-zero when
+      * write_trailer is called.
+      * muxing  : set by user
+      * demuxing: set by libavformat
+      */
      unsigned int nb_chapters;
      AVChapter **chapters;
  
       */
      int debug;
  #define FF_FDEBUG_TS        0x0001
 +
 +    /**
 +     * Transport stream id.
 +     * This will be moved into demuxer private options. Thus no API/ABI compatibility
 +     */
 +    int ts_id;
 +
 +    /**
 +     * Audio preload in microseconds.
 +     * Note, not all formats support this and unpredictable things may happen if it is used when not supported.
 +     * - encoding: Set by user via AVOptions (NO direct access)
 +     * - decoding: unused
 +     */
 +    int audio_preload;
 +
 +    /**
 +     * Max chunk time in microseconds.
 +     * Note, not all formats support this and unpredictable things may happen if it is used when not supported.
 +     * - encoding: Set by user via AVOptions (NO direct access)
 +     * - decoding: unused
 +     */
 +    int max_chunk_duration;
 +
 +    /**
 +     * Max chunk size in bytes
 +     * Note, not all formats support this and unpredictable things may happen if it is used when not supported.
 +     * - encoding: Set by user via AVOptions (NO direct access)
 +     * - decoding: unused
 +     */
 +    int max_chunk_size;
 +
 +    /**
 +     * forces the use of wallclock timestamps as pts/dts of packets
 +     * This has undefined results in the presence of B frames.
 +     * - encoding: unused
 +     * - decoding: Set by user via AVOptions (NO direct access)
 +     */
 +    int use_wallclock_as_timestamps;
 +
 +    /**
 +     * Avoid negative timestamps during muxing.
 +     *  0 -> allow negative timestamps
 +     *  1 -> avoid negative timestamps
 +     * -1 -> choose automatically (default)
 +     * Note, this only works when interleave_packet_per_dts is in use.
 +     * - encoding: Set by user via AVOptions (NO direct access)
 +     * - decoding: unused
 +     */
 +    int avoid_negative_ts;
 +
 +    /**
 +     * avio flags, used to force AVIO_FLAG_DIRECT.
 +     * - encoding: unused
 +     * - decoding: Set by user via AVOptions (NO direct access)
 +     */
 +    int avio_flags;
 +
 +    /**
 +     * The duration field can be estimated through various ways, and this field can be used
 +     * to know how the duration was estimated.
 +     * - encoding: unused
 +     * - decoding: Read by user via AVOptions (NO direct access)
 +     */
 +    enum AVDurationEstimationMethod duration_estimation_method;
 +
 +    /**
 +     * Skip initial bytes when opening stream
 +     * - encoding: unused
 +     * - decoding: Set by user via AVOptions (NO direct access)
 +     */
 +    unsigned int skip_initial_bytes;
 +
 +    /**
 +     * Correct single timestamp overflows
 +     * - encoding: unused
 +     * - decoding: Set by user via AVOPtions (NO direct access)
 +     */
 +    unsigned int correct_ts_overflow;
 +
 +    /**
 +     * Force seeking to any (also non key) frames.
 +     * - encoding: unused
 +     * - decoding: Set by user via AVOPtions (NO direct access)
 +     */
 +    int seek2any;
 +
 +    /**
 +     * Flush the I/O context after each packet.
 +     * - encoding: Set by user via AVOptions (NO direct access)
 +     * - decoding: unused
 +     */
 +    int flush_packets;
 +
 +    /**
 +     * format probing score.
 +     * The maximal score is AVPROBE_SCORE_MAX, its set when the demuxer probes
 +     * the format.
 +     * - encoding: unused
 +     * - decoding: set by avformat, read by user via av_format_get_probe_score() (NO direct access)
 +     */
 +    int probe_score;
 +
      /*****************************************************************
       * All fields below this line are not part of the public API. They
       * may not be used outside of libavformat and can be changed and
      /**
       * Offset to remap timestamps to be non-negative.
       * Expressed in timebase units.
 +     * @see AVStream.mux_ts_offset
       */
      int64_t offset;
  
       */
      AVRational offset_timebase;
  
 +    /**
 +     * IO repositioned flag.
 +     * This is set by avformat when the underlaying IO context read pointer
 +     * is repositioned, for example when doing byte based seeking.
 +     * Demuxers can use the flag to detect such changes.
 +     */
 +    int io_repositioned;
  } AVFormatContext;
  
 +int av_format_get_probe_score(const AVFormatContext *s);
 +
 +/**
 + * Returns the method used to set ctx->duration.
 + *
 + * @return AVFMT_DURATION_FROM_PTS, AVFMT_DURATION_FROM_STREAM, or AVFMT_DURATION_FROM_BITRATE.
 + */
 +enum AVDurationEstimationMethod av_fmt_ctx_get_duration_estimation_method(const AVFormatContext* ctx);
 +
  typedef struct AVPacketList {
      AVPacket pkt;
      struct AVPacketList *next;
@@@ -1351,6 -1090,7 +1362,6 @@@ const char *avformat_license(void)
   *
   * @see av_register_input_format()
   * @see av_register_output_format()
 - * @see av_register_protocol()
   */
  void av_register_all(void);
  
@@@ -1416,16 -1156,13 +1427,16 @@@ const AVClass *avformat_get_class(void)
   *
   * When muxing, should be called by the user before avformat_write_header().
   *
 + * User is required to call avcodec_close() and avformat_free_context() to
 + * clean up the allocation by avformat_new_stream().
 + *
   * @param c If non-NULL, the AVCodecContext corresponding to the new stream
   * will be initialized to use this codec. This is needed for e.g. codec-specific
   * defaults to be set, so codec should be provided if it is known.
   *
   * @return newly created stream or NULL on error.
   */
 -AVStream *avformat_new_stream(AVFormatContext *s, AVCodec *c);
 +AVStream *avformat_new_stream(AVFormatContext *s, const AVCodec *c);
  
  AVProgram *av_new_program(AVFormatContext *s, int id);
  
   */
  
  
 +#if FF_API_ALLOC_OUTPUT_CONTEXT
 +/**
 + * @deprecated deprecated in favor of avformat_alloc_output_context2()
 + */
 +attribute_deprecated
 +AVFormatContext *avformat_alloc_output_context(const char *format,
 +                                               AVOutputFormat *oformat,
 +                                               const char *filename);
 +#endif
 +
 +/**
 + * Allocate an AVFormatContext for an output format.
 + * avformat_free_context() can be used to free the context and
 + * everything allocated by the framework within it.
 + *
 + * @param *ctx is set to the created format context, or to NULL in
 + * case of failure
 + * @param oformat format to use for allocating the context, if NULL
 + * format_name and filename are used instead
 + * @param format_name the name of output format to use for allocating the
 + * context, if NULL filename is used instead
 + * @param filename the name of the filename to use for allocating the
 + * context, may be NULL
 + * @return >= 0 in case of success, a negative AVERROR code in case of
 + * failure
 + */
 +int avformat_alloc_output_context2(AVFormatContext **ctx, AVOutputFormat *oformat,
 +                                   const char *format_name, const char *filename);
 +
  /**
   * @addtogroup lavf_decoding
   * @{
@@@ -1495,15 -1203,6 +1506,15 @@@ AVInputFormat *av_probe_input_format(AV
  AVInputFormat *av_probe_input_format2(AVProbeData *pd, int is_opened, int *score_max);
  
  /**
 + * Guess the file format.
 + *
 + * @param is_opened Whether the file is already opened; determines whether
 + *                  demuxers with or without AVFMT_NOFILE are probed.
 + * @param score_ret The score of the best detection.
 + */
 +AVInputFormat *av_probe_input_format3(AVProbeData *pd, int is_opened, int *score_ret);
 +
 +/**
   * Probe a bytestream to determine the input format. Each time a probe returns
   * with a score that is too low, the probe buffer size is increased and another
   * attempt is made. When the maximum probe size is reached, the input format
   * @param logctx the log context
   * @param offset the offset within the bytestream to probe from
   * @param max_probe_size the maximum probe buffer size (zero for default)
 - * @return 0 in case of success, a negative value corresponding to an
 + * @return the score in case of success, a negative value corresponding to an
 + *         the maximal score is AVPROBE_SCORE_MAX
   * AVERROR code otherwise
   */
 +int av_probe_input_buffer2(AVIOContext *pb, AVInputFormat **fmt,
 +                           const char *filename, void *logctx,
 +                           unsigned int offset, unsigned int max_probe_size);
 +
 +/**
 + * Like av_probe_input_buffer2() but returns 0 on success
 + */
  int av_probe_input_buffer(AVIOContext *pb, AVInputFormat **fmt,
                            const char *filename, void *logctx,
                            unsigned int offset, unsigned int max_probe_size);
   */
  int avformat_open_input(AVFormatContext **ps, const char *filename, AVInputFormat *fmt, AVDictionary **options);
  
 +attribute_deprecated
 +int av_demuxer_open(AVFormatContext *ic);
 +
 +#if FF_API_FORMAT_PARAMETERS
 +/**
 + * Read packets of a media file to get stream information. This
 + * is useful for file formats with no headers such as MPEG. This
 + * function also computes the real framerate in case of MPEG-2 repeat
 + * frame mode.
 + * The logical file position is not changed by this function;
 + * examined packets may be buffered for later processing.
 + *
 + * @param ic media file handle
 + * @return >=0 if OK, AVERROR_xxx on error
 + * @todo Let the user decide somehow what information is needed so that
 + *       we do not waste time getting stuff the user does not need.
 + *
 + * @deprecated use avformat_find_stream_info.
 + */
 +attribute_deprecated
 +int av_find_stream_info(AVFormatContext *ic);
 +#endif
 +
  /**
   * Read packets of a media file to get stream information. This
   * is useful for file formats with no headers such as MPEG. This
  int avformat_find_stream_info(AVFormatContext *ic, AVDictionary **options);
  
  /**
 + * Find the programs which belong to a given stream.
 + *
 + * @param ic    media file handle
 + * @param last  the last found program, the search will start after this
 + *              program, or from the beginning if it is NULL
 + * @param s     stream index
 + * @return the next program which belongs to s, NULL if no program is found or
 + *         the last program is not among the programs of ic.
 + */
 +AVProgram *av_find_program_from_stream(AVFormatContext *ic, AVProgram *last, int s);
 +
 +/**
   * Find the "best" stream in the file.
   * The best stream is determined according to various heuristics as the most
   * likely to be what the user expects.
@@@ -1640,24 -1296,6 +1651,24 @@@ int av_find_best_stream(AVFormatContex
                          AVCodec **decoder_ret,
                          int flags);
  
 +#if FF_API_READ_PACKET
 +/**
 + * @deprecated use AVFMT_FLAG_NOFILLIN | AVFMT_FLAG_NOPARSE to read raw
 + * unprocessed packets
 + *
 + * Read a transport packet from a media file.
 + *
 + * This function is obsolete and should never be used.
 + * Use av_read_frame() instead.
 + *
 + * @param s media file handle
 + * @param pkt is filled
 + * @return 0 if OK, AVERROR_xxx on error
 + */
 +attribute_deprecated
 +int av_read_packet(AVFormatContext *s, AVPacket *pkt);
 +#endif
 +
  /**
   * Return the next frame of a stream.
   * This function returns what is stored in the file, and does not validate
@@@ -1712,7 -1350,6 +1723,7 @@@ int av_seek_frame(AVFormatContext *s, i
   * or if stream_index is -1, in AV_TIME_BASE units.
   * If flags contain AVSEEK_FLAG_ANY, then non-keyframes are treated as
   * keyframes (this may not be supported by all demuxers).
 + * If flags contain AVSEEK_FLAG_BACKWARD, it is ignored.
   *
   * @param stream_index index of the stream which is used as time base reference
   * @param min_ts smallest acceptable timestamp
@@@ -1740,17 -1377,6 +1751,17 @@@ int av_read_play(AVFormatContext *s)
   */
  int av_read_pause(AVFormatContext *s);
  
 +#if FF_API_CLOSE_INPUT_FILE
 +/**
 + * @deprecated use avformat_close_input()
 + * Close a media file (but not its codecs).
 + *
 + * @param s media file handle
 + */
 +attribute_deprecated
 +void av_close_input_file(AVFormatContext *s);
 +#endif
 +
  /**
   * Close an opened input AVFormatContext. Free it and all its contents
   * and set *s to NULL.
@@@ -1760,30 -1386,6 +1771,30 @@@ void avformat_close_input(AVFormatConte
   * @}
   */
  
 +#if FF_API_NEW_STREAM
 +/**
 + * Add a new stream to a media file.
 + *
 + * Can only be called in the read_header() function. If the flag
 + * AVFMTCTX_NOHEADER is in the format context, then new streams
 + * can be added in read_packet too.
 + *
 + * @param s media file handle
 + * @param id file-format-dependent stream ID
 + */
 +attribute_deprecated
 +AVStream *av_new_stream(AVFormatContext *s, int id);
 +#endif
 +
 +#if FF_API_SET_PTS_INFO
 +/**
 + * @deprecated this function is not supposed to be called outside of lavf
 + */
 +attribute_deprecated
 +void av_set_pts_info(AVStream *s, int pts_wrap_bits,
 +                     unsigned int pts_num, unsigned int pts_den);
 +#endif
 +
  #define AVSEEK_FLAG_BACKWARD 1 ///< seek backward
  #define AVSEEK_FLAG_BYTE     2 ///< seeking based on position in bytes
  #define AVSEEK_FLAG_ANY      4 ///< seek to any frame, even non-keyframes
@@@ -1891,25 -1493,6 +1902,25 @@@ enum AVCodecID av_guess_codec(AVOutputF
                              enum AVMediaType type);
  
  /**
 + * Get timing information for the data currently output.
 + * The exact meaning of "currently output" depends on the format.
 + * It is mostly relevant for devices that have an internal buffer and/or
 + * work in real time.
 + * @param s          media file handle
 + * @param stream     stream in the media file
 + * @param[out] dts   DTS of the last packet output for the stream, in stream
 + *                   time_base units
 + * @param[out] wall  absolute time when that packet whas output,
 + *                   in microsecond
 + * @return  0 if OK, AVERROR(ENOSYS) if the format does not support it
 + * Note: some formats or devices may not allow to measure dts and wall
 + * atomically.
 + */
 +int av_get_output_timestamp(struct AVFormatContext *s, int stream,
 +                            int64_t *dts, int64_t *wall);
 +
 +
 +/**
   * @}
   */
  
@@@ -1991,18 -1574,6 +2002,18 @@@ enum AVCodecID av_codec_get_id(const st
   */
  unsigned int av_codec_get_tag(const struct AVCodecTag * const *tags, enum AVCodecID id);
  
 +/**
 + * Get the codec tag for the given codec id.
 + *
 + * @param tags list of supported codec_id - codec_tag pairs, as stored
 + * in AVInputFormat.codec_tag and AVOutputFormat.codec_tag
 + * @param id codec id that should be searched for in the list
 + * @param tag A pointer to the found tag
 + * @return 0 if id was not found in tags, > 0 if it was found
 + */
 +int av_codec_get_tag2(const struct AVCodecTag * const *tags, enum AVCodecID id,
 +                      unsigned int *tag);
 +
  int av_find_default_stream_index(AVFormatContext *s);
  
  /**
@@@ -2137,60 -1708,11 +2148,60 @@@ const struct AVCodecTag *avformat_get_r
   * @return the table mapping RIFF FourCCs for audio to AVCodecID.
   */
  const struct AVCodecTag *avformat_get_riff_audio_tags(void);
 +
  /**
   * @}
   */
  
  /**
 + * Guess the sample aspect ratio of a frame, based on both the stream and the
 + * frame aspect ratio.
 + *
 + * Since the frame aspect ratio is set by the codec but the stream aspect ratio
 + * is set by the demuxer, these two may not be equal. This function tries to
 + * return the value that you should use if you would like to display the frame.
 + *
 + * Basic logic is to use the stream aspect ratio if it is set to something sane
 + * otherwise use the frame aspect ratio. This way a container setting, which is
 + * usually easy to modify can override the coded value in the frames.
 + *
 + * @param format the format context which the stream is part of
 + * @param stream the stream which the frame is part of
 + * @param frame the frame with the aspect ratio to be determined
 + * @return the guessed (valid) sample_aspect_ratio, 0/1 if no idea
 + */
 +AVRational av_guess_sample_aspect_ratio(AVFormatContext *format, AVStream *stream, AVFrame *frame);
 +
 +/**
 + * Guess the frame rate, based on both the container and codec information.
 + *
 + * @param ctx the format context which the stream is part of
 + * @param stream the stream which the frame is part of
 + * @param frame the frame for which the frame rate should be determined, may be NULL
 + * @return the guessed (valid) frame rate, 0/1 if no idea
 + */
 +AVRational av_guess_frame_rate(AVFormatContext *ctx, AVStream *stream, AVFrame *frame);
 +
 +/**
 + * Check if the stream st contained in s is matched by the stream specifier
 + * spec.
 + *
 + * See the "stream specifiers" chapter in the documentation for the syntax
 + * of spec.
 + *
 + * @return  >0 if st is matched by spec;
 + *          0  if st is not matched by spec;
 + *          AVERROR code if spec is invalid
 + *
 + * @note  A stream specifier can match several streams in the format.
 + */
 +int avformat_match_stream_specifier(AVFormatContext *s, AVStream *st,
 +                                    const char *spec);
 +
 +int avformat_queue_attached_pictures(AVFormatContext *s);
 +
 +
 +/**
   * @}
   */
  
@@@ -2,26 -2,25 +2,26 @@@
   * Matroska muxer
   * Copyright (c) 2007 David Conrad
   *
 - * This file is part of Libav.
 + * This file is part of FFmpeg.
   *
 - * Libav is free software; you can redistribute it and/or
 + * 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.
   *
 - * Libav is distributed in the hope that it will be useful,
 + * 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 Libav; if not, write to the Free Software
 + * License along with FFmpeg; if not, write to the Free Software
   * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
   */
  
  #include "avc.h"
  #include "avformat.h"
 +#include "avio_internal.h"
  #include "avlanguage.h"
  #include "flacenc.h"
  #include "internal.h"
@@@ -39,7 -38,6 +39,7 @@@
  #include "libavutil/opt.h"
  #include "libavutil/random_seed.h"
  #include "libavutil/samplefmt.h"
 +#include "libavutil/sha.h"
  
  #include "libavcodec/xiph.h"
  #include "libavcodec/mpeg4audio.h"
@@@ -67,7 -65,6 +67,7 @@@ typedef struct 
      uint64_t        pts;
      int             tracknum;
      int64_t         cluster_pos;        ///< file offset of the cluster containing the block
 +    int64_t         relative_pos;       ///< relative offset from the position of the cluster containing the block
  } mkv_cuepoint;
  
  typedef struct {
@@@ -78,7 -75,6 +78,7 @@@
  
  typedef struct {
      int             write_dts;
 +    int             has_cue;
  } mkv_track;
  
  #define MODE_MATROSKAv2 0x01
@@@ -107,8 -103,7 +107,9 @@@ typedef struct MatroskaMuxContext 
      int cluster_size_limit;
      int64_t cues_pos;
      int64_t cluster_time_limit;
 +
 +    uint32_t chapter_id_offset;
+     int wrote_chapters;
  } MatroskaMuxContext;
  
  
   * offset, 4 bytes for target EBML ID */
  #define MAX_SEEKENTRY_SIZE 21
  
 -/** per-cuepoint-track - 3 1-byte EBML IDs, 3 1-byte EBML sizes, 2
 +/** per-cuepoint-track - 4 1-byte EBML IDs, 4 1-byte EBML sizes, 3
   * 8-byte uint max */
 -#define MAX_CUETRACKPOS_SIZE 22
 +#define MAX_CUETRACKPOS_SIZE 32
  
  /** per-cuepoint - 2 1-byte EBML IDs, 2 1-byte EBML sizes, 8-byte uint max */
  #define MAX_CUEPOINT_SIZE(num_tracks) 12 + MAX_CUETRACKPOS_SIZE*num_tracks
  
 +/** Seek preroll value for opus */
 +#define OPUS_SEEK_PREROLL 80000000
 +
  
  static int ebml_id_size(unsigned int id)
  {
@@@ -136,7 -128,7 +137,7 @@@ static void put_ebml_id(AVIOContext *pb
  {
      int i = ebml_id_size(id);
      while (i--)
 -        avio_w8(pb, id >> (i*8));
 +        avio_w8(pb, (uint8_t)(id >> (i*8)));
  }
  
  /**
   */
  static void put_ebml_size_unknown(AVIOContext *pb, int bytes)
  {
 -    assert(bytes <= 8);
 +    av_assert0(bytes <= 8);
      avio_w8(pb, 0x1ff >> bytes);
 -    while (--bytes)
 -        avio_w8(pb, 0xff);
 +    ffio_fill(pb, 0xff, bytes - 1);
  }
  
  /**
@@@ -172,18 -165,18 +173,18 @@@ static void put_ebml_num(AVIOContext *p
      int i, needed_bytes = ebml_num_size(num);
  
      // sizes larger than this are currently undefined in EBML
 -    assert(num < (1ULL<<56)-1);
 +    av_assert0(num < (1ULL<<56)-1);
  
      if (bytes == 0)
          // don't care how many bytes are used, so use the min
          bytes = needed_bytes;
      // the bytes needed to write the given size would exceed the bytes
      // that we need to use, so write unknown size. This shouldn't happen.
 -    assert(bytes >= needed_bytes);
 +    av_assert0(bytes >= needed_bytes);
  
      num |= 1ULL << bytes*7;
      for (i = bytes - 1; i >= 0; i--)
 -        avio_w8(pb, num >> i*8);
 +        avio_w8(pb, (uint8_t)(num >> i*8));
  }
  
  static void put_ebml_uint(AVIOContext *pb, unsigned int elementid, uint64_t val)
      put_ebml_id(pb, elementid);
      put_ebml_num(pb, bytes, 0);
      for (i = bytes - 1; i >= 0; i--)
 -        avio_w8(pb, val >> i*8);
 +        avio_w8(pb, (uint8_t)(val >> i*8));
  }
  
  static void put_ebml_float(AVIOContext *pb, unsigned int elementid, double val)
@@@ -228,7 -221,7 +229,7 @@@ static void put_ebml_void(AVIOContext *
  {
      int64_t currentpos = avio_tell(pb);
  
 -    assert(size >= 2);
 +    av_assert0(size >= 2);
  
      put_ebml_id(pb, EBML_ID_VOID);
      // we need to subtract the length needed to store the size from the
          put_ebml_num(pb, size-1, 0);
      else
          put_ebml_num(pb, size-9, 8);
 -    while(avio_tell(pb) < currentpos + size)
 -        avio_w8(pb, 0);
 +    ffio_fill(pb, 0, currentpos + size - avio_tell(pb));
  }
  
  static ebml_master start_ebml_master(AVIOContext *pb, unsigned int elementid, uint64_t expectedsize)
@@@ -261,7 -255,9 +262,7 @@@ static void end_ebml_master(AVIOContex
  
  static void put_xiph_size(AVIOContext *pb, int size)
  {
 -    int i;
 -    for (i = 0; i < size / 255; i++)
 -        avio_w8(pb, 255);
 +    ffio_fill(pb, 255, size / 255);
      avio_w8(pb, size % 255);
  }
  
@@@ -378,7 -374,7 +379,7 @@@ static mkv_cues * mkv_start_cues(int64_
      return cues;
  }
  
 -static int mkv_add_cuepoint(mkv_cues *cues, int stream, int64_t ts, int64_t cluster_pos)
 +static int mkv_add_cuepoint(mkv_cues *cues, int stream, int64_t ts, int64_t cluster_pos, int64_t relative_pos)
  {
      mkv_cuepoint *entries = cues->entries;
  
  
      entries[cues->num_entries  ].pts = ts;
      entries[cues->num_entries  ].tracknum = stream + 1;
 -    entries[cues->num_entries++].cluster_pos = cluster_pos - cues->segment_offset;
 +    entries[cues->num_entries  ].cluster_pos = cluster_pos - cues->segment_offset;
 +    entries[cues->num_entries++].relative_pos = relative_pos;
  
      cues->entries = entries;
      return 0;
  }
  
 -static int64_t mkv_write_cues(AVIOContext *pb, mkv_cues *cues, int num_tracks)
 +static int64_t mkv_write_cues(AVIOContext *pb, mkv_cues *cues, mkv_track *tracks, int num_tracks)
  {
      ebml_master cues_element;
      int64_t currentpos;
  
          // put all the entries from different tracks that have the exact same
          // timestamp into the same CuePoint
 +        for (j = 0; j < num_tracks; j++)
 +            tracks[j].has_cue = 0;
          for (j = 0; j < cues->num_entries - i && entry[j].pts == pts; j++) {
 +            int tracknum = entry[j].tracknum - 1;
 +            av_assert0(tracknum>=0 && tracknum<num_tracks);
 +            if (tracks[tracknum].has_cue)
 +                continue;
 +            tracks[tracknum].has_cue = 1;
              track_positions = start_ebml_master(pb, MATROSKA_ID_CUETRACKPOSITION, MAX_CUETRACKPOS_SIZE);
 -            put_ebml_uint(pb, MATROSKA_ID_CUETRACK          , entry[j].tracknum   );
 -            put_ebml_uint(pb, MATROSKA_ID_CUECLUSTERPOSITION, entry[j].cluster_pos);
 +            put_ebml_uint(pb, MATROSKA_ID_CUETRACK           , entry[j].tracknum   );
 +            put_ebml_uint(pb, MATROSKA_ID_CUECLUSTERPOSITION , entry[j].cluster_pos);
 +            put_ebml_uint(pb, MATROSKA_ID_CUERELATIVEPOSITION, entry[j].relative_pos);
              end_ebml_master(pb, track_positions);
          }
          i += j - 1;
@@@ -519,7 -506,7 +520,7 @@@ static int mkv_write_codecprivate(AVFor
                  avio_write(dyn_cp, codec->extradata + 12,
                                     codec->extradata_size - 12);
          }
 -        else if (codec->extradata_size)
 +        else if (codec->extradata_size && codec->codec_id != AV_CODEC_ID_TTA)
              avio_write(dyn_cp, codec->extradata, codec->extradata_size);
      } else if (codec->codec_type == AVMEDIA_TYPE_VIDEO) {
          if (qt_id) {
              if (!codec->codec_tag)
                  codec->codec_tag = ff_codec_get_tag(ff_codec_bmp_tags, codec->codec_id);
              if (!codec->codec_tag) {
 -                av_log(s, AV_LOG_ERROR, "No bmp codec ID found.\n");
 -                ret = -1;
 +                av_log(s, AV_LOG_ERROR, "No bmp codec tag found for codec %s\n",
 +                       avcodec_get_name(codec->codec_id));
 +                ret = AVERROR(EINVAL);
              }
  
              ff_put_bmp_header(dyn_cp, codec, ff_codec_bmp_tags, 0);
          unsigned int tag;
          tag = ff_codec_get_tag(ff_codec_wav_tags, codec->codec_id);
          if (!tag) {
 -            av_log(s, AV_LOG_ERROR, "No wav codec ID found.\n");
 -            ret = -1;
 +            av_log(s, AV_LOG_ERROR, "No wav codec tag found for codec %s\n",
 +                   avcodec_get_name(codec->codec_id));
 +            ret = AVERROR(EINVAL);
          }
          if (!codec->codec_tag)
              codec->codec_tag = tag;
@@@ -565,7 -550,7 +566,7 @@@ static int mkv_write_tracks(AVFormatCon
      MatroskaMuxContext *mkv = s->priv_data;
      AVIOContext *pb = s->pb;
      ebml_master tracks;
 -    int i, j, ret;
 +    int i, j, ret, default_stream_exists = 0;
  
      ret = mkv_add_seekhead_entry(mkv->main_seekhead, MATROSKA_ID_TRACKS, avio_tell(pb));
      if (ret < 0) return ret;
      tracks = start_ebml_master(pb, MATROSKA_ID_TRACKS, 0);
      for (i = 0; i < s->nb_streams; i++) {
          AVStream *st = s->streams[i];
 +        default_stream_exists |= st->disposition & AV_DISPOSITION_DEFAULT;
 +    }
 +    for (i = 0; i < s->nb_streams; i++) {
 +        AVStream *st = s->streams[i];
          AVCodecContext *codec = st->codec;
          ebml_master subinfo, track;
          int native_id = 0;
  
          if (!bit_depth)
              bit_depth = av_get_bytes_per_sample(codec->sample_fmt) << 3;
 +        if (!bit_depth)
 +            bit_depth = codec->bits_per_coded_sample;
  
          if (codec->codec_id == AV_CODEC_ID_AAC)
              get_aac_sample_rates(s, codec, &sample_rate, &output_sample_rate);
          if ((tag = av_dict_get(st->metadata, "title", NULL, 0)))
              put_ebml_string(pb, MATROSKA_ID_TRACKNAME, tag->value);
          tag = av_dict_get(st->metadata, "language", NULL, 0);
 -        put_ebml_string(pb, MATROSKA_ID_TRACKLANGUAGE, tag ? tag->value:"und");
 +        if (mkv->mode != MODE_WEBM || codec->codec_id != AV_CODEC_ID_WEBVTT) {
 +            put_ebml_string(pb, MATROSKA_ID_TRACKLANGUAGE, tag ? tag->value:"und");
 +        } else if (tag && tag->value) {
 +            put_ebml_string(pb, MATROSKA_ID_TRACKLANGUAGE, tag->value);
 +        }
  
          // The default value for TRACKFLAGDEFAULT is 1, so add element
          // if we need to clear it.
 -        if (!(st->disposition & AV_DISPOSITION_DEFAULT))
 +        if (default_stream_exists && !(st->disposition & AV_DISPOSITION_DEFAULT))
              put_ebml_uint(pb, MATROSKA_ID_TRACKFLAGDEFAULT, !!(st->disposition & AV_DISPOSITION_DEFAULT));
  
 -        // look for a codec ID string specific to mkv to use,
 -        // if none are found, use AVI codes
 -        for (j = 0; ff_mkv_codec_tags[j].id != AV_CODEC_ID_NONE; j++) {
 -            if (ff_mkv_codec_tags[j].id == codec->codec_id) {
 -                put_ebml_string(pb, MATROSKA_ID_CODECID, ff_mkv_codec_tags[j].str);
 -                native_id = 1;
 -                break;
 +        if (st->disposition & AV_DISPOSITION_FORCED)
 +            put_ebml_uint(pb, MATROSKA_ID_TRACKFLAGFORCED, 1);
 +
 +        if (mkv->mode == MODE_WEBM && codec->codec_id == AV_CODEC_ID_WEBVTT) {
 +            const char *codec_id;
 +            if (st->disposition & AV_DISPOSITION_CAPTIONS) {
 +                codec_id = "D_WEBVTT/CAPTIONS";
 +                native_id = MATROSKA_TRACK_TYPE_SUBTITLE;
 +            } else if (st->disposition & AV_DISPOSITION_DESCRIPTIONS) {
 +                codec_id = "D_WEBVTT/DESCRIPTIONS";
 +                native_id = MATROSKA_TRACK_TYPE_METADATA;
 +            } else if (st->disposition & AV_DISPOSITION_METADATA) {
 +                codec_id = "D_WEBVTT/METADATA";
 +                native_id = MATROSKA_TRACK_TYPE_METADATA;
 +            } else {
 +                codec_id = "D_WEBVTT/SUBTITLES";
 +                native_id = MATROSKA_TRACK_TYPE_SUBTITLE;
 +            }
 +            put_ebml_string(pb, MATROSKA_ID_CODECID, codec_id);
 +        } else {
 +            // look for a codec ID string specific to mkv to use,
 +            // if none are found, use AVI codes
 +            for (j = 0; ff_mkv_codec_tags[j].id != AV_CODEC_ID_NONE; j++) {
 +                if (ff_mkv_codec_tags[j].id == codec->codec_id) {
 +                    put_ebml_string(pb, MATROSKA_ID_CODECID, ff_mkv_codec_tags[j].str);
 +                    native_id = 1;
 +                    break;
 +                }
              }
          }
  
 +        if (codec->codec_id == AV_CODEC_ID_OPUS) {
 +            put_ebml_uint(pb, MATROSKA_ID_SEEKPREROLL, OPUS_SEEK_PREROLL);
 +        }
 +
          if (mkv->mode == MODE_WEBM && !(codec->codec_id == AV_CODEC_ID_VP8 ||
 -                                        codec->codec_id == AV_CODEC_ID_VORBIS)) {
 +                                        codec->codec_id == AV_CODEC_ID_VP9 ||
 +                                      ((codec->codec_id == AV_CODEC_ID_OPUS)&&(codec->strict_std_compliance <= FF_COMPLIANCE_EXPERIMENTAL)) ||
 +                                        codec->codec_id == AV_CODEC_ID_VORBIS ||
 +                                        codec->codec_id == AV_CODEC_ID_WEBVTT)) {
              av_log(s, AV_LOG_ERROR,
 -                   "Only VP8 video and Vorbis audio are supported for WebM.\n");
 +                   "Only VP8,VP9 video and Vorbis,Opus(experimental, use -strict -2) audio and WebVTT subtitles are supported for WebM.\n");
              return AVERROR(EINVAL);
          }
  
          switch (codec->codec_type) {
              case AVMEDIA_TYPE_VIDEO:
                  put_ebml_uint(pb, MATROSKA_ID_TRACKTYPE, MATROSKA_TRACK_TYPE_VIDEO);
 -                put_ebml_uint(pb, MATROSKA_ID_TRACKDEFAULTDURATION, av_q2d(codec->time_base)*1E9);
 +                if(st->avg_frame_rate.num && st->avg_frame_rate.den && 1.0/av_q2d(st->avg_frame_rate) > av_q2d(codec->time_base))
 +                    put_ebml_uint(pb, MATROSKA_ID_TRACKDEFAULTDURATION, 1E9/av_q2d(st->avg_frame_rate));
 +                else
 +                    put_ebml_uint(pb, MATROSKA_ID_TRACKDEFAULTDURATION, av_q2d(codec->time_base)*1E9);
  
                  if (!native_id &&
                        ff_codec_get_tag(ff_codec_movvideo_tags, codec->codec_id) &&
                  // XXX: interlace flag?
                  put_ebml_uint (pb, MATROSKA_ID_VIDEOPIXELWIDTH , codec->width);
                  put_ebml_uint (pb, MATROSKA_ID_VIDEOPIXELHEIGHT, codec->height);
 -                if ((tag = av_dict_get(s->metadata, "stereo_mode", NULL, 0))) {
 -                    uint8_t stereo_fmt = atoi(tag->value);
 -                    int valid_fmt = 0;
 -
 -                    switch (mkv->mode) {
 -                    case MODE_WEBM:
 -                        if (stereo_fmt <= MATROSKA_VIDEO_STEREOMODE_TYPE_TOP_BOTTOM
 -                            || stereo_fmt == MATROSKA_VIDEO_STEREOMODE_TYPE_RIGHT_LEFT)
 -                            valid_fmt = 1;
 -                        break;
 -                    case MODE_MATROSKAv2:
 -                        if (stereo_fmt <= MATROSKA_VIDEO_STEREOMODE_TYPE_BOTH_EYES_BLOCK_RL)
 -                            valid_fmt = 1;
 -                        break;
 -                    }
  
 -                    if (valid_fmt)
 -                        put_ebml_uint (pb, MATROSKA_ID_VIDEOSTEREOMODE, stereo_fmt);
 +                if ((tag = av_dict_get(st->metadata, "stereo_mode", NULL, 0)) ||
 +                    (tag = av_dict_get( s->metadata, "stereo_mode", NULL, 0))) {
 +                    // save stereo mode flag
 +                    uint64_t st_mode = MATROSKA_VIDEO_STEREO_MODE_COUNT;
 +
 +                    for (j=0; j<MATROSKA_VIDEO_STEREO_MODE_COUNT; j++)
 +                        if (!strcmp(tag->value, ff_matroska_video_stereo_mode[j])){
 +                            st_mode = j;
 +                            break;
 +                        }
 +
 +                    if ((mkv->mode == MODE_WEBM && st_mode > 3 && st_mode != 11)
 +                        || st_mode >= MATROSKA_VIDEO_STEREO_MODE_COUNT) {
 +                        av_log(s, AV_LOG_ERROR,
 +                               "The specified stereo mode is not valid.\n");
 +                        return AVERROR(EINVAL);
 +                    } else
 +                        put_ebml_uint(pb, MATROSKA_ID_VIDEOSTEREOMODE, st_mode);
                  }
 +
 +                if ((tag = av_dict_get(st->metadata, "alpha_mode", NULL, 0)) ||
 +                    (tag = av_dict_get( s->metadata, "alpha_mode", NULL, 0)) ||
 +                    (codec->pix_fmt == AV_PIX_FMT_YUVA420P)) {
 +                    put_ebml_uint(pb, MATROSKA_ID_VIDEOALPHAMODE, 1);
 +                }
 +
                  if (st->sample_aspect_ratio.num) {
 -                    int d_width = codec->width*av_q2d(st->sample_aspect_ratio);
 +                    int64_t d_width = av_rescale(codec->width, st->sample_aspect_ratio.num, st->sample_aspect_ratio.den);
 +                    if (d_width > INT_MAX) {
 +                        av_log(s, AV_LOG_ERROR, "Overflow in display width\n");
 +                        return AVERROR(EINVAL);
 +                    }
                      put_ebml_uint(pb, MATROSKA_ID_VIDEODISPLAYWIDTH , d_width);
                      put_ebml_uint(pb, MATROSKA_ID_VIDEODISPLAYHEIGHT, codec->height);
 -                    put_ebml_uint(pb, MATROSKA_ID_VIDEODISPLAYUNIT, 3);
 +                }
 +
 +                if (codec->codec_id == AV_CODEC_ID_RAWVIDEO) {
 +                    uint32_t color_space = av_le2ne32(codec->codec_tag);
 +                    put_ebml_binary(pb, MATROSKA_ID_VIDEOCOLORSPACE, &color_space, sizeof(color_space));
                  }
                  end_ebml_master(pb, subinfo);
                  break;
                  break;
  
              case AVMEDIA_TYPE_SUBTITLE:
 -                put_ebml_uint(pb, MATROSKA_ID_TRACKTYPE, MATROSKA_TRACK_TYPE_SUBTITLE);
                  if (!native_id) {
                      av_log(s, AV_LOG_ERROR, "Subtitle codec %d is not supported.\n", codec->codec_id);
                      return AVERROR(ENOSYS);
                  }
 +
 +                if (mkv->mode != MODE_WEBM || codec->codec_id != AV_CODEC_ID_WEBVTT)
 +                    native_id = MATROSKA_TRACK_TYPE_SUBTITLE;
 +
 +                put_ebml_uint(pb, MATROSKA_ID_TRACKTYPE, native_id);
                  break;
              default:
                  av_log(s, AV_LOG_ERROR, "Only audio, video, and subtitles are supported for Matroska.\n");
 -                break;
 +                return AVERROR(EINVAL);
 +        }
 +
 +        if (mkv->mode != MODE_WEBM || codec->codec_id != AV_CODEC_ID_WEBVTT) {
 +            ret = mkv_write_codecprivate(s, pb, codec, native_id, qt_id);
 +            if (ret < 0) return ret;
          }
 -        ret = mkv_write_codecprivate(s, pb, codec, native_id, qt_id);
 -        if (ret < 0) return ret;
  
          end_ebml_master(pb, track);
  
@@@ -790,7 -711,7 +791,7 @@@ static int mkv_write_chapters(AVFormatC
      AVRational scale = {1, 1E9};
      int i, ret;
  
-     if (!s->nb_chapters)
+     if (!s->nb_chapters || mkv->wrote_chapters)
          return 0;
  
      ret = mkv_add_seekhead_entry(mkv->main_seekhead, MATROSKA_ID_CHAPTERS, avio_tell(pb));
          AVDictionaryEntry *t = NULL;
  
          chapteratom = start_ebml_master(pb, MATROSKA_ID_CHAPTERATOM, 0);
 -        put_ebml_uint(pb, MATROSKA_ID_CHAPTERUID, c->id);
 +        put_ebml_uint(pb, MATROSKA_ID_CHAPTERUID, c->id + mkv->chapter_id_offset);
          put_ebml_uint(pb, MATROSKA_ID_CHAPTERTIMESTART,
                        av_rescale_q(c->start, c->time_base, scale));
          put_ebml_uint(pb, MATROSKA_ID_CHAPTERTIMEEND,
      }
      end_ebml_master(pb, editionentry);
      end_ebml_master(pb, chapters);
+     mkv->wrote_chapters = 1;
      return 0;
  }
  
@@@ -878,33 -801,21 +881,33 @@@ static int mkv_write_tag(AVFormatContex
      end_ebml_master(s->pb, targets);
  
      while ((t = av_dict_get(m, "", t, AV_DICT_IGNORE_SUFFIX)))
 -        if (av_strcasecmp(t->key, "title"))
 +        if (av_strcasecmp(t->key, "title") && av_strcasecmp(t->key, "stereo_mode"))
              mkv_write_simpletag(s->pb, t);
  
      end_ebml_master(s->pb, tag);
      return 0;
  }
  
 +static int mkv_check_tag(AVDictionary *m)
 +{
 +    AVDictionaryEntry *t = NULL;
 +
 +    while ((t = av_dict_get(m, "", t, AV_DICT_IGNORE_SUFFIX)))
 +        if (av_strcasecmp(t->key, "title") && av_strcasecmp(t->key, "stereo_mode"))
 +            return 1;
 +
 +    return 0;
 +}
 +
  static int mkv_write_tags(AVFormatContext *s)
  {
 +    MatroskaMuxContext *mkv = s->priv_data;
      ebml_master tags = {0};
      int i, ret;
  
      ff_metadata_conv_ctx(s, ff_mkv_metadata_conv, NULL);
  
 -    if (av_dict_get(s->metadata, "", NULL, AV_DICT_IGNORE_SUFFIX)) {
 +    if (mkv_check_tag(s->metadata)) {
          ret = mkv_write_tag(s, s->metadata, 0, 0, &tags);
          if (ret < 0) return ret;
      }
      for (i = 0; i < s->nb_streams; i++) {
          AVStream *st = s->streams[i];
  
 -        if (!av_dict_get(st->metadata, "", 0, AV_DICT_IGNORE_SUFFIX))
 +        if (!mkv_check_tag(st->metadata))
              continue;
  
          ret = mkv_write_tag(s, st->metadata, MATROSKA_ID_TAGTARGETS_TRACKUID, i + 1, &tags);
      for (i = 0; i < s->nb_chapters; i++) {
          AVChapter *ch = s->chapters[i];
  
 -        if (!av_dict_get(ch->metadata, "", NULL, AV_DICT_IGNORE_SUFFIX))
 +        if (!mkv_check_tag(ch->metadata))
              continue;
  
 -        ret = mkv_write_tag(s, ch->metadata, MATROSKA_ID_TAGTARGETS_CHAPTERUID, ch->id, &tags);
 +        ret = mkv_write_tag(s, ch->metadata, MATROSKA_ID_TAGTARGETS_CHAPTERUID, ch->id + mkv->chapter_id_offset, &tags);
          if (ret < 0) return ret;
      }
  
@@@ -957,7 -868,6 +960,7 @@@ static int mkv_write_attachments(AVForm
          ebml_master attached_file;
          AVDictionaryEntry *t;
          const char *mimetype = NULL;
 +        uint64_t fileuid;
  
          if (st->codec->codec_type != AVMEDIA_TYPE_ATTACHMENT)
              continue;
              return AVERROR(EINVAL);
          }
  
 +        if (st->codec->flags & CODEC_FLAG_BITEXACT) {
 +            struct AVSHA *sha = av_sha_alloc();
 +            uint8_t digest[20];
 +            if (!sha)
 +                return AVERROR(ENOMEM);
 +            av_sha_init(sha, 160);
 +            av_sha_update(sha, st->codec->extradata, st->codec->extradata_size);
 +            av_sha_final(sha, digest);
 +            av_free(sha);
 +            fileuid = AV_RL64(digest);
 +        } else {
 +            fileuid = av_lfg_get(&c);
 +        }
 +        av_log(s, AV_LOG_VERBOSE, "Using %.16"PRIx64" for attachment %d\n",
 +               fileuid, i);
 +
          put_ebml_string(pb, MATROSKA_ID_FILEMIMETYPE, mimetype);
          put_ebml_binary(pb, MATROSKA_ID_FILEDATA, st->codec->extradata, st->codec->extradata_size);
 -        put_ebml_uint(pb, MATROSKA_ID_FILEUID, av_lfg_get(&c));
 +        put_ebml_uint(pb, MATROSKA_ID_FILEUID, fileuid);
          end_ebml_master(pb, attached_file);
      }
      end_ebml_master(pb, attachments);
@@@ -1024,22 -918,6 +1027,22 @@@ static int mkv_write_header(AVFormatCon
      if (!strcmp(s->oformat->name, "webm")) mkv->mode = MODE_WEBM;
      else                                   mkv->mode = MODE_MATROSKAv2;
  
 +    if (s->avoid_negative_ts < 0)
 +        s->avoid_negative_ts = 1;
 +
 +    for (i = 0; i < s->nb_streams; i++)
 +        if (s->streams[i]->codec->codec_id == AV_CODEC_ID_ATRAC3 ||
 +            s->streams[i]->codec->codec_id == AV_CODEC_ID_COOK ||
 +            s->streams[i]->codec->codec_id == AV_CODEC_ID_RA_288 ||
 +            s->streams[i]->codec->codec_id == AV_CODEC_ID_SIPR ||
 +            s->streams[i]->codec->codec_id == AV_CODEC_ID_RV10 ||
 +            s->streams[i]->codec->codec_id == AV_CODEC_ID_RV20) {
 +            av_log(s, AV_LOG_ERROR,
 +                   "The Matroska muxer does not yet support muxing %s\n",
 +                   avcodec_get_name(s->streams[i]->codec->codec_id));
 +            return AVERROR_PATCHWELCOME;
 +        }
 +
      mkv->tracks = av_mallocz(s->nb_streams * sizeof(*mkv->tracks));
      if (!mkv->tracks)
          return AVERROR(ENOMEM);
          put_ebml_binary(pb, MATROSKA_ID_SEGMENTUID, segment_uid, 16);
      }
  
 +    if (tag = av_dict_get(s->metadata, "creation_time", NULL, 0)) {
 +        // Adjust time so it's relative to 2001-01-01 and convert to nanoseconds.
 +        int64_t date_utc = (ff_iso8601_to_unix_time(tag->value) - 978307200) * 1000000000;
 +        uint8_t date_utc_buf[8];
 +        AV_WB64(date_utc_buf, date_utc);
 +        put_ebml_binary(pb, MATROSKA_ID_DATEUTC, date_utc_buf, 8);
 +    }
 +
      // reserve space for the duration
      mkv->duration = 0;
      mkv->duration_offset = avio_tell(pb);
      ret = mkv_write_tracks(s);
      if (ret < 0) return ret;
  
 +    for (i = 0; i < s->nb_chapters; i++)
 +        mkv->chapter_id_offset = FFMAX(mkv->chapter_id_offset, 1LL - s->chapters[i]->id);
 +
      if (mkv->mode != MODE_WEBM) {
          ret = mkv_write_chapters(s);
          if (ret < 0) return ret;
  
      av_init_packet(&mkv->cur_audio_pkt);
      mkv->cur_audio_pkt.size = 0;
 +    mkv->cluster_pos = -1;
  
      avio_flush(pb);
  
@@@ -1172,12 -1038,11 +1175,12 @@@ static int ass_get_duration(const uint8
      if (sscanf(p, "%*[^,],%d:%d:%d%*c%d,%d:%d:%d%*c%d",
                 &sh, &sm, &ss, &sc, &eh, &em, &es, &ec) != 8)
          return 0;
 -    start = 3600000*sh + 60000*sm + 1000*ss + 10*sc;
 -    end   = 3600000*eh + 60000*em + 1000*es + 10*ec;
 +    start = 3600000LL*sh + 60000LL*sm + 1000LL*ss + 10LL*sc;
 +    end   = 3600000LL*eh + 60000LL*em + 1000LL*es + 10LL*ec;
      return end - start;
  }
  
 +#if FF_API_ASS_SSA
  static int mkv_write_ass_blocks(AVFormatContext *s, AVIOContext *pb, AVPacket *pkt)
  {
      MatroskaMuxContext *mkv = s->priv_data;
  
      return max_duration;
  }
 +#endif
  
  static int mkv_strip_wavpack(const uint8_t *src, uint8_t **pdst, int *size)
  {
@@@ -1282,11 -1146,9 +1285,11 @@@ static void mkv_write_block(AVFormatCon
  {
      MatroskaMuxContext *mkv = s->priv_data;
      AVCodecContext *codec = s->streams[pkt->stream_index]->codec;
 -    uint8_t *data = NULL;
 -    int offset = 0, size = pkt->size;
 +    uint8_t *data = NULL, *side_data = NULL;
 +    int offset = 0, size = pkt->size, side_data_size = 0;
      int64_t ts = mkv->tracks[pkt->stream_index].write_dts ? pkt->dts : pkt->pts;
 +    uint64_t additional_id = 0;
 +    ebml_master block_group, block_additions, block_more;
  
      av_log(s, AV_LOG_DEBUG, "Writing block at offset %" PRIu64 ", size %d, "
             "pts %" PRId64 ", dts %" PRId64 ", duration %d, flags %d\n",
          offset = 8;
      }
  
 +    side_data = av_packet_get_side_data(pkt,
 +                                        AV_PKT_DATA_MATROSKA_BLOCKADDITIONAL,
 +                                        &side_data_size);
 +    if (side_data) {
 +        additional_id = AV_RB64(side_data);
 +        side_data += 8;
 +        side_data_size -= 8;
 +    }
 +
 +    if (side_data_size && additional_id == 1) {
 +        block_group = start_ebml_master(pb, MATROSKA_ID_BLOCKGROUP, 0);
 +        blockid = MATROSKA_ID_BLOCK;
 +    }
 +
      put_ebml_id(pb, blockid);
      put_ebml_num(pb, size+4, 0);
      avio_w8(pb, 0x80 | (pkt->stream_index + 1));     // this assumes stream_index is less than 126
      avio_write(pb, data + offset, size);
      if (data != pkt->data)
          av_free(data);
 +
 +    if (side_data_size && additional_id == 1) {
 +        block_additions = start_ebml_master(pb, MATROSKA_ID_BLOCKADDITIONS, 0);
 +        block_more = start_ebml_master(pb, MATROSKA_ID_BLOCKMORE, 0);
 +        put_ebml_uint(pb, MATROSKA_ID_BLOCKADDID, 1);
 +        put_ebml_id(pb, MATROSKA_ID_BLOCKADDITIONAL);
 +        put_ebml_num(pb, side_data_size, 0);
 +        avio_write(pb, side_data, side_data_size);
 +        end_ebml_master(pb, block_more);
 +        end_ebml_master(pb, block_additions);
 +        end_ebml_master(pb, block_group);
 +    }
  }
  
  static int srt_get_duration(uint8_t **buf)
@@@ -1381,44 -1217,6 +1384,44 @@@ static int mkv_write_srt_blocks(AVForma
      return duration;
  }
  
 +static int mkv_write_vtt_blocks(AVFormatContext *s, AVIOContext *pb, AVPacket *pkt)
 +{
 +    MatroskaMuxContext *mkv = s->priv_data;
 +    ebml_master blockgroup;
 +    int id_size, settings_size, size;
 +    uint8_t *id, *settings;
 +    int64_t ts = mkv->tracks[pkt->stream_index].write_dts ? pkt->dts : pkt->pts;
 +    const int flags = 0;
 +
 +    id_size = 0;
 +    id = av_packet_get_side_data(pkt, AV_PKT_DATA_WEBVTT_IDENTIFIER,
 +                                 &id_size);
 +
 +    settings_size = 0;
 +    settings = av_packet_get_side_data(pkt, AV_PKT_DATA_WEBVTT_SETTINGS,
 +                                       &settings_size);
 +
 +    size = id_size + 1 + settings_size + 1 + pkt->size;
 +
 +    av_log(s, AV_LOG_DEBUG, "Writing block at offset %" PRIu64 ", size %d, "
 +           "pts %" PRId64 ", dts %" PRId64 ", duration %d, flags %d\n",
 +           avio_tell(pb), size, pkt->pts, pkt->dts, pkt->duration, flags);
 +
 +    blockgroup = start_ebml_master(pb, MATROSKA_ID_BLOCKGROUP, mkv_blockgroup_size(size));
 +
 +    put_ebml_id(pb, MATROSKA_ID_BLOCK);
 +    put_ebml_num(pb, size+4, 0);
 +    avio_w8(pb, 0x80 | (pkt->stream_index + 1));     // this assumes stream_index is less than 126
 +    avio_wb16(pb, ts - mkv->cluster_pts);
 +    avio_w8(pb, flags);
 +    avio_printf(pb, "%.*s\n%.*s\n%.*s", id_size, id, settings_size, settings, pkt->size, pkt->data);
 +
 +    put_ebml_uint(pb, MATROSKA_ID_BLOCKDURATION, pkt->duration);
 +    end_ebml_master(pb, blockgroup);
 +
 +    return pkt->duration;
 +}
 +
  static void mkv_flush_dynbuf(AVFormatContext *s)
  {
      MatroskaMuxContext *mkv = s->priv_data;
@@@ -1443,7 -1241,6 +1446,7 @@@ static int mkv_write_packet_internal(AV
      int duration = pkt->duration;
      int ret;
      int64_t ts = mkv->tracks[pkt->stream_index].write_dts ? pkt->dts : pkt->pts;
 +    int64_t relative_packet_pos;
  
      if (ts == AV_NOPTS_VALUE) {
          av_log(s, AV_LOG_ERROR, "Can't write packet with unknown timestamp\n");
      }
  
      if (!s->pb->seekable) {
 -        if (!mkv->dyn_bc)
 -            avio_open_dyn_buf(&mkv->dyn_bc);
 +        if (!mkv->dyn_bc) {
 +            if ((ret = avio_open_dyn_buf(&mkv->dyn_bc)) < 0) {
 +                av_log(s, AV_LOG_ERROR, "Failed to open dynamic buffer\n");
 +                return ret;
 +            }
 +        }
          pb = mkv->dyn_bc;
      }
  
 -    if (!mkv->cluster_pos) {
 +    if (mkv->cluster_pos == -1) {
          mkv->cluster_pos = avio_tell(s->pb);
          mkv->cluster = start_ebml_master(pb, MATROSKA_ID_CLUSTER, 0);
          put_ebml_uint(pb, MATROSKA_ID_CLUSTERTIMECODE, FFMAX(0, ts));
          mkv->cluster_pts = FFMAX(0, ts);
      }
  
 +    relative_packet_pos = avio_tell(s->pb) - mkv->cluster.pos;
 +
      if (codec->codec_type != AVMEDIA_TYPE_SUBTITLE) {
          mkv_write_block(s, pb, MATROSKA_ID_SIMPLEBLOCK, pkt, keyframe << 7);
 +#if FF_API_ASS_SSA
      } else if (codec->codec_id == AV_CODEC_ID_SSA) {
          duration = mkv_write_ass_blocks(s, pb, pkt);
 +#endif
      } else if (codec->codec_id == AV_CODEC_ID_SRT) {
          duration = mkv_write_srt_blocks(s, pb, pkt);
 +    } else if (codec->codec_id == AV_CODEC_ID_WEBVTT) {
 +        duration = mkv_write_vtt_blocks(s, pb, pkt);
      } else {
          ebml_master blockgroup = start_ebml_master(pb, MATROSKA_ID_BLOCKGROUP, mkv_blockgroup_size(pkt->size));
 -        duration = pkt->convergence_duration;
 +        /* For backward compatibility, prefer convergence_duration. */
 +        if (pkt->convergence_duration > 0) {
 +            duration = pkt->convergence_duration;
 +        }
          mkv_write_block(s, pb, MATROSKA_ID_BLOCK, pkt, 0);
          put_ebml_uint(pb, MATROSKA_ID_BLOCKDURATION, duration);
          end_ebml_master(pb, blockgroup);
      }
  
      if (codec->codec_type == AVMEDIA_TYPE_VIDEO && keyframe) {
 -        ret = mkv_add_cuepoint(mkv->cues, pkt->stream_index, ts, mkv->cluster_pos);
 +        ret = mkv_add_cuepoint(mkv->cues, pkt->stream_index, ts, mkv->cluster_pos, relative_packet_pos);
          if (ret < 0) return ret;
      }
  
@@@ -1524,7 -1308,7 +1527,7 @@@ static int mkv_write_packet(AVFormatCon
          cluster_size = avio_tell(pb);
      }
  
 -    if (mkv->cluster_pos &&
 +    if (mkv->cluster_pos != -1 &&
          (cluster_size > mkv->cluster_size_limit ||
           cluster_time > mkv->cluster_time_limit ||
           (codec_type == AVMEDIA_TYPE_VIDEO && keyframe &&
                 " bytes, pts %" PRIu64 "dts %" PRIu64 "\n",
                 avio_tell(pb), pkt->pts, pkt->dts);
          end_ebml_master(pb, mkv->cluster);
 -        mkv->cluster_pos = 0;
 +        mkv->cluster_pos = -1;
          if (mkv->dyn_bc)
              mkv_flush_dynbuf(s);
          avio_flush(s->pb);
@@@ -1572,11 -1356,11 +1575,11 @@@ static int mkv_write_flush_packet(AVFor
      else
          pb = mkv->dyn_bc;
      if (!pkt) {
 -        if (mkv->cluster_pos) {
 +        if (mkv->cluster_pos != -1) {
              av_log(s, AV_LOG_DEBUG, "Flushing cluster at offset %" PRIu64
                     " bytes\n", avio_tell(pb));
              end_ebml_master(pb, mkv->cluster);
 -            mkv->cluster_pos = 0;
 +            mkv->cluster_pos = -1;
              if (mkv->dyn_bc)
                  mkv_flush_dynbuf(s);
              avio_flush(s->pb);
@@@ -1606,10 -1390,15 +1609,15 @@@ static int mkv_write_trailer(AVFormatCo
      if (mkv->dyn_bc) {
          end_ebml_master(mkv->dyn_bc, mkv->cluster);
          mkv_flush_dynbuf(s);
 -    } else if (mkv->cluster_pos) {
 +    } else if (mkv->cluster_pos != -1) {
          end_ebml_master(pb, mkv->cluster);
      }
  
+     if (mkv->mode != MODE_WEBM) {
+         ret = mkv_write_chapters(s);
+         if (ret < 0) return ret;
+     }
      if (pb->seekable) {
          if (mkv->cues->num_entries) {
              if (mkv->reserve_cues_space) {
                  currentpos = avio_tell(pb);
                  avio_seek(pb, mkv->cues_pos, SEEK_SET);
  
 -                cuespos = mkv_write_cues(pb, mkv->cues, s->nb_streams);
 +                cuespos = mkv_write_cues(pb, mkv->cues, mkv->tracks, s->nb_streams);
                  cues_end = avio_tell(pb);
                  if (cues_end > cuespos + mkv->reserve_cues_space) {
                      av_log(s, AV_LOG_ERROR, "Insufficient space reserved for cues: %d "
  
                  avio_seek(pb, currentpos, SEEK_SET);
              } else {
 -                cuespos = mkv_write_cues(pb, mkv->cues, s->nb_streams);
 +                cuespos = mkv_write_cues(pb, mkv->cues, mkv->tracks, s->nb_streams);
              }
  
              ret = mkv_add_seekhead_entry(mkv->main_seekhead, MATROSKA_ID_CUES, cuespos);
@@@ -1674,32 -1463,6 +1682,32 @@@ static int mkv_query_codec(enum AVCodec
      return 0;
  }
  
 +static const AVCodecTag additional_audio_tags[] = {
 +    { AV_CODEC_ID_ALAC,      0XFFFFFFFF },
 +    { AV_CODEC_ID_EAC3,      0XFFFFFFFF },
 +    { AV_CODEC_ID_MLP,       0xFFFFFFFF },
 +    { AV_CODEC_ID_OPUS,      0xFFFFFFFF },
 +    { AV_CODEC_ID_PCM_S16BE, 0xFFFFFFFF },
 +    { AV_CODEC_ID_PCM_S24BE, 0xFFFFFFFF },
 +    { AV_CODEC_ID_PCM_S32BE, 0xFFFFFFFF },
 +    { AV_CODEC_ID_QDM2,      0xFFFFFFFF },
 +    { AV_CODEC_ID_RA_144,    0xFFFFFFFF },
 +    { AV_CODEC_ID_RA_288,    0xFFFFFFFF },
 +    { AV_CODEC_ID_COOK,      0xFFFFFFFF },
 +    { AV_CODEC_ID_TRUEHD,    0xFFFFFFFF },
 +    { AV_CODEC_ID_NONE,      0xFFFFFFFF }
 +};
 +
 +static const AVCodecTag additional_video_tags[] = {
 +    { AV_CODEC_ID_PRORES,    0xFFFFFFFF },
 +    { AV_CODEC_ID_RV10,      0xFFFFFFFF },
 +    { AV_CODEC_ID_RV20,      0xFFFFFFFF },
 +    { AV_CODEC_ID_RV30,      0xFFFFFFFF },
 +    { AV_CODEC_ID_RV40,      0xFFFFFFFF },
 +    { AV_CODEC_ID_VP9,       0xFFFFFFFF },
 +    { AV_CODEC_ID_NONE,      0xFFFFFFFF }
 +};
 +
  #define OFFSET(x) offsetof(MatroskaMuxContext, x)
  #define FLAGS AV_OPT_FLAG_ENCODING_PARAM
  static const AVOption options[] = {
@@@ -1733,14 -1496,9 +1741,14 @@@ AVOutputFormat ff_matroska_muxer = 
      .flags             = AVFMT_GLOBALHEADER | AVFMT_VARIABLE_FPS |
                           AVFMT_TS_NONSTRICT | AVFMT_ALLOW_FLUSH,
      .codec_tag         = (const AVCodecTag* const []){
 -         ff_codec_bmp_tags, ff_codec_wav_tags, 0
 +         ff_codec_bmp_tags, ff_codec_wav_tags,
 +         additional_audio_tags, additional_video_tags, 0
      },
 +#if FF_API_ASS_SSA
      .subtitle_codec    = AV_CODEC_ID_SSA,
 +#else
 +    .subtitle_codec    = AV_CODEC_ID_ASS,
 +#endif
      .query_codec       = mkv_query_codec,
      .priv_class        = &matroska_class,
  };
@@@ -1762,7 -1520,6 +1770,7 @@@ AVOutputFormat ff_webm_muxer = 
      .priv_data_size    = sizeof(MatroskaMuxContext),
      .audio_codec       = AV_CODEC_ID_VORBIS,
      .video_codec       = AV_CODEC_ID_VP8,
 +    .subtitle_codec    = AV_CODEC_ID_WEBVTT,
      .write_header      = mkv_write_header,
      .write_packet      = mkv_write_flush_packet,
      .write_trailer     = mkv_write_trailer,
@@@ -1781,7 -1538,7 +1789,7 @@@ static const AVClass mka_class = 
  };
  AVOutputFormat ff_matroska_audio_muxer = {
      .name              = "matroska",
 -    .long_name         = NULL_IF_CONFIG_SMALL("Matroska"),
 +    .long_name         = NULL_IF_CONFIG_SMALL("Matroska Audio"),
      .mime_type         = "audio/x-matroska",
      .extensions        = "mka",
      .priv_data_size    = sizeof(MatroskaMuxContext),
      .write_trailer     = mkv_write_trailer,
      .flags             = AVFMT_GLOBALHEADER | AVFMT_TS_NONSTRICT |
                           AVFMT_ALLOW_FLUSH,
 -    .codec_tag         = (const AVCodecTag* const []){ ff_codec_wav_tags, 0 },
 +    .codec_tag         = (const AVCodecTag* const []){
 +        ff_codec_wav_tags, additional_audio_tags, 0
 +    },
      .priv_class        = &mka_class,
  };
  #endif