【蓝桥杯单片机12】实时时钟DS1302的基本操作

www.xmf393.com / 广东职业技术学院 欧浩源

实时时钟DS1302几乎是蓝桥杯“单片机设计与开发”每年必考的内容,虽然在竞赛现场有提供一个底层读写寄存器的库文件,但是作为备赛阶段,你应该搞清楚底层读写时序的代码实现。你会使用库文件开发,不一定会自己写底层;你会自己写驱动,就一定会使用库文件开发。你使用库文件开发的过程中碰到问题,或者需要调整时序的时候,如果没有过硬的功夫,那只能懵逼了。

1、什么是DS1302?
      DS1302是美国DALLAS公司推出的高性能、低功耗的实时时钟,附加31字节的静态RAM,采用SP三线接口与MCU进行同步通信,并可采用突发方式一次传送多个字节的时钟参数和RAM数据。实时时钟可提供秒、分、时、日、星期、月和年,一个月小于31天时可以自动调整,并具有润年补偿功能。
      简单来说,DS1302可以理解为一个电子手,里面带有一个31字节的内存。当然,基本的使用方法和我们平时使用电子手表差不多,你可以设定时间,也可以读取时间,只不过这些工作是通过SPI接口有MCU去完成而已。
      在DS1302中有两块存储器:日历时钟寄存器和今天RAM存储器。前者用于记录实时时间,后者用于记录其他数据。对于基本计时应用,重点关注的是日历时钟寄存器。设定时间参数就是往这些寄存器写入内容,读取实时时间也是从这些寄存器读出数据。

2、日历时钟寄存器
      DS1302有关日历和时钟的寄存器有12个,我们最常用的有7个。

      什么是BCD码?
      就是用十六进制来表示十进制。什么意思?怎么理解?
      例如,十六进制数0x13的值为整数19,但BCD码表示的是整数13。

3、控制字的格式
      DS1302将地址和读写控制放到一个字节里面,形成一个控制字,格式如下:

通过上面的控制字格式,大家就可以明白为什么DS1302读寄存器和写寄存器的地址是不一样的了,因为这个地址包含了读写控制位。为了方便程序设计,我们把读寄存器地址写寄存器地日历时钟寄存器方面用三个数组定义。

4、接口时序的实现
      DS1302的基本操作实际上非常简单,只有两个操作:其一是设定时间参数,其二是读取实时时间。不管是那个操作,MCU都要通过SPI接口进行数据交互,而SPI接口有其规定的时序,这个必须参考数据手册。
      控制字总是从最低位开始输出。在控制字指令输入后的下一个SCLK时钟信号的上升沿,数据被写入DS1302,数据的输入从最低位开始;在控制字指令输入后的下一个SCLK时钟信号的下降沿,数据从DS1302读出,数据的读出也是从最低位到最高位。
<1> 单字节写的时序

底层驱动代码实现可参考如下:

void DS1302_WriteByte(unsigned char addr, unsigned char dat)
{
unsigned char n;
RST = ;
_nop_();
SCLK = ;
_nop_();
RST = ;
_nop_(); for (n=; n<; n++) //发送要写入数据的内存地址
{
DSIO = addr & 0x01;
addr >>= ;
SCLK = ;
_nop_();
SCLK = ;
_nop_();
}
for (n=; n<; n++) //将指定内容写入该地址的内存
{
DSIO = dat & 0x01;
dat >>= ;
SCLK = ;
_nop_();
SCLK = ;
_nop_();
}
RST = ;
_nop_();
}

<2> 单字节读的时序

底层驱动代码实现可参考如下:

unsigned char DS1302_ReadByte(unsigned char addr)
{
unsigned char n,dat,tmp;
RST = ;
_nop_();
SCLK = ;
_nop_();
RST = ;
_nop_(); for(n=; n<; n++) //发送要读出数据的内存地址
{
DSIO = addr & 0x01;
addr >>= ;
SCLK = ;
_nop_();
SCLK = ;
_nop_();
} for(n=; n<; n++) //读出该地址内存的数据
{
tmp = DSIO;
dat = (dat>>) | (tmp<<);
SCLK = ;
_nop_();
SCLK = ;
_nop_();
} RST = ;
_nop_();
SCLK = ;
_nop_();
DSIO = ;
_nop_();
DSIO = ;
_nop_();
return dat;
}

有了上面两个底层的SPI接口数据读写代码,那么DS1302的基本操作就很容易实现了。

5、单元实训题目

6、实现源码参考

#include "reg52.h"
#include "intrins.h" sbit HC138_A = P2^;
sbit HC138_B = P2^;
sbit HC138_C = P2^; sbit SCLK = P1^;
sbit RST = P1^;
sbit DSIO = P2^;
unsigned char code READ_RTC_ADDR[] = {0x81, 0x83, 0x85, 0x87, 0x89, 0x8b, 0x8d};
unsigned char code WRITE_RTC_ADDR[] = {0x80, 0x82, 0x84, 0x86, 0x88, 0x8a, 0x8c};
unsigned char TIME[] = {0x30, 0x50, 0x23, 0x17, 0x02, 0x06, 0x18}; unsigned char code SMG_NoDot[] =
{0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,
0x80,0x90,0x88,0x80,0xc6,0xc0,0x86,0x8e,
0xbf,0x7f}; void DelaySMG(unsigned int time)
{
while(time--);
} void Init74HC138(unsigned char n)
{
switch(n)
{
case :
HC138_A = ;
HC138_B = ;
HC138_C = ;
break;
case :
HC138_A = ;
HC138_B = ;
HC138_C = ;
break;
case :
HC138_A = ;
HC138_B = ;
HC138_C = ;
break;
case :
HC138_A = ;
HC138_B = ;
HC138_C = ;
break;
case :
HC138_A = ;
HC138_B = ;
HC138_C = ;
break;
}
} void DispaySMG_Bit(unsigned char value, unsigned char pos)
{
Init74HC138();
P0 = (0x01 << pos);
Init74HC138();
P0 = value;
} void DS1302_WriteByte(unsigned char addr, unsigned char dat)
{
unsigned char n;
RST = ;
_nop_();
SCLK = ;
_nop_();
RST = ;
_nop_(); for (n=; n<; n++)
{
DSIO = addr & 0x01;
addr >>= ;
SCLK = ;
_nop_();
SCLK = ;
_nop_();
}
for (n=; n<; n++)
{
DSIO = dat & 0x01;
dat >>= ;
SCLK = ;
_nop_();
SCLK = ;
_nop_();
}
RST = ;
_nop_();
} unsigned char DS1302_ReadByte(unsigned char addr)
{
unsigned char n,dat,tmp;
RST = ;
_nop_();
SCLK = ;
_nop_();
RST = ;
_nop_(); for(n=; n<; n++)
{
DSIO = addr & 0x01;
addr >>= ;
SCLK = ;
_nop_();
SCLK = ;
_nop_();
} for(n=; n<; n++)
{
tmp = DSIO;
dat = (dat>>) | (tmp<<);
SCLK = ;
_nop_();
SCLK = ;
_nop_();
} RST = ;
_nop_();
SCLK = ;
_nop_();
DSIO = ;
_nop_();
DSIO = ;
_nop_();
return dat;
} void DS1302_Config()
{
unsigned char n;
DS1302_WriteByte(0x8E,0x00);
for (n=; n<; n++)
{
DS1302_WriteByte(WRITE_RTC_ADDR[n],TIME[n]);
}
DS1302_WriteByte(0x8E,0x80);
} void DS1302_ReadTime()
{
unsigned char n;
for (n=; n<; n++)
{
TIME[n] = DS1302_ReadByte(READ_RTC_ADDR[n]);
}
} void XMF_ShowRealTime()
{ DispaySMG_Bit(SMG_NoDot[TIME[]/],);
DelaySMG();
DispaySMG_Bit(0xff,);
DispaySMG_Bit(SMG_NoDot[TIME[]&0x0f],);
DelaySMG();
DispaySMG_Bit(0xff,);
DispaySMG_Bit(SMG_NoDot[],);
DelaySMG();
DispaySMG_Bit(0xff,); DispaySMG_Bit(SMG_NoDot[TIME[]/],);
DelaySMG();
DispaySMG_Bit(0xff,);
DispaySMG_Bit(SMG_NoDot[TIME[]&0x0f],);
DelaySMG();
DispaySMG_Bit(0xff,);
DispaySMG_Bit(SMG_NoDot[],);
DelaySMG();
DispaySMG_Bit(0xff,); DispaySMG_Bit(SMG_NoDot[TIME[]/],);
DelaySMG();
DispaySMG_Bit(0xff,);
DispaySMG_Bit(SMG_NoDot[TIME[]&0x0f],);
DelaySMG();
DispaySMG_Bit(0xff,);
} main()
{
DS1302_Config();
while()
{
DS1302_ReadTime();
XMF_ShowRealTime();
}
}

【蓝桥杯单片机12】实时时钟DS1302的基本操作的更多相关文章

  1. 【蓝桥杯单片机11】单总线温度传感器DS18B20的基本操作

    [蓝桥杯单片机11]单总线温度传感器DS18B20的基本操作 广东职业技术学院 欧浩源 单总线数字温度传感器DS18B20几乎成了各类单片机甚至ARM实验板的标配模块来,在蓝桥杯的往届省赛和国赛中,这 ...

  2. 【蓝桥杯单片机02】LED的基本控制

    [蓝桥杯单片机02]LED的基本控制 广东职业技术学院  欧浩源 在CT107D单片机综合训练平台实现LED的基本控制和其他单片机开发平台不一样,不单单是控制几个LED实现跑马灯这么简单.因为在这个平 ...

  3. MSP430:实时时钟-DS1302

    /* * DS1302.h * * Created on: 2013-11-27 * Author: Allen */ #ifndef DS1302_H_ #define DS1302_H_ #inc ...

  4. 蓝桥杯单片机CT107D 01 底层驱动基础

    代码下载 https://share.weiyun.com/5NHvLxG 这两个代码文件是其他底层驱动代码的基础: 包含了控制138573(间接控制数码管led和蜂鸣器等).delay延时函数.CT ...

  5. 轻松吃透实时时钟芯片DS1302软硬件设计,看完秒懂

    今天我们来讨论一款老掉牙的实时时钟芯片DS1302.什么是实时时钟(RealTime Clock, RTC)呢?为什么我们需要它呢?假设你使用单片机实现万年历应用,一般的做法是这样的:设置中断后判断1 ...

  6. 关于STM32驱动DS1302实时时钟的一点思考

    之前用51驱动过DS1302,没用多久就输出了正确的时间.当时以为这块芯片其实没啥,很简单.但是现在用STM32做项目,用到同样的芯片,以为这有何难,只要把那个程序拿过来复制黏贴改一下IO设置不就行了 ...

  7. [18/12/3]蓝桥杯 练习系统 入门级别 Fibonacci数列求模问题 题解思路

    前言略. 看到这个题目本来应该很高兴的,因为什么,因为太TM的基础了啊! 可是当你用常规方法尝试提交OJ时你会发现..hhh...运行超时..(开心地摇起了呆毛 //Fibonacci数列递归一般问题 ...

  8. 自制单片机之十三……时钟IC_DS1302

    在网上看了很久,发现初学者最有兴趣的就是DS1302时钟电路,也很自然,它是个做出来就让你觉得最实用的电路了,但实际上制做上并不简单,首先你要让你的显示部分(不管是数码管还是LCD)调试通过.然后把D ...

  9. STC8H开发(十四): I2C驱动RX8025T高精度实时时钟芯片

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

随机推荐

  1. mysql只修改年月日,时分秒不变

    比如时间为 “2014-12-11 13:12:01”修改为“2014-09-26 13:12::01”修改内容为修改yyyy-mm-dd为对应的日期,时间不变.UPDATE result_road_ ...

  2. nowcoder 203A Knight(贪心+打表)

    题目链接 题目描述 有一张无限大的棋盘,你要将马从(0,0)移到(n,m). 每一步中,如果马在(x,y),你可以将它移动到(x+1,y+2),(x+1,y-2),(x-1,y+2),(x-1,y-2 ...

  3. UVA 12633 Super Rooks on Chessboard(FFT)

    题意: 给你一个R*C的棋盘,棋盘上的棋子会攻击,一个棋子会覆盖它所在的行,它所在的列,和它所在的从左上到右下的对角线,那么问这个棋盘上没有被覆盖的棋盘格子数.数据范围R,C,N<=50000 ...

  4. P3165 [CQOI2014]排序机械臂

    题目描述 为了把工厂中高低不等的物品按从低到高排好序,工程师发明了一种排序机械臂.它遵循一个简单的排序规则,第一次操作找到高度最低的物品的位置 P1P_1P1​ ,并把左起第一个物品至 P1P_1P1 ...

  5. Intelligent Factorial Factorization LightOJ - 1035(水题)

    就是暴力嘛...很水的一个题... 不好意思交都... #include <iostream> #include <cstdio> #include <sstream&g ...

  6. js关闭当前页面窗口的问题

    有两种情况,如果当前页面窗口是由js代码打开的,那么可以直接用js关闭该窗口 如: window.close(); 如果该页面是由用户输入地址直接进去的,直接close是会无效的,此时需要这样做: w ...

  7. 学习Spring Boot:(四)应用日志

    前言 应用日志是一个系统非常重要的一部分,后来不管是开发还是线上,日志都起到至关重要的作用.这次使用的是 Logback 日志框架. 正文 Spring Boot在所有内部日志中使用Commons L ...

  8. CodeVs.1036 商务旅行 ( LCA 最近公共祖先 )

    CodeVs.1036 商务旅行 ( LCA 最近公共祖先 ) 题意分析 某首都城市的商人要经常到各城镇去做生意,他们按自己的路线去做,目的是为了更好的节约时间. 假设有N个城镇,首都编号为1,商人从 ...

  9. cgroup限制内存

    cgroup有个memory子系统,有两组对应的文件,一组带 memsw ,另一组不带. # docker ps -a # cd /sys/fs/cgroup/memory/docker/4b5619 ...

  10. [POI2011]ROT-Tree Rotations

    发现x的子树在后续处理中不会影响逆序对的情况(只关心有哪些值,相对位置已经不重要了) f[x]表示x为根的子树最小逆序对数 考虑左右儿子交换与否. 暴力是O(n^2)的 考虑线段树合并 左右儿子线段树 ...