学习日志-C51-DS1302可调时钟
现象:将时间显示在LCD1602上,并可通过按键进行时间调节
1.DS1302
DS1302:
写入与读取:
DS1302通过控制CE、SCLK、IO进行数据的写入与读取,由低位>>高位进行输入。
CE置1才可使能,进行数据读写,读写结束后CE复位为0。
写入数据:SCLK上升沿即可获得读写地址,8个周期后获得完整命令,在下一个上升沿即可写入数据。
读取数据:SCLK上升沿即可获得读写地址,8个周期后获得完整命令,紧接着的下降沿可读取数据。
因此写入数据经过了16个周期,而读取数据仅经过15个周期。
命令字:在进行读写前都要先获得命令字,确认读取命令以及地址,写入最后一位为0,读取最后一位为1。
在DS1302的时钟寄存器中,内容是BCD码格式,因此在写入时要将十进制转换为BCD码,读取时将BCD码转化为十进制。
十进制转BCD码:BCD=十进制/1016+十进制%10
BCD码转十进制:十进制=BCD/1610+BCD%16
另外在进行写入操作时,注意解除写保护,写入完毕后在恢复写保护。
2.代码实现
(1)DS1302
#include <REGX52.H>
sbit DS1302_SCLK=P3^6;//重定义
sbit DS1302_CE=P3^5;
sbit DS1302_IO=P3^4;
//寄存器写入地址
#define DS1302_SECOND 0x80 //秒
#define DS1302_MINUTE 0x82 //分
#define DS1302_HOUR 0x84//时
#define DS1302_DATE 0x86//日
#define DS1302_MONTH 0x88//月
#define DS1302_DAY 0x8A//星期
#define DS1302_YEAR 0x8C//年
#define DS1302_WP 0x8E//写保护
char DS1302_Time[]={24,6,16,15,01,55,6};
//设置时间,定义一组有符号的数据(有符号便于后面进行越界判断),年、月、日、小时、分、秒、星期
void DS1302_Init()//初始化
{
DS1302_CE=0;
DS1302_SCLK=0;
}
void DS1302_WriteByte(unsigned char Command,Data)//写入一个字节
{
unsigned char i;
DS1302_CE=1;//使能
for(i=0;i<8;i++)//重复8次,写入命名字
{
DS1302_IO=Command&(0x01<<i);//由低位开始写入
DS1302_SCLK=1;//置1,得到上升沿,数据写入
DS1302_SCLK=0;//复位
}
for(i=0;i<8;i++)//在进行写入时,完成命令写入后下一个周期上升沿即可开始写入数据,重复8次,完成一个字节数据写入
{
DS1302_IO=Data&(0x01<<i);
DS1302_SCLK=1;//上升沿写入
DS1302_SCLK=0;//复位
}
DS1302_CE=0;//使能复位
}
unsigned char DS1302_ReadByte(unsigned char Command)//读取一个字节
{
unsigned char i,Data=0x00;//局部变量定义
Command|=0x01;//command与0x01相或,则其他位不变,最后一位置1,变为读取状态
DS1302_CE=1;//使能
for(i=0;i<8;i++)//重复8次,确定命令字
{
DS1302_IO=Command&(0x01<<i);
DS1302_SCLK=0;//复位
DS1302_SCLK=1;//上升沿确定命令字
}
for(i=0;i<8;i++)
{
DS1302_SCLK=1;//上面8个循环结束为上升沿,由于结束后的第一个下降沿就会读取,第二个循环则以高电平开始。
DS1302_SCLK=0;//下降沿
if(DS1302_IO)
{Data|=(0x01<<i);}//将IO数据由低位开始赋值给Data
}
DS1302_CE=0;
DS1302_IO=0;//IO口置0
return Data;//返回Data
}
void DS1302_SetTime()//写入时间,将十进制转化为BCD码
{ DS1302_WriteByte(DS1302_WP,0x00); //解除写保护
DS1302_WriteByte(DS1302_YEAR,DS1302_Time[0]/10*16+DS1302_Time[0]%10);//写入年份
DS1302_WriteByte(DS1302_MONTH,DS1302_Time[1]/10*16+DS1302_Time[1]%10);
DS1302_WriteByte(DS1302_DATE,DS1302_Time[2]/10*16+DS1302_Time[2]%10);
DS1302_WriteByte(DS1302_HOUR,DS1302_Time[3]/10*16+DS1302_Time[3]%10);
DS1302_WriteByte(DS1302_MINUTE,DS1302_Time[4]/10*16+DS1302_Time[4]%10);
DS1302_WriteByte(DS1302_SECOND,DS1302_Time[5]/10*16+DS1302_Time[5]%10);
DS1302_WriteByte(DS1302_DAY,DS1302_Time[6]/10*16+DS1302_Time[6]%10);
DS1302_WriteByte(DS1302_WP,0x80);//恢复写保护 1000 0000
}
void DS1302_ReadTime()//读取时间,BCD转为十进制
{
unsigned char Num;
Num=DS1302_ReadByte(DS1302_YEAR);
DS1302_Time[0]=Num/16*10+Num%16;//读取到年份
Num=DS1302_ReadByte(DS1302_MONTH);
DS1302_Time[1]=Num/16*10+Num%16;
Num=DS1302_ReadByte(DS1302_DATE);
DS1302_Time[2]=Num/16*10+Num%16;
Num=DS1302_ReadByte(DS1302_HOUR);
DS1302_Time[3]=Num/16*10+Num%16;
Num=DS1302_ReadByte(DS1302_MINUTE);
DS1302_Time[4]=Num/16*10+Num%16;
Num=DS1302_ReadByte(DS1302_SECOND);
DS1302_Time[5]=Num/16*10+Num%16;
Num=DS1302_ReadByte(DS1302_DAY);
DS1302_Time[6]=Num/16*10+Num%16;
}
(2)显示时间
#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"
void TimeShow()//时间显示函数
{
DS1302_ReadTime();//读取时间
LCD_ShowNum(1,1,DS1302_Time[0],2);//显示年份
LCD_ShowNum(1,4,DS1302_Time[1],2);
LCD_ShowNum(1,7,DS1302_Time[2],2);
LCD_ShowNum(2,1,DS1302_Time[3],2);
LCD_ShowNum(2,4,DS1302_Time[4],2);
LCD_ShowNum(2,7,DS1302_Time[5],2);
}
(3)按键函数
#include <REGX52.H>
#include "Delay.h"
unsigned char Key()//获取按键键码函数
{
unsigned char KeyNumber=0;
if(P3_1==0){Delay(20);while(P3_1==0);Delay(20);KeyNumber=1;}
if(P3_0==0){Delay(20);while(P3_0==0);Delay(20);KeyNumber=2;}
if(P3_2==0){Delay(20);while(P3_2==0);Delay(20);KeyNumber=3;}
if(P3_3==0){Delay(20);while(P3_3==0);Delay(20);KeyNumber=4;}
return KeyNumber;
}
(4)时间修改
#include <REGX52.H>
#include "key.h"
#include "LCD1602.h"
#include "DS1302.h"
unsigned char KeyNum,MODE,TimeSelect,TimeSetFlashFlag;
void TimeSet()//时间修改函数
{
if(KeyNum==2)//按键2为修改位的选择
{
TimeSelect++;//时间修改选择位加1
TimeSelect%=6;//对6取余,即选择位仅为0~5
}
if(KeyNum==3)//按键3为修改数据增加
{
DS1302_Time[TimeSelect]++;//按下3,数值增加1
if(DS1302_Time[0]>99){DS1302_Time[0]=0;}//年上限越界判断
if(DS1302_Time[1]>12){DS1302_Time[1]=1;}//月上限越界判断,超过12自动置1
if(DS1302_Time[1]==1 || DS1302_Time[1]==3|| DS1302_Time[1]==5|| DS1302_Time[1]==7|| DS1302_Time[1]==8
|| DS1302_Time[1]==10|| DS1302_Time[1]==12)//日上限越界判断
{
if(DS1302_Time[2]>31){DS1302_Time[2]=1;}//1、3、5、7、8、10、12月均为31天,超过31天自动置1
}
else if(DS1302_Time[1]==4||DS1302_Time[1]==6||DS1302_Time[1]==9||DS1302_Time[1]==11)
{
if(DS1302_Time[2]>30){DS1302_Time[2]=1;}
}
else if(DS1302_Time[1]==2)//2月
{
if(DS1302_Time[0]%4==0)//闰年
{
if(DS1302_Time[2]>29)
{DS1302_Time[2]=1;}
}
if(DS1302_Time[0]%4)//平年
{
if(DS1302_Time[2]>28)
{DS1302_Time[2]=1;}
}
}
if(DS1302_Time[3]>23){DS1302_Time[3]=0;}//小时上限越界判断
if(DS1302_Time[4]>59){DS1302_Time[4]=0;}//分钟上限越界判断
if(DS1302_Time[5]>59){DS1302_Time[5]=0;}//秒上限越界判断
}
if(KeyNum==4)//按键4为修改数据减小
{
DS1302_Time[TimeSelect]--;//按下4,数值减少1
if(DS1302_Time[0]<0){DS1302_Time[0]=99;}//年下限越界判断
if(DS1302_Time[1]<1){DS1302_Time[1]=12;}//月下限越界判断
if(DS1302_Time[1]==1 || DS1302_Time[1]==3|| DS1302_Time[1]==5|| DS1302_Time[1]==7|| DS1302_Time[1]==8
|| DS1302_Time[1]==10|| DS1302_Time[1]==12)//日下限与上限越界判断
{
if(DS1302_Time[2]<1){DS1302_Time[2]=31;}
if(DS1302_Time[2]>31){DS1302_Time[2]=1;}
}
else if(DS1302_Time[1]==4||DS1302_Time[1]==6||DS1302_Time[1]==9||DS1302_Time[1]==11)
{
if(DS1302_Time[2]<1){DS1302_Time[2]=30;}
if(DS1302_Time[2]>30){DS1302_Time[2]=1;}
}
else if(DS1302_Time[1]==2)
{
if(DS1302_Time[0]%4==0)
{
if(DS1302_Time[2]<1){DS1302_Time[2]=29;}
if(DS1302_Time[2]>29){DS1302_Time[2]=1;}
}
if(DS1302_Time[0]%4)
{
if(DS1302_Time[2]<1){DS1302_Time[2]=28;}
if(DS1302_Time[2]>28){DS1302_Time[2]=1;}
}
}
if(DS1302_Time[3]<0){DS1302_Time[3]=23;}
if(DS1302_Time[4]<0){DS1302_Time[4]=59;}
if(DS1302_Time[5]<0){DS1302_Time[5]=59;}
}
//更新时间显示
if(TimeSelect==0 && TimeSetFlashFlag==1)//选中0位且闪烁
{LCD_ShowString(1,1,"__");}
else{LCD_ShowNum(1,1,DS1302_Time[0],2);}
if(TimeSelect==1 && TimeSetFlashFlag==1)
{LCD_ShowString(1,4,"__");}
else{LCD_ShowNum(1,4,DS1302_Time[1],2);}
if(TimeSelect==2 && TimeSetFlashFlag==1)
{LCD_ShowString(1,7,"__");}
else{LCD_ShowNum(1,7,DS1302_Time[2],2);}
if(TimeSelect==3 && TimeSetFlashFlag==1)
{LCD_ShowString(2,1,"__");}
else{LCD_ShowNum(2,1,DS1302_Time[3],2);}
if(TimeSelect==4 && TimeSetFlashFlag==1)
{LCD_ShowString(2,4,"__");}
else{LCD_ShowNum(2,4,DS1302_Time[4],2);}
if(TimeSelect==5 && TimeSetFlashFlag==1)
{LCD_ShowString(2,7,"__");}
else{LCD_ShowNum(2,7,DS1302_Time[5],2);}
LCD_ShowNum(1,14,TimeSelect,2);//显示修改选择位
LCD_ShowNum(2,14,TimeSetFlashFlag,2);
}
(5)主程序
#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"
#include "key.h"
#include "t0.h"
#include "TimeShow.h"
#include "timeset.h"
void main()
{//初始化
LCD_Init();
DS1302_Init();
Timer0Init();
LCD_ShowString(1,1," - -");
LCD_ShowString(2,1," : :");
DS1302_SetTime();//设置时间
while(1)
{
KeyNum=Key();//读取按键键码
if(KeyNum==1)//按键1更改运行模式
{
if(MODE==0){MODE=1;}
else if(MODE==1){MODE=0;DS1302_SetTime();}
}
switch(MODE)
{
case 0:TimeShow();break;//模式0,显示时间
case 1:TimeSet();break;//模式1,更改时间
}
}
}
void Timer0_Routine() interrupt 1 //中断程序
{
static unsigned int T0Count;
TL0 = 0x66;
TH0 = 0xFC; //初值
T0Count++;
if(T0Count>=500)//每500ms循环一次,标志位取反,实现选择位的闪烁
{
T0Count=0;
TimeSetFlashFlag=!TimeSetFlashFlag;//闪烁标志位取反
}
}