DS1302实时时钟工作原理及实验详解【时序定义、数据读写】

RTC(Real Time Clock):实时时钟,是一种集成电路,通常称为时钟芯片。

1.引脚定义和应用电路

2.内部结构

  1. 寄存器定义和命令字

寄存器:其中,第一行的CH表示时钟暂停控制位,置1表示时钟暂停,置0表示时钟静止;倒数第二行的WP表示write protect(写保护),置1表示写入操作无效;最后一行的TCS用于控制涓流充电,一般不进行设置。

命令字:

第0位表示读还是写(置1表示读,置0表示写);

第1-5位表示地址(秒地址为0,分钟地址为10,…);

第6位表示操作RAM还是时钟CK(置1表示操作RAM,置0表示操作CK);

第7位:固定为1;

举个例子,对时钟操作、秒地址操作、读:

10000001(0x81)

对时钟操作、秒地址操作、写:

10000000(0x80)

  1. 时序定义和数据读写

三个引脚:CE(使能端)、SCLK(时钟)和I/O(数据)。

规定:在时钟的上升沿,I/O口的数据将会被写入,在时钟的下降沿,时钟芯片的数据将会被读出。

根据时序图来写代码

①首先对DS1302进行引脚定义及初始化设置

//引脚定义
sbit DS1302_SCLK=P3^6;
sbit DS1302_IO=P3^4;
sbit DS1302_CE=P3^5;
 
/**
  * @brief  DS1302初始化
  * @param  无
  * @retval 无
  */
void DS1302_Init(void)
{
    DS1302_CE=0;
    DS1302_SCLK=0;
}

②向DS1302芯片中写入一个字节,从时序图可知,需要CE使能,SCLK来一个上升沿就往芯片里写入一个数据,从低位开始写。一个字节的数据输入IO口,先取高位。一个字节的数据输入IO口,先取低位。

/**
  * @brief  DS1302写一个字节
  * @param  Command 命令字/地址
  * @param  Data 要写入的数据
  * @retval 无
  */
void DS1302_WriteByte(unsigned char Command,Data)
{
      unsigned char i=0;
      DS1302_CE=1;//打开使能
      for(i=0;i<8;i++)
      {
        DS1302_IO=Command&(0x01<<i);//把Command中的数据从低位,一位一位的取出来给到IO口
        DS1302_SCLK=1;
        DS1302_SCLK=0;
      }
      for(i=0;i<8;i++)
      {
        DS1302_IO=Data&(0x01<<i);//把Data中的数据从低位,一位一位的取出来给到IO口
        DS1302_SCLK=1;
        DS1302_SCLK=0;
      }
      DS1302_CE=0;//关闭使能
//        DS1302_IO=Command&0x01;//把Command的第0位取出来
//        DS1302_SCLK=1;  //这里不需要加延时,因为单片机的运行周期是1us,而参考手册DS1302这里是 
                          纳秒级别的
//        DS1302_SCLK=0;  //来一个上升沿,把Command的第0位写进去
//    
//        DS1302_IO=Command&0x02;
//        DS1302_SCLK=1;
//        DS1302_SCLK=0;
//      .......
}

③从DS1302中读出一个字节的数据,根据时序图,需要向IO口写入一个地址数据,然后才可以从IO口中得到该地址的数据

/**
  * @brief  DS1302读一个字节
  * @param  Command 命令字/地址
  * @retval 读出的数据
  */
unsigned char DS1302_ReadByte(unsigned char Command)
{
        unsigned char i=0,Data=0x00;
        DS1302_CE=1;
        Command|=0x01; //直接把写的地址当作读的地址再给该地址最后一位置1变成读的地址,
                       //上边就少定义一个地址,而且读和写共用一个地址
        for(i=0;i<8;i++)
        {
            DS1302_IO=Command&(0x01<<i);//把Command中的数据从低位,一位一位的取出来给到IO口
            DS1302_SCLK=0;
            DS1302_SCLK=1;
        }
        for(i=0;i<8;i++)
        {
          DS1302_SCLK=1;
          DS1302_SCLK=0;
          if(DS1302_IO){Data|=(0x01<<i);}
        }
        DS1302_CE=0;
        DS1302_IO=0;//读取后将IO设置为0,否则读出的数据会出错
        return Data;
 
//        DS1302_SCLK=0;
//        DS1302_SCLK=1;
//        if(DS1302_IO){Data|=0x01;}   //这是读出来的最低位的数据,给到一个变量的最低位
//        
//        DS1302_SCLK=0;
//        DS1302_SCLK=1;
//        if(DS1302_IO){Data|=0x02;}
//        
//        DS1302_SCLK=0;
//        DS1302_SCLK=1;
//        if(DS1302_IO){Data|=0x04;}
//        .......
}

④写一个设置DS1302时间的子函数,设置时间之前必须先关闭DS1302的芯片写保护,设置完成之后再打开芯片的写保护

//秒分时日月天年的写地址定义,寄存器写入地址/指令定义
#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
 
//时间数组,索引0~6分别为年、月、日、时、分、秒、星期
unsigned char DS1302_Time[]={19,11,16,12,59,55,6};
 
/**
  * @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);
    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);//打开芯片写保护
 
}

⑤读取DS1302中的时间

/**
  * @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码,所以要把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;
}

⑥因为DS1302中的数据是以BCD码进行存储的,所以不管在设置时间或者读取时间都需要对数据进行BCD码与十进制的转换,因为我们习惯用十进制看时间。

⑦主函数

#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"
#include "Delay.h"
 
 
void main()
{
    
    LCD_Init();
    DS1302_Init();
    LCD_ShowString(1,1,"  -  -  ");//静态字符初始化显示
    LCD_ShowString(2,1,"  :  :  ");
    
    DS1302_SetTime();//设置时间
    while(1)
    {
          unsigned char i=0,j=0;
          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);//显示秒
    }
}


单字节写步骤(Write)

Step1:将CE置1;

Step2:命令字(I/O)发两个字节:第一个字节是命令字(先发最低位R/W),第二个字节是数据;【一位一位依次发送】

Step3:SCLK给上升沿,将命令字最低位写入单片机;

Step4:将SCLK置回0;

Step5:SCLK给上升沿,将命令字次低位写入单片机;

Step6:将SCLK置回0;

…(依次循环8次)

Step7:SCLK给上升沿,将数据最低位写入单片机;

Step8:将SCLK置回0;

Step9:SCLK给上升沿,将数据次低位写入单片机;

Step10:将SCLK置回0;

…(依次循环8次)

Step11:将CE置0;

(每个上升沿写入数据)

//单字节写,Command为命令字(地址),Data为要写入的数据
void DS1302_WriteByte(unsigned char Command,Data){
    unsigned char i;
    DS1302_CE=1;
    for(i=0;i<8;i++){
        DS1302_IO=Command&(0x01<<i);//0000 0001 左移
        DS1302_SCLK=1;
        DS1302_SCLK=0;
    }
    for(i=0;i<8;i++){
        DS1302_IO=Data&(0x01<<i);//0000 0001 左移
        DS1302_SCLK=1;
        DS1302_SCLK=0;
    }
    DS1302_CE=0;
}
单字节读步骤(Read)

Step1:将CE置1;

Step2:命令字(I/O)发两个字节:第一个字节是命令字(先发最低位R/W),第二个字节是数据;【一位一位依次发送】

Step3:SCLK给上升沿,将命令字最低位写入单片机;

Step4:将SCLK置回0;

Step5:SCLK给上升沿,将命令字次低位写入单片机;

Step6:将SCLK置回0;

…(依次循环8次)

Step7:SCLK给下降沿,将数据最低位读出;

Step8:将SCLK置回0;

Step9:SCLK给下降沿,将数据次低位读出;

Step10:将SCLK置回0;

…(依次循环8次)

Step11:将CE置0;

Step12:将I/O置0;

(每个下降沿读出数据)

【注意】:需要注意的是,从上述时序图中可以看出,单字节写(Write)有16个脉冲,而单字节读(Read)只有15个脉冲,因为当最后一个命令字的上升沿之后的下降沿数据马上就读出来了,如下图所示。

因此这里编写代码时需要调整顺序,对DS1302_SCLK先给0再给1,正好进行切分:

//DS1302读一个字节,Command为命令字(地址),Data为读出的数据
unsigned char DS1302_ReadByte(unsigned char Command){
    unsigned char i,Data=0x00;
    DS1302_CE=1;
    for(i=0;i<8;i++){
        DS1302_IO=Command&(0x01<<i);//0000 0001 左移
//        DS1302_SCLK=1;
//        DS1302_SCLK=0;
        DS1302_SCLK=0;
        DS1302_SCLK=1;        
    }
    for(i=0;i<8;i++){
        DS1302_SCLK=1;
        DS1302_SCLK=0;
        if(DS1302_IO){Data=Data|(0x01<<i);}
    }
    DS1302_CE=0;
    DS1302_IO=0;//清零

    return Data;
}

4.可调时钟代码()

物联沃分享整理
物联沃-IOTWORD物联网 » DS1302实时时钟工作原理及实验详解【时序定义、数据读写】

发表评论