一、使用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 uiSetData3 = 0; /*本程序中需要被设置的参数3*/
unsigned int uiSetData4 = 0; /*本程序中需要被设置的参数4*/
unsigned char ucTemp1 = 0; /*中间过渡变量*/
unsigned char ucTemp2 = 0; /*中间过渡变量*/
unsigned char ucTemp3 = 0; /*中间过渡变量*/
unsigned char ucTemp4 = 0; /*中间过渡变量*/
void Dig_Hc595_Drive(unsigned char, unsigned char);
/*根据原理图得出的共阴数码管字模表*/
code unsigned char Dig_Table[] =
{
0x3f, /*0 序号0*/
0x06, /*1 序号1*/
0x5b, /*2 序号2*/
0x4f, /*3 序号3*/
0x66, /*4 序号4*/
0x6d, /*5 序号5*/
0x7d, /*6 序号6*/
0x07, /*7 序号7*/
0x7f, /*8 序号8*/
0x6f, /*9 序号9*/
0x00, /*不显示 序号10*/
0x40, /*- 序号11*/
0x73, /*P 序号12*/
};
/**
* @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)
{
ucDigDot8 = 0;
ucDigDot7 = 0;
ucDigDot6 = 0;
ucDigDot5 = 0;
ucDigDot4 = 0;
ucDigDot3 = 0;
ucDigDot2 = 0;
ucDigDot1 = 0;
ET0 = 1;/*允许定时中断*/
TR0 = 1;/*启动定时中断*/
TR1 = 1;
ES = 1; /*允许串口中断*/
EA = 1;/*开总中断*/
}
/**
* @brief 初始化函数
* @param 无
* @retval 初始化单片机
**/
void Init(void)
{
LED = 0;
BEEP = 1;
Key_Gnd = 0;
Dig_Hc595_Drive(0x00, 0x00); /*关闭所有经过另外两个74HC595驱动的LED灯*/
Init_T0();
Init_USART();
/*
* 为了保证串口中断接收的数据不丢失,必须设置IP = 0x10,相当于把串口中断设置为最高优先级,
* 这个时候,串口中断可以打断任何其他的中断服务函数实现嵌套,
*/
IP = 0x10; /*把串口中断设置为最高优先级,必须的。*/
}
/**
* @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 ucSendData * @retval 在发送一串数据中,每个字节之间必须添加一个延时,用来等待串口发送完成。 * 不增加延时,单单靠发送完成标志位来判断还是容易出错,在51,PIC单片机中都是这么做。 * 在stm32单片机中,可以不增加延时,直接靠单片机自带的标志位来判断就很可靠。 **/ void eusart_send(unsigned char ucSendData) { ES = 0; /*关串口中断*/ TI = 0; /*清零串口发送完成中断请求标志*/ SBUF = ucSendData; /*发送一个字节*/ Delay_Short(400); /*每个字节之间的延时,这里非常关键,也是最容易出错的地方。延时的大小请根据实际项目来调整*/ TI = 0; /*清零串口发送完成中断请求标志*/ ES = 1; /*允许串口中断*/ } /** * @brief 一发一收的通讯服务程序 * @param 无 * @retval 无 **/ void communication_service(void) { unsigned int i; if(ucStatus == 1) /*处于正在通讯的过程中*/ { switch(ucSendStep) { case 0: /*通讯过程0 发送一串数据*/ switch(ucSendTotal) /*根据当前已经发送到第几条数据来决定发送哪些参数*/ { case 0: /*发送参数1*/ ucSendregBuf[0] = 0xeb; ucSendregBuf[1] = 0x00; ucSendregBuf[2] = 0x55; ucSendregBuf[3] = 0x01; ucSendregBuf[4] = 0x00; ucSendregBuf[5] = 0x02; ucSendregBuf[6] = uiSetData1 >> 8; /*把int类型的参数分解成两个字节的数据*/ ucSendregBuf[7] = uiSetData1; break; case 1: /*发送参数2*/ ucSendregBuf[0] = 0xeb; ucSendregBuf[1] = 0x00; ucSendregBuf[2] = 0x55; ucSendregBuf[3] = 0x02; ucSendregBuf[4] = 0x00; ucSendregBuf[5] = 0x02; ucSendregBuf[6] = uiSetData2 >> 8; /*把int类型的参数分解成两个字节的数据*/ ucSendregBuf[7] = uiSetData2; break; case 2: /*发送参数3*/ ucSendregBuf[0] = 0xeb; ucSendregBuf[1] = 0x00; ucSendregBuf[2] = 0x55; ucSendregBuf[3] = 0x03; ucSendregBuf[4] = 0x00; ucSendregBuf[5] = 0x02; ucSendregBuf[6] = uiSetData3 >> 8; /*把int类型的参数分解成两个字节的数据*/ ucSendregBuf[7] = uiSetData3;
上一篇:51单片机实现利用AT24C02进行掉电后的数据保存
下一篇:51单片机实现从机的串口收发
推荐阅读
史海拾趣
最后一家EPC公司深知创新是企业发展的核心动力。公司注重研发投入和人才培养,建立了专业的研发团队和完善的研发体系。公司不断跟踪行业发展趋势和技术动态,积极引进新技术、新工艺和新材料,推动产品升级和产业升级。同时,公司注重创新成果的转化和应用,将新技术、新工艺和新材料应用于实际项目中,提高了项目的竞争力和附加值。通过持续的创新和积累,公司逐渐在电子行业中形成了独特的技术优势和核心竞争力。
请注意,以上故事仅为框架性描述,并未包含具体的公司名称、项目细节和具体数字。您可以根据这些框架性描述进一步深入了解相关公司的发展情况,并结合实际案例进行具体分析和撰写。
为了进一步提升公司的竞争力和影响力,Caltron Devices Ltd公司积极实施国际化战略。公司加强与国外同行的交流与合作,引进先进的技术和管理经验,不断提升自身的创新能力。同时,Caltron还积极参加国际电子展会和论坛,展示公司的最新产品和技术成果,成功打开了全球市场的大门。
面对日益严峻的环境问题,Computer Conversions Corp积极响应环保号召,开始研发更加环保的计算机转换技术。公司不仅优化了生产工艺,减少了生产过程中的能源消耗和废弃物排放,还推出了一系列节能型转换设备,帮助客户在提升计算效率的同时,也降低了能源消耗。这一举措不仅赢得了客户的赞赏,也提升了公司的社会形象。
面对竞争激烈的半导体市场,Enable Semiconductor Corp公司明智地选择了细分市场作为突破口。公司通过对市场的深入调研,发现智能手机、可穿戴设备等领域对高性能半导体产品的需求日益增长。于是,公司及时调整产品策略,加大在这些领域的研发和市场投入。这一战略布局使得公司能够快速响应市场需求,赢得客户的信赖。
面对电子行业技术的快速迭代和市场竞争的加剧,Esc始终保持创新驱动的发展策略。公司不断加大对研发创新的投入力度,推动产品升级和产业升级。通过引入新技术、新工艺和新材料,Esc成功开发出一系列具有竞争力的新产品,并在市场上取得了显著的成绩。同时,公司还积极参与行业标准的制定和推广工作,推动整个电子行业的进步和发展。
现在要安装wince5.0,在msdn上下了一个装了 http://www.microsoft.com/downloads/details.aspx?familyid=4A4ED1F4-91D3-4DBE-986E-A812984318E5&displaylang=en 结果在vs的设备管理器中没有模拟器只是添加了一个项 看sdk下载页面说明“The Win ...… 查看全部问答∨ |
|
我在之前的系统中添加T9IME输入法编译时出现如下错误: BUILD: [01:0000001021:ERRORE] NMAKE : U1073: don\'t know how to make \'INCLUDES=\' BUILD: [01:0000001023:ERRORE] NMAKE.EXE -i -c BUILDMSG=Stop.   ...… 查看全部问答∨ |
来看看S3C6410在1024x768分辨率下的显示效果(立宇泰ARMSYS6410开发板) [img=http://www.hzlitai.com.cn/uploads/userup/0911/0414442K017.jpg[/img] 详情请看: http://blog.eeworld.net/sinpo1984/archive/2009/11/04/4767487.aspx… 查看全部问答∨ |
各位大虾好: 我是为实现8051F020上的串口0通讯,其中数据接收用中断的形式。 由于我期望的是,自定义6个字节数据为一帧,当收到一帧数据后,接收中断RI0置位,将数据读走。并同时保证下一字节数据的完整性。 ...… 查看全部问答∨ |
TATCL=TASSEL_1+MC_1;这条语句是不是相当于TACTL=TASSEL0+MC0;?因为头文件定义为:#define TASSEL0 (0x0100u) /* Timer A clock source select 1 */#define MC0&nb ...… 查看全部问答∨ |
今天参加了 TI Hercules 和 M4 的研讨会,由于上次我参加过 M4 的,所以这次被安排参加 Hercules 的研讨会。 Hercules 关键字:安全。 主要应用在医疗电子,工业控制和交通运输对安全性有严格要求的领域。 基于 65 nm 制程。最高主频 220Mhz,有 ...… 查看全部问答∨ |
|
如题,本人第一次做DSP的项目,目前在学2812,书看了2遍,对各个模块初步有了个了解,各块的程序也大致看了。现在已经开始着手弄硬件这块了,导师AD这块的要求是16位精度,然后处理速度还不能太慢,求大侠介绍有哪些可以考虑的外接AD芯片,价格不要 ...… 查看全部问答∨ |