Merge commit '12db891dcd57b305d6e3c1e22ea8204dc26edda1'
authorMichael Niedermayer <michaelni@gmx.at>
Sun, 16 Jun 2013 21:58:59 +0000 (23:58 +0200)
committerMichael Niedermayer <michaelni@gmx.at>
Sun, 16 Jun 2013 21:58:59 +0000 (23:58 +0200)
* commit '12db891dcd57b305d6e3c1e22ea8204dc26edda1':
  avf: move ff_write_chained to mux.c

Conflicts:
libavformat/mux.c
libavformat/utils.c

Merged-by: Michael Niedermayer <michaelni@gmx.at>
1  2 
libavformat/mux.c
libavformat/utils.c

diff --combined libavformat/mux.c
@@@ -1,21 -1,21 +1,21 @@@
  /*
 - * muxing functions for use within Libav
 + * muxing functions for use within FFmpeg
   * Copyright (c) 2000, 2001, 2002 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
   */
  
@@@ -27,7 -27,6 +27,7 @@@
  #include "libavutil/opt.h"
  #include "libavutil/dict.h"
  #include "libavutil/pixdesc.h"
 +#include "libavutil/timestamp.h"
  #include "metadata.h"
  #include "id3v2.h"
  #include "libavutil/avassert.h"
@@@ -48,7 -47,7 +48,7 @@@
  
  /**
   * @file
 - * muxing functions for use within Libav
 + * muxing functions for use within libavformat
   */
  
  /* fraction handling */
@@@ -101,88 -100,6 +101,88 @@@ static void frac_add(AVFrac *f, int64_
      f->num = num;
  }
  
 +AVRational ff_choose_timebase(AVFormatContext *s, AVStream *st, int min_precission)
 +{
 +    AVRational q;
 +    int j;
 +
 +    if (st->codec->codec_type == AVMEDIA_TYPE_AUDIO) {
 +        q = (AVRational){1, st->codec->sample_rate};
 +    } else {
 +        q = st->codec->time_base;
 +    }
 +    for (j=2; j<14; j+= 1+(j>2))
 +        while (q.den / q.num < min_precission && q.num % j == 0)
 +            q.num /= j;
 +    while (q.den / q.num < min_precission && q.den < (1<<24))
 +        q.den <<= 1;
 +
 +    return q;
 +}
 +
 +int avformat_alloc_output_context2(AVFormatContext **avctx, AVOutputFormat *oformat,
 +                                   const char *format, const char *filename)
 +{
 +    AVFormatContext *s = avformat_alloc_context();
 +    int ret = 0;
 +
 +    *avctx = NULL;
 +    if (!s)
 +        goto nomem;
 +
 +    if (!oformat) {
 +        if (format) {
 +            oformat = av_guess_format(format, NULL, NULL);
 +            if (!oformat) {
 +                av_log(s, AV_LOG_ERROR, "Requested output format '%s' is not a suitable output format\n", format);
 +                ret = AVERROR(EINVAL);
 +                goto error;
 +            }
 +        } else {
 +            oformat = av_guess_format(NULL, filename, NULL);
 +            if (!oformat) {
 +                ret = AVERROR(EINVAL);
 +                av_log(s, AV_LOG_ERROR, "Unable to find a suitable output format for '%s'\n",
 +                       filename);
 +                goto error;
 +            }
 +        }
 +    }
 +
 +    s->oformat = oformat;
 +    if (s->oformat->priv_data_size > 0) {
 +        s->priv_data = av_mallocz(s->oformat->priv_data_size);
 +        if (!s->priv_data)
 +            goto nomem;
 +        if (s->oformat->priv_class) {
 +            *(const AVClass**)s->priv_data= s->oformat->priv_class;
 +            av_opt_set_defaults(s->priv_data);
 +        }
 +    } else
 +        s->priv_data = NULL;
 +
 +    if (filename)
 +        av_strlcpy(s->filename, filename, sizeof(s->filename));
 +    *avctx = s;
 +    return 0;
 +nomem:
 +    av_log(s, AV_LOG_ERROR, "Out of memory\n");
 +    ret = AVERROR(ENOMEM);
 +error:
 +    avformat_free_context(s);
 +    return ret;
 +}
 +
 +#if FF_API_ALLOC_OUTPUT_CONTEXT
 +AVFormatContext *avformat_alloc_output_context(const char *format,
 +                                               AVOutputFormat *oformat, const char *filename)
 +{
 +    AVFormatContext *avctx;
 +    int ret = avformat_alloc_output_context2(&avctx, oformat, format, filename);
 +    return ret < 0 ? NULL : avctx;
 +}
 +#endif
 +
  static int validate_codec_tag(AVFormatContext *s, AVStream *st)
  {
      const AVCodecTag *avctag;
@@@ -230,9 -147,6 +230,9 @@@ static int init_muxer(AVFormatContext *
  
      if ((ret = av_opt_set_dict(s, &tmp)) < 0)
          goto fail;
 +    if (s->priv_data && s->oformat->priv_class && *(const AVClass**)s->priv_data==s->oformat->priv_class &&
 +        (ret = av_opt_set_dict(s->priv_data, &tmp)) < 0)
 +        goto fail;
  
      // some sanity checks
      if (s->nb_streams == 0 && !(of->flags & AVFMT_NOSTREAMS)) {
                  ret = AVERROR(EINVAL);
                  goto fail;
              }
 -
 -            if (av_cmp_q(st->sample_aspect_ratio,
 -                         codec->sample_aspect_ratio)) {
 +            if (av_cmp_q(st->sample_aspect_ratio, codec->sample_aspect_ratio)
 +                && FFABS(av_q2d(st->sample_aspect_ratio) - av_q2d(codec->sample_aspect_ratio)) > 0.004*av_q2d(st->sample_aspect_ratio)
 +            ) {
                  av_log(s, AV_LOG_ERROR, "Aspect ratio mismatch between muxer "
                                          "(%d/%d) and encoder layer (%d/%d)\n",
                         st->sample_aspect_ratio.num, st->sample_aspect_ratio.den,
          }
  
          if (of->codec_tag) {
 -            if (codec->codec_tag &&
 -                codec->codec_id == AV_CODEC_ID_RAWVIDEO &&
 -                !av_codec_get_tag(of->codec_tag, codec->codec_id) &&
 -                !validate_codec_tag(s, st)) {
 +            if (   codec->codec_tag
 +                && codec->codec_id == AV_CODEC_ID_RAWVIDEO
 +                && (   av_codec_get_tag(of->codec_tag, codec->codec_id) == 0
 +                    || av_codec_get_tag(of->codec_tag, codec->codec_id) == MKTAG('r', 'a', 'w', ' '))
 +                && !validate_codec_tag(s, st)) {
                  // the current rawvideo encoding system ends up setting
 -                // the wrong codec_tag for avi, we override it here
 +                // the wrong codec_tag for avi/mov, we override it here
                  codec->codec_tag = 0;
              }
              if (codec->codec_tag) {
                  if (!validate_codec_tag(s, st)) {
 -                    char tagbuf[32];
 +                    char tagbuf[32], cortag[32];
                      av_get_codec_tag_string(tagbuf, sizeof(tagbuf), codec->codec_tag);
 +                    av_get_codec_tag_string(cortag, sizeof(cortag), av_codec_get_tag(s->oformat->codec_tag, codec->codec_id));
                      av_log(s, AV_LOG_ERROR,
 -                           "Tag %s/0x%08x incompatible with output codec id '%d'\n",
 -                           tagbuf, codec->codec_tag, codec->codec_id);
 +                           "Tag %s/0x%08x incompatible with output codec id '%d' (%s)\n",
 +                           tagbuf, codec->codec_tag, codec->codec_id, cortag);
                      ret = AVERROR_INVALIDDATA;
                      goto fail;
                  }
@@@ -387,8 -299,6 +387,8 @@@ int avformat_write_header(AVFormatConte
  
      if (s->oformat->write_header) {
          ret = s->oformat->write_header(s);
 +        if (ret >= 0 && s->pb && s->pb->error < 0)
 +            ret = s->pb->error;
          if (ret < 0)
              return ret;
      }
      if ((ret = init_pts(s)) < 0)
          return ret;
  
 +    if (s->avoid_negative_ts < 0) {
 +        if (s->oformat->flags & (AVFMT_TS_NEGATIVE | AVFMT_NOTIMESTAMPS)) {
 +            s->avoid_negative_ts = 0;
 +        } else
 +            s->avoid_negative_ts = 1;
 +    }
 +
      return 0;
  }
  
  //FIXME merge with compute_pkt_fields
  static int compute_pkt_fields2(AVFormatContext *s, AVStream *st, AVPacket *pkt)
  {
 -    int delay = FFMAX(st->codec->has_b_frames, !!st->codec->max_b_frames);
 +    int delay = FFMAX(st->codec->has_b_frames, st->codec->max_b_frames > 0);
      int num, den, frame_size, i;
  
 -    av_dlog(s, "compute_pkt_fields2: pts:%" PRId64 " dts:%" PRId64 " cur_dts:%" PRId64 " b:%d size:%d st:%d\n",
 -            pkt->pts, pkt->dts, st->cur_dts, delay, pkt->size, pkt->stream_index);
 -
 -/*    if(pkt->pts == AV_NOPTS_VALUE && pkt->dts == AV_NOPTS_VALUE)
 - *      return AVERROR(EINVAL);*/
 +    av_dlog(s, "compute_pkt_fields2: pts:%s dts:%s cur_dts:%s b:%d size:%d st:%d\n",
 +            av_ts2str(pkt->pts), av_ts2str(pkt->dts), av_ts2str(st->cur_dts), delay, pkt->size, pkt->stream_index);
  
      /* duration field */
      if (pkt->duration == 0) {
  
      //XXX/FIXME this is a temporary hack until all encoders output pts
      if ((pkt->pts == 0 || pkt->pts == AV_NOPTS_VALUE) && pkt->dts == AV_NOPTS_VALUE && !delay) {
 +        static int warned;
 +        if (!warned) {
 +            av_log(s, AV_LOG_WARNING, "Encoder did not produce proper pts, making some up.\n");
 +            warned = 1;
 +        }
          pkt->dts =
  //        pkt->pts= st->cur_dts;
              pkt->pts = st->pts.val;
          ((!(s->oformat->flags & AVFMT_TS_NONSTRICT) &&
            st->cur_dts >= pkt->dts) || st->cur_dts > pkt->dts)) {
          av_log(s, AV_LOG_ERROR,
 -               "Application provided invalid, non monotonically increasing dts to muxer in stream %d: %" PRId64 " >= %" PRId64 "\n",
 -               st->index, st->cur_dts, pkt->dts);
 +               "Application provided invalid, non monotonically increasing dts to muxer in stream %d: %s >= %s\n",
 +               st->index, av_ts2str(st->cur_dts), av_ts2str(pkt->dts));
          return AVERROR(EINVAL);
      }
      if (pkt->dts != AV_NOPTS_VALUE && pkt->pts != AV_NOPTS_VALUE && pkt->pts < pkt->dts) {
 -        av_log(s, AV_LOG_ERROR, "pts < dts in stream %d\n", st->index);
 +        av_log(s, AV_LOG_ERROR, "pts (%s) < dts (%s) in stream %d\n",
 +               av_ts2str(pkt->pts), av_ts2str(pkt->dts), st->index);
          return AVERROR(EINVAL);
      }
  
 -    av_dlog(s, "av_write_frame: pts2:%"PRId64" dts2:%"PRId64"\n",
 -            pkt->pts, pkt->dts);
 +    av_dlog(s, "av_write_frame: pts2:%s dts2:%s\n",
 +            av_ts2str(pkt->pts), av_ts2str(pkt->dts));
      st->cur_dts = pkt->dts;
      st->pts.val = pkt->dts;
  
      return 0;
  }
  
 -/*
 +/**
 + * Make timestamps non negative, move side data from payload to internal struct, call muxer, and restore
 + * sidedata.
 + *
   * FIXME: this function should NEVER get undefined pts/dts beside when the
   * AVFMT_NOTIMESTAMPS is set.
   * Those additional safety checks should be dropped once the correct checks
   * are set in the callers.
   */
 -
  static int write_packet(AVFormatContext *s, AVPacket *pkt)
  {
 -    if (!(s->oformat->flags & (AVFMT_TS_NEGATIVE | AVFMT_NOTIMESTAMPS))) {
 -        AVRational time_base = s->streams[pkt->stream_index]->time_base;
 -        int64_t offset = 0;
 +    int ret, did_split;
  
 -        if (!s->offset && pkt->dts != AV_NOPTS_VALUE && pkt->dts < 0) {
 +    if (s->avoid_negative_ts > 0) {
 +        AVStream *st = s->streams[pkt->stream_index];
 +        int64_t offset = st->mux_ts_offset;
 +
 +        if (pkt->dts < 0 && pkt->dts != AV_NOPTS_VALUE && !s->offset) {
              s->offset = -pkt->dts;
 -            s->offset_timebase = time_base;
 +            s->offset_timebase = st->time_base;
 +        }
 +
 +        if (s->offset && !offset) {
 +            offset = st->mux_ts_offset =
 +                av_rescale_q_rnd(s->offset,
 +                                 s->offset_timebase,
 +                                 st->time_base,
 +                                 AV_ROUND_UP);
          }
 -        if (s->offset)
 -            offset = av_rescale_q(s->offset, s->offset_timebase, time_base);
  
          if (pkt->dts != AV_NOPTS_VALUE)
              pkt->dts += offset;
          if (pkt->pts != AV_NOPTS_VALUE)
              pkt->pts += offset;
 +
 +        av_assert2(pkt->dts == AV_NOPTS_VALUE || pkt->dts >= 0);
      }
 -    return s->oformat->write_packet(s, pkt);
 +
 +    did_split = av_packet_split_side_data(pkt);
 +    ret = s->oformat->write_packet(s, pkt);
 +    if (s->flush_packets && s->pb && s->pb->error >= 0)
 +        avio_flush(s->pb);
 +    if (did_split)
 +        av_packet_merge_side_data(pkt);
 +    return ret;
  }
  
  int av_write_frame(AVFormatContext *s, AVPacket *pkt)
      int ret;
  
      if (!pkt) {
 -        if (s->oformat->flags & AVFMT_ALLOW_FLUSH)
 -            return s->oformat->write_packet(s, pkt);
 +        if (s->oformat->flags & AVFMT_ALLOW_FLUSH) {
 +            ret = s->oformat->write_packet(s, NULL);
 +            if (s->flush_packets && s->pb && s->pb->error >= 0)
 +                avio_flush(s->pb);
 +            if (ret >= 0 && s->pb && s->pb->error < 0)
 +                ret = s->pb->error;
 +            return ret;
 +        }
          return 1;
      }
  
          return ret;
  
      ret = write_packet(s, pkt);
 +    if (ret >= 0 && s->pb && s->pb->error < 0)
 +        ret = s->pb->error;
  
      if (ret >= 0)
          s->streams[pkt->stream_index]->nb_frames++;
      return ret;
  }
  
 -void ff_interleave_add_packet(AVFormatContext *s, AVPacket *pkt,
 +#define CHUNK_START 0x1000
 +
 +int ff_interleave_add_packet(AVFormatContext *s, AVPacket *pkt,
                                int (*compare)(AVFormatContext *, AVPacket *, AVPacket *))
  {
      AVPacketList **next_point, *this_pktl;
 +    AVStream *st   = s->streams[pkt->stream_index];
 +    int chunked    = s->max_chunk_size || s->max_chunk_duration;
  
      this_pktl      = av_mallocz(sizeof(AVPacketList));
 +    if (!this_pktl)
 +        return AVERROR(ENOMEM);
      this_pktl->pkt = *pkt;
  #if FF_API_DESTRUCT_PACKET
      pkt->destruct  = NULL;           // do not free original but only the copy
  #endif
      pkt->buf       = NULL;
 -    av_dup_packet(&this_pktl->pkt);  // duplicate the packet if it uses non-alloced memory
 +    av_dup_packet(&this_pktl->pkt);  // duplicate the packet if it uses non-allocated memory
 +    av_copy_packet_side_data(&this_pktl->pkt, &this_pktl->pkt); // copy side data
  
      if (s->streams[pkt->stream_index]->last_in_packet_buffer) {
 -        next_point = &(s->streams[pkt->stream_index]->last_in_packet_buffer->next);
 -    } else
 +        next_point = &(st->last_in_packet_buffer->next);
 +    } else {
          next_point = &s->packet_buffer;
 +    }
  
 +    if (chunked) {
 +        uint64_t max= av_rescale_q_rnd(s->max_chunk_duration, AV_TIME_BASE_Q, st->time_base, AV_ROUND_UP);
 +        st->interleaver_chunk_size     += pkt->size;
 +        st->interleaver_chunk_duration += pkt->duration;
 +        if (   (s->max_chunk_size && st->interleaver_chunk_size > s->max_chunk_size)
 +            || (max && st->interleaver_chunk_duration           > max)) {
 +            st->interleaver_chunk_size      = 0;
 +            this_pktl->pkt.flags |= CHUNK_START;
 +            if (max && st->interleaver_chunk_duration > max) {
 +                int64_t syncoffset = (st->codec->codec_type == AVMEDIA_TYPE_VIDEO)*max/2;
 +                int64_t syncto = av_rescale(pkt->dts + syncoffset, 1, max)*max - syncoffset;
 +
 +                st->interleaver_chunk_duration += (pkt->dts - syncto)/8 - max;
 +            } else
 +                st->interleaver_chunk_duration = 0;
 +        }
 +    }
      if (*next_point) {
 +        if (chunked && !(this_pktl->pkt.flags & CHUNK_START))
 +            goto next_non_null;
 +
          if (compare(s, &s->packet_buffer_end->pkt, pkt)) {
 -            while (!compare(s, &(*next_point)->pkt, pkt))
 +            while (   *next_point
 +                   && ((chunked && !((*next_point)->pkt.flags&CHUNK_START))
 +                       || !compare(s, &(*next_point)->pkt, pkt)))
                  next_point = &(*next_point)->next;
 -            goto next_non_null;
 +            if (*next_point)
 +                goto next_non_null;
          } else {
              next_point = &(s->packet_buffer_end->next);
          }
      }
 -    assert(!*next_point);
 +    av_assert1(!*next_point);
  
      s->packet_buffer_end = this_pktl;
  next_non_null:
  
      s->streams[pkt->stream_index]->last_in_packet_buffer =
          *next_point                                      = this_pktl;
 +    return 0;
  }
  
  static int interleave_compare_dts(AVFormatContext *s, AVPacket *next,
      AVStream *st2 = s->streams[next->stream_index];
      int comp      = av_compare_ts(next->dts, st2->time_base, pkt->dts,
                                    st->time_base);
 +    if (s->audio_preload && ((st->codec->codec_type == AVMEDIA_TYPE_AUDIO) != (st2->codec->codec_type == AVMEDIA_TYPE_AUDIO))) {
 +        int64_t ts = av_rescale_q(pkt ->dts, st ->time_base, AV_TIME_BASE_Q) - s->audio_preload*(st ->codec->codec_type == AVMEDIA_TYPE_AUDIO);
 +        int64_t ts2= av_rescale_q(next->dts, st2->time_base, AV_TIME_BASE_Q) - s->audio_preload*(st2->codec->codec_type == AVMEDIA_TYPE_AUDIO);
 +        if (ts == ts2) {
 +            ts= ( pkt ->dts* st->time_base.num*AV_TIME_BASE - s->audio_preload*(int64_t)(st ->codec->codec_type == AVMEDIA_TYPE_AUDIO)* st->time_base.den)*st2->time_base.den
 +               -( next->dts*st2->time_base.num*AV_TIME_BASE - s->audio_preload*(int64_t)(st2->codec->codec_type == AVMEDIA_TYPE_AUDIO)*st2->time_base.den)* st->time_base.den;
 +            ts2=0;
 +        }
 +        comp= (ts>ts2) - (ts<ts2);
 +    }
  
      if (comp == 0)
          return pkt->stream_index < next->stream_index;
@@@ -663,59 -494,27 +663,59 @@@ int ff_interleave_packet_per_dts(AVForm
                                   AVPacket *pkt, int flush)
  {
      AVPacketList *pktl;
 -    int stream_count = 0;
 -    int i;
 +    int stream_count = 0, noninterleaved_count = 0;
 +    int64_t delta_dts_max = 0;
 +    int i, ret;
  
      if (pkt) {
 -        ff_interleave_add_packet(s, pkt, interleave_compare_dts);
 +        ret = ff_interleave_add_packet(s, pkt, interleave_compare_dts);
 +        if (ret < 0)
 +            return ret;
      }
  
 -    for (i = 0; i < s->nb_streams; i++)
 -        stream_count += !!s->streams[i]->last_in_packet_buffer;
 +    for (i = 0; i < s->nb_streams; i++) {
 +        if (s->streams[i]->last_in_packet_buffer) {
 +            ++stream_count;
 +        } else if (s->streams[i]->codec->codec_type == AVMEDIA_TYPE_SUBTITLE) {
 +            ++noninterleaved_count;
 +        }
 +    }
  
 -    if (stream_count && (s->nb_streams == stream_count || flush)) {
 +    if (s->nb_streams == stream_count) {
 +        flush = 1;
 +    } else if (!flush) {
 +        for (i=0; i < s->nb_streams; i++) {
 +            if (s->streams[i]->last_in_packet_buffer) {
 +                int64_t delta_dts =
 +                    av_rescale_q(s->streams[i]->last_in_packet_buffer->pkt.dts,
 +                                s->streams[i]->time_base,
 +                                AV_TIME_BASE_Q) -
 +                    av_rescale_q(s->packet_buffer->pkt.dts,
 +                                s->streams[s->packet_buffer->pkt.stream_index]->time_base,
 +                                AV_TIME_BASE_Q);
 +                delta_dts_max= FFMAX(delta_dts_max, delta_dts);
 +            }
 +        }
 +        if (s->nb_streams == stream_count+noninterleaved_count &&
 +           delta_dts_max > 20*AV_TIME_BASE) {
 +            av_log(s, AV_LOG_DEBUG, "flushing with %d noninterleaved\n", noninterleaved_count);
 +            flush = 1;
 +        }
 +    }
 +    if (stream_count && flush) {
 +        AVStream *st;
          pktl = s->packet_buffer;
          *out = pktl->pkt;
 +        st   = s->streams[out->stream_index];
  
          s->packet_buffer = pktl->next;
          if (!s->packet_buffer)
              s->packet_buffer_end = NULL;
  
 -        if (s->streams[out->stream_index]->last_in_packet_buffer == pktl)
 -            s->streams[out->stream_index]->last_in_packet_buffer = NULL;
 +        if (st->last_in_packet_buffer == pktl)
 +            st->last_in_packet_buffer = NULL;
          av_freep(&pktl);
 +
          return 1;
      } else {
          av_init_packet(out);
@@@ -754,8 -553,8 +754,8 @@@ int av_interleaved_write_frame(AVFormat
          if (st->codec->codec_type == AVMEDIA_TYPE_AUDIO && pkt->size == 0)
              return 0;
  
 -        av_dlog(s, "av_interleaved_write_frame size:%d dts:%" PRId64 " pts:%" PRId64 "\n",
 -                pkt->size, pkt->dts, pkt->pts);
 +        av_dlog(s, "av_interleaved_write_frame size:%d dts:%s pts:%s\n",
 +                pkt->size, av_ts2str(pkt->dts), av_ts2str(pkt->pts));
          if ((ret = compute_pkt_fields2(s, st, pkt)) < 0 && !(s->oformat->flags & AVFMT_NOTIMESTAMPS))
              return ret;
  
  
          if (ret < 0)
              return ret;
 +        if(s->pb && s->pb->error)
 +            return s->pb->error;
      }
  }
  
@@@ -806,18 -603,15 +806,18 @@@ int av_write_trailer(AVFormatContext *s
  
          if (ret < 0)
              goto fail;
 +        if(s->pb && s->pb->error)
 +            goto fail;
      }
  
      if (s->oformat->write_trailer)
          ret = s->oformat->write_trailer(s);
  
 -    if (!(s->oformat->flags & AVFMT_NOFILE))
 -        avio_flush(s->pb);
 -
  fail:
 +    if (s->pb)
 +       avio_flush(s->pb);
 +    if (ret == 0)
 +       ret = s->pb ? s->pb->error : 0;
      for (i = 0; i < s->nb_streams; i++) {
          av_freep(&s->streams[i]->priv_data);
          av_freep(&s->streams[i]->index_entries);
      return ret;
  }
  
 +int av_get_output_timestamp(struct AVFormatContext *s, int stream,
 +                            int64_t *dts, int64_t *wall)
 +{
 +    if (!s->oformat || !s->oformat->get_output_timestamp)
 +        return AVERROR(ENOSYS);
 +    s->oformat->get_output_timestamp(s, stream, dts, wall);
 +    return 0;
 +}
++
+ int ff_write_chained(AVFormatContext *dst, int dst_stream, AVPacket *pkt,
+                      AVFormatContext *src)
+ {
+     AVPacket local_pkt;
+     local_pkt = *pkt;
+     local_pkt.stream_index = dst_stream;
+     if (pkt->pts != AV_NOPTS_VALUE)
+         local_pkt.pts = av_rescale_q(pkt->pts,
+                                      src->streams[pkt->stream_index]->time_base,
+                                      dst->streams[dst_stream]->time_base);
+     if (pkt->dts != AV_NOPTS_VALUE)
+         local_pkt.dts = av_rescale_q(pkt->dts,
+                                      src->streams[pkt->stream_index]->time_base,
+                                      dst->streams[dst_stream]->time_base);
++    if (pkt->duration)
++        local_pkt.duration = av_rescale_q(pkt->duration,
++                                          src->streams[pkt->stream_index]->time_base,
++                                          dst->streams[dst_stream]->time_base);
+     return av_write_frame(dst, &local_pkt);
+ }
diff --combined libavformat/utils.c
@@@ -1,21 -1,21 +1,21 @@@
  /*
 - * various utility functions for use within Libav
 + * various utility functions for use within FFmpeg
   * Copyright (c) 2000, 2001, 2002 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
   */
  
@@@ -23,9 -23,7 +23,9 @@@
  #include "avio_internal.h"
  #include "internal.h"
  #include "libavcodec/internal.h"
 +#include "libavcodec/raw.h"
  #include "libavcodec/bytestream.h"
 +#include "libavutil/avassert.h"
  #include "libavutil/opt.h"
  #include "libavutil/dict.h"
  #include "libavutil/pixdesc.h"
@@@ -36,7 -34,6 +36,7 @@@
  #include "libavutil/mathematics.h"
  #include "libavutil/parseutils.h"
  #include "libavutil/time.h"
 +#include "libavutil/timestamp.h"
  #include "riff.h"
  #include "audiointerleave.h"
  #include "url.h"
  
  /**
   * @file
 - * various utility functions for use within Libav
 + * various utility functions for use within FFmpeg
   */
  
  unsigned avformat_version(void)
  {
 +    av_assert0(LIBAVFORMAT_VERSION_MICRO >= 100);
      return LIBAVFORMAT_VERSION_INT;
  }
  
  const char *avformat_configuration(void)
  {
 -    return LIBAV_CONFIGURATION;
 +    return FFMPEG_CONFIGURATION;
  }
  
  const char *avformat_license(void)
  {
  #define LICENSE_PREFIX "libavformat license: "
 -    return LICENSE_PREFIX LIBAV_LICENSE + sizeof(LICENSE_PREFIX) - 1;
 +    return LICENSE_PREFIX FFMPEG_LICENSE + sizeof(LICENSE_PREFIX) - 1;
 +}
 +
 +#define RELATIVE_TS_BASE (INT64_MAX - (1LL<<48))
 +
 +static int is_relative(int64_t ts) {
 +    return ts > (RELATIVE_TS_BASE - (1LL<<48));
 +}
 +
 +/**
 + * Wrap a given time stamp, if there is an indication for an overflow
 + *
 + * @param st stream
 + * @param timestamp the time stamp to wrap
 + * @return resulting time stamp
 + */
 +static int64_t wrap_timestamp(AVStream *st, int64_t timestamp)
 +{
 +    if (st->pts_wrap_behavior != AV_PTS_WRAP_IGNORE &&
 +        st->pts_wrap_reference != AV_NOPTS_VALUE && timestamp != AV_NOPTS_VALUE) {
 +        if (st->pts_wrap_behavior == AV_PTS_WRAP_ADD_OFFSET &&
 +            timestamp < st->pts_wrap_reference)
 +            return timestamp + (1ULL<<st->pts_wrap_bits);
 +        else if (st->pts_wrap_behavior == AV_PTS_WRAP_SUB_OFFSET &&
 +            timestamp >= st->pts_wrap_reference)
 +            return timestamp - (1ULL<<st->pts_wrap_bits);
 +    }
 +    return timestamp;
  }
  
 +#define MAKE_ACCESSORS(str, name, type, field) \
 +    type av_##name##_get_##field(const str *s) { return s->field; } \
 +    void av_##name##_set_##field(str *s, type v) { s->field = v; }
 +
 +MAKE_ACCESSORS(AVStream, stream, AVRational, r_frame_rate)
 +
  /* an arbitrarily chosen "sane" max packet size -- 50M */
  #define SANE_CHUNK_SIZE (50000000)
  
 +int ffio_limit(AVIOContext *s, int size)
 +{
 +    if(s->maxsize>=0){
 +        int64_t remaining= s->maxsize - avio_tell(s);
 +        if(remaining < size){
 +            int64_t newsize= avio_size(s);
 +            if(!s->maxsize || s->maxsize<newsize)
 +                s->maxsize= newsize - !newsize;
 +            remaining= s->maxsize - avio_tell(s);
 +            remaining= FFMAX(remaining, 0);
 +        }
 +
 +        if(s->maxsize>=0 && remaining+1 < size){
 +            av_log(NULL, remaining ? AV_LOG_ERROR : AV_LOG_DEBUG, "Truncating packet of size %d to %"PRId64"\n", size, remaining+1);
 +            size= remaining+1;
 +        }
 +    }
 +    return size;
 +}
 +
  /*
   * Read the data in sane-sized chunks and append to pkt.
   * Return the number of bytes read or an error.
   */
  static int append_packet_chunked(AVIOContext *s, AVPacket *pkt, int size)
  {
 -    int64_t chunk_size = size;
 -    int orig_pos       = pkt->pos; // av_grow_packet might reset pos
 +    int64_t orig_pos   = pkt->pos; // av_grow_packet might reset pos
      int orig_size      = pkt->size;
 -    int ret = 0;
 +    int ret;
  
      do {
          int prev_size = pkt->size;
           * When the caller requests a lot of data, limit it to the amount left
           * in file or SANE_CHUNK_SIZE when it is not known
           */
 -        if (size > SANE_CHUNK_SIZE) {
 -            int64_t filesize = avio_size(s) - avio_tell(s);
 -            chunk_size = FFMAX(filesize, SANE_CHUNK_SIZE);
 +        read_size = size;
 +        if (read_size > SANE_CHUNK_SIZE/10) {
 +            read_size = ffio_limit(s, read_size);
 +            // If filesize/maxsize is unknown, limit to SANE_CHUNK_SIZE
 +            if (s->maxsize < 0)
 +                read_size = FFMIN(read_size, SANE_CHUNK_SIZE);
          }
 -        read_size = FFMIN(size, chunk_size);
  
          ret = av_grow_packet(pkt, read_size);
          if (ret < 0)
  
          size -= read_size;
      } while (size > 0);
 +    if (size > 0)
 +        pkt->flags |= AV_PKT_FLAG_CORRUPT;
  
      pkt->pos = orig_pos;
      if (!pkt->size)
@@@ -197,23 -137,19 +197,23 @@@ int av_filename_number_test(const char 
      return filename && (av_get_frame_filename(buf, sizeof(buf), filename, 1)>=0);
  }
  
 -AVInputFormat *av_probe_input_format2(AVProbeData *pd, int is_opened, int *score_max)
 +AVInputFormat *av_probe_input_format3(AVProbeData *pd, int is_opened, int *score_ret)
  {
      AVProbeData lpd = *pd;
      AVInputFormat *fmt1 = NULL, *fmt;
 -    int score, id3 = 0;
 +    int score, nodat = 0, score_max=0;
 +    const static uint8_t zerobuffer[AVPROBE_PADDING_SIZE];
 +
 +    if (!lpd.buf)
 +        lpd.buf = zerobuffer;
  
      if (lpd.buf_size > 10 && ff_id3v2_match(lpd.buf, ID3v2_DEFAULT_MAGIC)) {
          int id3len = ff_id3v2_tag_len(lpd.buf);
          if (lpd.buf_size > id3len + 16) {
              lpd.buf += id3len;
              lpd.buf_size -= id3len;
 -        }
 -        id3 = 1;
 +        }else
 +            nodat = 1;
      }
  
      fmt = NULL;
          score = 0;
          if (fmt1->read_probe) {
              score = fmt1->read_probe(&lpd);
 +            if(fmt1->extensions && av_match_ext(lpd.filename, fmt1->extensions))
 +                score = FFMAX(score, nodat ? AVPROBE_SCORE_EXTENSION / 2 - 1 : 1);
          } else if (fmt1->extensions) {
              if (av_match_ext(lpd.filename, fmt1->extensions)) {
                  score = AVPROBE_SCORE_EXTENSION;
              }
          }
 -        if (score > *score_max) {
 -            *score_max = score;
 +        if (score > score_max) {
 +            score_max = score;
              fmt = fmt1;
 -        }else if (score == *score_max)
 +        }else if (score == score_max)
              fmt = NULL;
      }
 -
 -    /* a hack for files with huge id3v2 tags -- try to guess by file extension. */
 -    if (!fmt && is_opened && *score_max < AVPROBE_SCORE_EXTENSION / 2) {
 -        while ((fmt = av_iformat_next(fmt)))
 -            if (fmt->extensions && av_match_ext(lpd.filename, fmt->extensions)) {
 -                *score_max = AVPROBE_SCORE_EXTENSION / 2;
 -                break;
 -            }
 -    }
 -
 -    if (!fmt && id3 && *score_max < AVPROBE_SCORE_EXTENSION / 2 - 1) {
 -        while ((fmt = av_iformat_next(fmt)))
 -            if (fmt->extensions && av_match_ext("mp3", fmt->extensions)) {
 -                *score_max = AVPROBE_SCORE_EXTENSION / 2 - 1;
 -                break;
 -            }
 -    }
 +    if(nodat)
 +        score_max = FFMIN(AVPROBE_SCORE_EXTENSION / 2 - 1, score_max);
 +    *score_ret= score_max;
  
      return fmt;
  }
  
 +AVInputFormat *av_probe_input_format2(AVProbeData *pd, int is_opened, int *score_max)
 +{
 +    int score_ret;
 +    AVInputFormat *fmt= av_probe_input_format3(pd, is_opened, &score_ret);
 +    if(score_ret > *score_max){
 +        *score_max= score_ret;
 +        return fmt;
 +    }else
 +        return NULL;
 +}
 +
  AVInputFormat *av_probe_input_format(AVProbeData *pd, int is_opened){
      int score=0;
      return av_probe_input_format2(pd, is_opened, &score);
  }
  
 -static int set_codec_from_probe_data(AVFormatContext *s, AVStream *st, AVProbeData *pd, int score)
 +static int set_codec_from_probe_data(AVFormatContext *s, AVStream *st, AVProbeData *pd)
  {
      static const struct {
          const char *name; enum AVCodecID id; enum AVMediaType type;
          { "dts"      , AV_CODEC_ID_DTS       , AVMEDIA_TYPE_AUDIO },
          { "eac3"     , AV_CODEC_ID_EAC3      , AVMEDIA_TYPE_AUDIO },
          { "h264"     , AV_CODEC_ID_H264      , AVMEDIA_TYPE_VIDEO },
 +        { "loas"     , AV_CODEC_ID_AAC_LATM  , AVMEDIA_TYPE_AUDIO },
          { "m4v"      , AV_CODEC_ID_MPEG4     , AVMEDIA_TYPE_VIDEO },
          { "mp3"      , AV_CODEC_ID_MP3       , AVMEDIA_TYPE_AUDIO },
          { "mpegvideo", AV_CODEC_ID_MPEG2VIDEO, AVMEDIA_TYPE_VIDEO },
          { 0 }
      };
 -    AVInputFormat *fmt = av_probe_input_format2(pd, 1, &score);
 +    int score;
 +    AVInputFormat *fmt = av_probe_input_format3(pd, 1, &score);
  
 -    if (fmt) {
 +    if (fmt && st->request_probe <= score) {
          int i;
          av_log(s, AV_LOG_DEBUG, "Probe with size=%d, packets=%d detected %s with score=%d\n",
                 pd->buf_size, MAX_PROBE_PACKETS - st->probe_packets, fmt->name, score);
              }
          }
      }
 -    return !!fmt;
 +    return score;
  }
  
  /************************************************************/
  /* input media file */
  
 +int av_demuxer_open(AVFormatContext *ic){
 +    int err;
 +
 +    if (ic->iformat->read_header) {
 +        err = ic->iformat->read_header(ic);
 +        if (err < 0)
 +            return err;
 +    }
 +
 +    if (ic->pb && !ic->data_offset)
 +        ic->data_offset = avio_tell(ic->pb);
 +
 +    return 0;
 +}
 +
 +
  /** size of probe buffer, for guessing file type from file contents */
  #define PROBE_BUF_MIN 2048
  #define PROBE_BUF_MAX (1<<20)
@@@ -322,16 -241,13 +322,16 @@@ int av_probe_input_buffer(AVIOContext *
  {
      AVProbeData pd = { filename ? filename : "", NULL, -offset };
      unsigned char *buf = NULL;
 -    int ret = 0, probe_size;
 +    uint8_t *mime_type;
 +    int ret = 0, probe_size, buf_offset = 0;
  
      if (!max_probe_size) {
          max_probe_size = PROBE_BUF_MAX;
      } else if (max_probe_size > PROBE_BUF_MAX) {
          max_probe_size = PROBE_BUF_MAX;
      } else if (max_probe_size < PROBE_BUF_MIN) {
 +        av_log(logctx, AV_LOG_ERROR,
 +               "Specified probe size value %u cannot be < %u\n", max_probe_size, PROBE_BUF_MIN);
          return AVERROR(EINVAL);
      }
  
          return AVERROR(EINVAL);
      }
  
 +    if (!*fmt && pb->av_class && av_opt_get(pb, "mime_type", AV_OPT_SEARCH_CHILDREN, &mime_type) >= 0 && mime_type) {
 +        if (!av_strcasecmp(mime_type, "audio/aacp")) {
 +            *fmt = av_find_input_format("aac");
 +        }
 +        av_freep(&mime_type);
 +    }
 +
      for(probe_size= PROBE_BUF_MIN; probe_size<=max_probe_size && !*fmt;
          probe_size = FFMIN(probe_size<<1, FFMAX(max_probe_size, probe_size+1))) {
 -        int score = probe_size < max_probe_size ? AVPROBE_SCORE_MAX/4 : 0;
 -        int buf_offset = (probe_size == PROBE_BUF_MIN) ? 0 : probe_size>>1;
 +        int score = probe_size < max_probe_size ? AVPROBE_SCORE_RETRY : 0;
 +        void *buftmp;
  
          if (probe_size < offset) {
              continue;
          }
  
          /* read probe data */
 -        buf = av_realloc(buf, probe_size + AVPROBE_PADDING_SIZE);
 +        buftmp = av_realloc(buf, probe_size + AVPROBE_PADDING_SIZE);
 +        if(!buftmp){
 +            av_free(buf);
 +            return AVERROR(ENOMEM);
 +        }
 +        buf=buftmp;
          if ((ret = avio_read(pb, buf + buf_offset, probe_size - buf_offset)) < 0) {
              /* fail if error was not end of file, otherwise, lower score */
              if (ret != AVERROR_EOF) {
              score = 0;
              ret = 0;            /* error was end of file, nothing read */
          }
 -        pd.buf_size += ret;
 +        pd.buf_size = buf_offset += ret;
          pd.buf = &buf[offset];
  
          memset(pd.buf + pd.buf_size, 0, AVPROBE_PADDING_SIZE);
          /* guess file format */
          *fmt = av_probe_input_format2(&pd, 1, &score);
          if(*fmt){
 -            if(score <= AVPROBE_SCORE_MAX/4){ //this can only be true in the last iteration
 -                av_log(logctx, AV_LOG_WARNING, "Format detected only with low score of %d, misdetection possible!\n", score);
 +            if(score <= AVPROBE_SCORE_RETRY){ //this can only be true in the last iteration
 +                av_log(logctx, AV_LOG_WARNING, "Format %s detected only with low score of %d, misdetection possible!\n", (*fmt)->name, score);
              }else
 -                av_log(logctx, AV_LOG_DEBUG, "Probed with size=%d and score=%d\n", probe_size, score);
 +                av_log(logctx, AV_LOG_DEBUG, "Format %s probed with size=%d and score=%d\n", (*fmt)->name, probe_size, score);
          }
      }
  
      }
  
      /* rewind. reuse probe buffer to avoid seeking */
 -    if ((ret = ffio_rewind_with_probe_data(pb, buf, pd.buf_size)) < 0)
 -        av_free(buf);
 +    ret = ffio_rewind_with_probe_data(pb, &buf, pd.buf_size);
  
      return ret;
  }
@@@ -402,23 -307,21 +402,23 @@@ static int init_input(AVFormatContext *
  {
      int ret;
      AVProbeData pd = {filename, NULL, 0};
 +    int score = AVPROBE_SCORE_RETRY;
  
      if (s->pb) {
          s->flags |= AVFMT_FLAG_CUSTOM_IO;
          if (!s->iformat)
              return av_probe_input_buffer(s->pb, &s->iformat, filename, s, 0, s->probesize);
          else if (s->iformat->flags & AVFMT_NOFILE)
 -            return AVERROR(EINVAL);
 +            av_log(s, AV_LOG_WARNING, "Custom AVIOContext makes no sense and "
 +                                      "will be ignored with AVFMT_NOFILE format.\n");
          return 0;
      }
  
      if ( (s->iformat && s->iformat->flags & AVFMT_NOFILE) ||
 -        (!s->iformat && (s->iformat = av_probe_input_format(&pd, 0))))
 +        (!s->iformat && (s->iformat = av_probe_input_format2(&pd, 0, &score))))
          return 0;
  
 -    if ((ret = avio_open2(&s->pb, filename, AVIO_FLAG_READ,
 +    if ((ret = avio_open2(&s->pb, filename, AVIO_FLAG_READ | s->avio_flags,
                            &s->interrupt_callback, options)) < 0)
          return ret;
      if (s->iformat)
@@@ -443,7 -346,7 +443,7 @@@ static AVPacket *add_to_pktbuf(AVPacket
      return &pktl->pkt;
  }
  
 -static int queue_attached_pictures(AVFormatContext *s)
 +int avformat_queue_attached_pictures(AVFormatContext *s)
  {
      int i;
      for (i = 0; i < s->nb_streams; i++)
@@@ -468,10 -371,6 +468,10 @@@ int avformat_open_input(AVFormatContex
  
      if (!s && !(s = avformat_alloc_context()))
          return AVERROR(ENOMEM);
 +    if (!s->av_class){
 +        av_log(NULL, AV_LOG_ERROR, "Input context has not been properly allocated by avformat_alloc_context() and is not NULL either\n");
 +        return AVERROR(EINVAL);
 +    }
      if (fmt)
          s->iformat = fmt;
  
  
      if ((ret = init_input(s, filename, &tmp)) < 0)
          goto fail;
 +    avio_skip(s->pb, s->skip_initial_bytes);
  
      /* check filename in case an image number is expected */
      if (s->iformat->flags & AVFMT_NEEDNUMBER) {
      if (s->pb)
          ff_id3v2_read(s, ID3v2_DEFAULT_MAGIC, &id3v2_extra_meta);
  
 -    if (s->iformat->read_header)
 +    if (!(s->flags&AVFMT_FLAG_PRIV_OPT) && s->iformat->read_header)
          if ((ret = s->iformat->read_header(s)) < 0)
              goto fail;
  
 -    if (id3v2_extra_meta &&
 -        (ret = ff_id3v2_parse_apic(s, &id3v2_extra_meta)) < 0)
 -        goto fail;
 +    if (id3v2_extra_meta) {
 +        if (!strcmp(s->iformat->name, "mp3") || !strcmp(s->iformat->name, "aac") ||
 +            !strcmp(s->iformat->name, "tta")) {
 +            if((ret = ff_id3v2_parse_apic(s, &id3v2_extra_meta)) < 0)
 +                goto fail;
 +        } else
 +            av_log(s, AV_LOG_DEBUG, "demuxer does not support additional id3 data, skipping\n");
 +    }
      ff_id3v2_free_extra_meta(&id3v2_extra_meta);
  
 -    if ((ret = queue_attached_pictures(s)) < 0)
 +    if ((ret = avformat_queue_attached_pictures(s)) < 0)
          goto fail;
  
 -    if (s->pb && !s->data_offset)
 +    if (!(s->flags&AVFMT_FLAG_PRIV_OPT) && s->pb && !s->data_offset)
          s->data_offset = avio_tell(s->pb);
  
      s->raw_packet_buffer_remaining_size = RAW_PACKET_BUFFER_SIZE;
@@@ -555,62 -448,35 +555,62 @@@ fail
  
  /*******************************************************/
  
 +static void force_codec_ids(AVFormatContext *s, AVStream *st)
 +{
 +    switch(st->codec->codec_type){
 +    case AVMEDIA_TYPE_VIDEO:
 +        if(s->video_codec_id)   st->codec->codec_id= s->video_codec_id;
 +        break;
 +    case AVMEDIA_TYPE_AUDIO:
 +        if(s->audio_codec_id)   st->codec->codec_id= s->audio_codec_id;
 +        break;
 +    case AVMEDIA_TYPE_SUBTITLE:
 +        if(s->subtitle_codec_id)st->codec->codec_id= s->subtitle_codec_id;
 +        break;
 +    }
 +}
 +
  static void probe_codec(AVFormatContext *s, AVStream *st, const AVPacket *pkt)
  {
 -    if(st->codec->codec_id == AV_CODEC_ID_PROBE){
 +    if(st->request_probe>0){
          AVProbeData *pd = &st->probe_data;
 -        av_log(s, AV_LOG_DEBUG, "probing stream %d\n", st->index);
 +        int end;
 +        av_log(s, AV_LOG_DEBUG, "probing stream %d pp:%d\n", st->index, st->probe_packets);
          --st->probe_packets;
  
          if (pkt) {
 -            pd->buf = av_realloc(pd->buf, pd->buf_size+pkt->size+AVPROBE_PADDING_SIZE);
 +            uint8_t *new_buf = av_realloc(pd->buf, pd->buf_size+pkt->size+AVPROBE_PADDING_SIZE);
 +            if(!new_buf)
 +                goto no_packet;
 +            pd->buf = new_buf;
              memcpy(pd->buf+pd->buf_size, pkt->data, pkt->size);
              pd->buf_size += pkt->size;
              memset(pd->buf+pd->buf_size, 0, AVPROBE_PADDING_SIZE);
          } else {
 +no_packet:
              st->probe_packets = 0;
              if (!pd->buf_size) {
 -                av_log(s, AV_LOG_ERROR, "nothing to probe for stream %d\n",
 +                av_log(s, AV_LOG_WARNING, "nothing to probe for stream %d\n",
                         st->index);
 -                return;
              }
          }
  
 -        if (!st->probe_packets ||
 -            av_log2(pd->buf_size) != av_log2(pd->buf_size - pkt->size)) {
 -            set_codec_from_probe_data(s, st, pd, st->probe_packets > 0 ? AVPROBE_SCORE_MAX/4 : 0);
 -            if(st->codec->codec_id != AV_CODEC_ID_PROBE){
 +        end=    s->raw_packet_buffer_remaining_size <= 0
 +                || st->probe_packets<=0;
 +
 +        if(end || av_log2(pd->buf_size) != av_log2(pd->buf_size - pkt->size)){
 +            int score= set_codec_from_probe_data(s, st, pd);
 +            if(    (st->codec->codec_id != AV_CODEC_ID_NONE && score > AVPROBE_SCORE_RETRY)
 +                || end){
                  pd->buf_size=0;
                  av_freep(&pd->buf);
 -                av_log(s, AV_LOG_DEBUG, "probed stream %d\n", st->index);
 +                st->request_probe= -1;
 +                if(st->codec->codec_id != AV_CODEC_ID_NONE){
 +                    av_log(s, AV_LOG_DEBUG, "probed stream %d\n", st->index);
 +                }else
 +                    av_log(s, AV_LOG_WARNING, "probed stream %d failed\n", st->index);
              }
 +            force_codec_ids(s, st);
          }
      }
  }
@@@ -626,9 -492,15 +626,9 @@@ int ff_read_packet(AVFormatContext *s, 
          if (pktl) {
              *pkt = pktl->pkt;
              st = s->streams[pkt->stream_index];
 -            if (st->codec->codec_id != AV_CODEC_ID_PROBE || !st->probe_packets ||
 -                s->raw_packet_buffer_remaining_size < pkt->size) {
 -                AVProbeData *pd;
 -                if (st->probe_packets) {
 -                    probe_codec(s, st, NULL);
 -                }
 -                pd = &st->probe_data;
 -                av_freep(&pd->buf);
 -                pd->buf_size = 0;
 +            if (s->raw_packet_buffer_remaining_size <= 0)
 +                probe_codec(s, st, NULL);
 +            if(st->request_probe <= 0){
                  s->raw_packet_buffer = pktl->next;
                  s->raw_packet_buffer_remaining_size += pkt->size;
                  av_free(pktl);
                  if (st->probe_packets) {
                      probe_codec(s, st, NULL);
                  }
 +                av_assert0(st->request_probe <= 0);
              }
              continue;
          }
              continue;
          }
  
 -        st= s->streams[pkt->stream_index];
 +        if(!(s->flags & AVFMT_FLAG_KEEP_SIDE_DATA))
 +            av_packet_merge_side_data(pkt);
  
 -        switch(st->codec->codec_type){
 -        case AVMEDIA_TYPE_VIDEO:
 -            if(s->video_codec_id)   st->codec->codec_id= s->video_codec_id;
 -            break;
 -        case AVMEDIA_TYPE_AUDIO:
 -            if(s->audio_codec_id)   st->codec->codec_id= s->audio_codec_id;
 -            break;
 -        case AVMEDIA_TYPE_SUBTITLE:
 -            if(s->subtitle_codec_id)st->codec->codec_id= s->subtitle_codec_id;
 -            break;
 +        if(pkt->stream_index >= (unsigned)s->nb_streams){
 +            av_log(s, AV_LOG_ERROR, "Invalid stream index %d\n", pkt->stream_index);
 +            continue;
          }
  
 -        if(!pktl && (st->codec->codec_id != AV_CODEC_ID_PROBE ||
 -                     !st->probe_packets))
 +        st= s->streams[pkt->stream_index];
 +        pkt->dts = wrap_timestamp(st, pkt->dts);
 +        pkt->pts = wrap_timestamp(st, pkt->pts);
 +
 +        force_codec_ids(s, st);
 +
 +        /* TODO: audio: time filter; video: frame reordering (pts != dts) */
 +        if (s->use_wallclock_as_timestamps)
 +            pkt->dts = pkt->pts = av_rescale_q(av_gettime(), AV_TIME_BASE_Q, st->time_base);
 +
 +        if(!pktl && st->request_probe <= 0)
              return ret;
  
          add_to_pktbuf(&s->raw_packet_buffer, pkt, &s->raw_packet_buffer_end);
      }
  }
  
 +#if FF_API_READ_PACKET
 +int av_read_packet(AVFormatContext *s, AVPacket *pkt)
 +{
 +    return ff_read_packet(s, pkt);
 +}
 +#endif
 +
 +
  /**********************************************************/
  
 +static int determinable_frame_size(AVCodecContext *avctx)
 +{
 +    if (/*avctx->codec_id == AV_CODEC_ID_AAC ||*/
 +        avctx->codec_id == AV_CODEC_ID_MP1 ||
 +        avctx->codec_id == AV_CODEC_ID_MP2 ||
 +        avctx->codec_id == AV_CODEC_ID_MP3/* ||
 +        avctx->codec_id == AV_CODEC_ID_CELT*/)
 +        return 1;
 +    return 0;
 +}
 +
  /**
   * Get the number of samples of an audio frame. Return -1 on error.
   */
@@@ -729,13 -578,6 +729,13 @@@ int ff_get_audio_frame_size(AVCodecCont
      if (enc->frame_size > 1)
          return enc->frame_size;
  
 +    //For WMA we currently have no other means to calculate duration thus we
 +    //do it here by assuming CBR, which is true for all known cases.
 +    if(!mux && enc->bit_rate>0 && size>0 && enc->sample_rate>0 && enc->block_align>1) {
 +        if (enc->codec_id == AV_CODEC_ID_WMAV1 || enc->codec_id == AV_CODEC_ID_WMAV2)
 +            return  ((int64_t)size * 8 * enc->sample_rate) / enc->bit_rate;
 +    }
 +
      return -1;
  }
  
@@@ -752,9 -594,9 +752,9 @@@ void ff_compute_frame_duration(int *pnu
      *pden = 0;
      switch(st->codec->codec_type) {
      case AVMEDIA_TYPE_VIDEO:
 -        if (st->avg_frame_rate.num) {
 -            *pnum = st->avg_frame_rate.den;
 -            *pden = st->avg_frame_rate.num;
 +        if (st->r_frame_rate.num && !pc) {
 +            *pnum = st->r_frame_rate.den;
 +            *pden = st->r_frame_rate.num;
          } else if(st->time_base.num*1000LL > st->time_base.den) {
              *pnum = st->time_base.num;
              *pden = st->time_base.den;
      }
  }
  
 -static int is_intra_only(enum AVCodecID id)
 +static int is_intra_only(AVCodecContext *enc){
 +    const AVCodecDescriptor *desc;
 +
 +    if(enc->codec_type != AVMEDIA_TYPE_VIDEO)
 +        return 1;
 +
 +    desc = av_codec_get_codec_descriptor(enc);
 +    if (!desc) {
 +        desc = avcodec_descriptor_get(enc->codec_id);
 +        av_codec_set_codec_descriptor(enc, desc);
 +    }
 +    if (desc)
 +        return !!(desc->props & AV_CODEC_PROP_INTRA_ONLY);
 +    return 0;
 +}
 +
 +static int has_decode_delay_been_guessed(AVStream *st)
  {
 -    const AVCodecDescriptor *d = avcodec_descriptor_get(id);
 -    if (!d)
 -        return 0;
 -    if (d->type == AVMEDIA_TYPE_VIDEO && !(d->props & AV_CODEC_PROP_INTRA_ONLY))
 -        return 0;
 -    return 1;
 +    if(st->codec->codec_id != AV_CODEC_ID_H264) return 1;
 +    if(!st->info) // if we have left find_stream_info then nb_decoded_frames wont increase anymore for stream copy
 +        return 1;
 +#if CONFIG_H264_DECODER
 +    if(st->codec->has_b_frames &&
 +       avpriv_h264_has_num_reorder_frames(st->codec) == st->codec->has_b_frames)
 +        return 1;
 +#endif
 +    if(st->codec->has_b_frames<3)
 +        return st->nb_decoded_frames >= 7;
 +    else if(st->codec->has_b_frames<4)
 +        return st->nb_decoded_frames >= 18;
 +    else
 +        return st->nb_decoded_frames >= 20;
 +}
 +
 +static AVPacketList *get_next_pkt(AVFormatContext *s, AVStream *st, AVPacketList *pktl)
 +{
 +    if (pktl->next)
 +        return pktl->next;
 +    if (pktl == s->parse_queue_end)
 +        return s->packet_buffer;
 +    return NULL;
 +}
 +
 +static int update_wrap_reference(AVFormatContext *s, AVStream *st, int stream_index)
 +{
 +    if (s->correct_ts_overflow && st->pts_wrap_bits < 63 &&
 +        st->pts_wrap_reference == AV_NOPTS_VALUE && st->first_dts != AV_NOPTS_VALUE) {
 +        int i;
 +
 +        // reference time stamp should be 60 s before first time stamp
 +        int64_t pts_wrap_reference = st->first_dts - av_rescale(60, st->time_base.den, st->time_base.num);
 +        // if first time stamp is not more than 1/8 and 60s before the wrap point, subtract rather than add wrap offset
 +        int pts_wrap_behavior = (st->first_dts < (1LL<<st->pts_wrap_bits) - (1LL<<st->pts_wrap_bits-3)) ||
 +            (st->first_dts < (1LL<<st->pts_wrap_bits) - av_rescale(60, st->time_base.den, st->time_base.num)) ?
 +            AV_PTS_WRAP_ADD_OFFSET : AV_PTS_WRAP_SUB_OFFSET;
 +
 +        AVProgram *first_program = av_find_program_from_stream(s, NULL, stream_index);
 +
 +        if (!first_program) {
 +            int default_stream_index = av_find_default_stream_index(s);
 +            if (s->streams[default_stream_index]->pts_wrap_reference == AV_NOPTS_VALUE) {
 +                for (i=0; i<s->nb_streams; i++) {
 +                    s->streams[i]->pts_wrap_reference = pts_wrap_reference;
 +                    s->streams[i]->pts_wrap_behavior = pts_wrap_behavior;
 +                }
 +            }
 +            else {
 +                st->pts_wrap_reference = s->streams[default_stream_index]->pts_wrap_reference;
 +                st->pts_wrap_behavior = s->streams[default_stream_index]->pts_wrap_behavior;
 +            }
 +        }
 +        else {
 +            AVProgram *program = first_program;
 +            while (program) {
 +                if (program->pts_wrap_reference != AV_NOPTS_VALUE) {
 +                    pts_wrap_reference = program->pts_wrap_reference;
 +                    pts_wrap_behavior = program->pts_wrap_behavior;
 +                    break;
 +                }
 +                program = av_find_program_from_stream(s, program, stream_index);
 +            }
 +
 +            // update every program with differing pts_wrap_reference
 +            program = first_program;
 +            while(program) {
 +                if (program->pts_wrap_reference != pts_wrap_reference) {
 +                    for (i=0; i<program->nb_stream_indexes; i++) {
 +                        s->streams[program->stream_index[i]]->pts_wrap_reference = pts_wrap_reference;
 +                        s->streams[program->stream_index[i]]->pts_wrap_behavior = pts_wrap_behavior;
 +                    }
 +
 +                    program->pts_wrap_reference = pts_wrap_reference;
 +                    program->pts_wrap_behavior = pts_wrap_behavior;
 +                }
 +                program = av_find_program_from_stream(s, program, stream_index);
 +            }
 +        }
 +        return 1;
 +    }
 +    return 0;
  }
  
  static void update_initial_timestamps(AVFormatContext *s, int stream_index,
 -                                      int64_t dts, int64_t pts)
 +                                      int64_t dts, int64_t pts, AVPacket *pkt)
  {
      AVStream *st= s->streams[stream_index];
 -    AVPacketList *pktl= s->packet_buffer;
 +    AVPacketList *pktl= s->parse_queue ? s->parse_queue : s->packet_buffer;
 +    int64_t pts_buffer[MAX_REORDER_DELAY+1];
 +    int64_t shift;
 +    int i, delay;
  
 -    if(st->first_dts != AV_NOPTS_VALUE || dts == AV_NOPTS_VALUE || st->cur_dts == AV_NOPTS_VALUE)
 +    if(st->first_dts != AV_NOPTS_VALUE || dts == AV_NOPTS_VALUE || st->cur_dts == AV_NOPTS_VALUE || is_relative(dts))
          return;
  
 -    st->first_dts= dts - st->cur_dts;
 +    delay = st->codec->has_b_frames;
 +    st->first_dts= dts - (st->cur_dts - RELATIVE_TS_BASE);
      st->cur_dts= dts;
 +    shift = st->first_dts - RELATIVE_TS_BASE;
 +
 +    for (i=0; i<MAX_REORDER_DELAY+1; i++)
 +        pts_buffer[i] = AV_NOPTS_VALUE;
  
 -    for(; pktl; pktl= pktl->next){
 +    if (is_relative(pts))
 +        pts += shift;
 +
 +    for(; pktl; pktl= get_next_pkt(s, st, pktl)){
          if(pktl->pkt.stream_index != stream_index)
              continue;
 -        //FIXME think more about this check
 -        if(pktl->pkt.pts != AV_NOPTS_VALUE && pktl->pkt.pts == pktl->pkt.dts)
 -            pktl->pkt.pts += st->first_dts;
 +        if(is_relative(pktl->pkt.pts))
 +            pktl->pkt.pts += shift;
  
 -        if(pktl->pkt.dts != AV_NOPTS_VALUE)
 -            pktl->pkt.dts += st->first_dts;
 +        if(is_relative(pktl->pkt.dts))
 +            pktl->pkt.dts += shift;
  
          if(st->start_time == AV_NOPTS_VALUE && pktl->pkt.pts != AV_NOPTS_VALUE)
              st->start_time= pktl->pkt.pts;
 +
 +        if(pktl->pkt.pts != AV_NOPTS_VALUE && delay <= MAX_REORDER_DELAY && has_decode_delay_been_guessed(st)){
 +            pts_buffer[0]= pktl->pkt.pts;
 +            for(i=0; i<delay && pts_buffer[i] > pts_buffer[i+1]; i++)
 +                FFSWAP(int64_t, pts_buffer[i], pts_buffer[i+1]);
 +            if(pktl->pkt.dts == AV_NOPTS_VALUE)
 +                pktl->pkt.dts= pts_buffer[0];
 +        }
      }
 +
 +    if (update_wrap_reference(s, st, stream_index) && st->pts_wrap_behavior == AV_PTS_WRAP_SUB_OFFSET) {
 +        // correct first time stamps to negative values
 +        st->first_dts = wrap_timestamp(st, st->first_dts);
 +        st->cur_dts = wrap_timestamp(st, st->cur_dts);
 +        pkt->dts = wrap_timestamp(st, pkt->dts);
 +        pkt->pts = wrap_timestamp(st, pkt->pts);
 +        pts = wrap_timestamp(st, pts);
 +    }
 +
      if (st->start_time == AV_NOPTS_VALUE)
          st->start_time = pts;
  }
  static void update_initial_durations(AVFormatContext *s, AVStream *st,
                                       int stream_index, int duration)
  {
 -    AVPacketList *pktl= s->packet_buffer;
 -    int64_t cur_dts= 0;
 +    AVPacketList *pktl= s->parse_queue ? s->parse_queue : s->packet_buffer;
 +    int64_t cur_dts= RELATIVE_TS_BASE;
  
      if(st->first_dts != AV_NOPTS_VALUE){
          cur_dts= st->first_dts;
 -        for(; pktl; pktl= pktl->next){
 +        for(; pktl; pktl= get_next_pkt(s, st, pktl)){
              if(pktl->pkt.stream_index == stream_index){
                  if(pktl->pkt.pts != pktl->pkt.dts || pktl->pkt.dts != AV_NOPTS_VALUE || pktl->pkt.duration)
                      break;
                  cur_dts -= duration;
              }
          }
 -        pktl= s->packet_buffer;
 +        if(pktl && pktl->pkt.dts != st->first_dts) {
 +            av_log(s, AV_LOG_DEBUG, "first_dts %s not matching first dts %s in the queue\n", av_ts2str(st->first_dts), av_ts2str(pktl->pkt.dts));
 +            return;
 +        }
 +        if(!pktl) {
 +            av_log(s, AV_LOG_DEBUG, "first_dts %s but no packet with dts in the queue\n", av_ts2str(st->first_dts));
 +            return;
 +        }
 +        pktl= s->parse_queue ? s->parse_queue : s->packet_buffer;
          st->first_dts = cur_dts;
 -    }else if(st->cur_dts)
 +    }else if(st->cur_dts != RELATIVE_TS_BASE)
          return;
  
 -    for(; pktl; pktl= pktl->next){
 +    for(; pktl; pktl= get_next_pkt(s, st, pktl)){
          if(pktl->pkt.stream_index != stream_index)
              continue;
 -        if(pktl->pkt.pts == pktl->pkt.dts && pktl->pkt.dts == AV_NOPTS_VALUE
 +        if(pktl->pkt.pts == pktl->pkt.dts && (pktl->pkt.dts == AV_NOPTS_VALUE || pktl->pkt.dts == st->first_dts)
             && !pktl->pkt.duration){
              pktl->pkt.dts= cur_dts;
              if(!st->codec->has_b_frames)
                  pktl->pkt.pts= cur_dts;
 -            cur_dts += duration;
 -            if (st->codec->codec_type != AVMEDIA_TYPE_AUDIO)
 +//            if (st->codec->codec_type != AVMEDIA_TYPE_AUDIO)
                  pktl->pkt.duration = duration;
          }else
              break;
 +        cur_dts = pktl->pkt.dts + pktl->pkt.duration;
      }
 -    if(st->first_dts == AV_NOPTS_VALUE)
 +    if(!pktl)
          st->cur_dts= cur_dts;
  }
  
@@@ -1003,10 -717,6 +1003,10 @@@ static void compute_pkt_fields(AVFormat
      if((s->flags & AVFMT_FLAG_IGNDTS) && pkt->pts != AV_NOPTS_VALUE)
          pkt->dts= AV_NOPTS_VALUE;
  
 +    if (st->codec->codec_id != AV_CODEC_ID_H264 && pc && pc->pict_type == AV_PICTURE_TYPE_B)
 +        //FIXME Set low_delay = 0 when has_b_frames = 1
 +        st->codec->has_b_frames = 1;
 +
      /* do we have a video B-frame ? */
      delay= st->codec->has_b_frames;
      presentation_delayed = 0;
          pc && pc->pict_type != AV_PICTURE_TYPE_B)
          presentation_delayed = 1;
  
 -    if(pkt->pts != AV_NOPTS_VALUE && pkt->dts != AV_NOPTS_VALUE && pkt->dts > pkt->pts && st->pts_wrap_bits<63
 -       /*&& pkt->dts-(1LL<<st->pts_wrap_bits) < pkt->pts*/){
 -        pkt->dts -= 1LL<<st->pts_wrap_bits;
 +    if(pkt->pts != AV_NOPTS_VALUE && pkt->dts != AV_NOPTS_VALUE && st->pts_wrap_bits<63 && pkt->dts - (1LL<<(st->pts_wrap_bits-1)) > pkt->pts){
 +        if(is_relative(st->cur_dts) || pkt->dts - (1LL<<(st->pts_wrap_bits-1)) > st->cur_dts) {
 +            pkt->dts -= 1LL<<st->pts_wrap_bits;
 +        } else
 +            pkt->pts += 1LL<<st->pts_wrap_bits;
      }
  
      // some mpeg2 in mpeg-ps lack dts (issue171 / input_file.mpg)
      // we take the conservative approach and discard both
      // Note, if this is misbehaving for a H.264 file then possibly presentation_delayed is not set correctly.
      if(delay==1 && pkt->dts == pkt->pts && pkt->dts != AV_NOPTS_VALUE && presentation_delayed){
 -        av_log(s, AV_LOG_DEBUG, "invalid dts/pts combination\n");
 -        pkt->dts= pkt->pts= AV_NOPTS_VALUE;
 +        av_log(s, AV_LOG_DEBUG, "invalid dts/pts combination %"PRIi64"\n", pkt->dts);
 +        if(strcmp(s->iformat->name, "mov,mp4,m4a,3gp,3g2,mj2")) // otherwise we discard correct timestamps for vc1-wmapro.ism
 +            pkt->dts= AV_NOPTS_VALUE;
      }
  
 -    if (pkt->duration == 0 && st->codec->codec_type != AVMEDIA_TYPE_AUDIO) {
 +    if (pkt->duration == 0) {
          ff_compute_frame_duration(&num, &den, st, pc, pkt);
          if (den && num) {
              pkt->duration = av_rescale_rnd(1, num * (int64_t)st->time_base.den, den * (int64_t)st->time_base.num, AV_ROUND_DOWN);
 -
 -            if(pkt->duration != 0 && s->packet_buffer)
 -                update_initial_durations(s, st, pkt->stream_index, pkt->duration);
          }
      }
 +    if(pkt->duration != 0 && (s->packet_buffer || s->parse_queue))
 +        update_initial_durations(s, st, pkt->stream_index, pkt->duration);
  
      /* correct timestamps with byte offset if demuxers only have timestamps
         on packet boundaries */
      if(pkt->dts != AV_NOPTS_VALUE && pkt->pts != AV_NOPTS_VALUE && pkt->pts > pkt->dts)
          presentation_delayed = 1;
  
 -    av_dlog(NULL,
 -            "IN delayed:%d pts:%"PRId64", dts:%"PRId64" cur_dts:%"PRId64" st:%d pc:%p\n",
 -            presentation_delayed, pkt->pts, pkt->dts, st->cur_dts,
 -            pkt->stream_index, pc);
 +    av_dlog(NULL, "IN delayed:%d pts:%s, dts:%s cur_dts:%s st:%d pc:%p duration:%d\n",
 +           presentation_delayed, av_ts2str(pkt->pts), av_ts2str(pkt->dts), av_ts2str(st->cur_dts), pkt->stream_index, pc, pkt->duration);
      /* interpolate PTS and DTS if they are not present */
      //We skip H264 currently because delay and has_b_frames are not reliably set
      if((delay==0 || (delay==1 && pc)) && st->codec->codec_id != AV_CODEC_ID_H264){
              /* PTS = presentation timestamp */
              if (pkt->dts == AV_NOPTS_VALUE)
                  pkt->dts = st->last_IP_pts;
 -            update_initial_timestamps(s, pkt->stream_index, pkt->dts, pkt->pts);
 +            update_initial_timestamps(s, pkt->stream_index, pkt->dts, pkt->pts, pkt);
              if (pkt->dts == AV_NOPTS_VALUE)
                  pkt->dts = st->cur_dts;
  
              by knowing the future */
          } else if (pkt->pts != AV_NOPTS_VALUE ||
                     pkt->dts != AV_NOPTS_VALUE ||
 -                   pkt->duration              ||
 -                   st->codec->codec_type == AVMEDIA_TYPE_AUDIO) {
 +                   pkt->duration                ) {
              int duration = pkt->duration;
 -            if (!duration && st->codec->codec_type == AVMEDIA_TYPE_AUDIO) {
 -                ff_compute_frame_duration(&num, &den, st, pc, pkt);
 -                if (den && num) {
 -                    duration = av_rescale_rnd(1, num * (int64_t)st->time_base.den,
 -                                                 den * (int64_t)st->time_base.num,
 -                                                 AV_ROUND_DOWN);
 -                    if (duration != 0 && s->packet_buffer) {
 -                        update_initial_durations(s, st, pkt->stream_index,
 -                                                 duration);
 -                    }
 -                }
 -            }
  
 -            if (pkt->pts != AV_NOPTS_VALUE || pkt->dts != AV_NOPTS_VALUE ||
 -                duration) {
 -                /* presentation is not delayed : PTS and DTS are the same */
 -                if (pkt->pts == AV_NOPTS_VALUE)
 -                    pkt->pts = pkt->dts;
 -                update_initial_timestamps(s, pkt->stream_index, pkt->pts,
 -                                          pkt->pts);
 -                if (pkt->pts == AV_NOPTS_VALUE)
 -                    pkt->pts = st->cur_dts;
 -                pkt->dts = pkt->pts;
 -                if (pkt->pts != AV_NOPTS_VALUE)
 -                    st->cur_dts = pkt->pts + duration;
 -            }
 +            /* presentation is not delayed : PTS and DTS are the same */
 +            if (pkt->pts == AV_NOPTS_VALUE)
 +                pkt->pts = pkt->dts;
 +            update_initial_timestamps(s, pkt->stream_index, pkt->pts,
 +                                      pkt->pts, pkt);
 +            if (pkt->pts == AV_NOPTS_VALUE)
 +                pkt->pts = st->cur_dts;
 +            pkt->dts = pkt->pts;
 +            if (pkt->pts != AV_NOPTS_VALUE)
 +                st->cur_dts = pkt->pts + duration;
          }
      }
  
 -    if(pkt->pts != AV_NOPTS_VALUE && delay <= MAX_REORDER_DELAY){
 +    if(pkt->pts != AV_NOPTS_VALUE && delay <= MAX_REORDER_DELAY && has_decode_delay_been_guessed(st)){
          st->pts_buffer[0]= pkt->pts;
          for(i=0; i<delay && st->pts_buffer[i] > st->pts_buffer[i+1]; i++)
              FFSWAP(int64_t, st->pts_buffer[i], st->pts_buffer[i+1]);
          if(pkt->dts == AV_NOPTS_VALUE)
              pkt->dts= st->pts_buffer[0];
 -        if(st->codec->codec_id == AV_CODEC_ID_H264){ // we skipped it above so we try here
 -            update_initial_timestamps(s, pkt->stream_index, pkt->dts, pkt->pts); // this should happen on the first packet
 -        }
 -        if(pkt->dts > st->cur_dts)
 -            st->cur_dts = pkt->dts;
      }
 +    if(st->codec->codec_id == AV_CODEC_ID_H264){ // we skipped it above so we try here
 +        update_initial_timestamps(s, pkt->stream_index, pkt->dts, pkt->pts, pkt); // this should happen on the first packet
 +    }
 +    if(pkt->dts > st->cur_dts)
 +        st->cur_dts = pkt->dts;
  
 -    av_dlog(NULL,
 -            "OUTdelayed:%d/%d pts:%"PRId64", dts:%"PRId64" cur_dts:%"PRId64"\n",
 -            presentation_delayed, delay, pkt->pts, pkt->dts, st->cur_dts);
 +    av_dlog(NULL, "OUTdelayed:%d/%d pts:%s, dts:%s cur_dts:%s\n",
 +            presentation_delayed, delay, av_ts2str(pkt->pts), av_ts2str(pkt->dts), av_ts2str(st->cur_dts));
  
      /* update flags */
 -    if (is_intra_only(st->codec->codec_id))
 +    if (is_intra_only(st->codec))
          pkt->flags |= AV_PKT_FLAG_KEY;
      if (pc)
          pkt->convergence_duration = pc->convergence_duration;
@@@ -1169,9 -896,6 +1169,9 @@@ static int parse_packet(AVFormatContex
          av_init_packet(&flush_pkt);
          pkt = &flush_pkt;
          got_output = 1;
 +    } else if (!size && st->parser->flags & PARSER_FLAG_COMPLETE_FRAMES) {
 +        // preserve 0-size sync packets
 +        compute_pkt_fields(s, st, st->parser, pkt);
      }
  
      while (size > 0 || (pkt == &flush_pkt && got_output)) {
                                 pkt->pts, pkt->dts, pkt->pos);
  
          pkt->pts = pkt->dts = AV_NOPTS_VALUE;
 +        pkt->pos = -1;
          /* increment read pointer */
          data += len;
          size -= len;
          out_pkt.dts = st->parser->dts;
          out_pkt.pos = st->parser->pos;
  
 +        if(st->need_parsing == AVSTREAM_PARSE_FULL_RAW)
 +            out_pkt.pos = st->parser->frame_offset;
 +
          if (st->parser->key_frame == 1 ||
              (st->parser->key_frame == -1 &&
               st->parser->pict_type == AV_PICTURE_TYPE_I))
              out_pkt.flags |= AV_PKT_FLAG_KEY;
  
 -        compute_pkt_fields(s, st, st->parser, &out_pkt);
 +        if(st->parser->key_frame == -1 && st->parser->pict_type==AV_PICTURE_TYPE_NONE && (pkt->flags&AV_PKT_FLAG_KEY))
 +            out_pkt.flags |= AV_PKT_FLAG_KEY;
  
 -        if ((s->iformat->flags & AVFMT_GENERIC_INDEX) &&
 -            out_pkt.flags & AV_PKT_FLAG_KEY) {
 -            ff_reduce_index(s, st->index);
 -            av_add_index_entry(st, st->parser->frame_offset, out_pkt.dts,
 -                               0, 0, AVINDEX_KEYFRAME);
 -        }
 +        compute_pkt_fields(s, st, st->parser, &out_pkt);
  
          if (out_pkt.data == pkt->data && out_pkt.size == pkt->size) {
              out_pkt.buf   = pkt->buf;
@@@ -1311,17 -1035,17 +1311,17 @@@ static int read_frame_internal(AVFormat
          if (cur_pkt.pts != AV_NOPTS_VALUE &&
              cur_pkt.dts != AV_NOPTS_VALUE &&
              cur_pkt.pts < cur_pkt.dts) {
 -            av_log(s, AV_LOG_WARNING, "Invalid timestamps stream=%d, pts=%"PRId64", dts=%"PRId64", size=%d\n",
 +            av_log(s, AV_LOG_WARNING, "Invalid timestamps stream=%d, pts=%s, dts=%s, size=%d\n",
                     cur_pkt.stream_index,
 -                   cur_pkt.pts,
 -                   cur_pkt.dts,
 +                   av_ts2str(cur_pkt.pts),
 +                   av_ts2str(cur_pkt.dts),
                     cur_pkt.size);
          }
          if (s->debug & FF_FDEBUG_TS)
 -            av_log(s, AV_LOG_DEBUG, "ff_read_packet stream=%d, pts=%"PRId64", dts=%"PRId64", size=%d, duration=%d, flags=%d\n",
 +            av_log(s, AV_LOG_DEBUG, "ff_read_packet stream=%d, pts=%s, dts=%s, size=%d, duration=%d, flags=%d\n",
                     cur_pkt.stream_index,
 -                   cur_pkt.pts,
 -                   cur_pkt.dts,
 +                   av_ts2str(cur_pkt.pts),
 +                   av_ts2str(cur_pkt.dts),
                     cur_pkt.size,
                     cur_pkt.duration,
                     cur_pkt.flags);
          if (st->need_parsing && !st->parser && !(s->flags & AVFMT_FLAG_NOPARSE)) {
              st->parser = av_parser_init(st->codec->codec_id);
              if (!st->parser) {
 +                av_log(s, AV_LOG_VERBOSE, "parser not found for codec "
 +                       "%s, packets or times may be invalid.\n",
 +                       avcodec_get_name(st->codec->codec_id));
                  /* no parser available: just output the raw packets */
                  st->need_parsing = AVSTREAM_PARSE_NONE;
              } else if(st->need_parsing == AVSTREAM_PARSE_HEADERS) {
                  st->parser->flags |= PARSER_FLAG_COMPLETE_FRAMES;
              } else if(st->need_parsing == AVSTREAM_PARSE_FULL_ONCE) {
                  st->parser->flags |= PARSER_FLAG_ONCE;
 +            } else if(st->need_parsing == AVSTREAM_PARSE_FULL_RAW) {
 +                st->parser->flags |= PARSER_FLAG_USE_CODEC_TS;
              }
          }
  
              /* free packet */
              av_free_packet(&cur_pkt);
          }
 +        if (pkt->flags & AV_PKT_FLAG_KEY)
 +            st->skip_to_keyframe = 0;
 +        if (st->skip_to_keyframe) {
 +            av_free_packet(&cur_pkt);
 +            if (got_packet) {
 +                *pkt = cur_pkt;
 +            }
 +            got_packet = 0;
 +        }
      }
  
      if (!got_packet && s->parse_queue)
          ret = read_from_packet_buffer(&s->parse_queue, &s->parse_queue_end, pkt);
  
      if(s->debug & FF_FDEBUG_TS)
 -        av_log(s, AV_LOG_DEBUG, "read_frame_internal stream=%d, pts=%"PRId64", dts=%"PRId64", size=%d, duration=%d, flags=%d\n",
 +        av_log(s, AV_LOG_DEBUG, "read_frame_internal stream=%d, pts=%s, dts=%s, size=%d, duration=%d, flags=%d\n",
              pkt->stream_index,
 -            pkt->pts,
 -            pkt->dts,
 +            av_ts2str(pkt->pts),
 +            av_ts2str(pkt->dts),
              pkt->size,
              pkt->duration,
              pkt->flags);
@@@ -1390,19 -1100,15 +1390,19 @@@ int av_read_frame(AVFormatContext *s, A
  {
      const int genpts = s->flags & AVFMT_FLAG_GENPTS;
      int          eof = 0;
 +    int ret;
 +    AVStream *st;
  
 -    if (!genpts)
 -        return s->packet_buffer ? read_from_packet_buffer(&s->packet_buffer,
 -                                                          &s->packet_buffer_end,
 -                                                          pkt) :
 -                                  read_frame_internal(s, pkt);
 +    if (!genpts) {
 +        ret = s->packet_buffer ?
 +            read_from_packet_buffer(&s->packet_buffer, &s->packet_buffer_end, pkt) :
 +            read_frame_internal(s, pkt);
 +        if (ret < 0)
 +            return ret;
 +        goto return_packet;
 +    }
  
      for (;;) {
 -        int ret;
          AVPacketList *pktl = s->packet_buffer;
  
          if (pktl) {
  
              if (next_pkt->dts != AV_NOPTS_VALUE) {
                  int wrap_bits = s->streams[next_pkt->stream_index]->pts_wrap_bits;
 +                // last dts seen for this stream. if any of packets following
 +                // current one had no dts, we will set this to AV_NOPTS_VALUE.
 +                int64_t last_dts = next_pkt->dts;
                  while (pktl && next_pkt->pts == AV_NOPTS_VALUE) {
                      if (pktl->pkt.stream_index == next_pkt->stream_index &&
 -                        (av_compare_mod(next_pkt->dts, pktl->pkt.dts, 2LL << (wrap_bits - 1)) < 0) &&
 -                         av_compare_mod(pktl->pkt.pts, pktl->pkt.dts, 2LL << (wrap_bits - 1))) { //not b frame
 -                        next_pkt->pts = pktl->pkt.dts;
 +                        (av_compare_mod(next_pkt->dts, pktl->pkt.dts, 2LL << (wrap_bits - 1)) < 0)) {
 +                        if (av_compare_mod(pktl->pkt.pts, pktl->pkt.dts, 2LL << (wrap_bits - 1))) { //not b frame
 +                            next_pkt->pts = pktl->pkt.dts;
 +                        }
 +                        if (last_dts != AV_NOPTS_VALUE) {
 +                            // Once last dts was set to AV_NOPTS_VALUE, we don't change it.
 +                            last_dts = pktl->pkt.dts;
 +                        }
                      }
                      pktl = pktl->next;
                  }
 +                if (eof && next_pkt->pts == AV_NOPTS_VALUE && last_dts != AV_NOPTS_VALUE) {
 +                    // Fixing the last reference frame had none pts issue (For MXF etc).
 +                    // We only do this when
 +                    // 1. eof.
 +                    // 2. we are not able to resolve a pts value for current packet.
 +                    // 3. the packets for this stream at the end of the files had valid dts.
 +                    next_pkt->pts = last_dts + next_pkt->duration;
 +                }
                  pktl = s->packet_buffer;
              }
  
              /* read packet from packet buffer, if there is data */
              if (!(next_pkt->pts == AV_NOPTS_VALUE &&
 -                  next_pkt->dts != AV_NOPTS_VALUE && !eof))
 -                return read_from_packet_buffer(&s->packet_buffer,
 +                  next_pkt->dts != AV_NOPTS_VALUE && !eof)) {
 +                ret = read_from_packet_buffer(&s->packet_buffer,
                                                 &s->packet_buffer_end, pkt);
 +                goto return_packet;
 +            }
          }
  
          ret = read_frame_internal(s, pkt);
                            &s->packet_buffer_end)) < 0)
              return AVERROR(ENOMEM);
      }
 +
 +return_packet:
 +
 +    st = s->streams[pkt->stream_index];
 +    if (st->skip_samples) {
 +        uint8_t *p = av_packet_new_side_data(pkt, AV_PKT_DATA_SKIP_SAMPLES, 10);
 +        AV_WL32(p, st->skip_samples);
 +        av_log(s, AV_LOG_DEBUG, "demuxer injecting skip %d\n", st->skip_samples);
 +        st->skip_samples = 0;
 +    }
 +
 +    if ((s->iformat->flags & AVFMT_GENERIC_INDEX) && pkt->flags & AV_PKT_FLAG_KEY) {
 +        ff_reduce_index(s, st->index);
 +        av_add_index_entry(st, pkt->pos, pkt->dts, 0, 0, AVINDEX_KEYFRAME);
 +    }
 +
 +    if (is_relative(pkt->dts))
 +        pkt->dts -= RELATIVE_TS_BASE;
 +    if (is_relative(pkt->pts))
 +        pkt->pts -= RELATIVE_TS_BASE;
 +
 +    return ret;
  }
  
  /* XXX: suppress the packet queue */
@@@ -1535,8 -1201,7 +1535,8 @@@ void ff_read_frame_flush(AVFormatContex
              st->parser = NULL;
          }
          st->last_IP_pts = AV_NOPTS_VALUE;
 -        st->cur_dts = AV_NOPTS_VALUE; /* we set the current DTS to an unspecified origin */
 +        if(st->first_dts == AV_NOPTS_VALUE) st->cur_dts = RELATIVE_TS_BASE;
 +        else                                st->cur_dts = AV_NOPTS_VALUE; /* we set the current DTS to an unspecified origin */
          st->reference_dts = AV_NOPTS_VALUE;
  
          st->probe_packets = MAX_PROBE_PACKETS;
@@@ -1583,12 -1248,6 +1583,12 @@@ int ff_add_index_entry(AVIndexEntry **i
      if((unsigned)*nb_index_entries + 1 >= UINT_MAX / sizeof(AVIndexEntry))
          return -1;
  
 +    if(timestamp == AV_NOPTS_VALUE)
 +        return AVERROR(EINVAL);
 +
 +    if (is_relative(timestamp)) //FIXME this maintains previous behavior but we should shift by the correct offset once known
 +        timestamp -= RELATIVE_TS_BASE;
 +
      entries = av_fast_realloc(*index_entries,
                                index_entries_allocated_size,
                                (*nb_index_entries + 1) *
      if(index<0){
          index= (*nb_index_entries)++;
          ie= &entries[index];
 -        assert(index==0 || ie[-1].timestamp < timestamp);
 +        av_assert0(index==0 || ie[-1].timestamp < timestamp);
      }else{
          ie= &entries[index];
          if(ie->timestamp != timestamp){
  int av_add_index_entry(AVStream *st,
                         int64_t pos, int64_t timestamp, int size, int distance, int flags)
  {
 +    timestamp = wrap_timestamp(st, timestamp);
      return ff_add_index_entry(&st->index_entries, &st->nb_index_entries,
                                &st->index_entries_allocated_size, pos,
                                timestamp, size, distance, flags);
@@@ -1674,15 -1332,6 +1674,15 @@@ int av_index_search_timestamp(AVStream 
                                       wanted_timestamp, flags);
  }
  
 +static int64_t ff_read_timestamp(AVFormatContext *s, int stream_index, int64_t *ppos, int64_t pos_limit,
 +                                 int64_t (*read_timestamp)(struct AVFormatContext *, int , int64_t *, int64_t ))
 +{
 +    int64_t ts = read_timestamp(s, stream_index, ppos, pos_limit);
 +    if (stream_index >= 0)
 +        ts = wrap_timestamp(s->streams[stream_index], ts);
 +    return ts;
 +}
 +
  int ff_seek_frame_binary(AVFormatContext *s, int stream_index, int64_t target_ts, int flags)
  {
      AVInputFormat *avif= s->iformat;
      if (stream_index < 0)
          return -1;
  
 -    av_dlog(s, "read_seek: %d %"PRId64"\n", stream_index, target_ts);
 +    av_dlog(s, "read_seek: %d %s\n", stream_index, av_ts2str(target_ts));
  
      ts_max=
      ts_min= AV_NOPTS_VALUE;
          if(e->timestamp <= target_ts || e->pos == e->min_distance){
              pos_min= e->pos;
              ts_min= e->timestamp;
 -            av_dlog(s, "using cached pos_min=0x%"PRIx64" dts_min=%"PRId64"\n",
 -                    pos_min,ts_min);
 +            av_dlog(s, "using cached pos_min=0x%"PRIx64" dts_min=%s\n",
 +                    pos_min, av_ts2str(ts_min));
          }else{
 -            assert(index==0);
 +            av_assert1(index==0);
          }
  
          index= av_index_search_timestamp(st, target_ts, flags & ~AVSEEK_FLAG_BACKWARD);
 -        assert(index < st->nb_index_entries);
 +        av_assert0(index < st->nb_index_entries);
          if(index >= 0){
              e= &st->index_entries[index];
 -            assert(e->timestamp >= target_ts);
 +            av_assert1(e->timestamp >= target_ts);
              pos_max= e->pos;
              ts_max= e->timestamp;
              pos_limit= pos_max - e->min_distance;
 -            av_dlog(s, "using cached pos_max=0x%"PRIx64" pos_limit=0x%"PRIx64" dts_max=%"PRId64"\n",
 -                    pos_max,pos_limit, ts_max);
 +            av_dlog(s, "using cached pos_max=0x%"PRIx64" pos_limit=0x%"PRIx64" dts_max=%s\n",
 +                    pos_max, pos_limit, av_ts2str(ts_max));
          }
      }
  
      if ((ret = avio_seek(s->pb, pos, SEEK_SET)) < 0)
          return ret;
  
 +    ff_read_frame_flush(s);
      ff_update_cur_dts(s, st, ts);
  
      return 0;
@@@ -1754,35 -1402,30 +1754,35 @@@ int64_t ff_gen_search(AVFormatContext *
      int64_t start_pos, filesize;
      int no_change;
  
 -    av_dlog(s, "gen_seek: %d %"PRId64"\n", stream_index, target_ts);
 +    av_dlog(s, "gen_seek: %d %s\n", stream_index, av_ts2str(target_ts));
  
      if(ts_min == AV_NOPTS_VALUE){
          pos_min = s->data_offset;
 -        ts_min = read_timestamp(s, stream_index, &pos_min, INT64_MAX);
 +        ts_min = ff_read_timestamp(s, stream_index, &pos_min, INT64_MAX, read_timestamp);
          if (ts_min == AV_NOPTS_VALUE)
              return -1;
      }
  
 +    if(ts_min >= target_ts){
 +        *ts_ret= ts_min;
 +        return pos_min;
 +    }
 +
      if(ts_max == AV_NOPTS_VALUE){
          int step= 1024;
          filesize = avio_size(s->pb);
          pos_max = filesize - 1;
          do{
 -            pos_max -= step;
 -            ts_max = read_timestamp(s, stream_index, &pos_max, pos_max + step);
 +            pos_max = FFMAX(0, pos_max - step);
 +            ts_max = ff_read_timestamp(s, stream_index, &pos_max, pos_max + step, read_timestamp);
              step += step;
 -        }while(ts_max == AV_NOPTS_VALUE && pos_max >= step);
 +        }while(ts_max == AV_NOPTS_VALUE && pos_max > 0);
          if (ts_max == AV_NOPTS_VALUE)
              return -1;
  
          for(;;){
              int64_t tmp_pos= pos_max + 1;
 -            int64_t tmp_ts= read_timestamp(s, stream_index, &tmp_pos, INT64_MAX);
 +            int64_t tmp_ts= ff_read_timestamp(s, stream_index, &tmp_pos, INT64_MAX, read_timestamp);
              if(tmp_ts == AV_NOPTS_VALUE)
                  break;
              ts_max= tmp_ts;
          pos_limit= pos_max;
      }
  
 +    if(ts_max <= target_ts){
 +        *ts_ret= ts_max;
 +        return pos_max;
 +    }
 +
      if(ts_min > ts_max){
          return -1;
      }else if(ts_min == ts_max){
  
      no_change=0;
      while (pos_min < pos_limit) {
 -        av_dlog(s, "pos_min=0x%"PRIx64" pos_max=0x%"PRIx64" dts_min=%"PRId64" dts_max=%"PRId64"\n",
 -                pos_min, pos_max, ts_min, ts_max);
 +        av_dlog(s, "pos_min=0x%"PRIx64" pos_max=0x%"PRIx64" dts_min=%s dts_max=%s\n",
 +                pos_min, pos_max, av_ts2str(ts_min), av_ts2str(ts_max));
          assert(pos_limit <= pos_max);
  
          if(no_change==0){
              pos= pos_limit;
          start_pos= pos;
  
 -        ts = read_timestamp(s, stream_index, &pos, INT64_MAX); //may pass pos_limit instead of -1
 +        ts = ff_read_timestamp(s, stream_index, &pos, INT64_MAX, read_timestamp); //may pass pos_limit instead of -1
          if(pos == pos_max)
              no_change++;
          else
              no_change=0;
 -        av_dlog(s, "%"PRId64" %"PRId64" %"PRId64" / %"PRId64" %"PRId64" %"PRId64" target:%"PRId64" limit:%"PRId64" start:%"PRId64" noc:%d\n",
 -                pos_min, pos, pos_max, ts_min, ts, ts_max, target_ts,
 +        av_dlog(s, "%"PRId64" %"PRId64" %"PRId64" / %s %s %s target:%s limit:%"PRId64" start:%"PRId64" noc:%d\n",
 +                pos_min, pos, pos_max,
 +                av_ts2str(ts_min), av_ts2str(ts), av_ts2str(ts_max), av_ts2str(target_ts),
                  pos_limit, start_pos, no_change);
          if(ts == AV_NOPTS_VALUE){
              av_log(s, AV_LOG_ERROR, "read_timestamp() failed in the middle\n");
  
      pos = (flags & AVSEEK_FLAG_BACKWARD) ? pos_min : pos_max;
      ts  = (flags & AVSEEK_FLAG_BACKWARD) ?  ts_min :  ts_max;
 +#if 0
      pos_min = pos;
 -    ts_min = read_timestamp(s, stream_index, &pos_min, INT64_MAX);
 +    ts_min = ff_read_timestamp(s, stream_index, &pos_min, INT64_MAX, read_timestamp);
      pos_min++;
 -    ts_max = read_timestamp(s, stream_index, &pos_min, INT64_MAX);
 -    av_dlog(s, "pos=0x%"PRIx64" %"PRId64"<=%"PRId64"<=%"PRId64"\n",
 -            pos, ts_min, target_ts, ts_max);
 +    ts_max = ff_read_timestamp(s, stream_index, &pos_min, INT64_MAX, read_timestamp);
 +    av_dlog(s, "pos=0x%"PRIx64" %s<=%s<=%s\n",
 +            pos, av_ts2str(ts_min), av_ts2str(target_ts), av_ts2str(ts_max));
 +#endif
      *ts_ret= ts;
      return pos;
  }
@@@ -1879,8 -1514,6 +1879,8 @@@ static int seek_frame_byte(AVFormatCont
  
      avio_seek(s->pb, pos, SEEK_SET);
  
 +    s->io_repositioned = 1;
 +
      return 0;
  }
  
@@@ -1901,10 -1534,9 +1901,10 @@@ static int seek_frame_generic(AVFormatC
  
      if(index < 0 || index==st->nb_index_entries-1){
          AVPacket pkt;
 +        int nonkey=0;
  
          if(st->nb_index_entries){
 -            assert(st->index_entries);
 +            av_assert0(st->index_entries);
              ie= &st->index_entries[st->nb_index_entries-1];
              if ((ret = avio_seek(s->pb, ie->pos, SEEK_SET)) < 0)
                  return ret;
              if (read_status < 0)
                  break;
              av_free_packet(&pkt);
 -            if(stream_index == pkt.stream_index){
 -                if((pkt.flags & AV_PKT_FLAG_KEY) && pkt.dts > timestamp)
 +            if(stream_index == pkt.stream_index && pkt.dts > timestamp){
 +                if(pkt.flags & AV_PKT_FLAG_KEY)
                      break;
 +                if(nonkey++ > 1000 && st->codec->codec_id != AV_CODEC_ID_CDGRAPHICS){
 +                    av_log(s, AV_LOG_ERROR,"seek_frame_generic failed as this stream seems to contain no keyframes after the target timestamp, %d non keyframes found\n", nonkey);
 +                    break;
 +                }
              }
          }
          index = av_index_search_timestamp(st, timestamp, flags);
@@@ -1994,22 -1622,10 +1994,22 @@@ static int seek_frame_internal(AVFormat
  
  int av_seek_frame(AVFormatContext *s, int stream_index, int64_t timestamp, int flags)
  {
 -    int ret = seek_frame_internal(s, stream_index, timestamp, flags);
 +    int ret;
 +
 +    if (s->iformat->read_seek2 && !s->iformat->read_seek) {
 +        int64_t min_ts = INT64_MIN, max_ts = INT64_MAX;
 +        if ((flags & AVSEEK_FLAG_BACKWARD))
 +            max_ts = timestamp;
 +        else
 +            min_ts = timestamp;
 +        return avformat_seek_file(s, stream_index, min_ts, timestamp, max_ts,
 +                                  flags & ~AVSEEK_FLAG_BACKWARD);
 +    }
 +
 +    ret = seek_frame_internal(s, stream_index, timestamp, flags);
  
      if (ret >= 0)
 -        ret = queue_attached_pictures(s);
 +        ret = avformat_queue_attached_pictures(s);
  
      return ret;
  }
@@@ -2018,32 -1634,14 +2018,32 @@@ int avformat_seek_file(AVFormatContext 
  {
      if(min_ts > ts || max_ts < ts)
          return -1;
 +    if (stream_index < -1 || stream_index >= (int)s->nb_streams)
 +        return AVERROR(EINVAL);
 +
 +    if(s->seek2any>0)
 +        flags |= AVSEEK_FLAG_ANY;
 +    flags &= ~AVSEEK_FLAG_BACKWARD;
  
      if (s->iformat->read_seek2) {
          int ret;
          ff_read_frame_flush(s);
 +
 +        if (stream_index == -1 && s->nb_streams == 1) {
 +            AVRational time_base = s->streams[0]->time_base;
 +            ts = av_rescale_q(ts, AV_TIME_BASE_Q, time_base);
 +            min_ts = av_rescale_rnd(min_ts, time_base.den,
 +                                    time_base.num * (int64_t)AV_TIME_BASE,
 +                                    AV_ROUND_UP   | AV_ROUND_PASS_MINMAX);
 +            max_ts = av_rescale_rnd(max_ts, time_base.den,
 +                                    time_base.num * (int64_t)AV_TIME_BASE,
 +                                    AV_ROUND_DOWN | AV_ROUND_PASS_MINMAX);
 +        }
 +
          ret = s->iformat->read_seek2(s, stream_index, min_ts, ts, max_ts, flags);
  
          if (ret >= 0)
 -            ret = queue_attached_pictures(s);
 +            ret = avformat_queue_attached_pictures(s);
          return ret;
      }
  
  
      // Fall back on old API if new is not implemented but old is.
      // Note the old API has somewhat different semantics.
 -    if(s->iformat->read_seek || 1)
 -        return av_seek_frame(s, stream_index, ts, flags | ((uint64_t)ts - min_ts > (uint64_t)max_ts - ts ? AVSEEK_FLAG_BACKWARD : 0));
 +    if (s->iformat->read_seek || 1) {
 +        int dir = (ts - (uint64_t)min_ts > (uint64_t)max_ts - ts ? AVSEEK_FLAG_BACKWARD : 0);
 +        int ret = av_seek_frame(s, stream_index, ts, flags | dir);
 +        if (ret<0 && ts != min_ts && max_ts != ts) {
 +            ret = av_seek_frame(s, stream_index, dir ? max_ts : min_ts, flags | dir);
 +            if (ret >= 0)
 +                ret = av_seek_frame(s, stream_index, ts, flags | (dir^AVSEEK_FLAG_BACKWARD));
 +        }
 +        return ret;
 +    }
  
      // try some generic seek like seek_frame_generic() but with new ts semantics
 +    return -1; //unreachable
  }
  
  /*******************************************************/
@@@ -2097,71 -1686,43 +2097,71 @@@ static int has_duration(AVFormatContex
   */
  static void update_stream_timings(AVFormatContext *ic)
  {
 -    int64_t start_time, start_time1, end_time, end_time1;
 +    int64_t start_time, start_time1, start_time_text, end_time, end_time1;
      int64_t duration, duration1, filesize;
      int i;
      AVStream *st;
 +    AVProgram *p;
  
      start_time = INT64_MAX;
 +    start_time_text = INT64_MAX;
      end_time = INT64_MIN;
      duration = INT64_MIN;
      for(i = 0;i < ic->nb_streams; i++) {
          st = ic->streams[i];
          if (st->start_time != AV_NOPTS_VALUE && st->time_base.den) {
              start_time1= av_rescale_q(st->start_time, st->time_base, AV_TIME_BASE_Q);
 -            start_time = FFMIN(start_time, start_time1);
 +            if (st->codec->codec_type == AVMEDIA_TYPE_SUBTITLE || st->codec->codec_type == AVMEDIA_TYPE_DATA) {
 +                if (start_time1 < start_time_text)
 +                    start_time_text = start_time1;
 +            } else
 +                start_time = FFMIN(start_time, start_time1);
 +            end_time1 = AV_NOPTS_VALUE;
              if (st->duration != AV_NOPTS_VALUE) {
                  end_time1 = start_time1
                            + av_rescale_q(st->duration, st->time_base, AV_TIME_BASE_Q);
                  end_time = FFMAX(end_time, end_time1);
              }
 +            for(p = NULL; (p = av_find_program_from_stream(ic, p, i)); ){
 +                if(p->start_time == AV_NOPTS_VALUE || p->start_time > start_time1)
 +                    p->start_time = start_time1;
 +                if(p->end_time < end_time1)
 +                    p->end_time = end_time1;
 +            }
          }
          if (st->duration != AV_NOPTS_VALUE) {
              duration1 = av_rescale_q(st->duration, st->time_base, AV_TIME_BASE_Q);
              duration = FFMAX(duration, duration1);
          }
      }
 +    if (start_time == INT64_MAX || (start_time > start_time_text && start_time - start_time_text < AV_TIME_BASE))
 +        start_time = start_time_text;
 +    else if(start_time > start_time_text)
 +        av_log(ic, AV_LOG_VERBOSE, "Ignoring outlier non primary stream starttime %f\n", start_time_text / (float)AV_TIME_BASE);
 +
      if (start_time != INT64_MAX) {
          ic->start_time = start_time;
 -        if (end_time != INT64_MIN)
 -            duration = FFMAX(duration, end_time - start_time);
 +        if (end_time != INT64_MIN) {
 +            if (ic->nb_programs) {
 +                for (i=0; i<ic->nb_programs; i++) {
 +                    p = ic->programs[i];
 +                    if(p->start_time != AV_NOPTS_VALUE && p->end_time > p->start_time)
 +                        duration = FFMAX(duration, p->end_time - p->start_time);
 +                }
 +            } else
 +                duration = FFMAX(duration, end_time - start_time);
 +        }
      }
 -    if (duration != INT64_MIN) {
 +    if (duration != INT64_MIN && duration > 0 && ic->duration == AV_NOPTS_VALUE) {
          ic->duration = duration;
 -        if (ic->pb && (filesize = avio_size(ic->pb)) > 0) {
 +    }
 +        if (ic->pb && (filesize = avio_size(ic->pb)) > 0 && ic->duration != AV_NOPTS_VALUE) {
              /* compute the bitrate */
 -            ic->bit_rate = (double)filesize * 8.0 * AV_TIME_BASE /
 +            double bitrate = (double)filesize * 8.0 * AV_TIME_BASE /
                  (double)ic->duration;
 +            if (bitrate >= 0 && bitrate <= INT_MAX)
 +                ic->bit_rate = bitrate;
          }
 -    }
  }
  
  static void fill_all_stream_timings(AVFormatContext *ic)
  static void estimate_timings_from_bit_rate(AVFormatContext *ic)
  {
      int64_t filesize, duration;
 -    int bit_rate, i;
 +    int bit_rate, i, show_warning = 0;
      AVStream *st;
  
      /* if bit_rate is already set, we believe it */
          if (filesize > 0) {
              for(i = 0; i < ic->nb_streams; i++) {
                  st = ic->streams[i];
 -                duration= av_rescale(8*filesize, st->time_base.den, ic->bit_rate*(int64_t)st->time_base.num);
 -                if (st->duration == AV_NOPTS_VALUE)
 +                if (   st->time_base.num <= INT64_MAX / ic->bit_rate
 +                    && st->duration == AV_NOPTS_VALUE) {
 +                    duration= av_rescale(8*filesize, st->time_base.den, ic->bit_rate*(int64_t)st->time_base.num);
                      st->duration = duration;
 +                    show_warning = 1;
 +                }
              }
          }
      }
 +    if (show_warning)
 +        av_log(ic, AV_LOG_WARNING, "Estimating duration from bitrate, this may be inaccurate\n");
  }
  
 -#define DURATION_MAX_READ_SIZE 250000
 -#define DURATION_MAX_RETRY 3
 +#define DURATION_MAX_READ_SIZE 250000LL
 +#define DURATION_MAX_RETRY 4
  
  /* only usable for MPEG-PS streams */
  static void estimate_timings_from_pts(AVFormatContext *ic, int64_t old_offset)
                      duration -= st->start_time;
                  else
                      duration -= st->first_dts;
 -                if (duration < 0)
 -                    duration += 1LL<<st->pts_wrap_bits;
                  if (duration > 0) {
 -                    if (st->duration == AV_NOPTS_VALUE || st->duration < duration)
 +                    if (st->duration == AV_NOPTS_VALUE || st->info->last_duration<=0 ||
 +                        (st->duration < duration && FFABS(duration - st->info->last_duration) < 60LL*st->time_base.den / st->time_base.num))
                          st->duration = duration;
 +                    st->info->last_duration = duration;
                  }
              }
              av_free_packet(pkt);
@@@ -2316,16 -1872,14 +2316,16 @@@ static void estimate_timings(AVFormatCo
          file_size && ic->pb->seekable) {
          /* get accurate estimate from the PTSes */
          estimate_timings_from_pts(ic, old_offset);
 +        ic->duration_estimation_method = AVFMT_DURATION_FROM_PTS;
      } else if (has_duration(ic)) {
          /* at least one component has timings - we use them for all
             the components */
          fill_all_stream_timings(ic);
 +        ic->duration_estimation_method = AVFMT_DURATION_FROM_STREAM;
      } else {
 -        av_log(ic, AV_LOG_WARNING, "Estimating duration from bitrate, this may be inaccurate\n");
          /* less precise: use bitrate info */
          estimate_timings_from_bit_rate(ic);
 +        ic->duration_estimation_method = AVFMT_DURATION_FROM_BITRATE;
      }
      update_stream_timings(ic);
  
      }
  }
  
 -static int has_codec_parameters(AVStream *st)
 +static int has_codec_parameters(AVStream *st, const char **errmsg_ptr)
  {
      AVCodecContext *avctx = st->codec;
 -    int val;
 +
 +#define FAIL(errmsg) do {                                         \
 +        if (errmsg_ptr)                                           \
 +            *errmsg_ptr = errmsg;                                 \
 +        return 0;                                                 \
 +    } while (0)
 +
      switch (avctx->codec_type) {
      case AVMEDIA_TYPE_AUDIO:
 -        val = avctx->sample_rate && avctx->channels;
 +        if (!avctx->frame_size && determinable_frame_size(avctx))
 +            FAIL("unspecified frame size");
          if (st->info->found_decoder >= 0 && avctx->sample_fmt == AV_SAMPLE_FMT_NONE)
 -            return 0;
 +            FAIL("unspecified sample format");
 +        if (!avctx->sample_rate)
 +            FAIL("unspecified sample rate");
 +        if (!avctx->channels)
 +            FAIL("unspecified number of channels");
 +        if (st->info->found_decoder >= 0 && !st->nb_decoded_frames && avctx->codec_id == AV_CODEC_ID_DTS)
 +            FAIL("no decodable DTS frames");
          break;
      case AVMEDIA_TYPE_VIDEO:
 -        val = avctx->width;
 +        if (!avctx->width)
 +            FAIL("unspecified size");
          if (st->info->found_decoder >= 0 && avctx->pix_fmt == AV_PIX_FMT_NONE)
 -            return 0;
 +            FAIL("unspecified pixel format");
 +        if (st->codec->codec_id == AV_CODEC_ID_RV30 || st->codec->codec_id == AV_CODEC_ID_RV40)
 +            if (!st->sample_aspect_ratio.num && !st->codec->sample_aspect_ratio.num && !st->codec_info_nb_frames)
 +                FAIL("no frame in rv30/40 and no sar");
          break;
 -    default:
 -        val = 1;
 +    case AVMEDIA_TYPE_SUBTITLE:
 +        if (avctx->codec_id == AV_CODEC_ID_HDMV_PGS_SUBTITLE && !avctx->width)
 +            FAIL("unspecified size");
          break;
 +    case AVMEDIA_TYPE_DATA:
 +        if(avctx->codec_id == AV_CODEC_ID_NONE) return 1;
      }
 -    return avctx->codec_id != AV_CODEC_ID_NONE && val != 0;
 -}
  
 -static int has_decode_delay_been_guessed(AVStream *st)
 -{
 -    return st->codec->codec_id != AV_CODEC_ID_H264 ||
 -        st->info->nb_decoded_frames >= 6;
 +    if (avctx->codec_id == AV_CODEC_ID_NONE)
 +        FAIL("unknown codec");
 +    return 1;
  }
  
  /* returns 1 or 0 if or if not decoded data was returned, or a negative error */
@@@ -2396,7 -1933,6 +2396,7 @@@ static int try_decode_frame(AVStream *s
      const AVCodec *codec;
      int got_picture = 1, ret = 0;
      AVFrame *frame = avcodec_alloc_frame();
 +    AVSubtitle subtitle;
      AVPacket pkt = *avpkt;
  
      if (!frame)
  
      while ((pkt.size > 0 || (!pkt.data && got_picture)) &&
             ret >= 0 &&
 -           (!has_codec_parameters(st)         ||
 +           (!has_codec_parameters(st, NULL)   ||
             !has_decode_delay_been_guessed(st) ||
             (!st->codec_info_nb_frames && st->codec->codec->capabilities & CODEC_CAP_CHANNEL_CONF))) {
          got_picture = 0;
          case AVMEDIA_TYPE_AUDIO:
              ret = avcodec_decode_audio4(st->codec, frame, &got_picture, &pkt);
              break;
 +        case AVMEDIA_TYPE_SUBTITLE:
 +            ret = avcodec_decode_subtitle2(st->codec, &subtitle,
 +                                           &got_picture, &pkt);
 +            ret = pkt.size;
 +            break;
          default:
              break;
          }
          if (ret >= 0) {
              if (got_picture)
 -                st->info->nb_decoded_frames++;
 +                st->nb_decoded_frames++;
              pkt.data += ret;
              pkt.size -= ret;
              ret       = got_picture;
          }
      }
  
 +    if(!pkt.data && !got_picture)
 +        ret = -1;
 +
  fail:
      avcodec_free_frame(&frame);
      return ret;
@@@ -2506,7 -2034,6 +2506,7 @@@ enum AVCodecID ff_get_pcm_codec_id(int 
          default: return AV_CODEC_ID_NONE;
          }
      } else {
 +        bps  += 7;
          bps >>= 3;
          if (sflags & (1 << (bps - 1))) {
              switch (bps) {
  
  unsigned int av_codec_get_tag(const AVCodecTag * const *tags, enum AVCodecID id)
  {
 +    unsigned int tag;
 +    if (!av_codec_get_tag2(tags, id, &tag))
 +        return 0;
 +    return tag;
 +}
 +
 +int av_codec_get_tag2(const AVCodecTag * const *tags, enum AVCodecID id,
 +                      unsigned int *tag)
 +{
      int i;
      for(i=0; tags && tags[i]; i++){
 -        int tag= ff_codec_get_tag(tags[i], id);
 -        if(tag) return tag;
 +        const AVCodecTag *codec_tags = tags[i];
 +        while (codec_tags->id != AV_CODEC_ID_NONE) {
 +            if (codec_tags->id == id) {
 +                *tag = codec_tags->tag;
 +                return 1;
 +            }
 +            codec_tags++;
 +        }
      }
      return 0;
  }
@@@ -2585,8 -2097,8 +2585,8 @@@ static void compute_chapters_end(AVForm
  }
  
  static int get_std_framerate(int i){
 -    if(i<60*12) return i*1001;
 -    else        return ((const int[]){24,30,60,12,15})[i-60*12]*1000*12;
 +    if(i<60*12) return (i+1)*1001;
 +    else        return ((const int[]){24,30,60,12,15,48})[i-60*12]*1000*12;
  }
  
  /*
@@@ -2602,7 -2114,6 +2602,7 @@@ static int tb_unreliable(AVCodecContex
         || c->time_base.den <    5L*c->time_base.num
  /*       || c->codec_tag == AV_RL32("DIVX")
         || c->codec_tag == AV_RL32("XVID")*/
 +       || c->codec_tag == AV_RL32("mp4v")
         || c->codec_id == AV_CODEC_ID_MPEG2VIDEO
         || c->codec_id == AV_CODEC_ID_H264
         )
      return 0;
  }
  
 +#if FF_API_FORMAT_PARAMETERS
 +int av_find_stream_info(AVFormatContext *ic)
 +{
 +    return avformat_find_stream_info(ic, NULL);
 +}
 +#endif
 +
  int avformat_find_stream_info(AVFormatContext *ic, AVDictionary **options)
  {
 -    int i, count, ret, read_size, j;
 +    int i, count, ret, j;
 +    int64_t read_size;
      AVStream *st;
      AVPacket pkt1, *pkt;
      int64_t old_offset = avio_tell(ic->pb);
      int orig_nb_streams = ic->nb_streams;        // new streams might appear, no options for those
 +    int flush_codecs = ic->probesize > 0;
 +
 +    if(ic->pb)
 +        av_log(ic, AV_LOG_DEBUG, "File position before avformat_find_stream_info() is %"PRId64"\n", avio_tell(ic->pb));
  
      for(i=0;i<ic->nb_streams;i++) {
          const AVCodec *codec;
          AVDictionary *thread_opt = NULL;
          st = ic->streams[i];
  
 +        if (st->codec->codec_type == AVMEDIA_TYPE_VIDEO ||
 +            st->codec->codec_type == AVMEDIA_TYPE_SUBTITLE) {
 +/*            if(!st->time_base.num)
 +                st->time_base= */
 +            if(!st->codec->time_base.num)
 +                st->codec->time_base= st->time_base;
 +        }
          //only for the split stuff
          if (!st->parser && !(ic->flags & AVFMT_FLAG_NOPARSE)) {
              st->parser = av_parser_init(st->codec->codec_id);
 -            if(st->need_parsing == AVSTREAM_PARSE_HEADERS && st->parser){
 -                st->parser->flags |= PARSER_FLAG_COMPLETE_FRAMES;
 +            if(st->parser){
 +                if(st->need_parsing == AVSTREAM_PARSE_HEADERS){
 +                    st->parser->flags |= PARSER_FLAG_COMPLETE_FRAMES;
 +                } else if(st->need_parsing == AVSTREAM_PARSE_FULL_RAW) {
 +                    st->parser->flags |= PARSER_FLAG_USE_CODEC_TS;
 +                }
 +            } else if (st->need_parsing) {
 +                av_log(ic, AV_LOG_VERBOSE, "parser not found for codec "
 +                       "%s, packets or times may be invalid.\n",
 +                       avcodec_get_name(st->codec->codec_id));
              }
          }
          codec = st->codec->codec ? st->codec->codec :
                                : &thread_opt);
  
          //try to just open decoders, in case this is enough to get parameters
 -        if (!has_codec_parameters(st)) {
 +        if (!has_codec_parameters(st, NULL) && st->request_probe <= 0) {
              if (codec && !st->codec->codec)
                  avcodec_open2(st->codec, codec, options ? &options[i]
                                : &thread_opt);
      }
  
      for (i=0; i<ic->nb_streams; i++) {
 +#if FF_API_R_FRAME_RATE
 +        ic->streams[i]->info->last_dts = AV_NOPTS_VALUE;
 +#endif
          ic->streams[i]->info->fps_first_dts = AV_NOPTS_VALUE;
          ic->streams[i]->info->fps_last_dts  = AV_NOPTS_VALUE;
      }
              int fps_analyze_framecount = 20;
  
              st = ic->streams[i];
 -            if (!has_codec_parameters(st))
 +            if (!has_codec_parameters(st, NULL))
                  break;
              /* if the timebase is coarse (like the usual millisecond precision
                 of mkv), we need to analyze more frames to reliably arrive at
                  fps_analyze_framecount *= 2;
              if (ic->fps_probe_size >= 0)
                  fps_analyze_framecount = ic->fps_probe_size;
 +            if (st->disposition & AV_DISPOSITION_ATTACHED_PIC)
 +                fps_analyze_framecount = 0;
              /* variable fps and no guess at the real fps */
 -            if(   tb_unreliable(st->codec) && !st->avg_frame_rate.num
 -               && st->codec_info_nb_frames < fps_analyze_framecount
 +            if(   tb_unreliable(st->codec) && !(st->r_frame_rate.num && st->avg_frame_rate.num)
 +               && st->info->duration_count < fps_analyze_framecount
                 && st->codec->codec_type == AVMEDIA_TYPE_VIDEO)
                  break;
              if(st->parser && st->parser->parser->split && !st->codec->extradata)
                  /* if we found the info for all the codecs, we can stop */
                  ret = count;
                  av_log(ic, AV_LOG_DEBUG, "All info found\n");
 +                flush_codecs = 0;
                  break;
              }
          }
          /* we did not get all the codec info, but we read too much data */
          if (read_size >= ic->probesize) {
              ret = count;
 -            av_log(ic, AV_LOG_DEBUG, "Probe buffer size limit %d reached\n", ic->probesize);
 +            av_log(ic, AV_LOG_DEBUG, "Probe buffer size limit of %d bytes reached\n", ic->probesize);
 +            for (i = 0; i < ic->nb_streams; i++)
 +                if (!ic->streams[i]->r_frame_rate.num &&
 +                    ic->streams[i]->info->duration_count <= 1)
 +                    av_log(ic, AV_LOG_WARNING,
 +                           "Stream #%d: not enough frames to estimate rate; "
 +                           "consider increasing probesize\n", i);
              break;
          }
  
  
          if (ret < 0) {
              /* EOF or error*/
 -            AVPacket empty_pkt = { 0 };
 -            int err = 0;
 -            av_init_packet(&empty_pkt);
 -
 -            ret = -1; /* we could not have all the codec parameters before EOF */
 -            for(i=0;i<ic->nb_streams;i++) {
 -                st = ic->streams[i];
 -
 -                /* flush the decoders */
 -                if (st->info->found_decoder == 1) {
 -                    do {
 -                        err = try_decode_frame(st, &empty_pkt,
 -                                               (options && i < orig_nb_streams) ?
 -                                               &options[i] : NULL);
 -                    } while (err > 0 && !has_codec_parameters(st));
 -                }
 -
 -                if (err < 0) {
 -                    av_log(ic, AV_LOG_WARNING,
 -                           "decoding for stream %d failed\n", st->index);
 -                } else if (!has_codec_parameters(st)) {
 -                    char buf[256];
 -                    avcodec_string(buf, sizeof(buf), st->codec, 0);
 -                    av_log(ic, AV_LOG_WARNING,
 -                           "Could not find codec parameters (%s)\n", buf);
 -                } else {
 -                    ret = 0;
 -                }
 -            }
              break;
          }
  
              /* check for non-increasing dts */
              if (st->info->fps_last_dts != AV_NOPTS_VALUE &&
                  st->info->fps_last_dts >= pkt->dts) {
 -                av_log(ic, AV_LOG_WARNING, "Non-increasing DTS in stream %d: "
 +                av_log(ic, AV_LOG_DEBUG, "Non-increasing DTS in stream %d: "
                         "packet %d with DTS %"PRId64", packet %d with DTS "
                         "%"PRId64"\n", st->index, st->info->fps_last_dts_idx,
                         st->info->fps_last_dts, st->codec_info_nb_frames, pkt->dts);
              }
              st->info->fps_last_dts = pkt->dts;
              st->info->fps_last_dts_idx = st->codec_info_nb_frames;
 -
 -            /* check max_analyze_duration */
 -            if (av_rescale_q(pkt->dts - st->info->fps_first_dts, st->time_base,
 -                             AV_TIME_BASE_Q) >= ic->max_analyze_duration) {
 -                av_log(ic, AV_LOG_WARNING, "max_analyze_duration reached\n");
 +        }
 +        if (st->codec_info_nb_frames>1) {
 +            int64_t t=0;
 +            if (st->time_base.den > 0)
 +                t = av_rescale_q(st->info->codec_info_duration, st->time_base, AV_TIME_BASE_Q);
 +            if (st->avg_frame_rate.num > 0)
 +                t = FFMAX(t, av_rescale_q(st->codec_info_nb_frames, av_inv_q(st->avg_frame_rate), AV_TIME_BASE_Q));
 +
 +            if (t >= ic->max_analyze_duration) {
 +                av_log(ic, AV_LOG_WARNING, "max_analyze_duration %d reached at %"PRId64" microseconds\n", ic->max_analyze_duration, t);
                  break;
              }
 +            if (pkt->duration) {
 +                st->info->codec_info_duration        += pkt->duration;
 +                st->info->codec_info_duration_fields += st->parser && st->codec->ticks_per_frame==2 ? st->parser->repeat_pict + 1 : 2;
 +            }
 +        }
 +#if FF_API_R_FRAME_RATE
 +        {
 +            int64_t last = st->info->last_dts;
 +
 +            if(   pkt->dts != AV_NOPTS_VALUE && last != AV_NOPTS_VALUE && pkt->dts > last
 +               && pkt->dts - (uint64_t)last < INT64_MAX){
 +                double dts= (is_relative(pkt->dts) ?  pkt->dts - RELATIVE_TS_BASE : pkt->dts) * av_q2d(st->time_base);
 +                int64_t duration= pkt->dts - last;
 +
 +                if (!st->info->duration_error)
 +                    st->info->duration_error = av_mallocz(sizeof(st->info->duration_error[0])*2);
 +
 +//                 if(st->codec->codec_type == AVMEDIA_TYPE_VIDEO)
 +//                     av_log(NULL, AV_LOG_ERROR, "%f\n", dts);
 +                for (i=0; i<MAX_STD_TIMEBASES; i++) {
 +                    int framerate= get_std_framerate(i);
 +                    double sdts= dts*framerate/(1001*12);
 +                    for(j=0; j<2; j++){
 +                        int64_t ticks= llrint(sdts+j*0.5);
 +                        double error= sdts - ticks + j*0.5;
 +                        st->info->duration_error[j][0][i] += error;
 +                        st->info->duration_error[j][1][i] += error*error;
 +                    }
 +                }
 +                st->info->duration_count++;
 +                // ignore the first 4 values, they might have some random jitter
 +                if (st->info->duration_count > 3 && is_relative(pkt->dts) == is_relative(last))
 +                    st->info->duration_gcd = av_gcd(st->info->duration_gcd, duration);
 +            }
 +            if (pkt->dts != AV_NOPTS_VALUE)
 +                st->info->last_dts = pkt->dts;
          }
 +#endif
          if(st->parser && st->parser->parser->split && !st->codec->extradata){
              int i= st->parser->parser->split(st->codec, pkt->data, pkt->size);
              if (i > 0 && i < FF_MAX_EXTRADATA_SIZE) {
          count++;
      }
  
 +    if (flush_codecs) {
 +        AVPacket empty_pkt = { 0 };
 +        int err = 0;
 +        av_init_packet(&empty_pkt);
 +
 +        ret = -1; /* we could not have all the codec parameters before EOF */
 +        for(i=0;i<ic->nb_streams;i++) {
 +            const char *errmsg;
 +
 +            st = ic->streams[i];
 +
 +            /* flush the decoders */
 +            if (st->info->found_decoder == 1) {
 +                do {
 +                    err = try_decode_frame(st, &empty_pkt,
 +                                            (options && i < orig_nb_streams) ?
 +                                            &options[i] : NULL);
 +                } while (err > 0 && !has_codec_parameters(st, NULL));
 +
 +                if (err < 0) {
 +                    av_log(ic, AV_LOG_INFO,
 +                        "decoding for stream %d failed\n", st->index);
 +                }
 +            }
 +
 +            if (!has_codec_parameters(st, &errmsg)) {
 +                char buf[256];
 +                avcodec_string(buf, sizeof(buf), st->codec, 0);
 +                av_log(ic, AV_LOG_WARNING,
 +                       "Could not find codec parameters for stream %d (%s): %s\n"
 +                       "Consider increasing the value for the 'analyzeduration' and 'probesize' options\n",
 +                       i, buf, errmsg);
 +            } else {
 +                ret = 0;
 +            }
 +        }
 +    }
 +
      // close codecs which were opened in try_decode_frame()
      for(i=0;i<ic->nb_streams;i++) {
          st = ic->streams[i];
      for(i=0;i<ic->nb_streams;i++) {
          st = ic->streams[i];
          if (st->codec->codec_type == AVMEDIA_TYPE_VIDEO) {
 +            if(st->codec->codec_id == AV_CODEC_ID_RAWVIDEO && !st->codec->codec_tag && !st->codec->bits_per_coded_sample){
 +                uint32_t tag= avcodec_pix_fmt_to_codec_tag(st->codec->pix_fmt);
 +                if (avpriv_find_pix_fmt(ff_raw_pix_fmt_tags, tag) == st->codec->pix_fmt)
 +                    st->codec->codec_tag= tag;
 +            }
 +
              /* estimate average framerate if not set by demuxer */
 -            if (!st->avg_frame_rate.num && st->info->fps_last_dts != st->info->fps_first_dts) {
 -                int64_t delta_dts = st->info->fps_last_dts - st->info->fps_first_dts;
 -                int delta_packets = st->info->fps_last_dts_idx - st->info->fps_first_dts_idx;
 +            if (st->info->codec_info_duration_fields && !st->avg_frame_rate.num && st->info->codec_info_duration) {
                  int      best_fps = 0;
                  double best_error = 0.01;
  
                  av_reduce(&st->avg_frame_rate.num, &st->avg_frame_rate.den,
 -                          delta_packets*(int64_t)st->time_base.den,
 -                          delta_dts*(int64_t)st->time_base.num, 60000);
 +                          st->info->codec_info_duration_fields*(int64_t)st->time_base.den,
 +                          st->info->codec_info_duration*2*(int64_t)st->time_base.num, 60000);
  
                  /* round guessed framerate to a "standard" framerate if it's
                   * within 1% of the original estimate*/
                                best_fps, 12*1001, INT_MAX);
                  }
              }
 +            // the check for tb_unreliable() is not completely correct, since this is not about handling
 +            // a unreliable/inexact time base, but a time base that is finer than necessary, as e.g.
 +            // ipmovie.c produces.
 +            if (tb_unreliable(st->codec) && st->info->duration_count > 15 && st->info->duration_gcd > FFMAX(1, st->time_base.den/(500LL*st->time_base.num)) && !st->r_frame_rate.num)
 +                av_reduce(&st->r_frame_rate.num, &st->r_frame_rate.den, st->time_base.den, st->time_base.num * st->info->duration_gcd, INT_MAX);
 +            if (st->info->duration_count>1 && !st->r_frame_rate.num
 +                && tb_unreliable(st->codec)) {
 +                int num = 0;
 +                double best_error= 0.01;
 +
 +                for (j=0; j<MAX_STD_TIMEBASES; j++) {
 +                    int k;
 +
 +                    if(st->info->codec_info_duration && st->info->codec_info_duration*av_q2d(st->time_base) < (1001*12.0)/get_std_framerate(j))
 +                        continue;
 +                    if(!st->info->codec_info_duration && 1.0 < (1001*12.0)/get_std_framerate(j))
 +                        continue;
 +                    for(k=0; k<2; k++){
 +                        int n= st->info->duration_count;
 +                        double a= st->info->duration_error[k][0][j] / n;
 +                        double error= st->info->duration_error[k][1][j]/n - a*a;
 +
 +                        if(error < best_error && best_error> 0.000000001){
 +                            best_error= error;
 +                            num = get_std_framerate(j);
 +                        }
 +                        if(error < 0.02)
 +                            av_log(NULL, AV_LOG_DEBUG, "rfps: %f %f\n", get_std_framerate(j) / 12.0/1001, error);
 +                    }
 +                }
 +                // do not increase frame rate by more than 1 % in order to match a standard rate.
 +                if (num && (!st->r_frame_rate.num || (double)num/(12*1001) < 1.01 * av_q2d(st->r_frame_rate)))
 +                    av_reduce(&st->r_frame_rate.num, &st->r_frame_rate.den, num, 12*1001, INT_MAX);
 +            }
 +
 +            if (!st->r_frame_rate.num){
 +                if(    st->codec->time_base.den * (int64_t)st->time_base.num
 +                    <= st->codec->time_base.num * st->codec->ticks_per_frame * (int64_t)st->time_base.den){
 +                    st->r_frame_rate.num = st->codec->time_base.den;
 +                    st->r_frame_rate.den = st->codec->time_base.num * st->codec->ticks_per_frame;
 +                }else{
 +                    st->r_frame_rate.num = st->time_base.den;
 +                    st->r_frame_rate.den = st->time_base.num;
 +                }
 +            }
          }else if(st->codec->codec_type == AVMEDIA_TYPE_AUDIO) {
              if(!st->codec->bits_per_coded_sample)
                  st->codec->bits_per_coded_sample= av_get_bits_per_sample(st->codec->codec_id);
          }
      }
  
 +    if(ic->probesize)
      estimate_timings(ic, old_offset);
  
      compute_chapters_end(ic);
  
   find_stream_info_err:
      for (i=0; i < ic->nb_streams; i++) {
 -        if (ic->streams[i]->codec)
 +        st = ic->streams[i];
 +        if (ic->streams[i]->codec && ic->streams[i]->codec->codec_type != AVMEDIA_TYPE_AUDIO)
              ic->streams[i]->codec->thread_count = 0;
 +        if (st->info)
 +            av_freep(&st->info->duration_error);
          av_freep(&ic->streams[i]->info);
      }
 +    if(ic->pb)
 +        av_log(ic, AV_LOG_DEBUG, "File position after avformat_find_stream_info() is %"PRId64"\n", avio_tell(ic->pb));
      return ret;
  }
  
 -static AVProgram *find_program_from_stream(AVFormatContext *ic, int s)
 +AVProgram *av_find_program_from_stream(AVFormatContext *ic, AVProgram *last, int s)
  {
      int i, j;
  
 -    for (i = 0; i < ic->nb_programs; i++)
 -        for (j = 0; j < ic->programs[i]->nb_stream_indexes; j++)
 -            if (ic->programs[i]->stream_index[j] == s)
 -                return ic->programs[i];
 +    for (i = 0; i < ic->nb_programs; i++) {
 +        if (ic->programs[i] == last) {
 +            last = NULL;
 +        } else {
 +            if (!last)
 +                for (j = 0; j < ic->programs[i]->nb_stream_indexes; j++)
 +                    if (ic->programs[i]->stream_index[j] == s)
 +                        return ic->programs[i];
 +        }
 +    }
      return NULL;
  }
  
@@@ -3066,12 -2426,12 +3066,12 @@@ int av_find_best_stream(AVFormatContex
                          int flags)
  {
      int i, nb_streams = ic->nb_streams;
 -    int ret = AVERROR_STREAM_NOT_FOUND, best_count = -1;
 +    int ret = AVERROR_STREAM_NOT_FOUND, best_count = -1, best_bitrate = -1, best_multiframe = -1, count, bitrate, multiframe;
      unsigned *program = NULL;
      AVCodec *decoder = NULL, *best_decoder = NULL;
  
      if (related_stream >= 0 && wanted_stream_nb < 0) {
 -        AVProgram *p = find_program_from_stream(ic, related_stream);
 +        AVProgram *p = av_find_program_from_stream(ic, NULL, related_stream);
          if (p) {
              program = p->stream_index;
              nb_streams = p->nb_stream_indexes;
                  continue;
              }
          }
 -        if (best_count >= st->codec_info_nb_frames)
 +        count = st->codec_info_nb_frames;
 +        bitrate = avctx->bit_rate;
 +        multiframe = FFMIN(5, count);
 +        if ((best_multiframe >  multiframe) ||
 +            (best_multiframe == multiframe && best_bitrate >  bitrate) ||
 +            (best_multiframe == multiframe && best_bitrate == bitrate && best_count >= count))
              continue;
 -        best_count = st->codec_info_nb_frames;
 +        best_count = count;
 +        best_bitrate = bitrate;
 +        best_multiframe = multiframe;
          ret = real_stream_index;
          best_decoder = decoder;
          if (program && i == nb_streams - 1 && ret < 0) {
@@@ -3138,41 -2491,32 +3138,41 @@@ int av_read_pause(AVFormatContext *s
      return AVERROR(ENOSYS);
  }
  
 +void ff_free_stream(AVFormatContext *s, AVStream *st){
 +    av_assert0(s->nb_streams>0);
 +    av_assert0(s->streams[ s->nb_streams-1 ] == st);
 +
 +    if (st->parser) {
 +        av_parser_close(st->parser);
 +    }
 +    if (st->attached_pic.data)
 +        av_free_packet(&st->attached_pic);
 +    av_dict_free(&st->metadata);
 +    av_freep(&st->probe_data.buf);
 +    av_freep(&st->index_entries);
 +    av_freep(&st->codec->extradata);
 +    av_freep(&st->codec->subtitle_header);
 +    av_freep(&st->codec);
 +    av_freep(&st->priv_data);
 +    if (st->info)
 +        av_freep(&st->info->duration_error);
 +    av_freep(&st->info);
 +    av_freep(&s->streams[ --s->nb_streams ]);
 +}
 +
  void avformat_free_context(AVFormatContext *s)
  {
      int i;
 -    AVStream *st;
 +
 +    if (!s)
 +        return;
  
      av_opt_free(s);
      if (s->iformat && s->iformat->priv_class && s->priv_data)
          av_opt_free(s->priv_data);
  
 -    for(i=0;i<s->nb_streams;i++) {
 -        /* free all data in a stream component */
 -        st = s->streams[i];
 -        if (st->parser) {
 -            av_parser_close(st->parser);
 -        }
 -        if (st->attached_pic.data)
 -            av_free_packet(&st->attached_pic);
 -        av_dict_free(&st->metadata);
 -        av_freep(&st->probe_data.buf);
 -        av_free(st->index_entries);
 -        av_free(st->codec->extradata);
 -        av_free(st->codec->subtitle_header);
 -        av_free(st->codec);
 -        av_free(st->priv_data);
 -        av_free(st->info);
 -        av_free(st);
 +    for(i=s->nb_streams-1; i>=0; i--) {
 +        ff_free_stream(s, s->streams[i]);
      }
      for(i=s->nb_programs-1; i>=0; i--) {
          av_dict_free(&s->programs[i]->metadata);
      av_freep(&s->priv_data);
      while(s->nb_chapters--) {
          av_dict_free(&s->chapters[s->nb_chapters]->metadata);
 -        av_free(s->chapters[s->nb_chapters]);
 +        av_freep(&s->chapters[s->nb_chapters]);
      }
      av_freep(&s->chapters);
      av_dict_free(&s->metadata);
      av_free(s);
  }
  
 +#if FF_API_CLOSE_INPUT_FILE
 +void av_close_input_file(AVFormatContext *s)
 +{
 +    avformat_close_input(&s);
 +}
 +#endif
 +
  void avformat_close_input(AVFormatContext **ps)
  {
      AVFormatContext *s = *ps;
      avio_close(pb);
  }
  
 -AVStream *avformat_new_stream(AVFormatContext *s, AVCodec *c)
 +#if FF_API_NEW_STREAM
 +AVStream *av_new_stream(AVFormatContext *s, int id)
 +{
 +    AVStream *st = avformat_new_stream(s, NULL);
 +    if (st)
 +        st->id = id;
 +    return st;
 +}
 +#endif
 +
 +AVStream *avformat_new_stream(AVFormatContext *s, const AVCodec *c)
  {
      AVStream *st;
      int i;
          av_free(st);
          return NULL;
      }
 +    st->info->last_dts = AV_NOPTS_VALUE;
  
      st->codec = avcodec_alloc_context3(c);
      if (s->iformat) {
             but durations get some timestamps, formats with some unknown
             timestamps have their first few packets buffered and the
             timestamps corrected before they are returned to the user */
 -    st->cur_dts = 0;
 +    st->cur_dts = s->iformat ? RELATIVE_TS_BASE : 0;
      st->first_dts = AV_NOPTS_VALUE;
      st->probe_packets = MAX_PROBE_PACKETS;
 +    st->pts_wrap_reference = AV_NOPTS_VALUE;
 +    st->pts_wrap_behavior = AV_PTS_WRAP_IGNORE;
  
      /* default pts setting is MPEG-like */
      avpriv_set_pts_info(st, 33, 1, 90000);
  
      st->sample_aspect_ratio = (AVRational){0,1};
  
 +#if FF_API_R_FRAME_RATE
 +    st->info->last_dts      = AV_NOPTS_VALUE;
 +#endif
      st->info->fps_first_dts = AV_NOPTS_VALUE;
      st->info->fps_last_dts  = AV_NOPTS_VALUE;
  
@@@ -3309,11 -2630,6 +3309,11 @@@ AVProgram *av_new_program(AVFormatConte
          program->discard = AVDISCARD_NONE;
      }
      program->id = id;
 +    program->pts_wrap_reference = AV_NOPTS_VALUE;
 +    program->pts_wrap_behavior = AV_PTS_WRAP_IGNORE;
 +
 +    program->start_time =
 +    program->end_time   = AV_NOPTS_VALUE;
  
      return program;
  }
@@@ -3384,21 -2700,8 +3384,21 @@@ static void dump_metadata(void *ctx, AV
  
          av_log(ctx, AV_LOG_INFO, "%sMetadata:\n", indent);
          while((tag=av_dict_get(m, "", tag, AV_DICT_IGNORE_SUFFIX))) {
 -            if(strcmp("language", tag->key))
 -                av_log(ctx, AV_LOG_INFO, "%s  %-16s: %s\n", indent, tag->key, tag->value);
 +            if(strcmp("language", tag->key)){
 +                const char *p = tag->value;
 +                av_log(ctx, AV_LOG_INFO, "%s  %-16s: ", indent, tag->key);
 +                while(*p) {
 +                    char tmp[256];
 +                    size_t len = strcspn(p, "\x8\xa\xb\xc\xd");
 +                    av_strlcpy(tmp, p, FFMIN(sizeof(tmp), len+1));
 +                    av_log(ctx, AV_LOG_INFO, "%s", tmp);
 +                    p += len;
 +                    if (*p == 0xd) av_log(ctx, AV_LOG_INFO, " ");
 +                    if (*p == 0xa) av_log(ctx, AV_LOG_INFO, "\n%s  %-16s: ", indent, "");
 +                    if (*p) p++;
 +                }
 +                av_log(ctx, AV_LOG_INFO, "\n");
 +            }
          }
      }
  }
@@@ -3412,7 -2715,7 +3412,7 @@@ static void dump_stream_format(AVFormat
      int g = av_gcd(st->time_base.num, st->time_base.den);
      AVDictionaryEntry *lang = av_dict_get(st->metadata, "language", NULL, 0);
      avcodec_string(buf, sizeof(buf), st->codec, is_output);
 -    av_log(NULL, AV_LOG_INFO, "    Stream #%d.%d", index, i);
 +    av_log(NULL, AV_LOG_INFO, "    Stream #%d:%d", index, i);
      /* the pid is an important information, so we display it */
      /* XXX: add a generic system */
      if (flags & AVFMT_SHOW_IDS)
                    st->codec->width*st->sample_aspect_ratio.num,
                    st->codec->height*st->sample_aspect_ratio.den,
                    1024*1024);
 -        av_log(NULL, AV_LOG_INFO, ", PAR %d:%d DAR %d:%d",
 +        av_log(NULL, AV_LOG_INFO, ", SAR %d:%d DAR %d:%d",
                   st->sample_aspect_ratio.num, st->sample_aspect_ratio.den,
                   display_aspect_ratio.num, display_aspect_ratio.den);
      }
      if(st->codec->codec_type == AVMEDIA_TYPE_VIDEO){
          if(st->avg_frame_rate.den && st->avg_frame_rate.num)
              print_fps(av_q2d(st->avg_frame_rate), "fps");
 +#if FF_API_R_FRAME_RATE
 +        if(st->r_frame_rate.den && st->r_frame_rate.num)
 +            print_fps(av_q2d(st->r_frame_rate), "tbr");
 +#endif
          if(st->time_base.den && st->time_base.num)
              print_fps(1/av_q2d(st->time_base), "tbn");
          if(st->codec->time_base.den && st->codec->time_base.num)
@@@ -3488,9 -2787,8 +3488,9 @@@ void av_dump_format(AVFormatContext *ic
          av_log(NULL, AV_LOG_INFO, "  Duration: ");
          if (ic->duration != AV_NOPTS_VALUE) {
              int hours, mins, secs, us;
 -            secs = ic->duration / AV_TIME_BASE;
 -            us = ic->duration % AV_TIME_BASE;
 +            int64_t duration = ic->duration + 5000;
 +            secs = duration / AV_TIME_BASE;
 +            us = duration % AV_TIME_BASE;
              mins = secs / 60;
              secs %= 60;
              hours = mins / 60;
@@@ -3672,27 -2970,11 +3672,27 @@@ static void pkt_dump_internal(void *avc
          av_hex_dump(f, pkt->data, pkt->size);
  }
  
 +#if FF_API_PKT_DUMP
 +void av_pkt_dump(FILE *f, AVPacket *pkt, int dump_payload)
 +{
 +    AVRational tb = { 1, AV_TIME_BASE };
 +    pkt_dump_internal(NULL, f, 0, pkt, dump_payload, tb);
 +}
 +#endif
 +
  void av_pkt_dump2(FILE *f, AVPacket *pkt, int dump_payload, AVStream *st)
  {
      pkt_dump_internal(NULL, f, 0, pkt, dump_payload, st->time_base);
  }
  
 +#if FF_API_PKT_DUMP
 +void av_pkt_dump_log(void *avcl, int level, AVPacket *pkt, int dump_payload)
 +{
 +    AVRational tb = { 1, AV_TIME_BASE };
 +    pkt_dump_internal(avcl, NULL, level, pkt, dump_payload, tb);
 +}
 +#endif
 +
  void av_pkt_dump_log2(void *avcl, int level, AVPacket *pkt, int dump_payload,
                        AVStream *st)
  {
@@@ -3706,7 -2988,7 +3706,7 @@@ void av_url_split(char *proto, int prot
                    char *path, int path_size,
                    const char *url)
  {
 -    const char *p, *ls, *at, *col, *brk;
 +    const char *p, *ls, *ls2, *at, *at2, *col, *brk;
  
      if (port_ptr)               *port_ptr = -1;
      if (proto_size > 0)         proto[0] = 0;
  
      /* separate path from hostname */
      ls = strchr(p, '/');
 +    ls2 = strchr(p, '?');
      if(!ls)
 -        ls = strchr(p, '?');
 +        ls = ls2;
 +    else if (ls && ls2)
 +        ls = FFMIN(ls, ls2);
      if(ls)
          av_strlcpy(path, ls, path_size);
      else
      /* the rest is hostname, use that to parse auth/port */
      if (ls != p) {
          /* authorization (user[:pass]@hostname) */
 -        if ((at = strchr(p, '@')) && at < ls) {
 -            av_strlcpy(authorization, p,
 -                       FFMIN(authorization_size, at + 1 - p));
 +        at2 = p;
 +        while ((at = strchr(p, '@')) && at < ls) {
 +            av_strlcpy(authorization, at2,
 +                       FFMIN(authorization_size, at + 1 - at2));
              p = at + 1; /* skip '@' */
          }
  
@@@ -3813,14 -3091,6 +3813,14 @@@ int ff_hex_to_data(uint8_t *data, cons
      return len;
  }
  
 +#if FF_API_SET_PTS_INFO
 +void av_set_pts_info(AVStream *s, int pts_wrap_bits,
 +                     unsigned int pts_num, unsigned int pts_den)
 +{
 +    avpriv_set_pts_info(s, pts_wrap_bits, pts_num, pts_den);
 +}
 +#endif
 +
  void avpriv_set_pts_info(AVStream *s, int pts_wrap_bits,
                           unsigned int pts_num, unsigned int pts_den)
  {
          av_log(NULL, AV_LOG_WARNING, "st:%d has too large timebase, reducing\n", s->index);
  
      if(new_tb.num <= 0 || new_tb.den <= 0) {
 -        av_log(NULL, AV_LOG_ERROR, "Ignoring attempt to set invalid timebase for st:%d\n", s->index);
 +        av_log(NULL, AV_LOG_ERROR, "Ignoring attempt to set invalid timebase %d/%d for st:%d\n", new_tb.num, new_tb.den, s->index);
          return;
      }
      s->time_base = new_tb;
 +    av_codec_set_pkt_timebase(s->codec, new_tb);
      s->pts_wrap_bits = pts_wrap_bits;
  }
  
@@@ -3884,28 -3153,6 +3884,6 @@@ int ff_url_join(char *str, int size, co
      return strlen(str);
  }
  
- int ff_write_chained(AVFormatContext *dst, int dst_stream, AVPacket *pkt,
-                      AVFormatContext *src)
- {
-     AVPacket local_pkt;
-     local_pkt = *pkt;
-     local_pkt.stream_index = dst_stream;
-     if (pkt->pts != AV_NOPTS_VALUE)
-         local_pkt.pts = av_rescale_q(pkt->pts,
-                                      src->streams[pkt->stream_index]->time_base,
-                                      dst->streams[dst_stream]->time_base);
-     if (pkt->dts != AV_NOPTS_VALUE)
-         local_pkt.dts = av_rescale_q(pkt->dts,
-                                      src->streams[pkt->stream_index]->time_base,
-                                      dst->streams[dst_stream]->time_base);
-     if (pkt->duration)
-         local_pkt.duration = av_rescale_q(pkt->duration,
-                                           src->streams[pkt->stream_index]->time_base,
-                                           dst->streams[dst_stream]->time_base);
-     return av_write_frame(dst, &local_pkt);
- }
  void ff_parse_key_value(const char *str, ff_parse_key_val_cb callback_get_buf,
                          void *context)
  {
@@@ -4041,14 -3288,20 +4019,14 @@@ void ff_make_absolute_url(char *buf, in
  
  int64_t ff_iso8601_to_unix_time(const char *datestr)
  {
 -#if HAVE_STRPTIME
      struct tm time1 = {0}, time2 = {0};
      char *ret1, *ret2;
 -    ret1 = strptime(datestr, "%Y - %m - %d %T", &time1);
 -    ret2 = strptime(datestr, "%Y - %m - %dT%T", &time2);
 +    ret1 = av_small_strptime(datestr, "%Y - %m - %d %H:%M:%S", &time1);
 +    ret2 = av_small_strptime(datestr, "%Y - %m - %dT%H:%M:%S", &time2);
      if (ret2 && !ret1)
          return av_timegm(&time2);
      else
          return av_timegm(&time1);
 -#else
 -    av_log(NULL, AV_LOG_WARNING, "strptime() unavailable on this system, cannot convert "
 -                                 "the date string.\n");
 -    return 0;
 -#endif
  }
  
  int avformat_query_codec(AVOutputFormat *ofmt, enum AVCodecID codec_id, int std_compliance)
@@@ -4127,198 -3380,3 +4105,198 @@@ int ff_add_param_change(AVPacket *pkt, 
      }
      return 0;
  }
 +
 +AVRational av_guess_sample_aspect_ratio(AVFormatContext *format, AVStream *stream, AVFrame *frame)
 +{
 +    AVRational undef = {0, 1};
 +    AVRational stream_sample_aspect_ratio = stream ? stream->sample_aspect_ratio : undef;
 +    AVRational codec_sample_aspect_ratio  = stream && stream->codec ? stream->codec->sample_aspect_ratio : undef;
 +    AVRational frame_sample_aspect_ratio  = frame  ? frame->sample_aspect_ratio  : codec_sample_aspect_ratio;
 +
 +    av_reduce(&stream_sample_aspect_ratio.num, &stream_sample_aspect_ratio.den,
 +               stream_sample_aspect_ratio.num,  stream_sample_aspect_ratio.den, INT_MAX);
 +    if (stream_sample_aspect_ratio.num <= 0 || stream_sample_aspect_ratio.den <= 0)
 +        stream_sample_aspect_ratio = undef;
 +
 +    av_reduce(&frame_sample_aspect_ratio.num, &frame_sample_aspect_ratio.den,
 +               frame_sample_aspect_ratio.num,  frame_sample_aspect_ratio.den, INT_MAX);
 +    if (frame_sample_aspect_ratio.num <= 0 || frame_sample_aspect_ratio.den <= 0)
 +        frame_sample_aspect_ratio = undef;
 +
 +    if (stream_sample_aspect_ratio.num)
 +        return stream_sample_aspect_ratio;
 +    else
 +        return frame_sample_aspect_ratio;
 +}
 +
 +AVRational av_guess_frame_rate(AVFormatContext *format, AVStream *st, AVFrame *frame)
 +{
 +    AVRational fr = st->r_frame_rate;
 +
 +    if (st->codec->ticks_per_frame > 1) {
 +        AVRational codec_fr = av_inv_q(st->codec->time_base);
 +        AVRational   avg_fr = st->avg_frame_rate;
 +        codec_fr.den *= st->codec->ticks_per_frame;
 +        if (   codec_fr.num > 0 && codec_fr.den > 0 && av_q2d(codec_fr) < av_q2d(fr)*0.7
 +            && fabs(1.0 - av_q2d(av_div_q(avg_fr, fr))) > 0.1)
 +            fr = codec_fr;
 +    }
 +
 +    return fr;
 +}
 +
 +int avformat_match_stream_specifier(AVFormatContext *s, AVStream *st,
 +                                    const char *spec)
 +{
 +    if (*spec <= '9' && *spec >= '0') /* opt:index */
 +        return strtol(spec, NULL, 0) == st->index;
 +    else if (*spec == 'v' || *spec == 'a' || *spec == 's' || *spec == 'd' ||
 +             *spec == 't') { /* opt:[vasdt] */
 +        enum AVMediaType type;
 +
 +        switch (*spec++) {
 +        case 'v': type = AVMEDIA_TYPE_VIDEO;      break;
 +        case 'a': type = AVMEDIA_TYPE_AUDIO;      break;
 +        case 's': type = AVMEDIA_TYPE_SUBTITLE;   break;
 +        case 'd': type = AVMEDIA_TYPE_DATA;       break;
 +        case 't': type = AVMEDIA_TYPE_ATTACHMENT; break;
 +        default:  av_assert0(0);
 +        }
 +        if (type != st->codec->codec_type)
 +            return 0;
 +        if (*spec++ == ':') { /* possibly followed by :index */
 +            int i, index = strtol(spec, NULL, 0);
 +            for (i = 0; i < s->nb_streams; i++)
 +                if (s->streams[i]->codec->codec_type == type && index-- == 0)
 +                   return i == st->index;
 +            return 0;
 +        }
 +        return 1;
 +    } else if (*spec == 'p' && *(spec + 1) == ':') {
 +        int prog_id, i, j;
 +        char *endptr;
 +        spec += 2;
 +        prog_id = strtol(spec, &endptr, 0);
 +        for (i = 0; i < s->nb_programs; i++) {
 +            if (s->programs[i]->id != prog_id)
 +                continue;
 +
 +            if (*endptr++ == ':') {
 +                int stream_idx = strtol(endptr, NULL, 0);
 +                return stream_idx >= 0 &&
 +                    stream_idx < s->programs[i]->nb_stream_indexes &&
 +                    st->index == s->programs[i]->stream_index[stream_idx];
 +            }
 +
 +            for (j = 0; j < s->programs[i]->nb_stream_indexes; j++)
 +                if (st->index == s->programs[i]->stream_index[j])
 +                    return 1;
 +        }
 +        return 0;
 +    } else if (*spec == '#') {
 +        int sid;
 +        char *endptr;
 +        sid = strtol(spec + 1, &endptr, 0);
 +        if (!*endptr)
 +            return st->id == sid;
 +    } else if (!*spec) /* empty specifier, matches everything */
 +        return 1;
 +
 +    av_log(s, AV_LOG_ERROR, "Invalid stream specifier: %s.\n", spec);
 +    return AVERROR(EINVAL);
 +}
 +
 +void ff_generate_avci_extradata(AVStream *st)
 +{
 +    static const uint8_t avci100_1080p_extradata[] = {
 +        // SPS
 +        0x00, 0x00, 0x00, 0x01, 0x67, 0x7a, 0x10, 0x29,
 +        0xb6, 0xd4, 0x20, 0x22, 0x33, 0x19, 0xc6, 0x63,
 +        0x23, 0x21, 0x01, 0x11, 0x98, 0xce, 0x33, 0x19,
 +        0x18, 0x21, 0x02, 0x56, 0xb9, 0x3d, 0x7d, 0x7e,
 +        0x4f, 0xe3, 0x3f, 0x11, 0xf1, 0x9e, 0x08, 0xb8,
 +        0x8c, 0x54, 0x43, 0xc0, 0x78, 0x02, 0x27, 0xe2,
 +        0x70, 0x1e, 0x30, 0x10, 0x10, 0x14, 0x00, 0x00,
 +        0x03, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0xca,
 +        0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 +        // PPS
 +        0x00, 0x00, 0x00, 0x01, 0x68, 0xce, 0x33, 0x48,
 +        0xd0
 +    };
 +    static const uint8_t avci100_1080i_extradata[] = {
 +        // SPS
 +        0x00, 0x00, 0x00, 0x01, 0x67, 0x7a, 0x10, 0x29,
 +        0xb6, 0xd4, 0x20, 0x22, 0x33, 0x19, 0xc6, 0x63,
 +        0x23, 0x21, 0x01, 0x11, 0x98, 0xce, 0x33, 0x19,
 +        0x18, 0x21, 0x03, 0x3a, 0x46, 0x65, 0x6a, 0x65,
 +        0x24, 0xad, 0xe9, 0x12, 0x32, 0x14, 0x1a, 0x26,
 +        0x34, 0xad, 0xa4, 0x41, 0x82, 0x23, 0x01, 0x50,
 +        0x2b, 0x1a, 0x24, 0x69, 0x48, 0x30, 0x40, 0x2e,
 +        0x11, 0x12, 0x08, 0xc6, 0x8c, 0x04, 0x41, 0x28,
 +        0x4c, 0x34, 0xf0, 0x1e, 0x01, 0x13, 0xf2, 0xe0,
 +        0x3c, 0x60, 0x20, 0x20, 0x28, 0x00, 0x00, 0x03,
 +        0x00, 0x08, 0x00, 0x00, 0x03, 0x01, 0x94, 0x00,
 +        // PPS
 +        0x00, 0x00, 0x00, 0x01, 0x68, 0xce, 0x33, 0x48,
 +        0xd0
 +    };
 +    static const uint8_t avci50_1080i_extradata[] = {
 +        // SPS
 +        0x00, 0x00, 0x00, 0x01, 0x67, 0x6e, 0x10, 0x28,
 +        0xa6, 0xd4, 0x20, 0x32, 0x33, 0x0c, 0x71, 0x18,
 +        0x88, 0x62, 0x10, 0x19, 0x19, 0x86, 0x38, 0x8c,
 +        0x44, 0x30, 0x21, 0x02, 0x56, 0x4e, 0x6e, 0x61,
 +        0x87, 0x3e, 0x73, 0x4d, 0x98, 0x0c, 0x03, 0x06,
 +        0x9c, 0x0b, 0x73, 0xe6, 0xc0, 0xb5, 0x18, 0x63,
 +        0x0d, 0x39, 0xe0, 0x5b, 0x02, 0xd4, 0xc6, 0x19,
 +        0x1a, 0x79, 0x8c, 0x32, 0x34, 0x24, 0xf0, 0x16,
 +        0x81, 0x13, 0xf7, 0xff, 0x80, 0x02, 0x00, 0x01,
 +        0xf1, 0x80, 0x80, 0x80, 0xa0, 0x00, 0x00, 0x03,
 +        0x00, 0x20, 0x00, 0x00, 0x06, 0x50, 0x80, 0x00,
 +        // PPS
 +        0x00, 0x00, 0x00, 0x01, 0x68, 0xee, 0x31, 0x12,
 +        0x11
 +    };
 +    static const uint8_t avci100_720p_extradata[] = {
 +        // SPS
 +        0x00, 0x00, 0x00, 0x01, 0x67, 0x7a, 0x10, 0x29,
 +        0xb6, 0xd4, 0x20, 0x2a, 0x33, 0x1d, 0xc7, 0x62,
 +        0xa1, 0x08, 0x40, 0x54, 0x66, 0x3b, 0x8e, 0xc5,
 +        0x42, 0x02, 0x10, 0x25, 0x64, 0x2c, 0x89, 0xe8,
 +        0x85, 0xe4, 0x21, 0x4b, 0x90, 0x83, 0x06, 0x95,
 +        0xd1, 0x06, 0x46, 0x97, 0x20, 0xc8, 0xd7, 0x43,
 +        0x08, 0x11, 0xc2, 0x1e, 0x4c, 0x91, 0x0f, 0x01,
 +        0x40, 0x16, 0xec, 0x07, 0x8c, 0x04, 0x04, 0x05,
 +        0x00, 0x00, 0x03, 0x00, 0x01, 0x00, 0x00, 0x03,
 +        0x00, 0x64, 0x84, 0x00, 0x00, 0x00, 0x00, 0x00,
 +        // PPS
 +        0x00, 0x00, 0x00, 0x01, 0x68, 0xce, 0x31, 0x12,
 +        0x11
 +    };
 +    int size = 0;
 +    const uint8_t *data = 0;
 +    if (st->codec->width == 1920) {
 +        if (st->codec->field_order == AV_FIELD_PROGRESSIVE) {
 +            data = avci100_1080p_extradata;
 +            size = sizeof(avci100_1080p_extradata);
 +        } else {
 +            data = avci100_1080i_extradata;
 +            size = sizeof(avci100_1080i_extradata);
 +        }
 +    } else if (st->codec->width == 1440) {
 +        data = avci50_1080i_extradata;
 +        size = sizeof(avci50_1080i_extradata);
 +    } else if (st->codec->width == 1280) {
 +        data = avci100_720p_extradata;
 +        size = sizeof(avci100_720p_extradata);
 +    }
 +    if (!size)
 +        return;
 +    av_freep(&st->codec->extradata);
 +    st->codec->extradata_size = 0;
 +    st->codec->extradata = av_mallocz(size + FF_INPUT_BUFFER_PADDING_SIZE);
 +    if (!st->codec->extradata)
 +        return;
 +    memcpy(st->codec->extradata, data, size);
 +    st->codec->extradata_size = size;
 +}