本科毕业设计--基于atmega16单片机的电子时钟设计
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
基于ATmega16单片机的电子时钟设计
物理与电子工程学院电子信息科学与技术专业(职教)2009级罗德龙
指导教师宋培森
摘要:随着科学技术的发展电子时钟在日常的生活中应用的相当的多。
本文介绍了一款基于ATmega16单片机为主控芯片的数字电子时钟设计。
其中主要包括ATmega16单片机、独立键盘、DS1302芯片、LCD1602液晶显示、以及相关外围电路并在PROTUES仿真平台上运行。
关键词:ATmega16单片机;DS1302芯片;独立键盘;LCD1602
Abstract:With the development of science and technology electronic clock in the daily life of the application is quite long. This paper introduces a single chip processor as the main control chip based on ATmega16 digital electronic clock design. Include ATmega16 microcontroller, independent keyboard, DS1302 chip LCD1602, liquid crystal display (LCD), and related peripheral circuit and in PROTUES simulation platform operation.
Key words:ATmega16 microcontroller;DS1302 chip;Independent keyboard;LCD1602 1 前言
液晶显示器以其微功耗、体积小、显示内容丰富、超薄轻巧等许多优点,在袖珍式仪表和低功耗应用系统中得到越来越广泛的应用。
本文中详细介绍了基于ATmega16单片机控制下的162液晶屏显示设计,此设计基于ds1302的时钟电路方便实用,电路设计简单。
2 AVR单片机介绍
2.1 AVR单片机简介
AVR单片机是1997年由ATMEL公司研发出的增强型内置Flash的RISC精简指令集高速8位单片机。
AVR的单片机可以广泛应用于计算机外部设备、工业实时控制、仪表仪器、通讯设备、家用电器等各个领域。
1997年,由Atmel公司挪威设计中心的A先生和V先生,利用Atmel公司的Flash新技术,共同研发出RISC精简指令集高速8位单片机,简称AVR。
2.2 Atmega16简介
ATmega16是基于增强的AVR RISC结构的低功耗8 位CMOS微控制器。
由于其先进的指令集以及单时钟周期指令执行时间,ATmega16 的数据吞吐率高达 1 MIPS/MHz,从而可以减缓系统在功耗和处理速度之间的矛盾。
ATmega16 AVR 内核具有丰富的指令集和32 个通用工作寄存器。
所有的寄存器都直接与运算逻单元(ALU) 相连接,使得一条指令可以在一个时钟周期内同时访问两个独立的寄存器。
这种结构大大提高了代码效率,并且具有比普通的CISC 微控制器最高至10 倍的数据吞吐率。
ATmega16 有如下特点:16K字节的系统内可编程Flash(具有同时读写的能力),512 字节EEPROM,1K 字节SRAM,32 个通用I/O 口线,32 个通用工作寄存器,用于边界扫描的JTAG 接口,支持片内调试与编程,三个具有比较模式的灵活的定时器/ 计数器(T/C),片内/外中断,可编程串行USART,有起始条件检测器的通用串行接口,8路10位具有可选差分输入级可编程增益的ADC ,具有片内振荡器的可编程看门狗定时器,一个SPI 串行端口,以及六个可以通过软件进行选择的省电模式。
2.3 ATmega16产品特性
高性能、低功耗的8位AVR微处理器
先进的RISC 结构
131条指令
大多数指令执行时间为单个时钟周期
32个8位通用工作寄存器
全静态工作
工作于16MHz时性能高达16MIPS
只需两个时钟周期的硬件乘法器
非易失性程序和数据存储器
16K 字节的系统内可编程Flash,擦写寿命: 10,000次
具有独立锁定位的可选Boot代码区,通过片上Boot程序实现系统内编程,真正的同时读写操作
512 字节的EEPROM,擦写寿命: 100,000次
1K字节的片内SRAM
可以对锁定位进行编程以实现用户程序的加密
JTAG 接口( 与IEEE 1149.1 标准兼容)
符合JTAG 标准的边界扫描功能
支持扩展的片内调试功能
通过JTAG 接口实现对Flash、EEPROM、熔丝位和锁定位的编程外设特点
两个具有独立预分频器和比较器功能的8位定时器/计数器
一个具有预分频器、比较功能和捕捉功能的16位定时器/计数器
具有独立振荡器的实时计数器RTC
四通道PWM
8路10位ADC,8个单端通道,2个具有可编程增益(1x, 10x, 或200x)的差分通道
面向字节的两线接口
两个可编程的串行USART
可工作于主机/ 从机模式的SPI 串行接口
具有独立片内振荡器的可编程看门狗定时器
片内模拟比较器
特殊的处理器特点
上电复位以及可编程的掉电检测
片内经过标定的RC振荡器
片内/片外中断源
6种睡眠模式: 空闲模式、ADC 噪声抑制模式、省电模式、掉电模式、Standby 模式以及扩展的Standby模式 u I/O和封装
32个可编程的I/O口
2.4 工作电压:
ATmega16L:2.7 - 5.5V
ATmega16: 4.5 - 5.5V
2.5 ATmega16 引脚功能
VCC 电源正
GND 电源地
端口A (PA7..PA0)做为A/D 转换器的模拟输入端。
端口A 为8 位双向I/O 口,具有可编程的内部上拉电阻。
其输出缓冲器具
有对称的驱动特性,可以输出和吸收大电流。
作为输入使用时,若内部上拉电阻使能,端口被外部电路拉低时将输出电流。
在复位过程中,即使系统时钟还未起振,端口A 处于高阻状态。
端口B(PB7..PB0)为8位双向I/O 口,具有可编程的内部上拉电阻。
其输
出缓冲器具有对称的驱动特性,可以输出和吸收大电流。
作为输入使用时,若内部上拉电阻使能,端口被外部电路拉低时将输出电流。
在复位过程中,即使系统时钟还未起振,端口B 处于高阻状态。
端口B也可以用做其他不同的特殊功能.
端口C(PC7..PC0) 为8 位双向I/O 口,具有可编程的内部上拉电阻。
其输出缓冲器具有对称的驱动特性,可以输出和吸收大电流。
作为输入使用时,若内部上拉电阻使能,端口被外部电路拉低时将输出电流。
在复位过程中,即使系统时钟还未起振,端口C 处于高阻状态。
如果JTAG接口使能,即使复位出现引脚PC5(TDI)、 PC3(TMS)与 PC2(TCK)的上拉电阻被激活。
端口C 也可以用做其他不同的特殊功能。
端口D(PD7..PD0)为8 位双向I/O 口,具有可编程的内部上拉电阻。
其输出缓冲器具有对称的驱动特性,可以输出和吸收大电流。
作为输入使用时,若内部上拉电阻使能,则端口被外部电路拉低时将输出电流。
在复位过程中,即使系统时钟还未起振,端口D 处于高阻状态。
端口D 也可以用做其他不同的特殊功能.
RESET 复位输入引脚。
持续时间超过最小门限时间的低电平将引起系统复位。
持续时间小于门限间的脉冲不能保证可靠复位。
XTAL1 反向振荡放大器与片内时钟操作电路的输入端。
XTAL2 反向振荡放大器的输出端。
AVCC AVCC是端口A与A/D转换器的电源。
不使用ADC时,该引脚应直接与VCC连接。
使用ADC时应通过一个低通滤波器与VCC 连接。
AREF A/D 的模拟基准输入引脚。
2.6 AVR单片机的应用区域
AVR单片机应用区域包括:空调控制板、打印机控制板、智能电表、智能手电筒、LED控制屏和医疗设备等领域。
3162液晶显示屏介绍
162液晶模块是一种用5x7点阵图形来显示字符的液晶显示器,根据显示的容量可以分为1行16个字、2行16个字等。
3.1 162液晶屏显示屏的引脚定义
首先,我们来看162的引脚定义,162的引脚是很整齐的SIP单列直插封装。
表1是液晶屏的引脚定义。
表1 接口信号说明表
该液晶屏采用标准的16脚接口,我们只需要关注一下几个管脚:
1 脚:VSS 为地电源。
2 脚:VDD 接 5V 正电源。
3脚:VEE,液晶屏显示偏压信号,用于调整液晶屏的显示对比度,一般会外界电位器用以调整偏压信号,接正电源时对比度最弱,接地电源时对比度最高,对比度过高时会产生“阴影”,使用时可以通过一个10K的电位器调整对比度。
4脚:RS,数据/命令选择端,即对寄存器进行选择,高电平时选择数据寄存器、低电平时选择指令寄存器。
5脚:R/W,读写选择端,高电平时进行读操作,低电平时进行写操作。
6脚:E,使能信号,其实时162的逐句控制时钟信号,利用该信号的上升沿实现对液晶屏的数据传输。
7~14脚:8位双向数据线。
15脚:背光阳极。
16脚:背光阴极。
3.2 162液晶显示屏的指令说明
162液晶模块的读写操作、屏幕和光标的操作都是通过指令编程来实现的。
(说明:1为高电平、0为低电平)。
162液晶屏内部模块共有11条控制指令,如表2所示。
指令1:清显示,指令码01H,光标复位到地址00H位置
指令2:光标复位,光标返回到地址00H
指令3:光标和显示模式设置
I/D:光标移动方向,高电平右移,低电平左移
S: 屏幕上所有文字是否左移或者右移。
高电平表示有效,低电平则无效指令4:显示开关控制。
D:控制整体显示的开与关,高电平表示开显示,低电平表示关显示
C:控制光标的开与关,高电平表示有光标,低电平表示无光标
B:控制光标是否闪烁,高电平闪烁,低电平不闪烁
指令5:光标或显示移位 S/C:高电平时移动显示的文字,低电平时移动光标
指令6:功能设置命令
DL:高电平时为4位总线,低电平时为8位总线
N:低电平时为单行显示,高电平时双行显示
F: 低电平时显示5x7的点阵字符,高电平时显示5x10的点阵字符
指令7:字符发生器RAM地址设置
指令8:DDRAM地址设置
指令9:读忙信号和光标地址 BF:忙标志位,高电平表示忙,此时模块不能接收命令或者数据,如果为低电平表示不忙。
指令10:写数据
指令11:读数据
3.3 162液晶显示屏的时序
在操作液晶屏,我们应该对它的工作时序非常熟悉,这里介绍了162液晶显示屏的两个写时序:写指令和写入数据。
写指令,即设置162液晶显示屏的工作方式:需要把RS置为低电平,RW置为低电平,然后将数据送到数据口D0~D7,最后E引脚一个高脉冲将数据写入。
写数据,即在液晶屏上实现显示时:需要把RS置为高电平,RW置为低电平,然后将数据送到D0~D7,最后E引脚一个高脉冲将数据写入。
图1 162液晶显示屏时序图
注意:162液晶显示模块是一个慢显示器件,所以在执行每条指令之前一定要确认模块的忙标志为低电平,表示不忙,否则此指令失效。
当然,162提供了读忙信号的方法:当RS和RW共同为低电平时可以写入指令或者显示地址,当RS为低电平RW为高电平时可以读忙信号。
3.4 162液晶屏的RAM地址映射及标准字库表
液晶模块内部的字符发生存储器(CGROM)已经存储了128个不同的点阵字符图形,这些字符有:阿拉伯数字、英文字母的大小写和常用的符号等,每一个字符都有一个固定的代码,比如大写的英文字母“A”的代码是01000001B(41H),显示时模块把地址41H中的点阵字符图形显示出来,我们就能看到字母“A” 。
要显示字符时要先输入显示字符地址,也就是告诉模块在哪里显示字符。
液晶屏第一行的首地址是80H,第二行的首地址是C0H。
表3是液晶屏的GROM中的字符代码与图形对应关系。
表3 字符代码与图形对应表
4ds1302芯片介绍
DS1302是美国DALLAS公司推出的一种高性能、低功耗的实时时钟芯片,附加31字节静态RAM,采用SPI 三线接口与CPU进行同步通信,并可采用突发方式一次传送多个字节的时钟信号和RAM数据。
实时时钟可提供秒、分、时、日、星期、月和年,一个月小与31天时可以自动调整,且具有闰年补偿功能。
工作电压宽达2.5~5.5V 。
采用双电源供电(主电源和备用电源),可设置备用电源充电方式,提供了对后背电源进行涓细电流充电的能力。
DS1302的外部引脚分配如图2所示及内部结构如图3 所示。
DS1302用于数据记录,特别是对某些具有特殊意义的数据点的记录上,能实现数据与出现该数据的时间同时记录,因此广泛应用于测量系统中。
图2 引脚图
图3 内部结构
各引脚的功能为:
Vcc1 :主电源;Vcc2:备份电源。
当Vcc2>Vcc1+0.2V时,由Vcc2向DS1302供电,当Vcc2< Vcc1时,由Vcc1向DS1302供电。
SCLK:串行时钟,输入,控制数据的输入与输出;
I/O :三线接口时的双向数据线;
CE :输入信号,在读、写数据期间,必须为高。
该引脚有两个功能:第一,CE开始控制字访问移位寄存器的控制逻辑;其次,CE提供结束单字节或多字节数据传输的方法。
DS1302有下列几组寄存器:
①DS1302有关日历、时间的寄存器共有12个,其中有7 个寄存器(读
时81h ~8Dh ,写时80h ~8Ch ),存放的数据格式为BCD码形式。
小时寄存器(85h 、84h )的位7 用于定义DS1302是运行于12小时模式还是24小时模式。
当为高时,选择12小时模式。
在12小时模式时,位5 是,当为 1 时,表示 PM。
在 24小时模式时,位5 是第二个10小时位。
秒寄存器(81h 、80h )的位 7 定义为时钟暂停标志(CH)。
当该位置为1 时,时钟振荡器停止,DS1302处于低功耗状态;当该位置为0 时,时钟开始运行。
控制寄存器(8Fh 、8Eh )的位7 是写保护位(WP),其它 7 位均置为0 。
在任何的对时钟和RAM的写操作之前,WP位必须为0 。
当WP位为1 时,写保护位防止对任一寄存器的写操作。
② DS1302有关RAM的地址
DS1302 中附加31字节静态RAM的地址见表所示。
③DS1302的工作模式寄存器
所谓突发模式是指一次传送多个字节的时钟信号和RAM数据。
④此外,DS1302还有充电寄存器等。
2 读写时序说明
DS1302是SPI 总线驱动方式。
它不仅要向寄存器写入控制字,还需要读取相应寄存器的数据。
要想与DS1302通信,首先要先了解DS1302的控制字。
控制字(即地址及命令字节)
控制字的最高有效位(位7 )必须是逻辑 1 ,如果它为 0 ,则不能把数据写入到DS1302中。
位6 :如果为 0 ,则表示存取日历时钟数据,为1 表示存取RAM数据;位5 至位1 (A4~A0):指示操作单元的地址;位0 (最低有效位):如为0 ,表示要进行写操作,为1 表示进行读操作。
控制字总是从最低位开始输出。
在控制字指令输入后的下一个SCLK 时钟的上升沿时,数据被写入DS1302,数据输入从最低位(0位)开始。
同样,在紧跟8 位的控制字指令后的下一个SCLK 脉冲的下降沿,读出DS1302的数据,读出的数据也是从最低位到最高位。
数据读写时序如图4 。
图4 数据读写时序
5 仿真软件介绍
5.1 PROTUES简介
Protues软件是英国Labcenter electronics公司出版的EDA工具软件。
它不仅具有其它EDA工具软件的仿真功能,还能仿真单片机及外围器件。
它是目前最好的仿真单片机及外围器件的工具。
Proteus是世界上著名的EDA工具(仿真软件),从原理图布图、代码调试到单片机与外围电路协同仿真,一键切换到PCB设计,真正实现了从概念到产品的完整设计。
目前世界上唯一将电路仿真软件、PCB设计软件和虚拟模型仿真软件三合一的设计平台,其处理器模型支持8051、HC11、PIC10/12/16/18/24/30/DsPIC33、AVR、ARM、8086和MSP430等。
5.2 PROTUES的特点
1.原理布图
2.PCB自动或人工布线
3.SPICE电路仿真
4.互动的电路仿真。
用户甚至可以实时采用诸如RAM,ROM,键盘,马达,LED,
LCD,AD/DA,部分SPI器件,部分IIC器件。
5.仿真处理器及其外围电路。
可以仿真51系列、AVR、PIC、ARM、等常用主流单
片机。
还可以直接在基于原理图的虚拟原型上编程,再配合显示及输出,能看到运行后输入输出的效果。
配合系统配置的虚拟逻辑分析仪、示波器等,Protues 建立了完备的电子设计开发环境。
5.3 PROTUES 工作界面介绍
PROTUES 的工作界面是一种标准的windows 界面,如下图所示,包括标题栏、主菜单、标准工具栏、绘图工具栏、状态栏、对象选择按钮、预览对象方位控制按钮、仿真进程控制按钮、预览窗口、对象选择器窗口、图形编辑窗口。
图5 PROTUES 界面图
6 硬件设计
6.1系统框图
本系统所包含模块:
主控芯片ATmega16、LCD1602液晶显示模块、独立键盘模块、指示灯电路、DS1302时钟模块、以及电源。
另外还有背光电压显示电路。
图6
ATmega16LCD1602
指示灯
键盘电源
DS1302
6.2 PROTUES 仿真图及电路原理图
PROTUES 仿真图
电路原理图
6.3 独立键盘
该键盘是三路独立的按键回路,当D 端口读取的电平为高电平时不动作,说明没有按键按下;当任意一个电路的按键按下时,该段口读取的电平就为低电平执行相应的操作。
按钮主要功能:
K1:按下此按钮可以执行对时间的更改且相应的时间指示灯点亮;对应键值为:00001011。
K2:按下此按钮可以实现时间地址的切换并且相应的指示灯点亮;对应键值为:00001101。
K3:按下此按钮可以执行对时间的增1操作,对应键值为:00001110。
图7
6.4 背光电压显示
电位器的中间分为两路,一路接液晶的Vee 端,另一路接单片机PA3端。
经过软件处理后将值显示在液晶屏上。
模数转换:
AD 转换结果:ADC=Vin*1024/Vref ADC 多工选择寄存器- ADMUX
图8
图9
Bit 7:6 – REFS1:0: 参考电压选择 如Table 83所示,通过这几位可以选择参考电压。
如果在转换过程中改变了它们的设置,只有等到当前转换结束 (ADCSRA 寄存器的ADIF 置位 ) 之后改变才会起作用。
如果在AREF 引脚上施加了外部参考电压,内部参考电压就不能被选用了。
在本文我们选择AVCC,AREF 引脚外加滤波电容。
Bit 5 – ADLAR: ADC 转换结果 左对齐
ADLAR 影响ADC 转换结果在ADC 数据寄存器中的存放形式。
ADLAR 置位时转换结果为左对齐,否则为右对齐。
ADLAR 的改变将立即影响ADC 数据寄存器的内容,不论是否有转换正在进行。
关于这一位的完整描述请见P207“ADC 数据寄存器– ADCL 及ADCH ” 。
Bits 4:0 – MUX4:0: 模拟通道与增益选择位
通过这几位的设置,可以对连接到ADC 的模拟输入进行选择。
也可对差分通道增益进行选择。
如果在转换过程中改变这几位的值,那么只有到转换结束(ADCSRA 寄存器的ADIF 置位)后新的设置才有效。
表8 模拟通道与增益选择
ADC 控制和状态寄存器 A -ADCSRA
图10
Bit 7 – ADEN: ADC 使能
ADEN 置位即启动ADC ,否则ADC 功能关闭。
在转换过程中关闭ADC 将立即中止正在进行的转换。
Bit 6 – ADSC: ADC 开始转换 在单次转换模式下,ADSC 置位将启动一次ADC 转换。
在连续转换模式下,ADSC 置位将启动首次转换。
第一次转换(在ADC 启动之后置位ADSC ,或者在使能ADC 的同时置位ADSC)需要25 个ADC 时钟周期,而不是正常情况下的13个。
第一次转换执行ADC 初始化的工作。
在转换进行过程中读取ADSC 的返回值为"1”,直到转换结束。
ADSC 清零不产生任何动作。
Bit 5 – ADATE: ADC 自动触发使能
ADATE 置位将启动ADC 自动触发功能。
触发信号的上跳沿启动ADC 转换。
触发信号源通过SFIOR 寄存器的ADC 触发信号源选择位ADTS 设置。
Bit 4 – ADIF: ADC中断标志
在ADC转换结束,且数据寄存器被更新后,ADIF置位。
如果ADIE及SREG中的全局中断使能位I也置位,ADC转换结束中断服务程序即得以执行,同时ADIF硬件清零。
此外,还可以通过向此标志写1来清ADIF。
要注意的是,如果对ADCSRA进行读-修改-写操作,那么待处理的中断会被禁止。
这也适用于SBI及CBI指令。
Bit 3 – ADIE: ADC中断使能
若ADIE及SREG的位I置位,ADC转换结束中断即被使能。
Bits 2:0 – ADPS2:0: ADC 预分频器选择位
由这几位来确定XTAL与ADC输入时钟之间的分频因子。
ADC 数据寄存器- ADCL 及ADCH
ADLAR = 0
图11
ADLAR = 1
图12
ADC转换结束后,转换结果存于这两个寄存器之中。
如果采用差分通道,结果由2的补码形式表示。
读取ADCL之后,ADC数据寄存器一直要等到ADCH 也被读出才可以进行数据更新。
因此,如果转换结果为左对齐,且要求的精度不高于8比特,那么仅需读取ADCH就足够了。
否则必须先读出ADCL再读ADCH。
ADMUX寄存器的ADLAR及MUXn会影响转换结果在数据寄存器中的表示方式。
如果ADLAR为1,那么结果为左对齐;反之 ( 系统缺省设置 ),结果为右对齐。
6.5 指示灯电路
每路指示灯指示功能:
PC0路:秒寄存器显示
PC1路:分寄存器显示 PC2路:时寄存器显示 PC3路:日寄存器显示 PC4路:月寄存器显示 PC5路:星期寄存器显示 PC6路:年寄存器显示 PD4路:是否允许调节时间显示,
灯亮为允许。
图13
6.6 DS1302外围电路
22pf 电容和32768Hz 晶振组成振荡电路,
提供ds1302正常工作频率。
VCC2为主电源,VCC1为辅助电源。
Rst 为读写使能端,I/O
为三线接口时的双向数据线,SCLK 为串行时
钟,输入,控制数据的输入与输出。
图14
6.7 单片机外围电路
22pf 电容和8MHz 晶振组成振荡电路,提供单片机正常工作频率。
Reset 为上电复位端,
当通电后单片即不管以前的程序工作在哪
里,一切复位从“零”开始。
AVCC 是端口A 与A/D 转换器的电源。
AREF 是A/D 的模拟基准输入引脚。
图15
7 软件设计
在前面几节中,我们了解了ATmega16单片机、162液晶显示屏与ds1302的接口设计以及液晶屏的时序,那么这部分阐述的是单片机对液晶屏显示和ds1302的驱动控制。
软件工作流程如下:
22p
32768Hz 22p 22p
8MHz 22p
程序流程图
8结束语
经过几个月的研究与实验,终于完成了这篇论文。
在做这个课题的时侯遇到了许多困难,大部分通过查阅资料和看书以及研究别人成功的例子就能解决了,但少部分问题任不能解决,通过老师和同学的帮助终于解决。
感谢帮助我的老师和同学。
参考文献:
[1] HS162-4液晶显示使用说明资料.
[2] DS1302_数据手册.
[3] ATmega16中文资料.
[4] Proteus中文入门教程.
[5] 贺敬凯,刘德新,管明祥.单片机系统设计、仿真与应用。
西安电子科技大学出版社,2011.
[6]沈文. AVR单片机C语言开发入门指导.清华大学出版社,2003年.
[7]丁化成,耿德根,李军凯. AVR单片机应用设计.北京航空航天大学出版社,2002年.
[8]金春林,邱慧芳,张皆喜. AVR系列单片机C语言编程与应用实例.清华大学出版社,2003
年.
附件:
/******************************************************
设计介绍:
使用8MHz的晶振,ds1302芯片以及1602液晶屏。
使用
独立键盘设置或者调节时间。
13个10k的电阻,1个100
欧的电阻,8个led指示灯。
2个22pf的电容。
*******************************************************/
#include <mega16.h>
#define io PORTA.0
#define sclk PORTA.1
#define rst PORTA.2
#define rs PORTA.5
#define rw PORTA.6
#define e PORTA.7
#define uchar unsigned char
#define uint unsigned int
/*
uchar l_dat[] = {" BASE ON "};
uchar f_dat[] = {"ATMEGA16 DESIGN"};
*/
uchar set_time [7] = {0x00,0x00,0x13,0x21,0x07,0x06,0x12}; //初始时间缓冲区
uchar address=0x7e; //时间地址中间变量
uchar QIEHUAN; //切换时间标志位
uchar n=0; //定义时间调节变量
uchar INT0=0; //定义中断变量
uchar time[8] ={0x00}; //定义时、分、秒显示缓冲区
uchar day[8]={0x00}; //定义年、月、日、星期显示缓冲区
int d[3]={0,0,0}; //定义电压显示缓冲区
/***********延时函数*************/
void delayus(uint us) //1us延时函数
{
uint i;
us=us*5/4; //5/4是在8MHz晶振下,通过软件仿真反复实验得到的数值
for(i=0;i<us;i++);
}
void delayms(uint ms) //1ms延时函数
{
uint i,j;
for( i=0;i<ms;i++)
for(j=0;j<1141;j++); //1141是在8MHz晶振下,通过软件仿真反复实验得到的数值}
/*******液晶屏判断忙函数*********/
void lcd_busy()
{
uchar temp; //定义中间变量存取从lcd读取的值
DDRB=0x00; //定义B端口为输入
rs=0; //选择指令寄存器
rw=1; //进行读操作
do
{
e=1;
delayms(1); //延时1ms
temp=PINB; //读取B端口信号
e=0;
delayms(1); //延时1ms
}while(temp&0x80); //判断忙标志位是否为1
DDRB=0xff; //B端口置为输出,为想液晶屏写数据准备
}
/************* 写指令函数*********/
void lcd_wcmd(uchar cmd)
{
lcd_busy(); //判断液晶屏是否忙
delayms(1); //延时1ms
rs=0; //选择指令寄存器
rw=0; //进行写操作
e=0;
delayms(1); //延时1ms
PORTB=cmd; //把数据赋值给B端口
e=1; //开始写指令
delayms(1); //延时1ms
e=0;
delayms(1); //延时1ms
}
/**********写数据函数**********/
void lcd_wdat(uchar dat)
{
lcd_busy(); //判断液晶屏是否忙
delayms(1); //延时1ms
rs=1; //选择数据寄存器
rw=0; //进行写操作
e=0;
delayms(1); //延时1ms
PORTB=dat; //把数据赋值给B端口
e=1; //开始写数据
delayms(1); //延时1ms
e=0;
}
/*********lcd 初始化函数*********/
void lcd_init()
{
delayms(15); //等待LCD电源稳定
lcd_wcmd(0x38); //16*2显示,5*7点阵,8位数据delayms(5); //延时5ms
lcd_wcmd(0x0c); //显示开,关光标
delayms(5); //延时5ms
lcd_wcmd(0x06); //移动光标
delayms(5); //延时5ms
lcd_wcmd(0x01); //清除LCD的显示内容
delayms(5); //延时5ms
}
/*********写时间函数*******/
void ds1302writebyte(uchar dat)
{
uchar m; //定义循环变量
DDRA=0xf7; //设置ds1302的io端口为输入
sclk=0; //串行时钟低电平
delayus(10); //延时10us
for(m=0;m<8;m++) //开始传送8bit数据
{
io=dat&0x01; //读取最低位数据
delayus(10); //延时10us
sclk=0; //时钟低电平为传送数据准备
delayus(10); //延时10us
sclk=1; //时钟拉高,开始传送数据
dat>>=1; //数据右移一位,准备传送下位数据
}
}
/*********读时间函数*********/
uchar ds1302readbyte()
{
uchar n,dat; //定义循环变量,存取数据变量
DDRA=0xf6; //设置ds1302的io端口为输出
delayus(10); //延时10us
for(n=0;n<8;n++) //开始读取8bit数据
{
if(PINA.0==1) //读取A0端口并判断是否为1
dat=dat|0x80; //把要传送的当前位数据置为1,其余位保持不变else
dat=dat&0x7f; //把要传送的当前位数据置为0,其余位保持不变sclk=1; //时钟置为1,为传送数据准备
delayus(10); //延时10us
sclk=0; //时钟电平拉低,开始传送数据
dat>>=1; //数据右移一位,准备接受下位数据}
DDRA=0xf7; //设置ds1302的io端为输入
return dat; //返回数据
}
/***********向ds1302读数据函数*********/
uchar ds1302read(uchar cmd)
{
uchar dat; //定义存取ds1302读取数据的变量
rst=0; //初始化rst置为0
sclk=0; //初始化sclk置为0
rst=1; //初始化rst置为1,传输开始
ds1302writebyte(cmd); //传输命令字,要读取时间/日历地址
dat=ds1302readbyte(); //读取要得到的时间/日期
sclk=1; //时钟拉高
rst=0; //读取结束
return dat; //返回时间/日期
}
/*******向ds1302写数据函数************/
void ds1302write(uchar cmd,uchar dat)
{
rst=0; //初始化rst置为0
sclk=0; //初始化sclk置为0
rst=1; //初始化rst置为1,传输开始
ds1302writebyte(cmd); //传输命令字,要写入时间/日历地址
ds1302writebyte(dat); //写入要修改的时间/日期
sclk=1; //时钟拉高
rst=0; //读取结束
}
/********LCD显示的数据函数**********/
void lcd_disp ()
{ lcd_wcmd(0x02 | 0x80); //从第一行第3列开始显示
lcd_wdat(day[0]); //显示年十位
lcd_wdat(day[1]); //显示年个位
lcd_wdat(0x2d); //显示“—”
lcd_wdat(day[2]); //显示月十位
lcd_wdat(day[3]); //显示月个位
lcd_wdat(0x2d); // 显示“-”
lcd_wdat(day[4]); //显示日十位
lcd_wdat(day[5]); //显示日个位
lcd_wcmd(0x0f | 0x80); //第1行16位开始显示
lcd_wdat(day[6]); //显示星期
lcd_wcmd(0x40 | 0x80); //从第2行第1列开始显示
lcd_wdat(time[0]); //显示小时十位
lcd_wdat(time[1]); //显示小时个位
lcd_wdat(0x3a); //显示':'
lcd_wdat(time[2]); //显示分钟十位
lcd_wdat(time[3]); //显示分钟个位
lcd_wdat(0x3a); //显示':'
lcd_wdat(time[4]); //显示秒十位
lcd_wdat(time[5]); //显示秒个位
}
/*************时间函数******************/
void write_ds1302(uchar cmd,uchar dat)
{
sclk=0; //时钟置为0
rst=1; //rst置为1,开始传输数据
ds1302writebyte(cmd); //传送要写入寄存器的地址
ds1302writebyte(dat); //开始写入数据
sclk=0; //时钟置为0
rst=0; //rst置为0,停止写数据
}
/********以下是读取时间函数,负责读取当前的时间,并将读取到的时间转换为10进制数********/
void get_time()。