这些小活动你都参加了吗?快来围观一下吧!>>
电子产品世界» 论坛首页» 企业专区» GD32» GD32105R_START初体验之RTC实现万年历

共2条 1/1 1 跳转至

GD32105R_START初体验之RTC实现万年历

高工
2015-04-06 15:10:30 打赏

GD32 105R_STAR的后备寄存器(10个16位的寄存器)可以用于在关闭VDD时,保存20个字节的用户应用数据。RTC和后备寄存器不会被系统或电源复位源复位;当从待机模式唤醒时,也不会被复位。

楼主在本次试验中,采用的后备寄存器的BKP_DR1进行标志位的存储,供应用程序查询判断。如果标志位被更改,则进行RTC的初始化设置,否则跳过,代码如下:

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

*函数名称:void RtcInit(uint16 *time)

*函数功能: 按照默认值初始化实时时钟,秒值增加一次后产生中断.

*入口参数:time:初始化时间指针

*出口参数:

*日期:2015 4 6

*版本:V1.0

*作者:yrj

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

void RtcInit(uint16 *time)

{

if(BKP_ReadBackupRegister(BKP_DR1)!=0xAAAA) /*未设置过,开始初始化*/

{

RTC_Configuration(); /*初始化RTC*/

RTC_Set(time);/*默认时间 */

BKP_WriteBackupRegister(BKP_DR1,0xAAAA );

}

else /*系统继续计时*/

{

if(RCC_GetFlagStatus(RCC_FLAG_PORRST) != RESET) /*电复位*/

{}

else if(RCC_GetFlagStatus(RCC_FLAG_PINRST) != RESET) /*管脚复位*/

{}

RCC_RTCCLKCmd(ENABLE); //开启备份寄存器时钟

RTC_WaitForSynchro(); //

RTC_WaitForLastTask(); //等待RTC寄存器操作完成

RTC_ITConfig(RTC_IT_SEC,ENABLE); /*允许秒中断*/

RTC_WaitForLastTask(); /*等待RTC寄存器操作完成*/

}

}

根据当前的实际时钟,对RTC的初始化设置,并使其产生1秒的周期定时,RTC初始化代码如下:

/*

****************************************************************************

*函数名称:u8 RTC_Set(uint16 *tim )

*函数功能: 设置时钟,以1970年1月1日为基准//1970~2099年为合法年份

*入口参数:

*出口参数:

*日期:2015 4 6

*版本:V1.0

*作者:yrj

****************************************************************************

*/

/*平年的月份日期表*/

const uint8 mon_table[12]={31,28,31,30,31,30,31,31,30,31,30,31};


u8 RTC_Set(uint16 *tim )

{

uint16 syear=tim[0],smon=tim[1],sday=tim[2],hour=tim[3],min=tim[4],sec=tim[5];

u16 t;

u32 seccount=0;

if(syear<2000||syear>2099)return 1;/*syear范围1970-2099,此处设置范围为2000-2099 */

for(t=1970;t

{

if(Is_Leap_Year(t))seccount+=31622400;/*闰年的秒钟数*/

else seccount+=31536000; /*平年的秒钟数*/

}

smon-=1;

for(t=0;t

{

seccount+=(u32)mon_table[t]*86400;/*月份秒钟数相加*/

if(Is_Leap_Year(syear)&&t==1) seccount+=86400;/*闰年2月份增加一天的秒钟数*/

}

seccount+=(u32)(sday-1)*86400;/*把前面日期的秒钟数相加 */

seccount+=(u32)hour*3600;/*小时秒钟数*/

seccount+=(u32)min*60; /*分钟秒钟数*/

seccount+=sec;/*最后的秒钟加上去*/


/*设置时钟*/

PWR_BackupAccessCmd(ENABLE); //使能后备寄存器

RTC_WaitForLastTask();

RTC_SetCounter(seccount); /*设置时钟*/

RTC_WaitForLastTask();

return 0;

}

关于闰年平年的计算

/*

****************************************************************************

*函数名称:u8 Is_Leap_Year(u16 year)

*函数功能: //判断是否是闰年函数

*入口参数:

*出口参数:

*日期:2015 4 6

*版本:V1.0

*作者:yrj

****************************************************************************

*/

u8 Is_Leap_Year(u16 year)

{

if(year%4==0) /*必须能被4整除*/

{

if(year%100==0)

{

if(year%400==0) {return 1;}/*如果以00结尾,还要能被400整除*/

else {return 0;}

}

else

{ return 1; }

}

else{ return 0; }

}

在RTC的秒中断服务函数中,对年月日时分秒进行提取,用于万年历的显示,另外还可以根据具体的实际项目需要,进行设计开发,如每天0时定时开启风机等。。。

/*

****************************************************************************

*函数名称:void RtcException(void)

*函数功能: RTC中断服务函数 在函数中刷新实时日期时间,并发送消息邮箱

*入口参数:无

*出口参数:无

*日期:2015 4 6

*版本:V1.0

*作者:yrj

****************************************************************************

*/

void RtcException(void)

{

if(RTC_GetITStatus(RTC_IT_SEC)==SET)

{

RTC_WaitForLastTask();

RTC_ClearITPendingBit(RTC_IT_SEC);

RTC_WaitForLastTask();

/*读取当前时分秒*/

RTC_Get(); /* 刷新当前时间 提取 年、月、日、时、分、秒*/

if((0==timer.hour)&&(0==timer.min)&&(0==timer.sec))

{

//启动排风,或是设置闹钟。。。自由发挥喽

}

}

}

在GD32 105R 的RTC中断中,对于RTC计数值中年月日时分秒的提取:

/*

****************************************************************************

*函数名称:u8 RTC_Get(void)

*函数功能: 得到当前的时间 ,RTC2Timer

*入口参数:

*出口参数:

*日期:2015 4 6

*版本:V1.0

*作者:yrj

****************************************************************************

*/

u8 RTC_Get(void)

{

static u16 daycnt=0;

u32 timecount=0;

u32 temp=0;

u16 temp1=0;

timecount=RTC_GetCounter();

temp=timecount/86400; /*得到天数(秒钟数对应的)*/

if(daycnt!=temp)/*超过一天了*/

{

daycnt=temp;

temp1=1970; /*从1970年开始*/

while(temp>=365)

{

if(Is_Leap_Year(temp1))/*是闰年*/

{

if(temp>=366)temp-=366;/*闰年的秒钟数*/

else {temp1++;break;}

}

else temp-=365; /*平年 */

temp1++;

}

timer.w_year=temp1; /*得到年份*/

temp1=0;

while(temp>=28) /*超过了一个月*/

{

if(Is_Leap_Year(timer.w_year)&&temp1==1)/*当年是不是闰年/2月份*/

{

if(temp>=29)temp-=29;/*闰年的秒钟数*/

else break;

}

else

{

if(temp>=mon_table[temp1])temp-=mon_table[temp1];/*平年*/

else break;

}

temp1++;

}

timer.w_month=temp1+1; //得到月份

timer.w_date=temp+1; //得到日期

}

temp=timecount%86400; //得到秒钟数

timer.hour=temp/3600; //小时

timer.min=(temp%3600)/60; //分钟

timer.sec=(temp%3600)%60; //秒钟

timer.week=RTC_Get_Week(timer.w_year,timer.w_month,timer.w_date); /*获取星期*/

timer.year=timer.w_year-2000; //专为显示用年 14 7 22

return 0;

}

对于星期的判断:

/*

****************************************************************************

*函数名称:RTC_Get_Week(u16 year,u8 month,u8 day)

*函数功能: 获得现在是星期几,输入公历日期得到星期(只允许1901-2099年)

*入口参数:公历年月日

*出口参数:星期号

*日期:2015 4 6

*版本:V1.0

*作者:yrj

****************************************************************************

*/

u8 const table_week[12]={0,3,3,6,1,4,6,2,5,0,3,5}; /*月修正数据表*/

u8 RTC_Get_Week(u16 year,u8 month,u8 day)

{

u16 temp2;

u8 yearH,yearL;

yearH=year/100; yearL=year%100;

/*如果为21世纪,年份数加100 */

if (yearH>19)yearL+=100;

/* 所过闰年数只算1900年之后的 */

temp2=yearL+yearL/4;

temp2=temp2%7;

temp2=temp2+day+table_week[month-1];

if (yearL%4==0&&month<3)temp2--;

return(temp2%7);

}





关键词: GD32105R_STAR 万年历 STM32万

高工
2021-03-25 08:19:44 打赏
2楼

谢谢分享!


共2条 1/1 1 跳转至

回复

匿名不能发帖!请先 [ 登陆 注册]