
本帖最后由 xiaojie0513 于 2018-9-18 21:07 编辑 引言 在使用stm32或者其他单片机的时候,会经常使用到串口通讯,那么如何有效地接收数据呢?假如这段数据是不定长的有如何高效接收呢? DMA简介 什么是DMA DMA:全称Direct Memory Access,即直接存储器访问 DMA 传输将数据从一个地址空间复制到另外一个地址空间。CPU只需初始化DMA即可,传输动作本身是由 DMA 控制器来实现和完成。典型的例子就是移动一个外部内存的区块到芯片内部更快的内存区。这样的操作并没有让处理器参与处理,CPU可以干其他事情,当DMA传输完成的时候产生一个中断,告诉CPU我已经完成了,然后CPU知道了就可以去处理数据了,这样子提高了CPU的利用率,因为CPU是大脑,主要做数据运算的工作,而不是去搬运数据。DMA 传输对于高效能嵌入式系统算法和网络是很重要的。 在STM32的DMA资源STM32F1系列的MCU有两个DMA控制器(DMA2只存在于大容量产品中),DMA1有7个通道,DMA2有5个通道,每个通道专门用来管理来自于一个或者多个外设对存储器的访问请求。还有一个仲裁器来协调各个DMA请求的优先权。 而STM32F4/F7/H7系列的MCU有两个DMA控制器总共有16个数据流(每个DMA控制器8个),每一个DMA控制器都用于管理一个或多个外设的存储器访问请求。每个数据流总共可以有多达8个通道(或称请求)。每个通道都有一个仲裁器,用于处理 DMA 请求间的优先级。 " a7 K6 h" q8 x/ ~ DMA接收数据 % g u4 s* C0 L; l1 n ?6 l DMA在接收数据的时候,串口接收DMA在初始化的时候就处于开启状态,一直等待数据的到来,在软件上无需做任何事情,只要在初始化配置的时候设置好配置就可以了。等到接收到数据的时候,告诉CPU去处理即可。
其实,有很多方法:
DMA+串口空闲中断 这两个资源配合,简直就是天衣无缝啊,无论接收什么不定长的数据,管你数据有多少,来一个我就收一个,就像广东人吃“山竹”,来一个吃一个~(最近风好大,我好怕)。 可能很多人在学习stm32的时候,都不知道idle是啥东西,先看看stm32串口的状态寄存器: 当我们检测到触发了串口总线空闲中断的时候,我们就知道这一波数据传输完成了,然后我们就能得到这些数据,去进行处理即可。这种方法是最简单的,根本不需要我们做多的处理,只需要配置好,串口就等着数据的到来,dma也是处于工作状态的,来一个数据就自动搬运一个数据。 接收完数据时处理串口接收完数据是要处理的,那么处理的步骤是怎么样呢?
注意事项 STM32的IDLE的中断在串口无数据接收的情况下,是不会一直产生的,产生的条件是这样的,当清除IDLE标志位后,必须有接收到第一个数据后,才开始触发,一断接收的数据断流,没有接收到数据,即产生IDLE中断。如果中断发送数据帧的速率很快,MCU来不及处理此次接收到的数据,中断又发来数据的话,这里不能开启,否则数据会被覆盖。有两种方式解决:
实验效果: 当外部给单片机发送数 据的时候,假设这帧数据长度是1000个字节,那么在单片机接收到一个字节的时候并不会产生串口中断,只是DMA在背后默默地把数据搬运到你指定的缓冲区里面。当整帧数据发送完毕之后串口才会产生一次中断,此时可以利用DMA_GetCurrDataCounter()函数计算出本次的数据接受长度,从而进行数据处理。 串口的配置 /**0 P# P# |7 u4 j! B% M+ U很简单,基本与使用串口的时候一致,只不过一般我们是打开接收缓冲区非空中断,而现在是打开空闲中断——USART_ITConfig(DEBUG_USARTx, USART_IT_IDLE, ENABLE);。 * @brief USART GPIO 配置,工作参数配置( R" }) j( k5 m0 }- j* [" n2 E+ D * @param 无 * @retval 无 */! R' ~' C. _" [+ M* [7 { void USART_Config(void) {9 d, V; w- t: h6 P GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; b9 X6 {1 r% t7 G' K // 打开串口GPIO的时钟 DEBUG_USART_GPIO_APBxClkCmd(DEBUG_USART_GPIO_CLK, ENABLE);5 l2 `: n, U T , e O7 O* o3 d' w4 C9 W( K // 打开串口外设的时钟* \. E5 Q9 e& Z; B4 \ DEBUG_USART_APBxClkCmd(DEBUG_USART_CLK, ENABLE); // 将USART Tx的GPIO配置为推挽复用模式 GPIO_InitStructure.GPIO_Pin = DEBUG_USART_TX_GPIO_PIN;. ]0 N5 Z, H. R5 H GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(DEBUG_USART_TX_GPIO_PORT, &GPIO_InitStructure); // 将USART Rx的GPIO配置为浮空输入模式3 L4 u+ B* W& B! s% ~ GPIO_InitStructure.GPIO_Pin = DEBUG_USART_RX_GPIO_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(DEBUG_USART_RX_GPIO_PORT, &GPIO_InitStructure); 2 q8 a' G7 V2 d4 t9 I // 配置串口的工作参数! i+ a+ o. e: T8 h4 G // 配置波特率. q! X0 {0 w c USART_InitStructure.USART_BaudRate = DEBUG_USART_BAUDRATE; // 配置 针数据字长 USART_InitStructure.USART_WordLength = USART_WordLength_8b; // 配置停止位& E1 P8 P% z0 x6 \# e USART_InitStructure.USART_StopBits = USART_StopBits_1; // 配置校验位 USART_InitStructure.USART_Parity = USART_Parity_No ;4 s: m4 Y$ Q' w ]2 \. N* [5 G // 配置硬件流控制 USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;+ k. M( v# _, J) E' @: V7 e // 配置工作模式,收发一起. b" f# }8 w' U* _ USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;8 Z7 I* F2 M. o. I // 完成串口的初始化配置 USART_Init(DEBUG_USARTx, &USART_InitStructure);! m+ p, b' u9 H+ B3 I/ v // 串口中断优先级配置& [; A! C/ U9 M1 a2 _1 w NVIC_Configuration(); #if USE_USART_DMA_RX // 开启 串口空闲IDEL 中断4 f- F" I" d! h c* E% [ USART_ITConfig(DEBUG_USARTx, USART_IT_IDLE, ENABLE); j- c' A8 ^- r0 v& Y5 p8 R/ M // 开启串口DMA接收& D0 ^* P* j+ d1 P- Q4 r& O% B* U USART_DMACmd(DEBUG_USARTx, USART_DMAReq_Rx, ENABLE); / W; ~" k; K2 `+ `2 h( C! j /* 使能串口DMA */ USARTx_DMA_Rx_Config();$ H2 U7 d% M6 J8 \ u, V #else // 使能串口接收中断/ u% b; k5 x; ]# y- p+ g USART_ITConfig(DEBUG_USARTx, USART_IT_RXNE, ENABLE); #endif5 [+ w1 n+ v0 r# [6 ] #if USE_USART_DMA_TX ; C+ N1 \$ g, g: F4 w // 开启串口DMA发送4 V' b. a/ W& Q7 G+ O; c& G // USART_DMACmd(DEBUG_USARTx, USART_DMAReq_Tx, ENABLE); USARTx_DMA_Tx_Config();& L' l1 f) ~5 n* u #endif // 使能串口 USART_Cmd(DEBUG_USARTx, ENABLE); $ v: e7 i8 F% \( j3 r, { } 串口DMA配置 把DMA配置完成,就可以直接打开DMA了,让它处于工作状态,当有数据的时候就能直接搬运了。 #if USE_USART_DMA_RX static void USARTx_DMA_Rx_Config(void) { DMA_InitTypeDef DMA_InitStructure; // 开启DMA时钟 RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); // 设置DMA源地址:串口数据寄存器地址*/ DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)USART_DR_ADDRESS;5 u I: {1 }* y) N // 内存地址(要传输的变量的指针). w! r7 w S$ r- y( C DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)Usart_Rx_Buf;. n( S& @/ d2 C V8 b // 方向:从内存到外设 $ ?1 z) L4 `! M DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; // 传输大小 DMA_InitStructure.DMA_BufferSize = USART_RX_BUFF_SIZE;8 M; @' H$ ]. ^ ]6 w1 y* h; ] // 外设地址不增 . Y" \1 S. _3 N" A6 M& a7 N5 D$ U0 U DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;0 Y( P4 D- U! h7 d$ @+ b1 ~ // 内存地址自增 DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;+ ^. T0 l9 }$ O# Z5 r2 x4 W // 外设数据单位 ( e% \7 y6 U9 h& @& d DMA_InitStructure.DMA_PeripheralDataSize = * Y- F" V" ?# R( \8 g DMA_PeripheralDataSize_Byte; // 内存数据单位/ C2 h, {4 ^( H' m DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; // DMA模式,一次或者循环模式1 T+ _5 y2 {( c3 I3 F //DMA_InitStructure.DMA_Mode = DMA_Mode_Normal ; DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; // 优先级:中 1 Y) y4 z. r2 F7 d# M3 i6 T0 _+ } DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh; - i( t* ~( F/ W // 禁止内存到内存的传输 DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;' r5 l0 j0 ^2 V1 H // 配置DMA通道 DMA_Init(USART_RX_DMA_CHANNEL, &DMA_InitStructure); // 清除DMA所有标志# i5 l F* \$ t: m8 d ~/ s DMA_ClearFlag(DMA1_FLAG_TC5); DMA_ITConfig(USART_RX_DMA_CHANNEL, DMA_IT_TE, ENABLE); // 使能DMA+ h- B1 K, @; Z7 h+ W% Y2 |- {6 A/ g DMA_Cmd (USART_RX_DMA_CHANNEL,ENABLE); }* w5 n- O/ v1 K' v/ U #endif3 w1 X$ @- m. v1 t, H 接收完数据处理 因为接收完数据之后,会产生一个idle中断,也就是空闲中断,那么我们就可以在中断服务函数中知道已经接收完了,就可以处理数据了,但是中断服务函数的上下文环境是中断,所以,尽量是快进快出,一般在中断中将一些标志置位,供前台查询。在中断中先判断我们的产生在中断的类型是不是idle中断,如果是则进行下一步,否则就无需理会。 /********************************************************************4 R9 c. |' ?6 K/ R- t! O$ q3 y: J * @brief 串口中断服务函数3 b' Q& v1 A5 f7 c$ o * @author jiejie. S) X: E6 I# D" w6 D9 V1 ]6 e * @version V1.0 * @date 2018-xx-xx ******************************************************************8 { S) \. S" M' u" e. | */ % _& p+ E( U" P- Z# F void DEBUG_USART_IRQHandler(void)4 y) }8 q; Z, o. {7 y9 |" v: B { #if USE_USART_DMA_RX5 U2 w1 F# _: J& Z /* 使用串口DMA */% c; h* S9 E3 `# P. r C$ ~ if(USART_GetITStatus(DEBUG_USARTx,USART_IT_IDLE)!=RESET), W( a& {2 N. U9 u { /* 接收数据 */, r) Q; P4 ?3 Z) d Receive_DataPack(); // 清除空闲中断标志位9 B4 Z; S) R; ^# I; y" ~ USART_ReceiveData( DEBUG_USARTx ); } #else /* 接收中断 */ if(USART_GetITStatus(DEBUG_USARTx,USART_IT_RXNE)!=RESET) _' O( u4 y+ P: @4 Q5 }) j { + g5 f; c* Y2 W' N Receive_DataPack();$ m- c6 x1 _$ B! x }6 B4 ^' Q } #endif } Receive_DataPack() 这个才是真正的接收数据处理函数,为什么我要将这个函数单独封装起来呢?因为这个函数其实是很重要的,因为我的代码兼容普通串口接收与空闲中断,不一样的接收类型其处理也不一样,所以直接封装起来更好,在源码中通过宏定义实现选择接收的方式!更考虑了兼容操作系统的,可能我会在系统中使用dma+空闲中断,所以,供前台查询的信号量就有可能不一样,可能需要修改,我就把它封装起来了。不过无所谓,都是一样的。 /************************************************************* @brief Uart_DMA_Rx_Data+ I& G0 i' L/ f8 N* j4 b: l * @param NULL * @return NULL5 \1 a+ `5 @8 D7 \9 X7 M# I * @author jiejie * @github http://github.com/jiejieTop4 }' q8 _3 e0 L1 m * @date 2018-xx-xx' [1 s4 b. B$ [! [ F9 R: j! J * @version v1.04 t7 _8 r: ?1 ^2 S$ v, r * @note 使用串口 DMA 接收时调用的函数 ***********************************************************/) ^+ C) \7 y- h0 R #if USE_USART_DMA_RX- k6 V% r& f7 k0 [# t void Receive_DataPack(void)& J. ]& u* y6 ?7 K' W; t, _, N { /* 接收的数据长度 */ uint32_t buff_length;1 R/ y' b( U1 h! u- n( G * c6 Y; X( ]8 g. e+ l! H2 G; }) S$ V+ s /* 关闭DMA ,防止干扰 */) R* y4 F- P/ J; [ DMA_Cmd(USART_RX_DMA_CHANNEL, DISABLE); /* 暂时关闭dma,数据尚未处理 */ , H( [2 |! X' c% `" w3 t7 \! V ' c2 w+ F" c2 a# ^% r$ h$ L; { /* 清DMA标志位 */ DMA_ClearFlag( DMA1_FLAG_TC5 ); - p. T4 W& @* h4 t5 Q# { ! J/ k6 G. r3 u* \ /* 获取接收到的数据长度 单位为字节*/ buff_length = USART_RX_BUFF_SIZE - DMA_GetCurrDataCounter(USART_RX_DMA_CHANNEL);7 k4 I5 k$ ?7 H' a6 h /* 获取数据长度 */ Usart_Rx_Sta = buff_length; / _# f( d1 j# r6 _$ f PRINT_DEBUG("buff_length = %d\n ",buff_length); L' V; e, g# o0 m; H /* 重新赋值计数值,必须大于等于最大可能接收到的数据帧数目 */ USART_RX_DMA_CHANNEL->CNDTR = USART_RX_BUFF_SIZE; : j2 y" G8 J- ~0 p* R( X( O /* 此处应该在处理完数据再打开,如在 DataPack_Process() 打开*/ DMA_Cmd(USART_RX_DMA_CHANNEL, ENABLE); /* (OS)给出信号 ,发送接收到新数据标志,供前台程序查询 */0 y+ W4 F' K I5 E& f ) G$ z$ ^2 I0 z' Y9 Y, p, h /* 标记接收完成,在 DataPack_Handle 处理*/* O' e# [3 h3 q4 V- O$ J. d Usart_Rx_Sta |= 0xC000;, N7 {; ]0 @6 f: a8 }3 ]7 Q( T /* 3 ]7 E& H6 ]3 V DMA 开启,等待数据。注意,如果中断发送数据帧的速率很快,MCU来不及处理此次接收到的数据, 中断又发来数据的话,这里不能开启,否则数据会被覆盖。有2种方式解决: 1. 在重新开启接收DMA通道之前,将Rx_Buf缓冲区里面的数据复制到另外一个数组中,. L0 Q! g4 m# H% x 然后再开启DMA,然后马上处理复制出来的数据。- j$ S6 ?/ J1 U9 r ) O) |/ ] B9 f0 T1 N5 M& X 2. 建立双缓冲,重新配置DMA_MemoryBaseAddr的缓冲区地址,那么下次接收到的数据就会 保存到新的缓冲区中,不至于被覆盖。/ R0 u4 V' p# B* n9 Q0 p' W */ } f1使用dma是非常简单的,我在f4用dma的时候也遇到一些问题,最后看手册解决了,打算下一篇文章就写一下调试过程,没有什么是debug不能解决的,如果有,那就两次。今天台风天气,连着舍友的WiFi更新的文章~中国电信还是强,台风天气信号一点都不虚,我的移动卡一动不动-_-. 不设置回复可见,大家一起学习~感谢 本文为杰杰原创如需转载,请说明出处: |
这篇先前看过,好像是正点原子的例子改的,解析接收数据没有启到环形队列特点;比如当你接收到一帧数据时,(USART_RX_STA|=0x8000; //接收完成了);假设CPU处理外设多,没有及时处理这一帧数据,此时串口又来了一帧数据,就会丢数据了,没有用到环形队列缓存功能,
! }% J& U: x) g4 e G$ J. i" A$ G- B# J F
void USART1_IRQHandler(void) //串口1中断服务程序- h( Q- n( g. `/ P. r# s: ^
{
u8 Res;' ^& T/ E$ Z( [1 m; b2 G* X
#if SYSTEM_SUPPORT_OS //如果SYSTEM_SUPPORT_OS为真,则需要支持OS.% _5 `; \. }2 k& ^ o. w0 n/ O6 q
OSIntEnter();
#endif
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断(接收到的数据必须是0x0d 0x0a结尾); V4 z, _ [( o0 V4 I! e
{" }2 g; I3 q' v" h; }
Res =USART_ReceiveData(USART1); //读取接收到的数据
0 _5 r( R% e8 E5 Q2 C
if((USART_RX_STA&0x8000)==0)//接收未完成* N# q- T2 ~$ C- T, O
{
if(USART_RX_STA&0x4000)//接收到了0x0d2 ], ?4 i2 @- Z1 C7 r8 ]
{) N) M) Q) m3 J3 U
if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
else USART_RX_STA|=0x8000; //接收完成了 * d* f2 \+ P( Y
}
else //还没收到0X0D
{
if(Res==0x0d)USART_RX_STA|=0x4000;' ]5 T, f3 y' @! k5 J. o4 l0 A9 F
else. L: u* }9 h/ H
{6 S1 e2 _& }# ]7 k
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;$ m' E F: p7 L" @; `
USART_RX_STA++;
if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收
} " T4 g- B7 @1 ]5 G' d9 [
}
}
} . Q* p* p* D+ W' P0 h8 U
#if SYSTEM_SUPPORT_OS //如果SYSTEM_SUPPORT_OS为真,则需要支持OS.. N' s$ t8 o* c; W% _$ k" }
OSIntExit();
#endif! `# P; ]2 P* f9 ~
}
这个逻辑好像不对,
1、接收数据就是直接放在环形队列中,待CPU有空闲时根据环形队列队头与队尾位置,逐步取出接收到的数据进行处理;
2、如果你每接收到一个数据都去中断判断是否接收一帧数据,会影响CPU的效率,接收中断只接收数据,不做任何数据分析,来一个存一个,减少在中断里处理的时间;待CPU空闲时去环形队列中取数据、解析数据" A6 J; X s. o% Q! e2 I; l7 ~0 I
3、已经定义了一个环形队列,还去定义一个缓冲区好像有点多余,如果CPU没有时间处理那就不去环形队列取数据即可了,空闲时去取不就可以了。: b( K1 r' i! X
这里说的是你发的“STM32进阶之串口环形缓冲区实现”贴子,对于贴子内容是采用环形串口收接,没有说这篇贴子DMA处理;当然有些单片机没有DMA处理,采用环形接收的数据,看到“STM32进阶之串口环形缓冲区实现”这篇贴子,说了一下个人的理解,! t* k* Q" l( A E+ S' D3 W
对于这种环形队列加中断的方式接收数据和楼主探讨如何更高效接收与解析数据的方法。
有
STM32进阶之串口环形缓冲区实现, x5 U! ?! u" J
https://www.stmcu.org.cn/module/ ... amp;fromuid=3250941& R) T& C' X2 t4 r0 R' j+ \, b
(出处: 意法半导体STM32/STM8技术社区)+ _, ~8 x, A& V
你这是接收,跟缓冲区有啥关系啊?: L8 ] _$ K. O5 ]3 k# E, L% _
你接收完如果暂时没法处理可以将数据放到缓冲区去,然后有空再去处理,,,你这仅仅是接收而已,你在哪放入缓冲区呢
我看到我们一同事在用这种代码格式,我想他也写不出这样的格式呀,原来是出自正点原子
f1的dma是没有FIFO的吧,dma接收根本无需CPU去处理的好吗,先理解一下吧,,,,,dma接收的数据是放到指定的缓冲区的,并没有使用环形队列,,,,使用环形队列需要CPU的参与,,这是f1的,你可以在接收完重置dma存放数据的位置,也能起到缓存的作用9 h7 g2 G( l, B+ [8 T' q% R8 a$ I2 v
6 i4 k; }9 x7 z' |1 T* f
支持FIFO的dma无需这样子做。
每来一个字节的数据就进入中断一次,效率太低,,,
dma接收就可以了,够高效了,
不要断章取义,也不要出口成脏,你说的DMA效率高效,不一定每款单片机都有DMA,同时也不防碍讨论环形队列接收数据高效的方法。
没有出口成章啊,在一般应用中数据流没那么快,所以说高效不高效还得看项目需求。如果每一帧的间隔时间比较长,或者数据交互比较不频繁,我觉得用不用都可以
其实,反正我喜欢dma+空闲中断,,,我一般都用32的芯片,如果非要说环形缓冲区的话,,,,Linux的kfifo很经典,我也移植到32了,,到时候可以发出来看看,支持os