ffmpeg解密海思音频

在学了在学了! 2020-10-10 15:40:00 3742

使用海思语音编解码库进行 G711、G726、ADPCM 格式的编码,编码后的码流遵循以下表格中描述的帧结构,即在每帧码流数据的净荷数据之前填充有 4 个字节的帧头;使用语音编解码库进行以上格式的解码时,需要读取相应的帧头信息。

这4个字节的帧头内容即为如下数组中的值:
static char aryHeard[4] = {0,1,160,0}; //hisi audio header
1表示音频 160表示的实际长度为160 * short = 320

解码流程:
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
    #define  Hisi_AUDIO_HERDER_LEN          4                   //hisi 音频数据头  
    extern "C"  
    {  
    #include "libavcodec/avcodec.h"  
    #include "libavformat/avformat.h"  
    #include "libavutil/frame.h"  
    #include "libswscale/swscale.h"  
    #include "libavutil/imgutils.h"  
    }  
    //链接 ffmpeg lib库  

    AVCodec *codec;  
        AVCodecContext *c= NULL;  
        AVPacket avpkt;  
        AVFrame *decoded_frame = NULL;  
        avcodec_register_all();  
        av_init_packet(&avpkt);  
        /* find the mpeg audio decoder */  
        /* 
        ffmpeg g726编码器:AV_CODEC_ID_ADPCM_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解码器类型 
        */  
        codec = avcodec_find_decoder(AV_CODEC_ID_ADPCM_G726LE);  
        if (!codec)   
        {  
            fprintf(stderr, "codec not found\n");  
            return;  
        }  
        c = avcodec_alloc_context3(codec);  
        //采样率= 8000 每个采样用的bit数= 16 通道数= 1  
        /* 
        bits_per_coded_sample:表示编码压缩bit值与采样率的bit值之比。 
        如果为g726音频时,表示g726码流压缩与采样率比值。比如kbps码流压缩比为:k/8k = 5,kbps码流压缩比为k/8k = 2。 
        */  
        c->bits_per_coded_sample = 5;  
        c->channels = 1;  
        c->sample_fmt = AV_SAMPLE_FMT_S16;  
        c->sample_rate = 8000;  
        c->codec_type = AVMEDIA_TYPE_AUDIO;  
        //c->bit_rate = 16000;  
        int iRet = avcodec_open2(c, codec,NULL);  
        if ( iRet < 0 )   
        {  
            fprintf(stderr, "could not open codec\n");  
            return;  
        }  

        CString filePath = "";  
        CString newlFilePath = "";  
        char szFilter[] = {"g726 Files (*.g726_hisi)|*.g726_hisi||"};  
        CFileDialog dlg(TRUE,NULL,NULL,OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,szFilter,NULL);  
        if(dlg.DoModal() == IDOK)  
        {  
            filePath = dlg.GetPathName();  
            newlFilePath = filePath;  
            newlFilePath.Replace(".g726_hisi",".hisi2ff.pcm");  
            BOOL bRet = 0;  
            FILE * fpSrc = fopen(filePath.GetBuffer(filePath.GetLength()),"rb");  
            FILE * fpDst = fopen(newlFilePath.GetBuffer(newlFilePath.GetLength()),"wb+");  
            char szData[100] = {0};  
            char szOutData[320] = {0};  
            int  nDataLen = 100;  
            int  nOutDataLen = 320;  
            int  nReadedSize = 0;  
            unsigned short usHisiHeader[2] = {0};  
            if(fpSrc != NULL)  
            {  
                while(TRUE)  
                {  
                    //读取头标记  
                    nDataLen = Hisi_AUDIO_HERDER_LEN;  
                    nReadedSize = fread(szData,sizeof(char),nDataLen,fpSrc);  
                    if(nReadedSize < nDataLen)  
                    {  
                        break;  
                    }  
                    memcpy(usHisiHeader,szData,Hisi_AUDIO_HERDER_LEN);  
                    int nAudioFrameDataLen = (usHisiHeader[1] & 0x00ff) * sizeof(unsigned short);  
                    nDataLen = nAudioFrameDataLen;  
                    //读取音频帧数据  
                    nReadedSize = fread(szData,sizeof(char),nDataLen,fpSrc);  
                    if(nReadedSize < nDataLen)  
                    {  
                        break;  
                    }  
                    avpkt.data = (uint8_t *)szData;  
                    avpkt.size = nReadedSize;  
                    int got_frame = 0;  
                    if (!decoded_frame)   
                    {  
                        if (!(decoded_frame = avcodec_alloc_frame()))   
                        {  
                            return;  
                        }  
                    }   
                    else  
                    {  
                        avcodec_get_frame_defaults(decoded_frame);  
                    }  
                    int len = avcodec_decode_audio4(c, decoded_frame, &got_frame, &avpkt);  
                    if (len < 0)   
                    {  
                        return;  
                    }  
                    if (got_frame)   
                    {  
                        /* if a frame has been decoded, output it */  
                        int data_size = av_samples_get_buffer_size(NULL, c->channels,  
                            decoded_frame->nb_samples,  
                            c->sample_fmt, 1);  

                        fwrite(decoded_frame->data[0], 1, data_size, fpDst);  

                    }  
                }  

                fclose(fpSrc);  
                fclose(fpDst);  
                avcodec_close(c);  
                av_free(c);  
                av_free(decoded_frame);  
            }  

        }
    原文连接:https://blog.csdn.net/huabiaochen/article/details/83012355
声明:本文内容由易百纳平台入驻作者撰写,文章观点仅代表作者本人,不代表易百纳立场。如有内容侵权或者其他问题,请联系本站进行删除。
红包 点赞 收藏 评论 打赏
评论
0个
内容存在敏感词
手气红包
    易百纳技术社区暂无数据
相关专栏
置顶时间设置
结束时间
删除原因
  • 广告/SPAM
  • 恶意灌水
  • 违规内容
  • 文不对题
  • 重复发帖
打赏作者
易百纳技术社区
在学了在学了!
您的支持将鼓励我继续创作!
打赏金额:
¥1易百纳技术社区
¥5易百纳技术社区
¥10易百纳技术社区
¥50易百纳技术社区
¥100易百纳技术社区
支付方式:
微信支付
支付宝支付
易百纳技术社区微信支付
易百纳技术社区
打赏成功!

感谢您的打赏,如若您也想被打赏,可前往 发表专栏 哦~

举报反馈

举报类型

  • 内容涉黄/赌/毒
  • 内容侵权/抄袭
  • 政治相关
  • 涉嫌广告
  • 侮辱谩骂
  • 其他

详细说明

审核成功

发布时间设置
发布时间:
是否关联周任务-专栏模块

审核失败

失败原因
备注
拼手气红包 红包规则
祝福语
恭喜发财,大吉大利!
红包金额
红包最小金额不能低于5元
红包数量
红包数量范围10~50个
余额支付
当前余额:
可前往问答、专栏板块获取收益 去获取
取 消 确 定

小包子的红包

恭喜发财,大吉大利

已领取20/40,共1.6元 红包规则

    易百纳技术社区