📄 可调时钟.c
字号:
//******DS1302驱动程序*************
#include"REGX52.H"
#include "stdio.h"
#include <intrins.h>
#define delayNOP(); {_nop_();_nop_();_nop_();_nop_();};
#define AM(X) X
#define PM(X) (X+12) // 转成24小时制
#define DS1302_SECOND 0x80
#define DS1302_MINUTE 0x82
#define DS1302_HOUR 0x84
#define DS1302_WEEK 0x8A
#define DS1302_DAY 0x86
#define DS1302_MONTH 0x88
#define DS1302_YEAR 0x8C
#define DS1302_RAM(X) (0xC0+(X)*2) //用于计算 DS1302_RAM 地址的宏
sbit DS1302_CLK = P0^7; //实时时钟时钟线引脚
sbit DS1302_IO = P0^6; //实时时钟数据线引脚
sbit DS1302_RST = P0^5; //实时时钟复位线引脚
sbit ACC0 = ACC^0;
sbit ACC7 = ACC^7;
sbit rs = P0^0; //LCD数据/命令选择端(H/L)
sbit rw = P0^1; //LCD读/写选择端(H/L)
sbit ep = P0^2;
sbit PRE = P3^2;
sbit SET = P3^1;
unsigned char data DateString[9]; //lcd上排显示缓冲区
unsigned char data TimeString[9]; //lcd下排显示缓冲区
unsigned char Second, Minute,Hour,Week,Day, Month, Year;
unsigned char flag1,skey,vkey;
void pro_key();
// 延时程序
void delay(unsigned char ms)
{ while(ms--)
{ unsigned char i;
for(i = 0; i< 250; i++)
{
_nop_(); //执行一条_nop_()指令为一个机器周期
_nop_();
_nop_();
_nop_();
}
}
}
//测试LCD忙碌状态
bit lcd_busy()
{
bit result;
rs = 0;
rw = 1;
ep = 1;
_nop_();
_nop_();
_nop_();
_nop_();
result =(bit)(P1&0x80); //LCD的D0--D7中,D7=1为忙碌,D7=0为空闲
ep = 0;
return result;
}
//写入指令到LCD
void lcd_wcmd(char cmd)
{
while(lcd_busy()); //当lcd_busy为1时,再次检测LCD忙碌状态,lcd-busy为0时,开始写指令
rs = 0;
rw = 0;
ep = 0;
_nop_();
_nop_();
P1 = cmd;
_nop_();
_nop_();
_nop_();
_nop_();
ep = 1;
_nop_();
_nop_();
_nop_();
_nop_();
ep = 0;
}
//写入数据到LCD
void lcd_wdat(char dat)
{
while(lcd_busy()); //当lcd_busy为1时,再次检测LCD忙碌状态,lcd-busy为0时,开始写数据
rs = 1;
rw = 0;
ep = 0;
P1 = dat;
_nop_();
_nop_();
_nop_();
_nop_();
ep = 1;
_nop_();
_nop_();
_nop_();
_nop_();
ep = 0;
}
//LCD数据指针位置程序
void lcd_pos(char pos)
{
lcd_wcmd(pos|0x80); //数据指针=80+地址码(00H~27H,40H~67H)
}
//显示处理程序
void pro_display()
{ unsigned char i;
lcd_pos(0x00);
for (i=0;i<9;i++)
{lcd_wdat(DateString[i]);}
lcd_pos(0x40);
for (i=0;i<9;i++)
{lcd_wdat(TimeString[i]);}
}
//LCD初始化设定
void lcd_init()
{
lcd_wcmd(0x38); //设置LCD为16X2显示,5X7点阵,八位数据借口
delay(1);
lcd_wcmd(0x0c); //LCD开显示及光标设置(光标不闪烁,不显示"-")
delay(1);
lcd_wcmd(0x06); //LCD显示光标移动设置(光标地址指针加1,整屏显示不移动)
delay(1);
lcd_wcmd(0x01); //清除LCD的显示内容
delay(1);
}
void DS1302InputByte(unsigned char d) //实时时钟写入一字节(内部函数)
{
unsigned char i;
ACC = d;
for(i=8; i>0; i--)
{
DS1302_IO = ACC0; //相当于汇编中的 RRC
DS1302_CLK = 1;
DS1302_CLK = 0;
ACC = ACC >> 1;
}
}
unsigned char DS1302OutputByte(void) //实时时钟读取一字节(内部函数)
{
unsigned char i;
for(i=8; i>0; i--)
{
ACC = ACC >>1; //相当于汇编中的 RC
ACC7 = DS1302_IO;
DS1302_CLK = 1;
DS1302_CLK = 0;
}
return(ACC);
}
void Write1302(unsigned char ucAddr, unsigned char ucDa) //ucAddr: DS1302地址, ucData: 要写的数据
{
DS1302_RST = 0;
DS1302_CLK = 0;
DS1302_RST = 1;
DS1302InputByte(ucAddr); // 地址,命令
DS1302InputByte(ucDa); // 写1Byte数据
DS1302_CLK = 1;
DS1302_RST = 0;
}
unsigned char Read1302(unsigned char ucAddr) //读取DS1302某地址的数据
{
unsigned char ucData;
DS1302_RST = 0;
DS1302_CLK = 0;
DS1302_RST = 1;
DS1302InputByte(ucAddr|0x01); // 地址,命令
ucData = DS1302OutputByte(); // 读1Byte数据
DS1302_CLK = 1;
DS1302_RST = 0;
return(ucData);
}
void DS1302_SetProtect(bit flag) //是否写保护
{
if(flag)
Write1302(0x8E,0x10);
else
Write1302(0x8E,0x00);
}
void DS1302_SetTime(unsigned char Address, unsigned char Value) // 设置时间函数
{
DS1302_SetProtect(0);
Write1302(Address, ((Value/10)<<4 | (Value%10)));
}
void DS1302_GetTime()
{
unsigned char ReadValue;
ReadValue = Read1302(DS1302_SECOND);
Second = ((ReadValue&0x70)>>4)*10 + (ReadValue&0x0F);
ReadValue = Read1302(DS1302_MINUTE);
Minute = ((ReadValue&0x70)>>4)*10 + (ReadValue&0x0F);
ReadValue = Read1302(DS1302_HOUR);
Hour = ((ReadValue&0x70)>>4)*10 + (ReadValue&0x0F);
ReadValue = Read1302(DS1302_DAY);
Day = ((ReadValue&0x70)>>4)*10 + (ReadValue&0x0F);
ReadValue = Read1302(DS1302_WEEK);
Week = ((ReadValue&0x70)>>4)*10 + (ReadValue&0x0F);
ReadValue = Read1302(DS1302_MONTH);
Month = ((ReadValue&0x70)>>4)*10 + (ReadValue&0x0F);
ReadValue = Read1302(DS1302_YEAR);
Year = ((ReadValue&0x70)>>4)*10 + (ReadValue&0x0F);
}
void DateToStr()
{
DateString[0] =Year/10 + 48;
DateString[1] =Year%10 + 48;
DateString[2] = 0x2d;
DateString[3] =Month/10 +48;
DateString[4] =Month%10 +48;
DateString[5] = 0x2d;
DateString[6] =Day/10 +48;
DateString[7] =Day%10 +48;
DateString[8] = 0x20;
TimeString[0] =Hour/10 + 48;
TimeString[1] =Hour%10 + 48;
TimeString[2] = 0x2d;
TimeString[3] =Minute/10 + 48;
TimeString[4] =Minute%10 + 48;
TimeString[5] = 0x2d;
TimeString[6] =Second/10 + 48;
TimeString[7] =Second%10 + 48;
TimeString[8] = 0x20;
}
void Initial_DS1302(void)
{
unsigned char Second=Read1302(DS1302_SECOND);
if(Second&0x80)
DS1302_SetTime(DS1302_SECOND,0);
}
void DS1302_TimeStop(bit flag) // 是否将时钟停止
{
unsigned char Data;
Data=Read1302(DS1302_SECOND);
DS1302_SetProtect(0);
if(flag)
Write1302(DS1302_SECOND, Data|0x80);
else
Write1302(DS1302_SECOND, Data&0x7F);
}
//闰年的计算
bit leap_year()
{
bit leap;
if((Year%4==0&&Year%100!=0)||Year%400==0)//闰年的条件
leap=1;
else
leap=0;
return leap;
}
unsigned char scan_key()
{
skey=0x00; //给变量vkey置初值
skey|=PRE; //读取PRE键的状态
skey=skey<<1; //将PRE键的状态存于skey的B1位
skey|=SET; //读取SET键的状态,并存于skey的B0位
return skey; //返回skey的键值(即PRE,SET的状态)
}
//外部中断INT0中断处理程序
void int0() interrupt 0
{
TR0=0; //禁止Timer0
IE=0; //禁止中断
lcd_wcmd(0x0e); //显示光标"_",整个光标不闪烁
DateToStr();
pro_display();
lcd_pos(0x01); //使光标位于第一个调整项下
flag1=0;
vkey=0x03;
while(flag1^0x07)
{skey = scan_key();
//扫描按键状态
if (skey^vkey) //若skey与vkey相同,跳出循环,相异执行循环体
{ delay(10); //去按键抖动
skey = scan_key(); //转回扫描按键状态
if (skey^vkey) //若skey与vkey相同,跳出循环,相异执行循环体
{ vkey=skey; //将skey的值付给vkey
if (skey==0x01) //PRE键按下
{ flag1++; //调整标志位加1
switch (flag1) //将光标置于相应调整位置
{
case 1: lcd_pos(0x01);DS1302_TimeStop(1); break; //光标置年调整位置
case 2: lcd_pos(0x04);break; //光标置月调整位置
case 3: lcd_pos(0x07);break; //光标置日
case 4: lcd_pos(0x41);break;
//光标置小时设置位置
case 5: lcd_pos(0x44);break; //光标置分钟设置位置
case 6: lcd_pos(0x47);break; //光标置秒时报警设置位置调整位置
case 7: DS1302_TimeStop(0);break; //启动时钟
default:break;
}
}
if (skey==0x02) //SET键按下
pro_key(); //转设置按键处理程序
}
}
}
lcd_wcmd(0x0c); //设置LCD开显示及光标不闪烁,不显示"-"
lcd_wcmd(0x01); //清除LCD的显示内容
IE=0x81; //CPU开中断,INT0,INT1,开中断
TR0=1; //Timer0启动
delay(30);
}
//设置按键处理程序
void pro_key()
{
switch (flag1)
{
case 1:Year++;
DS1302_SetTime(0X8C, Year);
if (Year> 99) Year= 0;
DateToStr();
pro_display();
lcd_pos(0x01);break;
case 2:Month++;
DS1302_SetTime(0X88, Month);
if (Month>12) Month=1;
DateToStr();
pro_display();
lcd_pos(0x04);break;
case 3:Day++;
DS1302_SetTime(DS1302_DAY , Day);
if (Month==1||Month==3||Month==5||Month==7||Month==8||Month==10||Month==12)
if (Day>31) Day=1; //大月31天
if (Month==4||Month==6||Month==9||Month==11)
if (Day>30) Day=1; //小月30天
if (Month==2)
{if(leap_year()) //闰年的条件
{if (Day>29) Day=1;} //闰年2月为29天
else
{if (Day>28) Day=1;}} //平年2月为28天
DateToStr();
pro_display();
lcd_pos(0x07);break;
case 4:Hour++;
DS1302_SetTime(DS1302_HOUR , Hour);
if (Hour>23) Hour=0;
DateToStr();
pro_display();
lcd_pos(0x41);break;
case 5: Minute++;
DS1302_SetTime(DS1302_MINUTE , Minute);
if ( Minute>59) Minute=0;
DateToStr();
pro_display();
lcd_pos(0x44);break;
case 6:Second++;
DS1302_SetTime(DS1302_SECOND ,Second);
if (Second>59) Second=0;
DateToStr();
pro_display();
lcd_pos(0x47);break;
default: break ;
}
}
main()
{ IE=0X81;
IT0=0;
TR0=1;
TH0=0XDC;
TL0=0X00;
TR0=1;
lcd_init();
Initial_DS1302();
while(1)
{DS1302_GetTime();
DateToStr();
pro_display();
}
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -