单片机学习笔记————51单片机实现主机的串口收发

本文介绍了如何在proteus环境中实现51单片机的串口通信,包括绘制电路图、编写程序及进行仿真实验,详细阐述了主机的串口收发过程。
摘要由CSDN通过智能技术生成

proteus虚拟串口的实现:https://mp.csdn.net/console/editor/html/107251649

一、使用proteus绘制简单的电路图,用于后续仿真

 

二、编写程序

/********************************************************************************************************************
----	@Project:	USART
----	@File:	main.c
----	@Edit:	ZHQ
----	@Version:	V1.0
----	@CreationTime:	20200720
----	@ModifiedTime:	20200720
----	@Description:	实现功能:
----	显示和独立按键部分根据数码管显示的程序来改编,用S1,S5,S9,S13作为独立按键。
---- 	     一共有4个窗口。每个窗口显示一个参数。有两种更改参数的方式:
----	第一种:按键更改参数:
----	    第8,7,6,5位数码管显示当前窗口,P-1代表第1个窗口,P-2代表第2个窗口,P-3代表第3个窗口,P-4代表第1个窗口。
----	    第4,3,2,1位数码管显示当前窗口被设置的参数。范围是从0到9999。S1是加按键,按下此按键会依次增加当前窗口的参数。S5是减按键,按下此按键会依次减少当前窗口的参数。S9是切换窗口按键,按下此按键会依次循环切换不同的窗口。S13是启动发送数据和复位按键,当系统处于待机状态时,按下此按键会启动发送数据;当通讯超时蜂鸣器报警时,可以按下此键清除报警,返回到待机的状态。
----	
----	第二:通过串口把更改的参数发送给从机。
----	波特率是:9600.
----	通讯协议:EB 00 55  GG 00 02 XX XX  CY
----	其中第1,2,3位EB 00 55就是数据头
----	其中第4位GG就是数据类型。01代表更改参数1,02代表更改参数2,03代表更改参数3,04代表更改参数4,
----	其中第5,6位00 02就是有效数据长度。高位在左,低位在右。
----	其中从第7,8位XX XX是被更改的参数。高位在左,低位在右。
----	第9位CY是累加和,前面所有字节的累加。
----	一个完整的通讯必须发送完4串数据,每串数据之间的间隔时间不能超过10秒钟,否则认为通讯超时主机会重发数据,如果连续三次都没有返回,则引发蜂鸣器报警。如果接收到得数据校验正确,主机继续发送新的一串数据,直到把4串数据发送完毕为止。
----	
----	   系统处于待机状态时,LED灯一直亮,
----	   系统处于非待机状态时,LED灯闪烁,
----	   系统处于出错状态时,LED灯闪烁,并且蜂鸣器间歇鸣叫报警。
----	
----	通过电脑的串口助手来模拟从机,返回不同的应答
----	从机返回校验正确应答:eb 00 55 f5 00 00 35
----	从机返回校验出错应答:eb 00 55 fa 00 00 3a
----	单片机: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_key_time1 9	/*按键去抖动延时的时间*/
#define	const_key_time2 9	/*按键去抖动延时的时间*/
#define	const_key_time3 9	/*按键去抖动延时的时间*/
#define	const_key_time4 9	/*按键去抖动延时的时间*/
#define const_led_0_5s 32	/*大概0.5秒的时间*/
#define const_led_1s 64	/*大概1秒的时间*/
#define const_send_time_out 640	/*通讯超时出错的时间 大概10秒*/
#define const_rc_size 20	/*接收串口中断数据的缓冲区数组大小*/
#define const_receive_time 5	/*如果超过这个时间没有串口数据过来,就认为一串数据已经全部接收完,这个时间根据实际情况来调整大小*/
#define const_send_size 10	/*串口发送数据的缓冲区数组大小*/

#define const_voice_short 20	/*蜂鸣器短叫的持续时间*/

/*——————变量函数定义及声明——————*/
/*蜂鸣器的驱动IO口*/
sbit BEEP = P2^7;
/*LED*/
sbit LED = P3^5;

/*按键*/
sbit Key_S1 = P0^0;	/*对应S1键,加键*/
sbit Key_S2 = P0^1;	/*对应S5键,减键*/
sbit Key_S3 = P0^2;	/*对应S9键,切换窗口*/
sbit Key_S4 = P0^3;	/*对应S13键,复位*/
sbit Key_Gnd = P0^4;

/*数码管*/
sbit Dig_Hc595_Sh = P2^0;
sbit Dig_Hc595_St = P2^1;
sbit Dig_Hc595_Ds = P2^2;

unsigned char ucSendregBuf[const_send_size];   /*发送的缓冲区数组*/
unsigned int uiSendCnt = 0; /*用来识别串口是否接收完一串数据的计时器*/
unsigned char ucSendLock = 1;   /*串口服务程序的自锁变量,每次接收完一串数据只处理一次*/
unsigned int uiRcregTotal = 0;  /*代表当前缓冲区已经接收了多少个数据*/
unsigned char ucRcregBuf[const_rc_size];    /*接收串口中断数据的缓冲区数组*/
unsigned int uiRcMoveIndex = 0; /*用来解析数据协议的中间变量*/
unsigned char ucSendCntLock = 0;    /*串口计时器的原子锁*/
unsigned char ucRcType = 0; /*数据类型*/
unsigned int uiRcSize = 0; /*数据长度*/
unsigned char ucRcCy = 0;   /*校验累加和*/
unsigned char ucLedLock = 0;	/*原子锁*/
unsigned int uiLedCnt = 0;  /*控制Led闪烁的延时计时器*/
unsigned int uiSendTimeOutCnt = 0;  /*用来识别接收数据超时的计时器*/
unsigned char ucSendTimeOutLock = 0;    /*原子锁*/

unsigned char ucStatus = 0; /*当前状态变量 0代表待机 1代表正在通讯过程 2代表发送出错*/
unsigned char ucSendStep = 0;	/*发送数据的过程步骤*/
unsigned char ucErrorCnt = 0;	/*累计错误总数*/
unsigned char ucSendTotal = 0;	/*记录当前已经发送了多少串数据*/
unsigned char ucReceiveStatus = 0;	/*返回的数据状态 0代表待机 1代表校验正确 2代表校验出错*/

unsigned char ucKeySec = 0; /*被触发的按键编号*/
unsigned int uiKeyTimeCnt1 = 0; /*按键去抖动延时计数器*/
unsigned char ucKeyLock1 = 0;   /*按键触发后自锁的变量标志*/
unsigned int uiKeyTimeCnt2 = 0; /*按键去抖动延时计数器*/
unsigned char ucKeyLock2 = 0;   /*按键触发后自锁的变量标志*/
unsigned int uiKeyTimeCnt3 = 0; /*按键去抖动延时计数器*/
unsigned char ucKeyLock3 = 0;   /*按键触发后自锁的变量标志*/
unsigned int uiKeyTimeCnt4 = 0; /*按键去抖动延时计数器*/
unsigned char ucKeyLock4 = 0;   /*按键触发后自锁的变量标志*/

unsigned int uiVoiceCnt = 0;    /*蜂鸣器鸣叫的持续时间计数器*/
unsigned char ucVoiceLock = 0;  /*蜂鸣器鸣叫的原子锁*/

unsigned char ucDigShow8;   /*第8位数码管要显示的内容*/
unsigned char ucDigShow7;   /*第7位数码管要显示的内容*/
unsigned char ucDigShow6;   /*第6位数码管要显示的内容*/
unsigned char ucDigShow5;   /*第5位数码管要显示的内容*/
unsigned char ucDigShow4;   /*第4位数码管要显示的内容*/
unsigned char ucDigShow3;   /*第3位数码管要显示的内容*/
unsigned char ucDigShow2;   /*第2位数码管要显示的内容*/
unsigned char ucDigShow1;   /*第1位数码管要显示的内容*/

unsigned char ucDigDot8;   /*数码管8的小数点是否显示的标志*/
unsigned char ucDigDot7;   /*数码管7的小数点是否显示的标志*/
unsigned char ucDigDot6;   /*数码管6的小数点是否显示的标志*/
unsigned char ucDigDot5;   /*数码管5的小数点是否显示的标志*/
unsigned char ucDigDot4;   /*数码管4的小数点是否显示的标志*/
unsigned char ucDigDot3;   /*数码管3的小数点是否显示的标志*/
unsigned char ucDigDot2;   /*数码管2的小数点是否显示的标志*/
unsigned char ucDigDot1;   /*数码管1的小数点是否显示的标志*/

unsigned char ucDigShowTemp = 0;	/*临时中间变量*/
unsigned char ucDisplayDriveStep = 1; /*动态扫描数码管的步骤变量*/

unsigned char ucWd1Update = 1;	/*窗口1更新显示标志*/
unsigned char ucWd2Update = 0;	/*窗口2更新显示标志*/
unsigned char ucWd3Update = 0;	/*窗口3更新显示标志*/
unsigned char ucWd4Update = 0;	/*窗口4更新显示标志*/
unsigned char ucWd = 1;	/*本程序的核心变量,窗口显示变量。类似于一级菜单的变量。代表显示不同的窗口。*/
unsigned int uiSetData1 = 0;	/*本程序中需要被设置的参数1*/
unsigned int uiSetData2 = 0;	/*本程序中需要被设置的参数2*/
unsigned int uiSetDat
1.基本任务 (1)已知甲机 8 个开关,乙机 8 个发光二极管,利用它们的串口方式 1,波 特率自定义,实现:将甲机中 8 个开关所代表的数据传送到乙机,并在乙机的 8 个 LED 灯显示。请在 Proteus 中画出电路原理图,并编写程序仿真实现上述功能。 (2)已知单片机的 P0 口了 8 个发光二极管 LED0~LED7,现要求通过单片机串口收发上位机的命令,实现对这 8 个发光二极管的控制。PC 端采用串口调试程序进 行数据发送(如使用 stc-isp 烧写软件向单片机发送“88 FB AF XX FC FC”6 个字节的 命令,其中“88 FB AF”及“FC FC”为数据的帧头和帧尾,“XX”为 00~07 数据。 )单片机 使用串口中断进行数据接收,同时需要判断帧头和帧尾的正确性。判断帧头和帧尾完毕 后,若正确的话再判断“XX” 数据,对应“XX” 数据对 LED0~LED7 进行点亮、熄灭控 制;若不正确丢掉数据, 转入等待接收。 请在 Proteus 中画出电路原理图,并编写程 序仿真实现上述功能。 2.拓展任务 在以上基本任务 1 的基础上,奇校验,实现甲机和乙机的全双工通信,即甲机和乙 机都分别 8 个开关和 8 个发光二极管,甲机 8 个开关所代表的数据能传送到乙机并在 乙机的 8 个 LED 灯显示,同时乙机 8 个开关所代表的数据能传送到甲机并在甲机的 8 个 LED 灯显示,若校验出错则指示灯(自定义)闪烁。请在 Proteus 中画出电路原理图, 并编写程序仿真实现上述功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值