找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開(kāi)始

搜索
查看: 3804|回復(fù): 2
收起左側(cè)

DS1302與STC12單片機(jī)的連接電路和驅(qū)動(dòng)實(shí)現(xiàn)

[復(fù)制鏈接]
ID:912806 發(fā)表于 2021-8-27 11:07 | 顯示全部樓層 |閱讀模式
DS1302是低功耗帶RAM的實(shí)時(shí)時(shí)鐘電路, 常見(jiàn)的SOP8封裝體積很小, 它可以對(duì)年月日周時(shí)分秒進(jìn)行計(jì)時(shí), 具有閏年補(bǔ)償功能, 工作電壓為2.0V-5.5V, 采用三線接口與CPU進(jìn)行同步通信, 并可采用突發(fā)方式一次傳送多個(gè)字節(jié)的時(shí)鐘信號(hào)或RAM數(shù)據(jù). DS1302內(nèi)部有一個(gè)31byte的用于臨時(shí)性存放數(shù)據(jù)的RAM寄存器. DS1302是DS1202的升級(jí)產(chǎn)品, 與DS1202兼容, 但增加了主電源/后備電源雙電源引腳, 同時(shí)提供了對(duì)后備電源進(jìn)行涓細(xì)電流充電的能力.
幾個(gè)需要知道的點(diǎn):
  • 不帶溫度補(bǔ)償, 所以在溫度變化大的環(huán)境, 走時(shí)誤差會(huì)比較大, 需要有校對(duì)機(jī)制
  • 三線接口機(jī)制與SPI相似, 但不是SPI, 因?yàn)樗谕桓上實(shí)現(xiàn)的雙向IO
  • 自身不帶電源/電容, 掉電即重置, 需要自己管理好主電備電
  • 備電的充電方式是可以通過(guò)寄存器控制的, 分三級(jí)控制, 第一級(jí)開(kāi)關(guān), 第二級(jí)可選1級(jí)或2級(jí)二極管降壓, 第三級(jí)可選三種阻值
  • 額外的31個(gè)byte的存儲(chǔ)可以自由讀寫(xiě)
  • burst read有兩處, 一處是8個(gè)字節(jié)的control+時(shí)間, 另一處是31字節(jié)的ram, 都是一次性全部讀取, 但是可以編程控制讀到第幾個(gè)字節(jié)就停止.
  • 讀取的結(jié)果是BCD碼, 要轉(zhuǎn)換


Vcc1和Vcc2
DS1302的引腳中Vcc2為主電源, Vcc1為后備電源. 在主電源關(guān)閉的情況下, 也能保持時(shí)鐘的連續(xù)運(yùn)行. DS1302由Vcc1或Vcc2兩者中的較大者供電, 當(dāng)Vcc2大于Vcc1+0.2V時(shí), Vcc2給DS1302供電, 當(dāng)Vcc2小于Vcc1時(shí), DS1302由Vcc1供電.
注意: 供電不要低于3.6V, 否則在秒進(jìn)位時(shí)容易出現(xiàn)時(shí)間清零. 如果是在VCC端串聯(lián)電阻限流, 電阻不要超過(guò)3K.
RST腳
RST是復(fù)位/片選線, 通過(guò)把RST輸入驅(qū)動(dòng)置高電平來(lái)啟動(dòng)所有的數(shù)據(jù)傳送.
  • 當(dāng)RST為高電平時(shí), 允許地址/命令序列送入移位寄存器, 所有的數(shù)據(jù)傳送被初始化, 允許對(duì)DS1302進(jìn)行操作
  • 提供終止單字節(jié)或多字節(jié)數(shù)據(jù)傳送的方法, 如果在傳送過(guò)程中RST置為低電平,則會(huì)終止此次數(shù)據(jù)傳送,I/O引腳變?yōu)楦咦钁B(tài)
  • 上電運(yùn)行時(shí), 在Vcc>2.0V之前RST必須保持低電平. 只有在SCLK為低電平時(shí), 才能將RST置為高電平.
寫(xiě)入邏輯:
  • 將RST先拉低
  • 將SCLK拉低(如果不拉低, 且本來(lái)是高電平, 則瞬間就發(fā)出了一個(gè)bit, 這時(shí)還沒(méi)放數(shù)據(jù))
  • 將RST拉高, 形成一個(gè)上升沿
  • 在IO口放入一個(gè)bit (字節(jié)發(fā)送順序:從低位到高位)
  • 將SCLK拉高, 形成上升沿, 這時(shí)候DS1302就會(huì)將IO口的數(shù)據(jù)移入它的寄存器
  • 拉低SCLK, 為下一個(gè)字節(jié)做準(zhǔn)備
  • IO口再放一個(gè)bit, 再拉高, 直到發(fā)送完所有的16個(gè)字節(jié)
  • 最后拉低RST, 完成寫(xiě)入過(guò)程
寄存器地址
與時(shí)間相關(guān)的寄存器從0x80到0x8F, 其中偶數(shù)為寫(xiě)地址, 奇數(shù)為讀地址, 共控制8個(gè)字節(jié), 這8個(gè)字節(jié)可以通過(guò)burst方式一次性讀出
  1. #define DS1302_W_SECOND     0x80
  2. #define DS1302_W_MINUTE     0x82
  3. #define DS1302_W_HOUR       0x84
  4. #define DS1302_W_DAY        0x86
  5. #define DS1302_W_MONTH      0x88
  6. #define DS1302_W_WEEK       0x8A
  7. #define DS1302_W_YEAR       0x8C
  8. #define DS1302_W_PROTECT    0x8E

  9. #define DS1302_R_SECOND     0x81
  10. #define DS1302_R_MINUTE     0x83
  11. #define DS1302_R_HOUR       0x85
  12. #define DS1302_R_DAY        0x87
  13. #define DS1302_R_MONTH      0x89
  14. #define DS1302_R_WEEK       0x8B
  15. #define DS1302_R_YEAR       0x8D
  16. #define DS1302_R_PROTECT    0x8F
復(fù)制代碼

與控制相關(guān)的寄存器為如下幾個(gè), 也是偶數(shù)寫(xiě)奇數(shù)讀, 這些只能單個(gè)讀寫(xiě)

  1. #define DS1302_W_TK_CHARGER 0x90
  2. #define DS1302_W_CLK_BURST  0xBE
  3. #define DS1302_W_RAM_BURST  0xFE

  4. #define DS1302_R_TK_CHARGER 0x91
  5. #define DS1302_R_CLK_BURST  0xBF
  6. #define DS1302_R_RAM_BURST  0xFF
復(fù)制代碼
片內(nèi)的31個(gè)字節(jié)RAM起始地址為0xC0, 這31個(gè)字節(jié)可以通過(guò)burst方式一次性讀出
  1. #define DS1302_RAM_SIZE     0x31 // Ram Size in bytes
  2. #define DS1302_RAM_START    0xC0 // First byte Address
復(fù)制代碼

使用STC12 MCU讀取DS1302
接線圖, 測(cè)試中可以只接Vcc2或Vcc1, 可以串電阻或串二極管限流,
DS1302-sch3.png
實(shí)際的測(cè)試板, 接DS1302使用的是SOP8的測(cè)試座, 避免焊接, 中間的萬(wàn)能板僅僅是為了用一個(gè)IC座外接晶振, 可以忽略其他元件
DS1302-test2.jpg
測(cè)試座里的DS1302
DS1302-test1.jpg
代碼例子, 包含單個(gè)讀, 單個(gè)寫(xiě), burst讀時(shí)間, burst讀RAM, 基于 HML_FwLib_STC12 封裝庫(kù). 因?yàn)槭欠菢?biāo)準(zhǔn)SPI, 所以實(shí)際上只用到了串口打印, 直接把這個(gè)方法拎出來(lái)也行, 就不需要引入這個(gè)封裝庫(kù)了.
  1. /*****************************************************************************/
  2. /**
  3. * 連接方式
  4. *               |           |               |
  5. *               | --------- | ------------  |
  6. *               |1:Vcc2 VCC | 8:Vcc1        |
  7. *               |2:X1   OCS | 7:SCLK   P1_0 |
  8. *               |3:X2   OCS | 6:I/O    P1_1 |
  9. *               |4:GND  GND | 5:RST/CE P1_2 |
  10. *
  11. *****************************************************************************/
  12. #include "hml/hml.h"
  13. #include <stdio.h>

  14. #define DS1302_W_SECOND     0x80
  15. #define DS1302_W_MINUTE     0x82
  16. #define DS1302_W_HOUR       0x84
  17. #define DS1302_W_DAY        0x86
  18. #define DS1302_W_MONTH      0x88
  19. #define DS1302_W_WEEK       0x8A
  20. #define DS1302_W_YEAR       0x8C
  21. #define DS1302_W_PROTECT    0x8E
  22. #define DS1302_W_TK_CHARGER 0x90
  23. #define DS1302_W_CLK_BURST  0xBE
  24. #define DS1302_W_RAM_BURST  0xFE

  25. #define DS1302_R_SECOND     0x81
  26. #define DS1302_R_MINUTE     0x83
  27. #define DS1302_R_HOUR       0x85
  28. #define DS1302_R_DAY        0x87
  29. #define DS1302_R_MONTH      0x89
  30. #define DS1302_R_WEEK       0x8B
  31. #define DS1302_R_YEAR       0x8D
  32. #define DS1302_R_PROTECT    0x8F
  33. #define DS1302_R_TK_CHARGER 0x91
  34. #define DS1302_R_CLK_BURST  0xBF
  35. #define DS1302_R_RAM_BURST  0xFF

  36. #define DS1302_RAM_SIZE     0x31 // Ram Size in bytes
  37. #define DS1302_RAM_START    0xC0 // First byte Address

  38. #define HEX2BCD(v)  ((v) % 10 + (v) / 10 * 16)
  39. #define BCD2HEX(v)  ((v) % 16 + (v) / 16 * 10)

  40. #define DS1302_SCK P1_0
  41. #define DS1302_IO  P1_1
  42. #define DS1302_RST P1_2

  43. const uint8_t READ_RTC_ADDR[7]  = {0x81, 0x83, 0x85, 0x87, 0x89, 0x8b, 0x8d};
  44. const uint8_t WRITE_RTC_ADDR[7] = {0x80, 0x82, 0x84, 0x86, 0x88, 0x8a, 0x8c};
  45. uint8_t BUF[8] = { 0 };

  46. // Write one byte to DS1302
  47. void DS1302_WriteByte(uint8_t writeByte)
  48. {
  49.     for(uint8_t i=0; i < 8; i++)
  50.     {
  51.         if(1 == (writeByte & 0x01)){
  52.             DS1302_IO = 1;
  53.         }else{
  54.             DS1302_IO = 0;
  55.         }
  56.         DS1302_SCK = 1;
  57.         DS1302_SCK = 0;
  58.         writeByte >>= 1; // From low to high
  59.     }
  60. }

  61. // Read one byte from DS1302
  62. uint8_t DS1302_ReadByte(void)
  63. {
  64.     uint8_t dat, readByte = 0;
  65.     for(uint8_t i = 0; i < 8; i++)
  66.     {
  67.         //dat = DS1302_IO;
  68.         //readByte = (readByte>>1) | (dat << 7); // From low to high
  69.         readByte >>= 1;
  70.     if(DS1302_IO)
  71.         {
  72.       readByte |= 0x80;
  73.     }
  74.         DS1302_SCK = 1;
  75.         DS1302_SCK = 0;
  76.     }
  77.     return readByte;
  78. }

  79. void DS1302_WriteReg(uint8_t addr, uint8_t value)
  80. {
  81.     DS1302_RST = 0;
  82.     DS1302_SCK = 0;
  83.     DS1302_RST = 1;
  84.     DS1302_WriteByte(addr);
  85.     DS1302_WriteByte(value);
  86.     DS1302_SCK = 1;
  87.     DS1302_RST = 0;
  88. }

  89. uint8_t DS1302_ReadReg(uint8_t addr)
  90. {
  91.     uint8_t readByte = 0;
  92.     DS1302_RST = 0;
  93.     DS1302_SCK = 0;
  94.     DS1302_RST = 1;
  95.     DS1302_WriteByte(addr);
  96.     readByte = DS1302_ReadByte();
  97.     DS1302_SCK = 1;
  98.     DS1302_RST = 0;
  99.     return readByte;
  100. }

  101. void DS1302_ReadBurst(uint8_t cmd, uint8_t len, uint8_t *buf)
  102. {
  103.     uint8_t readByte = 0;
  104.     DS1302_RST = 0;
  105.     DS1302_SCK = 0;
  106.     DS1302_RST = 1;
  107.     DS1302_WriteByte(cmd);
  108.     while(len--)
  109.         *buf++ = DS1302_ReadByte();
  110.     DS1302_SCK = 1;
  111.     DS1302_RST = 0;
  112. }

  113. void DS1302_Init(void)
  114. {
  115.     DS1302_WriteReg(DS1302_W_PROTECT, 0x00); // write unprotect
  116.     DS1302_WriteReg(DS1302_W_TK_CHARGER, 0x01); // stop charger
  117.     DS1302_WriteReg(0XC0, 0x00);
  118.     DS1302_WriteReg(0XC2, 0x01);
  119.     DS1302_WriteReg(0XC4, 0x02);
  120.     DS1302_WriteReg(0XC6, 0x03);
  121.     DS1302_WriteReg(DS1302_W_PROTECT, 0x80); // write protect
  122. }

  123. void DS1302_print(uint8_t dat)
  124. {
  125.     printf_tiny("%x%x ", dat >> 4, dat & 0x0F);
  126. }

  127. void DS1302_printBuf(uint8_t len)
  128. {
  129.     for(uint8_t i = 0; i < len; i++)
  130.     {
  131.         printf_tiny("%x%x ", BUF[i] >> 4, BUF[i] & 0x0F);
  132.     }
  133.     printf_tiny("\n");
  134. }

  135. void main(void)
  136. {
  137.     UTIL_enablePrintf();
  138.     DS1302_Init();
  139.     while(1)
  140.     {
  141.         uint8_t dat = DS1302_ReadReg(DS1302_R_SECOND);
  142.         DS1302_print(dat);
  143.         dat = DS1302_ReadReg(DS1302_R_MINUTE);
  144.         DS1302_print(dat);
  145.         dat = DS1302_ReadReg(DS1302_R_HOUR);
  146.         DS1302_print(dat);
  147.         printf_tiny("\n");
  148.         DS1302_ReadBurst(DS1302_R_CLK_BURST, 8, BUF);
  149.         DS1302_printBuf(8);
  150.         DS1302_ReadBurst(DS1302_R_RAM_BURST, 8, BUF);
  151.         DS1302_printBuf(8);
  152.         sleep(491);
  153.     }
  154. }
復(fù)制代碼

我的8051筆記都在這個(gè)標(biāo)簽下: https://www.cnblogs.com/milton/tag/8051/


評(píng)分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎(jiǎng)勵(lì)!

查看全部評(píng)分

回復(fù)

使用道具 舉報(bào)

ID:624769 發(fā)表于 2021-8-28 16:39 | 顯示全部樓層
糾正一下, DS1302, 是SPI, 是早期的半雙工SPI, 也叫雙線制SPI(不算CE,現(xiàn)在普遍的都是三線全雙工SPI),如果你手頭有CH551這類低端的單片機(jī),你會(huì)發(fā)現(xiàn)SPI設(shè)置成雙線半雙工模式后,可以直接用SPI驅(qū)動(dòng)DS1302。效率也高,代碼也簡(jiǎn)單。
回復(fù)

使用道具 舉報(bào)

ID:912806 發(fā)表于 2021-8-29 21:16 | 顯示全部樓層
188610329 發(fā)表于 2021-8-28 16:39
糾正一下, DS1302, 是SPI, 是早期的半雙工SPI, 也叫雙線制SPI(不算CE,現(xiàn)在普遍的都是三線全雙工SPI),如 ...

多謝指導(dǎo), 長(zhǎng)知識(shí)了.
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表