zoukankan      html  css  js  c++  java
  • ffmpeg摄像头采集h264编码RTP发送

    一. 相关API说明

    1. av_register_all 
    2. avformat_network_init 
        不管是流媒体发送还是流媒体接收, 需要先执行该函数. 
    3. avformat_alloc_context 
        初始化一个AVFormatContext. 
    4. avcodec_alloc_frame
        初始化1个AVFrame. **需要注意的是,新版的FFmpeg使用av_frame_alloc来初始化一个AVFrame.**
    5. av_guess_format
        返回一个已经注册的最合适的匹配输入参数地输出格式.
    6. avio_open
        打开FFmpeg的输入输出文件. **需要注意的是, 新版的Ffmpeg使用avio_open2来打开输入输出文件.**
    7. avcodec_find_encoder
        通过code ID查找一个已经注册的音视频编码器.
    8. avformat_new_stream
        初始化一个AVStream. **销毁函数使用销毁AVFormatContext的avformat_free_context()就可以了**
    9. avcodec_get_context_defaults3
        用于设置该AVCodecContext的默认值.
    10. av_opt_set
        设置一个AVOption的值.
    11. avcodec_open2
        打开编码器.
    12. avformat_write_header
        写文件头.
    13. av_sdp_create
        打印sdp信息.
    14. av_init_packet
         使用默认值初始化AVPacket.
    15. avcodec_encode_video2
        编码一帧视频数据.
    16. av_rescale_q
        用于计算Packet的PTS.
    17. av_interleaved_write_frame
        写入一个AVPacket到输出文件.
    18. av_free_packet
        销毁一个AVPacket. 
    19. av_frame_free
        销毁一个AVFrame. 
    20. avcodec_close
        关闭编码器.
    21. av_free
        av_free()简单的封装了free.
    22. avformat_free_context
        销毁一个AVFormatContext.
    

    二. 工作流程

    1. 初始化, 依次完成以下工作
        av_register_all();
    
        //使用RTSP网络流时必须先执行
        avformat_network_init();                                                             
        pFrame = avcodec_alloc_frame();                                              
        fmtctx = avformat_alloc_context();         
    
         //设置流格式为RTP                                   
        fmtctx->oformat = av_guess_format("rtp", NULL, NULL);         
        //用指定IP和端口构造输出流地址
        sprintf(fmtctx->filename,"rtp://%s:%d",SendIp.toLatin1().data(),SendPort);
    
        //打开输出流
        avio_open(&fmtctx->pb,fmtctx->filename, AVIO_FLAG_WRITE);
    
        //查找编码器
        *codec = avcodec_find_encoder(codec_id);
    
        //初始化AVStream
        st = avformat_new_stream(oc, *codec);
    
        //设置AVCodecContext编码参数    
        avcodec_get_context_defaults3(c, *codec);
        c->codec_id = codec_id;
        c->codec_type = AVMEDIA_TYPE_VIDEO;
        c->width = capWidth;
        c->height = capHeight;
        c->time_base.den = frameRate;
        c->time_base.num = 1;
        c->pix_fmt = AV_PIX_FMT_YUV420P;   //PIX_FMT_YUV420P;
        if(oc->oformat->flags & AVFMT_GLOBALHEADER)
            c->flags|= CODEC_FLAG_GLOBAL_HEADER;
    
        av_opt_set(c->priv_data, "preset", "slow", 0);
        av_opt_set(c->priv_data, "tune","zerolatency",0);
        av_opt_set(c->priv_data, "x264opts","crf=26:vbv-maxrate=728:vbv-bufsize=3640:keyint=25",0);
    
        //打开编码器
        avcodec_open2(video_st->codec, video_codec, NULL);
    
        //写文件头
        avformat_write_header(fmtctx, NULL);
    
        //打印SDP信息, 该信息可用于Rtp流接收解码
        av_sdp_create(&fmtctx,1, sdp, sizeof(sdp));
    2. 视频编码, rtp传输.  该处使用一个单独的线程完成.
        while (1) 
        {
            //退出线程, 略
    
            //获取RGB图像
            cap>>srcMat;
    
            //格式转换, BGR->YUV420P
            convertMatBGR2YUV420(dstMat ,yMat,uMat,vMat);
    
            //构造AVPacket.
            av_init_packet(&pkt);
    
            pFrame->pts = video_st->codec->frame_number;
    
             //视频编码
            avcodec_encode_video2(c, &pkt,pFrame, &got_output);
            if (got_output) {
                if (c->coded_frame->key_frame)
                    pkt.flags |= AV_PKT_FLAG_KEY;
                pkt.stream_index = st->index;
    
                //计算PTS
                if (pkt.pts != AV_NOPTS_VALUE ) {
                    pkt.pts = av_rescale_q(pkt.pts,video_st->codec->time_base, video_st->time_base);
                }
                if(pkt.dts !=AV_NOPTS_VALUE ) {
                    pkt.dts = av_rescale_q(pkt.dts,video_st->codec->time_base, video_st->time_base);
                }
    
                //写入一个AVPacket到输出文件, 这里是一个输出流
                av_interleaved_write_frame(oc,&pkt);
            }
    
            //销毁AVPacket.
            av_free_packet(&pkt);
        }
    
        //销毁资源
        av_frame_free(&pFrame);
        avcodec_close(video_st->codec);
        av_free(video_st->codec);
        avformat_free_context(fmtctx);
    
    3. 其他, 略.
    

    三. 运行 
    程序运行后 
    点击”打开”按纽, 启动RTP传输. 
    这里写图片描述

    RTP接收 
    这里写图片描述

  • 相关阅读:
    项目实战9—企业级分布式存储应用与实战MogileFS、FastDFS
    项目详解4—haproxy 反向代理负载均衡
    项目实战4—HAProxy实现高级负载均衡实战和ACL控制
    项目实战2.1—nginx 反向代理负载均衡、动静分离和缓存的实现
    zabbix设置报警通知
    zabbix创建触发器
    zabbix的启动和关闭脚本
    zabbix监控第一台服务器
    zabbix的源码安装
    Linux命令之乐--iconv
  • 原文地址:https://www.cnblogs.com/lidabo/p/8663062.html
Copyright © 2011-2022 走看看