当前位置: 代码迷 >> 单片机 >> 单片机 DS1302 痛苦的调试过程,把小弟我的经验教训分享给大家,希望能给后来者一点帮助
  详细解决方案

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

热度:1104   发布时间:2016-04-28 16:35:54.0
单片机 DS1302 痛苦的调试过程,把我的经验教训分享给大家,希望能给后来者一点帮助
首先把我用的程序贴出来,是网上下的,我已经通过硬件测试,绝对没有问题
C/C++ code
#define WRITE_SECOND              0x80#define WRITE_MINUTE        0x82#define WRITE_HOUR                   0x84#define READ_SECOND                0x81#define READ_MINUTE                 0x83#define READ_HOUR                     0x85#define WRITE_PROTECT             0x8E //位寻址寄存器定义sbit ACC_7 = ACC^7;//管脚定义sbit SCLK = P3^5;                   // DS1302时钟信号         7脚sbit DIO= P3^6;                      // DS1302数据信号         6脚sbit CE = P3^7;                        // DS1302片选                 5脚 //地址、数据发送子程序void Write1302 ( unsigned char addr,dat )     {       unsigned char i,temp;       CE=0;                         //CE引脚为低,数据传送中止       SCLK=0;                    //清零时钟总线       CE = 1;                       //CE引脚为高,逻辑控制有效       //发送地址       for ( i=8; i>0; i-- ) //循环8次移位       {                   SCLK = 0;              temp = addr;              DIO = (bit)(temp&0x01);          //每次传输低字节              addr >>= 1;                //右移一位              SCLK = 1;       }       //发送数据       for ( i=8; i>0; i-- )        {                   SCLK = 0;              temp = dat;              DIO = (bit)(temp&0x01);                        dat >>= 1;                                 SCLK = 1;       }       CE = 0;         } //数据读取子程序unsigned char Read1302 ( unsigned char addr ){       unsigned char i,temp,dat1,dat2;       CE=0;                  SCLK=0;                    CE = 1;         //发送地址       for ( i=8; i>0; i-- )                      //循环8次移位       {                   SCLK = 0;              temp = addr;              DIO = (bit)(temp&0x01);          //每次传输低字节              addr >>= 1;                              //右移一位              SCLK = 1;       }       //读取数据       for ( i=8; i>0; i-- )        {              ACC_7=DIO;              SCLK = 0;             ACC>>=1;              SCLK = 1;       }            CE=0;       dat1=ACC;       dat2=dat1/16;                           //数据进制转换       dat1=dat1%16;                         //十六进制转十进制       dat1=dat1+dat2*10;       return (dat1);} //初始化DS1302void Initial(void)    {        Write1302 (WRITE_PROTECT,0X00);         //禁止写保护       Write1302 (WRITE_SECOND,0x56);          //秒位初始化       Write1302 (WRITE_MINUTE,0x34);         //分钟初始化       Write1302 (WRITE_HOUR,0x12);                     //小时初始化       Write1302 (WRITE_PROTECT,0x80);                 //允许写保护}


下面给大家说说我的痛苦调试经历
我使用的单片机的P1.6(CLK),P1.7(IO),P3.1(CE),可是时间读出来总是85,很郁闷,把1302拔掉,读出来还是85,知道是没有读到1302。没有办法,自己研究E文Datasheet,觉得上边下了的程序在读数据的时候似乎有点问题,似乎不是下降沿读数据而是上升沿,就自己该了程序,如下
C/C++ code
 //读取数据       for ( i=8; i>0; i-- )        {              [color=#FF0000]SCLK=0;[/color]              ACC_7=DIO;              SCLK = 0;             ACC>>=1;              [color=#FF0000]//SCLK = 1;[/color]       }     

结果还是一样,读出来是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()把转换的部分代码注释掉,哈哈,读出来果然就正确啦!