ffmpeg库解码海思G726库编码音频数据

http://blog.csdn.net/byxdaz/article/details/78430472

解码流程:

1、  读取海思g726音频数据,海思g726音频会多4个字节的海思头信息。

2、选择ffmpeg g726编码器进行解码。ffmpeg g726解码器包括:AV_CODEC_ID_ADPCM_G726、AV_CODEC_ID_ADPCM_G726LE。如果海思g726码流类型为asf时,请选择AV_CODEC_ID_ADPCM_G726解码器类型;如果海思g726码流类型为RFC3551标准时,请选择AV_CODEC_ID_ADPCM_G726LE解码器类型。

 实例代码:

[cpp]  view plain  copy
  1. #define  Hisi_AUDIO_HERDER_LEN          4                   //hisi 音频数据头  
  2. extern "C"  
  3. {  
  4. #include "libavcodec/avcodec.h"  
  5. #include "libavformat/avformat.h"  
  6. #include "libavutil/frame.h"  
  7. #include "libswscale/swscale.h"  
  8. #include "libavutil/imgutils.h"  
  9. }  
  10. //链接 ffmpeg lib库  
  11.   
  12. AVCodec *codec;  
  13.     AVCodecContext *c= NULL;  
  14.     AVPacket avpkt;  
  15.     AVFrame *decoded_frame = NULL;  
  16.     avcodec_register_all();  
  17.     av_init_packet(&avpkt);  
  18.     /* find the mpeg audio decoder */  
  19.     /* 
  20.     ffmpeg g726编码器:AV_CODEC_ID_ADPCM_G726 
  21.     ffmpeg g726解码器包括:AV_CODEC_ID_ADPCM_G726、AV_CODEC_ID_ADPCM_G726LE 
  22.     如果海思g726码流类型为asf时,请选择AV_CODEC_ID_ADPCM_G726解码器类型 
  23.     如果海思g726码流类型为RFC3551标准时,请选择AV_CODEC_ID_ADPCM_G726LE解码器类型 
  24.     */  
  25.     codec = avcodec_find_decoder(AV_CODEC_ID_ADPCM_G726LE);  
  26.     if (!codec)   
  27.     {  
  28.         fprintf(stderr, "codec not found\n");  
  29.         return;  
  30.     }  
  31.     c = avcodec_alloc_context3(codec);  
  32.     //采样率= 8000 每个采样用的bit数= 16 通道数= 1  
  33.     /* 
  34.     bits_per_coded_sample:表示编码压缩bit值与采样率的bit值之比。 
  35.     如果为g726音频时,表示g726码流压缩与采样率比值。比如kbps码流压缩比为:k/8k = 5,kbps码流压缩比为k/8k = 2。 
  36.     */  
  37.     c->bits_per_coded_sample = 5;  
  38.     c->channels = 1;  
  39.     c->sample_fmt = AV_SAMPLE_FMT_S16;  
  40.     c->sample_rate = 8000;  
  41.     c->codec_type = AVMEDIA_TYPE_AUDIO;  
  42.     //c->bit_rate = 16000;  
  43.     int iRet = avcodec_open2(c, codec,NULL);  
  44.     if ( iRet < 0 )   
  45.     {  
  46.         fprintf(stderr, "could not open codec\n");  
  47.         return;  
  48.     }  
  49.   
  50.     CString filePath = "";  
  51.     CString newlFilePath = "";  
  52.     char szFilter[] = {"g726 Files (*.g726_hisi)|*.g726_hisi||"};  
  53.     CFileDialog dlg(TRUE,NULL,NULL,OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,szFilter,NULL);  
  54.     if(dlg.DoModal() == IDOK)  
  55.     {  
  56.         filePath = dlg.GetPathName();  
  57.         newlFilePath = filePath;  
  58.         newlFilePath.Replace(".g726_hisi",".hisi2ff.pcm");  
  59.         BOOL bRet = 0;  
  60.         FILE * fpSrc = fopen(filePath.GetBuffer(filePath.GetLength()),"rb");  
  61.         FILE * fpDst = fopen(newlFilePath.GetBuffer(newlFilePath.GetLength()),"wb+");  
  62.         char szData[100] = {0};  
  63.         char szOutData[320] = {0};  
  64.         int  nDataLen = 100;  
  65.         int  nOutDataLen = 320;  
  66.         int  nReadedSize = 0;  
  67.         unsigned short usHisiHeader[2] = {0};  
  68.         if(fpSrc != NULL)  
  69.         {  
  70.             while(TRUE)  
  71.             {  
  72.                 //读取头标记  
  73.                 nDataLen = Hisi_AUDIO_HERDER_LEN;  
  74.                 nReadedSize = fread(szData,sizeof(char),nDataLen,fpSrc);  
  75.                 if(nReadedSize < nDataLen)  
  76.                 {  
  77.                     break;  
  78.                 }  
  79.                 memcpy(usHisiHeader,szData,Hisi_AUDIO_HERDER_LEN);  
  80.                 int nAudioFrameDataLen = (usHisiHeader[1] & 0x00ff) * sizeof(unsigned short);  
  81.                 nDataLen = nAudioFrameDataLen;  
  82.                 //读取音频帧数据  
  83.                 nReadedSize = fread(szData,sizeof(char),nDataLen,fpSrc);  
  84.                 if(nReadedSize < nDataLen)  
  85.                 {  
  86.                     break;  
  87.                 }  
  88.                 avpkt.data = (uint8_t *)szData;  
  89.                 avpkt.size = nReadedSize;  
  90.                 int got_frame = 0;  
  91.                 if (!decoded_frame)   
  92.                 {  
  93.                     if (!(decoded_frame = avcodec_alloc_frame()))   
  94.                     {  
  95.                         return;  
  96.                     }  
  97.                 }   
  98.                 else  
  99.                 {  
  100.                     avcodec_get_frame_defaults(decoded_frame);  
  101.                 }  
  102.                 int len = avcodec_decode_audio4(c, decoded_frame, &got_frame, &avpkt);  
  103.                 if (len < 0)   
  104.                 {  
  105.                     return;  
  106.                 }  
  107.                 if (got_frame)   
  108.                 {  
  109.                     /* if a frame has been decoded, output it */  
  110.                     int data_size = av_samples_get_buffer_size(NULL, c->channels,  
  111.                         decoded_frame->nb_samples,  
  112.                         c->sample_fmt, 1);  
  113.   
  114.                     fwrite(decoded_frame->data[0], 1, data_size, fpDst);  
  115.                       
  116.                 }  
  117.             }  
  118.   
  119.             fclose(fpSrc);  
  120.             fclose(fpDst);  
  121.             avcodec_close(c);  
  122.             av_free(c);  
  123.             av_free(decoded_frame);  
  124.         }  
  125.   
  126.     }  

代码下载
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值