本帖最后由 wulin 于 2018-11-12 12:47 编辑
串口接收中断每发生一次只能接收一个字节数据,如果一帧数据是8个字节有效数据,至少要增加一个数据头和一个数据尾,数据头用于判断起始位,数据尾用于判断结束位和验证一帧数据的正确性。
当收到第一个字节保存在缓存m=SBUF;,验证缓存m是否正确,正确继续接收并保存,否则抛弃等待正确数据头。当一帧数据接收完毕,设置一个标志,中断任务结束。主函数查询中断接收完毕标志后,解析处理和回传数据的任务全部放在主函数里完成。给你一个P2.5传感器串口发收数据串的程序参考。
//AA 00 10 00 4B 5B FF 串口助手发送模拟的P2.5传感器数据串
#include <reg51.h>
#define uint unsigned int
#define uchar unsigned char
sbit dula=P2^6; //段选
sbit wela=P2^7; //位选
uchar code table[]={ //共阴数码管0~F数组
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
uchar table0[] ="OK\n"; //用于串口助手返回验证
uchar table1[]="ERROR\n";//用于串口助手返回验证
uchar rec_buf[7]; //缓存
uint V_data; //收到的16位有效数据变量
uint swan,wan,qian,bai,shi,ge;//数码管显示位
bit flag=0; //接收完成标
/*************初始化串口**************/
void InitUART() //9600bps@11.0592MHz
{
PCON &= 0x7F; //波特率不倍速
SCON = 0x50; //8位数据,可变波特率
TMOD= 0x20; //设定定时器1为8位自动重装方式
TL1 = 0xFD; //设定定时初值
TH1 = 0xFD; //设定定时器重装值
ET1 = 0; //禁止定时器1中断
TR1 = 1; //启动定时器1
EA = 1; //开总中断
ES = 1; //开串口中断
}
/**********串口发送函数*************/
void SendOneByte(uchar c)
{
SBUF = c; //发送数据
while(!TI); //等待发送完成
TI = 0; //发送中断请求标志位清0
}
/************数据解析程序*************/
void analysis()
{
uchar i,j; //临时变量
if(flag==1) //一帧(7字节)数据串接收完成
{
ES=0; //关串口中断
flag=0; //接收完成标志清0
j=rec_buf[1]+rec_buf[2]+rec_buf[3]+rec_buf[4];//数据和
if(rec_buf[5]==j)//验证数据和
{
V_data=(rec_buf[1]<<8)|rec_buf[2];//恢复16位有效数据
for(i=0;i<3;i++)
SendOneByte(table0);//返回OK
}
else for(i=0;i<6;i++)
SendOneByte(table1);//返回ERROR
ES=1; //开串口中断
}
}
/*************数据分解***************/
void Transformation()
{
swan = V_data/100000; // 十万位
wan = V_data%100000/10000; // 万位
qian = V_data%10000/1000; // 千位
bai = V_data%1000/100; // 百位
shi = V_data%100/10; // 十位
ge = V_data%10; // 个位
}
/**********6位数码管显示程序************/
void display()
{
static uchar k=0; //分时显示变量
P0=0x00; //消隐
dula=1;
dula=0;
switch(k)
{
case 0:
P0=0x7e; //显示十万位码
wela=1;
wela=0;
P0=table[swan]; //显示十万位段码
dula=1;
dula=0;
k++;
break;
case 1:
P0=0x7d; //显示万位位码
wela=1;
wela=0;
P0=table[wan]; //显示万位段码
dula=1;
dula=0;
k++;
break;
case 2:
P0=0x7b; //显示千位位码
wela=1;
wela=0;
P0=table[qian]; //显示千位段码
dula=1;
dula=0;
k++;
break;
case 3:
P0=0x77; //显示百位位码
wela=1;
wela=0;
P0=table[bai]; //显示百位段码
dula=1;
dula=0;
k++;
break;
case 4:
P0=0xef; //显示十位位码
wela=1;
wela=0;
P0=table[shi]; //显示十位段码
dula=1;
dula=0;
k++;
break;
case 5:
P0=0xdf; //显示个位位码
wela=1;
wela=0;
P0=table[ge]; //显示个位段码
dula=1;
dula=0;
k=0;
break;
}
}
/**************主程序**************/
void main()
{
InitUART(); //初始化串口
while(1)
{
analysis(); //数据解析
Transformation();//数据分解
display(); //数码管显示
}
}
/*********串口中断服务程序**********/
void UARTInterrupt() interrupt 4
{
static uchar num=0; //静态计数变量
RI=0; //接收中断请求标志位清0
rec_buf[num]=SBUF; //接收到的数据串保存在缓存数组
if(rec_buf[0]==0xAA) //验证数据头(起始位)
{
num++;
if(num>=7)
{
flag=1; //接收完成标志置1
num=0; //计数变量清0
}
}
}
|