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); }