代码
#include "at89x52.h"
/***************************************************
* 类型定义,方便代码移植
***************************************************/
typedef unsigned char UINT8;
typedef unsigned int UINT16;
typedef unsigned long UINT32;
typedef char INT8;
typedef int INT16;
typedef long INT32;
typedef bit BOOL;
/***************************************************
* 大量宏定义,便于代码移植和阅读
***************************************************/
//--------------------------------
//----头部----
#define DCMD_CTRL_HEAD1 0x10 //PC下传控制包头部1
#define DCMD_CTRL_HEAD2 0x01 //PC下传控制包头部2
//----命令码----
#define DCMD_NULL 0x00 //命令码:空操作
#define DCMD_CTRL_BELL 0x01 //命令码:控制蜂鸣器
#define DCMD_CTRL_LED 0x02 //命令码:控制LED
#define DCMD_REQ_DATA 0x03 //命令码:请求数据
//----数据----
#define DCTRL_BELL_ON 0x01 //蜂鸣器响
#define DCTRL_BELL_OFF 0x02 //蜂鸣器禁鸣
#define DCTRL_LED_ON 0x03 //LED亮
#define DCTRL_LED_OFF 0x04 //LED灭
//--------------------------------
//----头部----
#define UCMD_CTRL_HEAD1 0x20 //MCU上传控制包头部1
#define UCMD_CTRL_HEAD2 0x01 //MCU上传控制包头部2
//----命令码----
#define UCMD_NULL 0x00 //命令码:空操作
#define UCMD_REQ_DATA 0x01 //命令码:请求数据
#define CTRL_FRAME_LEN 0x04 //帧长度(不包含数据和校验值)
#define PARITY_LEN 0x01 //检验值长度
#define EN_UART() ES=1 //允许串口中断
#define NOT_EN_UART() ES=0 //禁止串口中断
#define BELL(x) {if((x))P0_6=1 ;else P0_6=0;} //蜂鸣器控制宏函数
#define LED(x) {if((x))P2=0x00;else P2=0xFF;}//LED控制宏函数
#define TRUE 1
#define FALSE 0
#define HIGH 1
#define LOW 0
#define ON 1
#define OFF 0
#define NULL (void *)0
/*使用结构体对数据包进行封装
*方便操作数据
*/
typedef struct _PKT_PARITY
{
UINT8 m_ucHead1; //首部1
UINT8 m_ucHead2; //首部2
UINT8 m_ucOptCode; //操作码
UINT8 m_ucDataLength; //数据长度
UINT8 m_szDataBuf[16]; //数据
UINT8 m_ucParity; //校验值为1个字节
}PKT_PARITY;
/*使用共用体再一次对数据包进行封装
*操作数据更加方便
*/
typedef union _PKT_PARITY_EX
{
PKT_PARITY r;
UINT8 p[32];
} PKT_PARITY_EX;
PKT_PARITY_EX PktParityEx; //定义数据包变量
BOOL bLedOn=FALSE; //定义是否点亮LED布尔变量
BOOL bBellOn=FALSE; //定义是否蜂鸣器响布尔变量
BOOL bReqData=FALSE; //定义是否请求数据布尔变量
/******************************************************
*函数名称:OddParity
*输 入:buf 要校验的数据; len 校验数据的长
*输 出:校验值
*功 能:偶校验
*******************************************************/
BOOL OddParity(UINT8 *buf, UINT8 len)
{
UINT8 i,j;
UINT8 data_temp;
BOOL bParity;
bParity = 1;
for(j = 0; j < len;j++)
{
data_temp = *(buf + j);
for(i = 0; i < 8; i++)
{
if((data_temp & 0x01) == 0x01)
{
bParity ^= 1;
}
data_temp = data_temp >> 1;
}
}
return bParity;
}
/*************************************************************
* 函数名称:BufClr
* 输 入:dest 缓冲区; size 缓冲区大小
* 输 出:无
* 说 明:清空缓冲区
**************************************************************/
BOOL BufCpy(UINT8 * dest,UINT8 * src,UINT32 size)
{
if(NULL ==dest || NULL==src ||NULL==size)
{
return FALSE;
}
do
{
*dest++ = *src++;
}while(--size!=0);
return TRUE;
}
/****************************************************
** 函数名称: UartInit
** 输 入: 无
** 输 出: 无
** 功能描述: 串口初始化
*****************************************************/
void UartInit(void)
{
SCON=0x40;
T2CON=0x34;
RCAP2L=0xD9;
RCAP2H=0xFF;
REN=1;
ES=1;
}
/****************************************************
** 函数名称: UARTSendByte
** 输 入: b 单个字节
** 输 出: 无
** 功能描述: 串口 发送单个字节
*****************************************************/
void UARTSendByte(UINT8 b)
{
SBUF=b;
while(TI==0);
TI=0;
}
/****************************************************
** 函数名称: UARTSendByte
** 输 入: b 单个字节
** 输 出: 无
** 功能描述: 串口 发送单个字节
*****************************************************/
void UartSendNBytes(UINT8 *buf,UINT8 len)
{
while(len--)
{
UARTSendByte(*buf++);
}
}
/****************************************************
** 函数名称: main
** 输 入: 无
** 输 出: 无
** 功能描述: 函数主题
*****************************************************/
void main(void)
{
UINT8 i=0;
UINT8 ucCheckSum=0;
UartInit();//串口初始化
EA=1; //开总中断
while(1)
{
if(bLedOn) //是否点亮Led
{
LED(ON);
}
else
{
LED(OFF);
}
if(bBellOn)//是否响蜂鸣器
{
BELL(ON);
}
else
{
BELL(OFF);
}
if(bReqData)//是否请求数据
{
bReqData=FALSE;
NOT_EN_UART(); //禁止串口中断
PktParityEx.r.m_ucHead1=UCMD_CTRL_HEAD1;//MCU上传数据帧头部1
PktParityEx.r.m_ucHead2=UCMD_CTRL_HEAD2;//MCU上传数据帧头部2
PktParityEx.r.m_ucOptCode=UCMD_REQ_DATA;//MCU上传数据帧命令码
PktParityEx.r.m_ucParity=OddParity(PktParityEx.p,
CTRL_FRAME_LEN+
PktParityEx.r.m_ucDataLength);//计算校验值
/*
这样做的原因是因为有时写数据长度不一样,
导致PktParityEx.r.m_ucParity会出现为0的情况
所以使用BufCpy将校验值复制到相应的位置
*/
BufCpy(&PktParityEx.p[CTRL_FRAME_LEN+PktParityEx.r.m_ucDataLength],
&PktParityEx.r.m_ucParity,
PARITY_LEN);
UartSendNBytes(PktParityEx.p,
CTRL_FRAME_LEN+
PktParityEx.r.m_ucDataLength+
PARITY_LEN);//发送数据
EN_UART();//允许串口中断
}
}
}
/****************************************************
** 函数名称: UartIRQ
** 输 入: 无
** 输 出: 无
** 功能描述: 串口中断服务函数
*****************************************************/
void UartIRQ(void)interrupt 4
{
static UINT8 uccnt=0;
UINT8 uclen;
UINT8 ucParity;
if(RI) //是否接收到数据
{
RI=0;
PktParityEx.p[uccnt++]=SBUF;//获取单个字节
if(PktParityEx.r.m_ucHead1 == DCMD_CTRL_HEAD1)//是否有效的数据帧头部1
{
if(uccnt
关键字:通讯协议 奇偶校验
引用地址:51通讯协议—奇偶校验
{
if(uccnt>=2 && PktParityEx.r.m_ucHead2!=DCMD_CTRL_HEAD2)//是否有效的数据帧头部2
{
uccnt=0;
return;
}
}
else
{
uclen=CTRL_FRAME_LEN+PktParityEx.r.m_ucDataLength;//获取数据帧有效长度(不包括校验值)
ucParity=OddParity(PktParityEx.p,uclen);//计算校验值
/*
这样做的原因是因为有时写数据长度不一样,
导致PktParityEx.r.m_ucParity会出现为0的情况
所以使用BufCpy将校验值复制到相应的位置
*/
BufCpy(&PktParityEx.r.m_ucParity,
&PktParityEx.p[uclen],
PARITY_LEN);
if(ucParity!=PktParityEx.r.m_ucParity)//校验值是否匹配
{
uccnt=0;
return;
}
switch(PktParityEx.r.m_ucOptCode)//从命令码中获取相对应的操作
{
case DCMD_CTRL_BELL://控制蜂鸣器命令码
{
if(DCTRL_BELL_ON==PktParityEx.r.m_szDataBuf[0])//数据部分含控制码
{
bBellOn=TRUE;
}
else
{
bBellOn=FALSE;
}
}
break;
case DCMD_CTRL_LED://控制LED命令码
{
if(DCTRL_LED_ON==PktParityEx.r.m_szDataBuf[0])//数据部分含控制码
{
bLedOn=TRUE;
}
else
{
bLedOn=FALSE;
}
}
break;
case DCMD_REQ_DATA://请求数据命令码
{
bReqData=TRUE;
}
break;
}
uccnt=0;
return;
}
}
else
{
uccnt=0;
}
}
}
上一篇:51上移植ucosii的心得
下一篇:51通讯协议—CRC16校验
推荐阅读最新更新时间:2024-03-16 13:48
51单片机定时器0工作在模式2
51单片机定时器0工作在模式2,INT0(P3.2)引脚控制定时/计数器定时计数,定时/计数器0溢出中断,使P2.0引脚相连的LED灯闪烁。定时/计数器0溢出中断为250μs,与P2_0引脚相连的LED灯每2s亮灭一次。源程序如下: //51hei单片机网原创作品,版权所有. #include "reg_c51.h" #define reload_value 0x06 //计数值为250,若时钟频率为12MHz,相当于250μs unsigned char hex ={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90, 0x88,0x83,0xC6,0xA1,0x86,
[单片机]
Keil C51单片机开发心得简介
用Keil写C51程序是最好不过的了。用伟福也可以写,但伟福的编辑能力就相差太远了。个人喜欢用Keil写代码,然后用伟福硬件仿真(只有WAVE的仿真头)。不过Keil是好用,但写代码关键还是C51水平,也就是编程能力,写C程序就离不开模块化这个词了。C51和A51,要扬长避短,发挥C的长处,就得将C模块化好。刚刚开始的时候就只是熟悉C51的语法规则了,熟悉了这个然后再追求程序的可读性和可移植性。 下面是使用Keil C51的一点心得: 1、程序可读性的提高 要提高程序的可读性,就得养成一良好的编程习惯了,例如变量用小写,常量用大写,函数第一个字母用大写等等。这样成习惯以后,自己看代码就会一目了然。另外还要灵活的在KEIL中使用
[单片机]
如何驱动51单片机串口(uart通信)
使用51单片机上uart通信的驱动程序,软件特性可以在proteus上仿真,使用51单片机定时器T1溢出率做uart波特率时基,采用12MHz晶振,uart波特率采用4800。下面直接上代码: uart驱动接口 uart模块向外提供的接口,主要是用于初始化uart和发送数据的接口。 void uart_init(void); // 初始化串口 void uart_deinit(void); // 串口去初始化 void uart_send_data(u8 dat); // 串口发送 void uart_send_num(s32 num); // 串口发送数字 void uart_send_str(string str);
[单片机]
51单片机实现温度采集与显示(二)
下面简单介绍一下DS18B20: DS18B20 是由 DALLAS 半导体公司推出的一种的“一线总线(单总线) ” 接 口的温度传感器。 与传统的热敏电阻等测温元件相比, 它是一种新型的体积小、 适用电压宽、 与微处理器接口简单的数字化温度传感器。 通过P37模拟单总线通信读取数据和发送命令 上面是DS18B20 的示意图,可以看出他与51单片机相连的只有一个管脚——P37,那么二者是如何通信的呢?——采用软件模拟单总线通信 DS18B20 时序包括如下几种: 初始化时序、 写(0 和 1) 时序、 读(0 和 1) 时序。 DS18B20 发送所有的命令和数据都是字节的低位在前。 初始化时序:简单来说,就是主机发送复位
[单片机]
51单片机学习——1--点亮第一个LED
点亮LED 原理: 此单片机的LED灯就是个发光二极管,只能单向通行。图中LED阴极接IO口,阳极接VCC,也就是电源,为高电平,而正常情况下P2口默认输出高电平,所以只需令P2口输出低点平即可。 代码: #include reg52.h sbit led=P2^0;//通过位定义,令led代表P20口 void main() { while(1) { led=0;//令P20为低电平 } } 总结: 知识点: 设置管脚sbit name=Px^x(IO口) 令目标led灯对应的IO口输出低电平 拓展: 也可以直接对P2的所有IO口进行赋值输出。即1111 1111从高位到低位分别对应从左到右
[单片机]
c51: 用定时器T0 ,P0输出1s 方波,中断
/* T0工作方式2,自动重载8位定时器。 设 频率:f, 定时器初值 val ,定时时间 T。 关系式 T=(2^8-val)*12/f */ //任务:f=12MHZ ,T0工作方式2,定时250us,中断2000次,P0口翻转,使P0口输出1s 的方波。 #include reg52.h #define uint unsigned int //定义全局变量 uint counter; init_T0() { TMOD=0x02; //定时器工作方式 TL0=0x06; //时间常数 TH0=0x06; IE=0x82; //开中断 TR0=1; //T
[单片机]
51单片机实现通过串口用计数延时方式发送一串数据
一、使用proteus绘制简单的电路图,用于后续仿真 二、编写程序 /******************************************************************************************************************** ---- @Project: USART ---- @File: main.c ---- @Edit: ZHQ ---- @Version: V1.0 ---- @CreationTime: 20200714 ---- @ModifiedTime: 20200714 ----
[单片机]
基于SOC技术的C8051F020处理器实现动平衡测量系统的应用方案
由于旋转件不平衡量离心力的影响,在转动时,中心惯性主轴与回转轴线不重合,所以惯性力矩或惯性力偶矩的大小与方向会随着机械运动的循环而产生周期性变化,从而使得整个机械系统产生振动。由于振动对机械设备的工作精度、寿命等有很大影响,甚至可能损坏设备,所以大部分的旋转件需要做动平衡。 多数的动平衡测量系统的工作环境比较恶劣,周围存在很多其他设备,电磁和机械干扰可能同时存在,所以对测量系统的抗干扰性等要求更高。所以对现有测试系统的改造势在必行。提高系统集成度,减小系统复杂度,提高系统运算能力将有效解决上述问题。在此基础上我们采用了基于SOC技术的C8051F单片机作为系统核心。由于速度快,功能丰富,可以实现A/D转换、数字采集、操作控制、
[单片机]
小广播
热门活动
换一批
更多
设计资源 培训 开发板 精华推荐
最新单片机文章
更多精选电路图
更多热门文章
更多每日新闻
更多往期活动
11月14日历史上的今天
厂商技术中心
随便看看