DS1302时钟模块LCD1602显示
- DS1302时钟模块相关介绍
- 基本介绍
-
- 时序图
- 单字节写时序
- 单字节读时序
- 时钟/日历多字节(Burst)方式
- 代码
-
- 项目展示
DS1302时钟模块相关介绍
基本介绍
概述
DS1302 可慢速充电实时时钟芯片包含实时时钟/日历和 31 字节的非易失性静态 RAM。它经过一个简
单的串行接口与微处理器通信。实时时钟/日历可对秒,分,时,日,周,月,和年进行计数,对于小于
31 天的月,月末的日期自动进行调整,还具有闰年校正的功能。时钟可以采用 24 小时格式或带 AM(上
午)/PM(下午)的 12 小时格式。31 字节的== RAM== 可以用来临时保存一些重要数据。使用同步串行通信,
简化了 DS1302 与微处理器的通信。与时钟/RAM 通信仅需3 根线:(1)RST(复位),(2)I/O(数据线)
和(3)SCLK(串行时钟)。数据可以以每次一个字节的单字节形式或多达 31 字节的多字节形式传输。DS1302
能在非常低的功耗下工作,消耗小于 1µW 的功率便能保存数据和时钟信息。
特点
各引脚功能
各引脚的功能:
Vcc1:主电源;Vcc2:备份电源。 当Vcc2>Vcc1+0.2V时,由Vcc2向DS1302供电,当Vcc2<Vcc1时,由Vcc1向DS1302供电。
SCLK: 串行时钟,输入,控制数据的输入与输出;
I/O: 三线接口时的双向数据线;
CE: 输入信号,在读和写的器件必须为高。该引脚有两个功能:1.开始控制字访问移位寄存器的控制逻辑;2.提供结束单字节或多字节数据传输的方法。
相关寄存器
有关日历、时间的寄存器共有12个,其中有7个寄存器(读时81H——8DH,写时80H——8CH==)存放的数据格式是BCD码形式。如图所示
小时寄存器
位7用于定义DS1302是采用12小时模式还是24小时模式。高:12小时模式。在12小时模式时,位5 为1时,表示PM。在24小时模式时,为5是第二个10小时位。
秒寄存器
的就位7定义为时钟暂停标志(CH)。1:时钟振荡器停止,DS1302进入低功耗状态,当该位置为0时,时钟开始运行。
控制寄存器
的位7是写保护位(WP),其他7位均置0,对时钟和RAM进行写操作时,WP必须为0,也就是关闭写保护,当WP为1时,就是只读模式。
DS1302相关的RAM地址
DS1302中附加31字节静态RAM的地址如下图
DS1302的工作模式寄存器
所谓的突发模式(BURST模式)就是一次性传送多个字节的时钟信号和RAM数据。比如我可以一次性把时间和日期写入,也可以一次性的读出时间和日期。
通过对 31(十进制)位地址寻址(地址/命令位于 1 至 5=逻辑 1),可以把时钟/日历或 RAM 寄存器规定为多字节方式。如前所述,位 6 规定时钟或 RAM 而位 0 规定读或写。在时钟\日历寄存器中的地址 9 至 31或 RAM 寄存器中的地址 31 不能存储数据。在多字节方式中读或写从地址 0 的位 0 开始。当以多字节方式写时钟寄存器时,必须按数据传送的次序写最先 8 个寄存器。意思就是写我们不是只有7个吗?时分秒、年月日、周一共七个 但是吧它得字节数为8 所有要写够8次。
但是,当以多字节方式写 RAM 时,为了传送数据不必写所有 31 个字节。不管是否写了全部 31 个字节,所写的每一个字节都将传送至 RAM。
DS1302充电寄存器
这个我没整过,全部复制手册的。哈哈
这个寄存器控制 DS1302 的慢速充电特性。图 4 的简化电路表示慢速充电器的基本组成。慢速充电选择(TCS)位(位 4-7)控制慢速充电器的选择。为了防止偶然的因素使之工作,只有 1010 模式才能使慢
速充电器工作,所有其它的模式将禁止慢速充电器。DS1302 上电时,慢速充电器被禁止。二极管选择(DS)位(位 2-3)选择是一个二极管还是两个二极管连接在 Vcc2 与 Vcc1 之间。如果 DS 为 01,那么选择一个二极管;如果 DS 为 10,则选择两个二极管。如果 DS 为 00 或 11,那么充电器被禁止,与 TCS 无关。RS 位(位0-1)选择连接在 Vcc2 与 Vcc1 之间的电阻。电阻选择(RS)位选择的电阻如下:
如果 RS 为 00,充电器被禁止,与 TCS 无关。
二极管和电阻的选择用户根据电池和超容量电容充电所需的最大电流决定。最大充电电流可以如下列
所说明的那样进行计算。假定 5V 系统电源加到 Vcc2 而超容量电容接至 Vcc1。再假设慢速充电器工作时在Vcc2 和 Vcc1 之间接有一个二极管和电阻 R1。因而最大电流可计算如下:
Imax =(5.0V-二极管压降)/R1
=(5.0V-0.7V)/2kΩ
= 2.2mA
显而易见,当超容量电容充电时,Vcc2 和 Vcc1 之间的电压减少,因而充电电流将会减小。
时序图
我们首先得给DS1302时钟模块写入时间和日期,然后呢这个时钟会自己走,我们之后只需要读取时间和日期即可。我们直接看时序图 编写代码
单字节写时序
数据输入
跟随在输入写命令字节的 8 个 SCLK 周期之后,在下 8 个 SCLK 周期的上升沿输入数据。如果有额外的SCLK 周期,它们将被忽略。输入从位 0 开始。
上面时序图是前8位是地址,后8位是数据。
我们根据上面的时序图 来编写 单字节写的一个函数
发现上面的
void ds1302_write_byte(unsigned char addrOrData)
{
unsigned char i = 0;
for(i=0;i<8;i++)
{
DAT = addrOrData & 0x01;
addrOrData = addrOrData >> 1;
CLK = 1;
CLK = 0;
}
}
void ds1302_write(unsigned char addr,unsigned char data)
{
RST = 0;
SCLK = 0;
RST = 1;
ds1302_writeByte(addr);
ds1302_writeByte(dat);
RST = 0;
}
看上面的图,比如我想写入时 为18点 时的写寄存器是84H 数据就是18点 但是DS1302存储格式为 BCD 码那就是 0x18。
那写入 时 为18点 的代码则是:
ds1302_write(0x84,0x18);
单字节读时序
unsigned char ds1302_read_byte(unsigned char data)
{
unsigned char i;
for(i=0;i<8;i++)
{
dat = dat >> 1;
if(IO == 1)
dat = dat | 0x80;
else
dat = dat & 0x7f;
SCLK = 1;
SCLK = 0;
}
return data;
}
unsigned char ds1302_read_data(unsigned char addr)
{
unsigned char data;
RST = 0;
SCLK = 0;
RST = 1;
ds1302_write_byte(addr);
data = ds1302_read_byte(data);
RST = 0;
return data;
}
我们想读取时 里面时间 看看现在是几点 时 读寄存器是85H
time = ds1302_read_data(0x85);
写保护寄存器
写保护寄存器的位 7 是写保护位。开始 7 位(位 0-6)置为零,在读操作时总是读出零。在对时钟或
功能 字节数 脉冲数
CLOCK 8 72
RAM 31 256
DS1302
RAM 进行写操作之前,位 7 必须为零。当它为高电平时,写保护位禁止对任何其它寄存器进行写操作。
时钟/日历多字节(Burst)方式
所谓的突发模式(BURST模式)就是一次性传送多个字节的时钟信号和RAM数据。比如我可以一次性把时间和日期写入,也可以一次性的读出时间和日期。
代码
部分代码呈现
main.c
#include <reg52.h>
#include "lcd1602.h"
#include "ds1302.h"
void main()
{
lcd_init();
ds1302_init();
while(1)
{
ds1302_burst_read(DS1302_READ_TIME);
lcd_display_dateAndTime();
}
}
lcd1602.c
void lcd_init()
{
unsigned char array1[] = "date: - - ";
unsigned char array2[] = "time: : : ";
write_com(CURSOR_RIGHT);
write_com(CLOSE_DIS_CUR);
write_com(DIS_5X7);
write_com(CLEAR_SCREEN);
lcd_display_str(0,0,array1);
lcd_display_str(0,1,array2);
}
void lcd_display_dateAndTime()
{
lcd_display_byte(5,0,(time[7]>>4)+48);
lcd_display_byte(6,0,(time[7]&0x0f)+48);
lcd_display_byte(7,0,(time[6]>>4)+48);
lcd_display_byte(8,0,(time[6]&0x0f)+48);
lcd_display_byte(10,0,(time[4]>>4)+48);
lcd_display_byte(11,0,(time[4]&0x0f)+48);
lcd_display_byte(13,0,(time[3]>>4)+48);
lcd_display_byte(14,0,(time[3]&0x0f)+48);
lcd_display_byte(5,1,(time[2]>>4)+48);
lcd_display_byte(6,1,(time[2]&0x0f)+48);
lcd_display_byte(8,1,(time[1]>>4)+48);
lcd_display_byte(9,1,(time[1]&0x0f)+48);
lcd_display_byte(11,1,(time[0]>>4)+48);
lcd_display_byte(12,1,(time[0]&0x0f)+48);
lcd_display_byte(14,1,(time[5]&0x0f)+48);
}
ds1302.c
#include "ds1302.h"
unsigned char time[8] = {0x00,0x45,0x19,0x18,0x10,0x01,0x21,0x20};
unsigned char ds1302_readByte(unsigned char dat)
{
unsigned char i = 0;
for(i=0;i<8;i++)
{
dat = dat >> 1;
if(DAT == 1)
dat = dat | 0x80;
else
dat = dat & 0x7F;
CLK = 1;
CLK = 0;
}
return dat;
}
void ds1302_writeByte(unsigned char addrOrData)
{
unsigned char i = 0;
for(i=0;i<8;i++)
{
DAT = addrOrData & 0x01;
addrOrData = addrOrData >> 1;
CLK = 1;
CLK = 0;
}
}
void ds1302_writeCom(unsigned char addr,unsigned char dat)
{
RST = 0;
CLK = 0;
RST = 1;
ds1302_writeByte(addr);
ds1302_writeByte(dat);
RST = 0;
}
unsigned char ds1302_readData(unsigned char addr)
{
unsigned char dat = 0;
RST = 0;
CLK = 0;
RST = 1;
ds1302_writeByte(addr);
dat = ds1302_readByte(dat);
RST = 0;
return dat;
}
void ds1302_read_DateAndTime()
{
time[0] = ds1302_readData(DS1302_READ_YEAR);
time[1] = ds1302_readData(DS1302_READ_MONTH);
time[2] = ds1302_readData(DS1302_READ_DAY);
time[3] = ds1302_readData(DS1302_READ_HOUR);
time[4] = ds1302_readData(DS1302_READ_MINUTE);
time[5] = ds1302_readData(DS1302_READ_SECOND);
time[6] = ds1302_readData(DS1302_READ_WEEK);
}
void ds1302_init()
{
RST = 0;
CLK = 0;
ds1302_writeCom(DS1302_WRITE_PROTECT,0x00);
ds1302_burst_write(DS1302_WRITE_TIME);
ds1302_writeCom(DS1302_WRITE_PROTECT,0x80);
}
项目展示
如果觉得这篇文章对你有用。欢迎大家点赞、评论哈哈
需要整个工程代码,欢迎大家打赏,评论区留上你的邮箱 or vx or qq。o( ̄︶ ̄)o
继续加油!!!
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)