15.6 DS1302通信时序介绍

DS1302 我们前边也有提起过,是三根线,分别是 CE、I/O 和 SCLK,其中 CE 是使能线,SCLK 是时钟线,I/O 是数据线。前边我们介绍过了 SPI 通信,同学们发现没发现,这个 DS1302的通信线定义和 SPI 怎么这么像呢?

事实上,DS1302 的通信是 SPI 的变异种类,它用了 SPI 的通信时序,但是通信的时候没有完全按照 SPI 的规则来,下面我们一点点解剖 DS1302 的变异 SPI 通信方式。先看一下单字节写入操作,如图 15-11 所示。

图 15-11  DS1302 单字节写操作

然后我们再对比一下 CPOL=0/CPHA=0 情况下的 SPI 的操作时序,如图 15-12 所示。
15.6 DS1302通信时序介绍_第1张图片
图 15-12  CPOL=0/CPHA=0 通信时序


图 15-11 和图 15-12 的通信时序,其中 CE 和 SSEL 的使能控制是反的,对于通信写数据,都是在 SCK 的上升沿,从机进行采样,下降沿的时候,主机发送数据。DS1302 的时序里,单片机要预先写一个字节指令,指明要写入的寄存器的地址以及后续的操作是写操作,然后再写入一个字节的数据。

对于单字节读操作,我就不做对比了,把 DS1302 的时序图贴出来,大家自己看一下即可,如图 15-13 所示。
15.6 DS1302通信时序介绍_第2张图片
图 15-13  DS1302 单字节读操作

读操作有两处需要特别注意的地方。第一,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 文件程序源代码*****************************/
(此处省略,可参考之前章节的代码)
纯文本新窗口
    
    
    
    
  1. /*****************************main.c 文件程序源代码******************************/
  2. #include <reg52.h>
  3. sbit DS1302_CE = P1^7;
  4. sbit DS1302_CK = P3^5;
  5. sbit DS1302_IO = P3^4;
  6. bit flag200ms = 0; //200ms 定时标志
  7. unsigned char T0RH = 0; //T0 重载值的高字节
  8. unsigned char T0RL = 0; //T0 重载值的低字节
  9. void ConfigTimer0(unsigned int ms);
  10. void InitDS1302();
  11. unsigned char DS1302SingleRead(unsigned char reg);
  12. extern void InitLcd1602();
  13. extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);
  14. void main(){
  15. unsigned char i;
  16. unsigned char psec=0xAA; //秒备份,初值 AA 确保首次读取时间后会刷新显示
  17. unsigned char time[8]; //当前时间数组
  18. unsigned char str[12]; //字符串转换缓冲区
  19. EA = 1; //开总中断
  20. ConfigTimer0(1); //T0 定时 1ms
  21. InitDS1302(); //初始化实时时钟
  22. InitLcd1602(); //初始化液晶
  23. while (1){
  24. if (flag200ms){ //每 200ms 读取一次时间
  25. flag200ms = 0;
  26. for (i=0; i<7; i++){ //读取 DS1302 当前时间
  27. time[i] = DS1302SingleRead(i);
  28. }
  29. if (psec != time[0]){ //检测到时间有变化时刷新显示
  30. str[0] = '2'; //添加年份的高 2 位:20
  31. str[1] = '0';
  32. str[2] = (time[6] >> 4) + '0'; //“年”高位数字转换为 ASCII 码
  33. str[3] = (time[6]&0x0F) + '0'; //“年”低位数字转换为 ASCII 码
  34. str[4] = '-'; //添加日期分隔符
  35. str[5] = (time[4] >> 4) + '0'; //“月”
  36. str[6] = (time[4]&0x0F) + '0';
  37. str[7] = '-';
  38. str[8] = (time[3] >> 4) + '0'; //“日”
  39. str[9] = (time[3]&0x0F) + '0';
  40. str[10] = '\0';
  41. LcdShowStr(0, 0, str); //显示到液晶的第一行
  42. str[0] = (time[5]&0x0F) + '0'; //“星期”
  43. str[1] = '\0';
  44. LcdShowStr(11, 0, "week");
  45. LcdShowStr(15, 0, str); //显示到液晶的第一行
  46. str[0] = (time[2] >> 4) + '0'; //“时”
  47. str[1] = (time[2]&0x0F) + '0';
  48. str[2] = ':'; //添加时间分隔符
  49. str[3] = (time[1] >> 4) + '0'; //“分”
  50. str[4] = (time[1]&0x0F) + '0';
  51. str[5] = ':';
  52. str[6] = (time[0] >> 4) + '0'; //“秒”
  53. str[7] = (time[0]&0x0F) + '0';
  54. str[8] = '\0';
  55. LcdShowStr(4, 1, str); //显示到液晶的第二行
  56. psec = time[0]; //用当前值更新上次秒数
  57. }
  58. }
  59. }
  60. }
  61. /* 发送一个字节到 DS1302 通信总线上 */
  62. void DS1302ByteWrite(unsigned char dat){
  63. unsigned char mask;
  64. for (mask=0x01; mask!=0; mask<<=1){ //低位在前,逐位移出
  65. if ((mask&dat) != 0){ //首先输出该位数据
  66. DS1302_IO = 1;
  67. }else{
  68. DS1302_IO = 0;
  69. }
  70. DS1302_CK = 1; //然后拉高时钟
  71. DS1302_CK = 0; //再拉低时钟,完成一个位的操作
  72. }
  73. DS1302_IO = 1; //最后确保释放 IO 引脚
  74. }
  75. /* 由 DS1302 通信总线上读取一个字节 */
  76. unsigned char DS1302ByteRead(){
  77. unsigned char mask;
  78. unsigned char dat = 0;
  79. for (mask=0x01; mask!=0; mask<<=1){ //低位在前,逐位读取
  80. if (DS1302_IO != 0){ //首先读取此时的 IO 引脚,并设置 dat 中的对应位
  81. dat |= mask;
  82. }
  83. DS1302_CK = 1; //然后拉高时钟
  84. DS1302_CK = 0; //再拉低时钟,完成一个位的操作
  85. }
  86. return dat; //最后返回读到的字节数据
  87. }
  88. /* 用单次写操作向某一寄存器写入一个字节,reg-寄存器地址,dat-待写入字节 */
  89. void DS1302SingleWrite(unsigned char reg, unsigned char dat){
  90. DS1302_CE = 1; //使能片选信号
  91. DS1302ByteWrite((reg<<1)|0x80); //发送写寄存器指令
  92. DS1302ByteWrite(dat); //写入字节数据
  93. DS1302_CE = 0; //除能片选信号
  94. }
  95. /* 用单次读操作从某一寄存器读取一个字节,reg-寄存器地址,返回值-读到的字节 */
  96. unsigned char DS1302SingleRead(unsigned char reg){
  97. unsigned char dat;
  98. DS1302_CE = 1; //使能片选信号
  99. DS1302ByteWrite((reg<<1)|0x81); //发送读寄存器指令
  100. dat = DS1302ByteRead()//读取字节数据
  101. DS1302_CE = 0; //除能片选信号
  102. return dat;
  103. }
  104. /* DS1302 初始化,如发生掉电则重新设置初始时间 */
  105. void InitDS1302(){
  106. unsigned char i;
  107. unsigned char code InitTime[] = { //2013 年 10 月 8 日 星期二 12:30:00
  108. 0x00,0x30,0x12, 0x08, 0x10, 0x02, 0x13
  109. };
  110. DS1302_CE = 0; //初始化 DS1302 通信引脚
  111. DS1302_CK = 0;
  112. i = DS1302SingleRead(0); //读取秒寄存器
  113. if ((i & 0x80) != 0){ //由秒寄存器最高位 CH 的值判断 DS1302 是否已停止
  114. DS1302SingleWrite(7, 0x00); //撤销写保护以允许写入数据
  115. for (i=0; i<7; i++){ //设置 DS1302 为默认的初始时间
  116. DS1302SingleWrite(i, InitTime[i]);
  117. }
  118. }
  119. }
  120. /* 配置并启动 T0,ms-T0 定时时间 */
  121. void ConfigTimer0(unsigned int ms){
  122. unsigned long tmp; //临时变量
  123. tmp = 11059200 / 12; //定时器计数频率
  124. tmp = (tmp * ms) / 1000; //计算所需的计数值
  125. tmp = 65536 - tmp; //计算定时器重载值
  126. tmp = tmp + 12; //补偿中断响应延时造成的误差
  127. T0RH = (unsigned char)(tmp>>8); //定时器重载值拆分为高低字节
  128. T0RL = (unsigned char)tmp;
  129. TMOD &= 0xF0; //清零 T0 的控制位
  130. TMOD |= 0x01; //配置 T0 为模式 1
  131. TH0 = T0RH; //加载 T0 重载值
  132. TL0 = T0RL;
  133. ET0 = 1; //使能 T0 中断
  134. TR0 = 1; //启动 T0
  135. }
  136. /* T0 中断服务函数,执行 200ms 定时 */
  137. void InterruptTimer0() interrupt 1{
  138. static unsigned char tmr200ms = 0;
  139. TH0 = T0RH; //重新加载重载值
  140. TL0 = T0RL;
  141. tmr200ms++;
  142. if (tmr200ms >= 200){ //定时 200ms
  143. tmr200ms = 0;
  144. flag200ms = 1;
  145. }
  146. }
前边学习了 I2C 和 EEPROM 的底层读写时序,那么 DS1302 的底层读写时序程序的实现方法是与之类似的,这里就不过多解释了,大家自己认真揣摩一下。

你可能感兴趣的:(15.6 DS1302通信时序介绍)