ISD9160学习笔记03_ISD9160音频解码代码分析
录音例程涉及了录音和播放两大块内容,这篇笔记就先来说说播放,暂且先击破解码这部分功能。
我的锤子便签中有上个月记下的一句话,“斯蒂芬·平克说,写作之难,在于把网状思考,用树状结构,体现在线性展开的语句里。”这篇代码解析也有类似的困难,代码的网状结构,如何用文章这种线性载体来体现。我尽量挑出了主干,来讲解自己的理解。另外在文章最后添加了一个模块拓扑图来帮助消化。
我还是建议大家还是多琢磨下源码,代码的事还是让代码来说话,笔记是一个辅助的概括梳理。
本文作者twowinter,转载请注明:http://blog.csdn.net/iotisan/
查看代码主逻辑,主要是App_StartPlay和App_ProcessPlay这两个函数。下面就分别进行分析。
第一部分 App_StartPlay
BOOL App_StartPlay(void)
{
// Initiate NuLiteEx audio decode lib with callback functions stored in g_asAppCallBack[0]
NuLiteExApp_DecodeInitiate(&g_sApp.sNuLiteExAppDecode, (UINT8 *)&g_sApp.uTempBuf, ); // Start NuLiteEx decode lib to decode NuLiteEx file stored from address and played from audio channel 0.
// And decode the first frame of PCMs.
if ( NuLiteExApp_DecodeStartPlayByAddr(&g_sApp.sNuLiteExAppDecode, AUDIOROM_STORAGE_START_ADDR, ) == FALSE )
return FALSE; // Light playback led(PB9) for display status.
OUT4(); // Start Ultraio Timer & HW pwm for UltraIO curve output
ULTRAIO_START(); // Start to playback audio.
Playback_StartPlay();
}
可以看到App_StartPlay主要牵扯了NuLiteExApp和Playback两部分子函数。
重中之重 NuLiteExApp_DecodeStartPlayByAddr
由于对音频编解码这块比较陌生,我还是给对应代码做了中文注解方便消化。
BOOL NuLiteExApp_DecodeStartPlayByAddr(S_NULITEEX_APP_DECODE *psNuLiteExAppDecode, UINT32 u32NuLiteExStorageStartAddr, UINT8 u8PlaybackChannel)
{
UINT16 u16SampleRate;
// NuLiteEx解码库初始化对应的工作缓冲区,应用层传入temp缓存来方便解码库内部工作。另外根据传入的SPI地址从SPI取文件,获取采样率。
// NuLiteEx decoder initiates work buffer and returns sample rate.
if ( (u16SampleRate = NuLiteEx_DecodeInitiate( (UINT8*)psNuLiteExAppDecode->au32DecodeWorkBuf,
psNuLiteExAppDecode->pau8TempBuf,
u32NuLiteExStorageStartAddr,
g_asAppCallBack[psNuLiteExAppDecode->u8CallbackIndex].pfnReadDataCallback )) == )
return FALSE; // 给Playback模块对接对应的工作缓冲区,方便其下一步播放。
// Initiate and set output buffer variable(include frame size, buffer size etc.)
Playback_SetOutputBuf( &psNuLiteExAppDecode->sOutBufCtrl,
NULITEEXAPP_OUT_BUF_SIZE,
psNuLiteExAppDecode->i16OutBuf,
NULITEEXAPP_OUT_SAMPLES_PER_FRAME,
u16SampleRate ); // 工作缓冲区,置有效位。
// Trigger active flag of output buffer for NuLiteEx decoding
BUF_CTRL_SET_ACTIVE(&psNuLiteExAppDecode->sOutBufCtrl); // 工作缓冲区中的读写指针赋值。
// Pre-decode one frame
psNuLiteExAppDecode->sOutBufCtrl.u16BufWriteIdx = NULITEEXAPP_OUT_SAMPLES_PER_FRAME;
if ( NuLiteExApp_DecodeProcess(psNuLiteExAppDecode) == FALSE )
{
BUF_CTRL_SET_INACTIVE(&psNuLiteExAppDecode->sOutBufCtrl);
return FALSE;
}
psNuLiteExAppDecode->sOutBufCtrl.u16BufReadIdx = NULITEEXAPP_OUT_SAMPLES_PER_FRAME; // 记录当前播放的channel,用来停止播放。
// Record play channel index for stopping to play.
psNuLiteExAppDecode->u8PlaybackChannel = u8PlaybackChannel;
// 准备播放,把这里的循环缓冲区同playback共用。
// Add audio codec into channel and preper to play codec.
Playback_Add(psNuLiteExAppDecode->u8PlaybackChannel, &psNuLiteExAppDecode->sOutBufCtrl); return TRUE;
}
也很重要的Playback_StartPlay
void Playback_StartPlay(void)
{
INT16 *pi16PcmBuf; if( s_u8PlayCtrl == PLAYBACK_NOACTION ) // 这个s_u8PlayCtrl是playback模块内部处理的。
{
#if ( PLAYBACK_CHANNEL_COUNT > 1)
pi16PcmBuf = g_ai16DACSamples;
#else
pi16PcmBuf = &g_psDacBufCtrl->pi16Buf[g_psDacBufCtrl->u16BufReadIdx];// PCM数据缓冲区复制。
#endif #if ((APU_FILTER_ENABLE == 1)&&(APU_UPSAMPLE == 2))
NuDACFilterEx_Up2Initial(g_au8Up2WorkBuf);
#elif ((APU_FILTER_ENABLE == 1)&&(APU_UPSAMPLE == 4))
NuDACFilterEx_Up4Initial(g_au8Up4WorkBuf);
#endif
g_u8AppCtrl|=APPCTRL_PLAY;
s_u8PlayCtrl |= PLAYBACK_START;
#if (APU_ENABLE)
{
UINT8 u8Count; for( u8Count = ; u8Count < ; u8Count ++)
g_ai16DACSamples[u8Count] = ; //Clear virtual buffer
}
#endif Playback_ResetChannelVolume(); SPK_Start(); // 这里头开始调用DPWM来播放DPWM->DATA,DPWM_START_PLAY(DPWM); #if (APU_PDMA_ENABLE)
PdmaCtrl_Start(APU_PDMA_CH, (uint32_t *)pi16PcmBuf, (uint32_t *)&DPWM->DATA, );// 将PCM缓冲数据传到DPWM->DATA中。
#endif }
}
第二部分 App_ProcessPlay
App_ProcessPlay只调用了如下这个函数
BOOL NuLiteExApp_DecodeProcess(S_NULITEEX_APP_DECODE *psNuLiteExAppDecode)
{
INT16 *pi16OutBuf; // 环形缓冲区非激活状态,这个只有在应用层置位(按键停止、或者启动失败等情况)
if (BUF_CTRL_IS_INACTIVE(&psNuLiteExAppDecode->sOutBufCtrl))
return FALSE; // 环形缓冲区还有未读数据
if ( Playback_NeedUpdateOutputBuf(&psNuLiteExAppDecode->sOutBufCtrl) )
{
// 由核心库来判断这个文件是否解析完了
// Check end of file
if(NuLiteEx_DecodeIsEnd((UINT8*)psNuLiteExAppDecode->au32DecodeWorkBuf))
{
// Trigger inactive flag of output buffer to stop NuLiteEx decoding
BUF_CTRL_SET_INACTIVE(&psNuLiteExAppDecode->sOutBufCtrl);
// Use to represnt no active(or end) of decoding
psNuLiteExAppDecode->sOutBufCtrl.u16SampleRate = ;
return FALSE;
} // Record output data buffer pointer(for duplicate & process)
pi16OutBuf = (PINT16)&psNuLiteExAppDecode->sOutBufCtrl.pi16Buf[psNuLiteExAppDecode->sOutBufCtrl.u16BufWriteIdx]; // 核心库继续发挥其巨大作用,开足马力读取文件中PCM数据转到缓冲区。
NuLiteEx_DecodeProcess( (UINT8*)psNuLiteExAppDecode->au32DecodeWorkBuf,
psNuLiteExAppDecode->pau8TempBuf,
pi16OutBuf,
g_asAppCallBack[psNuLiteExAppDecode->u8CallbackIndex].pfnReadDataCallback,
g_asAppCallBack[psNuLiteExAppDecode->u8CallbackIndex].pfnUserEventCallback); // PlayBack依旧共享这个缓冲区,准备对数据进行进一步处理
// Update write index of output buffer and avoid buffer overflow
Playback_UpdateOutputBuf(&psNuLiteExAppDecode->sOutBufCtrl); // Duplicate data into buffer for using duplication callback function.
if ( psNuLiteExAppDecode->u8CtrlFlag&(NULITEEXAPP_CTRL_DUPLICATE_TO_BUF|NULITEEXAPP_CTRL_DUPLICATE_TO_FUNC) )
{
if ( psNuLiteExAppDecode->u8CtrlFlag & NULITEEXAPP_CTRL_DUPLICATE_TO_BUF )
BufCtrl_WriteWithCount(psNuLiteExAppDecode->psDuplicateOutBufCtrl, NULITEEXAPP_OUT_SAMPLES_PER_FRAME, pi16OutBuf );
else
psNuLiteExAppDecode->pfnDuplicateFunc(NULITEEXAPP_OUT_SAMPLES_PER_FRAME, pi16OutBuf);
}
}
return TRUE;
}
总结
源码拓扑结构
ISD9160学习笔记03_ISD9160音频解码代码分析的更多相关文章
- ISD9160学习笔记04_ISD9160音频编码代码分析
前言 录音例程涉及了录音和播放两大块内容,上篇笔记说了播放,这篇就来说说录音这块,也就是音频编码这部分功能. 上篇笔记中的这段话太装逼了,我决定再复制下,嘿嘿. “我的锤子便签中有上个月记下的一句话, ...
- memcached学习笔记——存储命令源码分析上篇
原创文章,转载请标明,谢谢. 上一篇分析过memcached的连接模型,了解memcached是如何高效处理客户端连接,这一篇分析memcached源码中的process_update_command ...
- input子系统学习笔记六 按键驱动实例分析下【转】
转自:http://blog.chinaunix.net/uid-20776117-id-3212095.html 本文接着input子系统学习笔记五 按键驱动实例分析上接续分析这个按键驱动实例! i ...
- tensorflow笔记:多层LSTM代码分析
tensorflow笔记:多层LSTM代码分析 标签(空格分隔): tensorflow笔记 tensorflow笔记系列: (一) tensorflow笔记:流程,概念和简单代码注释 (二) ten ...
- memcached学习笔记——存储命令源码分析下篇
上一篇回顾:<memcached学习笔记——存储命令源码分析上篇>通过分析memcached的存储命令源码的过程,了解了memcached如何解析文本命令和mencached的内存管理机制 ...
- ArcGIS案例学习笔记2_1_学校选址适宜性分析
ArcGIS案例学习笔记2_1_学校选址适宜性分析 计划时间:第二天上午 目的:学校选址,适宜性分析 内容:栅格数据分析 教程:pdf page=323 数据:chapter8/ex1/教育,生活,土 ...
- tensorflow笔记:多层CNN代码分析
tensorflow笔记系列: (一) tensorflow笔记:流程,概念和简单代码注释 (二) tensorflow笔记:多层CNN代码分析 (三) tensorflow笔记:多层LSTM代码分析 ...
- ISD9160学习笔记05_ISD9160语音识别代码分析
前言 语音识别是特别酷的功能,ISD9160的核心卖点就是这个语音识别,使用了Cybron VR 算法. 很好奇这颗10块钱以内的IC是如何实现人家百来块钱的方案.且听如下分析. 本文作者twowin ...
- 学习笔记:python3,代码。小例子习作(2017)
http://www.cnblogs.com/qq21270/p/7634025.html 学习笔记:python3,一些基本语句(一些基础语法的代码,被挪到这里了) 日期和时间操作 http://b ...
随机推荐
- maven学习日记(三)-------开发环境搭建(springmvc+hibernate4)各种maven错误汇总
1.maven编码 gbk 的不可映射字符 解决这个问题的思路: 在maven的编译插件中声明正确的字符集编码编码——编译使用的字符集编码与代码文件使用的字符集编码一致!! 安装系统之后,一般中文系统 ...
- 【转】【CentOS】【Python】Centos7安装Python3的方法
由于centos7原本就安装了Python2,而且这个Python2不能被删除,因为有很多系统命令,比如yum都要用到. [root@VM_105_217_centos Python-3.6.2]# ...
- selenium+java-查找页面中包含关键字的URL
package seleniumLearn1; import java.io.File; import java.io.FileOutputStream; import java.io.IOExcep ...
- Unity 各个组件参数总结
今天在蛮牛教育上学习了NGUI的课程下面给大家总结了一些小知识点希望对大家有帮助UICamera-可以添加到任何相机,包含事件系统.UICamera是每个UI的重要组成部分.它负责发送Camera中所 ...
- Dynamics CRM 2015/2016 Web API:聚合查询
各位小伙伴们,今天是博主2016年发的第一篇文章.首先祝大家新年快乐.工资Double,哈哈.今天我们来看一个比較重要的Feature--使用Web API运行FetchXML查询! 对的,各位.你们 ...
- 【转】C# 异常处理 throw和throw ex的区别 try catch finally的执行顺序(return)
[转]throw和throw ex的区别 之前,在使用异常捕获语句try...catch...throw语句时,一直没太留意几种用法的区别,前几天调试程序时无意中了解到几种使用方法是有区别的,网上一查 ...
- win7下安装双系统Ubuntu14.04后开机没有win7,直接进入Ubuntu
开机进入Ubuntu后,打开命令端,输入: sudo update-grub 然后重启,则解决问题
- Obj格式模型 读取
OBJ文件的结构 在一个OBJ文件中,首先有一些以v.vt或vn前缀开头的行指定了所有的顶点.纹理坐标.法线的坐标.然后再由一些以f开头的行指定每一个三角形所对应的顶点.纹理坐标和法线的索引.在顶点. ...
- js中的坑
for in vs hasOwnProperty == === 对象比较用===,值比较用==. 严格运算用=== http://www.zhihu.com/question/31442029 著作权 ...
- 在PC上运行安卓(Android)应用程序的几个方法
三种方法: 1.在PC安装一个安卓模拟器,在模拟器里面运行apk: 2.虚拟机安装 Android x86 然后在此系统里运行: 3.利用谷歌chrome浏览器运行(这是一个新颖.有前途.激动人心的方 ...