当前位置:首页 > 单片机 > 单片机
[导读]#include #include #include #include #include "pcf8563.h"#include "lcd.h"#include "keyboard.h"//时钟芯片数据接口PA0#define DATE_DT_set asm("sbi 0x1B,0")#define DATE_DT_clr asm("cbi 0x1B,0")//时钟芯片时

#include

#include

#include

#include

#include "pcf8563.h"

#include "lcd.h"

#include "keyboard.h"

//时钟芯片数据接口PA0

#define DATE_DT_set asm("sbi 0x1B,0")

#define DATE_DT_clr asm("cbi 0x1B,0")

//时钟芯片时钟接口PA1

#define DATE_CLK_set asm("sbi 0x1B,1")

#define DATE_CLK_clr asm("cbi 0x1B,1")

unsigned char old_minute,new_minute;

unsigned char number1[13]=

{

0x30, //0

0x31, //1

0x32, //2

0x33, //3

0x34, //4

0x35, //5

0x36, //6

0x37, //7

0x38, //8

0x39, //9

0x20, //空格

0x2E, //.

0x3A //:

};

void delayus(unsigned char i)

{

while(i)

i--;

}

// ************************************************ //

// *** This routine will send the I2C Start Bit *** //

// ************************************************ //

void I2C_Start (void) //I2C发送开始位

{

DDRA|=0x03; //将PA0数据端口(SDA),PA1时钟端口(SCL)设为输出

DATE_CLK_set; //将时钟端口(SCL)设为高

DATE_DT_set; //将数据端口(SDA)设为高

delayus(2);

DATE_DT_clr; //将数据端口(SDA)设为低

delayus(2);

DATE_DT_set;

}

// *********************************************** //

// *** This routine will send the I2C Stop Bit *** //

// *********************************************** //

void I2C_Stop (void) //I2C发送停止位

{

DDRA|=0x03; //将PA0数据端口(SDA),PA1时钟端口(SCL)设为输出

DATE_DT_clr; //将数据端口(SDA)设为低

DATE_CLK_set; //将时钟端口(SCL)设为高

delayus(2);

DATE_DT_set; //将数据端口(SDA)设为高

delayus(2);

}

// *********************************************************************** //

// *** 发送完毕检查校验位,有校验位返回1,无返回0 *** //

// *********************************************************************** //

unsigned char I2C_Ackn(void)

{

unsigned char errtime=255;

//DATE_CLK_clr; // 将时钟端口(SCL)设为低

DDRA|=0x02;

DDRA&=0xFE; // 设置数据口(SDA)为输入

delayus(2);

while(PINA&0x01)

{

errtime--;

if (!errtime) //errtime=0,没接收到

{

I2C_Stop();

return 0x00;

}

}

DATE_CLK_set;

delayus(2);

DATE_CLK_clr; // 将时钟端口(SCL)设为低

delayus(2);

return 0x01; //true

}

// ******************************************************** //

// *** This routine will write a byte to the I2C device *** //

// ******************************************************** //

void Write_I2C_Byte(unsigned char byte) //写一个字节到I2C设备

{

unsigned char i;

DDRA|=0x03; //将PA0数据端口(SDA)设为输出

for (i = 0; i < 8; i++) //传送8位数据

{

DATE_CLK_clr; //将时钟端口(SCL)设为低

if((byte & 0x80)) DATE_DT_set; // 设置 SDA 位

else DATE_DT_clr; // 清除 SDA 位

delayus(2);

DATE_CLK_set; //将时钟端口(SCL)设为高

asm("nop");

byte = byte << 1; //将输出数据左移一位

}

DATE_CLK_clr; // 校验标志位 (每传送8位,有一校验位)

if (I2C_Ackn()==0) // Check for acknowledge from I2C device

yonghudenglu();

//DATE_CLK_clr;

}

// ********************************************************* //

// *** This routine will read a byte from the I2C device *** //

// ********************************************************* //

unsigned char Read_I2C_Byte(void) //读取I2C设备的数据

{

unsigned char i,buff = 0;

delayus(2);

DDRA|=0x02; //PA1为时钟,输出

DDRA&=0xfe; //设置数据口(SDA)为输入

for (i = 0; i < 8; i++)

{

DATE_CLK_clr; // 将时钟端口(SCL)设为低

delayus(2);

DATE_CLK_set; //将时钟端口(SCL)设为高

delayus(2);

// 在 SDA 位上读取数据

if ( PINA&=0x01 )

buff++;

buff = (buff << 1);

delayus(2);

}

DDRA|=0x03; //设为输出,发送校验位

DATE_DT_clr;

delayus(2);

DATE_CLK_set;

delayus(2);

DATE_CLK_clr; //将时钟端口(SCL)设为高

//DATE_DT_clr;

return buff; // 返回读取值

}

//读8563寄存器

unsigned char rtc_read(unsigned char address)

{

unsigned char d;

I2C_Start();

Write_I2C_Byte(0xa2);

Write_I2C_Byte(address);

I2C_Start();

Write_I2C_Byte(0xa3);

d=Read_I2C_Byte();

d=d>>1;

I2C_Stop();

//for(;;){}

return d;

}

////////////////////////////////////////////////////////////////////////////////

//写8563寄存器

void rtc_write(unsigned char address,unsigned char data1)

{

I2C_Start();

Write_I2C_Byte(0xa2);

Write_I2C_Byte(address);

Write_I2C_Byte(data1);

I2C_Stop();

}

////////////////////////////////////////////////////////////////////////////////

void rtc_start(void)

{

rtc_write(0,0);

}

////////////////////////////////////////////////////////////////////////////////

void rtc_stop(void)

{

rtc_write(0,0x20);

}

void GetPCF8563(unsigned char *time)

{

CLI();

*time=(rtc_read(2)&0x7f); //寄存器0x02为秒寄存器

*(time+1)=(rtc_read(3)&0x7f); //寄存器0x03为分寄存器

*(time+2)=(rtc_read(4)&0x3f); //寄存器0x04为时寄存器

*(time+3)=(rtc_read(5)&0x3f); //寄存器0x05为天寄存器

*(time+4)=(rtc_read(7)&0x1f); //寄存器0x07为月寄存器

*(time+5)=(rtc_read(8)); //寄存器0x08为年寄存器

SEI();

}

unsigned char get_second(void) //获得当前秒数

{

unsigned i,j;

i=(rtc_read(2)&0x7f); //寄存器0x02为秒寄存器

j=(i&0x0f)+(i>>4)*10;

return j;

}

////////////////////////////////////////////////////////////////////////////////

void SetPCF8563(unsigned char adds,unsigned char data)

{

CLI();

rtc_stop();

rtc_write(adds,data);

rtc_start();

SEI();

}

//设置时间 (**年/**月/**日 **时:**分)

void set8563(void)

{

unsigned char maini=0,mainj=0,numb[12],newkey;

unsigned char sign=0;

unsigned char *time;

unsigned char displayn[12];

time=numb;

while((mainj<10)||(sign==0))

{

if(kbscan()!=0x20)

{

maini=kbscan();

while(kbscan()==maini)

asm("nop");

if(maini<10)

{

numb[mainj]=maini;

displayn[mainj]=number1[maini];

mainj++;

}

else if(maini==12)

{

if(mainj>0)

mainj--;

numb[mainj]=number1[10];

displayn[mainj]=number1[10];

}

else if((mainj>=10)&&(maini==13))

{

sign=1;

}

display(0x80,displayn[0],displayn[1]);

display(0x81,0x2f,displayn[2]);

display(0x82,displayn[3],0x2f);

display(0x83,displayn[4],displayn[5]);

display(0x84,0x20,0x20);

display(0x85,displayn[6],displayn[7]);

display(0x86,0x3a,displayn[8]);

display(0x87,displayn[9],0x20);

}

}

numb[0]=(numb[0]<<4)+(numb[1]&0x0F); //年

numb[2]=(numb[2]<<4)+(numb[3]&0x0F); //月

numb[4]=(numb[4]<<4)+(numb[5]&0x0F); //日

numb[6]=(numb[6]<<4)+(numb[7]&0x0F); //时

numb[8]=(numb[8]<<4)+(numb[9]&0x0F); //分

SetPCF8563(8,numb[0]); //设置年

SetPCF8563(7,numb[2]); //设置月

SetPCF8563(5,numb[4]); //设置日

SetPCF8563(4,numb[6]); //设置时

SetPCF8563(3,numb[8]); //设置分

}

//显示时间函数,屏幕第一行显示

void displaytime(unsigned char *time)

{

GetPCF8563(time);

new_minute=*(time+1);

if(new_minute!=old_minute)

{

display(0x80,0x30+(*(time+5)>>4),0x30+(*(time+5)&0x0F));

display(0x81,0x2f,0x30+(*(time+4)>>4));

display(0x82,0x30+(*(time+4)&0x0F),0x2f);

display(0x83,0x30+(*(time+3)>>4),0x30+(*(time+3)&0x0F));

display(0x84,0x20,0x20);

display(0x85,0x30+(*(time+2)>>4),0x30+(*(time+2)&0x0F));

display(0x86,0x3a,0x30+(*(time+1)>>4));

display(0x87,0x30+(*(time+1)&0x0F),0x20);

old_minute=new_minute;

}

//display(0x98,0x30+(*time>>4),0x30+(*time&0x0F));

}

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭