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.

你可能感兴趣的:(C/C++,单片机编程)