你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

关于USART接收中断的BUG和注意事项(转)

[复制链接]
疯de_恒 发布时间:2017-12-25 16:46
问题描述:
2 U6 O# e. ^4 Q8 }* @7 D, P在使用USART做串口通讯时,我只把接收中断打开,并设置抢占优先级为最低一个级别,而接收中断上一个优先级处理事情比较多,可能占用了2ms时间。当我使用9600波特率往下位机发送数据,速度非常快,就是一直按回车发!问题就出来,不到1分钟时间,通讯没有反应了。USART配置代码如下:
! E6 z1 f7 f4 z% i: qvoid uart_config(void)4 R5 z# v' {1 T) J4 l  M' @
{
6 u1 {* I' k  c8 H+ U0 s; z- T    USART_InitTypeDef USART_InitStructure;
& G* s8 t2 I4 }0 S: l+ c    USART_InitStructure.USART_BaudRate = UART_GetBaud(BaudRate);
, X" s7 r; ^2 K% v8 o& V    USART_InitStructure.USART_WordLength = USART_WordLength_8b;+ E; l- P1 c; m7 g! X8 z5 [- w5 b4 T
    USART_InitStructure.USART_StopBits = USART_StopBits_1;
2 ?! Z  e0 [" }4 f; q    USART_InitStructure.USART_Parity = USART_Parity_No;
0 b' ]) h5 r5 }    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;% r" w8 t/ _/ n9 ^/ l6 e  `: M  @
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
# N# L. @! a! E9 _    USART_InitStructure.USART_Clock = USART_Clock_Disable;0 S1 S: L1 D* e% F4 l; C: Y
    USART_InitStructure.USART_CPOL = USART_CPOL_Low;% p7 B* x$ a+ i: T( _: u
    USART_InitStructure.USART_CPHA = USART_CPHA_2Edge;1 P. ?0 F6 h9 [1 f  `/ N
    USART_InitStructure.USART_LastBit = USART_LastBit_Enable;
( z( G4 ~! S9 H* q    /* Configure USART1 */
: I8 s( U4 D' m# o& D    USART_Init(USART1, &USART_InitStructure);
' }7 J9 j# \7 ^, Y, X- k    /* Enable USART1 receive interrupt */
7 r  t' [9 z6 |& ~8 ?- r' Z  B    USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);. @8 k' j( L! p
    /* Enable the USART1 */
- c6 L) }/ K! m4 H& ]( p+ s% `    USART_Cmd(USART1, ENABLE);
% I# d4 m# b$ U* s  S) |8 |}
/ a7 |* o) O- O9 p2 Z
3 b+ R9 N; w6 V7 I3 n. @5 G1 `% _分析问题:4 w3 p$ v. J4 \. T9 u, v6 O7 z% {
; U$ B1 R) `2 t, H* p2 n; J
1.为什么没有通讯了?
# L' A2 j7 W7 o* f通过仿真器仿真,发现程序一直进入接收中断中,由于我没有使用中断发送,于是程序就死在了接收中断,也就没有数据发送到电脑上来。接收中断代码如下:
1 a) ^$ k7 F  t1 ~7 rvoid UART_Receive(void)
5 H+ T, L$ {" g3 Y$ B/ e{/ J1 m8 Y% V$ Z% h+ R# M3 c
   //正在处理上一条通讯,接收到数据不处理/ @0 `& N) r2 s  P% a9 J' m2 @
    if(bRecieveOK)
5 A  N: v7 S/ S9 {+ I: c$ u7 K5 B* k9 {    {( K, w  |7 K1 n& w5 X, U7 [
        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)% ^. T2 U" H4 D' f" g
            USART_ClearITPendingBit(USART1, USART_IT_RXNE);& ^: h$ j+ t( `& z5 r6 m
        return;//processing receive data,don't receive again
3 _5 _1 s+ G: b    }7 _9 z& ~1 s1 B' W
    if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
1 u2 y& i+ d; w8 T! O! }    {5 E6 @9 g+ |4 \; y
        if(MoudBusExpir == 0)
) o1 @; v; Y! A$ b+ [! G; t) a/ }        {( Y$ {. p3 W; Y+ m
            ucRcePtr = 0;( d7 O2 }  M+ n
            MoudBusExpir = 20;//50ms1 @. f" _. ^' {' A3 F/ V
        }( {3 B2 R0 w% R4 }. D
        else
: R: l. d9 L: E8 \- p% @            MoudBusExpir = 20;' L2 N0 N& Z0 [; w5 W+ p' {
        /* Read one byte from the receive data register */
$ E7 i! P5 @; n6 z, u# d) Q9 K: @        ucRS485Buff[ucRcePtr++] = USART_ReceiveData(USART1);
$ q) H+ O, ~" l$ ?        /* Clear the USART1 Receive interrupt */
& e& R9 _9 R' Q        USART_ClearITPendingBit(USART1, USART_IT_RXNE);
3 B% ]% d: w" y) o( p7 \    }
+ o! T$ m* K( I+ e
+ F& L9 w* ]& y8 {. X+ O
$ H4 W; T' Z) @; O; c8 D/ W! v2.为什么会一直跑到接收中断?
) n+ E4 W4 L" A! {8 e# p; X& J断点之后发现(USART_GetITStatus(USART1, USART_IT_RXNE)==RESET的,也就是说没有数据接收到也进了中断,而且在USART配置中我也只打开了接收中断!没有数据送过来应该是不可能进入中断的!- L/ {! m7 a, m# i/ ^3 I

5 q  _3 M) I( d3.响应了什么中断?
- e' H, M; E( B2 Y0 m: W( J  p我想通过函数(USART_GetITStatus()把所有中断状态都读出来,但失败了,USART_IT_XXX所有中断状态都是RESET!也就是说没有中断也进入到这个中断服务程序来了!?
9 I0 M  b" ]& u- ?6 W/ n
7 F4 ]* g' a: U! \: x' N4.找资料9 S& B: h/ e; L& Z( e: Y+ f
STM32F10x微控制器参考手册(2009年12月第10版)P541发现如下说明:3 l5 G! W) f: N8 ^/ ~2 {) t- ?
- v+ F* [8 ?) k
也就是说只要接收中断打开,即RXNEIE设置为1,那么ORE中断也自动打开了。- s5 ~: r0 @0 c1 B$ W
可是USART_GetITStatus(USART1, USART_IT_ORE )== RESET!!!!
2 F& o0 O# K" k; n! \& p" T找到USART_GetITStatus(USART1, USART_IT_RXNE)函数,发现只有当USART_IT_ERR中断使能时,才能读到ORE中断。
3 Q3 F& u7 p* `1 z$ E9 T/ s5 L在这里要指出这个BUG:产生ORE中断了,但使用USART_GetITStatus()函数却无法读到这个中断被SET起来!
2 R- o- z+ B! K3 Y& ^- o; f
: D' S7 A" V. n. e8 C/ \5.把ORE中断标志位清除
+ ]( X- M0 e! y既然找到了是什么中断,那么把相应的中断标志位清除,就应该可以了吧?
8 o8 I/ [6 v# i: l  c5 `USART_ClearITPendingBit(USART1, USART_IT_ORE);
1 ^0 S1 K+ S7 u* [# l' ^但是,结果是没有任何效果!清除之后,马上读ORE中断状态,USART_GetITStatus(USART1, USART_IT_ORE)==RESET.程序仍然跑死在接收中断。再使用另一个函数USART_GetFlagStatus(USART1, USART_FLAG_ORE) = SET,原来ORE标志位还没有清除。3 p$ G# U& A; O2 A/ K

  J  s9 c$ q# r6.问题解决
" P4 B- M) q/ p! V" T% S为什么清除不掉?头疼了,再找找资料,有发现,在P523页如下图:
0 `; ?# H3 r+ \- k
( z) O# ]7 i! n/ |接收中断程序改为:
: f# E6 \% b. o" j: d- Y8 }void UART_Receive(void)
& ^0 \4 @' ^! U{
; u6 {9 F. d9 x4 p0 ^    if (USART_GetFlagStatus(USART1, USART_FLAG_ORE) != RESET)//注意!不能使用if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)来判断
; A8 k' O( D5 j' l+ C. Y& g
    {) E: z+ c+ j# ?4 `) q: \% g) [2 t
        USART_ReceiveData(USART1);' R& c1 q8 _5 c! z" y& ?) k' i
    }9 {# w& R7 k- s: ^3 \5 p  M3 Z
   //正在处理上一条通讯,接收到数据不处理& g3 _% _& `* G4 E
    if(bRecieveOK)
' y6 g9 {4 ]7 r: C- j    {
/ S) h9 y0 Q( ?, j9 p        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
+ Y: q$ o" ^3 m$ A! l6 s            USART_ClearITPendingBit(USART1, USART_IT_RXNE);
! |* Q- d8 g7 o; ~! j$ N: G% C- R        return;//processing receive data,don't receive again
( [* M6 T9 L3 J* A. S    }6 E6 f( R: d8 z; [3 B
    if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)0 k/ X& C7 \: l) k4 H( h
    {
" Z- a9 {' U# O  |: i9 Y) {        if(MoudBusExpir == 0)2 P' {$ z  c+ v/ P0 C
        {6 u9 m+ E# T7 I7 x9 G2 O
            ucRcePtr = 0;
' P& Z& d  a5 d) M" `            MoudBusExpir = 20;//50ms& \2 e9 Q. U( i5 A+ \- j
        }: E: v) n5 j) ^( ]+ R' \0 g
        else
! s% X; j3 b5 z            MoudBusExpir = 20;
: K/ ~5 a5 V0 N# t! E1 f" K- l. ^        /* Read one byte from the receive data register */: L1 ?6 X% P  ?
        ucRS485Buff[ucRcePtr++] = USART_ReceiveData(USART1);
6 |' S4 m6 \) s: R, z6 [+ P! x5 h        /* Clear the USART1 Receive interrupt */! Y, M' P0 }* P; m4 @
        USART_ClearITPendingBit(USART1, USART_IT_RXNE);
) s# A: D5 a& F" K0 A, A    }. o2 d2 x) [$ E
4 L4 }- R2 G: H1 i; P' b1 I

; G. A+ n7 }$ `' r' I  K% b总结:* }  b% W' L0 M; B
注意问题:1.USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);使能了接收中断,那么ORE中断也同时被开启了。
  P$ z7 C1 o; W! D6 _" B
               2.ORE中断只能使用USART_GetFlagStatus(USART1, USART_FLAG_ORE) 读到(没有使能USART_IT_ERR中断时)
" W/ t% R9 E* J% IBUG建议:1.在STM32库中,能不能修改USART_GetITStatus()函数对USART_IT_ORE中断的处理?也就是我只要打开了接收中断,那么有ORE中断时,我也能使用USART_GetITStatus(USART1,USART_IT_ORE)读到.3 [- {# v( Q: b: z' f3 K* g
* U+ z' l1 I) k' H
! [( A' E) E  n6 T
6 h3 K& ?! \3 d! W3 j

1 C, k5 o* e! w7 X+ W0 E" _* u转自:关于USART接收中断的BUG和注意事项
1 b! T5 R+ Z9 j$ ^4 G, Z1 }http://bbs.21ic.com/icview-160999-1-1.html?fromuid=0
, E: K9 n' R  H) w(出处: 21ic电子技术论坛)
; {9 n  l' ?2 D; h+ N, C, j( V; g# a& o2 K$ \8 k( b% D
收藏 评论0 发布时间:2017-12-25 16:46

举报

0个回答

所属标签

相似分享

关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版