问题描述:

之前一直使用的单片机是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问题。的更多相关文章

  1. STM32 F4 SPI Accelerometer

    STM32 F4 SPI Accelerometer

  2. oled stm32的spi

    其实各种协议是很重要的,这篇文章就当做我对spi协议的一个整理吧. 必要的spi简介: https://www.cnblogs.com/zengsf/p/7221207.html?utm_source ...

  3. STM32的SPI口的DMA读写[原创www.cnblogs.com/helesheng]

    SPI是我最常用的接口之一,连接管脚仅为4根:在常见的芯片间通信方式中,速度远优于UART.I2C等其他接口.STM32的SPI口的同步时钟最快可到PCLK的二分之一,单个字节或字的通信时间都在us以 ...

  4. FPGA作为从机与STM32进行SPI协议通信---Verilog实现 [转]

    一.SPI协议简要介绍 SPI,是英语Serial Peripheral Interface的缩写,顾名思义就是串行外围设备接口.SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用 ...

  5. stm32之SPI通信协议

    SPI (Serial Peripheral interface),顾名思义就是串行外围设备接口.SPI是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为P ...

  6. FPGA作为从机与STM32进行SPI协议通信---Verilog实现

    一.SPI协议简要介绍 SPI,是英语Serial Peripheral Interface的缩写,顾名思义就是串行外围设备接口.SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用 ...

  7. STM32之spi管理模式

    1)sip管理模式分为:硬件管理和软件管理:主要由NSS .SSI.SSM决定: NSS是芯片上一个实实在在的引脚,SSI和SSM是SPI_CR1控制器里的的位. 值得注意的是:NSS分外部引脚和内部 ...

  8. STM32 HAL SPI读取MPU6500的设备ID异常

    1.问题背景 近前,使用STM32F4 HAL库的SPI读取MPU6500出现异常. 现象:读取ID失败,返回0,以为硬件焊接问题,各种排查,最后为了示波器测试方便,把读取ID的函数放到While(1 ...

  9. STM32之SPI时钟相位选择

    SPI的时钟模式分为四种,由SPI_CR1寄存器的两位CPOL,CPHA组合选择. CPOL 如果为1,则时钟的空闲电平为高电平:CPOL 如果为0,则时钟的空闲电平为低电平.空闲电平影响不大. CP ...

随机推荐

  1. 【转载】Dubbo与Zookeeper、SpringMVC整合和使用(负载均衡、容错)

    http://blog.csdn.net/congcong68/article/details/41113239 互联网的发展,网站应用的规模不断扩大,常规的垂直应用架构已无法应对,分布式服务架构以及 ...

  2. DWZ与KindEditor编辑器的整合

    DWZ自带的编辑器是xheditor,可能很多人用不习惯.就像我,习惯用kindeditor了.现在就来说说如何整合dwz和kindeditor. 一.打开DWZ的中的dwz.ui.js,进行修改. ...

  3. Android之开源项目工具库篇

    本文转自:http://www.trinea.cn/android/android-open-source-projects-dev-lib/ 本文中你可以找到那些精美App中各种有特性的View,如 ...

  4. Cocos2d-JS事件处理机制

    在很多图形用户技术中,事件处理机制一般都有三个重要的角色:事件.事件源和事件处理者.事件源是事件发生的场所,通常就是各个视图或控件,事件处理者是接收事件并对其进行处理的一段程序.事件处理机制中三个角色 ...

  5. tcpServer 浅显的发一代码

    接下来发出来的一段代码也是我从网上找的一个例子,具体的来源已经找不到了,跟作者说声抱歉 ,现在公司做机票,出于性能的原因,就重写一个底层的tcp请求(不是我写的) 下面测试的是个控制台应用程序 Htt ...

  6. 20150301—ASP.NET的Repeater

    Repeater与GridView等数据列表一样,都是用来显示数据库的信息的,其中Repeater是最基本的列表形式,其用法也比较灵活. 一.Repeater的位置: 工具箱-数据-Repeater ...

  7. (转).NET Memory Profiler 使用简介

    1         简介 .Net Memory Profiler(以下简称Profiler):专门针对于.NET程序,功能最全的内存分析工具,最大的特点是具有内存动态分析(Automatic Mem ...

  8. (转)hessian源码分析(一)------架构

    在计费中心的对外交互这块采用了hessian,有必要对hessian的运行机理和源码做一定的解析. 大致翻了翻源码后,发现hessian的主要结构分客户端与服务端,中间基于http传输.客户端主要做的 ...

  9. 8款给力HTML5/CSS3应用插件 可爱的HTML5笑脸

    1.HTML5/CSS3实现笑脸动画 非常可爱 今天我们要分享一款基于纯CSS3实现的笑脸动画,我们只要在面部滑动鼠标,即可让人物的眼睛嘴巴动起来,实现微笑的效果,还挺可爱的. 在线演示 源码下载 2 ...

  10. android ListView的怪异现象

    我们已经知道,当条目没显示一次,那个类重写的最后一个函数就执行一次,但是现在,发生了怪异现象!当窗体的属性设置为包裹的时候,会重复显示多次,所以,高度,宽度都要设置为充满类型才可以