天天看點

FFmpeg音視訊編解碼實踐總結PS:由于目前開發RTSP伺服器傳輸子產品時用到了h264檔案,是以攻了一段時間去實作h264的視訊編解碼,借用FFmpeg SDK實作了任意檔案格式之間的轉換,并實作了流媒體實時播放,目前音視訊同步需要稍加完善,視訊編碼代碼已成功移植到Visual Stdio平台,如有需要的留下郵箱以下文檔來自FFmpeg工程組(http://www.ffmpeg.com.cn/index.php開發事例)實作轉碼一個普通視訊檔案為視訊mpeg4,音頻mp3的功能的程式

PS:由于目前開發RTSP伺服器傳輸子產品時用到了h264檔案,是以攻了一段時間去實作h264的視訊編解碼,借用FFmpeg SDK實作了任意檔案格式之間的轉換,并實作了流媒體實時播放,目前音視訊同步需要稍加完善,視訊編碼代碼已成功移植到Visual Stdio平台,如有需要的留下郵箱

以下文檔來自FFmpeg工程組(http://www.ffmpeg.com.cn/index.php開發事例)

實作轉碼一個普通視訊檔案為視訊mpeg4,音頻mp3的功能的程式

本程式源引自FFmpeg工程組,實作轉碼一個普通視訊檔案為視訊mpeg4,音頻mp3的功能

#include <avcodec.h>
#include <avformat.h>
#include <stdio.h>
#include <avutil.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
main(int argc,char **argv)
{   
const char *input_file_name="/root/movies/ddh1.mpg";
  av_register_all();//注冊庫中所有可用的檔案格式和編碼器
  AVFormatContext *ic;
//輸入檔案處理部分
  ic=av_alloc_format_context();
if(av_open_input_file(&ic,input_file_name,NULL,0,NULL)!=0)
  {
     printf("can't open the file %s\n",input_file_name);
     exit(1);
  }//打開輸入檔案
  if(av_find_stream_info(ic)<0)
  {
     printf("can't find suitable codec parameters\n");
     exit(1);
  }//取出流資訊
  dump_format(ic,0,input_file_name,0);//列出輸入檔案的相關流資訊
  int i;
int videoindex=-1;int audioindex=-1;
for(i=0;i<ic->nb_streams;i++)
  {    
if(ic->streams[i]->codec->codec_type==CODEC_TYPE_VIDEO)
         {
            videoindex=i;
//printf("video\n");
         }
else if(ic->streams[i]->codec->codec_type==CODEC_TYPE_AUDIO)
         {
//printf("audio\n");
            audioindex=i;
         }
  }
if(videoindex==-1)
   {
          printf("can't find video stream\n");
          exit(1);
   }//沒有找到視訊流
  AVCodecContext *vCodecCtx;
  vCodecCtx=ic->streams[videoindex]->codec;//取得視訊流編碼上下文指針
  AVCodec *vCodec;
  vCodec=avcodec_find_decoder(vCodecCtx->codec_id);
if(vCodec==NULL)
  {
     printf("can't find suitable video decoder\n");
     exit(1);
  }//找到合适的視訊解碼器
  if(avcodec_open(vCodecCtx,vCodec)<0)
  {
     printf("can't open the video decoder\n");
     exit(1);
  }//打開該視訊解碼器
   if(audioindex==-1)
     {
        printf("can't find audio stream\n");
        exit(1);
     }//沒有找到音頻流
  AVCodecContext *aCodecCtx;
  aCodecCtx=ic->streams[audioindex]->codec;
  AVCodec *aCodec;
  aCodec=avcodec_find_decoder(aCodecCtx->codec_id);
if(aCodec==NULL)
  {
     printf("can't find suitable audio decoder\n");
     exit(1);
  }//找到合适的音頻解碼器
  if(avcodec_open(aCodecCtx,aCodec)<0)
  {
     printf("can't open the audio decoder\n");
     exit(1);
  }//打開該音頻解碼器
//下面為輸出檔案處理部分
    const char *output_file_name="/root/123.avi";
    AVOutputFormat *fmt;
    AVFormatContext *oc;
    AVCodecContext *oVcc,*oAcc;
    AVCodec *oVc,*oAc;
    AVStream *video_st,*audio_st;
    AVFrame *oVFrame,*oAFrame;
double video_pts;
    oVFrame=avcodec_alloc_frame();
    fmt=guess_format(NULL,output_file_name,NULL);
if(!fmt)
    {
           printf("could not deduce output format from outfile extension\n");
           exit(0);
    }//判斷是否可以判斷輸出檔案的編碼格式
    oc=av_alloc_format_context();
if(!oc)
    {
           printf("Memory error\n");
           exit(0);
    }
    oc->oformat=fmt;
    pstrcpy(oc->filename,sizeof(oc->filename),output_file_name);
    video_st=av_new_stream(oc,0);
if(!video_st)
    {
          printf("could not alloc video stream\n");
          exit(0);
    }
    oVcc=avcodec_alloc_context();
    oVcc=video_st->codec;
    oVcc->codec_id=CODEC_ID_MPEG4;
    oVcc->codec_type=CODEC_TYPE_VIDEO;
    oVcc->bit_rate=2500000;
    oVcc->width=704;
    oVcc->height=480;
    oVcc->time_base=vCodecCtx->time_base;
    oVcc->gop_size=vCodecCtx->gop_size;
//oVcc->pix_fmt=vCodecCtx->pix_fmt;
    oVcc->pix_fmt=vCodecCtx->pix_fmt;
    oVcc->max_b_frames=vCodecCtx->max_b_frames;
    video_st->r_frame_rate=ic->streams[videoindex]->r_frame_rate;
    audio_st=av_new_stream(oc,oc->nb_streams);
if(!audio_st)
    {
           printf("could not alloc audio stream\n");
           exit(0);
    }  
    avcodec_get_context_defaults2(audio_st->codec,CODEC_TYPE_AUDIO);
    oAcc=avcodec_alloc_context();
    oAcc=audio_st->codec;
    oAcc->codec_id=CODEC_ID_MP3;
    oAcc->codec_type=CODEC_TYPE_AUDIO;
    oAcc->bit_rate=aCodecCtx->bit_rate;
    oAcc->sample_rate=aCodecCtx->sample_rate;
    oAcc->channels=2;
if (av_set_parameters(oc, NULL) < 0) 
    {
           printf( "Invalid output format parameters\n");                         
              exit(0);                               
    }//設定必要的輸出參數
    strcpy(oc->title,ic->title);
    strcpy(oc->author,ic->author);
    strcpy(oc->copyright,ic->copyright);
    strcpy(oc->comment,ic->comment);
    strcpy(oc->album,ic->album);
    oc->year=ic->year;
    oc->track=ic->track;
    strcpy(oc->genre,ic->genre);
    dump_format(oc,0,output_file_name,1);//列出輸出檔案的相關流資訊
    oVc=avcodec_find_encoder(CODEC_ID_MPEG4);
if(!oVc)
    {
       printf("can't find suitable video encoder\n");
           exit(0);
    }//找到合适的視訊編碼器
    if(avcodec_open(oVcc,oVc)<0)
    {
           printf("can't open the output video codec\n");
           exit(0);
    }//打開視訊編碼器
    oAc=avcodec_find_encoder(CODEC_ID_MP3);
if(!oAc)
    {
           printf("can't find suitable audio encoder\n");
           exit(0);
    }//找到合适的音頻編碼器
    if(avcodec_open(oAcc,oAc)<0)
    {
           printf("can't open the output audio codec");
           exit(0);
    }//打開音頻編碼器
    /*if(url_exist(output_file_name))
    {
       printf("the output file name %s has exist,please select other\n",output_file_name);
       exit(0);
    }//判斷該輸出檔案是否已經存在*/
if (!(oc->flags & AVFMT_NOFILE))
    {
if(url_fopen(&oc->pb,output_file_name,URL_WRONLY)<0)
       {
              printf("can't open the output file %s\n",output_file_name);
              exit(0);
       }//打開輸出檔案
    }
if(!oc->nb_streams)
    {
           fprintf(stderr,"output file dose not contain any stream\n");
           exit(0);
    }//檢視輸出檔案是否含有流資訊
  if(av_write_header(oc)<0)
  {
      fprintf(stderr, "Could not write header for output file\n");
      exit(1);
  }[/i][/i]
AVPacket packet;
  uint8_t *ptr,*out_buf;
int out_size;
static short *samples=NULL;
static unsigned int samples_size=0;
  uint8_t *video_outbuf,*audio_outbuf;int video_outbuf_size,audio_outbuf_size;
  video_outbuf_size=400000;
  video_outbuf= (unsigned char *) malloc(video_outbuf_size);
  audio_outbuf_size = 10000;
  audio_outbuf = av_malloc(audio_outbuf_size);
int flag;int frameFinished;int len;int frame_index=0,ret;
while(av_read_frame(ic,&packet)>=0)//從輸入檔案中讀取一個包
       {
if(packet.stream_index==videoindex)//判斷是否為目前視訊流中的包
          {
         len=avcodec_decode_video(vCodecCtx,oVFrame,&frameFinished,packet.data,packet.size);//若為視訊包,解碼該視訊包
                 if(len<0)
                 {
                    printf("Error while decoding\n");
                    exit(0);
                 }
if(frameFinished)//判斷視訊祯是否讀完
         {
             fflush(stdout);
             oVFrame->pts=av_rescale(frame_index,AV_TIME_BASE*(int64_t)oVcc->time_base.num,oVcc->time_base.den);
             oVFrame->pict_type=0;
             out_size = avcodec_encode_video(oVcc, video_outbuf, video_outbuf_size, oVFrame);   
if (out_size > 0)            
             {                   
                 AVPacket pkt;               
                 av_init_packet(&pkt);                               
if(oVcc->coded_frame && oVcc->coded_frame->key_frame)                                       
                     pkt.flags |= PKT_FLAG_KEY;                                        
                     pkt.flags = packet.flags;                      
                     pkt.stream_index= video_st->index;                                               
                     pkt.data= video_outbuf;                                                         
                     pkt.size= out_size;                                             
                     ret=av_write_frame(oc, &pkt);                                       
             }
             frame_index++;
         }
else
             ret=av_write_frame(oc, &packet); 
//img_convert((AVPicture *)vFrame, PIX_FMT_RGB24, (AVPicture*)oVFrame, oVcc->pix_fmt,oVcc->width,oVcc-
>height);
//SaveFrame(vFrame,oVcc->width,oVcc->height,frame_index); 
                    if(ret!=0)
                    {
                      printf("while write video frame error\n");
                      exit(0);
                    }
          }
else if(packet.stream_index==audioindex)
      {
         len=packet.size;
         ptr=packet.data;
int ret=0;
while(len>0)
             {
                    out_buf=NULL;
                    out_size=0;
if(&packet)
               samples=av_fast_realloc(samples,&samples_size,FFMAX(packet.size*sizeof
(*samples),AVCODEC_MAX_AUDIO_FRAME_SIZE));
                    out_size=samples_size;
                    ret=avcodec_decode_audio(aCodecCtx,samples,&out_size,ptr,len);//若為音頻包,解碼該音頻包
                    if(ret<0)
                    {
                       printf("while decode audio failure\n");
                       exit(0);
                    }
            fflush(stdout);
            ptr+=ret;
            len-=ret;
if(out_size<=0)
continue;
            out_buf=(uint8_t *)samples;
            AVPacket pkt;
            av_init_packet(&pkt);
            pkt.size= avcodec_encode_audio(oAcc, audio_outbuf, audio_outbuf_size, out_buf);
            pkt.pts= av_rescale_q(oAcc->coded_frame->pts, oAcc->time_base, audio_st->time_base);
            pkt.flags |= PKT_FLAG_KEY;
            pkt.stream_index= audioindex;
            pkt.data= audio_outbuf;
if (av_write_frame(oc, &pkt) != 0) 
            {
               fprintf(stderr, "Error while writing audio frame\n");
               exit(1);
                }
         }
          } 
          av_free_packet(&packet);
       } 
av_write_trailer(oc);
for(i = 0; i < oc->nb_streams; i++) 
{            
  av_freep(&oc->streams[i]->codec);                       
  av_freep(&oc->streams[i]);                           
}
url_fclose(oc);
av_free(oc);
av_free(oVFrame);
av_free(out_buf);
avcodec_close(vCodecCtx);
avcodec_close(aCodecCtx);
av_close_input_file(ic);
}      
最近一直很忙  花時間整理了一份  希望大家多多交流ffmpeg視訊編解碼方面的知識  接下來會抽時間總結一篇RTP傳輸協定的文章  好了  廢話不說  附上連結自己下載下傳      
source download:  http://files.cnblogs.com/msopengl/ffmpegh264codec.rar      

作者:jemmyLiu 出處:http://www.cnblogs.com/msopengl/ 本文版權歸作者和部落格園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接配接,否則保留追究法律責任的權利。   一些接口已經老到已經很多事情記不清了,隻記得那年中國隊勇奪世界杯:  

FFmpeg音視訊編解碼實踐總結PS:由于目前開發RTSP伺服器傳輸子產品時用到了h264檔案,是以攻了一段時間去實作h264的視訊編解碼,借用FFmpeg SDK實作了任意檔案格式之間的轉換,并實作了流媒體實時播放,目前音視訊同步需要稍加完善,視訊編碼代碼已成功移植到Visual Stdio平台,如有需要的留下郵箱以下文檔來自FFmpeg工程組(http://www.ffmpeg.com.cn/index.php開發事例)實作轉碼一個普通視訊檔案為視訊mpeg4,音頻mp3的功能的程式

    更新下: 27:av_alloc_format_context -> avformat_alloc_context 28:if(av_open_input_file(&ic,input_file_name,NULL,0,NULL)!=0) -> if(avformat_open_input(&ic, input_file_name, NULL, NULL) !=0 ) 33: if(av_find_stream_info(ic)<0) -> if(avformat_find_stream_info(ic,NULL)<0) 38:dump_format(ic,0,input_file_name,0) -> av_dump_format(ic,0,input_file_name,0) 43: CODEC_TYPE_VIDEO -> AVMEDIA_TYPE_VIDEO 48: CODEC_TYPE_AUDIO -> AVMEDIA_TYPE_AUDIO 68: if(avcodec_open(vCodecCtx,vCodec)<0) -> if(avcodec_open2(vCodecCtx,vCodec,NULL)<0) 87: if(avcodec_open(aCodecCtx,aCodec)<0) -> if(avcodec_open2(aCodecCtx,aCodec,NULL)<0) 101: oVFrame=avcodec_alloc_frame() -> oVFrame=av_frame_alloc() 102: guess_format -> av_guess_format 108: av_alloc_format_context -> avformat_alloc_context 116: video_st=av_new_stream(oc,0) -> video_st=avformat_new_stream(oc,oVc) 122: oVcc=avcodec_alloc_context() -> oVcc=avcodec_alloc_context3(oVc) 124: CODEC_ID_H264 -> AV_CODEC_ID_H264 125: CODEC_TYPE_VIDEO -> AVMEDIA_TYPE_VIDEO 135: audio_st=av_new_stream(oc,oc->nb_streams) -> audio_st=avformat_new_stream(oc,oAc) 141: avcodec_get_context_defaults2(audio_st->codec,CODEC_TYPE_AUDIO) -> avcodec_get_context_defaults3(audio_st->codec,oAc) 142: oAcc=avcodec_alloc_context() -> oAcc=avcodec_alloc_context3(oAc) 144: CODEC_ID_MP3 -> AV_CODEC_ID_AAC (這種自己選擇) 145: CODEC_TYPE_AUDIO ->AVMEDIA_TYPE_AUDIO 146: …………  

轉載于:https://www.cnblogs.com/youngt/p/3732048.html