1 寄存器
DS1302的寄存器是很清楚的
可以看到ds1302的85H和84H分别对应了ds1302的读地址寄存器和写地址寄存器,我们如果这样去操做寄存器,例如 我现在定义这样的一个函数
ds1302_write_byte(unsigned char addr,unsigned char dat)
{
我下面会把函数给大家,大家只要知道这是一个写函数就可以了;
}
现在我们来使用它比如ds1302_write_byte(0x84,0x13);
那么我们加入一个执行元件,也就是说显示这个时间的屏幕,这边我以数码管为例,那么这个小时在数码管上显示的就应该是24小时制的 13时。
那么怎么让他显示12小时制的1点呢?
我们可以看寄存器,寄存器的bit7为0时选择的是24小时制为1时选择时的12小时制,当然12进制肯定得去选择它的上午和下午也就是bit5,当bit5为0时也就是AM上午的意思当bit5为1时为PM下午的意思,比如说我现在的时间是下午13点,也就是PM 1点那该怎么配置寄存器呢?
由图知bit7-bit4表示时间的十位 bit3-bit0表示时间的个位,
那么PM 1点我们就应给寄存器 1010 0001 对应16进制那就应该是0xa1;现在来引用我们的函数,
ds1302_write_byte(0x84,0xa1);
这样我们就把下午一点的内容写进了寄存器 那么我们改怎么去读取他呢?
我们上面说过了高四位代表这时间的十位,底四位代表时间的个位,那我们读取寄存器0x85的时候由于bit7是为1,上午bit5为0下午bit5为1,那么无论怎么样bit5,和bit7始终会影响高四位的值,那么我们该怎么去操作呢,移位,我们现在创建一个函数,
unsigned char ds1302_read_byte(u8 addr)
{
//下面我会给出函数;
}
那么我们来使用它unsigned char ds1302_read_byte(0x85);
我们新建一个变量来保存它,
unsigned char temp;
temp=ds1302_reda_type(0x85);
temp<<=3;//这里的操作就是把影响高三位的值给清楚掉
temp>>=3;//这里的操作是把值给移动回来以便后面去读取它
那么现在我怎么去取值呢?
//这边我插入一个数码管显示的函数,同理最后我会给大家
unsigned char gsmg_code[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};//共阳数码管0~f
smg_display(unsigned dat[],u8 pos)
{
//后面会给出
}
gsmg_code[0]=temp/16;取十位
gsmg_code[1]=temp%16;取个位
现在我们取到了值 ,那我们就去把他展示在数码管上面,
接下来我把我的函数给大家,有错误的地方还请改正,
#include "ds1302.h"
//显示小时-分钟-秒
u8 gREAD_RTC_ADDR[7]={0x81,0X83,0X85,0X87,0X89,0X8B,0X8D,};//秒分时天周月年
u8 gWRITE_RTC_ADDR[7]={0X80,0X82,0X84,0X86,0X88,0X8A,0X8C,};
u8 gWRITE_TIME[7]={0x57,0x59,0xb1,0x17,0x06,0x06,0x23};我这里显示的时间是12小时制,下午一点,
void ds1302_write_byte(u8 addr,u8 dat)
{
u8 i;
CE=0;
delay_ms(1);
SCLK=0;
delay_ms(1);
CE=1;
delay_ms(1);
for(i=0;i<8;i++)
{
IO=addr&0x01;
addr>>=1;
SCLK=1;
_nop_();
SCLK=0;
_nop_();
}
for(i=0;i<8;i++)
{
IO=dat&0x01;
dat>>=1;
SCLK=1;
_nop_();
SCLK=0;
_nop_();
}
CE=0;
_nop_();
SCLK=0;
_nop_();
}
u8 ds1302_read_byte(u8 addr)
{
u8 temp,i,value;
CE=0;
_nop_();
SCLK=0;
_nop_();
CE=1;
_nop_();
for(i=0;i<8;i++)
{
IO=addr&0x01;
addr>>=1;
SCLK=1;
_nop_();
SCLK=0;
_nop_();
}
for(i=0;i<8;i++)
{
temp=IO;
value=(temp<<7)|(value>>1);
SCLK=1;
_nop_();
SCLK=0;
_nop_();
}
CE=0;
_nop_();
SCLK=1;
_nop_();
IO = 0;//这个是因为我的板子这个端口没有上拉电阻,所以要这样去复位它
_nop_();
IO = 1;
_nop_();
return value;
}
void ds1302_initialize(void)
{
u8 i;
ds1302_write_byte(0x8e,0x00);
for(i=0;i<8;i++)
{
ds1302_write_byte(gWRITE_RTC_ADDR[i],gWRITE_TIME[i]);
}
ds1302_write_byte(0x8e,0x80);
}
void ds1302_read_time(void)
{
u8 i;
for(i=0;i<3;i++)
{
gWRITE_TIME[i]=ds1302_read_byte(gREAD_RTC_ADDR[i]);
}
}
数码管函数
#include "smg.h"
u8 gsmg_code[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};
/*******************************************************************************
* 函 数 名 : smg_display
* 函数功能 : 动态数码管显示
* 输 入 : dat:要显示的数据
pos:从左开始第几个位置开始显示,范围1-8
* 输 出 : 无
*******************************************************************************/
void smg_display(u8 dat[],u8 pos)
{
u8 i=0;
u8 pos_temp=pos-1;
for(i=pos_temp;i<8;i++)
{
switch(i)//位选
{
case 0: LSC=1;LSB=1;LSA=1;break;
case 1: LSC=1;LSB=1;LSA=0;break;
case 2: LSC=1;LSB=0;LSA=1;break;
case 3: LSC=1;LSB=0;LSA=0;break;
case 4: LSC=0;LSB=1;LSA=1;break;
case 5: LSC=0;LSB=1;LSA=0;break;
case 6: LSC=0;LSB=0;LSA=1;break;
case 7: LSC=0;LSB=0;LSA=0;break;
}
SMG_A_DP_PORT=dat[i-pos_temp];//传送段选数据
delay_10us(100);//延时一段时间,等待显示稳定
SMG_A_DP_PORT=0x00;//消音
}
}
smg以及ds1302的头文件
#ifndef _smg_H
#define _smg_H
#include "public.h"
#define SMG_A_DP_PORT P0 //使用宏定义数码管段码口
//定义数码管位选信号控制脚
sbit LSA=P2^2;
sbit LSB=P2^3;
sbit LSC=P2^4;
extern u8 gsmg_code[17];
void smg_display(u8 dat[],u8 pos);
#endif
#ifndef _DS1302_H
#define _DS1302_H
#include "public.h"
#include "intrins.h"
sbit SCLK=P3^6;
sbit IO=P3^4;
sbit CE=P3^5;
void ds1302_read_time(void);
void ds1302_initialize(void);
extern u8 gWRITE_TIME[7];
#endif
接下来就是我的主函数,
#include "smg.h"
#include "public.h"
#include "ds1302.h"
void main()
{
u8 time_buf[8];
ds1302_initialize();
while(1)
{
ds1302_read_time();
gWRITE_TIME[2]<<=3;
gWRITE_TIME[2]>>=3;
time_buf[0]=gsmg_code[gWRITE_TIME[2]/16];//小时十位
time_buf[1]=gsmg_code[gWRITE_TIME[2]%16];//小时个位
time_buf[2]=0x40;
time_buf[3]=gsmg_code[gWRITE_TIME[1]/16];//分钟十位
time_buf[4]=gsmg_code[gWRITE_TIME[1]%16];//分钟个位
time_buf[5]=0x40;
time_buf[6]=gsmg_code[gWRITE_TIME[0]/16];//秒十位
time_buf[7]=gsmg_code[gWRITE_TIME[0]%16];//秒个位
smg_display(time_buf,1);//数码管显示
}
}