[应用相关] STM32串口接收不定长数据原理与源程序

[复制链接]
1107|12
 楼主| 天灵灵地灵灵 发表于 2016-9-26 23:33 | 显示全部楼层 |阅读模式

今天说一下STM32单片机的接收不定长度字节数据的方法。由于STM32单片机带IDLE中断,所以利用这个中断,可以接收不定长字节的数据,由于STM32属于ARM单片机,所以这篇**的方法也适合其他的ARM单片机。

IDLE中断什么时候发生?

IDLE就是串口收到一帧数据后,发生的中断。什么是一帧数据呢?比如说给单片机一次发来1个字节,或者一次发来8个字节,这些一次发来的数据,就称为一帧数据,也可以叫做一包数据。

如何判断一帧数据结束,就是我们今天讨论的问题。因为很多项目中都要用到这个,因为只有接收到一帧数据以后,你才可以判断这次收了几个字节和每个字节的内容是否符合协议要求。

看了前面IDLE中断的定义,你就会明白了,一帧数据结束后,就会产生IDLE中断。这个中断真是太TMD有用了。省去了好多判断的麻烦。


 楼主| 天灵灵地灵灵 发表于 2016-9-26 23:35 | 显示全部楼层

如何配置好IDLE中断?

下面我们就配置好串口IDLE中断吧。

%E4%B8%B2%E5%8F%A3%E6%8E%A5%E6%94%B6%E5%A4%9A%E5%AD%97%E8%8A%821.png

这是串口CR1寄存器,其中,对bit4写1开启IDLE中断,对bit5写1开启接收数据中断。(注意:不同系列的STM32,对应的寄存器位可能不同)

(RXNE中断和IDLE中断的区别?

当接收到1个字节,就会产生RXNE中断,当接收到一帧数据,就会产生IDLE中断。比如给单片机一次性发送了8个字节,就会产生8次RXNE中断,1次IDLE中断。)

%E4%B8%B2%E5%8F%A3%E6%8E%A5%E6%94%B6%E5%A4%9A%E5%AD%97%E8%8A%822.png

这是状态寄存器,当串口接收到数据时,bit5就会自动变成1,当接收完一帧数据后,bit4就会变成1.

需要注意的是,在中断函数里面,需要把对应的位清0,否则会影响下一次数据的接收。比如RXNE接收数据中断,只要把接收到的一个字节读出来,就会清除这个中断。IDLE中断,如何是F0系列的单片机,需要用ICR寄存器来清除,如果是F1系列的单片机,清除方法是“先读SR寄存器,再读DR寄存器”。(我怎么知道?手册上写的)


%E4%B8%B2%E5%8F%A3%E6%8E%A5%E6%94%B6%E5%A4%9A%E5%AD%97%E8%8A%821.png
%E4%B8%B2%E5%8F%A3%E6%8E%A5%E6%94%B6%E5%A4%9A%E5%AD%97%E8%8A%822.png
 楼主| 天灵灵地灵灵 发表于 2016-9-26 23:36 | 显示全部楼层

下面以STM32F103为例给出源程序。

我们先来看程序中的主要部分。

串口初始化函数片段

空闲中断.png

如果你原来的串口初始化函数具有打开串口接收中断的话,实际上就是在初始化函数中多了一条打开空闲中断的语句。

串口中断函数

空闲中断2.png

串口中断函数里面,最重要的两条语句,就是上图中圈出来的两条语句。第一条语句用来判断是否接收到1个字节,第二条语句用来判断是否接收到1帧数据。(是不是感觉超级方便?妈妈再也不用担心我如何判断是否接收完1帧数据了。)

 楼主| 天灵灵地灵灵 发表于 2016-9-26 23:37 | 显示全部楼层

主函数

%E7%A9%BA%E9%97%B2%E4%B8%AD%E6%96%AD3.png

我写的这个主函数,是用来验证接收的正确性的。RxCounter表示的是这一帧数据有几个字节,接收完一帧数据,会在中断函数里面把ReceiveState置1,然后,通过串口把接收到的数据发送回串口。这样,既验证了接收了多少字节的正确性,又验证了接收到的数据是否正确。

%E7%A9%BA%E9%97%B2%E4%B8%AD%E6%96%AD4.png

上图是结果验证。


 楼主| 天灵灵地灵灵 发表于 2016-9-26 23:38 | 显示全部楼层
  1. /**
  2.   ******************************************************************************
  3.   * [url=home.php?mod=space&uid=288409]@file[/url]    串口接收不定长字节数据
  4.   * [url=home.php?mod=space&uid=247401]@brief[/url]   Main program body
  5.   ******************************************************************************
  6.   ******************************************************************************
  7.   */  

  8. /* Includes ------------------------------------------------------------------*/
  9. #include "stm32f10x.h"
  10. #include "uart.h"


  11. volatile uint8_t aRxBuffer[100]={0x00};
  12. volatile uint8_t RxCounter=0;
  13. volatile uint8_t ReceiveState=0;
  14. /**
  15.   * @brief  Main program.
  16.   * @param  None
  17.   * @retval None
  18.   */
  19. int main(void)
  20. {
  21.         uint8_t i=0;
  22.        
  23.         USART1_Init();
  24.        
  25.   while (1)
  26.   {
  27.                 if(ReceiveState==1)//如果接收到1帧数据
  28.                 {
  29.                         ReceiveState=0;
  30.                         i=0;

  31.                         while(RxCounter--)// 把接收到数据发送回串口
  32.                         {
  33.                                 USART_SendData(USART1, aRxBuffer[i++]);       
  34.                                 while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  35.                         }
  36.                         RxCounter=0;
  37.                 }
  38.   }
  39. }



 楼主| 天灵灵地灵灵 发表于 2016-10-9 22:28 | 显示全部楼层
用中断比较好,如果定时器弄不好了很麻烦的。
七颗咖啡豆 发表于 2016-10-10 16:54 | 显示全部楼层
所以我还是没明白,到底多少个字节为一帧。
dongnanxibei 发表于 2016-10-10 21:07 | 显示全部楼层
一帧数据结束后,就会产生IDLE中断。
雷雷 发表于 2016-10-11 20:21 来自手机 | 显示全部楼层
Rangar 发表于 2016-10-13 19:28 | 显示全部楼层
IDLE中断是什么优先级
Bjorn 发表于 2016-10-13 20:45 | 显示全部楼层
总线空闲中断是在检测到在接收数据后,数据总线上一个字节的时间内,没有再接到数据后发生。也就是RXNE位被置位之后,才开始检测,只被置位一次,除非再次检测到RXNE位被置位,然后才开始检测下一次的总线空闲。
LingTian 发表于 2016-10-16 08:47 | 显示全部楼层
 楼主| 天灵灵地灵灵 发表于 2016-10-24 20:51 | 显示全部楼层
IDLE就是串口收到一帧数据后,发生的中断
您需要登录后才可以回帖 登录 | 注册

本版积分规则

183

主题

3475

帖子

13

粉丝
快速回复 在线客服 返回列表 返回顶部