DS1302 的 BURST 模式

发布者:pingbashou最新更新时间:2017-11-16 来源: eefocus关键字:DS1302  BURST  模式 手机看文章 扫描二维码
随时随地手机看文章

进行产品开发的时候,逻辑的严谨性非常重要,如果一个产品或者程序逻辑上不严谨,就有可能出现功能上的错误。比如我们15.3.4节里的这个程序,我们再回顾一下,当单片机定时器时间到了 200 ms 后,我们连续把 DS1302 的时间参数的7个字节读了出来。但是不管怎么读,都会有一个时间差,在极端的情况下就会出现这样一种情况:假如我们当前的时间是00:00:59,我们先读秒,读到的秒是59,然后再去读分钟,而就在读完秒到还未开始读分钟的这段时间内,刚好时间进位了,变成了00:01:00这个时间,我们读到的分钟就是01,显示在液晶上就会出现一个00:01:59,这个时间很明显是错误的。出现这个问题的概率极小,但却是实实在在可能存在的。

为了解决这个问题,芯片厂家肯定要给我们提供一种解决方案,这就是 DS1302 的突发模式。突发模式也分为 RAM 突发模式和时钟突发模式,RAM 部分我们不讲,我们只看和时钟相关的 clock burst mode。

当我们写指令到 DS1302 的时候,只要我们将要写的5位地址全部写1,即读操作用 0xBF,写操作用 0xBE,这样的指令送给 DS1302 之后,它就会自动识别出来是 burst 模式,马上把所有的8个字节同时锁存到另外的8个字节的寄存器缓冲区内,这样时钟继续走,而我们读数据是从另外一个缓冲区内读取的。同样的道理,如果我们用 burst 模式写数据,那么我们也是先写到这个缓冲区内,最终 DS1302 会把这个缓冲区内的数据一次性送到它的时钟寄存器内。

要注意的是,不管是读还是写,只要使用时钟的 burst 模式,则必须一次性读写8个寄存器,要把时钟的寄存器完全读出来或者完全写进去。

下边就提供一个 burst 模式的例程给大家学习一下,程序的功能还是与上一节一样的。 /*Lcd1602.c 文件程序源代码***/ (此处省略,可参考之前章节的代码)

/*****************************main.c 文件程序源代码******************************/
#include 

sbit DS1302_CE = P1^7;
sbit DS1302_CK = P3^5;
sbit DS1302_IO = P3^4;

bit flag200ms = 0; //200ms 定时标志
unsigned char T0RH = 0; //T0 重载值的高字节
unsigned char T0RL = 0; //T0 重载值的低字节

void ConfigTimer0(unsigned int ms);
void InitDS1302();
void DS1302BurstRead(unsigned char *dat);
extern void InitLcd1602();
extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);

void main(){
    unsigned char psec=0xAA; //秒备份,初值 AA 确保首次读取时间后会刷新显示
    unsigned char time[8]; //当前时间数组
    unsigned char str[12]; //字符串转换缓冲区

    EA = 1; //开总中断
    ConfigTimer0(1); //T0 定时 1ms
    InitDS1302(); //初始化实时时钟
    InitLcd1602(); //初始化液晶

    while (1){
        if (flag200ms){ //每 200ms 读取依次时间
            flag200ms = 0;
            DS1302BurstRead(time); //读取 DS1302 当前时间

            if (psec != time[0]){ //检测到时间有变化时刷新显示
                str[0] = '2'; //添加年份的高 2 位:20
                str[1] = '0';
                str[2] = (time[6] >> 4) + '0'; //“年”高位数字转换为 ASCII 码
                str[3] = (time[6]&0x0F) + '0'; //“年”低位数字转换为 ASCII 码
                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]; //用当前值更新上次秒数
            }
        }
    }
}

/* 发送一个字节到 DS1302 通信总线上 */
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; //最后确保释放 IO 引脚
}
/* 由 DS1302 通信总线上读取一个字节 */
unsigned char DS1302ByteRead(){
    unsigned char mask;
    unsigned char dat = 0;

    for (mask=0x01; mask!=0; mask<<=1){ //低位在前,逐位读取
        if (DS1302_IO != 0){ //首先读取此时的 IO 引脚,并设置 dat 中的对应位
            dat |= mask;
        }
        DS1302_CK = 1; //然后拉高时钟
        DS1302_CK = 0; //再拉低时钟,完成一个位的操作
    }
    return dat; //最后返回读到的字节数据
}
/* 用单次写操作向某一寄存器写入一个字节,reg-寄存器地址,dat-待写入字节 */
void DS1302SingleWrite(unsigned char reg, unsigned char dat){
    DS1302_CE = 1; //使能片选信号
    DS1302ByteWrite((reg<<1)|0x80); //发送写寄存器指令
    DS1302ByteWrite(dat); //写入字节数据
    DS1302_CE = 0; //除能片选信号
}
/* 用单次读操作从某一寄存器读取一个字节,reg-寄存器地址,返回值-读到的字节 */
unsigned char DS1302SingleRead(unsigned char reg){
    unsigned char dat;

    DS1302_CE = 1; //使能片选信号
    DS1302ByteWrite((reg<<1)|0x81); //发送读寄存器指令
    dat = DS1302ByteRead(); //读取字节数据
    DS1302_CE = 0; //除能片选信号
    return dat;
}
/* 用突发模式连续写入 8 个寄存器数据,dat-待写入数据指针 */
void DS1302BurstWrite(unsigned char *dat){
    unsigned char i;

    DS1302_CE = 1;
    DS1302ByteWrite(0xBE); //发送突发写寄存器指令

    for (i=0; i<8; i++){ //连续写入 8 字节数据
        DS1302ByteWrite(dat[i]);
    }
    DS1302_CE = 0;
}
/* 用突发模式连续读取 8 个寄存器的数据,dat-读取数据的接收指针 */
void DS1302BurstRead(unsigned char *dat){
    unsigned char i;

    DS1302_CE = 1;
    DS1302ByteWrite(0xBF); //发送突发读寄存器指令
    for (i=0; i<8; i++){ //连续读取 8 个字节
        dat[i] = DS1302ByteRead();
    }
    DS1302_CE = 0;
}
/* DS1302 初始化,如发生掉电则重新设置初始时间 */
void InitDS1302(){
    unsigned char dat;
    unsigned char code InitTime[] = { //2013 年 10 月 8 日 星期二 12:30:00
        0x00,0x30,0x12, 0x08, 0x10, 0x02, 0x13
    };

    DS1302_CE = 0; //初始化 DS1302 通信引脚
    DS1302_CK = 0;
    dat = DS1302SingleRead(0); //读取秒寄存器

    if ((dat & 0x80) != 0){ //由秒寄存器最高位 CH 的值判断 DS1302 是否已停止
        DS1302SingleWrite(7, 0x00); //撤销写保护以允许写入数据
        DS1302BurstWrite(InitTime); //设置 DS1302 为默认的初始时间
    }
}
/* 配置并启动 T0,ms-T0 定时时间 */
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; //清零 T0 的控制位
    TMOD |= 0x01; //配置 T0 为模式 1
    TH0 = T0RH; //加载 T0 重载值
    TL0 = T0RL;
    ET0 = 1; //使能 T0 中断
    TR0 = 1; //启动 T0
}
/* T0 中断服务函数,执行 200ms 定时 */
void InterruptTimer0() interrupt 1{
    static unsigned char tmr200ms = 0;
    TH0 = T0RH; //重新加载重载值
    TL0 = T0RL;
    tmr200ms++;
    if (tmr200ms >= 200){ //定时 200ms
        tmr200ms = 0;
        flag200ms = 1;
    }
}


关键字:DS1302  BURST  模式 引用地址:DS1302 的 BURST 模式

上一篇:DS1302 通信时序介绍
下一篇:单片机 RS485 通信接口、控制线、原理图及程序实例

推荐阅读最新更新时间:2024-03-16 15:45

MSP430G2553 WDT的看门狗模式例子
#include io430.h int main( void ) { int i; // Stop watchdog timer to prevent time out reset WDTCTL = WDTPW + WDTHOLD; P1DIR |= BIT0; P1OUT &= ~BIT0; //暗 for(i=0;i 16000;i++); P1OUT |= BIT0; //亮 WDTCTL = WDT_ARST_1000; //启动看门狗为1000ms定时 while(1) { for(i=0;i 1000;i++); //主函数任务
[单片机]
基于AVR单片机的DS1302子程序
摘要: 目 的: 建立DS1302操作库 目标系统: 基于AVR单片机 应用软件: ICCAVR /********************************************************************* 目 的: 建立DS1302操作库 目标系统: 基于AVR单片机 应用软件: ICCAVR *********************************************************************/ /*01010101010101010
[单片机]
Google与苹果激战犹酣:模式之争短期难分胜负
    导读:MarketWatch专栏作家辛奈尔(John Shinal)撰文指出,谷歌与苹果之间的竞争,其实是两种不同模式的竞争,而以各自目前的成就看来,他们之间要分出胜负,恐怕还要很长时间。 以下即辛奈尔的评论文章全文: 苹果iOS作业系统与谷歌安卓作业系统之间归根结底其实是一场哲学层面的战争——同时控制移动设备的硬件和软件,或者只控制软件,到底哪一种才是更好的选择? 最终的赢家只能由消费者选择出来,而战火恐怕还要延续许多年。 不过,我们看到的是,谷歌(GOOG)针对iPhone和iPad的地图应用程序推出之后,受到了广泛的好评,而这无疑再度证明了搜索巨头强大的软件开发能力,以及他们对软件的重视。具有讽刺意味的是,他们的成功
[手机便携]
模式51单片机心形流水灯+呼吸灯+蜂鸣器音乐
1.基础硬件DIY设计 电路硬件: STC89C52RC 蜂鸣器/24个LED/4个四角按键 1)整体原理图 2)PCB电路 3)3D_PCB TOP层 2.单片机程序设计 1)呼吸灯 //呼吸灯 while(1) { for(high=1;high cycle;high++) { P2=P1=P0=0XFF; delay2(high); P2=P1=P0=0X00; low=cycle-high; delay2(low); } for(low=1;low cycle;low++) { P2=P1=P0=0XFF;; high=cycle-low;
[单片机]
多<font color='red'>模式</font>51单片机心形流水灯+呼吸灯+蜂鸣器音乐
无人店只是新零售模式的小马甲
新零售不是炒作概念,更不是制造新闻,而是真正面向用户提供更好的产品和服务,更高的效率和更舒适的体验。日前引爆媒体圈的阿里无人店,其本质不是新零售,只是“披着新零售外衣”的马甲。下面就随网络通信小编一起来了解一下相关内容吧。 没有售货员、收银员,甚至不用掏钱包、也不用掏手机,顾客只需自主选购,离开超市时,支付宝内便会自动扣款,完成整个购物过程。随着阿里首家无人超市——"淘咖啡"正式的亮相,被外界誉为一个”智能化、无人化“的新零售时代已经到来。 事实正的如此吗?在智家电看来,所谓的无人店只是马甲,其本质不是要展示无人的特点,只是一个由各种物联网新技术构建的未来模式。而且这一模式并非新零售的本质和代表,更不可能成为未来的趋势,只是一个
[网络通信]
stm32控制DS1302
在ds1302.h文件中: #ifndef _STM32F103_DS1302_H_ #define _STM32F103_DS1302_H_ //*****************DS1302控制命令******************* #define RdSec 0x81 #define RdMin 0x83 #define RdHour 0x85 #define RdDate 0x87 #define RdMonth 0x89 #define RdWeek 0x8b #define RdYear 0x8d #define RdControl 0x8f #define RdTrickleCharge 0x91 #d
[单片机]
谷歌要求:安卓11相机默认不可设置为“美颜”模式
随着计算摄影的兴起,智能手机相机拍摄的照片越来越脱离现实。无论是由于创意的色彩平衡,合成多张高清照片,使用AI塑造细节和清晰度,还是更常见的相机“美颜”增强,这些通常看起来都比真实照片更好。   但这种“虚假”的拍摄在国外社交媒体上受到争议,被指控造成各种病态现象,包括身体畸形增加,皮肤增白以及儿童成人饮食失调。   外媒XDA-Developers爆料,据一份泄露的文件称,谷歌似乎计划反对这一流行但有害的功能。   Google适用于Android 11的Android兼容性定义文档(CDD)最近泄露,建议的更改之一是禁止OEM厂商在相机应用中默认使用面部调整算法。   IT之家了解到,值得注意的是,这并不意味着安卓11相机应用
[手机便携]
开关模式LED驱动器的调光技术
led发出的可见光的亮度概念相当容易理解。 LED调光方法 对开关模式驱动 电路 的LED进行 调光有两种常用方法:脉宽调制调光和模拟调光。两种方法都对经过LED 或LED串的时间平均 电流 进行控制,但在衡量两种调光电路的优缺点时,两者之间的差异也很明显。   图1显示采用 降压 拓扑的一个 LED驱动器。Vin必须始终高于LED 和RSNS上的 电压 之和。电感电流为LED电流。该电流通过监控CS引脚的电压进行调节。当CS开始低于设置的电压时,流经 L1、LED和RSNS的电流脉冲的占空比增加,从而增加LED的平均电流。 模拟调光 LED的模拟调光是对LED电流的每个周期
[电源管理]
开关<font color='red'>模式</font>LED驱动器的调光技术
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved