[!--tag.name--]

DS1302时钟芯片简单操作及BCD相关注意事项

标签:DS1302   

《DS1302时钟芯片简单操作及BCD相关注意事项》这篇文章中存在错误,具体如下:

*错误描述:
*联系邮箱:

ds1302是具有时钟功能的芯片,一旦启动,可以自动计时,内部含有年月日时分秒寄存器等。

先说下我这几天遇到的问题,其实归结起来满简单的一个问题,针对ds1302芯片的读写字节操作满简单的,见附表的datasheet,但这里要强调的是往ds1302芯片写数据或者是读数据,在程序执行上一般会分别调用先后调用往ds1302写一个字节或者读一个字节的方法(当然这两个方法得自己写)。不过记得在调用这两个方法的过程中要保持CE(即芯片的置位端)持续为高电平,切不可写完一个字节就将置位端拉低,接着要写下一个字节又把置位端拉高。现在看看下面的代码:

sbit clk = P3^6;//时钟

sbit io = P3^4; //数据

sbit reset = P3^5;// DS1302复位

/写一字节到ds1302

void write_byte(uchar dat)

{

uchar i = 0;

reset = 1; //置位端高电平允许写

for(i=0; i<8; i++)

{

clk = 0;

io = dat&0x01; //从最低位开始写

clk = 1; //时钟信号上升沿接收数据

dat = dat >> 1; //数据右移

}

reset = 0;

}

//从ds1302读一个字节

uchar read_byte()

{

uchar rev = 0x00;

uchar i = 0;

reset = 1; //置位端高电平允许读

for(i=0; i<8; i++)

{

clk = 0;

if(io)

{

rev |= 0x80; //读取的数据放在高位端

}

rev >>= 1; //数据右移

clk = 1; //时钟信号上升沿接收数据

}

return rev;

}

void write_data_ds1302(uchar address, uchar dat)

{

reset=0;

_nop_();

clk=0;

_nop_();

reset = 1;

_nop_(); //启动

write_byte(address); //发送地址

write_byte(dat); //发送数据

clk = 1;

reset = 0; //恢复

}

在主函数里

write_data_ds1302(0x82, 0x36); //在分钟寄存器里写入数据

minute = read_data_ds1302(0x83); //读出分钟寄存器里的数据

write_data(minute);

得到的minute却是0x7e,或者0x00这样明显错误的答案!!

原因在于write_byte(uchar dat)中最后一行代码reset = 0;相当于拉低置位端,数据后续的传输被破坏了。

真正的函数方法只要这样写就可以了

sbit SCK = P3^6;//时钟

sbit SDA = P3^4; //数据

sbit RST = P3^5;// DS1302复位

//写一字节到ds1302

void write_byte(uchar dat)

{

uchar i = 0;

SCK = 0;

RST = 1; //置位端高电平允许写

for(i=0; i<8; i++)

{

SCK = 0;

delay(1);

SDA = dat&0x01; //从最低位开始写

dat >>= 1; //数据右移

SCK = 1; //时钟信号上升沿接收数据

}

}

//从ds1302读一个字节

uchar read_byte()

{

uchar rev = 0x00;

uchar i = 0;

for(i=0; i<8; i++)

{

if(SDA)

{

rev |= 0x80; //读取的数据放在高位端

}

SCK = 0;

rev >>= 1; //数据右移

SCK = 1;

}

return rev;

}

void write_data_ds1302(uchar address, uchar dat)

{

RST = 0;

SCK = 0;

RST = 1;

write_byte(address);//发送地址

write_byte(dat);//发送数据

RST = 0; //恢复

}

uchar read_data_ds1302(uchar address)

{

uchar rev = 0x00;

RST = 0;

SCK = 0;

delay(4);

RST = 1;

write_byte(address);

rev = read_byte();

return rev;

}

这里的write_data_ds1032(uchar address, uchar dat)指的是写入地址后接着写入一个字节的数据,如果还要写第二个字节数据,应当把末行的RST = 0; 去掉,同时调用write_byte(uchar dat)方法。

在这里谈下ds1302 数据的注意事项,比如write_data_ds1302(0x84, 18); //在分钟寄存器里写入数据 这里将18无论以十进制的方法或者十六进制0x12的方法存到寄存器中,在读取出来是0x18,因为存入到寄存器里的八位数据,在寄存器的八位中,寄存器的前四位存放十位,后四位存放个位。(因为分钟寄存器存放的数据最大也就59即0x59).所以minute = read_data_ds1302(0x85); //读出分钟寄存器里的数据 将得到0x18.