美文网首页Android直播
FFMPEG解协议、解封装流程

FFMPEG解协议、解封装流程

作者: 野码道人 | 来源:发表于2019-12-31 11:00 被阅读0次

    1、av_register_all();

    初始化 libavformat, 注册所有 muxers、demuxers and protocols

    2、avformat_network_init();

    对网络库进行全局初始化

    3、avformat_open_input(&fmtCtx, fileName, NULL, &dict);

    打开一个输入流并且读取头部信息,关闭流的时候需要调用avformat_close_input()

    参数说明:

    • AVFormatContext **ps, 格式化的上下文。如果传入的是一个AVFormatContext类型的指针,则该空间须自己手动清理,若传入的指针为空,则FFmpeg会内部自己创建。
    • const char *url, 传入的地址。支持http,RTSP,以及普通的本地文件。地址最终会存入到AVFormatContext结构体当中。
    • AVInputFormat *fmt, 指定输入的封装格式。一般传NULL,由FFmpeg自行探测。
    • AVDictionary **options, 其它参数设置。它是一个字典,用于参数传递,可以设置私有信息,以及协议类型,不传则写NULL

    秒开说明:

    • fps_probe_size = 1
    • max_analyze_duration = 300 *1000
    • probesize = 400 * 1024

    当只有视频的时候:probesize = 1000;
    只有音频的时候:probesize = 4000;

    注意:点播场景连接一旦建立不会断开,但是直播场景几秒内没有数据CDN会断开与客户端的连接,所以直播场景端上需要无限重试机制

    4、avformat_find_stream_info(fmtCtx, NULL);

    获取视频流信息,有些格式当中没有头部信息,如flv格式,调用avformat_open_input()在打开文件之后就没有参数,也就无法获取到里面的信息。调用此函数,会试着去探测文件的格式

    秒开说明:
    当只有视频或只有音频的时候avformat_find_stream_info可以做如下优化,下面贴上源码和修改后的代码

    源码:

    int avformat_find_stream_info(AVFormatContext *ic, AVDictionary **options)
    {
        int i, count = 0, ret = 0, j;
        int64_t read_size;
        AVStream *st;
        AVCodecContext *avctx;
        AVPacket pkt1, *pkt;
        int64_t old_offset  = avio_tell(ic->pb);
        // new streams might appear, no options for those
        int orig_nb_streams = ic->nb_streams;
        int flush_codecs;
        int64_t max_analyze_duration = ic->max_analyze_duration;
        int64_t max_stream_analyze_duration;
        int64_t max_subtitle_analyze_duration;
        int64_t probesize = ic->probesize;
        int eof_reached = 0;
        int *missing_streams = av_opt_ptr(ic->iformat->priv_class, ic->priv_data, "missing_streams");
    
        flush_codecs = probesize > 0;
    
        av_opt_set(ic, "skip_clear", "1", AV_OPT_SEARCH_CHILDREN);
    
        max_stream_analyze_duration = max_analyze_duration;
        max_subtitle_analyze_duration = max_analyze_duration;
        if (!max_analyze_duration) {
            max_stream_analyze_duration =
            max_analyze_duration        = 5*AV_TIME_BASE;
            max_subtitle_analyze_duration = 30*AV_TIME_BASE;
            if (!strcmp(ic->iformat->name, "flv"))
                max_stream_analyze_duration = 90*AV_TIME_BASE;
            if (!strcmp(ic->iformat->name, "mpeg") || !strcmp(ic->iformat->name, "mpegts"))
                max_stream_analyze_duration = 7*AV_TIME_BASE;
        }
    
        if (ic->pb)
            av_log(ic, AV_LOG_DEBUG, "Before avformat_find_stream_info() pos: %"PRId64" bytes read:%"PRId64" seeks:%d nb_streams:%d\n",
                   avio_tell(ic->pb), ic->pb->bytes_read, ic->pb->seek_count, ic->nb_streams);
    
        for (i = 0; i < ic->nb_streams; i++) {
            const AVCodec *codec;
            AVDictionary *thread_opt = NULL;
            st = ic->streams[i];
            avctx = st->internal->avctx;
    
            if (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO ||
                st->codecpar->codec_type == AVMEDIA_TYPE_SUBTITLE) {
    /*            if (!st->time_base.num)
                    st->time_base = */
                if (!avctx->time_base.num)
                    avctx->time_base = st->time_base;
            }
    
            /* check if the caller has overridden the codec id */
    #if FF_API_LAVF_AVCTX
    FF_DISABLE_DEPRECATION_WARNINGS
            if (st->codec->codec_id != st->internal->orig_codec_id) {
                st->codecpar->codec_id   = st->codec->codec_id;
                st->codecpar->codec_type = st->codec->codec_type;
                st->internal->orig_codec_id = st->codec->codec_id;
            }
    FF_ENABLE_DEPRECATION_WARNINGS
    #endif
            // only for the split stuff
            if (!st->parser && !(ic->flags & AVFMT_FLAG_NOPARSE) && st->request_probe <= 0) {
                st->parser = av_parser_init(st->codecpar->codec_id);
                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->codecpar->codec_id));
                }
            }
    
            if (st->codecpar->codec_id != st->internal->orig_codec_id)
                st->internal->orig_codec_id = st->codecpar->codec_id;
    
            ret = avcodec_parameters_to_context(avctx, st->codecpar);
            if (ret < 0)
                goto find_stream_info_err;
            if (st->request_probe <= 0)
                st->internal->avctx_inited = 1;
    
            codec = find_probe_decoder(ic, st, st->codecpar->codec_id);
    
            /* Force thread count to 1 since the H.264 decoder will not extract
             * SPS and PPS to extradata during multi-threaded decoding. */
            av_dict_set(options ? &options[i] : &thread_opt, "threads", "1", 0);
    
            if (ic->codec_whitelist)
                av_dict_set(options ? &options[i] : &thread_opt, "codec_whitelist", ic->codec_whitelist, 0);
    
            /* Ensure that subtitle_header is properly set. */
            if (st->codecpar->codec_type == AVMEDIA_TYPE_SUBTITLE
                && codec && !avctx->codec) {
                if (avcodec_open2(avctx, codec, options ? &options[i] : &thread_opt) < 0)
                    av_log(ic, AV_LOG_WARNING,
                           "Failed to open codec in %s\n",__FUNCTION__);
            }
    
            // Try to just open decoders, in case this is enough to get parameters.
            if (!has_codec_parameters(st, NULL) && st->request_probe <= 0) {
                if (codec && !avctx->codec)
                    if (avcodec_open2(avctx, codec, options ? &options[i] : &thread_opt) < 0)
                        av_log(ic, AV_LOG_WARNING,
                               "Failed to open codec in %s\n",__FUNCTION__);
            }
            if (!options)
                av_dict_free(&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;
        }
    
        read_size = 0;
        for (;;) {
            int analyzed_all_streams;
            if (ff_check_interrupt(&ic->interrupt_callback)) {
                ret = AVERROR_EXIT;
                av_log(ic, AV_LOG_DEBUG, "interrupted\n");
                break;
            }
    
            /* check if one codec still needs to be handled */
            for (i = 0; i < ic->nb_streams; i++) {
                int fps_analyze_framecount = 20;
                int count;
    
                st = ic->streams[i];
                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
                 * the correct fps. */
                if (av_q2d(st->time_base) > 0.0005)
                    fps_analyze_framecount *= 2;
                if (!tb_unreliable(st->internal->avctx))
                    fps_analyze_framecount = 0;
                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 */
                count = (ic->iformat->flags & AVFMT_NOTIMESTAMPS) ?
                           st->info->codec_info_duration_fields/2 :
                           st->info->duration_count;
                if (!(st->r_frame_rate.num && st->avg_frame_rate.num) &&
                    st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) {
                    if (count < fps_analyze_framecount)
                        break;
                }
                // Look at the first 3 frames if there is evidence of frame delay
                // but the decoder delay is not set.
                if (st->info->frame_delay_evidence && count < 2 && st->internal->avctx->has_b_frames == 0)
                    break;
                if (!st->internal->avctx->extradata &&
                    (!st->internal->extract_extradata.inited ||
                     st->internal->extract_extradata.bsf) &&
                    extract_extradata_check(st))
                    break;
                if (st->first_dts == AV_NOPTS_VALUE &&
                    !(ic->iformat->flags & AVFMT_NOTIMESTAMPS) &&
                    st->codec_info_nb_frames < ((st->disposition & AV_DISPOSITION_ATTACHED_PIC) ? 1 : ic->max_ts_probe) &&
                    (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO ||
                     st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO))
                    break;
            }
            analyzed_all_streams = 0;
            if (!missing_streams || !*missing_streams)
            if (i == ic->nb_streams) {
                analyzed_all_streams = 1;
                /* 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");
                    flush_codecs = 0;
                    break;
                }
            }
            /* We did not get all the codec info, but we read too much data. */
            if (read_size >= probesize) {
                ret = count;
                av_log(ic, AV_LOG_DEBUG,
                       "Probe buffer size limit of %"PRId64" bytes reached\n", probesize);
                for (i = 0; i < ic->nb_streams; i++)
                    if (!ic->streams[i]->r_frame_rate.num &&
                        ic->streams[i]->info->duration_count <= 1 &&
                        ic->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO &&
                        strcmp(ic->iformat->name, "image2"))
                        av_log(ic, AV_LOG_WARNING,
                               "Stream #%d: not enough frames to estimate rate; "
                               "consider increasing probesize\n", i);
                break;
            }
    
            /* NOTE: A new stream can be added there if no header in file
             * (AVFMTCTX_NOHEADER). */
            ret = read_frame_internal(ic, &pkt1);
            if (ret == AVERROR(EAGAIN))
                continue;
    
            if (ret < 0) {
                /* EOF or error*/
                eof_reached = 1;
                break;
            }
    
            pkt = &pkt1;
    
            if (!(ic->flags & AVFMT_FLAG_NOBUFFER)) {
                ret = ff_packet_list_put(&ic->internal->packet_buffer,
                                         &ic->internal->packet_buffer_end,
                                         pkt, 0);
                if (ret < 0)
                    goto find_stream_info_err;
            }
    
            st = ic->streams[pkt->stream_index];
            if (!(st->disposition & AV_DISPOSITION_ATTACHED_PIC))
                read_size += pkt->size;
    
            avctx = st->internal->avctx;
            if (!st->internal->avctx_inited) {
                ret = avcodec_parameters_to_context(avctx, st->codecpar);
                if (ret < 0)
                    goto find_stream_info_err;
                st->internal->avctx_inited = 1;
            }
    
            if (pkt->dts != AV_NOPTS_VALUE && st->codec_info_nb_frames > 1) {
                /* 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_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_first_dts =
                    st->info->fps_last_dts  = AV_NOPTS_VALUE;
                }
                /* Check for a discontinuity in dts. If the difference in dts
                 * is more than 1000 times the average packet duration in the
                 * sequence, we treat it as a discontinuity. */
                if (st->info->fps_last_dts != AV_NOPTS_VALUE &&
                    st->info->fps_last_dts_idx > st->info->fps_first_dts_idx &&
                    (pkt->dts - (uint64_t)st->info->fps_last_dts) / 1000 >
                    (st->info->fps_last_dts     - (uint64_t)st->info->fps_first_dts) /
                    (st->info->fps_last_dts_idx - st->info->fps_first_dts_idx)) {
                    av_log(ic, AV_LOG_WARNING,
                           "DTS discontinuity 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_first_dts =
                    st->info->fps_last_dts  = AV_NOPTS_VALUE;
                }
    
                /* update stored dts values */
                if (st->info->fps_first_dts == AV_NOPTS_VALUE) {
                    st->info->fps_first_dts     = pkt->dts;
                    st->info->fps_first_dts_idx = st->codec_info_nb_frames;
                }
                st->info->fps_last_dts     = pkt->dts;
                st->info->fps_last_dts_idx = st->codec_info_nb_frames;
            }
            if (st->codec_info_nb_frames>1) {
                int64_t t = 0;
                int64_t limit;
    
                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 == 0
                    && st->codec_info_nb_frames>30
                    && st->info->fps_first_dts != AV_NOPTS_VALUE
                    && st->info->fps_last_dts  != AV_NOPTS_VALUE)
                    t = FFMAX(t, av_rescale_q(st->info->fps_last_dts - st->info->fps_first_dts, st->time_base, AV_TIME_BASE_Q));
    
                if (analyzed_all_streams)                                limit = max_analyze_duration;
                else if (avctx->codec_type == AVMEDIA_TYPE_SUBTITLE) limit = max_subtitle_analyze_duration;
                else                                                     limit = max_stream_analyze_duration;
    
                if (t >= limit) {
                    av_log(ic, AV_LOG_VERBOSE, "max_analyze_duration %"PRId64" reached at %"PRId64" microseconds st:%d\n",
                           limit,
                           t, pkt->stream_index);
                    if (ic->flags & AVFMT_FLAG_NOBUFFER)
                        av_packet_unref(pkt);
                    break;
                }
                if (pkt->duration) {
                    if (avctx->codec_type == AVMEDIA_TYPE_SUBTITLE && pkt->pts != AV_NOPTS_VALUE && st->start_time != AV_NOPTS_VALUE && pkt->pts >= st->start_time) {
                        st->info->codec_info_duration = FFMIN(pkt->pts - st->start_time, st->info->codec_info_duration + pkt->duration);
                    } else
                        st->info->codec_info_duration += pkt->duration;
                    st->info->codec_info_duration_fields += st->parser && st->need_parsing && avctx->ticks_per_frame ==2 ? st->parser->repeat_pict + 1 : 2;
                }
            }
            if (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) {
    #if FF_API_R_FRAME_RATE
                ff_rfps_add_frame(ic, st, pkt->dts);
    #endif
                if (pkt->dts != pkt->pts && pkt->dts != AV_NOPTS_VALUE && pkt->pts != AV_NOPTS_VALUE)
                    st->info->frame_delay_evidence = 1;
            }
            if (!st->internal->avctx->extradata) {
                ret = extract_extradata(st, pkt);
                if (ret < 0)
                    goto find_stream_info_err;
            }
    
            /* 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 AV_CODEC_CAP_CHANNEL_CONF is set this will force decoding of at
             * least one frame of codec data, this makes sure the codec initializes
             * the channel configuration and does not only trust the values from
             * the container. */
            try_decode_frame(ic, st, pkt,
                             (options && i < orig_nb_streams) ? &options[i] : NULL);
    
            if (ic->flags & AVFMT_FLAG_NOBUFFER)
                av_packet_unref(pkt);
    
            st->codec_info_nb_frames++;
            count++;
        }
    
        if (eof_reached) {
            int stream_index;
            for (stream_index = 0; stream_index < ic->nb_streams; stream_index++) {
                st = ic->streams[stream_index];
                avctx = st->internal->avctx;
                if (!has_codec_parameters(st, NULL)) {
                    const AVCodec *codec = find_probe_decoder(ic, st, st->codecpar->codec_id);
                    if (codec && !avctx->codec) {
                        AVDictionary *opts = NULL;
                        if (ic->codec_whitelist)
                            av_dict_set(&opts, "codec_whitelist", ic->codec_whitelist, 0);
                        if (avcodec_open2(avctx, codec, (options && stream_index < orig_nb_streams) ? &options[stream_index] : &opts) < 0)
                            av_log(ic, AV_LOG_WARNING,
                                   "Failed to open codec in %s\n",__FUNCTION__);
                        av_dict_free(&opts);
                    }
                }
    
                // EOF already reached while reading the stream above.
                // So continue with reoordering DTS with whatever delay we have.
                if (ic->internal->packet_buffer && !has_decode_delay_been_guessed(st)) {
                    update_dts_from_pts(ic, stream_index, ic->internal->packet_buffer);
                }
            }
        }
    
        if (flush_codecs) {
            AVPacket empty_pkt = { 0 };
            int err = 0;
            av_init_packet(&empty_pkt);
    
            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(ic, 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);
                    }
                }
            }
        }
    
        ff_rfps_calculate(ic);
    
        for (i = 0; i < ic->nb_streams; i++) {
            st = ic->streams[i];
            avctx = st->internal->avctx;
            if (avctx->codec_type == AVMEDIA_TYPE_VIDEO) {
                if (avctx->codec_id == AV_CODEC_ID_RAWVIDEO && !avctx->codec_tag && !avctx->bits_per_coded_sample) {
                    uint32_t tag= avcodec_pix_fmt_to_codec_tag(avctx->pix_fmt);
                    if (avpriv_find_pix_fmt(avpriv_get_raw_pix_fmt_tags(), tag) == avctx->pix_fmt)
                        avctx->codec_tag= tag;
                }
    
                /* estimate average framerate if not set by demuxer */
                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;
                    AVRational codec_frame_rate = avctx->framerate;
    
                    if (st->info->codec_info_duration        >= INT64_MAX / st->time_base.num / 2||
                        st->info->codec_info_duration_fields >= INT64_MAX / st->time_base.den ||
                        st->info->codec_info_duration        < 0)
                        continue;
                    av_reduce(&st->avg_frame_rate.num, &st->avg_frame_rate.den,
                              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. */
                    for (j = 0; j < MAX_STD_TIMEBASES; j++) {
                        AVRational std_fps = { get_std_framerate(j), 12 * 1001 };
                        double error       = fabs(av_q2d(st->avg_frame_rate) /
                                                  av_q2d(std_fps) - 1);
    
                        if (error < best_error) {
                            best_error = error;
                            best_fps   = std_fps.num;
                        }
    
                        if (ic->internal->prefer_codec_framerate && codec_frame_rate.num > 0 && codec_frame_rate.den > 0) {
                            error       = fabs(av_q2d(codec_frame_rate) /
                                               av_q2d(std_fps) - 1);
                            if (error < best_error) {
                                best_error = error;
                                best_fps   = std_fps.num;
                            }
                        }
                    }
                    if (best_fps)
                        av_reduce(&st->avg_frame_rate.num, &st->avg_frame_rate.den,
                                  best_fps, 12 * 1001, INT_MAX);
                }
    
                if (!st->r_frame_rate.num) {
                    if (    avctx->time_base.den * (int64_t) st->time_base.num
                        <= avctx->time_base.num * avctx->ticks_per_frame * (int64_t) st->time_base.den) {
                        av_reduce(&st->r_frame_rate.num, &st->r_frame_rate.den,
                                  avctx->time_base.den, (int64_t)avctx->time_base.num * avctx->ticks_per_frame, INT_MAX);
                    } else {
                        st->r_frame_rate.num = st->time_base.den;
                        st->r_frame_rate.den = st->time_base.num;
                    }
                }
                if (st->display_aspect_ratio.num && st->display_aspect_ratio.den) {
                    AVRational hw_ratio = { avctx->height, avctx->width };
                    st->sample_aspect_ratio = av_mul_q(st->display_aspect_ratio,
                                                       hw_ratio);
                }
            } else if (avctx->codec_type == AVMEDIA_TYPE_AUDIO) {
                if (!avctx->bits_per_coded_sample)
                    avctx->bits_per_coded_sample =
                        av_get_bits_per_sample(avctx->codec_id);
                // set stream disposition based on audio service type
                switch (avctx->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;
                }
            }
        }
    
        if (probesize)
            estimate_timings(ic, old_offset);
    
        av_opt_set(ic, "skip_clear", "0", AV_OPT_SEARCH_CHILDREN);
    
        if (ret >= 0 && ic->nb_streams)
            /* We could not have all the codec parameters before EOF. */
            ret = -1;
        for (i = 0; i < ic->nb_streams; i++) {
            const char *errmsg;
            st = ic->streams[i];
    
            /* if no packet was ever seen, update context now for has_codec_parameters */
            if (!st->internal->avctx_inited) {
                if (st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO &&
                    st->codecpar->format == AV_SAMPLE_FMT_NONE)
                    st->codecpar->format = st->internal->avctx->sample_fmt;
                ret = avcodec_parameters_to_context(st->internal->avctx, st->codecpar);
                if (ret < 0)
                    goto find_stream_info_err;
            }
            if (!has_codec_parameters(st, &errmsg)) {
                char buf[256];
                avcodec_string(buf, sizeof(buf), st->internal->avctx, 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;
            }
        }
    
        compute_chapters_end(ic);
    
        /* update the stream parameters from the internal codec contexts */
        for (i = 0; i < ic->nb_streams; i++) {
            st = ic->streams[i];
    
            if (st->internal->avctx_inited) {
                int orig_w = st->codecpar->width;
                int orig_h = st->codecpar->height;
                ret = avcodec_parameters_from_context(st->codecpar, st->internal->avctx);
                if (ret < 0)
                    goto find_stream_info_err;
    #if FF_API_LOWRES
                // The decoder might reduce the video size by the lowres factor.
                if (st->internal->avctx->lowres && orig_w) {
                    st->codecpar->width = orig_w;
                    st->codecpar->height = orig_h;
                }
    #endif
            }
    
    #if FF_API_LAVF_AVCTX
    FF_DISABLE_DEPRECATION_WARNINGS
            ret = avcodec_parameters_to_context(st->codec, st->codecpar);
            if (ret < 0)
                goto find_stream_info_err;
    
    #if FF_API_LOWRES
            // The old API (AVStream.codec) "requires" the resolution to be adjusted
            // by the lowres factor.
            if (st->internal->avctx->lowres && st->internal->avctx->width) {
                st->codec->lowres = st->internal->avctx->lowres;
                st->codec->width = st->internal->avctx->width;
                st->codec->height = st->internal->avctx->height;
            }
    #endif
    
            if (st->codec->codec_tag != MKTAG('t','m','c','d')) {
                st->codec->time_base = st->internal->avctx->time_base;
                st->codec->ticks_per_frame = st->internal->avctx->ticks_per_frame;
            }
            st->codec->framerate = st->avg_frame_rate;
    
            if (st->internal->avctx->subtitle_header) {
                st->codec->subtitle_header = av_malloc(st->internal->avctx->subtitle_header_size);
                if (!st->codec->subtitle_header)
                    goto find_stream_info_err;
                st->codec->subtitle_header_size = st->internal->avctx->subtitle_header_size;
                memcpy(st->codec->subtitle_header, st->internal->avctx->subtitle_header,
                       st->codec->subtitle_header_size);
            }
    
            // Fields unavailable in AVCodecParameters
            st->codec->coded_width = st->internal->avctx->coded_width;
            st->codec->coded_height = st->internal->avctx->coded_height;
            st->codec->properties = st->internal->avctx->properties;
    FF_ENABLE_DEPRECATION_WARNINGS
    #endif
    
            st->internal->avctx_inited = 0;
        }
    
    find_stream_info_err:
        for (i = 0; i < ic->nb_streams; i++) {
            st = ic->streams[i];
            if (st->info)
                av_freep(&st->info->duration_error);
            avcodec_close(ic->streams[i]->internal->avctx);
            av_freep(&ic->streams[i]->info);
            av_bsf_free(&ic->streams[i]->internal->extract_extradata.bsf);
            av_packet_free(&ic->streams[i]->internal->extract_extradata.pkt);
        }
        if (ic->pb)
            av_log(ic, AV_LOG_DEBUG, "After avformat_find_stream_info() pos: %"PRId64" bytes read:%"PRId64" seeks:%d frames:%d\n",
                   avio_tell(ic->pb), ic->pb->bytes_read, ic->pb->seek_count, count);
        return ret;
    }
    

    修改后的代码:

    int fast_avformat_find_stream_info(AVFormatContext *ic, AVDictionary **options)
    {
        int i, count = 0, ret = 0, j;
        int64_t read_size;
        AVStream *st;
        AVCodecContext *avctx;
        AVPacket pkt1, *pkt;
        int64_t old_offset  = avio_tell(ic->pb);
        // new streams might appear, no options for those
        int orig_nb_streams = ic->nb_streams;
        int flush_codecs;
        int64_t max_analyze_duration = ic->max_analyze_duration;
        int64_t max_stream_analyze_duration;
        int64_t max_subtitle_analyze_duration;
        int64_t probesize = ic->probesize;
        int eof_reached = 0;
        int *missing_streams = av_opt_ptr(ic->iformat->priv_class, ic->priv_data, "missing_streams");
        
        flush_codecs = probesize > 0;
        
        av_opt_set(ic, "skip_clear", "1", AV_OPT_SEARCH_CHILDREN);
        
        max_stream_analyze_duration = max_analyze_duration;
        max_subtitle_analyze_duration = max_analyze_duration;
        if (!max_analyze_duration) {
            max_stream_analyze_duration =
            max_analyze_duration        = 5*AV_TIME_BASE;
            max_subtitle_analyze_duration = 30*AV_TIME_BASE;
            if (!strcmp(ic->iformat->name, "flv"))
                max_stream_analyze_duration = 90*AV_TIME_BASE;
            if (!strcmp(ic->iformat->name, "mpeg") || !strcmp(ic->iformat->name, "mpegts"))
                max_stream_analyze_duration = 7*AV_TIME_BASE;
        }
        
        if (ic->pb)
            av_log(ic, AV_LOG_INFO, "Before avformat_find_stream_info() pos: %"PRId64" bytes read:%"PRId64" seeks:%d nb_streams:%d\n",
                   avio_tell(ic->pb), ic->pb->bytes_read, ic->pb->seek_count, ic->nb_streams);
        
        for (i = 0; i < ic->nb_streams; i++) {
            const AVCodec *codec;
            AVDictionary *thread_opt = NULL;
            st = ic->streams[i];
            avctx = st->internal->avctx;
            
            if (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO ||
                st->codecpar->codec_type == AVMEDIA_TYPE_SUBTITLE) {
                /*            if (!st->time_base.num)
                 st->time_base = */
                if (!avctx->time_base.num)
                    avctx->time_base = st->time_base;
            }
            
            /* check if the caller has overridden the codec id */
    #if FF_API_LAVF_AVCTX
            FF_DISABLE_DEPRECATION_WARNINGS
            if (st->codec->codec_id != st->internal->orig_codec_id) {
                st->codecpar->codec_id   = st->codec->codec_id;
                st->codecpar->codec_type = st->codec->codec_type;
                st->internal->orig_codec_id = st->codec->codec_id;
            }
            FF_ENABLE_DEPRECATION_WARNINGS
    #endif
            // only for the split stuff
            if (!st->parser && !(ic->flags & AVFMT_FLAG_NOPARSE) && st->request_probe <= 0) {
                st->parser = av_parser_init(st->codecpar->codec_id);
                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->codecpar->codec_id));
                }
            }
            
            if (st->codecpar->codec_id != st->internal->orig_codec_id)
                st->internal->orig_codec_id = st->codecpar->codec_id;
            
            ret = avcodec_parameters_to_context(avctx, st->codecpar);
            if (ret < 0)
                goto find_stream_info_err;
            if (st->request_probe <= 0)
                st->internal->avctx_inited = 1;
            
            codec = find_probe_decoder(ic, st, st->codecpar->codec_id);
            
            /* Force thread count to 1 since the H.264 decoder will not extract
             * SPS and PPS to extradata during multi-threaded decoding. */
            av_dict_set(options ? &options[i] : &thread_opt, "threads", "1", 0);
            
            if (ic->codec_whitelist)
                av_dict_set(options ? &options[i] : &thread_opt, "codec_whitelist", ic->codec_whitelist, 0);
            
            /* Ensure that subtitle_header is properly set. */
            if (st->codecpar->codec_type == AVMEDIA_TYPE_SUBTITLE
                && codec && !avctx->codec) {
                if (avcodec_open2(avctx, codec, options ? &options[i] : &thread_opt) < 0)
                    av_log(ic, AV_LOG_WARNING,
                           "Failed to open codec in %s\n",__FUNCTION__);
            }
            
            // Try to just open decoders, in case this is enough to get parameters.
            if (!has_codec_parameters(st, NULL) && st->request_probe <= 0) {
                if (codec && !avctx->codec)
                    if (avcodec_open2(avctx, codec, options ? &options[i] : &thread_opt) < 0)
                        av_log(ic, AV_LOG_WARNING,
                               "Failed to open codec in %s\n",__FUNCTION__);
            }
            if (!options)
                av_dict_free(&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;
        }
        
        read_size = 0;
        int _analyze_finish = 0;
        for (;;) {
            int analyzed_all_streams;
            if (ff_check_interrupt(&ic->interrupt_callback)) {
                ret = AVERROR_EXIT;
                av_log(ic, AV_LOG_DEBUG, "interrupted\n");
                break;
            }
            
            int _video_only = 0;
            int _audio_only = 0;
            if(probesize == 1000)
            {
                _video_only = 1;
                av_log(ic, AV_LOG_DEBUG, "specified stream video only \n");
            }
            
            if(probesize == 4000)
            {
                _audio_only = 1;
                av_log(ic, AV_LOG_DEBUG, "specified stream audio only \n");
            }
            
            /* check if one codec still needs to be handled */
            for (i = 0; i < ic->nb_streams; i++) {
                int fps_analyze_framecount = 20;
                
                st = ic->streams[i];
                if(_video_only && st->codec->codec_type == AVMEDIA_TYPE_AUDIO)
                {
                    continue;
                }
                if(_audio_only && st->codec->codec_type == AVMEDIA_TYPE_VIDEO)
                {
                    continue;
                }
                if (!has_codec_parameters(st, NULL))
                    break;
                if (st->parser && st->parser->parser->split &&
                    !st->codec->extradata && (!strstr(ic->filename, ".m3u8") && !strstr(ic->filename, ".M3U8")))
                {
                    av_log(ic,AV_LOG_DEBUG,"has no extradata \n");
                    break;
                }
                
                if(st->start_time == AV_NOPTS_VALUE)
                {
                    av_log(ic,AV_LOG_DEBUG,"has no start time");
                    break;
                }
                
                if((_video_only && st->codec->codec_type == AVMEDIA_TYPE_VIDEO) || (_audio_only && st->codec->codec_type == AVMEDIA_TYPE_AUDIO))
                    _analyze_finish = 1;
                
                if(ic->fps_probe_size == 1)
                {
                    av_log(ic,AV_LOG_DEBUG,"avformat_find_stream_info skip full detect \n");
                    continue;
                }
                /* 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 (!tb_unreliable(st->internal->avctx))
                    fps_analyze_framecount = 0;
                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 (!(st->r_frame_rate.num && st->avg_frame_rate.num) &&
                    st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) {
                    int count = (ic->iformat->flags & AVFMT_NOTIMESTAMPS) ?
                    st->info->codec_info_duration_fields/2 :
                    st->info->duration_count;
                    if (count < fps_analyze_framecount)
                        break;
                }
                if (!st->internal->avctx->extradata &&
                    (!st->internal->extract_extradata.inited ||
                     st->internal->extract_extradata.bsf) &&
                    extract_extradata_check(st))
                    break;
                if (st->first_dts == AV_NOPTS_VALUE &&
                    !(ic->iformat->flags & AVFMT_NOTIMESTAMPS) &&
                    st->codec_info_nb_frames < ((st->disposition & AV_DISPOSITION_ATTACHED_PIC) ? 1 : ic->max_ts_probe) &&
                    (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO ||
                     st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO))
                    break;
            }
            analyzed_all_streams = 0;
            if (!missing_streams || !*missing_streams || _analyze_finish)
                if (i == ic->nb_streams) {
                    analyzed_all_streams = 1;
                    /* 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) || ic->fps_probe_size == 1) {
                        /* 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 >= probesize) {
                ret = count;
                av_log(ic, AV_LOG_DEBUG,
                       "Probe buffer size limit of %"PRId64" bytes reached\n", probesize);
                for (i = 0; i < ic->nb_streams; i++)
                    if (!ic->streams[i]->r_frame_rate.num &&
                        ic->streams[i]->info->duration_count <= 1 &&
                        ic->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO &&
                        strcmp(ic->iformat->name, "image2"))
                        av_log(ic, AV_LOG_WARNING,
                               "Stream #%d: not enough frames to estimate rate; "
                               "consider increasing probesize\n", i);
                break;
            }
            
            /* NOTE: A new stream can be added there if no header in file
             * (AVFMTCTX_NOHEADER). */
            ret = read_frame_internal(ic, &pkt1);
            if (ret == AVERROR(EAGAIN))
                continue;
            
            if (ret < 0) {
                /* EOF or error*/
                eof_reached = 1;
                break;
            }
            
            pkt = &pkt1;
            
            if (!(ic->flags & AVFMT_FLAG_NOBUFFER)) {
                ret = ff_packet_list_put(&ic->internal->packet_buffer, &ic->internal->packet_buffer_end, pkt, 0);
                if (ret < 0)
                    goto find_stream_info_err;
            }
            
            st = ic->streams[pkt->stream_index];
            if (!(st->disposition & AV_DISPOSITION_ATTACHED_PIC))
                read_size += pkt->size;
            
            avctx = st->internal->avctx;
            if (!st->internal->avctx_inited) {
                ret = avcodec_parameters_to_context(avctx, st->codecpar);
                if (ret < 0)
                    goto find_stream_info_err;
                st->internal->avctx_inited = 1;
            }
            
            if (pkt->dts != AV_NOPTS_VALUE && st->codec_info_nb_frames > 1) {
                /* 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_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_first_dts =
                    st->info->fps_last_dts  = AV_NOPTS_VALUE;
                }
                /* Check for a discontinuity in dts. If the difference in dts
                 * is more than 1000 times the average packet duration in the
                 * sequence, we treat it as a discontinuity. */
                if (st->info->fps_last_dts != AV_NOPTS_VALUE &&
                    st->info->fps_last_dts_idx > st->info->fps_first_dts_idx &&
                    (pkt->dts - st->info->fps_last_dts) / 1000 >
                    (st->info->fps_last_dts     - st->info->fps_first_dts) /
                    (st->info->fps_last_dts_idx - st->info->fps_first_dts_idx)) {
                    av_log(ic, AV_LOG_WARNING,
                           "DTS discontinuity 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_first_dts =
                    st->info->fps_last_dts  = AV_NOPTS_VALUE;
                }
                
                /* update stored dts values */
                if (st->info->fps_first_dts == AV_NOPTS_VALUE) {
                    st->info->fps_first_dts     = pkt->dts;
                    st->info->fps_first_dts_idx = st->codec_info_nb_frames;
                }
                st->info->fps_last_dts     = pkt->dts;
                st->info->fps_last_dts_idx = st->codec_info_nb_frames;
            }
            if (st->codec_info_nb_frames>1) {
                int64_t t = 0;
                int64_t limit;
                
                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 == 0
                    && st->codec_info_nb_frames>30
                    && st->info->fps_first_dts != AV_NOPTS_VALUE
                    && st->info->fps_last_dts  != AV_NOPTS_VALUE)
                    t = FFMAX(t, av_rescale_q(st->info->fps_last_dts - st->info->fps_first_dts, st->time_base, AV_TIME_BASE_Q));
                
                if (analyzed_all_streams)                                limit = max_analyze_duration;
                else if (avctx->codec_type == AVMEDIA_TYPE_SUBTITLE) limit = max_subtitle_analyze_duration;
                else                                                     limit = max_stream_analyze_duration;
                
                if (t >= limit) {
                    av_log(ic, AV_LOG_VERBOSE, "max_analyze_duration %"PRId64" reached at %"PRId64" microseconds st:%d\n",
                           limit,
                           t, pkt->stream_index);
                    if (ic->flags & AVFMT_FLAG_NOBUFFER)
                        av_packet_unref(pkt);
                    break;
                }
                if (pkt->duration) {
                    if (avctx->codec_type == AVMEDIA_TYPE_SUBTITLE && pkt->pts != AV_NOPTS_VALUE && pkt->pts >= st->start_time) {
                        st->info->codec_info_duration = FFMIN(pkt->pts - st->start_time, st->info->codec_info_duration + pkt->duration);
                    } else
                        st->info->codec_info_duration += pkt->duration;
                    st->info->codec_info_duration_fields += st->parser && st->need_parsing && avctx->ticks_per_frame ==2 ? st->parser->repeat_pict + 1 : 2;
                }
            }
    #if FF_API_R_FRAME_RATE
            if (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO)
                ff_rfps_add_frame(ic, st, pkt->dts);
    #endif
            if (!st->internal->avctx->extradata) {
                ret = extract_extradata(st, pkt);
                if (ret < 0)
                    goto find_stream_info_err;
            }
            
            /* 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 AV_CODEC_CAP_CHANNEL_CONF is set this will force decoding of at
             * least one frame of codec data, this makes sure the codec initializes
             * the channel configuration and does not only trust the values from
             * the container. */
            try_decode_frame(ic, st, pkt,
                             (options && i < orig_nb_streams) ? &options[i] : NULL);
            
            if (ic->flags & AVFMT_FLAG_NOBUFFER)
                av_packet_unref(pkt);
            
            st->codec_info_nb_frames++;
            count++;
        }
        
        if (eof_reached) {
            int stream_index;
            for (stream_index = 0; stream_index < ic->nb_streams; stream_index++) {
                st = ic->streams[stream_index];
                avctx = st->internal->avctx;
                if (!has_codec_parameters(st, NULL)) {
                    const AVCodec *codec = find_probe_decoder(ic, st, st->codecpar->codec_id);
                    if (codec && !avctx->codec) {
                        AVDictionary *opts = NULL;
                        if (ic->codec_whitelist)
                            av_dict_set(&opts, "codec_whitelist", ic->codec_whitelist, 0);
                        if (avcodec_open2(avctx, codec, (options && stream_index < orig_nb_streams) ? &options[stream_index] : &opts) < 0)
                            av_log(ic, AV_LOG_WARNING,
                                   "Failed to open codec in %s\n",__FUNCTION__);
                        av_dict_free(&opts);
                    }
                }
                
                // EOF already reached while reading the stream above.
                // So continue with reoordering DTS with whatever delay we have.
                if (ic->internal->packet_buffer && !has_decode_delay_been_guessed(st)) {
                    update_dts_from_pts(ic, stream_index, ic->internal->packet_buffer);
                }
            }
        }
        
        if (flush_codecs) {
            AVPacket empty_pkt = { 0 };
            int err = 0;
            av_init_packet(&empty_pkt);
            
            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(ic, 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);
                    }
                }
            }
        }
        
        ff_rfps_calculate(ic);
        
        for (i = 0; i < ic->nb_streams; i++) {
            st = ic->streams[i];
            avctx = st->internal->avctx;
            if (avctx->codec_type == AVMEDIA_TYPE_VIDEO) {
                if (avctx->codec_id == AV_CODEC_ID_RAWVIDEO && !avctx->codec_tag && !avctx->bits_per_coded_sample) {
                    uint32_t tag= avcodec_pix_fmt_to_codec_tag(avctx->pix_fmt);
                    if (avpriv_find_pix_fmt(avpriv_get_raw_pix_fmt_tags(), tag) == avctx->pix_fmt)
                        avctx->codec_tag= tag;
                }
                
                /* estimate average framerate if not set by demuxer */
                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;
                    AVRational codec_frame_rate = avctx->framerate;
                    
                    if (st->info->codec_info_duration        >= INT64_MAX / st->time_base.num / 2||
                        st->info->codec_info_duration_fields >= INT64_MAX / st->time_base.den ||
                        st->info->codec_info_duration        < 0)
                        continue;
                    av_reduce(&st->avg_frame_rate.num, &st->avg_frame_rate.den,
                              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. */
                    for (j = 0; j < MAX_STD_TIMEBASES; j++) {
                        AVRational std_fps = { get_std_framerate(j), 12 * 1001 };
                        double error       = fabs(av_q2d(st->avg_frame_rate) /
                                                  av_q2d(std_fps) - 1);
                        
                        if (error < best_error) {
                            best_error = error;
                            best_fps   = std_fps.num;
                        }
                        
                        if (ic->internal->prefer_codec_framerate && codec_frame_rate.num > 0 && codec_frame_rate.den > 0) {
                            error       = fabs(av_q2d(codec_frame_rate) /
                                               av_q2d(std_fps) - 1);
                            if (error < best_error) {
                                best_error = error;
                                best_fps   = std_fps.num;
                            }
                        }
                    }
                    if (best_fps)
                        av_reduce(&st->avg_frame_rate.num, &st->avg_frame_rate.den,
                                  best_fps, 12 * 1001, INT_MAX);
                }
                
                if (!st->r_frame_rate.num) {
                    if (    avctx->time_base.den * (int64_t) st->time_base.num
                        <= avctx->time_base.num * avctx->ticks_per_frame * (int64_t) st->time_base.den) {
                        av_reduce(&st->r_frame_rate.num, &st->r_frame_rate.den,
                                  avctx->time_base.den, (int64_t)avctx->time_base.num * avctx->ticks_per_frame, INT_MAX);
                    } else {
                        st->r_frame_rate.num = st->time_base.den;
                        st->r_frame_rate.den = st->time_base.num;
                    }
                }
                if (st->display_aspect_ratio.num && st->display_aspect_ratio.den) {
                    AVRational hw_ratio = { avctx->height, avctx->width };
                    st->sample_aspect_ratio = av_mul_q(st->display_aspect_ratio,
                                                       hw_ratio);
                }
            } else if (avctx->codec_type == AVMEDIA_TYPE_AUDIO) {
                if (!avctx->bits_per_coded_sample)
                    avctx->bits_per_coded_sample =
                    av_get_bits_per_sample(avctx->codec_id);
                // set stream disposition based on audio service type
                switch (avctx->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;
                }
            }
        }
        
        if (probesize)
            estimate_timings(ic, old_offset);
        
        av_opt_set(ic, "skip_clear", "0", AV_OPT_SEARCH_CHILDREN);
        
        if (ret >= 0 && ic->nb_streams)
        /* We could not have all the codec parameters before EOF. */
            ret = -1;
        for (i = 0; i < ic->nb_streams; i++) {
            const char *errmsg;
            st = ic->streams[i];
            
            /* if no packet was ever seen, update context now for has_codec_parameters */
            if (!st->internal->avctx_inited) {
                if (st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO &&
                    st->codecpar->format == AV_SAMPLE_FMT_NONE)
                    st->codecpar->format = st->internal->avctx->sample_fmt;
                ret = avcodec_parameters_to_context(st->internal->avctx, st->codecpar);
                if (ret < 0)
                    goto find_stream_info_err;
            }
            if (!has_codec_parameters(st, &errmsg)) {
                char buf[256];
                avcodec_string(buf, sizeof(buf), st->internal->avctx, 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;
            }
        }
        
        compute_chapters_end(ic);
        
        /* update the stream parameters from the internal codec contexts */
        for (i = 0; i < ic->nb_streams; i++) {
            st = ic->streams[i];
            
            if (st->internal->avctx_inited) {
                int orig_w = st->codecpar->width;
                int orig_h = st->codecpar->height;
                ret = avcodec_parameters_from_context(st->codecpar, st->internal->avctx);
                if (ret < 0)
                    goto find_stream_info_err;
                // The decoder might reduce the video size by the lowres factor.
                if (av_codec_get_lowres(st->internal->avctx) && orig_w) {
                    st->codecpar->width = orig_w;
                    st->codecpar->height = orig_h;
                }
            }
            
    #if FF_API_LAVF_AVCTX
            FF_DISABLE_DEPRECATION_WARNINGS
            ret = avcodec_parameters_to_context(st->codec, st->codecpar);
            if (ret < 0)
                goto find_stream_info_err;
            
            // The old API (AVStream.codec) "requires" the resolution to be adjusted
            // by the lowres factor.
            if (av_codec_get_lowres(st->internal->avctx) && st->internal->avctx->width) {
                av_codec_set_lowres(st->codec, av_codec_get_lowres(st->internal->avctx));
                st->codec->width = st->internal->avctx->width;
                st->codec->height = st->internal->avctx->height;
            }
            
            if (st->codec->codec_tag != MKTAG('t','m','c','d')) {
                st->codec->time_base = st->internal->avctx->time_base;
                st->codec->ticks_per_frame = st->internal->avctx->ticks_per_frame;
            }
            st->codec->framerate = st->avg_frame_rate;
            
            if (st->internal->avctx->subtitle_header) {
                st->codec->subtitle_header = av_malloc(st->internal->avctx->subtitle_header_size);
                if (!st->codec->subtitle_header)
                    goto find_stream_info_err;
                st->codec->subtitle_header_size = st->internal->avctx->subtitle_header_size;
                memcpy(st->codec->subtitle_header, st->internal->avctx->subtitle_header,
                       st->codec->subtitle_header_size);
            }
            
            // Fields unavailable in AVCodecParameters
            st->codec->coded_width = st->internal->avctx->coded_width;
            st->codec->coded_height = st->internal->avctx->coded_height;
            st->codec->properties = st->internal->avctx->properties;
            FF_ENABLE_DEPRECATION_WARNINGS
    #endif
            
            st->internal->avctx_inited = 0;
        }
        
    find_stream_info_err:
        for (i = 0; i < ic->nb_streams; i++) {
            st = ic->streams[i];
            if (st->info)
                av_freep(&st->info->duration_error);
            avcodec_close(ic->streams[i]->internal->avctx);
            av_freep(&ic->streams[i]->info);
            av_bsf_free(&ic->streams[i]->internal->extract_extradata.bsf);
            av_packet_free(&ic->streams[i]->internal->extract_extradata.pkt);
        }
        if (ic->pb)
            av_log(ic, AV_LOG_INFO, "After avformat_find_stream_info() pos: %"PRId64" bytes read:%"PRId64" seeks:%d frames:%d\n",
                   avio_tell(ic->pb), ic->pb->bytes_read, ic->pb->seek_count, count);
        return ret;
    }
    

    5、av_find_best_stream(fmtCtx, AVMEDIA_TYPE_AUDIO, -1, -1, NULL, 0);

    获取音视频流的索引,通过索引取出AVStream,获取相应的音视频解码规格参数概要信息

    音频参数:

    • channels
    • sample_rate
    • codec_id
    • time_base
    • sample_fmt
    • channel_layout
    • bit_rate

    解析到音频参数后可以打开音频解码器

    视频参数:

    • time_base
    • r_frame_rate
    • avg_frame_rate
    • width
    • height
    • codec_id
    • pix_fmt
    • bit_rate

    解析到视频参数后,这里需要区分场景

    H.264码流分Annex-B和AVCC(AVC1)格式。H.265码流是Annex-B和HVCC格式。
    区别有两点:参数组织格式;分隔符。
    
    • AVCC:使用NALU长度(固定字节,通常为4字节)分隔NAL;在头部包含extradata(或sequence header)的结构体。
    • Annex-B:使用start code分隔NAL(start code为三字节或四字节,0x000001或0x00000001,一般是四字节);SPS和PPS按流的方式写在头部。

    移动端直播场景http-flv和hls需要区别处理, mp4/flv/mkv是AVCC/HVCC格式,ts是Annex-B格式
    ,前者需要解析extradata,获取sps、pps等参数,ffmepg提供了工具函数av_bitstream_filter_init("h264_mp4toannexb/hevc_mp4toannexb"),后者直接解码即可,走完上述流程后,由于头里面可能没有sps、pps等信息,所以视频解码器启动需要延后,待到参数获取到之后启动解码器

    6、av_read_frame(fmtCtx, &pkt);

    循环调用av_read_frame(fmtCtx, &pkt)函数,通过pkt.stream_index区分音视频,放入待解码队列

    相关文章

      网友评论

        本文标题:FFMPEG解协议、解封装流程

        本文链接:https://www.haomeiwen.com/subject/xsrboctx.html