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

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

[复制链接]
cuyebiren 发布时间:2016-12-6 21:42
" W0 w- k- Y5 ?' ?. R7 U- V' c3 a
STM32电机培训online,大佬带你玩电机
8 ?8 k! `! @- \【开源】做了一个STLINK/V2-UART二合一编程器1 i- P- b# M7 A+ u, J

; i0 l: ]2 |1 c6 Y5 a【实战经验】UART应用异常案例分析
6 z& x' X+ e3 k% M9 C8 X5 c$ u# U; p

2 s4 d& ^  o8 b/ M0 N9 T  H/ RHAL库是比较全面的,封装比较彻底的,也是功能比较强大的。: v$ }. l% E8 b
使用HAL库,我们直接调用它的API函数,不用关心它的底层操作过程。
% V' h, q" i& X7 s3 A+ A使用HAL库,省去了好多繁琐的处理过程,不再需要我们自己写如等待等过程。
0 L7 t4 m2 O% D  zHAL库也包含如Ethernet、USB等高级外设的驱动。$ O0 x% z! |9 D5 H* \+ n+ V
对于初接触它的人来说,尤其是用惯了标准库的人,总会有各种不适应和排斥感。
' s# i0 j0 _9 H8 s就拿UART来说,我们通过中断方式接受或发送数据。如果仿真调试的话,会发现UART有开关中断的现象,而不是中断一直开着。
) x0 @  R3 ?, B' m% F
3 N( F  {1 X7 L) Y! N下面,就讲解UART常用函数以及中断处理过程。
; b1 T5 _0 F. |% J8 \
# V' E* \5 ?2 R' G& V4 f; PHAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout);
. h9 ^- ~% W  ~* }串口发送,发送指定长度的数据。如果超时没发送完成,则不再发送,返回超时标志(HAL_TIMEOUT)。
: e" e$ @) x& B& |  S3 b" P4 xHAL_StatusTypeDef HAL_UART_Receive(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout);
$ B7 i! L. I+ u串口接收,接收指定长度的数据。如果超时没接收完成,则不再接收数据到指定缓冲区,返回超时标志(HAL_TIMEOUT)。1 [0 F: R& W# X8 T7 _

' r4 c- ]& K  p; K# I
  p3 a: G4 c; \( ]* D2 S  q
HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);5 X# ]& d. `* C( @! q$ x
串口中断发送,以中断方式发送指定长度的数据。- w8 R( B" d2 Y; Z( Q7 J
大致过程是,把 发送缓冲区指针 指向 要发送的数据,设置 发送长度,发送计数器初值,然后使能串口发送中断,触发串口中断。
2 w+ X1 F& w+ M1 z, W+ E9 ^- M再然后,串口中断函数处理,直到数据发送完成,而后关闭中断,不再发送数据串口发送完成回调函数* ^& C8 i* m* }) b
HAL_StatusTypeDef HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
8 b- t. T3 s6 @& `7 {串口中断接收,以中断方式接收指定长度数据。5 |! z' K/ V' L  \0 L6 @; R
大致过程是,把 接收缓冲区指针 指向 要存放接收数据的数组,设置 接收长度,接收计数器初值,然后使能串口接收中断。接收到数据时,会触发串口中断。
( F+ R3 b) J; T% T再然后,串口中断函数处理,直到接收到指定长度数据,而后关闭中断,不再触发接收中断,调用串口接收完成回调函数。4 J  O4 g- _# i, g- Y

& Y9 ^$ g( O# Y2 r

2 K1 e- ?3 l( SHAL_StatusTypeDef HAL_UART_Transmit_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);0 m8 S: q  b* t+ |- Q
串口DMA发送,以DMA方式发送指定长度的数据。- u1 t! E' m6 ~" }6 S6 t' i; [
过程是,发送缓冲区指针 指向 要发送的数据,设置 发送长度,发送计数器初值,设置 DMA传输完成中断的回调函数,使能DMA控制器中断,使能DMA控制器传输,使能UART的DMA传输请求。
$ S  g# Y! i8 X, x4 {, R2 O然后,UART便会发送数据,直到发送完成,触发DMA中断。
. Z6 \. t5 L2 h) s  XDMA中断处理,如果 DMA模式 是 循环模式,则 直接 调用 DMA传输完成中断的回调函数。
. J" T' O, h2 @$ M2 n# p如果 DMA模式 是 正常模式,则 先 关闭DMA传输完成中断,不再触发DMA中断再 调用 DMA传输完成中断的回调函数。4 Z* _3 A. Y( ]% x% Q
DMA传输完成中断的回调函数处理过程,如果 DMA模式 是 循环模式,则 直接 调用 串口发送完成回调函数。2 |/ ^6 ?& R% m% F  k! N
如果 DMA模式 是 正常模式,则 先关闭 UART的DMA传输请求, 再 使能串口传输完成中断,直到传输完成,触发中断。0 J; S2 a) N1 t! G# T9 l. u  `
串口传输完成中断处理,关闭中断,调用串口发送完成回调函数
5 p1 g+ i8 A5 i7 x# u* M% i4 lHAL_StatusTypeDef HAL_UART_Receive_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);  f/ z9 k* l2 J6 Z2 [* s$ }8 I
串口DMA接收,以DMA方式接收指定长度的数据。
" h$ W/ }. q; a8 E, T1 v过程是,接收缓冲区指针 指向 要存放接收数据的数组,设置 接收长度,接收计数器初值,设置 DMA传输完成中断的回调函数,使能DMA控制器中断,使能DMA控制器传输,使能UART的DMA传输请求。
/ x  Z, D  U5 m1 V# B- Q' h, Y然后,UART接收到数据,便会通过DMA把数据存到接收缓冲区,直到接收到指定长度数据,触发DMA中断。. v# n; K% X6 p9 c8 z
DMA中断处理,如果 DMA模式 是 循环模式,则 直接 调用 DMA传输完成中断的回调函数。
- |6 I, \7 X6 L- T4 \7 i# h如果 DMA模式 是 正常模式,则 先 关闭DMA传输完成中断,不再触发DMA中断再 调用 DMA传输完成中断的回调函数。
3 `+ l( v+ I: _2 y# h- mDMA传输完成中断的回调函数处理过程,如果 DMA模式 是 循环模式,则 直接 调用 串口接收完成回调函数。
- E3 P. P' L! R# D' U4 {, m如果 DMA模式 是 正常模式,则 先关闭 UART的DMA传输请求,调用 串口接收完成回调函数。" m" y" w- c5 a+ d. p- q4 b* l! ]9 o

6 D8 S4 j2 G' A8 E# @# l0 v
7 W+ m- [% S! T  n

3 G5 g( K6 L, T( p! D: P由于函数较多和过长的缘故,下面仅以HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)为例,分析源码:
# R7 V* l7 a$ P, V- j. [/ G- ~/**
: e; q) Q# s' A% V  * @brief  Sends an amount of data in non blocking mode.
- \1 F- R6 h- `* l4 z2 e; D  * @param  huart: pointer to a UART_HandleTypeDef structure that contains# ?) o: u* D4 u$ h7 T
  *                the configuration information for the specified UART module.
; L2 m- _$ E- k" h# u  * @param  pData: Pointer to data buffer8 s6 t: N# [+ Y1 e8 c' c; n
  * @param  Size: Amount of data to be sent
( p! n7 g( a7 {! g3 S; c- Q  * @retval HAL status
  v! s% \; U9 m8 f; q9 `  */                   串口中断发送,以中断方式发送指定长度的数据。6 L$ ^) Y: N; X7 N2 y3 I
HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)
# x( G+ R/ X3 e: [) Y' {, [{               
8 I: ]8 B) R. j" [" _, i1 R  /* Check that a Tx process is not already ongoing */
# {" ?! Q: z: T9 z) E  if(huart->gState == HAL_UART_STATE_READY)    如果 串口空闲,则执行以下语句。
' {" X( d) Y! C  {% P9 w1 q1 s: ~
    if((pData == NULL ) || (Size == 0U))      如果发送数据为空或者发送长度为0,则返回错误。. x" F! Q8 w+ ]* G# }
    {( p0 [! o$ ]1 [0 A
      return HAL_ERROR;
4 t+ K8 B$ t; g  }! {! U$ O- b    }
( L, x8 h6 v% G$ Y5 D
8 Y1 ^2 b, r* E+ s' g    /* Process Locked */
+ a( }. f& j& z7 r4 L5 }: m7 B    __HAL_LOCK(huart);                上锁。
; `/ z7 C, d* M7 C7 ]2 y
2 ]( f! ]# T. u) f" |: F4 X% _    huart->pTxBuffPtr = pData;               结构体变量 huart 的 参数设置。发送缓冲区,发送长度,发送计数器。* w, {& ]* O0 C* M7 n- f% p
    huart->TxXferSize = Size;
! K  l0 N5 F/ l( U+ p    huart->TxXferCount = Size;& _$ A& a' S  S+ J" o# ?: a
1 |0 m4 J7 C: E
    huart->ErrorCode = HAL_UART_ERROR_NONE;, y- w; w' s/ k' W5 G
    huart->gState = HAL_UART_STATE_BUSY_TX;           状态设为 发送繁忙。' J* `) @8 u: I  ?

6 m; b0 {6 P; I' X5 v2 b( w8 G    /* Process Unlocked */
3 g2 z3 ^) t" Y" B" v    __HAL_UNLOCK(huart);               解锁。8 B% ^# _8 u1 q- s

, b4 p. p2 U! H) `  E. ?6 y* K+ X    /* Enable the UART Transmit data register empty Interrupt */
$ E0 Q6 S- h  s0 S" X" h    SET_BIT(huart->Instance->CR1, USART_CR1_TXEIE);              使能UART发送数据寄存器空中断,则会触发串口中断(发送中断)。% n( M3 s  U. @  s$ H) h/ T3 K
6 G" _& b( ^: z2 X0 g1 u! U5 d
    return HAL_OK;; t/ {% Y) Z6 }+ w4 ?
  }
; r4 u- h- O( D5 w* w& o  else               如果 串口忙,则返回 忙状态。
& g+ O# t4 H+ O+ Y: g  {. I0 X2 W0 ^2 y. V
    return HAL_BUSY;   
, K; V1 N( ^' T, g5 _) C1 [2 f  }
! p+ [, T. @1 w}
# V+ e  n' D% E' ?' i: C
. {8 m" w3 R! P7 U/ v3 `
7 f, p! C9 O) U3 P/ a
1 收藏 25 评论39 发布时间:2016-12-6 21:42

举报

39个回答
any012 回答时间:2016-12-7 09:09:59
感觉HAL库的中断接收不是那么方便。9 {$ V1 Q' l$ S4 p* C
有时候其他设备发送过来的数据比较快,用HAL库的中断接受的话,会做很多的判断处理,等到回调函数执行的时候,有可能下个数据就快来了。
sgsong 回答时间:2017-10-16 11:53:56
/**. e8 W9 J* x4 f, g
  * @brief  Sends an amount of data in non blocking mode.+ i8 j, `( c8 `# ]2 l
  * @param  huart: pointer to a UART_HandleTypeDef structure that contains# \3 ^* v8 r% c
  *                the configuration information for the specified UART module., E% J: K- c1 f5 l. s, Y- x# A
  * @param  pData: Pointer to data buffer+ k2 }. J* I) @3 S
  * @param  Size: Amount of data to be sent
& B- d$ Y' `2 h* w3 Y  * @retval HAL status* P- d, {! }. t* }5 L* q
  */                   串口中断发送,以中断方式发送指定长度的数据。$ y. ]- F/ @# f5 }, j7 n8 X" j
HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)
6 `& ^& X9 z" r{               
: ^; E$ q4 ]+ V2 G9 ~0 J# ?  Y  /* Check that a Tx process is not already ongoing */; {$ r& |2 n2 _( K
  if(huart->gState == HAL_UART_STATE_READY)    如果 串口空闲,则执行以下语句。
: n  G) T, a3 j& j: h  {
8 g9 O  |# x& J) k0 ?2 ]    if((pData == NULL ) || (Size == 0U))      如果发送数据为空或者发送长度为0,则返回错误。, W, N1 Y* e3 T+ w
    {
9 p2 f3 w7 _3 P2 B, f. M" {      return HAL_ERROR;
$ u: f! @- r+ L) l! F/ v    }
1 R- j) Q2 C( a6 Z- A9 [1 D7 e7 U. F! K8 P- _# X8 j+ R; W
    /* Process Locked */$ {4 F$ I& H8 D* [$ G$ H
    __HAL_LOCK(huart);                上锁。
% @0 a3 I6 ^: Q- R/ T" S4 B" m
( }7 C; i$ m2 p% ]& v    huart->pTxBuffPtr = pData;               结构体变量 huart 的 参数设置。发送缓冲区,发送长度,发送计数器。* b4 m3 V) e8 `* m
    huart->TxXferSize = Size;
. r; r( T5 I; G: H+ \    huart->TxXferCount = Size;
1 s+ l/ d* B+ s" ~& C* s2 a3 u. |
3 w# z% G# ]# [    huart->ErrorCode = HAL_UART_ERROR_NONE;
7 T/ T3 x4 \. k2 \    huart->gState = HAL_UART_STATE_BUSY_TX;           状态设为 发送繁忙。
5 `( _! d* N# D: E- K& d/ F+ W% G+ K$ I. z1 W
    /* Process Unlocked */8 H3 K, ^+ `* N& L% S
    __HAL_UNLOCK(huart);               解锁。* r+ v  w8 r  B: ~  f: B

0 B' t! U8 N1 O1 }- r: l. H   /* Enable the UART Transmit data register empty Interrupt */7 k% [- W  f3 [$ Z+ ]. ?' F' ]
    SET_BIT(huart->Instance->CR1, USART_CR1_TXEIE);              使能UART发送数据寄存器空中断,则会触发串口中断(发送中断)。
$ c7 d! H8 p8 r/ L& \* v+ l/ _
% D$ \) p; m( B% W    return HAL_OK;
4 `8 f. |  y( d  }: D% k& v5 H; z$ ^' x' G7 R
  else               如果 串口忙,则返回 忙状态。- H9 z4 P2 [+ B/ S
  {5 p6 ^# S" i6 v. [) L
    return HAL_BUSY;   - r$ x( H& a" I1 ?4 n  ^
  }
* ?  [. ?* a; j. V; z}1 H- o) X$ ~# }' y
# t3 X5 O2 B5 W) b. i8 e5 C5 c( k) o
有个疑问,这个代码发送时,给整个串口上锁了(不是串口发送上锁),如果配置为全双工,这时中断接收数据怎么办?数据来时,串口一直处于忙的状态,数据量大时,数据会不会丢失?这种情况应该就不是全双工了。没找到同时收发数据的官方例程,都是一发一收的。
any012 回答时间:2016-12-7 09:49:14
本帖最后由 any012 于 2016-12-7 09:53 编辑 / |4 ~+ p, S6 X! t# }: e: g
cuyebiren 发表于 2016-12-7 09:19
' e1 {1 k( q, D2 y9 A( u( w你可以测试一下,以 115200 波特率 或 更高 长时间测试,看看会不会出现这种情况。。。或者 使用 DMA 方 ...
7 L4 T. R# ?# }9 ?# l
我的这里要求比较特殊。
: c( \; H  V8 q( h, l$ c! Ostm32的spi作为从设备,中断接收。接收过来的帧信息是固定帧首固定长度,于是我再接收完成回调函数里判断帧首,然后再调用中断模式的接收函数,使设备不断处于接收状态。
! x2 G! _; J! W) j结果,接收的数据不对,后来逐步排查,发现是接收完一个数据后,调用中断模式接收函数,但再次中断时,直接进入了overrun错误处理。应该是发送过来的数据太快,处理的太慢了。7 n3 [- }4 U7 |) Z" Y8 r
主设备用的也是stm32,硬件spi,256分频。
) ?. p$ }- m0 v除了接收部分,同时我也要发送数据。开始想在中断接收的回调函数里,调用普通的发送函数,可这个函数的第三个参数是超时判断用的,如果我接收的数据是帧尾,回调函数里我要发送下一个发送帧的帧首,但下一帧信息什么时候来,这个不确定。而一旦超时,就自动关断SPI中断及SPI了。$ Q+ t7 M' e6 r: E

2 @2 g  z& q! j8 e% ?6 t+ x7 D8 m以上问题,我最后在工程里,按标准库的方式写,通过了。4 O# K0 T* j& w2 f8 c+ e
) w7 n8 l. g, G
另一个问题是串口中断接收,是作为modbus从设备的,这个不要求接收的同时再发送数据了。但来的每帧信息,里面各数据之间的间隔很短。. C8 q! m* D5 N  V6 ?; N, g
我之前从论坛找了个其他网友分享的标准库版modbus文件,挺好用的。改用HAL库没调成功,感觉还是接收到一个数据后,要判断处理的太多。& z2 x+ g  r7 r& ^0 y
* q- {! X3 @* m: U4 k
目前HAL库的教程,只找到了硬石科技的STM32F103的教程,正点原子的也有一般STM32F4的HAL教程。
1 R% ^- c: I3 d3 s) `但两个教程里,貌似都没有接收中断这类的例子。, \8 [6 S' x7 R# T+ L/ s
原子的教程,sys文件夹下串口部分,貌似也是按类似标准库的方式处理的。5 N4 ~! f3 @5 F0 y: n" b9 p' I6 ]
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
0 p: i$ ]2 [# X绝对高手!!!!!!多谢了

/ U# I# d) h1 P7 l, r0 ]4 R+ j' j我只是研究了下HAL库的源码,做到“知其所以然”而已啦!
斜阳 回答时间:2016-12-7 09:10:19
mARK一下
cuyebiren 回答时间:2016-12-7 09:19:18
any012 发表于 2016-12-7 09:09- R; B7 c. t/ |( j$ @1 _" O
感觉HAL库的中断接收不是那么方便。1 ~% S+ l& v9 w
有时候其他设备发送过来的数据比较快,用HAL库的中断接受的话,会做很 ...
( A) ~! @# m" N/ K* K, h0 W
你可以测试一下,以 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
! Y8 s% N( ~$ }9 t/ Q我的这里要求比较特殊。
# e5 X4 F% Q) F" n8 I' Pstm32的spi作为从设备,中断接收。接收过来的帧信息是固定帧首固定长度,于是我再 ...
" p& P9 c+ w  T! D0 n
超时 值 可大可小。超时判断也是可以取消的,好像就是把 超时值 设为 0XFFFFFFFF 。
( F( b* N" a5 x1 Z中断处理函数 中 的 各种判断 是很快的,就是 几个 if 语句而已。除非是 真的 出现 了 传输错误 或 溢出 错误。
/ v8 ^/ Y+ Z+ Y9 ?% E中断回调函数 越短越好,切记 不要在 中断函数中 加延时。
3 F1 I2 {+ ]7 ]( R% D具体 怎么实现各种功能,建议参考官方例程。
cuyebiren 回答时间:2016-12-7 13:25:40
zhoupxa 发表于 2016-12-7 11:56/ B+ a$ t7 Y& k- u
使用HAL库在某些情况下还是有局限性的,如果要求更高的话,还是寄存器编程吧 ...

' U& F2 I+ `4 t5 H2 `HAL库的各种判断,以及中断开关做的是比较好的,处理过程也是比较完整的。标准库仅是封装了操作寄存器的函数。你做低功耗,或者Flash比较小,代码又比较多时,寄存器编程更好。但通常情况下,这种情况的概率很小。。。用库可以把我们从编写大量寄存器中解放出来。。。
zbber 回答时间:2016-12-7 16:41:40
好帖,赞一个,顶楼主4 r; |2 O* E% u" ?5 b
*numb 回答时间:2017-3-14 14:48:52
DMA的中断回调函数在哪,找也找不到
123下一页

所属标签

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