51串口接收函数.docx
- 文档编号:5096233
- 上传时间:2022-12-13
- 格式:DOCX
- 页数:9
- 大小:20.67KB
51串口接收函数.docx
《51串口接收函数.docx》由会员分享,可在线阅读,更多相关《51串口接收函数.docx(9页珍藏版)》请在冰豆网上搜索。
51串口接收函数
51串口接收函数
篇一:
搞定单片机多字节串口接收
搞定单片机多字节串口接收
工作了一年多,写了不少单片机串口程序。
感觉串口多字节接收部分的逻辑相对于配置寄存器跟串口回复来说,是有点难度的——寄存器配置基本上都是死的,串口回复多字节跟回复一字节只是多了一个循环。
串口接收程序是基于串口中断的,单片机的串口每次接收到一字节数据产生一次中断,然后再读取某个寄存器就可以得到串口接收的数据了。
然而在实际应用当中,基本上不会有单字节接收的情况。
一般都是基于一定串口通信协议的多字节通信。
在422或者485通信中,还可能是一个主机(一般是计算机)带多个从机(相应的有单片机的板卡)。
这就要求我们的单片机能够在连续接收到的串口数据序列中识别出符合自己板卡对应的通信协议,来进行控制操作,不符合则不进行任何操作。
简而言之就是,单片机要在一串数据中找到符合一定规律的几个字节的数据。
先来说下怎样定串口协议吧。
这个协议指的不是串口底层的协议,而是前面提到的数据帧协议。
一般都是有帧头(2~3个字节吧),数据(长度根据需要),结束位(1位,有时候设计成校验字节,最简单的校验也就是前面所有数据求和)。
比如0xaa0x55+(数据部分省略)+校验和(除了aa55之外数据的和),如果要是多板卡的话有时候还要在帧头后面加一个板选字节(相当于3字节帧头了)。
第一次写串口接收程序的时候,我首先想到的就是定义一个全局变量(实际上最好是定义局部静态变量),初始值设置为0,然后每进一次中断+1,然后加到串口通信协议的长度的时候再清零。
然后判断帧头、校验。
写完了之后我自己都觉得不对,一旦数据错开了一位,后面就永远都接收不到数了。
无奈看了一下前辈们的代码,跟我的思路差不多,只不过那个计数值跟接收到的数据时同时判断的,而且每次中断都要判断,一旦不对计数的那个变量就清零。
废话少说,直接上一段代码让大家看看就明白了。
(通信协议姑且按照简单的aa55一个字节数据一个字节校验,代码是基于51单片机的)。
接收成功则在中断程序中把串口接收成功标志位置1。
下面是全局变量定义
unsignedcharreceive[4]={0,0,0,0};//接收缓存
bituart_flag;//串口接收成功标志
然后串口中断部分
voidserinterrupt4
{
staticunsignedcharcount;//串口接收计数的变量
RI=0;//手动清某个寄存器,大家都懂的
receive[count]=SBUF;
if//同时判断count跟收到的数据
{
count=1;
}
elseif
{
count=2;
}
elseif
{
count++;
}
elseif//判断校验和,数据多的话是求//和,或者其他的校验方法,也可能是固定的帧尾
{
count=0;
uart_flag=1;//串口接收成功标志,为1时在主程序中回复,然后清零
ES=0;//关中断,回复完了再ES=1;
}
else
{
count=0;//判断不满足条件就将计数值清零
}
}
第一次做的串口大概就按照这个方法写完了(我后来看过其他的代码,有人用switch语句写的,逻辑跟这个也差不多,不过我还是感觉用ifelse来写清晰一些),
不过在测试的时候发现了bug,如果数据帧发送一半,然后突然停止,再来重新发,就会丢失一帧的数据。
比如先接受到aa55,然后断了,再进来aa550101,就不受控制了。
后来我也想到一个bug,如果在多设备通信中,属于其他设备的的帧数据最后一位是aa(或者最后两位为aa55,或者最后3位为aa55板选),下一次通信的数据就接收不到了。
当时对于数据突然中断的bug,没有想到很好的解决办法,不过这种情况几率极小,所以一直用这个方法写也没有问题。
多设备通信最后一位恰好是aa的几率也很小,出问题的可能也很小。
当时项目里面的控制数据跟校验恰好不可能出现aa,于是我把if改成了if其他都没变,解决了,没有bug了。
后来我又写了几次单片机程序,才想到了一些解决问题的方法——不过改天再接着写吧,太累了,明天还要上班呢。
在后来的项目中,真的遇到了数据位跟校验位都可能出现aa的情况。
我考虑到每次数据都是连续发送的(至少我们用labwindows做的上位机程序是这样的),成功接收到了一帧数据是要有一定时间回
复的,也就是说如果接收到一半,但是很长时间没接收到数据,把计数值count清零就ok啦。
涉及时间的问题自然要用定时器来实现啦。
这次的通信协议如下,串口波特率19200,2个帧头aa55,一个板选,6字节数据,一个校验字节(除帧头外其他数据的和)。
全局变量定义
unsignedcharboardAddr;//板选地址,通过检测几个io引脚,具体怎么得到的就不写了,很简单的unsignedcharg_DatRev[10]={0};//接收缓存
bitretFlag=0;//为1代表串口接收到了一帧数据
串口初始化函数,晶振22.1184
voidinit_uart
{
SCON=0x50;//串口方式1允许接收
TMOD=0x21;//定时器1,方式2,8位自动重载,同时配置定时器0,工作方式1PCON=0x80;//波特率加倍
TH1=0xfa;
TL1=0xfa;//写入串口定时器初值
TH0=/256;//写入定时器0初值,串口传输一个字节时间为(1/19200)*10,计算得0.52ms
TL0=%256;//定时器0定时大约1ms多
EA=1;
ET0=1;//波特率:
1920022.1184M初值:
250
IE|=0x90;
TR1=1;
}
串口中断函数
voidUART_INTinterrupt4
{
staticunsignedcharcount;//串口接收计数的变量
RI=0;
g_DatRev[count]=SBUF;
if//帧头
{
count=1;
}
else
if
{
count=2;
}
elseif
{
CK=g_DatRev[count];
count=3;
}
elseif
{
CK+=g_DatRev[count];
count++;
}
elseif
{
ES=0;
retFlag=1;
count=0;
}
else
{
count=0;
}
resettimer;
}
//判断count不为0的话就启动定时器
voidresettimer
{
TR0=0;
TH0=/256;
TL0=%256;
if
{
TR0=1;
}
}
定时器中断函数
voidT0_timeinterrupt1
{
TR0=0;
TH0=/256;
TL0=%256;
count=0;
}
这种方法的确是本人自己想出来的,别人可能也这样做过,但我这个绝对不是抄袭或者模仿来的。
这样写的确可以避免前面提到过的bug,不过代价是多用了一个定时器的资源,而且中断函数里的内容更多了,占用了更多的时间。
要是能把第一种方法改进一下就好了,主要是那个校验不能为aa的那个bug,因为毕竟传输到一半突然断了的可能性是非常小的。
后来我想第一个判断if好像有点太严格了,考虑到第二字节的帧头,跟板选地址不可能为aa,于是把这个改写为if,这样就把bug出现的几率降到了非常小,也只是在前一帧结尾数据恰好为aa55板选的时候才出现,几率是多少大家自己算一下吧,呵呵。
这样我自己觉得,昨天写的那种方法改进到这个程度,应该算可以啦,反正我是很满意了。
实际上我还想过其他的方法,比如缓存的数组采用移位寄存的方式。
拿前面的4个字节的协议为例。
voidserinterrupt4
{
unsignedchari;
RI=0;
for
{
receive[i]=receive[i+1];
}
篇二:
51单片机串口接收的函数模块
51单片机串口接收的函数模块.txt机会就像秃子头上一根毛,你抓住就抓住了,抓不住就没了。
我和你说了10分钟的话,但却没有和你产生任何争论。
那么,我们之间一定有个人变得虚伪无比!
过错是短暂的遗憾,错过是永远的遗憾。
相遇是缘,相知是份,相爱是约定,相守才是真爱。
标题:
51单片机串口接收的函数模块Zz
发信站:
郁金香BBS站,站内信件
//**************************************************************************//本函数模块来自"51单片机世界",作者版主丁丁(聂小猛)。
//主页地址/
//本程序为本站推出的"MCU51-63K仿真器"及"DX-51多功能试验板"而设计的,详情请到本站查看.
//"51单片机世界"版权所有,未经允许,不得抄袭作为商业用途
//**************************************************************************//串口中断服务程序,仅需做简单调用即可完成串口输入输出的处理
//串口输入设有缓冲区,大小可任意设置。
//可供使用的函数名:
/*
externchargetbyte;//从接收缓冲区取一个byte,如不想等待则在调用前检测inbufsign是否为1。
externputbyte;//放入一个字节到发送缓冲区
externputstring;//发送一个字符串到串口
externputhex;//发送一个字节的hex码,分成两个字节发。
externbitinbufsign;
#defineCRputstring//发送一个回车换行
externvoidserial_init;
externputint;
*/
//*************************************************************************#include
#defineucharunsignedchar
#defineuintunsignedint
#defineILEN16//32//设置串口中断接收缓冲区大小
idataucharinbuf[ILEN];
dataucharidata*inlast=inbuf;//最后由中断进入接收缓冲区的字节位置
dataucharidata*getlast=inbuf;//最后取走的字节位置
bitinbufsign;//接收缓冲区非空标志有=1
bitinbufful;//输入缓冲区满标志满=1
#defineCRputstring//CR=回车换行
//*****************************
//放入一个字节到发送缓冲区
putbyte
{
SBUF=c;
while;TI=0;
}
//***************************************
//发送一个定义在程序存储区的字符串到串口
putstring
{
for//遇到停止符0结束
putbyte;
}
//*************************************
//发送一个字节的hex码,分成两个字节发。
ucharcodehex_[]={"0123456789ABCDEF"};
puthex
{
intch;
ch=&0x0f;
putbyte;
ch=c&0x0f;
putbyte;
}
//*************************************
//从接收缓冲区取一个byte,如不想等待则在调用前检测inbufsign是否为1。
uchargetbyte
{
charidatac;
while;//缓冲区空等待
ES=0;
c=*getlast;//取数据
getlast++;//最后取走的数据位置加一
inbufful=0;//输入缓冲区的满标志清零
ifgetlast=inbuf;//地址到顶部回到底部
ifinbufsign=0;//地址相等置接收缓冲区空空标志,再取数前要检该标志
ES=1;
return;//取回数据
}
/*
//*************************
//以十进制的形式输出一个整型数到串口
putint
{
uchari[6];
i[0]=0x30+n/10000;
i[1]=0x30+/1000;
i[2]=0x30+/100;
i[3]=0x30+/10;
i[4]=0x30+n%10;
i[5]=0;
putstring;
}
//*********************
//模拟接收到一个字符,
putinbuf
{ES=0;if
{*inlast=c;
inlast++;
ifinlast=inbuf;
ifinbufful=1;
inbufsign=1;
}
ES=1;
}
*/
//*****************************************
//串口中断处理
serialinterrupt4
{
if
{
RI=0;
if
{
*inlast=SBUF;//放入数据
inlast++;//最后放入的位置加一
inbufsign=1;
ifinlast=inbuf;//地址到顶部回到底部
ifinbufful=1;//接收缓冲区满置满标志
}
}
}
//*****************************
//串口初始化0xfd=19200,0xfa=9600,0xf4=4800,0xe8=2400,0xd0=1200110592
voidserial_init
{
//用t1作波特率
SCON=0x50;//mode1:
8-bitUART,enablereceiverTMOD|=0x21;//timer1mode2:
8-Bitreloadth01X16PCON|=0x00;//SMOD=0
TH1=0xfd;//baud*1/*reloadvalue9600baudTL1=0xfd;
TR1=1;//timer1run
//用t2作波特率
/*
T2CON=0x30;
RCAP2H=0xFF;
RCAP2L=0xb8;//70=4800,b8=9600,dc=19200,ee=38400,f4=57600,fa=115200TR2=1;ET2=1;
SCON=0x50;//SCON=0xd0;
*/
ES=1;
REN=1;
EA=1;
SM2=1;//SM2=1时收到的第9位为1才置位RI标志
TI=1;
}
篇三:
单片机C51串口中断接收和发送范例
//这是一个单片机C51串口接收(中断)和发送例程,可以用来测试51单片机的中断接收
//和查询发送,另外我觉得发送没有必要用中断,因为程序的开销是一样
#include
#include
#defineINBUF_LEN4//数据长度
unsignedcharinbuf1[INBUF_LEN];
unsignedcharchecksum,count3;
bitread_flag=0;
voidinit_serialxxm
{
SCON=0x50;//SCON:
serailmode1,8-bitUART,enableucvrTMOD|=0x20;//TMOD:
timer1,mode2,8-bitreloadPCON|=0x80;//SMOD=1;
TH1=0xF4;//Baud:
4800fosc="11".0592MHz
IE|=0x90;//EnableSerialInterrupt
TR1=1;//timer1run
//TI="1";
}
//向串口发送一个字符
voidsend_char_xx
{
SBUF="ch";
while;
TI=0;
}
//向串口发送一个字符串,strlen为该字符串长度
voidsend_string_xx{
unsignedintk=0;
do
{
send_char_xx);
k++;
}while;
}
//串口接收中断函数
voidserialinterrupt4using3
{
if
{
unsignedcharch;
RI=0;
ch="SBUF";
if
{
count3=0;
inbuf1[count3]=ch;
checksum=ch-128;
}
else
{
count3++;
inbuf1[count3]=ch;
checksum^=ch;
if)&&)
{
read_flag=1;//如果串口接收的数据达到INBUF_LEN个,且校验没错,
//就置位取数标志
}
}
}
}
main
{
init_serialxxm;//初始化串口
while
{
if//如果取数标志已置位,就将读到的数从串口发出{
read_flag=0;//取数标志清0
send_string_xx;
}
}
}
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 51 串口 接收 函数