问题描述: 之前一直使用的单片机是LPC2109,对其SPI很熟悉。基本就是原本拿来稍作修改就用。, T3 ? x; j$ f3 b+ ~
由于某种原因需要使用STM32,然后设备的驱动是之前写好的,只修改了一些硬件控制端口,由于硬件驱动使用到了SPI接口,而我是把SPI接口提供了出来,本来以为简单修改SPI配置到对应单片机就行了。简单看了STM3的SPI配置,轻车熟路改代码,瞬间体现了良好的接口有哈。2 u+ F8 g8 X; S* W, B
编译,生成目标文件,下载运行。0 ^& m% F# ]! a, E8 R) H
并没有出现预想的结果。由于之前的设备驱动是能用的,所以排除设备驱动问题。
& g( H( N% J ]" m7 B开始以为是由于对STM32端口配置的不熟悉导致的、看手册,看别人代码,没发现问题。) I B9 n( T; A6 J# T
debug........
! U6 c* M4 K" w6 Q4 p8 R; U: C问题定在SPI代码上。查看配置,一样啊。郁闷!!!
/ {1 _- E5 @' H6 B/ v3 b把自己配置考到别人能用的代码中,可以使用。更加郁闷!!!!
% j. ?9 h% _# V. X9 Ddebug看寄存器。对比能运行代码寄存器状态。发现运行到一段代码的时候寄存器不同
; C r9 l7 ]2 a% C$ sSPI_CR 0x0043+ U/ J. {: `6 I' H
SPI_CR 0x0002. Y9 N( h& F9 R
看datasheet.OVR置位。问题应该就在这了。可是为什么呢?????? / T. _5 c9 ?' b( G* H: Y. f
溢出错误(OVR)
; [$ l0 }. k3 E. E& O7 M- I 溢出错误表示连续传输多个数据时,后一个数据覆盖了前一个数据而产生的错误。
: }1 E! \% O8 Q; W 状态标志SPIF表示的是数据传输正在进行中,它对数据的传输有较大的影响。主器件的SPIF有效由数据寄存器的空标志SPTE=0产生,而从器件的SPIF有效则只能由收到的第一个SCK的跳变产生,且又由于从器件的SPIF和主器件发出的SCK是异步的,因此从器件的传输标志SPIF从相对于主器件的传输标志SPIF主有一定的滞后。如图4所示,在主器件连续发送两个数据的时候将有可能导致从器件的传输标志和主器件下一个数据的传输标志相重叠(图4中虚线和阴影部分),第一个收到的数据必然被覆盖,第二个数据的收/发也必然出错,产生溢出错误 ; _" U* N' I# t' R5 f' V# g" M
' X$ T! `1 u# f5 ?) H7 @图4溢出错误
M/ ?: T* P1 B+ e, W- L: g 通过对从器件的波形分析发现,counter=8后的第一个时钟周期,数据最后一位的传输已经完成。在数据已经收/发完毕的情况下,counter=8状态的长短对数据的正确性没有影响,因此可以缩短counter=8的状态,以避免前一个SPIF和后一个SPIF相重叠。这样,从硬件上避免了这一阶段的溢出错误。' n/ ~: u' Q$ X0 B- E
但是,如果从器件工作速度不够快或者软件正在处理其他事情,在SPI接口接收到的数据尚未被读取的情况下,又接收到一个新的数据,溢出错误还是会发生的。此时,SPI接口保护前一个数据不被覆盖,舍弃新收到的数据,置溢出标志OVR=1;另外发出中断信号(如果该中断允许),通知从器件及时读取数据。
s% x1 \5 }* F) r9 u4 O9 K23.4.7 错误标志位) w B5 |! l5 m) g" W
I2S 单元有2个错误标志位。
C# r" z0 C, L! g2 d9 G: } V: }下溢标志位(UDR)' p% N9 d: t; ~" t
在从发送模式下,如果数据传输的第一个时钟边沿到达时,新的数据仍然没有写入SPI_DR寄存
0 B& s6 S' G" p4 B器,该标志位会被置’1’ 。在寄存器SPI_I2SCFGR的I2SMOD 位置’1’ 后,该标志位才有效。如果
* I& ^' o% B# C ~. R N寄存器SPI_CR2的ERRIE位为’1’ ,就会产生中断。; h. t6 {3 A7 E. q* X
通过对寄存器SPI_SR进行读操作来清除该标志位。, G: L, l- f- z3 {2 W1 B" {4 X
上溢标志位(OVR)/ h+ H" R: S* @! D, ?9 e
如果还没有读出前一个接收到的数据时,又接收到新的数据,即产生上溢,该标志位置’1’ ,如0 V+ O" q' V0 P" V$ h, k; p
果寄存器SPI_CR2的ERRIE位为’1’ ,则产生中断指示发生了错误。+ U# ], C0 j8 G8 Z
这时,接收缓存的内容,不会刷新为从发送设备送来的新数据。对寄存器SPI_DR的读操作返回
1 J8 l: Y2 n8 P: e6 f3 f5 }3 `最后一个正确接收到的数据。其他所有在上溢发生后由发送设备发出的16位数据都会丢失。& ^0 U5 T F7 }# W# m" t9 F- |+ G* O
通过先读寄存器SPI_SR再读寄存器SPI_DR,来清除该标志位。 - void SPI_write_byte(u8 data)% ~7 R, n% u+ p B; e3 I4 W5 h
- {
! g7 w# w) d0 e- J - S0SPDR = data;- i, @1 z# [3 G! Q0 ]
- while ((S0SPSR & 0x80) == 0);# J* E$ ~9 j6 j
- }
# ^0 u1 M: u0 M& L# o1 ~# Q - 1 f1 J* z0 V- E k8 l6 o- F
- u8 SPI_read_byte(void)
5 J0 F L, ?* D: }3 Y0 W - {- T8 F, l1 x7 U. g$ J4 L J4 \
- S0SPDR = 0xff;
- F0 o' `+ s" }- w, K4 l - while((S0SPSR & 0x80) == 0);% v& P: D7 y% n: a& G
- return (S0SPDR);$ f2 Z3 ]3 u" X& h6 X) X
- }
复制代码 ! b Q$ h) ~! @# K/ v
整个工程修改的代码如下(注释代码为不能正常工作的): - /*---------------------------------------------------------------------------*/
. w- H) Z2 h1 W% [ - // void SPI_write_byte(u8 data)! l+ |6 D; C8 n- c
- // {9 N- s s% k* i2 J; b) }3 y
- // while (!(SPI1->SR & (1 << 1)));/ H/ v% s3 B9 P. |1 z+ E* v
- // SPI1->DR = data;
- E, H5 F; Y! @" _; } - // }9 z! e/ z9 o% ?$ \ ?) C" c
& H; Y, G' q$ B& Z0 q9 P- // u8 SPI_read_byte(void)$ r& t* ~) E7 I) D
- // {
! A8 }# g5 q8 p, V. W - // while (!(SPI1->SR & 1));3 [) h! |, `2 e* t1 d7 }3 r
- // return SPI1->DR;* f) a& D: f2 n/ F- ]3 }
- // }
0 ?! N1 Q5 x: U
; S# b: b: Y- {6 ^* g4 p" q- u8 spi_rw(u8 data)
8 q% i4 e: l# P/ M- _5 W - {
& @' e, h0 A5 M- s6 S- v - while (!(SPI1->SR & (1 << 1)));9 v7 T2 e# w4 a4 J: @
- SPI1->DR = data;/ j# s* u7 x0 a" M
- while (!(SPI1->SR & 1));/ r0 ?# P, Y, J0 c# |' B5 `
- return SPI1->DR;! x# q7 ^0 f& E5 H6 |5 c
- }) k b2 F3 c2 {4 ~/ ]6 ?5 |
- /*---------------------------------------------------------------------------*/
' G3 c1 u; K' F; v, J6 ]$ G - // SPI_write_byte(op | (address & ADDR_MASK));
T& O" d, C& E& m% k: W- V - // SPI_write_byte(data);- l1 u) s+ a. b# }$ S3 e; W- X
- spi_rw(op | (address & ADDR_MASK));- S2 I) S9 X( K- m; D& j
- spi_rw(data);3 r: w, W( K! q. }* m1 w
- /*---------------------------------------------------------------------------*/ s* x( `; y6 o5 u+ R
- // SPI_write_byte(RBM);+ o# k+ B0 @5 J8 }
- spi_rw(RBM);0 Y* e/ Q% B E$ c& o* W
- // *data = SPI_read_byte();6 \- L8 d3 e! z$ \6 Q
- *data = spi_rw(0xff);
5 M3 ]. Z0 o# i - /*---------------------------------------------------------------------------*/( l0 D% o' K# _! `0 `
- // SPI_write_byte(WBM);
a, C" I. g4 A) C7 b- r - spi_rw(WBM);
7 X e0 L5 R4 n - // SPI_write_byte(*data);
" q" w8 z3 I( O1 k. u - spi_rw(*data);( _3 U C* G0 s: X
- /*---------------------------------------------------------------------------*/
复制代码
/ Y+ n) {* S6 q) [看完基本就明白问题所在了... . e/ j4 r" ^; E8 t; S, @" j% k
分析问题: 我是按照LPC的SPI配置的,而现在的是STM32,问题关键就在于STM32的接受缓冲空和发送缓冲非空的标志是不同的。而LPC单片机是相同的。仔细分析我写的代码,实际上每次执行都缺少了对状态的判断,从而导致了数据的溢出。
0 _4 H: h/ R8 h4 J解决问题: 修改代码如下,问题解决。 - u8 SPI_write_byte(u8 data)1 u- t& d- n. Q# S
- {
& k. G8 `) c2 b6 N. g6 {' R% o$ y F8 W' y - while (!(SPI1->SR & (1 << 1)));
5 I" R& A5 G6 ?2 W! M - SPI1->DR = data;8 L5 B/ @# `, Y# ?3 h+ P
- while (!(SPI1->SR & 1));* ~6 A; r$ W G. ?+ L! s
- return SPI1->DR;
# _9 k3 A4 n7 j) q$ @ - }; \1 A; l5 B1 X2 M2 \- I
- 0 o( a0 b T: h/ Z5 ~
- u8 SPI_read_byte(void)" n& U* N% b, ~) L
- {9 \' v. f7 q5 X9 ]( R6 E. C/ T
- while (!(SPI1->SR & (1 << 1)));
U9 }4 w/ u! n3 @, [ - SPI1->DR = 0xff;
. i. u N+ `* V5 E3 s7 L - while (!(SPI1->SR & 1));
+ p" n2 }* o& ]$ ?% ` - return SPI1->DR; d3 ~0 t, {' V* B( q1 U" E
- }
复制代码总结: 问题出在思维的定势,先入为主的思想导致了错误的思维,也体现了对问题的分析能力,以及编码的随意性。哎血的教训啊。。。
. h5 E- `9 _; B; u# j |
感谢分享!解决了我对一段驱动代码的困惑。