STM32的SPI问题。
问题描述:
之前一直使用的单片机是LPC2109,对其SPI很熟悉。基本就是原本拿来稍作修改就用。
由于某种原因需要使用STM32,然后设备的驱动是之前写好的,只修改了一些硬件控制端口,由于硬件驱动使用到了SPI接口,而我是把SPI接口提供了出来,本来以为简单修改SPI配置到对应单片机就行了。简单看了STM3的SPI配置,轻车熟路改代码,瞬间体现了良好的接口有哈。
编译,生成目标文件,下载运行。
并没有出现预想的结果。由于之前的设备驱动是能用的,所以排除设备驱动问题。
开始以为是由于对STM32端口配置的不熟悉导致的、看手册,看别人代码,没发现问题。
debug........
问题定在SPI代码上。查看配置,一样啊。郁闷!!!
把自己配置考到别人能用的代码中,可以使用。更加郁闷!!!!
debug看寄存器。对比能运行代码寄存器状态。发现运行到一段代码的时候寄存器不同
SPI_CR 0x0043
SPI_CR 0x0002
看datasheet.OVR置位。问题应该就在这了。可是为什么呢??????
搜此问题,此处出自这里
溢出错误(OVR)
溢出错误表示连续传输多个数据时,后一个数据覆盖了前一个数据而产生的错误。
状态标志SPIF表示的是数据传输正在进行中,它对数据的传输有较大的影响。主器件的SPIF有效由数据寄存器的空标志SPTE=0产生,而从器件的SPIF有效则只能由收到的第一个SCK的跳变产生,且又由于从器件的SPIF和主器件发出的SCK是异步的,因此从器件的传输标志SPIF从相对于主器件的传输标志SPIF主有一定的滞后。如图4所示,在主器件连续发送两个数据的时候将有可能导致从器件的传输标志和主器件下一个数据的传输标志相重叠(图4中虚线和阴影部分),第一个收到的数据必然被覆盖,第二个数据的收/发也必然出错,产生溢出错误
图4溢出错误
通过对从器件的波形分析发现,counter=8后的第一个时钟周期,数据最后一位的传输已经完成。在数据已经收/发完毕的情况下,counter=8状态的长短对数据的正确性没有影响,因此可以缩短counter=8的状态,以避免前一个SPIF和后一个SPIF相重叠。这样,从硬件上避免了这一阶段的溢出错误。
但是,如果从器件工作速度不够快或者软件正在处理其他事情,在SPI接口接收到的数据尚未被读取的情况下,又接收到一个新的数据,溢出错误还是会发生的。此时,SPI接口保护前一个数据不被覆盖,舍弃新收到的数据,置溢出标志OVR=1;另外发出中断信号(如果该中断允许),通知从器件及时读取数据。
23.4.7 错误标志位
I2S 单元有2个错误标志位。
下溢标志位(UDR)
在从发送模式下,如果数据传输的第一个时钟边沿到达时,新的数据仍然没有写入SPI_DR寄存
器,该标志位会被置’1’ 。在寄存器SPI_I2SCFGR的I2SMOD 位置’1’ 后,该标志位才有效。如果
寄存器SPI_CR2的ERRIE位为’1’ ,就会产生中断。
通过对寄存器SPI_SR进行读操作来清除该标志位。
上溢标志位(OVR)
如果还没有读出前一个接收到的数据时,又接收到新的数据,即产生上溢,该标志位置’1’ ,如
果寄存器SPI_CR2的ERRIE位为’1’ ,则产生中断指示发生了错误。
这时,接收缓存的内容,不会刷新为从发送设备送来的新数据。对寄存器SPI_DR的读操作返回
最后一个正确接收到的数据。其他所有在上溢发生后由发送设备发出的16位数据都会丢失。
通过先读寄存器SPI_SR再读寄存器SPI_DR,来清除该标志位。
void SPI_write_byte(u8 data)
{
S0SPDR = data;
while ((S0SPSR & 0x80) == );
} u8 SPI_read_byte(void)
{
S0SPDR = 0xff;
while((S0SPSR & 0x80) == );
return (S0SPDR);
}
整个工程修改的代码如下(注释代码为不能正常工作的):
/*---------------------------------------------------------------------------*/
// void SPI_write_byte(u8 data)
// {
// while (!(SPI1->SR & (1 << 1)));
// SPI1->DR = data;
// } // u8 SPI_read_byte(void)
// {
// while (!(SPI1->SR & 1));
// return SPI1->DR;
// } u8 spi_rw(u8 data)
{
while (!(SPI1->SR & ( << )));
SPI1->DR = data;
while (!(SPI1->SR & ));
return SPI1->DR;
}
/*---------------------------------------------------------------------------*/
// SPI_write_byte(op | (address & ADDR_MASK));
// SPI_write_byte(data);
spi_rw(op | (address & ADDR_MASK));
spi_rw(data);
/*---------------------------------------------------------------------------*/
// SPI_write_byte(RBM);
spi_rw(RBM);
// *data = SPI_read_byte();
*data = spi_rw(0xff);
/*---------------------------------------------------------------------------*/
// SPI_write_byte(WBM);
spi_rw(WBM);
// SPI_write_byte(*data);
spi_rw(*data);
/*---------------------------------------------------------------------------*/
看完基本就明白问题所在了...
分析问题:
我是按照LPC的SPI配置的,而现在的是STM32,问题关键就在于STM32的接受缓冲空和发送缓冲非空的标志是不同的。而LPC单片机是相同的。仔细分析我写的代码,实际上每次执行都缺少了对状态的判断,从而导致了数据的溢出。
解决问题:
修改代码如下,问题解决。
u8 SPI_write_byte(u8 data)
{
while (!(SPI1->SR & ( << )));
SPI1->DR = data;
while (!(SPI1->SR & ));
return SPI1->DR;
} u8 SPI_read_byte(void)
{
while (!(SPI1->SR & ( << )));
SPI1->DR = 0xff;
while (!(SPI1->SR & ));
return SPI1->DR;
}
总结:
问题出在思维的定势,先入为主的思想导致了错误的思维,也体现了对问题的分析能力,以及编码的随意性。哎血的教训啊。。。
STM32的SPI问题。的更多相关文章
- STM32 F4 SPI Accelerometer
STM32 F4 SPI Accelerometer
- oled stm32的spi
其实各种协议是很重要的,这篇文章就当做我对spi协议的一个整理吧. 必要的spi简介: https://www.cnblogs.com/zengsf/p/7221207.html?utm_source ...
- STM32的SPI口的DMA读写[原创www.cnblogs.com/helesheng]
SPI是我最常用的接口之一,连接管脚仅为4根:在常见的芯片间通信方式中,速度远优于UART.I2C等其他接口.STM32的SPI口的同步时钟最快可到PCLK的二分之一,单个字节或字的通信时间都在us以 ...
- FPGA作为从机与STM32进行SPI协议通信---Verilog实现 [转]
一.SPI协议简要介绍 SPI,是英语Serial Peripheral Interface的缩写,顾名思义就是串行外围设备接口.SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用 ...
- stm32之SPI通信协议
SPI (Serial Peripheral interface),顾名思义就是串行外围设备接口.SPI是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为P ...
- FPGA作为从机与STM32进行SPI协议通信---Verilog实现
一.SPI协议简要介绍 SPI,是英语Serial Peripheral Interface的缩写,顾名思义就是串行外围设备接口.SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用 ...
- STM32之spi管理模式
1)sip管理模式分为:硬件管理和软件管理:主要由NSS .SSI.SSM决定: NSS是芯片上一个实实在在的引脚,SSI和SSM是SPI_CR1控制器里的的位. 值得注意的是:NSS分外部引脚和内部 ...
- STM32 HAL SPI读取MPU6500的设备ID异常
1.问题背景 近前,使用STM32F4 HAL库的SPI读取MPU6500出现异常. 现象:读取ID失败,返回0,以为硬件焊接问题,各种排查,最后为了示波器测试方便,把读取ID的函数放到While(1 ...
- STM32之SPI时钟相位选择
SPI的时钟模式分为四种,由SPI_CR1寄存器的两位CPOL,CPHA组合选择. CPOL 如果为1,则时钟的空闲电平为高电平:CPOL 如果为0,则时钟的空闲电平为低电平.空闲电平影响不大. CP ...
随机推荐
- 【AngularJs】---JSONP跨域访问数据传输
大家会自然想到只有一个字母之差的JSON吧~ JSON(JavaScript Object Notation)和JSONP(JSON with Padding)虽然只有一个字母的差别,但其实他们根本不 ...
- Jfinal----Handler之责任链设计模式
Jfinal handler的处理采用了责任链设计模式 有关责任链模式,推荐看: <JAVA与模式>之责任链模式 1.实现Handler只需要继承Handler public class ...
- sql语句将本地服务器中的数据插入到外网服务器中
--将本地的数据库中的某张表中的数据导入到180的数据库中 --这个要在本地的数据库运行 exec sp_addlinkedserver 'srv_lnk', '', 'SQLOLEDB','xxx. ...
- Linux 系统中用户切换(su user与 su - user 的区别)
1. Linux系统中用户切换的命令为su,语法为: su [-fmp] [-c command] [-s shell] [--help] [--version] [-] [USER [ARG]] 参 ...
- 桌面虚拟化之部署DDC-5.6
1. 打开管理软件 2. 选择桌面部署 3. 如果没有数据库则使用默认的 4. 导入许可证文件(当然未申请可试用30天) 5. 主机类型选择无(这里未做服务器虚拟化) 6. 最后完成初步配置 配置计算 ...
- HTML+CSS学习笔记(8)- CSS选择器
标签:HTML+CSS 什么是选择器? 每一条css样式声明(定义)由两部分组成,形式如下: 选择器{ 样式; } 在{}之前的部分就是"选择器","选择器"指 ...
- (转)国内外三个不同领域巨头分享的Redis实战经验及使用场景
随着应用对高性能需求的增加,NoSQL逐渐在各大名企的系统架构中生根发芽.这里我们将为大家分享社交巨头新浪微博.传媒巨头Viacom及图片分享领域佼佼者Pinterest带来的Redis实践,首先我们 ...
- (转)hessian源码分析(一)------架构
在计费中心的对外交互这块采用了hessian,有必要对hessian的运行机理和源码做一定的解析. 大致翻了翻源码后,发现hessian的主要结构分客户端与服务端,中间基于http传输.客户端主要做的 ...
- debian终端菱形乱码修复
最简安装debian的时候由于没有中文字库,若选择看中文环境会出现菱形乱码.先把zh.utf8换为us.utf8看着好顺眼些.按空格键取消已选的zh.utf8选项按空格键选择us.utf8选项ok
- Google Ajax Library API使用方法(JQuery)
Google Ajax Library API使用方法 1.传统方式: <script src="//ajax.googleapis.com/ajax/libs/jquery/1.7. ...