• DS1302与STC12的连接电路和驱动实现


    简介

    DS1302是低功耗带RAM的实时时钟电路, 常见的SOP8封装体积很小, 它可以对年月日周时分秒进行计时, 具有闰年补偿功能, 工作电压为2.0V-5.5V, 采用三线接口与CPU进行同步通信, 并可采用突发方式一次传送多个字节的时钟信号或RAM数据. DS1302内部有一个31byte的用于临时性存放数据的RAM寄存器. DS1302是DS1202的升级产品, 与DS1202兼容, 但增加了主电源/后备电源双电源引脚, 同时提供了对后备电源进行涓细电流充电的能力.

    几个需要知道的点:

    • 不带温度补偿, 所以在温度变化大的环境, 走时误差会比较大, 需要有校对机制
    • 三线接口机制与SPI相似, 但不是SPI, 因为它在同一根线上实现的双向IO
    • 自身不带电源/电容, 掉电即重置, 需要自己管理好主电备电
    • 备电的充电方式是可以通过寄存器控制的, 分三级控制, 第一级开关, 第二级可选1级或2级二极管降压, 第三级可选三种阻值
    • 额外的31个byte的存储可以自由读写
    • burst read有两处, 一处是8个字节的control+时间, 另一处是31字节的ram, 都是一次性全部读取, 但是可以编程控制读到第几个字节就停止.
    • 读取的结果是BCD码, 要转换

    pin脚结构

    对于SOP8封装, 将字面朝自己, 小圆点朝左上方, 从小圆点开始逆时针数分别为1至8脚

    Left Right
    1:Vcc2 8:Vcc1
    2:X1 7:SCLK
    3:X2 6:I/O
    4:GND 5:RST/CE
    • X1,X2: X1和X2是振荡源, 可以直接外接 32.768 kHz 晶振
    • RST: 重置
    • I/O: 数据IO, 串行数据输入输出端(双向)
    • SCLK: 串口时钟
    • Vcc1: 备用电
    • Vcc2: 供电

    Vcc1和Vcc2

    DS1302的引脚中Vcc2为主电源, Vcc1为后备电源. 在主电源关闭的情况下, 也能保持时钟的连续运行. DS1302由Vcc1或Vcc2两者中的较大者供电, 当Vcc2大于Vcc1+0.2V时, Vcc2给DS1302供电, 当Vcc2小于Vcc1时, DS1302由Vcc1供电.

    注意: 供电不要低于3.6V, 否则在秒进位时容易出现时间清零. 如果是在VCC端串联电阻限流, 电阻不要超过3K.

    RST脚

    RST是复位/片选线, 通过把RST输入驱动置高电平来启动所有的数据传送.

    • 当RST为高电平时, 允许地址/命令序列送入移位寄存器, 所有的数据传送被初始化, 允许对DS1302进行操作
    • 提供终止单字节或多字节数据传送的方法, 如果在传送过程中RST置为低电平,则会终止此次数据传送,I/O引脚变为高阻态
    • 上电运行时, 在Vcc>2.0V之前RST必须保持低电平. 只有在SCLK为低电平时, 才能将RST置为高电平.

    写入逻辑:

    1. 将RST先拉低
    2. 将SCLK拉低(如果不拉低, 且本来是高电平, 则瞬间就发出了一个bit, 这时还没放数据)
    3. 将RST拉高, 形成一个上升沿
    4. 在IO口放入一个bit (字节发送顺序:从低位到高位)
    5. 将SCLK拉高, 形成上升沿, 这时候DS1302就会将IO口的数据移入它的寄存器
    6. 拉低SCLK, 为下一个字节做准备
    7. IO口再放一个bit, 再拉高, 直到发送完所有的16个字节
    8. 最后拉低RST, 完成写入过程

    寄存器地址

    与时间相关的寄存器从0x80到0x8F, 其中偶数为写地址, 奇数为读地址, 共控制8个字节, 这8个字节可以通过burst方式一次性读出

    #define DS1302_W_SECOND     0x80
    #define DS1302_W_MINUTE     0x82
    #define DS1302_W_HOUR       0x84
    #define DS1302_W_DAY        0x86
    #define DS1302_W_MONTH      0x88
    #define DS1302_W_WEEK       0x8A
    #define DS1302_W_YEAR       0x8C
    #define DS1302_W_PROTECT    0x8E
    
    #define DS1302_R_SECOND     0x81
    #define DS1302_R_MINUTE     0x83
    #define DS1302_R_HOUR       0x85
    #define DS1302_R_DAY        0x87
    #define DS1302_R_MONTH      0x89
    #define DS1302_R_WEEK       0x8B
    #define DS1302_R_YEAR       0x8D
    #define DS1302_R_PROTECT    0x8F
    

    与控制相关的寄存器为如下几个, 也是偶数写奇数读, 这些只能单个读写

    #define DS1302_W_TK_CHARGER 0x90
    #define DS1302_W_CLK_BURST  0xBE
    #define DS1302_W_RAM_BURST  0xFE
    
    #define DS1302_R_TK_CHARGER 0x91
    #define DS1302_R_CLK_BURST  0xBF
    #define DS1302_R_RAM_BURST  0xFF
    

    片内的31个字节RAM起始地址为0xC0, 这31个字节可以通过burst方式一次性读出

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

    使用STC12 MCU读取DS1302

    接线图, 测试中可以只接Vcc2或Vcc1, 可以串电阻或串二极管限流, 代码中使用的管脚与图中不一样, 具体看代码中的说明

    实际的测试板, 接DS1302使用的是SOP8的测试座, 避免焊接, 中间的万能板仅仅是为了用一个IC座外接晶振, 可以忽略其他元件

    测试座里的DS1302

    代码例子, 包含单个读, 单个写, burst读时间, burst读RAM, 基于 HML_FwLib_STC12 封装库. 因为是非标准SPI, 所以实际上只用到了串口打印, 直接把这个方法拎出来也行, 就不需要引入这个封装库了.

    /*****************************************************************************/
    /** 
     * 连接方式
     *               |           |               |
     *               | --------- | ------------  |
     *               |1:Vcc2 VCC | 8:Vcc1        |
     *               |2:X1   OCS | 7:SCLK   P1_0 |
     *               |3:X2   OCS | 6:I/O    P1_1 |
     *               |4:GND  GND | 5:RST/CE P1_2 |
     *
     *****************************************************************************/
    #include "hml/hml.h"
    #include <stdio.h>
    
    #define DS1302_W_SECOND     0x80
    #define DS1302_W_MINUTE     0x82
    #define DS1302_W_HOUR       0x84
    #define DS1302_W_DAY        0x86
    #define DS1302_W_MONTH      0x88
    #define DS1302_W_WEEK       0x8A
    #define DS1302_W_YEAR       0x8C
    #define DS1302_W_PROTECT    0x8E
    #define DS1302_W_TK_CHARGER 0x90
    #define DS1302_W_CLK_BURST  0xBE
    #define DS1302_W_RAM_BURST  0xFE
    
    #define DS1302_R_SECOND     0x81
    #define DS1302_R_MINUTE     0x83
    #define DS1302_R_HOUR       0x85
    #define DS1302_R_DAY        0x87
    #define DS1302_R_MONTH      0x89
    #define DS1302_R_WEEK       0x8B
    #define DS1302_R_YEAR       0x8D
    #define DS1302_R_PROTECT    0x8F
    #define DS1302_R_TK_CHARGER 0x91
    #define DS1302_R_CLK_BURST  0xBF
    #define DS1302_R_RAM_BURST  0xFF
    
    #define DS1302_RAM_SIZE     0x31 // Ram Size in bytes
    #define DS1302_RAM_START    0xC0 // First byte Address
    
    #define HEX2BCD(v)  ((v) % 10 + (v) / 10 * 16)
    #define BCD2HEX(v)  ((v) % 16 + (v) / 16 * 10)
    
    #define DS1302_SCK P1_0
    #define DS1302_IO  P1_1
    #define DS1302_RST P1_2
    
    const uint8_t READ_RTC_ADDR[7]  = {0x81, 0x83, 0x85, 0x87, 0x89, 0x8b, 0x8d};
    const uint8_t WRITE_RTC_ADDR[7] = {0x80, 0x82, 0x84, 0x86, 0x88, 0x8a, 0x8c};
    uint8_t BUF[8] = { 0 };
    
    // Write one byte to DS1302
    void DS1302_WriteByte(uint8_t writeByte)
    {
        for(uint8_t i=0; i < 8; i++)
        {
            if(1 == (writeByte & 0x01)){
                DS1302_IO = 1;
            }else{
                DS1302_IO = 0;
            }
            DS1302_SCK = 1;
            DS1302_SCK = 0;
            writeByte >>= 1; // From low to high
        }
    }
     
    // Read one byte from DS1302
    uint8_t DS1302_ReadByte(void)
    {
        uint8_t dat, readByte = 0;
        for(uint8_t i = 0; i < 8; i++)
        {
            //dat = DS1302_IO;
            //readByte = (readByte>>1) | (dat << 7); // From low to high
            readByte >>= 1;
        if(DS1302_IO)
            {
          readByte |= 0x80;
        }
            DS1302_SCK = 1;
            DS1302_SCK = 0;
        }
        return readByte;
    }
    
    void DS1302_WriteReg(uint8_t addr, uint8_t value)
    {
        DS1302_RST = 0;
        DS1302_SCK = 0;
        DS1302_RST = 1;
        DS1302_WriteByte(addr);
        DS1302_WriteByte(value);
        DS1302_SCK = 1;
        DS1302_RST = 0;
    }
     
    uint8_t DS1302_ReadReg(uint8_t addr)
    {
        uint8_t readByte = 0;
        DS1302_RST = 0;
        DS1302_SCK = 0;
        DS1302_RST = 1;
        DS1302_WriteByte(addr);
        readByte = DS1302_ReadByte();
        DS1302_SCK = 1;
        DS1302_RST = 0;
        return readByte;
    }
    
    void DS1302_ReadBurst(uint8_t cmd, uint8_t len, uint8_t *buf) 
    {
        uint8_t readByte = 0;
        DS1302_RST = 0;
        DS1302_SCK = 0;
        DS1302_RST = 1;
        DS1302_WriteByte(cmd);
        while(len--)
            *buf++ = DS1302_ReadByte();
        DS1302_SCK = 1;
        DS1302_RST = 0;
    }
    
    void DS1302_Init(void)
    {
        DS1302_WriteReg(DS1302_W_PROTECT, 0x00); // write unprotect
        DS1302_WriteReg(DS1302_W_TK_CHARGER, 0x01); // stop charger
        DS1302_WriteReg(0XC0, 0x00);
        DS1302_WriteReg(0XC2, 0x01);
        DS1302_WriteReg(0XC4, 0x02);
        DS1302_WriteReg(0XC6, 0x03);
        DS1302_WriteReg(DS1302_W_PROTECT, 0x80); // write protect
    }
    
    void DS1302_print(uint8_t dat)
    {
        printf_tiny("%x%x ", dat >> 4, dat & 0x0F);
    }
    
    void DS1302_printBuf(uint8_t len)
    {
        for(uint8_t i = 0; i < len; i++)
        {
            printf_tiny("%x%x ", BUF[i] >> 4, BUF[i] & 0x0F);
        }
        printf_tiny("
    ");
    }
    
    void main(void)
    {
        UTIL_enablePrintf();
        DS1302_Init();
        while(1)
        {
            uint8_t dat = DS1302_ReadReg(DS1302_R_SECOND);
            DS1302_print(dat);
            dat = DS1302_ReadReg(DS1302_R_MINUTE);
            DS1302_print(dat);
            dat = DS1302_ReadReg(DS1302_R_HOUR);
            DS1302_print(dat);
            printf_tiny("
    ");
            DS1302_ReadBurst(DS1302_R_CLK_BURST, 8, BUF);
            DS1302_printBuf(8);
            DS1302_ReadBurst(DS1302_R_RAM_BURST, 8, BUF);
            DS1302_printBuf(8);
            sleep(491);
        }
    }
    
    

    参考

  • 相关阅读:
    迁移博客到Github Pages
    局域网传输文件的一点研究
    0CTF 2017 部分Web的某些不一样的思路
    WPScan初体验
    几种语言的CGI编程
    博客园站内短消息XSS
    nc分析代理服务器
    一个PHP混淆后门的分析
    瘟疫公司中国版(Android)手动破解内购
    参加 Tokyo Westerns / MMA CTF 2nd 2016 经验与感悟 TWCTF 2016 WriteUp
  • 原文地址:https://www.cnblogs.com/milton/p/15192778.html
Copyright © 2020-2023  润新知