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

基于STM32开发串口DMA空闲中断经验分享

[复制链接]
攻城狮Melo 发布时间:2023-3-29 18:42
  做STM32开发,串口可以说真的是无处不在,芯片与芯片之间的通信可以用TTL电平,多机通信可以用RS485工控总线,对外上位机需要RS232串口通信,那么作为一款经典的单片机系列STM32,应该怎么处理串口报文最有效快捷呢,无数的实践证明,大量的项目经验告诉我们答案永远是串口空闲中断。" R5 R8 [4 C& s+ d

; x) }$ S0 S! ?8 y! a  ]8 t      那么什么是串口空闲中断呢,大家都知道串口在空闲的时候默认是高电平,在发数据的时候有起始位,数据位,结束位等等,传统的串口处理数据的方式是一个个字节接收,这样做猛地一看好像没有啥问题,但却留下很多隐患,比如:1.固定的报头,一个个字节接收少接收一字节,导致报文接收异常;2.同时开两个115200波特率的串口,因为两个串口本身是独立的,一个个字节接收数据导致频繁进入中断,存在中断嵌套的风险;3.驱动底层代码会非常繁琐,把一个STM32硬生生玩成一个FPGA,判断报头报尾,CRC校验还需要C语言做成类似Verilog状态机的逻辑等等,笔者在这里不在赘述了,因为不是这篇博客的重点。+ j( H& L' V- h# b, M. }" I8 B% G+ `
, A% t2 T+ _. f
       串口空闲中断可以简单地理解成,当串口不发数据的时候产生中断,然后我们可以巧妙地把这些数据通过DMA搬运到指定变量地址上,这样当主机不发数据的时候,即可以得到整条报文的数据,和报文的长度,之后再对整条报文进行处理,包括CRC校验,逻辑层赋值,应用层控制等其他相关操作,这样就极大地减少了频繁进入中断的次数,也优化了底层逻辑,因为收到是一整条报文而不是一个个字节,如图1所示,笔者这里用到的STM32F103RCT6芯片,PCB上把STM32的PB6和PB7作为UART1,查阅芯片手册可以看到如图2所示,DMA1的通道4和通道5是UART1的对应的DMA通道,所以我们在编写完初始化USART1_Init后,还需要如图3所示打开两个通道的中断。
; F% w: Z+ n, e  c' P6 U3 M/ {/ t) x: \$ [
5d3d67356ae948beb6b1b6a58b458b93.png
7 d$ E9 n( Y; P! K5 X+ N% U      6 D; l4 w6 a$ j# s! a" G5 Y
dc864cbe53864d6494b50e8781e92fae.png
* q5 ^: l+ |4 I

4 e3 L1 v: [) G  v1 P5 N2 h 3b9049c2eefa4930aa49ee867aea090a.png
" ^. E! _, J# {5 B

: e9 X: b- @$ j9 i7 o$ d  h e8bddc9a676d4c13a1f75e352057c499.png
$ B; J5 u7 g# k$ C d08ba5f61bf24d60b590d9a47f476a8e.png 0 k8 Y' T" ~+ c6 u

! g! W8 n$ t) g+ Y. G! R
7 N. s, v$ q3 K: W# |5 V+ ?7 X 02a09f5becc44467ad864b90381815e5.png
9 V0 f/ m9 B. y7 r1 v' Q8 }: L  J3 A7 O' W' y, I; S) {+ V+ N1 w
2 f7 t5 I* E" `
图1 USART1的初始化函数" Z5 K0 x& Q  A/ `
/ i5 N, R* K; A0 _4 [# D) _
a51b018d5ee14acd936a9ca32026cf88.png
) r4 f9 T. _$ z$ Q% h% R& h2 W. `6 V: ?& O% I6 Q9 ]
图2 STM32F103系列的DMA通道示意图
  r5 [9 U2 u) l0 i# \5 S) B' \* [1 L, x* Z, c, c- C; Z( T
e492323adc7447ac9ffb411698907d27.png
: }3 S! k$ m7 P0 G" }6 K
8 d* p. a( V" m% h4 I6 w 图3 DMA通道4和通道5中断开启
8 N0 {# q# O, o5 U; s
1 h) J9 ~2 n' y$ Q+ G7 A. U       如图4所示,定义了串口收发的数据报文,这里面flag在串口空闲中断里置位,表示接收到一包数据,在主函数里再进行数据解包处理操作,切记在STM32中断里不要做过多的判断和等待或者其他复杂操作,假设来了一个中断,CPU处理了10秒钟,那么大家都知道STM32也好,ARM LIUNX也好运行的原理,中断来了要压栈保护现场,屏蔽IRQ等其他中断,如果一个中断处理太长时间,那么对于用户来说体验会非常差,其他操作都不被CPU响应,仿佛是死机了一样,毫无实时性可言,如图5所示是串口空闲中断处理函数,这里需要说明几个地方:1.因为开始DMA接收的数据长度设定的是2048个字节,所以USART1_Type.RX_Count = 2048-Uart1Handle.hdmarx->Instance->CNDTR这句话,代表把整包报文的数据字节长度赋值给USART1_Type.RX_Count;2.需要在空闲中断里先关DMA,再使能DMA才能在下一次串口空闲中断产生的时候触发;3.可以在串口空闲中断处简单地判断报头,简化了逻辑层的设计。
: A3 v/ O8 }8 b+ w" [/ \3 ~0 W1 J% d6 C$ m2 i
1fbe428cb2e745cfb11dfc888d6501d0.png
& E7 c8 e  j3 s( E7 \: q5 p2 {

. K' p9 b, }9 p; N* p6 I图4 串口收发数据报文定义  r) ~) Y( {' _/ `

3 }$ [' b% I5 ]6 g$ Y
a05524e5329f4be8965e7037444c3c6d.png
+ v5 \, i# U' _6 T7 {7 {# G 图5 串口空闲中断处理函数- N2 w9 i9 z( J
————————————————/ u9 T  H" }$ j6 ~5 ]1 F9 N
版权声明:青青豌豆+ k, h8 W) f. ~4 G* `' ?1 s
如有侵权请联系删除+ \6 t+ b- U1 _3 V, J: ~' b+ ?! J

% J: c) \( z; x
46009ea4ae2542119d017ef8bcb49abc.png
收藏 评论0 发布时间:2023-3-29 18:42

举报

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