Skip to content
Snippets Groups Projects
utils.c 124 KiB
Newer Older
  • Learn to ignore specific revisions
  • Michael Niedermayer's avatar
    Michael Niedermayer committed
            st->last_IP_pts = AV_NOPTS_VALUE;
    
    static void av_estimate_timings(AVFormatContext *ic, int64_t old_offset)
    
    {
        int64_t file_size;
    
        /* get the file size, if possible */
        if (ic->iformat->flags & AVFMT_NOFILE) {
            file_size = 0;
        } else {
    
            file_size = avio_size(ic->pb);
    
            if (file_size < 0)
                file_size = 0;
        }
        ic->file_size = file_size;
    
    
        if ((!strcmp(ic->iformat->name, "mpeg") ||
             !strcmp(ic->iformat->name, "mpegts")) &&
    
            file_size && ic->pb->seekable) {
    
            /* get accurate estimate from the PTSes */
    
            av_estimate_timings_from_pts(ic, old_offset);
    
        } else if (av_has_duration(ic)) {
    
            /* at least one component has timings - we use them for all
    
               the components */
            fill_all_stream_timings(ic);
        } else {
    
            av_log(ic, AV_LOG_WARNING, "Estimating duration from bitrate, this may be inaccurate\n");
    
            /* less precise: use bitrate info */
    
            av_estimate_timings_from_bit_rate(ic);
        }
        av_update_stream_timings(ic);
    
    #if 0
        {
            int i;
            AVStream *st;
            for(i = 0;i < ic->nb_streams; i++) {
                st = ic->streams[i];
    
            printf("%d: start_time: %0.3f duration: %0.3f\n",
                   i, (double)st->start_time / AV_TIME_BASE,
    
            printf("stream: start_time: %0.3f duration: %0.3f bitrate=%d kb/s\n",
                   (double)ic->start_time / AV_TIME_BASE,
    
                   (double)ic->duration / AV_TIME_BASE,
                   ic->bit_rate / 1000);
        }
    #endif
    }
    
    
    static int has_codec_parameters(AVCodecContext *enc)
    {
        int val;
        switch(enc->codec_type) {
    
            val = enc->sample_rate && enc->channels && enc->sample_fmt != AV_SAMPLE_FMT_NONE;
    
            if(!enc->frame_size &&
               (enc->codec_id == CODEC_ID_VORBIS ||
    
                enc->codec_id == CODEC_ID_AAC ||
    
                enc->codec_id == CODEC_ID_MP1 ||
                enc->codec_id == CODEC_ID_MP2 ||
    
                enc->codec_id == CODEC_ID_SPEEX))
    
            val = enc->width && enc->pix_fmt != PIX_FMT_NONE;
    
        return enc->codec_id != CODEC_ID_NONE && val != 0;
    
    static int has_decode_delay_been_guessed(AVStream *st)
    {
        return st->codec->codec_id != CODEC_ID_H264 ||
    
            st->codec_info_nb_frames >= 6 + st->codec->has_b_frames;
    
    static int try_decode_frame(AVStream *st, AVPacket *avpkt)
    
    {
        int16_t *samples;
        AVCodec *codec;
    
        int got_picture, data_size, ret=0;
    
    Baptiste Coudurier's avatar
    Baptiste Coudurier committed
        if(!st->codec->codec){
            codec = avcodec_find_decoder(st->codec->codec_id);
            if (!codec)
                return -1;
            ret = avcodec_open(st->codec, codec);
            if (ret < 0)
                return ret;
        }
    
        if(!has_codec_parameters(st->codec) || !has_decode_delay_been_guessed(st)){
    
    Baptiste Coudurier's avatar
    Baptiste Coudurier committed
            switch(st->codec->codec_type) {
    
                avcodec_get_frame_defaults(&picture);
    
    Baptiste Coudurier's avatar
    Baptiste Coudurier committed
                ret = avcodec_decode_video2(st->codec, &picture,
                                            &got_picture, avpkt);
                break;
    
    Baptiste Coudurier's avatar
    Baptiste Coudurier committed
                data_size = FFMAX(avpkt->size, AVCODEC_MAX_AUDIO_FRAME_SIZE);
                samples = av_malloc(data_size);
                if (!samples)
                    goto fail;
                ret = avcodec_decode_audio3(st->codec, samples,
                                            &data_size, avpkt);
                av_free(samples);
                break;
            default:
                break;
            }
    
    unsigned int ff_codec_get_tag(const AVCodecTag *tags, enum CodecID id)
    
    {
        while (tags->id != CODEC_ID_NONE) {
            if (tags->id == id)
                return tags->tag;
            tags++;
        }
        return 0;
    }
    
    
    enum CodecID ff_codec_get_id(const AVCodecTag *tags, unsigned int tag)
    
        int i;
        for(i=0; tags[i].id != CODEC_ID_NONE;i++) {
            if(tag == tags[i].tag)
                return tags[i].id;
        }
        for(i=0; tags[i].id != CODEC_ID_NONE; i++) {
    
            if (ff_toupper4(tag) == ff_toupper4(tags[i].tag))
    
    unsigned int av_codec_get_tag(const AVCodecTag * const *tags, enum CodecID id)
    
            int tag= ff_codec_get_tag(tags[i], id);
    
    enum CodecID av_codec_get_id(const AVCodecTag * const *tags, unsigned int tag)
    
            enum CodecID id= ff_codec_get_id(tags[i], tag);
    
            if(id!=CODEC_ID_NONE) return id;
        }
        return CODEC_ID_NONE;
    }
    
    
    static void compute_chapters_end(AVFormatContext *s)
    {
    
        unsigned int i, j;
        int64_t max_time = s->duration + (s->start_time == AV_NOPTS_VALUE) ? 0 : s->start_time;
    
        for (i = 0; i < s->nb_chapters; i++)
    
            if (s->chapters[i]->end == AV_NOPTS_VALUE) {
    
                AVChapter *ch = s->chapters[i];
                int64_t   end = max_time ? av_rescale_q(max_time, AV_TIME_BASE_Q, ch->time_base)
                                         : INT64_MAX;
    
                for (j = 0; j < s->nb_chapters; j++) {
                    AVChapter *ch1 = s->chapters[j];
                    int64_t next_start = av_rescale_q(ch1->start, ch1->time_base, ch->time_base);
                    if (j != i && next_start > ch->start && next_start < end)
                        end = next_start;
                }
                ch->end = (end == INT64_MAX) ? ch->start : end;
    
    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;
    
    /*
     * Is the time base unreliable.
     * This is a heuristic to balance between quick acceptance of the values in
     * the headers vs. some extra checks.
    
     * Old DivX and Xvid often have nonsense timebases like 1fps or 2fps.
     * MPEG-2 commonly misuses field repeat flags to store different framerates.
    
     * And there are "variable" fps files this needs to detect as well.
     */
    static int tb_unreliable(AVCodecContext *c){
        if(   c->time_base.den >= 101L*c->time_base.num
           || c->time_base.den <    5L*c->time_base.num
    
    /*       || c->codec_tag == AV_RL32("DIVX")
           || c->codec_tag == AV_RL32("XVID")*/
    
           || c->codec_id == CODEC_ID_MPEG2VIDEO
           || c->codec_id == CODEC_ID_H264
           )
    
        int i, count, ret, read_size, j;
    
        int64_t old_offset = avio_tell(ic->pb);
    
        for(i=0;i<ic->nb_streams;i++) {
    
            st = ic->streams[i];
    
            if (st->codec->codec_id == CODEC_ID_AAC) {
                st->codec->sample_rate = 0;
                st->codec->frame_size = 0;
                st->codec->channels = 0;
            }
    
            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;
    
            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;
                }
    
            codec = avcodec_find_decoder(st->codec->codec_id);
    
            /* Force decoding of at least one frame of codec data
             * this makes sure the codec initializes the channel configuration
             * and does not trust the values from the container.
             */
            if (codec && codec->capabilities & CODEC_CAP_CHANNEL_CONF)
                st->codec->channels = 0;
    
    
            /* Ensure that subtitle_header is properly set. */
            if (st->codec->codec_type == AVMEDIA_TYPE_SUBTITLE
                && codec && !st->codec->codec)
                avcodec_open(st->codec, codec);
    
    
            //try to just open decoders, in case this is enough to get parameters
            if(!has_codec_parameters(st->codec)){
    
                if (codec && !st->codec->codec)
    
        for (i=0; i<ic->nb_streams; i++) {
    
            ic->streams[i]->info->last_dts = AV_NOPTS_VALUE;
    
            if(url_interrupt_cb()){
    
                ret= AVERROR_EXIT;
    
                av_log(ic, AV_LOG_DEBUG, "interrupted\n");
    
            /* check if one codec still needs to be handled */
            for(i=0;i<ic->nb_streams;i++) {
                st = ic->streams[i];
    
                /* variable fps and no guess at the real fps */
    
                if(   tb_unreliable(st->codec) && !(st->r_frame_rate.num && st->avg_frame_rate.num)
    
                   && st->info->duration_count<20 && st->codec->codec_type == AVMEDIA_TYPE_VIDEO)
    
                if(st->parser && st->parser->parser->split && !st->codec->extradata)
    
                if(st->first_dts == AV_NOPTS_VALUE)
                    break;
    
            }
            if (i == ic->nb_streams) {
                /* NOTE: if the format has no header, then we need to read
                   some packets to get most of the streams, so we cannot
                   stop here */
    
                if (!(ic->ctx_flags & AVFMTCTX_NOHEADER)) {
    
                    /* if we found the info for all the codecs, we can stop */
                    ret = count;
    
                    av_log(ic, AV_LOG_DEBUG, "All info found\n");
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            /* we did not get all the codec info, but we read too much data */
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                ret = count;
    
                av_log(ic, AV_LOG_DEBUG, "Probe buffer size limit %d reached\n", ic->probesize);
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                break;
            }
    
            /* NOTE: a new stream can be added there if no header in file
               (AVFMTCTX_NOHEADER) */
            ret = av_read_frame_internal(ic, &pkt1);
    
            if (ret < 0 && ret != AVERROR(EAGAIN)) {
    
                /* EOF or error */
                ret = -1; /* we could not have all the codec parameters before EOF */
    
                for(i=0;i<ic->nb_streams;i++) {
                    st = ic->streams[i];
    
                    if (!has_codec_parameters(st->codec)){
                        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);
    
            pkt= add_to_pktbuf(&ic->packet_buffer, &pkt1, &ic->packet_buffer_end);
    
            if ((ret = av_dup_packet(pkt)) < 0)
                goto find_stream_info_err;
    
            read_size += pkt->size;
    
            if (st->codec_info_nb_frames>1) {
                if (st->time_base.den > 0 && av_rescale_q(st->info->codec_info_duration, st->time_base, AV_TIME_BASE_Q) >= ic->max_analyze_duration) {
    
                    av_log(ic, AV_LOG_WARNING, "max_analyze_duration reached\n");
    
                st->info->codec_info_duration += pkt->duration;
    
                int64_t duration= pkt->dts - last;
    
                if(pkt->dts != AV_NOPTS_VALUE && last != AV_NOPTS_VALUE && duration>0){
    
                    double dur= duration * av_q2d(st->time_base);
    
    
    //                if(st->codec->codec_type == AVMEDIA_TYPE_VIDEO)
    
    //                    av_log(NULL, AV_LOG_ERROR, "%f\n", dur);
    
                    if (st->info->duration_count < 2)
                        memset(st->info->duration_error, 0, sizeof(st->info->duration_error));
                    for (i=1; i<FF_ARRAY_ELEMS(st->info->duration_error); i++) {
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                        int framerate= get_std_framerate(i);
                        int ticks= lrintf(dur*framerate/(1001*12));
                        double error= dur - ticks*1001*12/(double)framerate;
    
                        st->info->duration_error[i] += error*error;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                    }
    
                    // ignore the first 4 values, they might have some random jitter
    
                    if (st->info->duration_count > 3)
                        st->info->duration_gcd = av_gcd(st->info->duration_gcd, duration);
    
                if (last == AV_NOPTS_VALUE || st->info->duration_count <= 1)
                    st->info->last_dts = pkt->dts;
    
            if(st->parser && st->parser->parser->split && !st->codec->extradata){
                int i= st->parser->parser->split(st->codec, pkt->data, pkt->size);
    
                    st->codec->extradata= av_malloc(st->codec->extradata_size + FF_INPUT_BUFFER_PADDING_SIZE);
    
                    memcpy(st->codec->extradata, pkt->data, st->codec->extradata_size);
    
                    memset(st->codec->extradata + i, 0, FF_INPUT_BUFFER_PADDING_SIZE);
    
            /* if still no information, we try to open the codec and to
               decompress the frame. We try to avoid that in most cases as
    
               it takes longer and uses more memory. For MPEG-4, we need to
               decompress for QuickTime. */
    
            if (!has_codec_parameters(st->codec) || !has_decode_delay_been_guessed(st))
    
                try_decode_frame(st, pkt);
    
    Diego Biurrun's avatar
    Diego Biurrun committed
        // close codecs which were opened in try_decode_frame()
    
        for(i=0;i<ic->nb_streams;i++) {
            st = ic->streams[i];
    
            if(st->codec->codec)
                avcodec_close(st->codec);
    
            if (st->codec_info_nb_frames>2 && !st->avg_frame_rate.num && st->info->codec_info_duration)
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
                av_reduce(&st->avg_frame_rate.num, &st->avg_frame_rate.den,
    
                         (st->codec_info_nb_frames-2)*(int64_t)st->time_base.den,
    
                          st->info->codec_info_duration*(int64_t)st->time_base.num, 60000);
    
            if (st->codec->codec_type == AVMEDIA_TYPE_VIDEO) {
    
                if(st->codec->codec_id == CODEC_ID_RAWVIDEO && !st->codec->codec_tag && !st->codec->bits_per_coded_sample)
    
                    st->codec->codec_tag= avcodec_pix_fmt_to_codec_tag(st->codec->pix_fmt);
    
                // 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 > 1 && !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 && !st->r_frame_rate.num
    
                   && tb_unreliable(st->codec) /*&&
    
                   //FIXME we should not special-case MPEG-2, but this needs testing with non-MPEG-2 ...
    
                   st->time_base.num*duration_sum[i]/st->info->duration_count*101LL > st->time_base.den*/){
    
                    double best_error= 2*av_q2d(st->time_base);
    
                    best_error = best_error*best_error*st->info->duration_count*1000*12*30;
    
                    for (j=1; j<FF_ARRAY_ELEMS(st->info->duration_error); j++) {
                        double error = st->info->duration_error[j] * get_std_framerate(j);
    
    //                    if(st->codec->codec_type == AVMEDIA_TYPE_VIDEO)
    
    //                        av_log(NULL, AV_LOG_ERROR, "%f %f\n", get_std_framerate(j) / 12.0/1001, error);
    
                        if(error < best_error){
                            best_error= 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);
    
                // set stream disposition based on audio service type
                switch (st->codec->audio_service_type) {
                case AV_AUDIO_SERVICE_TYPE_EFFECTS:
                    st->disposition = AV_DISPOSITION_CLEAN_EFFECTS;    break;
                case AV_AUDIO_SERVICE_TYPE_VISUALLY_IMPAIRED:
                    st->disposition = AV_DISPOSITION_VISUAL_IMPAIRED;  break;
                case AV_AUDIO_SERVICE_TYPE_HEARING_IMPAIRED:
                    st->disposition = AV_DISPOSITION_HEARING_IMPAIRED; break;
                case AV_AUDIO_SERVICE_TYPE_COMMENTARY:
                    st->disposition = AV_DISPOSITION_COMMENT;          break;
                case AV_AUDIO_SERVICE_TYPE_KARAOKE:
                    st->disposition = AV_DISPOSITION_KARAOKE;          break;
                }
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    
        av_estimate_timings(ic, old_offset);
    
        /* correct DTS for B-frame streams with no timestamps */
    
        for(i=0;i<ic->nb_streams;i++) {
            st = ic->streams[i];
    
            if (st->codec->codec_type == AVMEDIA_TYPE_VIDEO) {
    
                if(b-frames){
                    ppktl = &ic->packet_buffer;
                    while(ppkt1){
                        if(ppkt1->stream_index != i)
                            continue;
                        if(ppkt1->pkt->dts < 0)
                            break;
                        if(ppkt1->pkt->pts != AV_NOPTS_VALUE)
                            break;
                        ppkt1->pkt->dts -= delta;
                        ppkt1= ppkt1->next;
                    }
                    if(ppkt1)
                        continue;
                    st->cur_dts -= delta;
                }
            }
        }
    #endif
    
     find_stream_info_err:
        for (i=0; i < ic->nb_streams; i++)
            av_freep(&ic->streams[i]->info);
    
    static AVProgram *find_program_from_stream(AVFormatContext *ic, 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];
        return NULL;
    }
    
    int av_find_best_stream(AVFormatContext *ic,
                            enum AVMediaType type,
                            int wanted_stream_nb,
                            int related_stream,
                            AVCodec **decoder_ret,
                            int flags)
    {
        int i, nb_streams = ic->nb_streams, stream_number = 0;
        int ret = AVERROR_STREAM_NOT_FOUND, best_count = -1;
        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);
            if (p) {
                program = p->stream_index;
                nb_streams = p->nb_stream_indexes;
            }
        }
        for (i = 0; i < nb_streams; i++) {
            AVStream *st = ic->streams[program ? program[i] : i];
            AVCodecContext *avctx = st->codec;
            if (avctx->codec_type != type)
                continue;
            if (wanted_stream_nb >= 0 && stream_number++ != wanted_stream_nb)
                continue;
    
            if (st->disposition & (AV_DISPOSITION_HEARING_IMPAIRED|AV_DISPOSITION_VISUAL_IMPAIRED))
                continue;
    
            if (decoder_ret) {
                decoder = avcodec_find_decoder(ic->streams[i]->codec->codec_id);
                if (!decoder) {
                    if (ret < 0)
                        ret = AVERROR_DECODER_NOT_FOUND;
                    continue;
                }
            }
            if (best_count >= st->codec_info_nb_frames)
                continue;
            best_count = st->codec_info_nb_frames;
    
            ret = program ? program[i] : i;
    
            best_decoder = decoder;
            if (program && i == nb_streams - 1 && ret < 0) {
                program = NULL;
                nb_streams = ic->nb_streams;
                i = 0; /* no related stream found, try again with everything */
            }
        }
        if (decoder_ret)
            *decoder_ret = best_decoder;
        return ret;
    }
    
    
    /*******************************************************/
    
    int av_read_play(AVFormatContext *s)
    {
    
        if (s->iformat->read_play)
            return s->iformat->read_play(s);
    
            return ffio_read_pause(s->pb, 0);
    
    }
    
    int av_read_pause(AVFormatContext *s)
    {
    
        if (s->iformat->read_pause)
            return s->iformat->read_pause(s);
    
            return ffio_read_pause(s->pb, 1);
    
    void av_close_input_stream(AVFormatContext *s)
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    {
    
        avformat_free_context(s);
    }
    
    void avformat_free_context(AVFormatContext *s)
    {
        int i;
        AVStream *st;
    
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        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);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
            }
    
            av_metadata_free(&st->metadata);
    
            av_free(st->index_entries);
    
            av_free(st->codec->extradata);
    
            av_free(st->codec->subtitle_header);
    
            av_free(st->filename);
    
    #endif
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    
            av_freep(&s->programs[i]->provider_name);
            av_freep(&s->programs[i]->name);
    
    #endif
    
            av_metadata_free(&s->programs[i]->metadata);
    
            av_freep(&s->programs[i]->stream_index);
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        av_freep(&s->programs);
    
            av_free(s->chapters[s->nb_chapters]->title);
    
    #endif
    
            av_metadata_free(&s->chapters[s->nb_chapters]->metadata);
    
        }
        av_freep(&s->chapters);
    
        av_metadata_free(&s->metadata);
    
        av_free(s);
    
    void av_close_input_file(AVFormatContext *s)
    {
    
        AVIOContext *pb = s->iformat->flags & AVFMT_NOFILE ? NULL : s->pb;
    
        av_close_input_stream(s);
        if (pb)
    
    AVStream *av_new_stream(AVFormatContext *s, int id)
    {
        AVStream *st;
    
    
    #if FF_API_MAX_STREAMS
        if (s->nb_streams >= MAX_STREAMS){
            av_log(s, AV_LOG_ERROR, "Too many streams\n");
            return NULL;
        }
    #else
    
        AVStream **streams;
    
        if (s->nb_streams >= INT_MAX/sizeof(*streams))
            return NULL;
        streams = av_realloc(s->streams, (s->nb_streams + 1) * sizeof(*streams));
        if (!streams)
            return NULL;
        s->streams = streams;
    #endif
    
        if (!(st->info = av_mallocz(sizeof(*st->info)))) {
            av_free(st);
            return NULL;
        }
    
        if (s->iformat) {
            /* no default bitrate if decoding */
    
        st->start_time = AV_NOPTS_VALUE;
        st->duration = AV_NOPTS_VALUE;
    
            /* we set the current DTS to 0 so that formats without any timestamps
               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->first_dts = AV_NOPTS_VALUE;
    
        st->probe_packets = MAX_PROBE_PACKETS;
    
        /* default pts setting is MPEG-like */
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        st->last_IP_pts = AV_NOPTS_VALUE;
    
        for(i=0; i<MAX_REORDER_DELAY+1; i++)
            st->pts_buffer[i]= AV_NOPTS_VALUE;
    
        st->reference_dts = AV_NOPTS_VALUE;
    
        st->sample_aspect_ratio = (AVRational){0,1};
    
    
    AVProgram *av_new_program(AVFormatContext *ac, int id)
    {
        AVProgram *program=NULL;
        int i;
    
    #ifdef DEBUG_SI
        av_log(ac, AV_LOG_DEBUG, "new_program: id=0x%04x\n", id);
    #endif
    
        for(i=0; i<ac->nb_programs; i++)
            if(ac->programs[i]->id == id)
                program = ac->programs[i];
    
        if(!program){
            program = av_mallocz(sizeof(AVProgram));
            if (!program)
                return NULL;
            dynarray_add(&ac->programs, &ac->nb_programs, program);
            program->discard = AVDISCARD_NONE;
        }
        program->id = id;
    
        return program;
    }
    
    
    AVChapter *ff_new_chapter(AVFormatContext *s, int id, AVRational time_base, int64_t start, int64_t end, const char *title)
    
            if(s->chapters[i]->id == id)
                chapter = s->chapters[i];
    
        if(!chapter){
            chapter= av_mallocz(sizeof(AVChapter));
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
            if(!chapter)
    
            dynarray_add(&s->chapters, &s->nb_chapters, chapter);
    
    Aurelien Jacobs's avatar
    Aurelien Jacobs committed
        av_free(chapter->title);
    
    #endif
    
        av_metadata_set2(&chapter->metadata, "title", title, 0);
    
        chapter->time_base= time_base;
    
        chapter->start = start;
    
    Michael Niedermayer's avatar
    Michael Niedermayer committed
        chapter->end   = end;
    
    /************************************************************/
    /* output media file */
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
    
    
    int av_set_parameters(AVFormatContext *s, AVFormatParameters *ap)
    {
        int ret;
    
        if (s->oformat->priv_data_size > 0) {
            s->priv_data = av_mallocz(s->oformat->priv_data_size);
            if (!s->priv_data)
    
            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 (s->oformat->set_parameters) {
            ret = s->oformat->set_parameters(s, ap);
            if (ret < 0)
                return ret;
        }
        return 0;
    }
    
    
    static int validate_codec_tag(AVFormatContext *s, AVStream *st)
    {
        const AVCodecTag *avctag;
        int n;
        enum CodecID id = CODEC_ID_NONE;
        unsigned int tag = 0;
    
        /**
         * Check that tag + id is in the table
         * If neither is in the table -> OK
         * If tag is in the table with another id -> FAIL
         * If id is in the table with another tag -> FAIL unless strict < normal
         */
        for (n = 0; s->oformat->codec_tag[n]; n++) {
            avctag = s->oformat->codec_tag[n];
            while (avctag->id != CODEC_ID_NONE) {
                if (ff_toupper4(avctag->tag) == ff_toupper4(st->codec->codec_tag)) {
                    id = avctag->id;
                    if (id == st->codec->codec_id)
                        return 1;
                }
                if (avctag->id == st->codec->codec_id)
                    tag = avctag->tag;
                avctag++;
            }
        }
        if (id != CODEC_ID_NONE)
            return 0;
        if (tag && (st->codec->strict_std_compliance >= FF_COMPLIANCE_NORMAL))
            return 0;
        return 1;
    }
    
    
        // some sanity checks
    
        if (s->nb_streams == 0 && !(s->oformat->flags & AVFMT_NOSTREAMS)) {
    
            av_log(s, AV_LOG_ERROR, "no streams\n");
    
        for(i=0;i<s->nb_streams;i++) {
            st = s->streams[i];
    
            switch (st->codec->codec_type) {
    
                if(st->codec->sample_rate<=0){
                    av_log(s, AV_LOG_ERROR, "sample rate not set\n");
    
                if(!st->codec->block_align)
                    st->codec->block_align = st->codec->channels *
                        av_get_bits_per_sample(st->codec->codec_id) >> 3;
    
                if(st->codec->time_base.num<=0 || st->codec->time_base.den<=0){ //FIXME audio too?
                    av_log(s, AV_LOG_ERROR, "time base not set\n");
    
                if((st->codec->width<=0 || st->codec->height<=0) && !(s->oformat->flags & AVFMT_NODIMENSIONS)){
    
                    av_log(s, AV_LOG_ERROR, "dimensions not set\n");
    
                if(av_cmp_q(st->sample_aspect_ratio, st->codec->sample_aspect_ratio)){
                    av_log(s, AV_LOG_ERROR, "Aspect ratio mismatch between encoder and muxer layer\n");
    
                if(st->codec->codec_tag && st->codec->codec_id == CODEC_ID_RAWVIDEO && av_codec_get_tag(s->oformat->codec_tag, st->codec->codec_id) == 0 && !validate_codec_tag(s, st)){
                    //the current rawvideo encoding system ends up setting the wrong codec_tag for avi, we override it here
                    st->codec->codec_tag= 0;
                }
    
                    if (!validate_codec_tag(s, st)) {
    
                        char tagbuf[32];
                        av_get_codec_tag_string(tagbuf, sizeof(tagbuf), st->codec->codec_tag);
    
                        av_log(s, AV_LOG_ERROR,
    
                               "Tag %s/0x%08x incompatible with output codec id '%d'\n",
                               tagbuf, st->codec->codec_tag, st->codec->codec_id);
    
                        return AVERROR_INVALIDDATA;
                    }
    
                }else
                    st->codec->codec_tag= av_codec_get_tag(s->oformat->codec_tag, st->codec->codec_id);
            }
    
    
            if(s->oformat->flags & AVFMT_GLOBALHEADER &&
                !(st->codec->flags & CODEC_FLAG_GLOBAL_HEADER))
              av_log(s, AV_LOG_WARNING, "Codec for stream %d does not use global headers but container format requires global headers\n", i);
    
        if (!s->priv_data && s->oformat->priv_data_size > 0) {
    
            s->priv_data = av_mallocz(s->oformat->priv_data_size);
            if (!s->priv_data)
    
        ff_metadata_mux_compat(s);
    
        /* set muxer identification string */
    
        if (s->nb_streams && !(s->streams[0]->codec->flags & CODEC_FLAG_BITEXACT)) {
    
            av_metadata_set2(&s->metadata, "encoder", LIBAVFORMAT_IDENT, 0);
    
        if(s->oformat->write_header){
            ret = s->oformat->write_header(s);
            if (ret < 0)
                return ret;
        }
    
    
        /* init PTS generation */
        for(i=0;i<s->nb_streams;i++) {
    
                den = (int64_t)st->time_base.num * st->codec->sample_rate;
    
                den = (int64_t)st->time_base.num * st->codec->time_base.den;
    
            if (den != AV_NOPTS_VALUE) {
                if (den <= 0)
                    return AVERROR_INVALIDDATA;
                av_frac_init(&st->pts, 0, 0, den);
            }
    
    //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 num, den, frame_size, i;
    
        av_dlog(s, "av_write_frame: 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 -1;*/
    
        /* duration field */
    
        if (pkt->duration == 0) {
            compute_frame_duration(&num, &den, st, NULL, pkt);
            if (den && num) {
    
                pkt->duration = av_rescale(1, num * (int64_t)st->time_base.den * st->codec->ticks_per_frame, den * (int64_t)st->time_base.num);
    
        if(pkt->pts == AV_NOPTS_VALUE && pkt->dts != AV_NOPTS_VALUE && delay==0)
            pkt->pts= pkt->dts;
    
    
        //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){
    
            pkt->dts=
    //        pkt->pts= st->cur_dts;
            pkt->pts= st->pts.val;
        }
    
    
        //calculate dts from pts
    
        if(pkt->pts != AV_NOPTS_VALUE && pkt->dts == AV_NOPTS_VALUE && delay <= MAX_REORDER_DELAY){
    
            st->pts_buffer[0]= pkt->pts;
            for(i=1; i<delay+1 && st->pts_buffer[i] == AV_NOPTS_VALUE; i++)
    
                st->pts_buffer[i]= pkt->pts + (i-delay-1) * pkt->duration;
    
            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(st->cur_dts && st->cur_dts != AV_NOPTS_VALUE && st->cur_dts >= pkt->dts){
    
                   "Application provided invalid, non monotonically increasing dts to muxer in stream %d: %"PRId64" >= %"PRId64"\n",
    
            return -1;
        }
        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_DEBUG, "av_write_frame: pts2:%"PRId64" dts2:%"PRId64"\n", pkt->pts, pkt->dts);
    
        st->cur_dts= pkt->dts;
        st->pts.val= pkt->dts;
    
    
            frame_size = get_audio_frame_size(st->codec, pkt->size);
    
            /* HACK/FIXME, we skip the initial 0 size packets as they are most
               likely equal to the encoder delay, but it would be better if we
               had the real timestamps from the encoder */
    
            if (frame_size >= 0 && (pkt->size || st->pts.num!=st->pts.den>>1 || st->pts.val)) {
    
                av_frac_add(&st->pts, (int64_t)st->time_base.den * frame_size);
    
            av_frac_add(&st->pts, (int64_t)st->time_base.den * st->codec->time_base.num);