POP_UP_MESSAGE_CONTENT
3. Q:为什么我用仿真器单步调试检测不到SD2068返回ACK呢?
A:因为SD2068的I2C总线有0.5s超时自动复位的功能。
4. Q:为什么从RTC读出来时间有跳秒的现象。如9秒直接跳到16秒,而且会大于60?
A:RTC采用BCD码计时,BCD码的用0x10表示十进制的10秒,用十进制来看就是会看到从9跳到16(0x10=16)。
5. Q:为什么有时候我读到的时间是乱码?
A:1. SD2068已完全掉电,再上电读到的时间是随机数(可能是乱码);
2. SD2068的VBAT脚悬空也可能会读到乱码;
3. 总速度太快或通信程序的时序中有不符合规范的地方(见附录);
4. 控制SCL、SDA的IO口被复用;
5. 芯片的SDA端口没有加上拉电阻。
6. Q:为什么小时会跑到25点、26点……?
A:没有正确设置12/24小时制。在小时寄存器(02H)的最高位设置12/24小时制。
7. Q:为什么小时显示90点、91点……?
A:读出小时数据后没有把最高位(12/24小时控制位)屏蔽掉。
8. Q:时钟芯片产生中断输出的时候的INT脚是高电平还是低电平?
A:INT端口是一个N沟道开路输出口,使用时通常接一个上拉电阻。没有中断输出的时候是高电平,产生中断后变为低电平。
9. Q:时钟芯片的INT脚在不用的时候可以悬空吗?
A:可以的。
10. Q:SD2068不起振是什么原因?
A:1. 晶振两脚之间有走线、受污、凝结等原因存在泄漏。请洗净干燥后再试;
2. 晶振电路的走线过长,在PCB布线时晶振电路的走线应尽量短且尽可能靠近IC;
3. 负载电容或匹配电容与晶振不匹配或者电容质量有问题(请参考11.Q);
4. PCB板受潮,导致阻抗失配而不能起振;
5. 晶体质量有问题。
11. Q:SD2068应该使用多大的匹配电容?
A:负载电容为6pf的晶体通常使用7pf的匹配电容;负载电容为12.5pf的晶体通常使用15pf的匹配电容。
12. Q:SD2068的功耗很大,后备电池很快就没电了?
A:正常情况下SD2068的功耗约为1uA。
1. 上拉电阻接到了电池端。可参考SD2068手册中介绍应用参考电路;
2. 可能芯片已损坏,换一个芯片后看是否正常。
13. Q:为什么2月30日也可以设置进去,芯片不能容错吗?
A:SD2068芯片没有容错功能,请务必保证写入的时间。设定不存在的时间或日期数据将导致计数器不能正常计数。
14. Q:我给SD2068设定好年、月、日以后它可以自动调整星期吗?
A:日历对应的星期系统不能自动调整,可以通过一定的算法来实现,下面介绍一种常用的公式:
A:最常见的公式:
W = [Y-1] + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + D
Y 是年份数,D 是这一天在这一年中的累积天数,也就是这一天在这一年中是第几天。
B:最好用的是蔡勒公式:
W = [C/4] - 2C + y + [y/4] + [13 * (M+1) / 5] + d - 1
C 是世纪数减一,y 是年份后两位,M 是月份,d 是日数。1 月和 2 月要按上一年的 13 月和14 月 来算,这时 C 和 y 均按上一年取值。
两个公式中的[...]均指只取计算结果的整数部分。算出来的 W 除以 7, 余数是几就是星期几。如果 余数是 0,则为星期日。
15. Q:怎么从软件上判断SD2058掉电了?
A:读0FH寄存器,判断该寄存器的最低位(RTCF)是否为1,若为1表明SD2068掉电。
16. Q:为什么可以读到SD2068的时间,而不能写时间?
A:SD2068内置有三个保护软锁(WRTC1~WRTC3),需要把这三个锁位置1后才可以对寄存器进行写操作。置1解锁的顺序为:WRTC1->WRTC2、WRTC3;当写寄存器操作完成后,把这三个锁位置0对寄存器数据进行保护。置0上锁的顺序为WRTC2、WRTC3-> WRTC1。具体可以参考相关官网的Demo例程。
附录:程序问题举例
1.在切换端口方向的时候引入了STOP信号。
这个函数开始便设置IO口的输入输出,本没有错。SCL在此操作之前是低电平,但是一些单片机把SCL端口置为输出的时候会使SCL变成高电平,那么问题来了,把SDA置为输入的时候SDA也会由低电平变成高电平,这就形成了一个STOP信号了,总线立刻被释放掉了,之前发送的数据也清零了,当然就等不到ACK回应了。
SCL线只有一个方向,一直都是输出,只要在START函数里设置一次就可以了。修改的方法是把 DDRA|=0x08 这条语句屏蔽掉就好了。
2.发送STOP信号的时候引入START也可能会是I2C从器件工作不正常。
上面发送STOP信号的这个函数,看上去没什么错误,但是如果在调用这个函数之前SCL线的状态是高电平,那在这个函数开始调用的时候便会产生一个START信号。比较保险的做法是先拉低SCL,再拉低SDA,把SDA1_CLR和SCL1_CLR调换一下位置就可以了。
3. 对I2C协议总线协议不熟悉导致读指定地址寄存器失败。
上面的程序看起来好像也没有问题,但实际上是没有了解I2C的传输协议,想当然的认为发送了读命令以后再发送一个地址就可以读到指定寄存器的数据了。正确的时序图如下:
寄存器的地址不是用读命令发送出去的,而是用写命令发送出去的。正确的写法如下:
4. 启动信号保持时间、数据建立或数据保持的时间太短等,可能会导致总线不能响应。
上面这个程序的问题就在于SDA数据建立的时间太短。对一些处理速度快的MCU,数据是发送出去了,RTC是响应不到的。正确的做法是在p_Clock_CLK=1; 的前面加一个延时:
5. 器件地址错误导致读写寄存器失败。
上面的程序很显然是对器件地址理解上的错误和I2C总线操作流程不熟悉,没有搞清楚器件地址的读写操作。