测试一
测试代码:别忘了之前调整点阵的跳线
#include <STC89C5xRC.H>
#include "LCD1602.h"void main()
{LCD_Init();LCD_ShowString(1,1,"RTC");while(1){}
}
------------------------------------
测试二
DS1302.C
#include <STC89C5xRC.H>
#include "DS1302.h"//引脚定义
sbit DS1302_SCLK=P3^6;
sbit DS1302_IO=P3^4;
sbit DS1302_CE=P3^5;//如果给定的 SCLK 引脚在电路板上确实连接到 P3.6,那么可以直接操作 SCLK,代码会按照定义进行相应操作。
//这样定义是为了方便不同型号,只需要修改P36就可以。void DS1302_Init(void)
{DS1302_CE=0; //手册中标注平时默认为0,CE只有在1的时候才工作DS1302_SCLK=0;
}void DS1302_WriteByte(unsigned char Command,Data)
{unsigned char i;DS1302_CE=1;for(i=0;i<8;i++){DS1302_IO=Command&(0x01<<i);DS1302_SCLK=1;DS1302_SCLK=0;}/*DS1302_IO=Command&0x01; //取出第0位DS1302_SCLK=1;DS1302_SCLK=0; 这里置1后立马置0需要考虑时钟芯片的操作时间(可能反应不过来)*/for(i=0;i<8;i++){DS1302_IO=Data&(0x01<<i);DS1302_SCLK=1;DS1302_SCLK=0;}DS1302_CE=0;}unsigned char DS1302_ReadByte(unsigned char Command)
{unsigned char i,Data=0x00;//局部变量初始化不一定是0,全局默认是DS1302_CE=1;for(i=0;i<8;i++){DS1302_IO=Command&(0x01<<i);DS1302_SCLK=0;DS1302_SCLK=1; //一个脉冲控制两个,一个结束另一个直接开始,要调整顺序//第二部分的脉冲操作是芯片自己负责,自己不用操作,读出只需要置1}for(i=0;i<8;i++){DS1302_SCLK=1; //最开始连续两次1,电平不会有变化(没有变化)DS1302_SCLK=0; //如果是先0后1,搞完后SCLK是1,但是最后SCLK是0if(DS1302_IO){Data|=(0x01<<i);}}DS1302_CE=0;DS1302_IO=0; //读取后将IO设置为0,否则读出的数据会出错return Data;
}
main.c
#include <STC89C5xRC.H>
#include "LCD1602.h"
#include "DS1302.h"unsigned char second;void main()
{LCD_Init();LCD_ShowString(1,1,"RTC");DS1302_WriteByte(0x8E,0x00);//前面一直显示128/129,加上这句好了DS1302_WriteByte(0x80,0x03);second=DS1302_ReadByte(0x81);LCD_ShowNum(2,1,second,3);while(1){}
}
DS1302_WriteByte(0x8E,0x00):DS1302 实时时钟模块内部有一个控制寄存器0x8E,用于设置是否允许对数据进行读写。
DS1302 的秒数是以 BCD(Binary-Coded Decimal)格式存储的。
一、时钟
DS1302.C
#include <STC89C5xRC.H>//引脚定义
sbit DS1302_SCLK=P3^6;
sbit DS1302_IO=P3^4;
sbit DS1302_CE=P3^5;
//如果给定的 SCLK 引脚在电路板上确实连接到 P3.6,那么可以直接操作 SCLK,代码会按照定义进行相应操作。
//这样定义是为了方便不同型号,只需要修改P36就可以。//时间数组,索引0~6分别为年、月、日、时、分、秒、星期
unsigned char DS1302_Time[]={25,2,14,12,59,55,5};//寄存器写入地址/指令定义
#define DS1302_SECOND 0x80
#define DS1302_MINUTE 0x82
#define DS1302_HOUR 0x84
#define DS1302_DATE 0x86
#define DS1302_MONTH 0x88
#define DS1302_DAY 0x8A
#define DS1302_YEAR 0x8C
#define DS1302_WP 0x8E/*** @brief DS1302初始化* @param 无* @retval 无*/
void DS1302_Init(void)
{DS1302_CE=0; //手册中标注平时默认为0,CE只有在1的时候才工作DS1302_SCLK=0;
}/*** @brief DS1302写一个字节* @param Command 命令字/地址* @param Data 要写入的数据* @retval 无*/
void DS1302_WriteByte(unsigned char Command,Data)
{unsigned char i;DS1302_CE=1;for(i=0;i<8;i++){DS1302_IO=Command&(0x01<<i);DS1302_SCLK=1;DS1302_SCLK=0;}/*DS1302_IO=Command&0x01; //取出第0位DS1302_SCLK=1;DS1302_SCLK=0; 这里置1后立马置0需要考虑时钟芯片的操作时间(可能反应不过来)*/for(i=0;i<8;i++){DS1302_IO=Data&(0x01<<i);DS1302_SCLK=1;DS1302_SCLK=0;}DS1302_CE=0;}/*** @brief DS1302读一个字节* @param Command 命令字/地址* @retval 读出的数据*/
unsigned char DS1302_ReadByte(unsigned char Command)
{unsigned char i,Data=0x00;//局部变量初始化不一定是0,全局默认是Command|=0x01;//这里较测试2代码改了,写的地址+1就是读的地址,不用重复定义很多地址DS1302_CE=1;for(i=0;i<8;i++){DS1302_IO=Command&(0x01<<i);DS1302_SCLK=0;DS1302_SCLK=1; //一个脉冲控制两个,一个结束另一个直接开始,要调整顺序//第二部分的脉冲操作是芯片自己负责,自己不用操作,读出只需要置1}for(i=0;i<8;i++){DS1302_SCLK=1; //最开始连续两次1,电平不会有变化(没有变化)DS1302_SCLK=0; //如果是先0后1,搞完后SCLK是1,但是最后SCLK是0if(DS1302_IO){Data|=(0x01<<i);}}DS1302_CE=0;DS1302_IO=0; //读取后将IO设置为0,否则读出的数据会出错return Data;
}/*** @brief DS1302设置时间,调用之后,DS1302_Time数组的数字会被设置到DS1302中* @param 无* @retval 无*/
void DS1302_SetTime(void)
{DS1302_WriteByte(DS1302_WP,0x00);DS1302_WriteByte(DS1302_YEAR,DS1302_Time[0]/10*16+DS1302_Time[0]%10);//十进制转BCD码后写入DS1302_WriteByte(DS1302_MONTH,DS1302_Time[1]/10*16+DS1302_Time[1]%10);DS1302_WriteByte(DS1302_DATE,DS1302_Time[2]/10*16+DS1302_Time[2]%10);DS1302_WriteByte(DS1302_HOUR,DS1302_Time[3]/10*16+DS1302_Time[3]%10);DS1302_WriteByte(DS1302_MINUTE,DS1302_Time[4]/10*16+DS1302_Time[4]%10);DS1302_WriteByte(DS1302_SECOND,DS1302_Time[5]/10*16+DS1302_Time[5]%10);DS1302_WriteByte(DS1302_DAY,DS1302_Time[6]/10*16+DS1302_Time[6]%10);DS1302_WriteByte(DS1302_WP,0x80);
}/*** @brief DS1302读取时间,调用之后,DS1302中的数据会被读取到DS1302_Time数组中* @param 无* @retval 无*/
void DS1302_ReadTime(void)
{unsigned char Temp;Temp=DS1302_ReadByte(DS1302_YEAR);DS1302_Time[0]=Temp/16*10+Temp%16;//BCD码转十进制后读取Temp=DS1302_ReadByte(DS1302_MONTH);DS1302_Time[1]=Temp/16*10+Temp%16;Temp=DS1302_ReadByte(DS1302_DATE);DS1302_Time[2]=Temp/16*10+Temp%16;Temp=DS1302_ReadByte(DS1302_HOUR);DS1302_Time[3]=Temp/16*10+Temp%16;Temp=DS1302_ReadByte(DS1302_MINUTE);DS1302_Time[4]=Temp/16*10+Temp%16;Temp=DS1302_ReadByte(DS1302_SECOND);DS1302_Time[5]=Temp/16*10+Temp%16;Temp=DS1302_ReadByte(DS1302_DAY);DS1302_Time[6]=Temp/16*10+Temp%16;
}
main.c
#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"void main()
{LCD_Init();DS1302_Init();LCD_ShowString(1,1," - - ");//静态字符初始化显示LCD_ShowString(2,1," : : ");DS1302_SetTime();//设置时间while(1){DS1302_ReadTime();//读取时间LCD_ShowNum(1,1,DS1302_Time[0],2);//显示年LCD_ShowNum(1,4,DS1302_Time[1],2);//显示月LCD_ShowNum(1,7,DS1302_Time[2],2);//显示日LCD_ShowNum(2,1,DS1302_Time[3],2);//显示时LCD_ShowNum(2,4,DS1302_Time[4],2);//显示分LCD_ShowNum(2,7,DS1302_Time[5],2);//显示秒}
}
DS1302 实时时钟芯片在初始化并设置好时间之后,会内部开始计时。每秒钟,芯片的 Second 寄存器会自动递增,从而导致在每次调用 DS1302_ReadTime() 时获取的值是自动更新的。