#ifndef __SET_H_
#define __SET_H_
#include "absacc.h"
#include "reg51.h"
/*************** 这里设置程序初始化时显示的时间 ****************/
#define SET_HOUR 12 /* 设置初始化小时 */
#define SET_MINUTE 00 /* 设置初始化分钟 */
#define SET_SECOND 00 /* 设置初始化秒数 */
/************************* 系统地址 ****************************/
#define BASE_PORT 0x8000 /* 选通基地址 */
#define KEY_LINE BASE_PORT+1 /* 键盘行线地址 */
#define KEY_COLUMN BASE_PORT+2 /* 键盘列线地址 */
#define LED_SEG BASE_PORT+4 /* 数码管段选地址 */
#define LED_BIT BASE_PORT+2 /* 数码管位选地址 */
#define LED_ON(x) XBYTE[LED_BIT]=(0x01<<x)&NBSP
#define LED_OFF XBYTE[LED_SEG]=0x00
/*LED 显示空 */
/************** 在设置模式下对秒分时的宏定义 *****************/
#define SECOND 0 /* 对应数码管右边两位 */
#define MINUTE 1 /* 对应数码管中间两位 */
#define HOUR 2 /* 对应数码管左边两位 */
/******************** 定义四种工作模式 ***********************/
#define CLOCK clockstr /* 时钟模式 */
#define ALART alartstr /* 闹钟模式 */
#define DATE datestr /* 日期模式 */
#define TIMER timerstr /* 秒表模式 */
sbit P3_1=P3^1; /*外接蜂鸣器的管脚 */
bdata bit Alart_EN=0; /* 闹铃功能允许位 0:禁止闹铃 1:允许闹铃 */
bdata bit IsBeep=0; /* 响铃标志位 0:未响铃 1:正在响铃 */
//定义函数
void sys_init(void); /* 系统的初始化程序 */
void display(void); /* 动态刷新一次数码管子程序 */
unsigned char getkeycode(void); /* 获取键值子程序 */
void keyprocess(unsigned char); /* 键值处理子程序 */
struct{ /* 时间结构体变量 */
unsigned char s;
unsigned char m;
unsigned char h;
}clock={SET_SECOND,SET_MINUTE,SET_HOUR};
struct{ /* 闹铃时间结构体变量 */
unsigned char m;
unsigned char h;
}alart={SET_MINUTE,SET_HOUR};
struct{ /* 日期结构体变量 */
unsigned int year;
unsigned char month;
unsigned char day;
}date={6,1,1};
struct{ /* 秒表时间结构体变量 */
unsigned char ms;
unsigned char s;
unsigned char m;
}timer={0,0,0};
#include "set.h"
/**************** 以下是所有子函数的声明 *********************/
void clockplus(void); /* 时间加 1S 的子程序 */
void update_clockstr(void); /* 更新时间显示编码 */
void update_alartstr(void); /* 更新闹钟时间的显示编码 */
void update_datestr(void); /*更新日期显示编码 */
void update_timerstr(void); /* 更新秒表时间的显示编码 */
void deley(int); /* 延时子程序 */
void update_dispbuf(unsigned char *); /* 更新显示缓冲区 */
unsigned char getmonthdays(unsigned int,unsigned char);/* 计算某月的天数子程序 */
/* 功能键功能子函数 */
void Akey(void); /* 当前设置位 +1 开关闹钟 开关秒表 */
void Bkey(void); /* 当前设置位 -1 开关闹钟 */
void Ckey(void); /* 设置位选择 秒表清零 */
void Dkey(void); /*切换四种工作模式 */
/********************** 全局变量声明部分 *********************/
unsigned char led[10]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F};/* 从 0~9 的
LED 编码 */
unsigned char ledchar[3]={0x5c,0x54,0x71};/*o n f*/
//unsigned char key[24]={ /* 键值代码数组 对应键位: */
// 0x70,0x71,0x72,0x73,0x74,0x75, /* 7 8 9 A TRACE RESET*/
// 0xb0,0xb1,0xb2,0xb3,0xb4,0xb5, /* 4 5 6 B STEP MON */
// 0xd0,0xd1,0xd2,0xd3,0xd4,0xd5, /* 1 2 3 C HERE LAST */
// 0xe0,0xe1,0xe2,0xe3,0xe4,0xe5}; /* 0 F E D EXEC NEXT */
unsigned char dispbuf[6]; /* 显示缓冲区数组 */
unsigned char clockstr[6]; /* 时间显示的数码管编码数组 */
unsigned char alartstr[6]; /* 闹钟显示的数码管编码数组 */
unsigned char datestr[6]; /* 日期显示的数码管编码数组 */
unsigned char timerstr[6]; /*秒表显示的数码管编码数组 */
unsigned int itime=0,idot; /* 定时器 0 中断计数 */
unsigned char itime1=0; /* 定时器 1 中断计数 */
bdata bit IsSet=0; /*设置模式标志位 0:正常走时 1:设置模式 */
unsigned char SetSelect=0; /* 在设置模式 IsSet=1 时,正在被设置的位 ,对应上面的宏 */
unsigned char *CurrentMode; /* 标 志 当 前 正 设 置 的 功 能 , 如 CurrentMode=CLOCK 或CurrentMode=ALART 等*/
void timerplus(void);
/************************** 函数部分 *************************/
void sys_init(void)
{
TMOD=0x22; /* 定时器 0 和 1 都设置为工作方式 2,基准定时 250×2=500us=0.5ms*/
TH0=6; /* 定时器 0 中断服务用来产生 1 秒时钟定时及闹钟蜂鸣器蜂鸣脉冲 */
TL0=6; /* 定时器 1 中断服务留给秒表使用,产生 1/100 秒定时 */
TH1=6;
TL1=6;
ET0=1;
ET1=1;
EA=1;
TR0=1;
update_clockstr(); /* 初始化时钟显示编码数组 */
update_alartstr(); /* 初始化闹钟显示编码数组 */
update_datestr(); /* 初始化日期显示编码数组 */
update_timerstr(); /* 初始化秒表显示编码数组 */
update_dispbuf(clockstr);/* 初始化显示缓冲数组 */
CurrentMode=CLOCK; /* 默认的显示摸式为时钟 */
P3_1=0; /* 蜂鸣器接线引脚复位 */
}
void timer0(void) interrupt 1 using 1 /* 定时器 0 中断服务器,用来产生 1 秒定时 */
{
itime++;
if(itime==1000)
{
if(IsSet) /* 在设置模式下,对正在设置的位闪烁显示 */
{
dispbuf[SetSelect*2]=0; /* 对正在设置的位所对应的显示缓冲区
元素赋 0,使 LED 灭*/
dispbuf[SetSelect*2+1]=0;
}
if(IsBeep) P3_1=!P3_1; /* 闹钟模式时,产生峰鸣器响脉冲 */
if(CurrentMode==CLOCK)
{
dispbuf[2]=dispbuf[2]&0x7f;
dispbuf[4]=dispbuf[4]&0x7f;
}
}
if(itime==2000) /* 两千次计数为 1S 2000×0.5ms=1s*/
{
itime=0; /*定时 1s 时间到,软计数清零 */
clockplus(); /* 时间结构体变量秒数加 1 */
update_clockstr(); /* 更新时间显示编码数组 */
if(CurrentMode!=TIMER) update_dispbuf(CurrentMode); /* 用时间编码数组更新
显示缓冲区 */
}
}
void timer1(void) interrupt 3 using 2 /* 定时器 1 中断服务器,用来产生 1/100 秒定时 */
{
idot++;
if(++itime1==20) /*20*0.5ms=10ms*/
{
itime1=0;
timerplus();
update_timerstr();
if(CurrentMode==TIMER)
{
update_dispbuf(timerstr);
dispbuf[2]=dispbuf[2]&0x7f; /* 关闭小数点的显示 */
dispbuf[4]=dispbuf[4]&0x7f;
if(idot<1000) /* 闪烁显示小数点 */
{
dispbuf[2]=dispbuf[2]|0x80;
dispbuf[4]=dispbuf[4]|0x80;
}else{
dispbuf[2]=dispbuf[2]&0x7f;
dispbuf[4]=dispbuf[4]&0x7f;
}
}
}
if(idot==2000) idot=0;
}
/* 功能模块子函数 */
void clockplus(void) /* 时间加 1s 判断分,时子函数 */
{
if(++clock.s==60) /* 秒位判断 */
{
clock.s=0;
if(++clock.m==60) /* 分位判断 */
{
clock.m=0;
if(++clock.h==24) /* 时位判断 */
{
clock.h=0;
if(++date.day==(getmonthdays(date.year,date.month)+1))
{
date.day=1;
if(++date.month==13) date.month=1;
}
}
}
}
}
void timerplus() /* 秒表 1/100 秒位加 1,判断秒、分子程序 */
{
if(++timer.ms==100)
{
timer.ms=0;
if(++timer.s==60)
{
timer.s=0;
if(++timer.m==60)
{
timer.m=0;
}
}
}
}
void update_clockstr(void) /* 更新时钟显示代码数组 clockstr*/
{
clockstr[0]=led[clock.s%10]; /* 给元素 0 赋相应数码管显示编码, 编码序号是秒数的个位 */
clockstr[1]=led[(int)(clock.s/10)]; /* 给元素 1 赋相应数码管显示编码,编码序号是秒数的十
位*/
clockstr[2]=led[clock.m%10]; /* 以下类推 */
clockstr[3]=led[(int)(clock.m/10)];
clockstr[4]=led[clock.h%10];
clockstr[5]=led[(int)(clock.h/10)];
}
void update_alartstr(void) /* 更新闹钟显示代码数组 alartstr*/
{ /*右边两位显示 on:闹钟开启 of:闹钟关闭 */
if(Alart_EN) alartstr[0]=ledchar[1];/* 显示字母 n*/
else alartstr[0]=ledchar[2]; /* 显示字母 f*/
alartstr[1]=ledchar[0]; /*显示字母 o*/
alartstr[2]=led[alart.m%10];
alartstr[3]=led[(int)(alart.m/10)];
alartstr[4]=led[alart.h%10];
alartstr[5]=led[(int)(alart.h/10)];
}
void update_datestr(void) /* 更新日期显示代码数组 datestr*/
{
datestr[0]=led[date.day%10];
datestr[1]=led[(int)(date.day/10)];
datestr[2]=led[date.month%10];
datestr[3]=led[(int)(date.month/10)];
datestr[4]=led[date.year%10];
datestr[5]=led[(int)(date.year/10)];
}
void update_timerstr(void) /* 更新秒表显示代码数组 timerstr*/
{
timerstr[0]=led[timer.ms%10];
timerstr[1]=led[(int)(timer.ms/10)];
timerstr[2]=led[timer.s%10];
timerstr[3]=led[(int)(timer.s/10)];
timerstr[4]=led[timer.m%10];
timerstr[5]=led[(int)(timer.m/10)];
}
void display(void) /*刷新显示六位 LED 一次 */
{
unsigned char i;
for(i=0;i<6;i++)
{
LED_ON(i); /*选通相应位 */
XBYTE[LED_SEG]=dispbuf[i]; /* 写显示段码 */
deley(50); /* 延时显示 */
LED_OFF; /* 写 LED 全灭段码 */
}
}
void update_dispbuf(unsigned char *str) /*更新显示缓冲区子函数 ,参数为要用来更新缓冲
区的源字符数组的首地址 */
{
dispbuf[0]=str[0]; /* 将要更新的源字符数组内容 COPY 至 dispbuf 数组,用作显示缓
冲区 */
dispbuf[1]=str[1];
dispbuf[2]=str[2]|0x80; /* 默认把时位和分位后面的小数点显示出来,根据需要再取
舍*/
dispbuf[3]=str[3];
dispbuf[4]=str[4]|0x80;
dispbuf[5]=str[5];
}
void deley(int i) /* 延时子函数 */
{
while(i --);
}
unsigned char getkeycode(void) /* 键盘扫描子程序,返回获得的键码 */
{
unsigned char keycode; /* 键码变量 ,一开始存行码 */
unsigned char scancode=0x20; /* 列扫描码 */
unsigned char icolumn=0; /* 键的列号 */
display(); /* 用刷新数码管显示的时间去抖 */
XBYTE[KEY_COLUMN]=0x00;
keycode=XBYTE[KEY_LINE]&0x0f; /* 从行端口读入四位行码 */
while((scancode&0x3f)!=0) /* 取 scancode 的低六位,只要没变为全 0,则执行循环 */
{
XBYTE[KEY_COLUMN]=(~scancode)&0x3f; /* 给列赋扫描码,第一次为 011111*/
if((XBYTE[KEY_LINE]&0x0f)==keycode) break; /* 检测按键所在的列跳出循环 */
scancode=scancode>>1; /*列扫描码右移一位 */
icolumn++; /* 列号加 1*/
}
keycode=keycode<<4; /* 把行码移到高四位 */
keycode=keycode|icolumn; /* 由行码和列码组成键码 */
//等待按键放开
XBYTE[KEY_COLUMN]=0x00;
while((XBYTE[KEY_LINE]&0x0f)!=0x0f) display();
return keycode;
}
void keyprocess(unsigned char keycode) /* 键值处理函数 */
{
switch (keycode)
{
case 0x73: Akey();
break;
case 0xB3: Bkey();
break;
case 0xD3: Ckey();
break;
case 0xE3: Dkey();
break;
default: break;
}
update_dispbuf(CurrentMode);
}
unsigned char getmonthdays(unsigned int year,unsigned char month)/* 得到某月的天数 */
{
unsigned char days;
switch (month)
{
case 4:
case 6:
case 9:
case 11:days=30;
break;
case 2: if(year%4==0) days=29;
else days=28;
break;
default:days=31;
break;
}
return days;
}
/* 功能键子函数部分 */
void Akey(void) /* 对当前设置位进行加一操作,如果设置秒位,则给秒位清零 */
{
if(CurrentMode==TIMER) /*秒表模式下启闭走时 */
{ TR1=!TR1;
return;
}
if(!IsSet) return; /* 如果不是设置模式退出 */
switch (SetSelect)
{
case SECOND:if(CurrentMode==CLOCK)
{
clock.s=0; /* 如果当前被设置位是秒位,则清零秒位 */
update_clockstr();
}
if(CurrentMode==ALART)
{
Alart_EN=!Alart_EN;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(++date.day==(getmonthdays(date.year,date.month)+1))
date.day=1;
update_datestr();
}
if(CurrentMode==TIMER)
{
TR1=!TR1;
}
break;
case MINUTE:if(CurrentMode==CLOCK)
{
if(++clock.m==60) clock.m=0; /* 如果当前被设置分位, 则分
位加 1*/
update_clockstr();
}
if(CurrentMode==ALART)
{
if(++alart.m==60) alart.m=0;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(++date.month==13) date.month=1;
update_datestr();
}
break;
case HOUR: if(CurrentMode==CLOCK)
{
if(++clock.h==24) clock.h=0; /* 如果当前被设置时位,则时
位加 1*/
update_clockstr();
}
if(CurrentMode==ALART)
{
if(++alart.h==24) alart.h=0;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(++date.year==100) date.year=0;
update_datestr();
}
break;
default: break;
}
update_dispbuf(CurrentMode);
}
void Bkey(void) /* 对当前设置位进行减一操作,如果设置秒分,则给秒位清零 ,类比 Akey()
函数 */
{
if(!IsSet) return;
switch (SetSelect)
{
case SECOND:if(CurrentMode==CLOCK)
{
clock.s=0;
update_clockstr();
}
if(CurrentMode==ALART)
{
Alart_EN=!Alart_EN;
update_alartstr();
}
if(CurrentMode==DATE)
{
if( --date.day==0x00)
date.day=getmonthdays(date.year,date.month);
update_datestr();
}
break;
case MINUTE:if(CurrentMode==CLOCK)
{
if( --clock.m==0xff) clock.m=59;
update_clockstr();
}
if(CurrentMode==ALART)
{
if( --alart.m==0xff) alart.m=59;
update_alartstr();
}
if(CurrentMode==DATE)
{
if( --date.month==0x00) date.month=12;
update_datestr();
}
break;
case HOUR: if(CurrentMode==CLOCK)
{
if( --clock.h==0xff) clock.h=23;
update_clockstr();
}
if(CurrentMode==ALART)
{
if( --alart.h==0xff) alart.h=23;
update_alartstr();
}
if(CurrentMode==DATE)
{
if( --date.year==0xffff) date.year=99;
update_datestr();
}
break;
default: break;
}
update_dispbuf(CurrentMode);
}
void Ckey(void) /* 正常走时模式和设置模式的切换 */
{
if(CurrentMode==TIMER)
{
TR1=0; /* 初始化定时器 1 设置,停止秒表记时 */
TH1=6;
TL1=6;
timer.ms=0; /* 初始化秒表数组 */
timer.s=0;
timer.m=0;
update_timerstr();
}else
{
if(IsSet==0) /* 在非秒表模式下,第一次按下 C 键进入设置模式 ,设置时位 */
{
IsSet=1; /* 置位标志位,进入设置模式 */
SetSelect=HOUR;
return;
} /* 第二次按 C 键设置分位,第三次按键设置秒位,第四次按键完成退出设置 */
if(SetSelect==0) /* 按到第四次,即设置完秒位后,将标志位 IsSet 置 0,完成设置 */
{
IsSet=0; /* 复位标志位,进入正常走时模式 */
return;
}
if(SetSelect>0) SetSelect --; /* 设置位的标志变量 SetSelect=0:时位 1:分位 2:秒位 */
}
}
void Dkey(void) /* 工作状态切换:时钟、闹钟、日期、秒表 */
{
if(CurrentMode==CLOCK) /* 切换至闹钟,同时开关闹钟 */
{ CurrentMode=ALART;
Alart_EN=!Alart_EN;
update_alartstr();
return;
}
if(CurrentMode==ALART) /* 切换至日期 */
{
CurrentMode=DATE;
return;
}
if(CurrentMode==DATE) /* 切换至秒表,同时关闭设置模式 */
{
CurrentMode=TIMER;
IsSet=0;
return;
}
if(CurrentMode==TIMER) /*切换至时钟 */
{
CurrentMode=CLOCK;
return;
}
}
#include "set.h"
void main(void)
{
sys_init();
while(1)
{
XBYTE[KEY_COLUMN,0x00]; /*给键盘列线赋全零扫描码,判断是否有键按下 */
while((XBYTE[KEY_LINE]&0x0f)==0x0f) /* 检测是否有键按下,无则一直进行 LED 的刷
新显示 */
{
if(Alart_EN&&(clock.h==alart.h)&&(clock.m==alart.m)) {IsBeep=1;}
else
{
IsBeep=0;
P3_1=0;
}
display();
}
keyprocess(getkeycode()); /* 有键按下时得到键值,并送入键值处理程序 */
display(); /* 可要可不要 */
}
}