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

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

[复制链接]
cuyebiren 发布时间:2016-12-6 21:42

# `" v4 C5 v* o( q& w2 `STM32电机培训online,大佬带你玩电机9 D6 e9 ], I+ @+ T$ S7 {' W
【开源】做了一个STLINK/V2-UART二合一编程器1 ?1 O: ^; G5 X1 C
; z6 H) f" ^6 @9 K
【实战经验】UART应用异常案例分析) t/ R* `6 l. _  W5 \7 F( }$ x  Z

2 j. h! l% E) B( C. _( Z  F: }0 @% Z$ R- C$ _4 N& J0 r# H
HAL库是比较全面的,封装比较彻底的,也是功能比较强大的。1 m& P+ F8 j- c
使用HAL库,我们直接调用它的API函数,不用关心它的底层操作过程。* H! C  U$ p, ?
使用HAL库,省去了好多繁琐的处理过程,不再需要我们自己写如等待等过程。
& ?# [; R3 `5 o" _9 S7 AHAL库也包含如Ethernet、USB等高级外设的驱动。
) j8 S( r8 m. J% k) v对于初接触它的人来说,尤其是用惯了标准库的人,总会有各种不适应和排斥感。: O9 [$ n3 n2 u% t8 w9 l
就拿UART来说,我们通过中断方式接受或发送数据。如果仿真调试的话,会发现UART有开关中断的现象,而不是中断一直开着。( T6 S1 l- D1 i- `
% l, {5 Z/ Y1 u& k4 Z# _+ Y
下面,就讲解UART常用函数以及中断处理过程。
; g9 q8 V" q% e7 ~/ b- S8 A
0 m4 B$ y5 N, l( Q! ^7 H+ {( IHAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout);
9 c4 O) h  X  W3 m2 k# _- v, l串口发送,发送指定长度的数据。如果超时没发送完成,则不再发送,返回超时标志(HAL_TIMEOUT)。
  B6 v7 H5 n8 pHAL_StatusTypeDef HAL_UART_Receive(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout);
' k3 T+ [6 ^! i) M串口接收,接收指定长度的数据。如果超时没接收完成,则不再接收数据到指定缓冲区,返回超时标志(HAL_TIMEOUT)。
/ K) h  a4 F' {% q
( s7 d+ z' P3 y6 H

0 m9 g  N$ R' J2 EHAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);  O" Q  T0 c& Q/ C. s1 J
串口中断发送,以中断方式发送指定长度的数据。
9 T/ S6 S- g! \6 J大致过程是,把 发送缓冲区指针 指向 要发送的数据,设置 发送长度,发送计数器初值,然后使能串口发送中断,触发串口中断。. y/ q( ]% `4 a& b4 `5 |( ^" n
再然后,串口中断函数处理,直到数据发送完成,而后关闭中断,不再发送数据串口发送完成回调函数
& f2 m1 w. y) j( U, s2 Z' xHAL_StatusTypeDef HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
4 p# m4 R" p9 w) ]8 Z# u. ~6 S' R2 F串口中断接收,以中断方式接收指定长度数据。
' U4 o2 x. u# O大致过程是,把 接收缓冲区指针 指向 要存放接收数据的数组,设置 接收长度,接收计数器初值,然后使能串口接收中断。接收到数据时,会触发串口中断。( _: u& o0 d& O/ f2 \+ [; `/ z
再然后,串口中断函数处理,直到接收到指定长度数据,而后关闭中断,不再触发接收中断,调用串口接收完成回调函数。
4 ?" `1 N' B9 L% j
* o# U( [( a; a4 k4 ~7 Z

4 b8 L2 g, x, c8 |+ L0 zHAL_StatusTypeDef HAL_UART_Transmit_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);
. l* Z# t" n& n串口DMA发送,以DMA方式发送指定长度的数据。
3 L1 `+ ~' B( e% G3 r# g% z过程是,发送缓冲区指针 指向 要发送的数据,设置 发送长度,发送计数器初值,设置 DMA传输完成中断的回调函数,使能DMA控制器中断,使能DMA控制器传输,使能UART的DMA传输请求。6 D! X5 c& c, O; j5 k
然后,UART便会发送数据,直到发送完成,触发DMA中断。+ _' b2 r( t' @0 B6 F2 N8 k
DMA中断处理,如果 DMA模式 是 循环模式,则 直接 调用 DMA传输完成中断的回调函数。
- W( Y* p5 A" ~: \* ^! [如果 DMA模式 是 正常模式,则 先 关闭DMA传输完成中断,不再触发DMA中断再 调用 DMA传输完成中断的回调函数。2 o! o& G* Y9 D/ A1 U$ ]
DMA传输完成中断的回调函数处理过程,如果 DMA模式 是 循环模式,则 直接 调用 串口发送完成回调函数。/ g/ U6 m0 r  f7 T
如果 DMA模式 是 正常模式,则 先关闭 UART的DMA传输请求, 再 使能串口传输完成中断,直到传输完成,触发中断。2 w4 i) d9 s7 T  X
串口传输完成中断处理,关闭中断,调用串口发送完成回调函数
) W/ \/ P0 w( D! W+ xHAL_StatusTypeDef HAL_UART_Receive_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);6 J) w3 v* {! k4 r- Z  g
串口DMA接收,以DMA方式接收指定长度的数据。
* i6 v" V. W$ A过程是,接收缓冲区指针 指向 要存放接收数据的数组,设置 接收长度,接收计数器初值,设置 DMA传输完成中断的回调函数,使能DMA控制器中断,使能DMA控制器传输,使能UART的DMA传输请求。
0 K; h% v+ H, e+ i% u' }, p7 A然后,UART接收到数据,便会通过DMA把数据存到接收缓冲区,直到接收到指定长度数据,触发DMA中断。
) Z" o$ w7 S- Q) n: b4 j8 cDMA中断处理,如果 DMA模式 是 循环模式,则 直接 调用 DMA传输完成中断的回调函数。
3 a9 @9 i& d/ ~1 W# D. x$ B如果 DMA模式 是 正常模式,则 先 关闭DMA传输完成中断,不再触发DMA中断再 调用 DMA传输完成中断的回调函数。5 k# o& y. ?  u% g: h+ \- O/ s
DMA传输完成中断的回调函数处理过程,如果 DMA模式 是 循环模式,则 直接 调用 串口接收完成回调函数。0 A8 |+ _, E% H8 Y/ M6 ]
如果 DMA模式 是 正常模式,则 先关闭 UART的DMA传输请求,调用 串口接收完成回调函数。+ r$ `7 f+ A1 t5 W1 Y5 B

! v+ g6 U' ^6 R7 d7 z, C- ^0 y% U% h; |1 w7 M" R* Y, K
, [% J1 @' F! F$ O7 [2 t
由于函数较多和过长的缘故,下面仅以HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)为例,分析源码:
# }2 D+ [% Q% V8 C" w3 U/**
6 T0 g6 R* w6 j0 u: x: p: \$ D) ^  * @brief  Sends an amount of data in non blocking mode.  ^5 G4 [- d! R. L( n: X
  * @param  huart: pointer to a UART_HandleTypeDef structure that contains( F# b) }) z: Z/ S+ f
  *                the configuration information for the specified UART module.) [  }- T' q: ^. M; t) M
  * @param  pData: Pointer to data buffer' T( u9 J0 i6 ~6 b8 |
  * @param  Size: Amount of data to be sent( J, J8 _% R3 \; P' s( r7 W
  * @retval HAL status! ?6 a. k# ]% S* ]; _( h
  */                   串口中断发送,以中断方式发送指定长度的数据。
! D& ~2 |# j3 I. W' JHAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)" x6 o% z) @& h/ k6 j
{               
  Y. Y8 Y+ s0 U7 j  /* Check that a Tx process is not already ongoing */0 [7 g4 {3 ~" s3 `
  if(huart->gState == HAL_UART_STATE_READY)    如果 串口空闲,则执行以下语句。
9 q9 o! R' Y* @  {
! ^$ T! T4 `5 ]+ b; S" D/ i; Z    if((pData == NULL ) || (Size == 0U))      如果发送数据为空或者发送长度为0,则返回错误。" H( F4 v" X  X3 H4 F
    {
' x# O  v+ |/ J' s% n+ a5 T7 Q      return HAL_ERROR;
/ g1 T1 b/ J  L2 N8 ?" ]    }
' I% o2 z* F2 g6 V+ u) M9 c. b% U+ P
    /* Process Locked *// c$ J: C& G3 x2 T
    __HAL_LOCK(huart);                上锁。
. \' |5 X. z/ ~% z
9 P( d) J- |$ R& @; n% b( i$ r: M    huart->pTxBuffPtr = pData;               结构体变量 huart 的 参数设置。发送缓冲区,发送长度,发送计数器。
( u" M' e4 p4 p    huart->TxXferSize = Size;
2 v) y0 }& m9 H1 [0 P/ _    huart->TxXferCount = Size;
8 H3 \0 J/ o( y: W' p# b( y% ]7 _  O4 e0 @, @% m/ G6 B" g
    huart->ErrorCode = HAL_UART_ERROR_NONE;) L7 b+ @$ P2 o0 m
    huart->gState = HAL_UART_STATE_BUSY_TX;           状态设为 发送繁忙。: X  L/ A1 d. i" H) f. g/ m
8 P. {% f! }; O
    /* Process Unlocked */6 w& p; ^6 J, N  N9 n  c
    __HAL_UNLOCK(huart);               解锁。
# p/ X1 i# M5 g8 q' A/ v$ V" m2 u! \# g; [- [
   /* Enable the UART Transmit data register empty Interrupt */
7 n- N" T. b1 ^/ [  ]5 T1 e; }) B    SET_BIT(huart->Instance->CR1, USART_CR1_TXEIE);              使能UART发送数据寄存器空中断,则会触发串口中断(发送中断)。: F- T* p4 p& o) E3 g

5 @) X! U" S7 b. ]/ w0 K5 r    return HAL_OK;
9 @9 I6 q' w0 h0 _  }6 J, y! R( V7 D& p2 a$ b8 W
  else               如果 串口忙,则返回 忙状态。
- R* F1 N' m% A  {. n" H8 r' ?) Q% q6 M. z
    return HAL_BUSY;   3 b+ r, R; S9 @
  }
- m- G. n! \3 Y0 _}" Q1 ~, V! r# D; f, t# c
5 Z: _0 n1 a$ s" P9 [
! h- @, o" \+ m* [/ y
1 收藏 25 评论39 发布时间:2016-12-6 21:42

举报

39个回答
any012 回答时间:2016-12-7 09:09:59
感觉HAL库的中断接收不是那么方便。
" A" J( `$ R* H- h* G0 e有时候其他设备发送过来的数据比较快,用HAL库的中断接受的话,会做很多的判断处理,等到回调函数执行的时候,有可能下个数据就快来了。
sgsong 回答时间:2017-10-16 11:53:56
/**
: T! v: H, D8 }% J; ]) i9 H5 [+ n7 }  * @brief  Sends an amount of data in non blocking mode.& F( E( I" e! d& _' ^! F4 `
  * @param  huart: pointer to a UART_HandleTypeDef structure that contains
  p; ?; P; z1 O2 w0 C% k  *                the configuration information for the specified UART module./ N: _- I5 x6 c( @% Y4 k
  * @param  pData: Pointer to data buffer- |( l# s# h, }) Z% T" s; R* L2 X
  * @param  Size: Amount of data to be sent1 l8 _$ ]  b$ p) w: O6 ~# b
  * @retval HAL status/ l" |) `9 s7 ]) n4 S6 b2 F" |
  */                   串口中断发送,以中断方式发送指定长度的数据。
, j% A  y9 k/ K9 j: n' U7 Y  sHAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)2 D! `0 b  T" T" k+ y: [
{               
- Y  h1 e; q! B  /* Check that a Tx process is not already ongoing */
4 X- C/ K: H6 p2 i- c  if(huart->gState == HAL_UART_STATE_READY)    如果 串口空闲,则执行以下语句。
4 w( e: @; d& g$ `( b9 [) r+ e$ g  T: r) @  {
/ Z' }9 A/ L. m" q    if((pData == NULL ) || (Size == 0U))      如果发送数据为空或者发送长度为0,则返回错误。
5 O* @3 a# `, f( w0 a    {; V% D; c: V8 s# w% G. \0 u- j
      return HAL_ERROR;4 n7 x7 T0 }5 m) F2 q
    }
2 M$ R3 E/ }. h+ s  P+ s. |5 l+ u/ V2 n* N
    /* Process Locked */% o8 T) O5 i6 O! a3 ]* ^- ]8 i7 _
    __HAL_LOCK(huart);                上锁。
- |7 {3 t5 X: O! R* M5 n
# W5 _; s0 b4 @' \7 ^+ b' S    huart->pTxBuffPtr = pData;               结构体变量 huart 的 参数设置。发送缓冲区,发送长度,发送计数器。
: Z' f8 A! e4 J; `    huart->TxXferSize = Size;; s6 A6 b% S4 _& N! P4 p0 m. J& M
    huart->TxXferCount = Size;+ B1 z& c( ^" e7 y+ t

9 Z2 M* T5 b+ u" U    huart->ErrorCode = HAL_UART_ERROR_NONE;
" W7 h. R3 l! x3 a( H0 V3 x, q& i    huart->gState = HAL_UART_STATE_BUSY_TX;           状态设为 发送繁忙。: c$ [& w9 a0 G" q( S7 `

9 D# E& A* j1 y    /* Process Unlocked */! n: h. z2 `& ?' |" c( Q* g
    __HAL_UNLOCK(huart);               解锁。
3 Q7 D( Y4 p' z# L! w1 i0 K# l& _# p5 Z: ^! w
   /* Enable the UART Transmit data register empty Interrupt */1 }: y1 L, v& M1 g1 u/ o$ P
    SET_BIT(huart->Instance->CR1, USART_CR1_TXEIE);              使能UART发送数据寄存器空中断,则会触发串口中断(发送中断)。
9 U$ }- Y2 b1 ~' u
. Z, w" \- P: ^0 C    return HAL_OK;
4 Z9 l7 \& c7 N0 V, C  g' G  }
7 N" W- A6 V1 E1 @  I/ x1 s2 P  else               如果 串口忙,则返回 忙状态。0 K5 t, c* [  ?/ f3 P( n
  {
' q. K) F( |- j% A4 o6 b9 e' c( ?# `    return HAL_BUSY;   ; m" g' S2 f% p2 C
  }
: b; j' \. v5 P3 B* T}/ j! i5 `. V2 V6 h
6 Q0 u+ W& Z: O
有个疑问,这个代码发送时,给整个串口上锁了(不是串口发送上锁),如果配置为全双工,这时中断接收数据怎么办?数据来时,串口一直处于忙的状态,数据量大时,数据会不会丢失?这种情况应该就不是全双工了。没找到同时收发数据的官方例程,都是一发一收的。
any012 回答时间:2016-12-7 09:49:14
本帖最后由 any012 于 2016-12-7 09:53 编辑
5 j& N1 h* x9 m- N9 Q# B+ n
cuyebiren 发表于 2016-12-7 09:195 _3 O. }6 N2 t% e* h
你可以测试一下,以 115200 波特率 或 更高 长时间测试,看看会不会出现这种情况。。。或者 使用 DMA 方 ...
" l3 ?* D  l5 p4 U: v7 ~
我的这里要求比较特殊。! Z9 k7 @4 [% A+ L
stm32的spi作为从设备,中断接收。接收过来的帧信息是固定帧首固定长度,于是我再接收完成回调函数里判断帧首,然后再调用中断模式的接收函数,使设备不断处于接收状态。
! ]" i: }0 u; W# \8 L( \4 ?结果,接收的数据不对,后来逐步排查,发现是接收完一个数据后,调用中断模式接收函数,但再次中断时,直接进入了overrun错误处理。应该是发送过来的数据太快,处理的太慢了。
3 v. X3 E3 ^; }* D8 s主设备用的也是stm32,硬件spi,256分频。
, ^% @7 }; P# ?9 ~4 X  y0 ^# U除了接收部分,同时我也要发送数据。开始想在中断接收的回调函数里,调用普通的发送函数,可这个函数的第三个参数是超时判断用的,如果我接收的数据是帧尾,回调函数里我要发送下一个发送帧的帧首,但下一帧信息什么时候来,这个不确定。而一旦超时,就自动关断SPI中断及SPI了。- }' N" L$ C- ^) E2 ?1 ?* f1 l; ]2 g

; o7 V' n7 V( |% k1 a# a以上问题,我最后在工程里,按标准库的方式写,通过了。
8 m. l- r: y/ O' W; x( V
/ [: n2 ]& y0 N0 {) ~另一个问题是串口中断接收,是作为modbus从设备的,这个不要求接收的同时再发送数据了。但来的每帧信息,里面各数据之间的间隔很短。- P8 J1 u5 m) F% u  S5 f- w  z9 b; j
我之前从论坛找了个其他网友分享的标准库版modbus文件,挺好用的。改用HAL库没调成功,感觉还是接收到一个数据后,要判断处理的太多。
; C8 K3 Z8 }7 S" [5 z6 x* H1 e: i9 X, Z
目前HAL库的教程,只找到了硬石科技的STM32F103的教程,正点原子的也有一般STM32F4的HAL教程。3 d( ^( a8 c3 n9 D* s5 z
但两个教程里,貌似都没有接收中断这类的例子。% J4 Q. Z$ O" a) e, g# i
原子的教程,sys文件夹下串口部分,貌似也是按类似标准库的方式处理的。
+ {' Q& N% A7 o: k; p/ B% Q! Q: H
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
: a" Q1 H$ j4 o* q0 g6 @% c绝对高手!!!!!!多谢了
( G7 d  h# a7 o2 l7 z# m
我只是研究了下HAL库的源码,做到“知其所以然”而已啦!
斜阳 回答时间:2016-12-7 09:10:19
mARK一下
cuyebiren 回答时间:2016-12-7 09:19:18
any012 发表于 2016-12-7 09:09* c! b8 f0 l4 l! g9 [; j5 I, T
感觉HAL库的中断接收不是那么方便。. Z% U2 w, h4 I
有时候其他设备发送过来的数据比较快,用HAL库的中断接受的话,会做很 ...
  M, f. X% `# a" P& }) V
你可以测试一下,以 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
4 D4 L8 F2 [+ [% X9 e我的这里要求比较特殊。
% E1 R' ?2 F/ C% J& N! x: C* S& X0 Mstm32的spi作为从设备,中断接收。接收过来的帧信息是固定帧首固定长度,于是我再 ...

; g0 e7 w) \& b: I- R超时 值 可大可小。超时判断也是可以取消的,好像就是把 超时值 设为 0XFFFFFFFF 。
9 m, o6 G2 T& g! u" y$ a中断处理函数 中 的 各种判断 是很快的,就是 几个 if 语句而已。除非是 真的 出现 了 传输错误 或 溢出 错误。( ?% p( [0 k9 m  @0 F4 ?
中断回调函数 越短越好,切记 不要在 中断函数中 加延时。
5 F' X4 V  Q% Q% a9 G! W/ Y具体 怎么实现各种功能,建议参考官方例程。
cuyebiren 回答时间:2016-12-7 13:25:40
zhoupxa 发表于 2016-12-7 11:56
  ?+ i& E& x) n6 H使用HAL库在某些情况下还是有局限性的,如果要求更高的话,还是寄存器编程吧 ...

7 _! t/ s3 u7 o0 EHAL库的各种判断,以及中断开关做的是比较好的,处理过程也是比较完整的。标准库仅是封装了操作寄存器的函数。你做低功耗,或者Flash比较小,代码又比较多时,寄存器编程更好。但通常情况下,这种情况的概率很小。。。用库可以把我们从编写大量寄存器中解放出来。。。
zbber 回答时间:2016-12-7 16:41:40
好帖,赞一个,顶楼主
& J4 v6 y7 ]3 y6 c# |* A, s
*numb 回答时间:2017-3-14 14:48:52
DMA的中断回调函数在哪,找也找不到
123下一页

所属标签

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