Skip to content
Snippets Groups Projects
utils.c 122 KiB
Newer Older
  • Learn to ignore specific revisions
  •         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++) {
    
                /* if the timebase is coarse (like the usual millisecond precision
                   of mkv), we need to analyze more frames to reliably arrive at
                   the correct fps */
                if (av_q2d(st->time_base) > 0.0005)
                    fps_analyze_framecount *= 2;
    
                if (ic->fps_probe_size >= 0)
                    fps_analyze_framecount = ic->fps_probe_size;
    
                /* 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 < fps_analyze_framecount
                   && 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;
    
        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++) {
    
            int real_stream_index = program ? program[i] : i;
            AVStream *st = ic->streams[real_stream_index];
    
            AVCodecContext *avctx = st->codec;
            if (avctx->codec_type != type)
                continue;
    
            if (wanted_stream_nb >= 0 && real_stream_index != wanted_stream_nb)
    
                continue;
    
            if (st->disposition & (AV_DISPOSITION_HEARING_IMPAIRED|AV_DISPOSITION_VISUAL_IMPAIRED))
                continue;
    
            if (decoder_ret) {
    
                decoder = avcodec_find_decoder(st->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;
    
            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 avio_pause(s->pb, 0);
    
    }
    
    int av_read_pause(AVFormatContext *s)
    {
    
        if (s->iformat->read_pause)
            return s->iformat->read_pause(s);
    
            return avio_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;
    
    
        av_opt_free(s);
        if (s->iformat && s->iformat->priv_class)
            av_opt_free(s->priv_data);
    
    
    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);
    
    Fabrice Bellard's avatar
    Fabrice Bellard committed
        }
    
            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_metadata_free(&s->chapters[s->nb_chapters]->metadata);
    
        }
        av_freep(&s->chapters);
    
        av_metadata_free(&s->metadata);
    
        av_freep(&s->streams);
    
        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;
    
        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;
    
        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);
    
        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)
    
        /* 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, "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 -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);
    
    }
    
    int av_write_frame(AVFormatContext *s, AVPacket *pkt)
    {
    
        int ret = compute_pkt_fields2(s, s->streams[pkt->stream_index], pkt);
    
        if(ret<0 && !(s->oformat->flags & AVFMT_NOTIMESTAMPS))
    
        ret= s->oformat->write_packet(s, pkt);
        return ret;
    
    void ff_interleave_add_packet(AVFormatContext *s, AVPacket *pkt,
                                  int (*compare)(AVFormatContext *, AVPacket *, AVPacket *))
    {
        AVPacketList **next_point, *this_pktl;
    
        this_pktl = av_mallocz(sizeof(AVPacketList));
        this_pktl->pkt= *pkt;
    
        pkt->destruct= NULL;             // do not free original but only the copy
        av_dup_packet(&this_pktl->pkt);  // duplicate the packet if it uses non-alloced memory
    
        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 = &s->packet_buffer;
    
    
        if(*next_point){
            if(compare(s, &s->packet_buffer_end->pkt, pkt)){
                while(!compare(s, &(*next_point)->pkt, pkt)){
    
                    next_point= &(*next_point)->next;
                }
    
            }else{
                next_point = &(s->packet_buffer_end->next);
    
        assert(!*next_point);
    
        s->packet_buffer_end= this_pktl;
    
        s->streams[pkt->stream_index]->last_in_packet_buffer=
        *next_point= this_pktl;
    
    static int ff_interleave_compare_dts(AVFormatContext *s, AVPacket *next, AVPacket *pkt)
    
    {
        AVStream *st = s->streams[ pkt ->stream_index];
        AVStream *st2= s->streams[ next->stream_index];
    
        int comp = av_compare_ts(next->dts, st2->time_base, pkt->dts,
                                 st->time_base);
    
        if (comp == 0)
    
            return pkt->stream_index < next->stream_index;
    
    int av_interleave_packet_per_dts(AVFormatContext *s, AVPacket *out, AVPacket *pkt, int flush){