STM32F407作为I2S从机,位时钟和同步时钟有外部codec提供,使用DMA传输数据。
播放wav文件时的,操作顺序如下:
1.打开文件,解析文件头信息;
2.配置MCU端I2S;
3.设置并启动DMA,循环模式;
4.设置外部codec
芯片,产生位时钟和同步时钟;
注:2,4步需要根据文件进行设置。
5.在DMA传输过半中断和传输完成中断中,读取wav文件的音频数据,填充DMA缓冲区;
6.播放结束后,关闭DMA和I2S;
7.关闭codec芯片。
播放下一曲文件时,重复以上7个步骤。
遇到的问题是:播放第一个wav文件,完全正常;播放第二个或后面的文件时,出现沙沙声或者声音变大了,
用示波器观察到数据错位了。经多次观察,错位的位数不固定。
请问有高手遇到过类似的问题吗?
STM32作为I2S从机时,由于位时钟和同步时钟都是由外部COCEC芯片提供,所以可能会出现数据不对齐的情况。这时候SPI状态寄存器SR的bit8会被硬件置1,也就是说STM32是知道出现格式错误的,只是没有自动完成重新对齐。
LRCK是用于左右声道的同步的。比如:按照飞利浦标准I2S格式,LRCK下降沿后延迟1个BCLK,开始传输左声道数据;LRCK上升沿后延迟1个BCLK,开始传输右声道数据。这是不应该出现数据错位的。所以,这可以算是STM32的I2S的一个BUG吧。
问题现象,高16b和低16b错位。STM32的I2S数据帧长度只有两种,16b和32b。16b数据可以封装成16b帧或者32b帧;24b和32b数据都被封装成32b帧。当播放32b帧的时候,有可能出现高16b和低16b错位的情况,可以通过示波器观察看到。此时STM32的I2S并不能检测出帧错误,这时候SPI状态寄存器SR的bit8不会被硬件置1。这样,用户程序及不知道出了错误,无法通过重启I2S和DMA来重新同步I2S数据。
没有完全解决,问题描述都写出来了。
另外说明,我用STM32作为I2S的从机,但是作为从机时,STM32并没有MCLK输入。
切换波特率的时候,我并没有对内部MCLK进行调整,这可能是一个很重要原因。
所以,我建议尽量用STM32作为I2S作为主机,并把MCLK向外输出。
如果可能,可以用12.288M晶振作为MCU主时钟,并配置把这个时钟向外输出,连接到外部CODE。这样,所有I2S电路的MCLK使用的都是12.288M。
一周热门 更多>