11585|30

1

帖子

0

TA的资源

一粒金砂(初级)

楼主
 

单片机 DS1302 痛苦的调试过程,把我的经验教训分享给大家,希望能给后来者一点帮助 [复制链接]

首先把我用的程序贴出来,是网上下的,我已经通过硬件测试,绝对没有问题

  1. #define WRITE_SECOND              0x80
  2. #define WRITE_MINUTE        0x82
  3. #define WRITE_HOUR                   0x84
  4. #define READ_SECOND                0x81
  5. #define READ_MINUTE                 0x83
  6. #define READ_HOUR                     0x85
  7. #define WRITE_PROTECT             0x8E

  8. //位寻址寄存器定义
  9. sbit ACC_7 = ACC^7;
  10. //管脚定义
  11. sbit SCLK = P3^5;                   // DS1302时钟信号         7脚
  12. sbit DIO= P3^6;                      // DS1302数据信号         6脚
  13. sbit CE = P3^7;                        // DS1302片选                 5脚

  14. //地址、数据发送子程序
  15. void Write1302 ( unsigned char addr,dat )     
  16. {
  17.        unsigned char i,temp;
  18.        CE=0;                         //CE引脚为低,数据传送中止
  19.        SCLK=0;                    //清零时钟总线
  20.        CE = 1;                       //CE引脚为高,逻辑控制有效
  21.        //发送地址
  22.        for ( i=8; i>0; i-- ) //循环8次移位
  23.        {     
  24.               SCLK = 0;
  25.               temp = addr;
  26.               DIO = (bit)(temp&0x01);          //每次传输低字节
  27.               addr >>= 1;                //右移一位
  28.               SCLK = 1;
  29.        }
  30.       
  31. //发送数据
  32.        for ( i=8; i>0; i-- )
  33.        {     
  34.               SCLK = 0;
  35.               temp = dat;
  36.               DIO = (bit)(temp&0x01);         
  37.               dat >>= 1;                  
  38.               SCLK = 1;
  39.        }
  40.        CE = 0;         
  41. }

  42. //数据读取子程序
  43. unsigned char Read1302 ( unsigned char addr )
  44. {
  45.        unsigned char i,temp,dat1,dat2;
  46.        CE=0;           
  47.        SCLK=0;            
  48.        CE = 1;  
  49.        //发送地址
  50.        for ( i=8; i>0; i-- )                      //循环8次移位
  51.        {     
  52.               SCLK = 0;
  53.               temp = addr;
  54.               DIO = (bit)(temp&0x01);          //每次传输低字节
  55.               addr >>= 1;                              //右移一位
  56.               SCLK = 1;
  57.        }
  58.        //读取数据
  59.        for ( i=8; i>0; i-- )
  60.        {
  61.               ACC_7=DIO;
  62.               SCLK = 0;
  63.              ACC>>=1;
  64.               SCLK = 1;
  65.        }     
  66.        CE=0;
  67.        dat1=ACC;
  68.        dat2=dat1/16;                           //数据进制转换
  69.        dat1=dat1%16;                         //十六进制转十进制
  70.        dat1=dat1+dat2*10;
  71.        return (dat1);
  72. }

  73. //初始化DS1302
  74. void Initial(void)   
  75. {
  76.        Write1302 (WRITE_PROTECT,0X00);         //禁止写保护
  77.        Write1302 (WRITE_SECOND,0x56);          //秒位初始化
  78.        Write1302 (WRITE_MINUTE,0x34);         //分钟初始化
  79.        Write1302 (WRITE_HOUR,0x12);                     //小时初始化
  80.        Write1302 (WRITE_PROTECT,0x80);                 //允许写保护
  81. }

复制代码


下面给大家说说我的痛苦调试经历
我使用的单片机的P1.6(CLK),P1.7(IO),P3.1(CE),可是时间读出来总是85,很郁闷,把1302拔掉,读出来还是85,知道是没有读到1302。没有办法,自己研究E文Datasheet,觉得上边下了的程序在读数据的时候似乎有点问题,似乎不是下降沿读数据而是上升沿,就自己该了程序,如下

  1. //读取数据
  2.        for ( i=8; i>0; i-- )
  3.        {
  4.               [color=#FF0000]SCLK=0;[/color]
  5.               ACC_7=DIO;
  6.               SCLK = 0;
  7.              ACC>>=1;
  8.               [color=#FF0000]//SCLK = 1;[/color]
  9.        }     

复制代码

结果还是一样,读出来是85.(其实人总是喜欢去怀疑别人,而愿意相信自己是对的,大家注意了我改的是错的,最上面的程序是对的)
这个时候就傻了,不知道哪里有问题,也不知道为什么什么都读不到,第一步“读取”都失败了,下面还有写入,还有存储数据根本就无法谈起,最怕就是这种问题,明明觉得没有问题,却总是得不到想要的结果,现在直接读不到任何结果。
在无奈和苦闷中,我把CE接到了P1.5,(不管是不是这个问题,试试看啦,因为真的想不到其他的了),想不到奇迹竟然出现了,还真的瞎猫逮到了死耗子,居然能读出数据了,能读到初始化后的时,分,秒的数值了。兴奋啊,赶紧趁热打铁,又写了个定时程序,一秒钟把DS1302的“秒”读出来用数码管显示,呵呵,真的像我预料的一样,每一秒钟数码管的数值都会变化,现在更加兴奋了,简单的时钟已经实现了。
可是没过几分钟就发现,秒钟跳动极其不稳定,有时候正常,有时候会乱跳,从20跳到40,从5跳到18...
现在我开始怀疑自己了,怀疑自己把别人的时序改掉引起的,于是又改了回去,这下终于稳定了!

接着又写了修改时间的操作,没有费周折,一下子就搞定了

(这里告一段落,因为到这里基本的读写时间寄存器都已经成功。)

下面开始读写DS1302的31个Bytes的Ram,程序就不贴出来了,还是用最上面的程序,只是寄存器地址用相应的RAM地址就好了。
我的程序显示把数据写进去,然后在把它读出来用数码管显示,按照预想,读出来的应该跟写进去的一样才对,因为是对同一个寄存器进行读写啊,可是让人郁闷的事情又发生了,读出来的值跟写进去的值死活就是不一样,写进去是一个值,读出来始终都是另外一个值,试过N多次都是同样的结果。
直到写入2222(我把前两外跟后两位分别写到两个寄存器),读出来是1616,虽然不是我想要的结果,但是却让我很兴奋,因为我觉得这两者之间肯定存在着某种联系,这是我想到了BCD码转换,(大家都知道DS1302的时钟寄存器读出来是BCD),所以在上面的程序中读1302的函数Read1302()的最后是把BCD转成一般的16进制。大家把22换成16进制看看,是不是正好是16啊!所以我认为读1302的时钟寄存器需要转换,而读RAM不需要,就在Read1302()把转换的部分代码注释掉,哈哈,读出来果然就正确啦!

(这下都OK啦,后来自己写了一个很庞大的程序,用键盘修改时间和RAM数据,然后读出来用数码管显示,一切运行正常。不过事情并没有到这里结束,下面又让人更头痛的问题,下面这个问题的原因和现象在网上没有任何资料,大家继续往下看)

先贴我写的一段程序(初始化用的,读时钟和RAM)

  1. void initial(void)   
  2. { //程序比较乱,大家只要知道:先读时间,只要时间不是00 00和85 85那就读RAM

  3.            //DS1302初始化,开启时钟
  4.            Write1302(WRITE_PROTECT,0X00);                         //取消写保护
  5.            delay(2);
  6.        Write1302(WRITE_SECOND,0x01);                          //秒位初始化,开启时钟
  7.            delay(2);
  8.        Write1302(WRITE_PROTECT,0x80);
  9.            //时钟和存储数据的读取
  10.            hour=Read1302(READ_HOUR,1);
  11.            delay(1);
  12.            minute=Read1302(READ_MINUTE,1);
  13.            if(((hour*100+minute)!=0) && ((hour*100+minute)!=8585)){
  14.                            step=Read1302(SEP_ADD+1,0);
  15.                         if(step==0){
  16.                                 step=2;
  17.                         }
  18.                            QtyValue[0]=Read1302(PLAN_QTY_ADD+1,0)*100 + Read1302(PLAN_QTY_ADD+3,0);
  19.                         delay(1);
  20.                            QtyValue[1]=Read1302(ESTM_QTY_ADD+1,0)*100 + Read1302(ESTM_QTY_ADD+3,0);
  21.                         delay(1);
  22.                     QtyValue[2]=Read1302(ACTL_QTY_ADD+1,0)*100 + Read1302(ACTL_QTY_ADD+3,0);
  23.                         delay(1);
  24.                     paraQty=Read1302(PAH_QTY_ADD+1,0)+2;
  25.                         delay(1);
  26.                     paraStartMode=(bit)Read1302(PAH_MODE_ADD+1,0);
  27.                         delay(1);
  28.                     readPAH2();
  29.                         delay(1);
  30.                     rest=paraStartMode;
  31.                         delay(1);
  32.                     for(paraEditPoint=0;paraEditPoint>3;paraEditPoint++){
  33.                                  resetHour[paraEditPoint]=Read1302(RESET_TIME_ADD+1+paraEditPoint*4,0);
  34.                                 delay(1);
  35.                                  resetMinute[paraEditPoint]=Read1302(RESET_TIME_ADD+1+paraEditPoint*4+2,0);
  36.                                 delay(1);
  37.                            }
  38.            }
  39.                 //else{
  40.                 //        QtyValue[2]=hour*100+minute;
  41.                 //}

  42.            //模式指针初始化
  43.         beep();//初始化完后,蜂鸣器叫一声            
  44. }
复制代码


程序写完了,在自己焊的板子上也调试OK了,下面就做了一块PCB板,焊好元器件之后把单片机烧了程序放到里面,上电,没有任何反应,连蜂鸣器都不叫,数码管也不亮,我在初始化程序最后加了beep()函数,如果不叫,那就说明初始化没有完成。没有办法,只能用万用表量单片机和DS1302的引脚,看看能否有所发现,当我量到1302的 IO脚的时候,蜂鸣器突然叫了,数码管也亮了,只不过读出来的是85和23(其中23是正确的,85是没有读到)。
这说明什么呢?说明我用万用表接触IO的时候,系统才成功初始化,就是被我触发的!我后来又试了很多次,确实是被我触发的,当我一个表笔接地,一个表笔接IO时候,才能触发它初始化完毕(也就是读取时钟和RAM完毕)。我后来直接用导线连接地线和IO,同样能触发系统初始化。
这时候我就纳闷了,为什么在面包板上可以,上了PCB就不行了呢?线路都一样啊,这是在面包板上DS1302紧挨着单片机,在PCB上拉远了一点,隔了一个芯片的距离,难道是线太长了??PCB应该比面包板更稳定才是啊,只有大概4公分的距离,不会又什么影响吧?再说网上从来没有资料说线长对1302读写有影响啊!!但是一个读到85,一个读到23,显示是读到了一个,只是不稳定,所以只能读到一个。
没有办法,继续上网查资料,后来发现有人说IO口一定要加上拉电阻,要不然会读出85,哎呀,如获救命稻草!立刻回来加了4.7K的上拉电阻,满怀希望地上电,却读到8585,也就是两个值都没有读到!怎么会这样?!!不是说上拉电阻能避免读到85吗?效果怎么相反啊?!!郁闷哪?谁来救我?没有人!网上再也找不到其他资料了!
只能去掉上拉电阻,继续读写,发现真的很不稳定,有时候读到8523,有时候读到8585!
这时候这能来排除最后的可能性了,我怀疑是线太长,于是直接重新焊了一个1302在单片机的引脚上,这样不可能再长了把!!

上电!真的好了,一切都正常了,原来真的是线太长引起的!
大家没有见过这种情况吧,我也很纳闷!

写出来给后来人一些参考,让他们少走一些弯路

最新回复

楼主辛苦啦,顶下,我同学正在做这个  详情 回复 发表于 2010-7-1 14:48
点赞 关注

回复
举报

1

帖子

0

TA的资源

一粒金砂(初级)

沙发
 
谢谢啊
 
 

回复

2

帖子

0

TA的资源

一粒金砂(中级)

板凳
 
有人用过SD2068没,小弟读里面的数据出错。
 
 
 

回复

29

帖子

0

TA的资源

一粒金砂(高级)

4
 
最近在用C写这个程序,dsp控制的,头疼。
参考了楼主程序,大赞~
 
 
 

回复

2

帖子

0

TA的资源

一粒金砂(初级)

5
 
谢谢,mark
 
 
 

回复

1

帖子

0

TA的资源

一粒金砂(初级)

6
 
很好很强大~~~~~~~~~~~~~
 
 
 

回复

1

帖子

0

TA的资源

一粒金砂(初级)

7
 
原来我在写1302程序的时候也很痛苦,最后发现原来sda和scl必须要接上拉。哈哈。最后哭笑不得。
 
 
 

回复

11

帖子

0

TA的资源

一粒金砂(初级)

8
 
我最喜欢你说的“其实人总是喜欢去怀疑别人,而愿意相信自己是对的”,这点对大型项目尤为重要,有问题时先要想想自己的东西有没有问题。
 
 
 

回复

1

帖子

0

TA的资源

一粒金砂(初级)

9
 
楼主不错,谢谢分享,1302还是在学校的时候用过,没走过什么弯路,还是很简单的一个东西
 
 
 

回复

1

帖子

0

TA的资源

一粒金砂(初级)

10
 
很好的精神!
 
 
 

回复

4

帖子

0

TA的资源

一粒金砂(初级)

11
 
呵呵,可以做为layout的实例了。
 
 
 

回复

5

帖子

0

TA的资源

一粒金砂(初级)

12
 
谢谢发帖分享,楼主辛苦了
 
 
 

回复

5

帖子

0

TA的资源

一粒金砂(初级)

13
 
1302倒版的太多了。。试下用sd2068.。。i2c口。调试起来应该简单的多。。
 
 
 

回复

3

帖子

0

TA的资源

一粒金砂(初级)

14
 
其实编程序和调试就这样的
弄得多了 有点经验就会少走一点弯路
 
 
 

回复

1

帖子

0

TA的资源

一粒金砂(初级)

15
 
你看一下1302的复位过程就应该首先想到了。
你的芯片电源旁边有小电容吗?退耦之用。
 
 
 

回复

1

帖子

0

TA的资源

一粒金砂(初级)

16
 
hao nan
 
 
 

回复

3

帖子

0

TA的资源

一粒金砂(初级)

17
 
痛苦几回就成长了。
 
 
 

回复

1

帖子

0

TA的资源

一粒金砂(初级)

18
 
很好很强大
 
 
 

回复

6

帖子

0

TA的资源

一粒金砂(初级)

19
 
up
 
 
 

回复

1

帖子

0

TA的资源

一粒金砂(初级)

20
 
学习
 
 
 

回复
您需要登录后才可以回帖 登录 | 注册

随便看看
查找数据手册?

EEWorld Datasheet 技术支持

相关文章 更多>>
关闭
站长推荐上一条 1/6 下一条

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 国产芯 安防电子 汽车电子 手机便携 工业控制 家用电子 医疗电子 测试测量 网络通信 物联网

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
快速回复 返回顶部 返回列表