I2S 总线学习:2-I2S驱动WM8978
背景
为了了解I2S总线所对应的硬件设计,下文转载了《STM32:I2S驱动WM8978》。
以加深对I2S总线的了解。
正文
最近项目中使用STM32F4驱动音频IC:WM8978。
由于STM32的I2S接口只有一个数据引脚,因此在设计引脚的时候,就需要确定是录音还是放音。
WM8978为DAC+ADC芯片,本身并不具备编解码的功能。
1)WM8978可通过I2S接口接收PCM数据,转为模拟信号输出,此为DAC过程,即放音;
2)WM8978可接收模拟信号转为数字信号,通过I2S接口传输给MCU,此为ADC过程,即录音。
3)WM8978还使用I2C接口配置其工作参数,比如音量,EQ,3D环绕等。WM8978本身可直连1W/8欧的小喇叭。(在下文中没有使用)
1.GPIO配置
我使用的是I2S3,对着硬件工程师给的原理图,再使用STM32CubeMX对照各个管脚看看是否有此映射。不得不说,新版的STM32CubeMX使用起来有些不顺。我只喜欢使用STM32CubeMX查看资源,却不喜欢这个软件的代码,架构有些不合我意。
我使用的还是传统的库,版本为V1.4.0。
GPIO_InitTypeDef GPIO_InitStructure;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA |
RCC_AHB1Periph_GPIOB |
RCC_AHB1Periph_GPIOC, ENABLE); //使能外设GPIOB,GPIOC时钟
//PB3/4/5 复用功能输出
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_4| GPIO_Pin_5;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化
//PC7复用功能输出
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
GPIO_Init(GPIOC, &GPIO_InitStructure);//初始化
//PA15复用功能输出
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15;
GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化
//这些AF...等等,注意看stm32f4xx_gpio.h的相关定义,特别是ext,否则会有问题
GPIO_PinAFConfig(GPIOB, GPIO_PinSource3, GPIO_AF_SPI3); // _CK
GPIO_PinAFConfig(GPIOB, GPIO_PinSource4, GPIO_AF_I2S3ext); // _EXT_SD GPIO_AF_I2S3ext
GPIO_PinAFConfig(GPIOB, GPIO_PinSource5, GPIO_AF_SPI3); // _SD
GPIO_PinAFConfig(GPIOC, GPIO_PinSource7, GPIO_AF_SPI3); //_MCK
GPIO_PinAFConfig(GPIOA, GPIO_PinSource15, GPIO_AF_SPI3); //_WS
关键在于AF的配置,有GPIO_AF_SPI3,GPIO_AF5_SPI3,GPIO_AF7_SPI3等等,令人模糊,还好库文件有说明。不同型号的MCU有不同的用法,要注意。
2.I2S寄存器配置
由于STM32的I2S与SPI是混在一起,有些资源共用,有些不共用,所以使用起来要注意。
void I2S3_Init(u16 I2S_Standard, u16 I2S_Mode, u16 I2S_Clock_Polarity, u16 I2S_DataFormat)
{
I2S_InitTypeDef I2S_InitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI3, ENABLE); //使能SPI2时钟
RCC_APB1PeriphResetCmd(RCC_APB1Periph_SPI3, ENABLE); //复位SPI2
RCC_APB1PeriphResetCmd(RCC_APB1Periph_SPI3, DISABLE); //结束复位
I2S_InitStructure.I2S_Mode = I2S_Mode; //IIS模式
I2S_InitStructure.I2S_Standard = I2S_Standard; //IIS标准
I2S_InitStructure.I2S_DataFormat = I2S_DataFormat; //IIS数据长度
I2S_InitStructure.I2S_MCLKOutput = I2S_MCLKOutput_Disable; //主时钟输出禁止
I2S_InitStructure.I2S_AudioFreq = I2S_AudioFreq_Default; //IIS频率设置
I2S_InitStructure.I2S_CPOL = I2S_Clock_Polarity; //空闲状态时钟电平
I2S_Init(SPI3, &I2S_InitStructure);
SPI_I2S_DMACmd(SPI3, SPI_I2S_DMAReq_Tx, ENABLE); //SPI3 TX DMA请求使能.
I2S_Cmd(SPI3, ENABLE); //SPI3 /I2S EN使能.
}
3.DMA配置
首先得查看手册上的DMA通道
你会看到I2S3_EXT_TX,不过其实并不是使用这个,而是SPI3_TX。有两个Stream可选择,我使用Stream 5.
void I2S3_TX_DMA_Init(u8 *buf0, u8 *buf1, u16 num)
{
NVIC_InitTypeDef NVIC_InitStructure;
DMA_InitTypeDef DMA_InitStructure;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE); //DMA1时钟使能
DMA_DeInit(DMA1_Stream5);
while (DMA_GetCmdStatus(DMA1_Stream5) != DISABLE) {} //等待DMA1_Stream1可配置
/* 配置 DMA Stream */
DMA_InitStructure.DMA_Channel = DMA_Channel_0; //通道0 SPI3_TX通道
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&SPI3->DR; //外设地址为:(u32)&SPI3->DR
DMA_InitStructure.DMA_Memory0BaseAddr = (u32)buf0; //DMA 存储器0地址
DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral; //存储器到外设模式
DMA_InitStructure.DMA_BufferSize = num; //数据传输量
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设非增量模式
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //存储器增量模式
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//外设数据格式
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;//存储器数据长度:16位
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; // 使用循环模式
DMA_InitStructure.DMA_Priority = DMA_Priority_High; //高优先级
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable; //不使用FIFO模式
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_1QuarterFull;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single; //外设突发单次传输
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; //存储器突发单次传输
DMA_Init(DMA1_Stream5, &DMA_InitStructure);
DMA_DoubleBufferModeConfig(DMA1_Stream5, (u32)buf1, DMA_Memory_0); //双缓冲模式配置
DMA_DoubleBufferModeCmd(DMA1_Stream5, ENABLE); //双缓冲模式开启
DMA_ITConfig(DMA1_Stream5, DMA_IT_TC, ENABLE); //开启传输完成中断
NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream5_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x00;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x01;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //使能中断
NVIC_Init(&NVIC_InitStructure);
}
//DMA1_Stream5中断服务函数
void DMA1_Stream5_IRQHandler(void)
{
if(DMA_GetITStatus(DMA1_Stream5, DMA_IT_TCIF5) == SET) ////DMA1_Stream5,传输完成标志
{
DMA_ClearITPendingBit(DMA1_Stream5, DMA_IT_TCIF5);
/*
此处加入传输完成处理
*/
}
}
5.播放音乐
当MCU与WM8978配置好后,播放音乐的过程为:
1)首先需要获取PCM数据。可以直接从WAV文件获取,也可以从MP3等文件解码得到,并根据文件信息,设置I2S的采样率。
2)将数据填充至之前所设置的DMA内存。
3)使用DMA_Cmd(DMA1_Stream5, ENABLE); 开启DMA传输,即可播放。
4)DMA传输完成,触发中断,可继续按(2)进行数据填充。
5)如果想暂停音乐,只需要暂时不主动进行数据填充。
6)使用DMA_Cmd(DMA1_Stream5, DISABLE);即可停止音乐播放。
如果最后发现无法发出声音,可使用逻辑分析仪检测相应IO口,正常会有很明显的波形。如图

如果I2S的管脚已经有完整的波形,还是没有输出声音,注意是不是买了假货!因为我就遇到过。
6.硬件
如果不想加功放,可提高SPKVDD电压(最高7V,以数据手册为准)。
数字信号与模拟信号不要混在一起,稍微隔离一下。这是电路基本常识,但是总是发现有人乱来。
在编写软件前,一定要把硬件工程师的电路图详细看一遍,很有可能会发现很多BUG,这样会最大限度避免很多无用功。
I2S 总线学习:2-I2S驱动WM8978的更多相关文章
- I2S音频总线学习
IIS音频总线学习(一)数字音频技术 一.声音的基本概念 声音是通过一定介质传播的连续的波. 图1 声波 重要指标: 振幅:音量的大小 周期:重复出现的时间间隔 频率:指信号每秒钟变化的次数 声音按频 ...
- I2S总线协议理解
I2S总线 Inter IC Sound总线又称集成电路内置音频总线. I2S对数字音频设备之间的音频数据传输而制定的一种总线标准. 采用了沿独立的导线传输时钟与数据信号的设计,通过将数据和时钟信号分 ...
- 迅为4412开发板Linux驱动教程——总线_设备_驱动注册流程详解
本文转自:http://www.topeetboard.com 视频下载地址: 驱动注册:http://pan.baidu.com/s/1i34HcDB 设备注册:http://pan.baidu.c ...
- CAN总线学习系列之— CAN总线特点介绍
CAN总线学习系列之— CAN总线特点介绍 CAN 总线作为一种工业界的流行总线广泛应于工业自动化.多种控制设备.交通工具.医疗仪器以及建筑.环境控制等各个行业中,它是是一种多主机局域网,所以这样 一 ...
- 迅为4412开发板Linux驱动教程——总线_设备_驱动注冊流程具体解释
视频下载地址: 驱动注冊:http://pan.baidu.com/s/1i34HcDB 设备注冊:http://pan.baidu.com/s/1kTlGkcR 总线_设备_驱动注冊流程具体解释 • ...
- 从需求的角度去理解Linux系列:总线、设备和驱动
笔者成为博客专家后整理以前原创的嵌入式Linux系列博文,现推出以让更多的读者受益. <从需求的角度去理解linux系列:总线.设备和驱动>是一篇有关如何学习嵌入式Linux系统的方法论文 ...
- i2c总线,设备,驱动之间的关系
------ 总线上先添加好所有具体驱动,i2c.c遍历i2c_boardinfo链表,依次建立i2c_client, 并对每一个i2c_client与所有这个线上的驱动匹配,匹配上,就调用这个驱动的 ...
- linux设备驱动归纳总结(九):1.platform总线的设备和驱动【转】
本文转载自:http://blog.chinaunix.net/uid-25014876-id-111745.html linux设备驱动归纳总结(九):1.platform总线的设备和驱动 xxxx ...
- linux设备驱动归纳总结(八):2.总线、设备和驱动的关系【转】
本文转载自:http://blog.chinaunix.net/uid-25014876-id-110295.html linux设备驱动归纳总结(八):2.总线.设备和驱动的关系 xxxxxxxxx ...
随机推荐
- 与英特尔分道扬镳,苹果的5G业务掉队了吗?
5G概念已经大热,越来越多的厂商推出相关产品,中国骄傲之华为不仅在5G通信标准制定方面参与感非常强,也先于竞争对手推出5G智能终端,连同三星/Vivo等也纷纷推出5G终端,而作为智能手机市场绝对的利润 ...
- 对FPM 模块进行参数优化!
Nginx 的 PHP 解析功能实现如果是交由 FPM 处理的,为了提高 PHP 的处理速度,可对FPM 模块进行参数跳转.FPM 优化参数:pm 使用哪种方式启动 fpm 进程,可以说 static ...
- org.springframework.data.redis.RedisConnectionFailureException
org.springframework.data.redis.RedisConnectionFailureException: Cannot get Jedis connection; nested ...
- 《程序之美系列(套装共6册)》[美]斯宾耐立思 等 (作者) epub+mobi+azw3
<架构之美>内容包括:facebook的架构如何建立在以数据为中心的应用生态系统之上.xen的创新架构对操作系统未来的影响.kde项目的社群过程如何让软件的架构从粗略的草图成为漂亮的系统. ...
- ThinkPHP6源码:从Http类的实例化看依赖注入是如何实现的
ThinkPHP 6 从原先的 App 类中分离出 Http 类,负责应用的初始化和调度等功能,而 App 类则专注于容器的管理,符合单一职责原则. 以下源码分析,我们可以从 App,Http 类的实 ...
- 基于Goolgle最新NavigationDrawer实现全屏水平平移
常见实现App 上面侧边栏菜单之前使用SlidingMenu,现在发现Goolgle原生NavigationDrawer也挺好用.但是细心的开发者们发现NavigationDrawer没有类似Slid ...
- 物联网协议CoAP协议学习
CoAP:Constrained Application Protocol协议是为物联网中资源受限的设备制定的应用层协议,即简化版的基于UDP的HTTP协议.其核心内容为资源抽象.REST式交互可扩展 ...
- notepad++一次去掉所有空行,然后加上2个空行
打开替换窗口,查找我的目标中填写: ^\r\n 替换为中不填,空着, 点击全部替换按钮. 如何给所有行添加2行空行: 打开替换窗口,查找目标中填写: \r\n 替换为中填写: \r\n\r\n\r\n ...
- gitlab两种连接方式:ssh和http配置介绍 --转自 散尽浮华
gitlab环境部署好后,创建project工程,在本地或远程下载gitlab代码,有两种方式:ssh和http 1)ssh方式:这是一种相对安全的方式 这要求将本地的公钥上传到gitlab中,如下图 ...
- virtual column make sqlserver using function index
In sqlserver, it is impossible that if we want to create an function index. Doesn`t means we can not ...