版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、搞定单片机多字节串口接收工作了一年多,写了不少单片机串口程序。感觉串口多字节接收部分的逻辑相对于配置寄存器跟串口回复来说,是有点难度的。寄存器配置基本上都是死的,串口回复多字节跟回复一字节只是多了一个循环。串口接收程序是基于串口中断的,单片机的串口每次接收到一字节数据产生一次中断,然后再读取某个寄存器就可以得到串口接收的数据了。然而在实际应用当中,基本上不会有单字节接收的情况。一般都是基于一定串口通信协议的多字节通信。在422或者485通信中,还可能是一个主机(一般是计算机)带多个从机(相应的有单片机的板卡)。这就要求我们的单片机能够在连续接收到的串口数据序列中识别出符合自己板卡对应的通信协议
2、,来进行控制操作,不符合则不进行任何操作。简而言之就是,单片机要在一串数据中找到符合一定规律的几个字节的数据。先来说下怎样定串口协议吧。这个协议指的不是串口底层的协议,而是前面提到的数据帧协议。一般都是有帧头(23个字节吧),数据(长度根据需要),结束位(1位,有时候设计成校验字节,最简单的校验也就是前面所有数据求和)。比如Oxaa0 x55+(数据部分省略)+校验和(除了aa55之外数据的和),如果要是多板卡的话有时候还要在帧头后面加一个板选字节(相当于3字节帧头了)。第一次写串口接收程序的时候,我首先想到的就是定义一个全局变量(实际上最好是定义局部静态变量),初始值设置为0,然后每进一次中
3、断+1,然后加到串口通信协议的长度的时候再清零。然后判断帧头、校验。写完了之后我自己都觉得不对,一旦数据错开了一位,后面就永远都接收不到数了。无奈看了一下前辈们的代码,跟我的思路差不多,只不过那个计数值跟接收到的数据时同时判断的,而且每次中断都要判断,一旦不对计数的那个变量就清零。废话少说,直接上一段代码让大家看看就明白了。(通信协议姑且按照简单的aa55一个字节数据一个字节校验,代码是基于51单片机的)。接收成功则在中断程序中把串口接收成功标志位置1。下面是全局变量定义unsignedcharreceive4=0,0,0,0;接收缓存bituart_flag;串口接收成功标志然后串口中断部分
4、voidser()interrupt4staticunsignedcharcount;/串口接收计数的变量R|=0;手动清某个寄存器,大家都懂的receivecount=SBUF;if(count=0&receivecount=0 xaa)同时判断count跟收到的数据count=1;elseif(count=1&receivecount=0 x55)count=2;elseif(count=2)count+;elseif(count=3&receivecount=receive2)/判断校验和,数据多的话是求和,或者其他的校验方法,也可能是固定的帧尾count=0;uart_flag=1;/
5、串口接收成功标志,为1时在主程序中回复,然后清零ES=O;关中断,回复完了再ES=1;elsecount=0;判断不满足条件就将计数值清零第一次做的串口大概就按照这个方法写完了(我后来看过其他的代码,有人用switch语句写的,逻辑跟这个也差不多,不过我还是感觉用ifelse来写清晰一些),不过在测试的时候发现了bug,如果数据帧发送一半,然后突然停止,再来重新发,就会丢失一帧的数据。比如先接受到aa55,然后断了,再进来aa550101,就不受控制了。后来我也想到一个bug,如果在多设备通信中,属于其他设备的的帧数据最后一位是aa(或者最后两位为aa55,或者最后3位为aa55板选),下一次
6、通信的数据就接收不到了。当时对于数据突然中断的bug,没有想到很好的解决办法,不过这种情况几率极小,所以一直用这个方法写也没有问题。多设备通信最后一位恰好是aa的几率也很小,出问题的可能也很小。当时项目里面的控制数据跟校验恰好不可能出现aa,于是我把if(count=0&receivecount=0 xaa)改成了if(receivecount=0 xaa)其他都没变,解决了,没有bug了。后来我又写了几次单片机程序,才想到了一些解决问题的方法定定不过改天再接着写吧,太累了,明天还要上班呢。在后来的项目中,真的遇到了数据位跟校验位都可能出现aa的情况。我考虑到每次数据都是连续发送的(至少我们用
7、labwindows做的上位机程序是这样的),成功接收到了一帧数据是要有一定时间回复的,也就是说如果接收到一半,但是很长时间没接收到数据,把计数值count清零就ok啦。涉及时间的问题自然要用定时器来实现啦。这次的通信协议如下,串口波特率19200,2个帧头aa55,一个板选,6字节数据,一个校验字节(除帧头外其他数据的和)。全局变量定义unsignedcharboardAddr;/板选地址,通过检测几个io引脚,具体怎么得到的就不写了,很简单的unsignedcharg_DatRev10=0;接收缓存bitretFlag=0;为1代表串口接收到了一帧数据串口初始化函数,晶振22.1184vo
8、idinit_uart()SCON=0 x50;串口方式1允许接收TMOD=0 x21;定时器1,方式2,8位自动重载,同时配置定时器0,工作方式1PCON=0 x80;/波特率加倍TH1=0 xfa;TL1=0 xfa;写入串口定时器初值TH0=(65536-2000)/256;写入定时器0初值,串口传输一个字节时间为(1/19200)*10,计算得0.52msTL0=(65536-2000)%256;/定时器0定时大约1ms多EA=1;ET0=1;波特率:1920022.1184M初值:250(0 xfa)IE|=0 x90;TR1=1;串口中断函数voidUART_INT(void)in
9、terruptstaticunsignedcharcount;/串口接收计数的变量RI=0;g_DatRevcount=SBUF;/帧头if(g_DatRevcount=0 xaa&count=0)count=1;elseif(count=1&g_DatRevcount=0 x55)count=2;_CK=g_DatRevcount;elseif(count=3&count=0&count=2&receivecount=Oxaa),这样就把bug出现的几率降到了非常小,也只是在前一帧结尾数据恰好为aa55板选的时候才出现,几率是多少大家自己算一下吧,呵呵。这样我自己觉得,昨天写的那种方法改进到
10、这个程度,应该算可以啦,反正我是很满意了。实际上我还想过其他的方法,比如缓存的数组采用移位寄存的方式。拿前面的4个字节的协议为例。voidser()interrupt4unsignedchari;RI=0;for(i=0;i3;i+)receivei=receivei+1;receive3=SBUF;if(reveiveO=Oxaa&receive1=0 x55&receive2=receive3)ret_flag=1;ES=0;这段代码看上去可是简单明了,这样判断可是不错啊,同时判断帧头跟校验不会产生前面提到的bug。说实话当时我刚想出这种方法并写出来的时候,马上就被我给否了。那个for循环
11、可真是很占时间的啊,延时函数都是这样写的。每次都循环一下,这延时太长,通信速度太快的话就不能接收到下一字节数据了。最要命的是这个时间的长度是随着通信协议帧的字节数增加而增加的,如果一次要接收几十个字节,肯定就玩完了。这种方法我一次都没用过。不过我居然又想出来了这种方法的改良措施,是前两天刚想出来的,呵呵,还没有实践过呢。下面代码的协议就按第二段程序(定时器清零的那个协议,一共10字节)全局变量bitret_flag;unsignedcharreceive256=0;unsignedcharboardaddress;中断函数voidser()interrupt4staticunsignedcha
12、ri=0;staticunsignedchartotal=0;RI=0;receivei=SBUF;total=total-receivei-7+receivei-1;if(receivei-9=0 xaa&receivei-8=0 x55&receivei-7=boardaddress&receivei=total)ret_flag=1;ES=0;i+;之所以要定义256个长度的数组,就是为了能够让数组首尾相接。因为0-1=255,255+1=0。而且我在计算校验的时候也改进了算法,不会因为数据长度的增加而增加计算校验值的时间。这种方法也是我不久前才想出来的,所以还没有经过实际的验证。上面的
13、代码可能会有逻辑上的错误,如果真有错误,有网友看出来的话,请在下面留言告诉我。这个方法也是我原创的哦,别人也肯能会想到,不过我这个绝对不是抄袭别人的。上面的代码最大的缺点就是变量定义的太多了,太占ram资源了,编译的时候可能会出现错误,毕竟51单片机才128字节的ram(有的资源也很丰富的,比如c8051系列的),这一下子就是256字节的变量。不过对于资源多一些的单片机,这样写还是可以的。要是能有4bit在一起的数据类型就好了,呵呵,verilog代码里面是可以的,C语言里貌似不行啊。要想能在例如51单片机上运行,只能按照下面的折中方式了,也就是把i相关的量都与一个0 x0f全局变量bitret_flag;unsignedcharreceive16=0;可以考虑在定义时加上idata,毕竟还可能是32或者64长度的数组呢unsignedcharidatareceive16=0;unsignedcharboardaddress;中断函数voidser()interrupt4staticunsignedchari=0;staticunsignedchartotal=0;RI=0;receivei&0 x0f=SBUF;total=total-receive(i-7)&0 x0f+recei
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 郑州汽车工程职业学院《软件测试》2023-2024学年第一学期期末试卷
- 浙江大学《管理研究方法与科研写作》2023-2024学年第一学期期末试卷
- 漳州职业技术学院《MATAB应用》2023-2024学年第一学期期末试卷
- 升级硬件设施推动数据中心转型
- 保险创新产品发布汇报模板
- 双十一金融风控模板
- 专业基础-房地产经纪人《专业基础》名师预测卷3
- 企业文化讲座
- 农学研究实战解读
- 教师助人为乐先进事迹材料
- 教育部中国特色学徒制课题:基于中国特色学徒制的新形态教材建设与应用研究
- 2025年护理质量与安全管理工作计划
- (T8联考)2025届高三部分重点中学12月第一次联考评物理试卷(含答案详解)
- 工程施工扬尘防治教育培训
- 红薯采购合同模板
- 2023年河南省公务员录用考试《行测》真题及答案解析
- 2024年安徽省公务员录用考试《行测》真题及答案解析
- 山西省太原市重点中学2025届物理高一第一学期期末统考试题含解析
- 充电桩项目运营方案
- 2024年农民职业农业素质技能考试题库(附含答案)
- 高考对联题(对联知识、高考真题及答案、对应练习题)
评论
0/150
提交评论