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

HAL库UART的几个常用函数讲解+中断处理过程讲解  

[复制链接]
cuyebiren 发布时间:2016-12-6 21:42
* k/ V+ l  U9 @' J& _: \. Q/ j
STM32电机培训online,大佬带你玩电机
+ f/ [& W0 C4 \7 p, ?8 q2 y【开源】做了一个STLINK/V2-UART二合一编程器) s* Z3 h$ X% m) a
0 |% U  J1 S9 \/ d  W4 Z9 Q) Y
【实战经验】UART应用异常案例分析
+ m7 j% f) h1 s( f: Y8 V/ S/ n5 w& B& ]/ a$ N4 E

2 B8 f, {8 e1 B, LHAL库是比较全面的,封装比较彻底的,也是功能比较强大的。
) c  T3 g! M! X" W2 P6 h+ h使用HAL库,我们直接调用它的API函数,不用关心它的底层操作过程。
' j/ D) v. h" _' b! l使用HAL库,省去了好多繁琐的处理过程,不再需要我们自己写如等待等过程。
! m, {! h% n8 iHAL库也包含如Ethernet、USB等高级外设的驱动。
& ?: A7 B/ K7 X3 ^5 Q+ R对于初接触它的人来说,尤其是用惯了标准库的人,总会有各种不适应和排斥感。
( E' e' I  u' T就拿UART来说,我们通过中断方式接受或发送数据。如果仿真调试的话,会发现UART有开关中断的现象,而不是中断一直开着。
  m& O" D: h, G
5 f( r# {0 O& w8 q) V下面,就讲解UART常用函数以及中断处理过程。$ H! F3 o' |! l& s" w
4 T& h4 ?. s/ f4 T
HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout);: _6 E$ O9 P6 ~& [; g8 h
串口发送,发送指定长度的数据。如果超时没发送完成,则不再发送,返回超时标志(HAL_TIMEOUT)。4 l2 b1 ?+ V8 H% H9 C5 h
HAL_StatusTypeDef HAL_UART_Receive(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout);9 k! r' ~  A4 {, M
串口接收,接收指定长度的数据。如果超时没接收完成,则不再接收数据到指定缓冲区,返回超时标志(HAL_TIMEOUT)。
; \2 e6 l$ D* m3 @7 C9 h
( Q6 m+ ?* u7 B" v9 e- R
  {7 S6 G/ S; [/ l4 K
HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
/ r" J5 _- u  Q! a+ X/ W/ n/ \2 X串口中断发送,以中断方式发送指定长度的数据。
# Q. R2 C% i% k, x大致过程是,把 发送缓冲区指针 指向 要发送的数据,设置 发送长度,发送计数器初值,然后使能串口发送中断,触发串口中断。; p4 J0 \5 w3 Y
再然后,串口中断函数处理,直到数据发送完成,而后关闭中断,不再发送数据串口发送完成回调函数
$ q/ Y. L! u2 gHAL_StatusTypeDef HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
- r3 T$ r9 ]) t* B串口中断接收,以中断方式接收指定长度数据。9 b( F! q; [4 C
大致过程是,把 接收缓冲区指针 指向 要存放接收数据的数组,设置 接收长度,接收计数器初值,然后使能串口接收中断。接收到数据时,会触发串口中断。) S+ ]$ h. J( x3 s5 c6 I
再然后,串口中断函数处理,直到接收到指定长度数据,而后关闭中断,不再触发接收中断,调用串口接收完成回调函数。$ O2 l. _- o& `( W9 A

- u+ E, Z% \" S6 |/ {8 i
; j* D+ G9 k3 ]1 g( k8 |
HAL_StatusTypeDef HAL_UART_Transmit_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
) `, Q- Y0 Z5 S3 h4 p9 x串口DMA发送,以DMA方式发送指定长度的数据。
& l% |* P3 k; o0 o! T过程是,发送缓冲区指针 指向 要发送的数据,设置 发送长度,发送计数器初值,设置 DMA传输完成中断的回调函数,使能DMA控制器中断,使能DMA控制器传输,使能UART的DMA传输请求。0 D: P) N6 g3 }
然后,UART便会发送数据,直到发送完成,触发DMA中断。8 f6 n6 y3 ]( t3 {9 B
DMA中断处理,如果 DMA模式 是 循环模式,则 直接 调用 DMA传输完成中断的回调函数。- O2 M4 a- R5 Q0 w9 n* G$ P( v
如果 DMA模式 是 正常模式,则 先 关闭DMA传输完成中断,不再触发DMA中断再 调用 DMA传输完成中断的回调函数。
; O, E9 u" W$ l/ w8 jDMA传输完成中断的回调函数处理过程,如果 DMA模式 是 循环模式,则 直接 调用 串口发送完成回调函数。5 Z, @6 N& z. ]
如果 DMA模式 是 正常模式,则 先关闭 UART的DMA传输请求, 再 使能串口传输完成中断,直到传输完成,触发中断。
& n3 a8 F6 w' s. Q串口传输完成中断处理,关闭中断,调用串口发送完成回调函数1 \4 r; c) ~! j
HAL_StatusTypeDef HAL_UART_Receive_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
. Z9 j2 }6 L$ S' p! t+ r- m0 X5 X$ |' @; C串口DMA接收,以DMA方式接收指定长度的数据。6 o5 J) x5 M! q
过程是,接收缓冲区指针 指向 要存放接收数据的数组,设置 接收长度,接收计数器初值,设置 DMA传输完成中断的回调函数,使能DMA控制器中断,使能DMA控制器传输,使能UART的DMA传输请求。
. M9 E5 b4 b" I" S6 h然后,UART接收到数据,便会通过DMA把数据存到接收缓冲区,直到接收到指定长度数据,触发DMA中断。
- x! G2 v0 `2 C: {+ dDMA中断处理,如果 DMA模式 是 循环模式,则 直接 调用 DMA传输完成中断的回调函数。
/ B9 X. P3 Y1 t如果 DMA模式 是 正常模式,则 先 关闭DMA传输完成中断,不再触发DMA中断再 调用 DMA传输完成中断的回调函数。8 y$ k6 _$ _8 P% a
DMA传输完成中断的回调函数处理过程,如果 DMA模式 是 循环模式,则 直接 调用 串口接收完成回调函数。( U, h" D0 Q2 j" A4 c" z! |1 {, n
如果 DMA模式 是 正常模式,则 先关闭 UART的DMA传输请求,调用 串口接收完成回调函数。8 N, W4 I& A- [" w1 c9 d2 x

1 r/ b0 Y* {2 X+ _( v0 i
* x. Q. t# H, S7 T8 u

3 F/ ?" d* ]5 d6 m+ d* U由于函数较多和过长的缘故,下面仅以HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)为例,分析源码:+ N- c3 V* v7 h: U$ @7 B
/**1 |: x0 q, W! u6 G% |( ~, d$ `; m
  * @brief  Sends an amount of data in non blocking mode.
+ }) {3 r+ y' o" J( B  * @param  huart: pointer to a UART_HandleTypeDef structure that contains& O! X) ?1 t/ P( L2 D( h
  *                the configuration information for the specified UART module.$ M- I; t; R  H# E; h7 U
  * @param  pData: Pointer to data buffer
) P6 V" `6 F3 z  j- R  * @param  Size: Amount of data to be sent
* i, `% {( m0 T0 Z4 f  * @retval HAL status
9 }1 z5 m6 ^" T: E) F  */                   串口中断发送,以中断方式发送指定长度的数据。* z* ?5 g$ s, V
HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)
! W' a7 @+ u! B/ ~! N{                9 ?* ]; b3 w7 t+ c! o3 ^/ L
  /* Check that a Tx process is not already ongoing */
9 q+ c2 B8 N' [. D/ [- U  if(huart->gState == HAL_UART_STATE_READY)    如果 串口空闲,则执行以下语句。
' S! r; B/ q& T9 U- C+ B0 n  {- z3 C$ V! C. W# R% h) X: G: e
    if((pData == NULL ) || (Size == 0U))      如果发送数据为空或者发送长度为0,则返回错误。
- l" y7 p7 J- R! j  x8 y    {
- e6 s( O: e4 ^" d( p! B      return HAL_ERROR;8 f& {( S. m& _% `$ x
    }, B. }) @8 e3 [7 K7 H

2 M) w' r! l" C7 p8 @    /* Process Locked */
0 m7 {% H  y/ h( a0 d/ o" h  ?  _* o    __HAL_LOCK(huart);                上锁。
9 A. N  k. a! [% ]' I6 {- u: k( ^9 T7 c8 J& ^8 u7 p
    huart->pTxBuffPtr = pData;               结构体变量 huart 的 参数设置。发送缓冲区,发送长度,发送计数器。2 p" O3 X* v" Y$ T
    huart->TxXferSize = Size;  d' U' a; l4 E! W0 I
    huart->TxXferCount = Size;
. j: S5 r6 S' k
) A0 f: G4 g7 a5 N2 N$ I# A9 F    huart->ErrorCode = HAL_UART_ERROR_NONE;
4 K. _$ K9 i' F! k7 h/ U    huart->gState = HAL_UART_STATE_BUSY_TX;           状态设为 发送繁忙。7 q- f( Q7 R1 _' o1 S
& J5 N; q' ]7 [
    /* Process Unlocked */" e1 g9 Q+ R( ~3 H6 I8 g" e1 ?7 w3 X1 q
    __HAL_UNLOCK(huart);               解锁。% P: M. M4 s) H( t5 w* Q6 \, S
8 |5 E6 v5 g9 C2 S. _) h% o8 ]
   /* Enable the UART Transmit data register empty Interrupt */. Q: [9 Z1 s1 g1 i$ c+ J
    SET_BIT(huart->Instance->CR1, USART_CR1_TXEIE);              使能UART发送数据寄存器空中断,则会触发串口中断(发送中断)。
3 Z1 q3 C& Z( M9 [% ?7 y$ y% h7 l0 ^8 \) A. _# L( G3 i0 e4 Z
    return HAL_OK;
) S7 z8 ?% H$ |9 P! M4 Y  }% Q5 W. E) ~; K! d! w
  else               如果 串口忙,则返回 忙状态。+ Y- Z1 X5 j2 c
  {1 A; Q4 m3 ?! u  p
    return HAL_BUSY;   
' L& W! y, Q* ~4 |8 k5 b4 o  }, r' t5 H/ `0 U2 |
}" u7 Y& a% Q; y" d1 \: A) J; y( l
; L8 ~0 J* E) j4 Z2 |
0 c% V4 Q% _9 f: z. J8 y- G4 U
1 收藏 25 评论39 发布时间:2016-12-6 21:42

举报

39个回答
any012 回答时间:2016-12-7 09:09:59
感觉HAL库的中断接收不是那么方便。
- ?- |0 m6 _! X) P  W7 ?3 B+ d有时候其他设备发送过来的数据比较快,用HAL库的中断接受的话,会做很多的判断处理,等到回调函数执行的时候,有可能下个数据就快来了。
sgsong 回答时间:2017-10-16 11:53:56
/**
8 E) F0 G' P+ U7 U( S' Q- |# G- t( d" Q  * @brief  Sends an amount of data in non blocking mode.
1 o+ t$ Y+ U* W% y9 e  * @param  huart: pointer to a UART_HandleTypeDef structure that contains
) o: ]. L+ M0 U3 B( Z: K7 V  *                the configuration information for the specified UART module.
4 Z) O* m1 {  d. O  * @param  pData: Pointer to data buffer8 w" r0 N8 ^( Q9 r7 n
  * @param  Size: Amount of data to be sent
8 o; e+ \0 m/ t" D# W( n( }3 Q  ~! p" Y  * @retval HAL status
- v" M9 {: b0 `' ]/ p  */                   串口中断发送,以中断方式发送指定长度的数据。9 O- X7 _2 D/ ]2 o
HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)
  D6 u! g9 ~3 e8 }2 P: q& m4 v{               
6 G; M' c& a0 \9 C# Z  /* Check that a Tx process is not already ongoing */
  F. o4 F5 V. C/ }9 G  if(huart->gState == HAL_UART_STATE_READY)    如果 串口空闲,则执行以下语句。* C8 R) D" a- |( w
  {8 p% j# ?% Q$ V& s9 \& N; D
    if((pData == NULL ) || (Size == 0U))      如果发送数据为空或者发送长度为0,则返回错误。
: x- t; y" Q9 ]+ [3 ]    {
/ r( f1 Q' r2 }0 ~- y; ~" e% S$ \      return HAL_ERROR;& ]8 F# c' N9 g: f3 N% T. r
    }
7 u* e3 C2 N0 y1 a; h9 m6 G3 p7 |$ a* u6 v
    /* Process Locked */8 _2 V  A- N" s! Y& }
    __HAL_LOCK(huart);                上锁。
+ ]  T# L7 @' D7 Z9 ^5 I' \$ `% p4 D% l8 Q/ b0 Q5 S
    huart->pTxBuffPtr = pData;               结构体变量 huart 的 参数设置。发送缓冲区,发送长度,发送计数器。- o9 q. X( o% d
    huart->TxXferSize = Size;
5 `0 @6 w9 l9 @; S" W- e    huart->TxXferCount = Size;6 g5 O0 y% k1 r* r
" H* V8 C( W' z/ E  p$ [
    huart->ErrorCode = HAL_UART_ERROR_NONE;
. Y9 p+ y- t2 i8 y% k    huart->gState = HAL_UART_STATE_BUSY_TX;           状态设为 发送繁忙。
+ h/ z! H/ ^4 o4 J' y
) _0 C4 w1 J6 {' b: y    /* Process Unlocked */% k( m8 [: S7 F$ ~0 y! j4 Z
    __HAL_UNLOCK(huart);               解锁。
4 ~1 K, h6 G' c/ i% W
. C$ G  B: N% f, P# k* L   /* Enable the UART Transmit data register empty Interrupt */, z# Q) [& _: R1 V/ A
    SET_BIT(huart->Instance->CR1, USART_CR1_TXEIE);              使能UART发送数据寄存器空中断,则会触发串口中断(发送中断)。
- v' b# ]' _+ e. _6 z) k+ Y7 A( `' M: G- T
    return HAL_OK;
9 Z3 f, D- H/ j5 z  }+ v5 W; ?. ]5 R" P! ^% g6 k5 t( {
  else               如果 串口忙,则返回 忙状态。0 d6 l$ z  C5 b/ ~! f; V
  {
( \+ I) {/ p  r; E    return HAL_BUSY;   
( S# M2 i' a5 a  }: l. w; j6 f" x, a& T8 e
}0 p$ a2 z  }. M5 }

9 h! R0 K9 Q. {: h* Y# k1 a- ~% h有个疑问,这个代码发送时,给整个串口上锁了(不是串口发送上锁),如果配置为全双工,这时中断接收数据怎么办?数据来时,串口一直处于忙的状态,数据量大时,数据会不会丢失?这种情况应该就不是全双工了。没找到同时收发数据的官方例程,都是一发一收的。
any012 回答时间:2016-12-7 09:49:14
本帖最后由 any012 于 2016-12-7 09:53 编辑
8 C. g* P7 [+ \8 A5 |/ j6 V
cuyebiren 发表于 2016-12-7 09:19+ J6 ]3 t# n  J* F) s; ]
你可以测试一下,以 115200 波特率 或 更高 长时间测试,看看会不会出现这种情况。。。或者 使用 DMA 方 ...

2 p! H5 @2 c, H$ `1 ~( Y* R1 Z我的这里要求比较特殊。
% h; K6 p+ H( W- V* _- `3 B) nstm32的spi作为从设备,中断接收。接收过来的帧信息是固定帧首固定长度,于是我再接收完成回调函数里判断帧首,然后再调用中断模式的接收函数,使设备不断处于接收状态。
  L# L3 V. ]6 t$ C3 ~结果,接收的数据不对,后来逐步排查,发现是接收完一个数据后,调用中断模式接收函数,但再次中断时,直接进入了overrun错误处理。应该是发送过来的数据太快,处理的太慢了。  I3 H/ v- d3 q+ }
主设备用的也是stm32,硬件spi,256分频。9 m; }0 p) l* h# j$ m0 B6 a2 z* ~  J
除了接收部分,同时我也要发送数据。开始想在中断接收的回调函数里,调用普通的发送函数,可这个函数的第三个参数是超时判断用的,如果我接收的数据是帧尾,回调函数里我要发送下一个发送帧的帧首,但下一帧信息什么时候来,这个不确定。而一旦超时,就自动关断SPI中断及SPI了。
5 M3 J: ^7 |6 u; H0 T( }$ P" v  i- u8 r8 s4 e% L; [! R' e
以上问题,我最后在工程里,按标准库的方式写,通过了。
* l6 P2 q' j6 H/ l2 s; ^  R
5 ?8 W" j# e: v9 T  `另一个问题是串口中断接收,是作为modbus从设备的,这个不要求接收的同时再发送数据了。但来的每帧信息,里面各数据之间的间隔很短。
  r( l( f8 m( |我之前从论坛找了个其他网友分享的标准库版modbus文件,挺好用的。改用HAL库没调成功,感觉还是接收到一个数据后,要判断处理的太多。) Q) k5 t0 ?/ I5 @4 X3 S7 D
! ~1 [2 D. N, y1 H) p
目前HAL库的教程,只找到了硬石科技的STM32F103的教程,正点原子的也有一般STM32F4的HAL教程。* b; p, ~$ ^0 i+ G
但两个教程里,貌似都没有接收中断这类的例子。
) G8 }7 N5 ?2 d5 o7 u原子的教程,sys文件夹下串口部分,貌似也是按类似标准库的方式处理的。
5 y3 y( B; u& J1 o3 U6 C8 k
lisingch 回答时间:2016-12-7 07:21:08
喜欢,赞!
andypanfan 回答时间:2016-12-7 08:38:39
绝对高手!!!!!!多谢了
高二毛 回答时间:2016-12-7 08:53:06
感谢分享,很详细
cuyebiren 回答时间:2016-12-7 08:56:21
andypanfan 发表于 2016-12-7 08:38
+ W- ^+ U; ~0 W6 U* e0 D% i/ \绝对高手!!!!!!多谢了

/ U) Z, X* ?( X- l我只是研究了下HAL库的源码,做到“知其所以然”而已啦!
斜阳 回答时间:2016-12-7 09:10:19
mARK一下
cuyebiren 回答时间:2016-12-7 09:19:18
any012 发表于 2016-12-7 09:09- S" o4 [0 W1 Q
感觉HAL库的中断接收不是那么方便。. A6 H% `! J9 @" y2 _
有时候其他设备发送过来的数据比较快,用HAL库的中断接受的话,会做很 ...

! N0 u, g) x! X' N! Q" q; b你可以测试一下,以 115200 波特率 或 更高 长时间测试,看看会不会出现这种情况。。。或者 使用 DMA 方式,应该 不会出现这种情况。。。感觉 HAL 库设计的初衷,是进行 大批量数据 吞吐。。。就是 函数参数 的 size 不是 1,而是相对比较大的值。。。
jackten 回答时间:2016-12-7 10:29:06
谢谢分享                  
zhoupxa 回答时间:2016-12-7 11:56:38
使用HAL库在某些情况下还是有局限性的,如果要求更高的话,还是寄存器编程吧
cuyebiren 回答时间:2016-12-7 13:16:02
any012 发表于 2016-12-7 09:49
: E+ T% X: X: l* P9 }我的这里要求比较特殊。. w. Z$ Y$ ~3 O# l# |5 Y: j
stm32的spi作为从设备,中断接收。接收过来的帧信息是固定帧首固定长度,于是我再 ...
) i7 |8 @' [1 j0 b4 n/ J1 G1 z
超时 值 可大可小。超时判断也是可以取消的,好像就是把 超时值 设为 0XFFFFFFFF 。
, f" Z) r0 q2 R( }5 ^" m$ I6 r- g中断处理函数 中 的 各种判断 是很快的,就是 几个 if 语句而已。除非是 真的 出现 了 传输错误 或 溢出 错误。
" Y6 r# U1 P- n中断回调函数 越短越好,切记 不要在 中断函数中 加延时。# L# S1 b8 ^) @( X; _8 o
具体 怎么实现各种功能,建议参考官方例程。
cuyebiren 回答时间:2016-12-7 13:25:40
zhoupxa 发表于 2016-12-7 11:56( g% f0 X" |3 h' e, d$ q- C
使用HAL库在某些情况下还是有局限性的,如果要求更高的话,还是寄存器编程吧 ...

8 f5 m; ?! r+ g0 l* UHAL库的各种判断,以及中断开关做的是比较好的,处理过程也是比较完整的。标准库仅是封装了操作寄存器的函数。你做低功耗,或者Flash比较小,代码又比较多时,寄存器编程更好。但通常情况下,这种情况的概率很小。。。用库可以把我们从编写大量寄存器中解放出来。。。
zbber 回答时间:2016-12-7 16:41:40
好帖,赞一个,顶楼主
! V1 d; \0 O9 n8 f  S
*numb 回答时间:2017-3-14 14:48:52
DMA的中断回调函数在哪,找也找不到
123下一页

所属标签

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版