DS1302 我们前边也有提起过,是三根线,分别是 CE、I/O 和 SCLK,其中 CE 是使能线,SCLK 是时钟线,I/O 是数据线。前边我们介绍过了 SPI 通信,同学们发现没发现,这个 DS1302的通信线定义和 SPI 怎么这么像呢? 
  事实上,DS1302 的通信是 SPI 的变异种类,它用了 SPI 的通信时序,但是通信的时候没有完全按照 SPI 的规则来,下面我们一点点解剖 DS1302 的变异 SPI 通信方式。先看一下单字节写入操作,如图 15-11 所示。 
                  
   图 15-11 和图 15-12 的通信时序,其中 CE 和 SSEL 的使能控制是反的,对于通信写数据,都是在 SCK 的上升沿,从机进行采样,下降沿的时候,主机发送数据。DS1302 的时序里,单片机要预先写一个字节指令,指明要写入的寄存器的地址以及后续的操作是写操作,然后再写入一个字节的数据。 
  对于单字节读操作,我就不做对比了,把 DS1302 的时序图贴出来,大家自己看一下即可,如图 15-13 所示。 
                   
   读操作有两处需要特别注意的地方。第一,DS1302 的时序图上的箭头都是针对 DS1302来说的,因此读操作的时候,先写第一个字节指令,上升沿的时候 DS1302 来锁存数据,下降沿我们用单片机发送数据。到了第二个字数据,由于我们这个时序过程相当于CPOL=0/CPHA=0,前沿发送数据,后沿读取数据,第二个字节是 DS1302 下降沿输出数据,我们的单片机上升沿来读取,因此箭头从 DS1302 角度来说,出现在了下降沿。 
  第二个需要注意的地方就是,我们的单片机没有标准的 SPI 接口,和 I2C 一样需要用 IO口来模拟通信过程。在读 DS1302 的时候,理论上 SPI 是上升沿读取,但是程序是用 IO 口模拟的,所以数据的读取和时钟沿的变化不可能同时了,必然就有一个先后顺序。通过实验发现,如果先读取 IO 线上的数据,再拉高 SCLK 产生上升沿,那么读到的数据一定是正确的,而颠倒顺序后数据就有可能出错。这个问题产生的原因还是在于 DS1302 的通信协议与标准SPI 协议存在的差异造成的,如果是标准 SPI 的数据线,数据会一直保持到下一个周期的下降沿才会变化,所以读取数据和上升沿的先后顺序就无所谓了;但 DS1302 的 IO 线会在时钟上升沿后被 DS1302 释放,也就是撤销强推挽输出变为弱下拉状态,而此时在 51 单片机引脚内部上拉的作用下,IO 线上的实际电平会慢慢上升,从而导致在上升沿产生后再读取 IO 数据的话就可能会出错。因此这里的程序我们按照先读取 IO 数据,再拉高 SCLK 产生上升沿的顺序。 
  下面我们就写一个程序,先将 2013 年 10 月 8 号星期二 12 点 30 分 00 秒这个时间写到DS1302 内部,让 DS1302 正常运行,然后再不停的读取 DS1302 的当前时间,并显示在我们的液晶屏上。 
  /***************************Lcd1602.c 文件程序源代码*****************************/
  (此处省略,可参考之前章节的代码)
 
-   
- #include <reg52.h>  
-   
- sbit DS1302_CE = P1^7;  
- sbit DS1302_CK = P3^5;  
- sbit DS1302_IO = P3^4;  
- bit flag200ms = 0;   
- unsigned char T0RH = 0;   
- unsigned char T0RL = 0;   
-   
- void ConfigTimer0(unsigned int ms);  
- void InitDS1302();  
- unsigned char DS1302SingleRead(unsigned char reg);  
- extern void InitLcd1602();  
- extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);  
-   
- void main(){  
-     unsigned char i;  
-     unsigned char psec=0xAA;   
-     unsigned char time[8];   
-     unsigned char str[12];   
-     EA = 1;   
-     ConfigTimer0(1);   
-     InitDS1302();   
-     InitLcd1602();   
-      
-     while (1){  
-         if (flag200ms){   
-             flag200ms = 0;  
-             for (i=0; i<7; i++){   
-                 time[i] = DS1302SingleRead(i);  
-             }  
-             if (psec != time[0]){   
-                 str[0] = '2';   
-                 str[1] = '0';  
-                 str[2] = (time[6] >> 4) + '0';   
-                 str[3] = (time[6]&0x0F) + '0';   
-                 str[4] = '-';   
-                 str[5] = (time[4] >> 4) + '0';   
-                 str[6] = (time[4]&0x0F) + '0';  
-                 str[7] = '-';  
-                 str[8] = (time[3] >> 4) + '0';   
-                 str[9] = (time[3]&0x0F) + '0';  
-                 str[10] = '\0';  
-                 LcdShowStr(0, 0, str);   
-                 str[0] = (time[5]&0x0F) + '0';   
-                 str[1] = '\0';  
-                 LcdShowStr(11, 0, "week");  
-                 LcdShowStr(15, 0, str);   
-                 str[0] = (time[2] >> 4) + '0';   
-                 str[1] = (time[2]&0x0F) + '0';  
-                 str[2] = ':';   
-                 str[3] = (time[1] >> 4) + '0';   
-                 str[4] = (time[1]&0x0F) + '0';  
-                 str[5] = ':';  
-                 str[6] = (time[0] >> 4) + '0';   
-                 str[7] = (time[0]&0x0F) + '0';  
-                 str[8] = '\0';  
-                 LcdShowStr(4, 1, str);   
-                 psec = time[0];   
-             }  
-         }  
-     }  
- }  
-   
- void DS1302ByteWrite(unsigned char dat){  
-     unsigned char mask;  
-   
-     for (mask=0x01; mask!=0; mask<<=1){   
-         if ((mask&dat) != 0){   
-             DS1302_IO = 1;  
-         }else{  
-             DS1302_IO = 0;  
-         }  
-         DS1302_CK = 1;   
-         DS1302_CK = 0;   
-     }  
-     DS1302_IO = 1;   
- }  
-   
- unsigned char DS1302ByteRead(){  
-     unsigned char mask;  
-     unsigned char dat = 0;  
-      
-     for (mask=0x01; mask!=0; mask<<=1){   
-         if (DS1302_IO != 0){   
-             dat |= mask;  
-         }  
-         DS1302_CK = 1;   
-         DS1302_CK = 0;   
-     }  
-     return dat;   
- }  
-   
- void DS1302SingleWrite(unsigned char reg, unsigned char dat){  
-     DS1302_CE = 1;   
-     DS1302ByteWrite((reg<<1)|0x80);   
-     DS1302ByteWrite(dat);   
-     DS1302_CE = 0;   
- }  
-   
- unsigned char DS1302SingleRead(unsigned char reg){  
-     unsigned char dat;  
-     DS1302_CE = 1;   
-     DS1302ByteWrite((reg<<1)|0x81);   
-     dat = DS1302ByteRead()  
-     DS1302_CE = 0;   
-     return dat;  
- }  
-   
- void InitDS1302(){  
-     unsigned char i;  
-     unsigned char code InitTime[] = {   
-         0x00,0x30,0x12, 0x08, 0x10, 0x02, 0x13  
-     };  
-      
-     DS1302_CE = 0;   
-     DS1302_CK = 0;  
-     i = DS1302SingleRead(0);   
-      
-     if ((i & 0x80) != 0){   
-         DS1302SingleWrite(7, 0x00);   
-         for (i=0; i<7; i++){   
-             DS1302SingleWrite(i, InitTime[i]);  
-         }  
-     }  
- }  
-   
- void ConfigTimer0(unsigned int ms){  
-     unsigned long tmp;   
-     tmp = 11059200 / 12;   
-     tmp = (tmp * ms) / 1000;   
-     tmp = 65536 - tmp;   
-     tmp = tmp + 12;   
-     T0RH = (unsigned char)(tmp>>8);   
-     T0RL = (unsigned char)tmp;  
-     TMOD &= 0xF0;   
-     TMOD |= 0x01;   
-     TH0 = T0RH;   
-     TL0 = T0RL;  
-     ET0 = 1;   
-     TR0 = 1;   
- }  
-   
- void InterruptTimer0() interrupt 1{  
-     static unsigned char tmr200ms = 0;  
-     TH0 = T0RH;   
-     TL0 = T0RL;  
-     tmr200ms++;  
-     if (tmr200ms >= 200){   
-         tmr200ms = 0;  
-         flag200ms = 1;  
-     }  
- }   
 
  前边学习了 I2C 和 EEPROM 的底层读写时序,那么 DS1302 的底层读写时序程序的实现方法是与之类似的,这里就不过多解释了,大家自己认真揣摩一下。