一、使用proteus绘制简单的电路图,用于后续仿真
二、编写程序
/********************************************************************************************************************
---- @Project: USART
---- @File: main.c
---- @Edit: ZHQ
---- @Version: V1.0
---- @CreationTime: 20200703
---- @ModifiedTime: 20200710
---- @Description:
---- 通讯协议:XX YY EB 00 55
---- 其中后三位 EB 00 55就是我所说的数据尾,它的有效数据XX YY在数据尾的前面。
---- 任意时刻,单片机从电脑“串口调试助手”上位机收到的一串数据中,只要此数据中包含关键字EB 00 55 ,并且此关键字前面两个字节的数据XX YY 分别为01 02,那么蜂鸣器鸣叫一声表示接收的数据尾和有效数据都是正确的。
---- 单片机:AT89C52
********************************************************************************************************************/
#include "reg52.h"
/*——————宏定义——————*/
#define FOSC 11059200L
#define BAUD 9600
#define T1MS (65536-FOSC/12/500) /*0.5ms timer calculation method in 12Tmode*/
#define const_voice_short 19 /*蜂鸣器短叫的持续时间*/
#define const_rc_size 10 /*接收串口中断数据的缓冲区数组大小*/
#define const_receive_time 5 /*如果超过这个时间没有串口数据过来,就认为一串数据已经全部接收完,这个时间根据实际情况来调整大小*/
/*——————变量函数定义及声明——————*/
/*蜂鸣器的驱动IO口*/
sbit BEEP = P2^7;
/*LED*/
sbit LED = P3^5;
unsigned int uiSendCnt = 0; /*用来识别串口是否接收完一串数据的计时器*/
unsigned char ucSendLock = 1; /*串口服务程序的自锁变量,每次接收完一串数据只处理一次*/
unsigned int uiRcregTotal = 0; /*代表当前缓冲区已经接收了多少个数据*/
unsigned char ucRcregBuf[const_rc_size]; /*接收串口中断数据的缓冲区数组*/
unsigned int uiRcMoveIndex = 0; /*用来解析数据协议的中间变量*/
unsigned int uiVoiceCnt = 0; /*蜂鸣器鸣叫的持续时间计数器*/
/**
* @brief 定时器0初始化函数
* @param 无
* @retval 初始化T0
**/
void Init_T0(void)
{
TMOD = 0x01; /*set timer0 as mode1 (16-bit)*/
TL0 = T1MS; /*initial timer0 low byte*/
TH0 = T1MS >> 8; /*initial timer0 high byte*/
}
/**
* @brief 串口初始化函数
* @param 无
* @retval 初始化T0
**/
void Init_USART(void)
{
SCON = 0x50;
TMOD = 0x21;
TH1=TL1=-(FOSC/12/32/BAUD);
}
/**
* @brief 外围初始化函数
* @param 无
* @retval 初始化外围
* 让数码管显示的内容转移到以下几个变量接口上,方便以后编写更上一层的窗口程序。
* 只要更改以下对应变量的内容,就可以显示你想显示的数字。
**/
void Init_Peripheral(void)
{
ET0 = 1;/*允许定时中断*/
TR0 = 1;/*启动定时中断*/
TR1 = 1;
ES = 1; /*允许串口中断*/
EA = 1;/*开总中断*/
}
/**
* @brief 初始化函数
* @param 无
* @retval 初始化单片机
**/
void Init(void)
{
LED = 0;
Init_T0();
Init_USART();
}
/**
* @brief 延时函数
* @param 无
* @retval 无
**/
void Delay_Long(unsigned int uiDelayLong)
{
unsigned int i;
unsigned int j;
for(i=0;i for(j=0;j<500;j++) /*内嵌循环的空指令数量*/ { ; /*一个分号相当于执行一条空语句*/ } } } ///** //* @brief 延时函数 //* @param 无 //* @retval 无 //**/ //void Delay_Short(unsigned int uiDelayShort) //{ // unsigned int i; // for(i=0;i // ; /*一个分号相当于执行一条空语句*/ // } //} /** * @brief 串口服务程序 * @param 无 * @retval 在main函数里 * 识别一串数据是否已经全部接收完了的原理: * 在规定的时间里,如果没有接收到任何一个字节数据,那么就认为一串数据被接收完了,然后就进入数据协议 * 解析和处理的阶段。这个功能的实现要配合定时中断,串口中断的程序一起阅读,要理解他们之间的关系。 **/ void usart_service(void) { /*如果超过了一定的时间内,再也没有新数据从串口来*/ if(uiSendCnt >= const_receive_time && ucSendLock == 1) { ucSendLock = 0; /*处理一次就锁起来,不用每次都进来,除非有新接收的数据*/ /*下面的代码进入数据协议解析和数据处理的阶段*/ uiRcMoveIndex = uiRcregTotal; /*由于是判断数据尾,所以下标移动变量从数组的最尾端开始向0移动*/ while(uiRcMoveIndex >= 5) /*如果处理的数据量大于等于5(2个有效数据,3个数据尾)说明还没有把缓冲区的数据处理完*/ { /*数据尾eb 00 55的判断*/ if(ucRcregBuf[uiRcMoveIndex - 3] == 0xeb && ucRcregBuf[uiRcMoveIndex - 2] == 0x00 && ucRcregBuf[uiRcMoveIndex - 1] == 0x55) { /*有效数据01 02的判断*/ if(ucRcregBuf[uiRcMoveIndex - 5] == 0x01 && ucRcregBuf[uiRcMoveIndex - 4] == 0x02) { uiVoiceCnt = const_voice_short; /*蜂鸣器发出声音,说明数据尾和有效数据都接收正确*/ LED = ~LED; /*LED亮灭*/ } break; /*退出循环*/ } uiRcMoveIndex --; /*因为是判断数据尾,下标向着0的方向移动*/ } uiRcregTotal = 0; /*清空缓冲的下标,方便下次重新从0下标开始接受新数据*/ } } /** * @brief 定时器0中断函数 * @param 无 * @retval 无 **/ void ISR_T0(void) interrupt 1 { TF0 = 0; /*清除中断标志*/ TR0 = 0; /*关中断*/ if(uiSendCnt < const_receive_time) /*如果超过这个时间没有串口数据过来,就认为一串数据已经全部接收完*/ { uiSendCnt ++; /*表面上这个数据不断累加,但是在串口中断里,每接收一个字节它都会被清零,除非这个中间没有串口数据过来*/ ucSendLock = 1; /*开自锁标志*/ } if(uiVoiceCnt != 0) { uiVoiceCnt --; BEEP = 0; } else { ; BEEP = 1; } TL0 = T1MS; /*initial timer0 low byte*/ TH0 = T1MS >> 8; /*initial timer0 high byte*/ TR0 = 1; /*开中断*/ } /** * @brief 串口接收数据中断 * @param 无 * @retval 无 **/ void usart_receive(void) interrupt 4 { if(RI == 1) { RI = 0; ++ uiRcregTotal; if(uiRcregTotal > const_rc_size) { uiRcregTotal = const_rc_size; } ucRcregBuf[uiRcregTotal - 1] = SBUF; /*将串口接收到的数据缓存到接收缓冲区里*/ uiSendCnt = 0; /*及时喂狗,虽然main函数那边不断在累加,但是只要串口的数据还没发送完毕,那么它永远也长不大,因为每个中断都被清零。*/ } else { TI = 0; } } /*————————————主函数————————————*/ /** * @brief 主函数 * @param 无 * @retval 实现LED灯闪烁 **/ void main() { /*单片机初始化*/ Init(); /*延时,延时时间一般是0.3秒到2秒之间,等待外围芯片和模块上电稳定*/ Delay_Long(100); /*单片机外围初始化*/ Init_Peripheral(); while(1) { usart_service(); } } 三、仿真实现
上一篇:51单片机实现判断数据头来接收一串数据的串口通用程序框架
下一篇:单片机学习笔记————proteus实现虚拟串口
推荐阅读
史海拾趣
有一种三脚红外接收器件,只要发射940um的脉冲过去,大概在4--10个脉冲什么之后接收端都输出低电平,什么频率都一样,有点象接收头,只是接收头有一固定载波   ...… 查看全部问答∨ |
|
evc如何在一个窗体中创建两种字体? 我在一个窗口中创建了两种字体,结果就所有要显示的字都没有显示出来, 之前我创建了一种字体,还能显示,是不是我没有DELETEOBJECT的缘故?我该如和使用deleteobject()?? 望大侠拔刀相助,。。。。… 查看全部问答∨ |
|
自动调谐算法[1]完成后,将提供两个PID值集合:最小超调量集合和最小设置时间集合。最小超调量集合保护待测器件不受到热伤害,对于在器件最大指定温度附近的温度设定点是非常有益的(参见图1)。对于没有接近最大指定温度的设定点,可以利用最小设 ...… 查看全部问答∨ |
|
怎样实现神州四号开发板与MATLAB的连接,实现利用MATLAB实现:在电脑上显示电压变... 1. 利用stm32f107学习板,熟悉所学案例。2. 在stm32f107学习板上,利用UCGUI实现电子钟,其中时间可通过按键调整。(必做)3. 在上题基础上,利用串口调试小程序,实 ...… 查看全部问答∨ |