单片机多字节串口接收(转)

  工作了一年多,写了不少单片机串口程序。感觉串口多字节接收部分的逻辑相对于配置寄存器跟串口回复来说,是有点难度的——寄存器配置基本上都是死的,串口回复多字节跟回复一字节只是多了一个循环。

  串口接收程序是基于串口中断的,单片机的串口每次接收到一字节数据产生一次中断,然后再读取某个寄存器就可以得到串口接收的数据了。然而在实际应用当中,基本上不会有单字节接收的情况。一般都是基于一定串口通信协议的多字节通信。在422或者485通信中,还可能是一个主机(一般是计算机)带多个从机(相应的有单片机的板卡)。这就要求我们的单片机能够在连续接收到的串口数据序列中识别出符合自己板卡对应的通信协议,来进行控制操作,不符合则不进行任何操作。简而言之就是,单片机要在一串数据中找到符合一定规律的几个字节的数据。

  先来说下怎样定串口协议吧。这个协议指的不是串口底层的协议,而是前面提到的数据帧协议。一般都是有帧头(2~3个字节吧),数据(长度根据需要),结束位(1位,有时候设计成校验字节,最简单的校验也就是前面所有数据求和)。

  比如0xaa 0x55 +(数据部分省略)+校验和(除了aa 55

  之外数据的和),如果要是多板卡的话有时候还要在帧头后面加一个板选字节(相当于3字节帧头了)。

  第一次写串口接收程序的时候,我首先想到的就是定义一个全局变量(实际上最好是定义局部静态变量),初始值设置为0,然后每进一次中断+1,然后加到串口通信协议的长度的时候再清零。然后判断帧头、校验。写完了之后我自己都觉得不对,一旦数据错开了一位,后面就永远都接收不到数了。无奈看了一下前辈们的代码,跟我的思路差不多,只不过那个计数值跟接收到的数据时同时判断的,而且每次中断都要判断,一旦不对计数的那个变量就清零。

  废话少说,直接上一段代码让大家看看就明白了。(通信协议姑且按照简单的aa 55 一个字节数据

  一个字节校验,代码是基于51单片机的)。接收成功则在中断程序中把串口接收成功标志位置1。

  下面是全局变量定义

  unsigned char receive[4]={0,0,0,0};//接收缓存

  bit uart_flag;//串口接收成功标志

  然后串口中断部分

  ```c

  void ser()interrupt 4

  {

  static unsigned char count;//串口接收计数的变量

  RI=0;//手动清某个寄存器,大家都懂的

  receive[count]=SBUF;

  if(count==0&&receive[count]==0xaa)//同时判断count跟收到的数据

  {

  count=1;

  }

  else if(count==1&&receive[count]==0x55)

  {

  count=2;

  }

  else if(count==2)

  {

  count++;

  }

  else if(count==3&&receive[count]== receive

  [2])//判断校验和,数据多的话是求//和,或者其他的校验方法,也可能是固定的帧尾

  {

  count=0;

  uart_flag =1;//串口接收成功标志,为1时在主程序中回复,然后清零

  ES=0; //关中断,回复完了再ES=1;

  }

  else

  {

  count=0;//判断不满足条件就将计数值清零

  }

  }

  ```

  第一次做的串口大概就按照这个方法写完了(我后来看过其他的代码,有人用switch语句写的,逻辑跟这个也差不多,不过我还是感觉用if

  else来写清晰一些),

  不过在测试的时候发现了bug,如果数据帧发送一半,然后突然停止,再来重新发,就会丢失一帧的数据。比如先接受到aa 55,然后断了,再进来aa 55

  01 01,就不受控制了。后来我也想到一个bug,如果在多设备通信中,属于其他设备的的帧数据最后一位是aa(或者最后两位为aa 55 ,或者最后3位为aa 55

  板选),下一次通信的数据就接收不到了。

  当时对于数据突然中断的bug,没有想到很好的解决办法,不过这种情况几率极小,所以一直用这个方法写也没有问题。多设备通信最后一位恰好是aa的几率也很小,出问题的可能也很小。当时项目里面的控制数据跟校验恰好不可能出现aa,于是我把if(count==0&&receive[count]==0xaa)改成了if(receive[count]==0xaa)其他都没变,解决了,没有bug了。

  后来我又写了几次单片机程序,才想到了一些解决问题的方法——不过改天再接着写吧,太累了,明天还要上班呢。

  在后来的项目中,真的遇到了数据位跟校验位都可能出现aa的情况。我考虑到每次数据都是连续发送的(至少我们用labwindows做的上位机程序是这样的),成功接收到了一帧数据是要有一定时间回复的,也就是说如果接收到一半,但是很长时间没接收到数据,把计数值count清零就ok啦。涉及时间的问题自然要用定时器来实现啦。

  这次的通信协议如下,串口波特率19200,2个帧头aa 55 ,一个板选,6字节数据,一个校验字节(除帧头外其他数据的和)。

  全局变量定义

  unsigned char boardAddr;//板选地址,通过检测几个io引脚,具体怎么得到的就不写了,很简单的

  unsigned char g_DatRev [10]={0};//接收缓存

  bit retFlag=0;//为1代表串口接收到了一帧数据

  串口初始化函数,晶振22.1184

  ```c

  void init_uart()

  {

  SCON = 0x50; //串口方式1允许接收

  TMOD = 0x21; //定时器1,方式2,8位自动重载,同时配置定时器0,工作方式1

  PCON = 0x80; // 波特率加倍

  TH1 = 0xfa;

  TL1 = 0xfa; //写入串口定时器初值

  TH0=(65536-2000)/256; //写入定时器0初值,串口传输一个字节时间为(1/19200)*10,计算得0.52ms

  TL0=(65536-2000)%256; //定时器0定时大约1ms多

  EA=1;

  ET0=1; //波特率:19200 22.1184M 初值:250(0xfa)

  IE |= 0x90;

  TR1 = 1;

  }

  ```

  串口中断函数

  ```c

  void UART_INT(void) interrupt 4

  {

  static unsigned char count;//串口接收计数的变量

  RI = 0;

  g_DatRev[count] = SBUF;

  if(g_DatRev[count]==0xaa&&count==0) //帧头

  {

  count=1;

  }

  else if(count==1&&g_DatRev[count]==0x55)

  {

  count=2;

  }

  else if (count==2&&g_DatRev[2] == boardAddr)

  {

  CK = g_DatRev[count];

  count=3;

  }

  else if(count>=3&&count<9)

  {

  CK += g_DatRev[count];

  count ++;

  }

  else if(count == 9&&CK==g_DatRev[9])

  {

  ES = 0;

  retFlag = 1;

  count=0;

  }

  else

  {

  count=0;

  }

  resettimer();

  }

  //判断count不为0的话就启动定时器

  void resettimer()

  {

  TR0=0;

  TH0=(65536-2000)/256;

  TL0=(65536-2000)%256;

  if(count!=0)

  {

  TR0=1;

  }

  }

  定时器中断函数

  void T0_time()interrupt 1

  {

  TR0=0;

  TH0=(65536-2000)/256;

  TL0=(65536-2000)%256;

  count=0;

  }

  ```

  这种方法的确是本人自己想出来的,别人可能也这样做过,但我这个绝对不是抄袭或者模仿来的。这样写的确可以避免前面提到过的bug,不过代价是多用了一个定时器的资源,而且中断函数里的内容更多了,占用了更多的时间。

  要是能把第一种方法改进一下就好了,主要是那个校验不能为aa的那个bug,因为毕竟传输到一半突然断了的可能性是非常小的。后来我想第一个判断if(count==0&&receive[count]==0xaa)好像有点太严格了,考虑到第二字节的帧头,跟板选地址不可能为aa,于是把这个改写为if(count>=0&&count<=2&&

  receive[count]==0xaa),这样就把bug出现的几率降到了非常小,也只是在前一帧结尾数据恰好为 aa 55 板选

  的时候才出现,几率是多少大家自己算一下吧,呵呵。这样我自己觉得,昨天写的那种方法改进到这个程度,应该算可以啦,反正我是很满意了。

  实际上我还想过其他的方法,比如缓存的数组采用移位寄存的方式。拿前面的4个字节的协议为例。

  ```c

  void ser()interrupt 4

  {

  unsigned char i;

  RI=0;

  for(i=0;i<3;i++)

  {

  receive[i]=receive[i+1];

  }

  receive[3]=SBUF;

  if(reveive[0]==0xaa&&receive[1]==0x55&&receive[2]==receive[3])

  {

  ret_flag=1;

  ES = 0;

  }

  }

  ```

  这段代码看上去可是简单明了,这样判断可是不错啊,同时判断帧头跟校验不会产生前面提到的bug。说实话当时我刚想出这种方法并写出来的时候,马上就被我给否了。那个for循环可真是很占时间的啊,延时函数都是这样写的。每次都循环一下,这延时太长,通信速度太快的话就不能接收到下一字节数据了。最要命的是这个时间的长度是随着通信协议帧的字节数增加而增加的,如果一次要接收几十个字节,肯定就玩完了。这种方法我一次都没用过。

  不过我居然又想出来了这种方法的改良措施,是前两天刚想出来的,呵呵,还没有实践过呢。

  下面代码的协议就按第二段程序(定时器清零的那个协议,一共10字节)

  全局变量

  bit ret_flag;

  unsigned char receive[256]={0};

  unsigned char boardaddress;

  中断函数

  ```c

  void ser()interrupt 4

  {

  static unsigned char i=0;

  static unsigned char total=0;

  RI=0;

  receive[i]=SBUF;

  total=total-receive[i-7]+receive[i-1];

  if(receive[i-9]==0xaa&&receive[i-8]==0x55

  &&receive[i-7]==boardaddress&&receive[i]==total

  )

  {

  ret_flag=1;

  ES = 0;

  }

  i++;

  }

  ```

  之所以要定义256个长度的数组,就是为了能够让数组“首尾相接”。因为0 -1 = 255 , 255+1 =

  0。而且我在计算校验的时候也改进了算法,不会因为数据长度的增加而增加计算校验值的时间。这种方法也是我不久前才想出来的,所以还没有经过实际的验证。上面的代码可能会有逻辑上的错误,如果真有错误,有网友看出来的话,请在下面留言告诉我。这个方法也是我原创的哦,别人也肯能会想到,不过我这个绝对不是抄袭别人的。

  上面的代码最大的缺点就是变量定义的太多了,太占ram资源了,编译的时候可能会出现错误,毕竟51单片机才128字节的ram(有的资源也很丰富的,比如c8051系列的),这一下子就是256字节的变量。不过对于资源多一些的单片机,这样写还是可以的。要是能有4bit在一起的数据类型就好了,呵呵,verilog代码里面是可以的,C语言里貌似不行啊。

  要想能在例如51单片机上运行,只能按照下面的折中方式了,也就是把i相关的量都与一个0x0f

  全局变量

  bit ret_flag;

  unsigned char receive[16]={0};// 可以考虑在定义时加上idata,毕竟还可能是32

  //或者64长度的数组呢unsigned char idata receive[16]={0};

  unsigned char boardaddress;

  中断函数

  ```c

  void ser()interrupt 4

  {

  static unsigned char i=0;

  static unsigned char total=0;

  RI=0;

  receive[i&0x0f]=SBUF;

  total=total-receive[(i-7)&0x0f]+receive[(i-1)&0x0f];

  if(receive[(i-9)&0x0f]==0xaa&&receive[(i-8)&0x0f]==0x55

  &&receive[(i-7)&0x0f]==boardaddress&&receive[i&0x0f]==total

  )

  {

  ret_flag=1;

  ES = 0;

  }

  i++;

  }

  ```

  stm32相关视频学习资料

  (stm32 USART串口应用)

  http://www.makeru.com.cn/live/1392_1164.html?s=45051

  基于STM32讲解串口操作

  http://www.makeru.com.cn/live/1758_490.html?s=45051

  通过Z-stack协议栈实现串口透传

  http://www.makeru.com.cn/live/1758_330.html?s=45051

  通信协议 - UART串口协议

  http://www.makeru.com.cn/live/3576_1437.html?s=45051

  stm32之SPI通信

  http://www.makeru.com.cn/live/3523_1795.html?s=45051

  SPI通信协议驱动norFlash

  http://www.makeru.com.cn/live/4034_2151.html?s=45051

原文地址:https://www.cnblogs.com/923327iu/p/12367654.html

时间: 2024-08-25 10:53:22

单片机多字节串口接收(转)的相关文章

【转】搞定单片机多字节串口接收(串口多字节接收发送的高阶研究)

搞定单片机多字节串口接收(串口多字节接收发送的高阶研究) 原文地址:http://bbs.ednchina.com/BLOG_ARTICLE_3007162.HTM 工作了一年多,写了不少单片机串口程序.感觉串口多字节接收部分的逻辑相对于配置寄存器跟串口回复来说,是有点难度的——寄存器配置基本上都是死的,串口回复多字节跟回复一字节只是多了一个循环. 串口接收程序是基于串口中断的,单片机的串口每次接收到一字节数据产生一次中断,然后再读取某个寄存器就可以得到串口接收的数据了.然而在实际应用当中,基本

AVR单片机教程——串口接收

上一讲中,我们实现了单片机开发板向电脑传输数据.在这一讲中,我们将通过电脑向单片机发送指令,让单片机根据指令控制LED.这一次,两端的TX与RX需要交叉连接,单片机TX连接串口工具RX也是需要的,因为程序会根据指令反馈信息. 为了简单起见,我们的程序只需要控制4个板载LED.指令包含两个字节:第一个字节为r.y.g.b中的一个,分别表示红.黄.绿.蓝灯:第二个字节为0或1,表示灯不亮或亮. 然而,a2这样的指令是没有意义的,却是可能出现的.即使用户已经熟悉了这8条指令,也可能会不小心打错.我们应

51单片机之串口通信(三)

51单片机之串口通信(三) 已有 47 次阅读2015-12-29 00:21 |个人分类:51单片机| 单片机, 通信 用串口实现发送和接收同时可操作: 电脑显示结果如图: 源程序: /*项目名称:串口发送接收实验项目内容:串口发送接收同时进行,计算机发送数据给单片机控制P0口,从而控制LED灯的亮灭,单片机发送数据到计算机,计算机显示出来时间:2015年12月23日项目负责人:YUAN*/#include <reg52.h>typedef unsigned char uChar8;type

51单片机之串口通信(一)

一.基础知识 1.串行通信和并行通信:目前用的比较多的是串行通信.串行通信优点是连接简单,传输距离远:缺点是传输速度慢. 2.串行通信:分为同步通信和异步通信:异步通信是指发送和接收设备利用各自的时钟控制数据的发送和接收. 3.串行通信的传输方向:单工,半双工,全双工. 4.波特率:每秒钟传输2进制代码的位数,如1个字节为10位,每秒传输240个字节,则波特率为10*240=2400bps,单位是bps: 5.RS-232C:25个管脚,我们用到的是PGND(保护接地).TXD(发送数据),RX

stm32的串口接收字符串以十六进制数

#include "pbdata.h" uint8_t TxBuffer1[] = "USART Interrupt Example: This isUSART1 DEMO"; uint8_t RxBuffer1[],rec_f,tx_flag; volatile uint8_t TxCounter1 = 0x00; volatile uint8_t RxCounter1 = 0x00; uint32_t Rec_Len; int main(void) { u8 a

2018最新mfc作为上位机接收硬件端USB或串口数据显示成图片 解决串口接收数据丢字节丢包问题

本文用的是VS2013MFC写串口数据接收: 第一步:首先建立一个MFC工程,成功后会跳出一个对话框,直接在对话框上点击右键->点击插入ACTIVAE控件->选择MicrosoftCommunications Control, version 6.0 成功后会显示一个电话的图标在对话框上,运行起来不会显示的 不用担心这个美观问题.如果没有这个插件的话,可能是版本太低  可以自己下载一个补上 第二步:大概的窗体搞好:   那个显示图片的大框是PICTURE控件变量 然后就要项目->类向导中

018_STM32程序移植之_串口接收中文

(一)在平时数据传输中很少用到接收中文的情况,但是最近需要用到就花了半天时间来弄弄 (二)接收原理,从现在接收情况分析:一个中文占两个数据的空间,也就是两个十六进制可以转化成为一个中文 (三)示例情况,用Hex Editor来看看中文 "你好",可以看到四个十六进制数据:0xc4,0xe3,0xba,0xc3 (四)我们的目的还是单片机通过串口来进行数据的接收,用CH340短接T和R看看发送"你好"也是会接收到十六进制0xc4,0xe3,0xba,0xc3 (五)从

AVR单片机教程——串口发送

到目前为止,我们的开发板只能处理很小量的数据:读取几个引脚电平,输出几个LED,顶多用数码管显示一个两位数字.至于输入一个指令.输出一条调试信息,甚至用scanf和printf来输入输出,在已经接触过的这些器件上是难以想象的.而本讲"串口发送"与下一讲"串口接收",将打开这一扇大门. 硬件 本讲的主题是UART(Universal Asynchronous Receiver-Transmitter,通用异步收发器),俗称串口.实际上串口是串行接口的统称,在单片机领域

基础问题:在一个 Activity 中定义的串口接收程序,如果 Activity 切换到其它 Activity 后还能接收到串口数据吗?

============问题描述============ RT:基础问题:在一个 Activity 中定义的串口接收程序,如果 Activity 切换到其它 Activity 后还能接收到串口数据吗? 我的程序有两个 Activity,在启动后的 Activity 中已经验证了接收与发送数据. 后继做了第二个 Activity,此时从第二个 Activity 返回 第一个 Activity 时串口的接收线程会出错.调试了一下,串口再次被初始化了. 问题: (1)不想串口被反复初始化,应该如何做?