ffmpeg /x264视频流编解码末尾丢帧问题分析和解决

8 篇文章 1 订阅
7 篇文章 0 订阅

一:问题

int avcodec_encode_video2 ( AVCodecContext *  avctx,


AVPacket *  avpkt,


const AVFrame *  frame,


int *  got_packet_ptr 

)


Encode a frame of video.
Takes input raw video data from frame and writes the next output packet, if available, to avpkt. The output packet does not necessarily contain data for the most recent frame, as encoders can delay and reorder input frames internally as needed.

int avcodec_decode_video2 ( AVCodecContext *  avctx,


AVFrame *  picture,


int *  got_picture_ptr,


const AVPacket *  avpkt 

)


Decode the video frame of size avpkt->size from avpkt->data into picture.

Some decoders may support multiple frames in a single AVPacket, such decoders would then just decode the first frame.



官网api的提到:

Takes input raw video data from frame and writes the next output packet, if available, to avpkt. The output packet does not necessarily contain data for the most recent frame, as encoders can delay and reorder input frames internally as needed.


二原因:

拿编码来说,丢帧原因如图:




这被称为编码延迟。延迟原因又分为两种,一是计算延迟,而是缓存延迟。

所以:

h->frames.i_delay =
                                 param->i_sync_lookahead +                        // 前向考虑帧数
                                max ( param->i_bframe,                               // B帧数量
                                          param->rc.i_lookahead)  +                 // 码率控制前向考虑帧数

通过设置参数将编码和获取的帧间隔缩小到0,使用“zerolatency"的tune设定编码模型,详细:

三 解决

编码延迟就需要在编码所有yuv数据之后必须flush。

int vflush_encoder(AVFormatContext *fmt_ctx, unsigned int stream_index) {
    int ret = 0;
    int got_frame;
    AVPacket enc_pkt;
    if (!(fmt_ctx->streams[stream_index]->codec->codec->capabilities
            & CODEC_CAP_DELAY))
        return 0;
    av_init_packet(&enc_pkt);
    while (IS_GOING) {
        enc_pkt.data = NULL;
        enc_pkt.size = 0;
        ret = avcodec_encode_video2(fmt_ctx->streams[stream_index]->codec,
                &enc_pkt, NULL, &got_frame);

        if (ret < 0)
            break;
        if (!got_frame) {
            ret = 0;
            break;
        }
        ret = av_write_frame(fmt_ctx, &enc_pkt);
        printf("Flush Encoder: Succeed to encode 1 frame!\tsize:%5d\n",
                enc_pkt.size);
        av_free_packet(&enc_pkt);

        if (ret < 0)
            break;
    }

    return ret;
}



解码同理,在 av_read_frame到没有avpacket可以读取之后,要继续 调用 avcodec_encode_video2,将视频末尾帧解码出来。



  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是使用FFmpeg解码H.264视频流的示例代码: ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <errno.h> #include <unistd.h> #include <fcntl.h> #include <sys/time.h> #include <sys/types.h> #include <sys/stat.h> extern "C" { #include <libavcodec/avcodec.h> } int main(int argc, char **argv) { AVCodec *codec = NULL; AVCodecContext *codec_ctx = NULL; AVFrame *frame = NULL; AVPacket packet; int ret, i, video_stream_index; int got_frame = 0; int frame_count = 0; int video_width, video_height; struct timeval start_time, end_time; if (argc < 2) { printf("Usage: %s <input_file>\n", argv[0]); return -1; } avcodec_register_all(); // 打开文件并读取视频流信息 AVFormatContext *format_ctx = NULL; if (avformat_open_input(&format_ctx, argv[1], NULL, NULL) != 0) { printf("Couldn't open input file\n"); return -1; } if (avformat_find_stream_info(format_ctx, NULL) < 0) { printf("Couldn't find stream information\n"); return -1; } // 查找视频流,并初始化解码器 for (i = 0; i < format_ctx->nb_streams; i++) { if (format_ctx->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) { video_stream_index = i; codec_ctx = avcodec_alloc_context3(codec); avcodec_parameters_to_context(codec_ctx, format_ctx->streams[i]->codecpar); codec = avcodec_find_decoder(codec_ctx->codec_id); if (!codec) { printf("Unsupported codec\n"); return -1; } if (avcodec_open2(codec_ctx, codec, NULL) < 0) { printf("Could not open codec\n"); return -1; } video_width = codec_ctx->width; video_height = codec_ctx->height; break; } } // 初始化AVFrame并分配内存 frame = av_frame_alloc(); if (!frame) { printf("Could not allocate frame\n"); return -1; } gettimeofday(&start_time, NULL); // 读取视频流并解码 while (av_read_frame(format_ctx, &packet) >= 0) { if (packet.stream_index == video_stream_index) { ret = avcodec_decode_video2(codec_ctx, frame, &got_frame, &packet); if (ret < 0) { printf("Error decoding video frame\n"); return -1; } if (got_frame) { printf("Decoded frame %d\n", frame_count++); // 在这里可以处理解码后的,例如渲染到屏幕上 } } av_packet_unref(&packet); } gettimeofday(&end_time, NULL); double elapsed_time = (end_time.tv_sec - start_time.tv_sec) + (end_time.tv_usec - start_time.tv_usec) / 1000000.0; printf("Decoded %d frames in %f seconds (average fps: %f)\n", frame_count, elapsed_time, frame_count / elapsed_time); avcodec_free_context(&codec_ctx); avformat_close_input(&format_ctx); av_frame_free(&frame); return 0; } ``` 这段代码打开指定文件并读取视频流信息,查找视频流并初始化解码器,然后循环读取视频流并解码每个视频。在解码每个后,您可以将其渲染到屏幕上或进行其他处理。最后,它将打印解码的数量和解码时间,然后释放所有资源。 请注意,为了简化代码,这个示例忽略了错误处理和内存释放。在实际应用中,您需要确保正确地处理和释放所有资源,以避免内存泄漏和其他问题

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值