基于PCF8563和stc12C5A60S2制作的万年历

   这个星期和同学一起完成了这个课程设计,对单片机也有了更加近一步的了解。这次课程设计做的是基于芯片PCF8563和单片机stc12C5A60S2的万年历,在制作过程中,遇到了好几个问题,有些是硬件电路搭建时的疏忽导致的电路错误,有些是程序编写时的错误,经过反复的检查和调试后,最终完成了这次课程设计。做完这次课设,最大的感受就是搭建实验电路时必须非常严谨,检查每一根线路的正确性,在调试的时候,一定要耐心,不断进行检查。

/*************SMG1.H*************/
#define uint unsigned int
#define uchar unsigned char

uint c;

unsigned char rd[7];
sbit P20=P2^0;
sbit P21=P2^1;
sbit P22=P2^2;
sbit P23=P2^3;
sbit P24=P2^4;
sbit P25=P2^5;
sbit P26=P2^6;
sbit P27=P2^7;
sbit P07=P0^7;

uchar code num[10] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};

void delay_time1(unsigned short t)
{
unsigned short i,n;
for(n=0;n<t;n++)
  for(i=0;i<500;i++)
         {;}
}

void display_time()
{
	rd[2]&=0x3f;
	P2=0xFF;				//关闭所有位选
	P20=0;					//开某一位
	c=rd[2]/16;				//赋值
	P0=num[c];				//送入P0
	delay_time1(1);			//延时
	
	P2=0xFF;
	P21=0;
	c=rd[2]%16;
  	P0=num[c];
	P07=1;
	delay_time1(1);
	
	P2=0xFF;
	P22=0;
	c=rd[1]/16;
	P0=num[c];
	delay_time1(1);
	
	P2=0xFF;
	P23=0;
	c=rd[1]%16;
	P0=num[c];
	P07=1;
	delay_time1(1);
	
	P2=0xFF;
	P24=0;
	c=rd[0]/16;
	P0=num[c];
	delay_time1(1);
	
	P2=0xFF;
	P25=0;
	c=rd[0]%16;
	P0=num[c];
	delay_time1(1);
}

void display_date()
{
	rd[6]&=0x1f;
	P2=0xFF;
	P20=0;
	c=rd[6]/16;
	P0=num[c];
	delay_time1(1);
	
	P2=0xFF;
	P21=0;
	c=rd[6]%16;
  	P0=num[c];
	P07=1;
	delay_time1(1);
	
	rd[5]&=0x1f;
	P2=0xFF;
	P22=0;
	c=rd[5]/16;
	P0=num[c];
	delay_time1(1);
	
	P2=0xFF;
	P23=0;
	c=rd[5]%16;
	P0=num[c];
	P07=1;
	delay_time1(1);
	
	rd[3]&=0x3f;
	P2=0xFF;
	P24=0;
	c=rd[3]/16;
	P0=num[c];
	delay_time1(1);
	
	P2=0xFF;
	P25=0;
	c=rd[3]%16;
	P0=num[c];
	delay_time1(1);
}
//#endif

/**********VIIC_C51.H*****************************/
#ifdef uchar
#define READYDEF1
#else
#define uchar unsigned char
#endif
extern bit ISendByte(uchar sla,uchar c);
extern bit ISendStr(uchar sla,uchar suba,uchar *s,uchar no);
extern bit IRcvByte(uchar sla,uchar *c);
extern bit IRcvStr(uchar sla,uchar suba,uchar *s,uchar no);
#ifndef READYDEF
#undef uchar
#endif

#include <reg52.h>
#include <intrins.h>
#define uchar unsigned char
#define uint unsigned  int
#define _Nop()  _nop_()
sbit SDA=P1^7;
sbit SCL=P1^6;
bit ack;


void delay12()
{
   uchar i;
   i=12;
   while(--i);
   
}
void Start_I2c()
{
SDA=1;
_Nop();
SCL=1;
delay12();
SDA=0;
delay12();
SCL=0;
_Nop();
_Nop();
}

void Stop_I2c()
{
SDA=0;
_Nop();
SCL=1;
delay12();
SDA=1;
delay12();
}


void SendByte(uchar c)
{
uchar BitCnt;
for(BitCnt=0;BitCnt<8;BitCnt++)
{
	if((c<<BitCnt)&0x80)SDA=1;
	else SDA=0;
	 _Nop();
	 SCL=1;
delay12();
	 SCL=0;
 }
 _Nop();
 _Nop();
 SDA=1;
delay12();
 SCL=1;
delay12();
 if(SDA==1)ack=0;
 else ack=1;
 SCL=0;
delay12();
 }

uchar RcvByte()
{
uchar retc;
uchar BitCnt;
retc=0;
SDA=1;
for(BitCnt=0;BitCnt<8;BitCnt++)

 {
 _Nop();
 SCL=0;
delay12();
 SCL=1;
delay12();
 retc=retc<<1;
	if(SDA==1)retc=retc+1;
 _Nop();
 _Nop();

}

SCL=0;
_Nop();
_Nop();
return(retc);
}

void Ack_I2c(bit a)
{
if(a==0)SDA=0;
else SDA=1;
_Nop();
_Nop();
_Nop();
SCL=1;
delay12();
SCL=0;
delay12();


}


//bit ISendByte(uchar sla,uchar c)
//{
//Start_I2c();
//SendByte(sla);
//  if(ack==0)return(0);
//  SendByte(c);
//  if(ack==0)return(0);
//  Stop_I2c();
//  return(1);
//}



bit ISendStr(uchar sla,uchar suba,uchar *s,uchar no)
{
 uchar i;
 Start_I2c();
 SendByte(sla);
 if(ack==0)return(0);
  SendByte(suba);
 if(ack==0)return(0);

 for(i=0;i<no;i++)
 {
 SendByte(*s);

  if(ack==0)return(0);
  s++;
 
 }
Stop_I2c();
return(1);
}


//bit IRcvByte(uchar sla,uchar *c)
//{
//Start_I2c();
// SendByte(sla+1);
// if(ack==0)return(0);
// *c=RcvByte();
// Ack_I2c(1);
// Stop_I2c();
// return(1);
// }


bit IRcvStr(uchar sla,uchar suba,uchar *s,uchar no)
{
uchar i;
Start_I2c();
 SendByte(sla);
 if(ack==0)return(0);
  SendByte(suba);
 if(ack==0)return(0);
 Start_I2c();
  SendByte(sla+1);
 if(ack==0)return(0);
 for(i=0;i<no-1;i++)
 {
 *s=RcvByte();
 Ack_I2c(0);
 s++;

 }
*s=RcvByte();
Ack_I2c(1);
Stop_I2c();
return(1);

}


/**********wannian.c*****************************/
#include "stc12.h"
#include "VIIC_C51.h"
#include "smg1.h"

#define PCF8563 0xA2
#define WRADDR 0x00
#define RDADDR 0x02
#define CON_STATUS1 0x01
#define MINALARMADDR 0x09
#define HOUALARMADDR 0x0A

sbit BUZZ=P1^1;

int flag1,flag2;
unsigned char rd[7];
unsigned char time=100;
unsigned char rcon_sta;
unsigned char wcon_sta;
unsigned char alarmflag=0;

unsigned char DelayNS(unsigned char no)
{
    unsigned char i,j;
    for(;no>0;no--)
         for(i=0;i<100;i++)
              for(j=0;j<100;j++);
     return 0;
}


void delay_time(unsigned short t)
{
	unsigned short i,n;
	for(n=0;n<t;n++)
	for(i=0;i<10000;i++)
	  {;}
}

void initmcu()
{
	EA=1;
	IT0=1;
	EX0=1;
	ET0=1;
	TMOD=0x01;
	TH0=(65536-50000)/256;
	TL0=(65536-50000)%256;
}

void tt0() interrupt 1 using 1
{
	TH0=(65536-50000)/256;
	TL0=(65536-50000)%256;
	time--;
	if(time==0){BUZZ=1;TR0=0;time=100;}
 }

void intt0() interrupt 0 using 0
{
	BUZZ=0;
	TR0=1;
	alarmflag=1;
}
void main()
{ 
    
	unsigned char code td[9]={0x00,0x12,0x40,0x30,0x12,0x01,0x04,0x12,0x16};//2016.11.30.11.59.40
	unsigned char minalarm=0x31;//定时12.00响
	unsigned char houralarm=0x12;
	delay_time(10);
	initmcu();
	ISendStr(PCF8563,WRADDR,td,0x5);
	DelayNS(1);
	ISendStr(PCF8563,WRADDR+5,&td[5],0x4);
	DelayNS(1);
	ISendStr(PCF8563,MINALARMADDR,&minalarm,0x1);
	ISendStr(PCF8563,HOUALARMADDR,&houralarm,0x1);
	IRcvStr(PCF8563,CON_STATUS1,&rcon_sta,0x01);
	wcon_sta=rcon_sta|0x02;
	ISendStr(PCF8563,CON_STATUS1,&wcon_sta,0x01);
	 while(1)
	 {
	 	DelayNS(1);
	 	IRcvStr(PCF8563,RDADDR,rd,0x7);			//读现在的秒、分、时、日、星期、月、年
		if(alarmflag==1)									
		{
			IRcvStr(PCF8563,CON_STATUS1,&rcon_sta,0x01);	//清PCF8563的AF标志位
      		wcon_sta=rcon_sta&0xF7;							//清AF位,但其他位不改变
       		ISendStr(PCF8563,CON_STATUS1,&wcon_sta,0x01);
       		alarmflag=0;		
		}
		if(flag2==0)			//两段程序交替显示可以通过flag1和flag2调节时间。
		{
			display_time();		//时间显示
			flag1++;
		}
		if(flag1==100)			//flag2控制时间显示时间,flag1控制日期显示时间
		{
			display_date();		//日期显示
			flag2++;
			if(flag2==50)
			{
				flag1=0;
				flag2=0;
			}
		}		
	 }	
}



实现实时时钟,电源拔插时间不重置(内置电池供电维持时间变化)。功能有查看日期,查看星期,查看闹钟和分别的调整时间,调整日期,调整闹钟,调整星期。且默认情况下不可以调整,当按下可调控键之后才可以调整、再次按可调控键关闭变为不可调整状态。还有整点报时功能(有个小瑕疵就是闹钟正好是整点的时候和这个整点报时同时间的时候不会报时和闹钟,这个很容易改好,因为下午3点要答辩了,懒得改了机子老是写不进去,学校穷便宜机子没办法)。然后这个闹钟可以设置星期几几点闹也可以设置为普通的每天的这个时间点闹铃,这些都是可以调控的。时钟所有的功能都有,只差一个秒表,,这个很简单,,,外部中断来一个或者定时器T1中断来一个都可以,我没弄,因为我这个已经代码很长了,头疼、加中断还得加显示函数和秒表变化函数if分大于60 时++啥的,但因为这个采用的显示是低四位高四位控制的,我强行加一个也比较麻烦所有就没加了。欢迎下载干货,难看懂的都有备注,写了断断续续一周+时间左右(恕在下才疏学浅,因为书上上课的时候没学过I2C总线和pcf8563所以写的比较久。)部分代码如下: #define MAIN_Fosc 22118400L //定义主时钟 频率也是计数计时周期一秒的计数值 #include "STC15Fxxxx.H" /***********************************************************/ #define DIS_DOT 0x20 #define DIS_BLACK 0x10 #define DIS_ 0x11 /****************************** 用户定义宏***********************************/ #define Timer0_Reload (65536UL -(MAIN_Fosc / 1000)) //Timer 0 中断频率,1000次/秒 频率倍数计数即周期 周期为1秒 除以一千就是1000次每秒 /*****************************************************************************/ /************* 本地常量声明 **************/ u8 code t_display[]={ //标准字库 // 0 1 2 3 4 5 6 7 8 9 A B C D E F //共阴 0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71, //black - H J K L N o P U t G Q r M y 0x00,0x40,0x76,0x1E,0x70,0x38,0x37,0x5C,0x73,0x3E,0x78,0x3d,0x67,0x50,0x37,0x6e, 0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF,0x46}; //0. 1. 2. 3. 4. 5. 6. 7. 8. 9. -1 u8code T_COM[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80}; //位码 取反就是书上213面 0从低位到高位 /************* IO口定义 **************/ sbit P_HC595_SER = P4^0; //pin 14 SER datainput sbit P_HC595_RCLK = P5^4; //pin 12 RCLk store(latch) clock sbit P_HC595_SRCLK = P4^3; //pin 11 SRCLK Shift data clock sbit led=P1^7; sbit led1=P4^7; sbit key=P3^2; /************* 本地变量声明 **************/ u8 LED8[8]; //显示缓冲 u8 display_index; //显示位索引 bit B_1ms; //1ms标志 u8IO_KeyState, IO_KeyState1, IO_KeyHoldCnt; //行列键盘变量 u8 KeyHoldCnt; //键按下计时 u8 KeyCode; //给用户使用的键码, 1~16有效 u8 cnt50ms; u8hello; u8nao; u8minute2; u8KeyCode0; u8 hour,hour1,minute,minute1,second,day,week,week1,month,year; //RTC变量 u16 msecond; u16msecond1; u8hello; u8naofu; u8zhuangtai; u8xunhuan; u8tuinao; //闹钟控制开启退出 u8xinqinao;//星期·闹钟 /************* 本地函数声明 **************/ void CalculateAdcKey(u16 adc); void IO_KeyScan(void); //50ms call void WriteNbyte(u8 addr, u8 *p, u8 number); void ReadNbyte( u8 addr, u8 *p, u8 number); void DisplayRTC(void); void DisplayRTC1(void); void DisplayRTC2(void); void ReadRTC(void);//读取时钟 void ReadRTC1(void); void ReadRTC2(void); void ReadRTC3(void); void WriteRTC(void); void WriteRTC1(void); void WriteRTC2(void); void WriteRTC3(void); void DisplayRTC3(void); void DisplayRTC4(void); void DisplayRTC5(void); /**************** 外部函数声明和外部变量声明*****************/ /**********************************************/ voidmain(void) { u8 i; P0M1= 0; P0M0 = 0; //设置为准双向口 P1M1= 0; P1M0 = 0; //设置为准双向口 P2M1= 0; P2M0 = 0; //设置为准双向口 P3M1= 0; P3M0 = 0; //设置为准双向口 P4M1= 0; P4M0 = 0; //设置为准双向口 P5M1= 0; P5M0 = 0; //设置为准双向口 P6M1= 0; P6M0 = 0; //设置为准双向口 P7M1= 0; P7M0 = 0; //设置为准双向口 display_index= 0; AUXR= 0x80;//T0时钟无分频 TMOD=0x00; //Timer0 set as 1T, 16 bits timer auto-reload,T0时钟无分频,16位自动重装 TH0= (u8)(Timer0_Reload / 256); //定时器取其高低8位为状态值 TL0= (u8)(Timer0_Reload % 256); ET0= 1; //Timer0 interrupt enable T0中断允许 TR0= 1; //Tiner0 run T0启动 EA= 1; //打开总中断 ,总中断允许 hello=0; if(nao==1){}//断电重置 elseif(nao==2){} else { nao=0; } for(i=0;i= 60) F0 = 1; //错误 if(minute>= 60) F0 = 1; //错误 if(hour >= 24) F0= 1; //错误 if(F0==1) //有错误, 默认12:00:00 { second= 0; minute= 0; hour = 12; WriteRTC(); } if(day>=32) F0=2; if(week>=8) F0=2; if(month>=13)F0=2; if(year>=100)F0=2; if(F0==2) //有错误, 默认12:00:00 { day=6; week=3; month=6; year=18; WriteRTC1(); } if(minute1>= 60) F0 = 3; //错误 if(hour1 >= 24) F0= 3; //错误 if(F0==3) { minute1=30; hour1=7; WriteRTC2(); } if(week1>=8) { WriteRTC3(); } DisplayRTC(); KeyHoldCnt= 0; //键按下计时 KeyCode= 0; //给用户使用的键码,1~16有效 IO_KeyState= 0; IO_KeyState1= 0; IO_KeyHoldCnt= 0; cnt50ms= 0; zhuangtai=0; KeyCode0=0; tuinao=0; minute2=61; naofu=0;//控制闹钟加1加5与分钟同步 xunhuan=0;//控制循环闹钟 xinqinao=0;//星期闹钟控制 if(msecond1==0){} else { msecond1=0; } while(1) { if(B_1ms) //1ms到 { B_1ms= 0; if(++msecond>= 500) //1秒到 刷新时钟 { if(hello==0) { ReadRTC(); DisplayRTC(); if(minute==0) //整点报时 { if(hour==hour1) { if(minute==minute1) {}else{ led1=0; } }else { led1=0; } } else{ led1=1; } if(hour==hour1) //闹钟实现块 { if(second==0)//保证和时钟分钟同步++乘其下一分钟前进行+5 +1操作 { msecond1=0; naofu=1; } else { if(naofu==1) { if(++msecond1>=60) {msecond1=0; naofu=0; } } } if(minute==minute1-1)//取出闹钟分钟值 { ReadRTC2(); minute2=minute1; }
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值