zoukankan      html  css  js  c++  java
  • (转)使用FFMpeg进行H264编码 .

    使用FFMpeg可以很方便的对音视频进行编码,并且写文件。

    下面的代码是将5幅1280*720大小的图片进行编码,并且写到文件中。

    代码有些乱,但希望能抛砖引玉,对学习这方面的朋友有帮助。

      1 CFile file[5];
      2 BYTE *szTxt[5];
      3 
      4 int nWidth = 0;
      5 int nHeight= 0;
      6 
      7 int nDataLen=0;
      8 
      9 int nLen;
     10 
     11 CString csFileName;
     12 for (int fileI = 1; fileI <= 5; fileI ++)
     13 {
     14   csFileName.Format("e:\\pics\\%d.bmp", fileI);
     15   file[fileI - 1].Open(csFileName,CFile::modeRead | CFile::typeBinary);
     16   nLen = file[fileI - 1].GetLength();
     17 
     18   szTxt[fileI -1] = new BYTE[nLen];
     19   file[fileI - 1].Read(szTxt[fileI - 1], nLen);
     20   file[fileI - 1].Close();
     21 
     22   //BMP bmi;//BITMAPINFO bmi;
     23   //int nHeadLen = sizeof(BMP);
     24   BITMAPFILEHEADER bmpFHeader;
     25   BITMAPINFOHEADER bmpIHeader;
     26   memcpy(&bmpFHeader,szTxt[fileI -1],sizeof(BITMAPFILEHEADER));
     27 
     28   int nHeadLen = bmpFHeader.bfOffBits - sizeof(BITMAPFILEHEADER);
     29   memcpy(&bmpIHeader,szTxt[fileI - 1]+sizeof(BITMAPFILEHEADER),nHeadLen);
     30 
     31 nWidth = bmpIHeader.biWidth;// 464;// bmi.bmpInfo.bmiHeader.biWidth;// ;
     32   nHeight = bmpIHeader.biHeight;//362;// bmi.bmpInfo.bmiHeader.biHeight;// ;
     33 
     34   szTxt[fileI - 1] += bmpFHeader.bfOffBits;
     35   nDataLen = nLen-bmpFHeader.bfOffBits;
     36 }
     37 
     38 av_register_all();
     39 avcodec_register_all();
     40 AVFrame *m_pRGBFrame =  new AVFrame[1];  //RGB帧数据  
     41 AVFrame *m_pYUVFrame = new AVFrame[1];;  //YUV帧数据
     42 AVCodecContext *c= NULL;
     43 AVCodecContext *in_c= NULL;
     44 AVCodec *pCodecH264; //编码器
     45 uint8_t * yuv_buff;//
     46 
     47 //查找h264编码器
     48 pCodecH264 = avcodec_find_encoder(CODEC_ID_H264);
     49 if(!pCodecH264)
     50 {
     51   fprintf(stderr, "h264 codec not found\n");
     52   exit(1);
     53 }
     54 
     55 c= avcodec_alloc_context3(pCodecH264);
     56 c->bit_rate = 3000000;// put sample parameters 
     57 c->width =nWidth;// 
     58 c->height = nHeight;// 
     59 
     60 // frames per second 
     61 AVRational rate;
     62 rate.num = 1;
     63 rate.den = 25;
     64 c->time_base= rate;//(AVRational){1,25};
     65 c->gop_size = 10; // emit one intra frame every ten frames 
     66 c->max_b_frames=1;
     67 c->thread_count = 1;
     68 c->pix_fmt = PIX_FMT_YUV420P;//PIX_FMT_RGB24;
     69 
     70 //av_opt_set(c->priv_data, /*"preset"*/"libvpx-1080p.ffpreset", /*"slow"*/NULL, 0);
     71 //打开编码器
     72 if(avcodec_open2(c,pCodecH264,NULL)<0)
     73   TRACE("不能打开编码库");
     74 
     75 int size = c->width * c->height;
     76 
     77 yuv_buff = (uint8_t *) malloc((size * 3) / 2); // size for YUV 420 
     78 
     79 //将rgb图像数据填充rgb帧
     80 uint8_t * rgb_buff = new uint8_t[nDataLen];
     81 
     82 //图象编码
     83 int outbuf_size=100000;
     84 uint8_t * outbuf= (uint8_t*)malloc(outbuf_size); 
     85 int u_size = 0;
     86 FILE *f=NULL; 
     87 char * filename = "e:\\pics\\myData.h264";
     88 f = fopen(filename, "wb");
     89 if (!f)
     90 {
     91   TRACE( "could not open %s\n", filename);
     92   exit(1);
     93 }
     94 
     95 //初始化SwsContext
     96 SwsContext * scxt = sws_getContext(c->width,c->height,PIX_FMT_BGR24,c->width,c->height,PIX_FMT_YUV420P,SWS_POINT,NULL,NULL,NULL);
     97 
     98 AVPacket avpkt;
     99 
    100 //AVFrame *pTFrame=new AVFrame
    101 for (int i=0;i<250;++i)
    102 {
    103 
    104   //AVFrame *m_pYUVFrame = new AVFrame[1];
    105   
    106   int index = (i / 25) % 5;
    107   memcpy(rgb_buff,szTxt[index],nDataLen);
    108 
    109   avpicture_fill((AVPicture*)m_pRGBFrame, (uint8_t*)rgb_buff, PIX_FMT_RGB24, nWidth, nHeight);
    110   
    111   //将YUV buffer 填充YUV Frame
    112   avpicture_fill((AVPicture*)m_pYUVFrame, (uint8_t*)yuv_buff, PIX_FMT_YUV420P, nWidth, nHeight);
    113 
    114   // 翻转RGB图像
    115   m_pRGBFrame->data[0]  += m_pRGBFrame->linesize[0] * (nHeight - 1);
    116   m_pRGBFrame->linesize[0] *= -1;                   
    117   m_pRGBFrame->data[1]  += m_pRGBFrame->linesize[1] * (nHeight / 2 - 1);
    118   m_pRGBFrame->linesize[1] *= -1;
    119   m_pRGBFrame->data[2]  += m_pRGBFrame->linesize[2] * (nHeight / 2 - 1);
    120   m_pRGBFrame->linesize[2] *= -1;
    121 
    122 
    123   //将RGB转化为YUV
    124   sws_scale(scxt,m_pRGBFrame->data,m_pRGBFrame->linesize,0,c->height,m_pYUVFrame->data,m_pYUVFrame->linesize);
    125   
    126   int got_packet_ptr = 0;
    127   av_init_packet(&avpkt);
    128   avpkt.data = outbuf;
    129   avpkt.size = outbuf_size;
    130   u_size = avcodec_encode_video2(c, &avpkt, m_pYUVFrame, &got_packet_ptr);
    131   if (u_size == 0)
    132   {
    133    fwrite(avpkt.data, 1, avpkt.size, f);
    134   }
    135 }
    136 
    137 fclose(f); 
    138 delete []m_pRGBFrame;
    139 delete []m_pYUVFrame;
    140 delete []rgb_buff;
    141 free(outbuf);
    142 avcodec_close(c);
    143 av_free(c);
  • 相关阅读:
    《JavaScript &amp; jQuery交互式Web前端开发》之JavaScript基础指令
    hash_map原理及C++实现
    开源大数据引擎:Greenplum 数据库架构分析
    摆脱命令行,Ubuntu下配置Android开发环境
    JS学习十四天----server端运行JS代码
    【苦读官方文档】2.Android应用程序基本原理概述
    UVA
    android页面间传递对象
    大话设计模式C++版——建造者模式
    poj 3370 Halloween treats
  • 原文地址:https://www.cnblogs.com/billcoco/p/2553948.html
Copyright © 2011-2022 走看看