一文搞懂如何通过SPI+PWM模拟I2S

前言

        I2S是一种数字音频接口标准,全称为Inter-IC Sound。它是一种串行接口,主要用于音频传输。但是由于有一些MCU可能没有I2S外设,然后你还需要用I2S进行语音播放,这个时候你就要用到本文的内容了。


一、什么是I2S?

        I2S是一种数字音频接口标准,全称为Inter-IC Sound。它是一种串行接口,主要用于连接数字音频处理器(例如MCU等)与数字音频器件(例如音频解码器、放大器等)之间的数据传输。I2S接口支持高质量音频传输,采用标准化的数据格式,可以在不同的音频设备之间实现互操作性。I2S接口在多种音频应用中广泛应用,例如智能音频系统、数码音乐播放器、智能家居音箱等。

        I2S一般有四种不同音频标准,包括 I2S Philips 标准、MSB 和 LSB 对齐标准,以及 PCM 标准。下面开始简单介绍下这四种协议:

(1)I2S Philips 标准

        I2S Philips 标准使用 WS 信号来指示当前正在发送的数据所属的通道。该信号从当前通道数据的第一个位(MSB) 之前的一个时钟开始有效,发送方在时钟信号 (CK) 的下降沿改变数据,接收方在上升沿读取数据。 WS 信号也在 CK 的下降沿变化。值得注意的是下图红框中,真正的数据是WS拉低后延迟一个时钟才开始发送的。

(2)MSB 对齐标准

        MSB 对齐标准同时生成 WS 信号和第一个数据位(即 MSBit)

(3)LSB 对齐标准

        LSB 对齐标准与 MSB 对齐标准类似(对于 16 位和 32 位全精度帧格式,没有任何不同)

(4)PCM 标准

        对于 PCM 标准,无需使用通道信息。有两种 PCM 模式(短帧和长帧),对于长帧同步,在主模式下会将 WS 信号持续 13 个周期。对于短帧同步, WS 同步信号的持续时间仅为一个周期。

        看完了上面的四种协议,对于SPI比较熟悉的人估计已经知道了,I2S的MSB和LSB协议和SPI非常相像,甚至STM32的部分MCU把SPI和I2S做成了一个外设,通过寄存器来选择功能,接下来讲解如何通过SPI+PWM的方式模拟I2S。


二、通过SPI+PWM模拟I2S

        首先我们再来看下I2S MSB协议,可以看到协议有三个波形,分别为CLK、WS、DATA,数据的发送方要在时钟信号 (CLK) 的下降沿改变数据,接收方在上升沿读取数据。 WS 信号也在 CLK 的下降沿变化。

        下图是SPI的MISO和CLK的波形,发送方的数据也是在CLK下降沿改变,接受方在数据的上升沿读取数据,这种方式和I2S的CLK和DATA的信号一模一样,因此我们就有了I2S的CLK引脚和DATA引脚的波形。

        那么还差一个WS的信号,这个WS的频率是根据CLK的频率计算得来的,也就是

                                            Freq(WS) = Freq(CLK) / 32(I2S的位数X2)

        所以,只要WS的信号和CLK的信号能够一起产生,且WS的频率是CLK频率的三十二分之一(假定I2S位数为16,而一个WS周期包含了左右声道两个16bit数据),这样就可以了。

        因此,我们需要两个条件,一个是CLK和WS的信号一起产生且CLK和DATA的信号也要一起产生;而SPI主机模式下,CLK和DATA在SPI使能之后的一段时间内才能有信号,这个信号的延时是不定的,这个跟MCU的外设设计以及编译器的优化程度有关。因为我们没有办法让CLK和WS一起产生。

        那么如何让CLK、WS、DATA一起产生呢?那就是使用SPI的从机发送模式,SPI的从机发送模式会在收到CLK之后发送自己的数据,且是同步产生的。

        有的MCU在SPI从机模式下要求设置必须CS引脚,如果是这样的话,那就随便定义一个CS引脚,然后把CS引脚拉低就好了

        那么,我们就只需要在MCU的外部产生两个PWM,一个作为I2S的CLK信号且连接到SPI的SPI_CLK上,另一个作为I2S的WS信号,然后SPI的SPI_DATA作为I2S_DATA信号进行输出。且让两个PWM同时输出,这样就实现了CLK、WS、DATA同时产生的目的。如下图:


三、实机测试

接下来我在亮牛半导体的LN882H WIFI芯片上测试下SPI+PWM模拟I2S。

首先定义I2S的引脚:

	/* I2S引脚接法:	CK 		-> 	A8和B5WS 		-> 	A2DATA 	-> 	B8*/hal_gpio_pin_afio_select(GPIOB_BASE,GPIO_PIN_5,SPI0_CLK);           //SPI的CLKhal_gpio_pin_afio_select(GPIOB_BASE,GPIO_PIN_8,SPI0_MISO);          //SPI的MISOhal_gpio_pin_afio_select(GPIOA_BASE,GPIO_PIN_2,ADV_TIMER_PWM0);     //PWM通道0hal_gpio_pin_afio_select(GPIOA_BASE,GPIO_PIN_8,ADV_TIMER_PWM2);     //PWM通道2hal_gpio_pin_afio_en(GPIOB_BASE,GPIO_PIN_5,HAL_ENABLE);hal_gpio_pin_afio_en(GPIOB_BASE,GPIO_PIN_8,HAL_ENABLE);hal_gpio_pin_afio_en(GPIOA_BASE,GPIO_PIN_2,HAL_ENABLE);hal_gpio_pin_afio_en(GPIOA_BASE,GPIO_PIN_8,HAL_ENABLE);

然后初始化SPI从机模式:

    /* 2. 配置SPI */spi_init_type_def spi_init;memset(&spi_init,0,sizeof(spi_init));spi_init.spi_baud_rate_prescaler = SPI_BAUDRATEPRESCALER_2;       //设置波特率(从机此操作无影响)spi_init.spi_mode      = SPI_MODE_SLAVE;                          //设置从模式spi_init.spi_data_size = SPI_DATASIZE_16B;                        //设置数据16bit宽度spi_init.spi_first_bit = SPI_FIRST_BIT_MSB;                       //设置帧格式为MSBspi_init.spi_cpol      = SPI_CPOL_LOW;                            //设置时钟极性0spi_init.spi_cpha      = SPI_CPHA_1EDGE;                          //设置时钟相位0hal_spi_init(SPI0_BASE,&spi_init);                                //初始化SPIhal_spi_en(SPI0_BASE,HAL_DISABLE);                                //SPI使能hal_spi_ssoe_en(SPI0_BASE,HAL_DISABLE);                           //关闭CS OUTPUT/* 初始化SPI的发送DMA */dma_init_t_def dma_init;memset(&dma_init,0,sizeof(dma_init));dma_init.dma_mem_addr = (uint32_t)song_temp;    //设置DMA内存地址(设置音频数据地址)dma_init.dma_data_num = 0;                      //设置DMA传输次数dma_init.dma_dir = DMA_READ_FORM_MEM;           //设置DMA方向dma_init.dma_mem_inc_en = DMA_MEM_INC_EN;       //设置DMA内存是否增长dma_init.dma_p_addr = SPI0_DATA_REG;            //设置DMA外设地址dma_init.dma_mem_size = DMA_MEM_SIZE_16_BIT;dma_init.dma_p_size   = DMA_P_SIZE_16_BIT;hal_dma_init(DMA_CH_4,&dma_init);               //初始化DMAhal_dma_en(DMA_CH_4,HAL_DISABLE);               //DMA使能

再设置两个独立通道的PWM,PWM0为8K,PWM1为8*16*2K,由于LN882H APB总线为40M,因此无法输出很准的8*16*2K,只能输出一个近似值256410Hz(目标是256000Hz),为了让WS和CLK匹配,只能让实际输出的CLK除以32,这就是WS的实际频率了。(这样做的结果就是实际8K采样的音频在播放的时候可能会有一点差异)。

    adv_tim_init_t_def adv_tim_init;memset(&adv_tim_init,0,sizeof(adv_tim_init));adv_tim_init.adv_tim_clk_div     = 0;                           //设置时钟分频,0为不分频adv_tim_init.adv_tim_load_value  = 4992 - 2;                    //设置PWM频率为8Kadv_tim_init.adv_tim_cmp_a_value = 4992/2 ;                     //设置通道a比较值,占空比为 50%adv_tim_init.adv_tim_cmp_b_value = 4992/2 ;                     //设置通道b比较值,占空比为 50%adv_tim_init.adv_tim_cnt_mode    = ADV_TIMER_CNT_MODE_INC;      //向上计数模式adv_tim_init.adv_tim_cha_en      = ADV_TIMER_CHA_EN;            //使能通道ahal_adv_tim_init(ADV_TIMER_0_BASE,&adv_tim_init);               //初始化TIMER0hal_adv_tim_a_en(ADV_TIMER_0_BASE,HAL_DISABLE);                 //先失能TIMER0memset(&adv_tim_init,0,sizeof(adv_tim_init));adv_tim_init.adv_tim_clk_div     = 0;                           //设置时钟分频,0为不分频adv_tim_init.adv_tim_load_value  = 156 - 2;                     //设置PWM频率为8*16*2Kadv_tim_init.adv_tim_cmp_a_value = 156/2 ;                      //设置通道a比较值,占空比为 50%adv_tim_init.adv_tim_cmp_b_value = 156/2 ;                      //设置通道b比较值,占空比为 50%adv_tim_init.adv_tim_cnt_mode    = ADV_TIMER_CNT_MODE_INC;      //向上计数模式adv_tim_init.adv_tim_cha_en      = ADV_TIMER_CHA_EN;            //使能通道ahal_adv_tim_init(ADV_TIMER_1_BASE,&adv_tim_init);               //初始化ADV_TIMER1hal_adv_tim_a_en(ADV_TIMER_1_BASE,HAL_DISABLE);                 //先失能TIMER1

使能DMA,开始播放音频数据

//关闭DMA,重新配置音频传输数量
hal_spi_en(SPI0_BASE,HAL_DISABLE); 
hal_adv_tim_a_en(ADV_TIMER_1_BASE,HAL_DISABLE);
hal_adv_tim_a_en(ADV_TIMER_0_BASE,HAL_DISABLE);
hal_dma_en(DMA_CH_4,HAL_DISABLE);
hal_dma_set_data_num(DMA_CH_4,30000);//开启DMA,开始播放音频
hal_dma_en(DMA_CH_4,HAL_ENABLE);
hal_adv_tim_a_en(ADV_TIMER_1_BASE,HAL_ENABLE);
hal_adv_tim_a_en(ADV_TIMER_0_BASE,HAL_ENABLE);
hal_spi_en(SPI0_BASE,HAL_ENABLE); //等待播放完成,然后关闭DMA,并延时
while(hal_dma_get_data_num(DMA_CH_4) != 0);
hal_adv_tim_a_en(ADV_TIMER_0_BASE,HAL_DISABLE);
hal_adv_tim_a_en(ADV_TIMER_1_BASE,HAL_DISABLE);
OS_MsDelay(1000); 

结果:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/264928.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【第三届】:“玄铁杯”RISC-V应用创新大赛(基于yolov5和OpenCv算法 — 智能警戒哨兵)

文章目录 前言 一、智能警戒哨兵是什么? 二、方案流程图 三、硬件方案 四、软件方案 五、演示视频链接 总结 前言 最近参加了第三届“玄铁杯”RISC-V应用创新大赛,我的创意题目是基于 yolov5和OpenCv算法 — 智能警戒哨兵 先介绍一下比赛&#xf…

企业欠税信息API:实现税务管理的智能化与高效化

前言 随着经济的发展和社会的进步,企业欠税问题逐渐凸显,成为制约经济发展的重要因素。为了解决这一问题,企业欠税信息API应运而生。它通过先进的技术手段,提供了一种全新的欠税信息查询方式,帮助企业实现税务管理的智…

【EI会议征稿】2024年遥感、测绘与图像处理国际学术会议(RSMIP2024)

2024年遥感、测绘与图像处理国际学术会议(RSMIP2024) 2024 International Conference on Remote Sensing, Mapping and Image Processing 2024年遥感、测绘与图像处理国际学术会议(RSMIP2024)将于2024年1月19日-21日在中国厦门举行。会议主要围绕遥感、测绘与图像处理等研究领…

如何实现远程公共网络下访问Windows Node.js服务端

文章目录 前言1.安装Node.js环境2.创建node.js服务3. 访问node.js 服务4.内网穿透4.1 安装配置cpolar内网穿透4.2 创建隧道映射本地端口 5.固定公网地址 前言 Node.js 是能够在服务器端运行 JavaScript 的开放源代码、跨平台运行环境。Node.js 由 OpenJS Foundation&#xff0…

Jol-分析Java对象的内存布局

Jol-分析Java对象的内存布局 Open JDK提供的JOL(Java Object Layout)工具为我们方便分析、了解一个Java对象在内存当中的具体布局情况。本文实验环境为64位HotSpot虚拟机。 Java对象的内存布局 Java的实例对象、数组对象在内存中的组成包括:对象头、实例数据和内存…

windows 10多用户同时远程登陆配置【笔记】

系统环境&多用户访问情况: 1、【win】【R】键入【gpedit.msc】 2、依次选择【计算机配置】→ 【管理模板】 → 【Windows组件】 → 【远程桌面服务】 → 【远程桌面会话主机】 →【连接】 2.1、右键 【允许用户通过使用远程桌面服务进行远程连接】 编辑 …

Hiera实战:使用Hiera实现图像分类任务(二)

文章目录 训练部分导入项目使用的库设置随机因子设置全局参数图像预处理与增强读取数据设置Loss设置模型设置优化器和学习率调整策略设置混合精度,DP多卡,EMA定义训练和验证函数训练函数验证函数调用训练和验证方法 运行以及结果查看测试完整的代码 在上…

Windows安装Maven

一、Maven 是什么? Maven 是一个项目管理和整合工具。Maven 为开发者提供了一套完整的构建生命周期框架。开发团队几乎不用花多少时间就能够自动完成工程的基础构建配置,因为 Maven 使用了一个标准的目录结构和一个默认的构建生命周期。 在有多个开发团…

三. LiDAR和Camera融合的BEV感知算法-BEV-SAN

目录 前言0. 简述1. 算法动机&开创性思路2. 主体结构3. 损失函数4. 性能对比总结下载链接参考 前言 自动驾驶之心推出的《国内首个BVE感知全栈系列学习教程》,链接。记录下个人学习笔记,仅供自己参考。 本次课程我们来学习下课程第三章——LiDAR和Ca…

【教程】逻辑回归怎么做多分类

目录 一、逻辑回归模型介绍 1.1 逻辑回归模型简介 1.2 逻辑回归二分类模型 1.3 逻辑回归多分类模型 二、如何实现逻辑回归二分类 2.1 逻辑回归二分类例子 2.2 逻辑回归二分类实现代码 三、如何实现一个逻辑回归多分类 3.1 逻辑回归多分类问题 3.1 逻辑回归多分类的代…

IT领域的鄙视链现象分析

1 前言 在当今快节奏的科技领域,IT行业内部不可避免地存在着一种微妙而又显而易见的“鄙视链”。这种链条似乎在技能、编程语言、框架和工具的选择上形成了一种看似无休止的等级制度,而每个人都试图站在这个链条的顶端。 在这个看似平等开放的行业中&a…

Python:核心知识点整理大全12-笔记

目录 6.3.3 按顺序遍历字典中的所有键 6.3.4 遍历字典中的所有值 6.4 嵌套 6.4.1 字典列表 aliens.py 6.4.2 在字典中存储列表 pizza.py favorite_languages.py 注意 往期快速传送门👆(在文章最后): 6.3.3 按顺序遍历字…