在消费电子产品中,往往都会用到音频系统来播放音乐、进行通话等多媒体应用,此外,对于一些需语音提示的产品,音频部分都是不可或缺的功能。笔者此处就s3c2416的音频驱动实现作一个简单的介绍。

1. IIS音频总线

s3c2416支持IIS、PCM、AC97这三种音频接口,此处只分析IIS音频接口。IIS接口(Inter-IC Sound)在20世纪80年代首先被飞利浦公司用于消费音频,为数字音频设备之间的音频数据传输而制定的一种总线标准。IIS有以下三个主要的信号:

1) 串行时钟SCLK,也叫做位时钟(BCLK)。数字音频的每一位均需对应一个SCLK脉冲,因此位时钟频率应大于等于2采样频率采样位数。乘以2表示每个采样会产生左声道和右声道的数据。

2) 帧时钟LRCK,也叫做左右声道切换时钟(WS)。LRCK为1时表示传输的是右声道的数据,为0时表示传输的是左声道的数据,因此IIS是非常适合于立体声系统的。LRCK是一个占空比约50%的方波,这个频率是需要尽可能与采样频率一致的,不然无法体现原来的音频本质。

3) 串行数据SDATA,用二进制补码表示音频数据。在串行时钟SCLK脉冲下,数据一位一位出现在SDATA线上。对于具体的IIS主机或设备,为支持全双工(例如通话时需同时支持放音与录音),串行数据线分串行输入SDI和串行输出SDO这两根。SDI用来传输采样设备数字化后的录音数据,SDO用来传输需播放的音频数据。

有时为了IIS主控制与IIS设备能够更好的同步,还需要传输一路时钟信号MCLK,也叫做主时钟。主要用于IIS设备A/D、D/A采样时的采样时钟,一般是采样频率的256倍、384倍、512倍、768倍。在满足要求的条件下,应尽可能选用较低的主时钟。

2. WM8960音频编解码器

WM8960是欧胜微电子推出的一款低功耗、高质量的立体编码解码器。该芯片内置有麦克风接口、立体声耳机驱动器以及D类立体声扬声器驱动,24比特模数转换器(ADC)和数模转换器(DAC)。

WM8960具有三对左右声道的模拟输入,其中INPUT1专用于Mic输入,支持单端或差分的Mic信号接入。这个输入具有一个程控放大器(PGA),并且可用自动电平控制(ALC)对Mic信号进行增益放大。其它的INPUT2、3可做为Mic差分接入的同相输入或线输入。

WM8960具有一对左右声道的耳机输出,16欧负载时,输出40mW。一对D类左右声道扬声器输出,每声道8欧负载,在1W输出功率时,具有87%的效率。一路左右声道混合输出。

3. WM8960驱动编写

声音是模拟信号,cpu是不能处理模拟信号的,并且认为模拟信号也是不具有传输性的。因此音频编解码器至少具有三个主要功能部分:模数转换器(ADC)、数模转换器(DAC)、程控放大器(PGA)。ADC用来采样外部的模拟声音信号(如Mic录音),进行离散化后,转换成数字音频,通过音频总线(如IIS)传输给cpu,cpu再对数字音频进行处理,如调频、混合、存储等。DAC用来把从cpu过来的数字音频信号还原成原来的模拟声音信号,DAC转换后的离散化PCM调制信号再通过滤波器真实还原出原来的模拟声音。PGA可在各个阶段对音频信号进行可编程的增益放大,例如音量的控制(可参考WM8960_HeadphoneVolume()函数),Mic灵敏度的调节(可参考WM8960_RecorderVolume()函数)等。

WM8960在使用前必须进行初始化,即需配置音频接口IIS的参数(可参考WM8960_Init()的实现),若进行录音,需配置录音路径的上电、接通,并进行增益的设定(具体见WM8960_RecorderStart()函数的实现)。若进行放音,需配置是耳机、扬声器等的话音路径,进行增益设定(可参考WM8960_HeadphoneStart()函数的实现)。IIS是音频接口,只能传输音频信号,因此WM8960还需另外的IIC接口,通过IIC总线写寄存器对这些配置进行设定。IIC驱动编写在前面的章节有详细的介绍,此处不再细说,WM8960模块驱动WM8960.c如下:

  1. #include "IIC.h"
  2. #include "WM8960.h"
  3. #define VolumeLevel 7
  4. static int RecorderVolume;
  5. static int HeadphoneVolume;
  6. // WM8960寄存器不能通过IIC读,开辟缓存记录寄存器的变化
  7. static unsigned short WM8960_Reg[56] = {
  8. 0x0097, 0x0097, 0x0000, 0x0000,
  9. 0x0000, 0x0008, 0x0000, 0x000a,
  10. 0x01c0, 0x0000, 0x00ff, 0x00ff,
  11. 0x0000, 0x0000, 0x0000, 0x0000,
  12. 0x0000, 0x007b, 0x0100, 0x0032,
  13. 0x0000, 0x00c3, 0x00c3, 0x01c0,
  14. 0x0000, 0x0000, 0x0000, 0x0000,
  15. 0x0000, 0x0000, 0x0000, 0x0000,
  16. 0x0100, 0x0100, 0x0050, 0x0050,
  17. 0x0050, 0x0050, 0x0000, 0x0000,
  18. 0x0000, 0x0000, 0x0040, 0x0000,
  19. 0x0000, 0x0050, 0x0050, 0x0000,
  20. 0x0002, 0x0037, 0x004d, 0x0080,
  21. 0x0008, 0x0031, 0x0026, 0x00e9,
  22. };
  23. static void WM8960_WriteReg(unsigned char RegAddr, unsigned short Value)
  24. {
  25. unsigned char Data;
  26. unsigned char Addr;
  27. // WM8960只有7位的寄存器地址,外加寄存器值第8位,构成8位数据
  28. Addr = (RegAddr<<1) |((Value>>8) & 0x1);
  29. // WM8960有9位的寄存器值,最高位与寄存器地址一齐发送
  30. Data = (unsigned char)Value; // 低8位寄存器值
  31. IIC_WriteBytes(WM8960_SlaveAddr,Addr, &Data, 1);
  32. WM8960_Reg[RegAddr] = Value; // 写成功后更新寄存器的值
  33. }
  34. static unsigned short WM8960_ReadReg(unsigned char RegAddr)
  35. {
  36. return WM8960_Reg[RegAddr]; // 返换缓存的WM8960寄存器的值(9位)
  37. }
  38. unsigned char WM8960_HeadphoneVolume(unsigned char Control)
  39. {
  40. // -10db ~ 6db (0x6f ~ 0x7f)
  41. unsigned char Level;
  42. if (Control == VolumeDown) { // 耳机音量减
  43. if ((0x7f-0x6f)/VolumeLevel ==0) {
  44. HeadphoneVolume--;
  45. } else {
  46. HeadphoneVolume -=(0x7f-0x6f)/VolumeLevel;
  47. }
  48. if (HeadphoneVolume < 0) {
  49. HeadphoneVolume = 0;
  50. }
  51. } else {// 耳机音量加
  52. if ((0x7f-0x6f)/VolumeLevel ==0) {
  53. HeadphoneVolume++;
  54. } else {
  55. HeadphoneVolume +=(0x7f-0x6f)/VolumeLevel;
  56. }
  57. if (HeadphoneVolume >VolumeLevel) {
  58. HeadphoneVolume =VolumeLevel;
  59. }
  60. }
  61. if (HeadphoneVolume == 0) {
  62. Level = 0; // 静音
  63. } else {
  64. Level =((0x7f-0x6f)*HeadphoneVolume)/VolumeLevel + 0x6f;
  65. }
  66. // Headphone Volume Updata
  67. WM8960_WriteReg(0x02,(1<<8)|Level);
  68. WM8960_WriteReg(0x03,(1<<8)|Level);
  69. return ((HeadphoneVolume*100)/VolumeLevel);// 返回音量百分比
  70. }
  71. void WM8960_HeadphoneStop()
  72. {
  73. unsigned short RegValue;
  74. RegValue = WM8960_ReadReg(0x1a);
  75. RegValue =~((1<<8)|(1<<7)|(1<<6)|(1<<5));
  76. WM8960_WriteReg(0x1a, RegValue);
  77. }
  78. void WM8960_HeadphoneStart()
  79. {
  80. unsigned short RegValue;
  81. // DAC Left/Right,LOUT1/ROUT1Output Buffer Power up
  82. WM8960_WriteReg(0x1a, 0x01e0);
  83. // Left DAC Digital Volume -28db
  84. WM8960_WriteReg(0x0a, 0x01c5);
  85. // Right DAC Digital Volume -28db
  86. WM8960_WriteReg(0x0b, 0x01c5);
  87. // DAC Digital No mute
  88. WM8960_WriteReg(0x05, 0x0000);
  89. // Left DAC to Left Output Mixer
  90. WM8960_WriteReg(0x22, 0x0100);
  91. // Right DAC to Left Output Mixer
  92. WM8960_WriteReg(0x25, 0x0100);
  93. // Left/Right Output Mixer Enable
  94. RegValue = WM8960_ReadReg(0x2f);
  95. RegValue |= (1<<2) |(1<<3);
  96. WM8960_WriteReg(0x2f, RegValue);
  97. }
  98. unsigned char WM8960_RecorderVolume(unsigned char Control)
  99. {
  100. // -10db ~ 10db (0xaf ~ 0xd7)
  101. unsigned char Level;
  102. if (Control == VolumeDown) { //Mic灵敏度调低
  103. if ((0xd7-0xaf)/VolumeLevel ==0) {
  104. RecorderVolume--;
  105. } else {
  106. RecorderVolume -=(0xd7-0xaf)/VolumeLevel;
  107. }
  108. if (RecorderVolume < 0) {
  109. RecorderVolume = 0;
  110. }
  111. } else {// Mic灵敏度调高
  112. if ((0xd7-0xaf)/VolumeLevel ==0) {
  113. RecorderVolume++;
  114. } else {
  115. RecorderVolume += (0xd7-0xaf)/VolumeLevel;
  116. }
  117. if (RecorderVolume >VolumeLevel) {
  118. RecorderVolume =VolumeLevel;
  119. }
  120. }
  121. Level =((0xd7-0xaf)*RecorderVolume)/VolumeLevel + 0xaf;
  122. // Left ADC Digital Volume Control
  123. WM8960_WriteReg(0x15, (1<<8)| Level);
  124. return ((RecorderVolume*100)/VolumeLevel);// 返回音量百分比
  125. }
  126. void WM8960_RecorderStop()
  127. {
  128. unsigned short RegValue;
  129. RegValue = WM8960_ReadReg(0x19);
  130. RegValue &=~((1<<5)|(1<<3)|(1<<1));
  131. WM8960_WriteReg(0x19, RegValue);
  132. }
  133. void WM8960_RecorderStart()
  134. {
  135. unsigned short RegValue;
  136. // Mic单端接入到LINPUT1
  137. // VREF,Vmid Divider,PGA Left,ADCLeft,MICBIAS,Master Clock上电或使能
  138. WM8960_WriteReg(0x19, 0x00ea);
  139. // Left Channel Input PGA Enable
  140. RegValue = WM8960_ReadReg(0x2f);
  141. RegValue |= (1<<5);
  142. WM8960_WriteReg(0x2f, RegValue);
  143. // Connect Left Input PGA to LeftInput Boost mixer
  144. // Left Channel Input PGA BoostGain 0db
  145. WM8960_WriteReg(0x20, 0x0108);
  146. // PGA放大0db,Mic到ADC增益30db,ADC full scale level is 1.0Vrms
  147. WM8960_WriteReg(0x00, 0x13f);//Left Input PGA Volume 30db
  148. }
  149. void WM8960_Init(void)
  150. {
  151. unsigned short RegValue;
  152. WM8960_WriteReg(0xf, 0); // 复位WM8960寄存器
  153. // VMID 2x50k,VREF上电
  154. WM8960_WriteReg(0x19, 0x00c0); //WM8960_POWER1
  155. WM8960_WriteReg(0x1a, 0x0000); //WM8960_POWER2
  156. WM8960_WriteReg(0x2f, 0x0000); //WM8960_POWER3
  157. // Slow Clock Enable,left/rightdata = left ADC
  158. RegValue = WM8960_ReadReg(0x17);
  159. WM8960_WriteReg(0x17, RegValue |(1<<2) | (1<<0));
  160. // SYSCLK 使用MCLK时钟,ADC/DAC采样率SYSCLK/256/1.5,即384fs
  161. WM8960_WriteReg(0x04, 0x0048); //WM8960_CLOCK1
  162. // DAC静音,不使用去加重
  163. WM8960_WriteReg(0x05, 0x0008); //WM8960_DACCTL1
  164. // 音频接口配置为I2S Format, 16位字长,slave mode
  165. WM8960_WriteReg(0x07, 0x0002); //WM8960_IFACE1
  166. // ADCLRC/GPIO1作为Jackdetect input,Mic基准电压0.9 * AVDD
  167. WM8960_WriteReg(0x30, 0x0002);
  168. // ADCLRC/GPIO1 Pin FunctionSelect GPIO pin
  169. WM8960_WriteReg(0x09, 0x0040);
  170. // Headphone Volume Update, 0db
  171. WM8960_WriteReg(0x02, 0x0179);
  172. WM8960_WriteReg(0x03, 0x0179);
  173. // Left ADC Digital Volume Control0db
  174. WM8960_WriteReg(0x15, 0x01c3);
  175. RecorderVolume = VolumeLevel/2;
  176. HeadphoneVolume = VolumeLevel/2;
  177. }

WM8960模块驱动头文件WM8960.h如下:

  1. #ifndef __WM8960_H__
  2. #define __WM8960_H__
  3. #ifdef __cplusplus
  4. extern "C" {
  5. #endif
  6. // WM8960 7位IIC设备地址
  7. #define WM8960_SlaveAddr 0x1a
  8. #define VolumeDown 0x0
  9. #define VolumeUp 0x1
  10. void WM8960_Init(void);
  11. void WM8960_RecorderStart(void);
  12. void WM8960_RecorderStop(void);
  13. unsigned char WM8960_RecorderVolume(unsigned char Control);
  14. void WM8960_HeadphoneStart(void);
  15. void WM8960_HeadphoneStop(void);
  16. unsigned char WM8960_HeadphoneVolume(unsigned char Control);
  17. #ifdef __cplusplus
  18. }
  19. #endif
  20. #endif /*__WM8960_H__*/

4. IIS接口驱动

IIS最主要配置模式寄存器(IISMOD)和IIS时钟预分频寄存器(IISPSR)。IISPSR用来产生IIS基时钟,即用作编解码时钟(codec clock),与音频编解码器约定好这个时钟后,例如384fs,是可以得到传输的音频信号的采样频率fs的,由采样率可以得到IIS总线上左右声道切换时钟频率(与采样率一致)。在约定好采样位数后,通过采样频率fs可以得到IIS总线的位时钟(BCLK),这个时钟通常为2fs采样位数。这些信号同步之后,cpu与音频编解码器即能正确地一一对应传输音频数据。IISMOD主要用来设置采样位数、编解码时钟、位时钟、数据格式等信息。

音频是实时播放/录音的,任何cpu的IIS控制器均只有非常有限的FIFO,不能根据FIFO的状态采用查询或中断的方式来读写音频FIFO,因为cpu根本无法及时加载或保存音频数据,必然造成播放或录音的断续。例如,cpu需要从固态存储器加载音频数据或保存音频数据到固态存储器中,这都是需要较长的时间的。因此对于音频驱动,实用的只能是采用DMA传输方式,在IIS初始化中,除了IIS引脚配置外,还需配置DMA,并开启相应通道的DMA传输中断,具体实现可参考IIS_Init()函数。

对于实际的应用系统中,音频的播放/录音,均是边播放(录音),边加载(保存)音频数据的,这样只需较小的音频缓存空间,加载/保存时间也被分化在较短的时间内。为了避免播放/录音的断续,通常是要采用双缓存的,即一个缓存空间在播放/录音时,另一个缓存空间就在加载/保存音频数据,通过DMA中断可快速切换使用的缓存空间。具体可参考DMA_IRQ()中断处理的实现。

在需播放音频时,必须先配置IIS Tx的相关参数,包括音频数据的采样率,采样位数,声道数。不同的采样率,IIS接口的时钟配置不一样,不同采样位数、声道数,音频数据在FIFO中的格式并不一样,因此需要告诉接口函数这些参数,具体实现可参考IIS_TxInit()函数。

解码出的PCM音频数据需要写入播放缓存中,音频数据在缓存中的格式有特定的要求,这部分的实现可参考IIS_WriteBuffer()函数。音频缓存写好后,即可通过IIS_TxStart()函数启用IIS接口传输,DMA此后会根据FIFO的状态自动传输缓存中的音频数据到FIFO中,不断解码出的PCM音频数据需不断调用IIS_WriteBuffer()函数写入缓存中,即可实现连续播放音频。

在需录音时,必须先配置IIS Rx的相关参数,包括录音的采样率,采样位数。通常对于电话级应用,采样率可设置成8k,采样位数8位,对于cd音乐级应用,采样率可设置成44.1k、16位,这已经是天籁之音了,再高的采样频率,人耳 也无法辨别。这部分的实现可参考IIS_RxInit()函数,IIS接收通道准备好后,即可通过IIS_RxStart()函数接收录制的数字音频。

录制的PCM音频数据在缓存中对不同采样位数有不同的格式,音频数据需要从缓存中读出,通过处理后(如有损无损压缩、混频等音频处理),再保存进固态存储器中。从缓存读出录制的音频可参考IIS_ReadBuffer()函数的实现,录制时音频数据源源不断地存放进缓存中,因此需不断地调用IIS_ReadBuffer()函数读出录音数据,进行处理后保存。