RTC相关库函数
void RCC_RTCCLKConfig(uint32_t CLKSource);//时钟源选择
void RCC_RTCCLKCmd(FunctionalState NewState)//时钟使能
ErrorStatus RTC_Init(RTC_InitTypeDef* RTC_InitStruct);
typedef struct
{ uint32_t RTC_HourFormat; //小时格式:24/12
uint32_t RTC_AsynchPrediv; //异步分频 系数
uint32_t RTC_SynchPrediv; //同步分频系数
}RTC_InitTypeDe
void RTC_ITConfig(uint32_t RTC_IT, FunctionalState NewState);
FlagStatus RTC_GetFlagStatus(uint32_t RTC_FLAG);
void RTC_ClearFlag(uint32_t RTC_FLAG);
ITStatus RTC_GetITStatus(uint32_t RTC_IT);
void RTC_ClearITPendingBit(uint32_t RTC_IT);
void RTC_WriteProtectionCmd(FunctionalState NewState);//取消写保护
ErrorStatus RTC_EnterInitMode(void);//进入配置模式,RTC_ISR_INITF位设置为1
void RTC_ExitInitMode(void)//退出初始化模式。
其他相关函数
uint32_t RTC_ReadBackupRegister(uint32_t RTC_BKP_DR);
void RTC_WriteBackupRegister(uint32_t RTC_BKP_DR, uint32_t Data)
void RTC_ITConfig(uint32_t RTC_IT, FunctionalState NewState);
RTC一般配置步骤
1)使能电源时钟,并使能 RTC 及 RTC 后备寄存器写访问。
我们要访问 RTC 和 RTC 备份区域就必须先使能电源时钟,然后使能 RTC 即后备区域访问。电源时钟使能,通过 RCC_APB1ENR 寄存器来设置;RTC 及 RTC 备份寄存器的写访问,通过 PWR_CR 寄存器的 DBP 位设置。库函数设置方法为:
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);//使能 PWR 时钟 PWR_BackupAccessCmd(ENABLE); //使能后备寄存器访问
2)开启外部低速振荡器,选择 RTC 时钟,并使能。
这个步骤,只需要在 RTC 初始化的时候执行一次即可,不需要每次上电都执行,这些操作 都是通过 RCC_BDCR 寄存器来实现的。
开启 LSE 的库函数为:
RCC_LSEConfig(RCC_LSE_ON);//LSE 开启
同时,选择 RTC 时钟源以及使能时钟函数为:
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE); //设选择 LSE 作为 RTC 时钟 RCC_RTCCLKCmd(ENABLE); //使能 RTC 时钟
3)初始化 RTC,设置 RTC 的分频,以及配置 RTC 参数。
初始化 RTC 是通过函数 RTC_Init 实现的:
ErrorStatus RTC_Init(RTC_InitTypeDef* RTC_InitStruct);
RTC 初始化参数结构体 RTC_InitTypeDef 定义:
typedef struct
{
uint32_t RTC_HourFormat;
uint32_t RTC_AsynchPrediv;
uint32_t RTC_SynchPrediv;
}RTC_InitTypeDef;
结构体一共只有三个成员变量:
参数 RTC_HourFormat 用来设置 RTC 的时间格式,也就是设置 CR 寄存器的 FMT 位。如果设置为 24 小时格式参数值可选择 RTC_HourFormat_24,12 小时格式, 参数值可以选择 RTC_HourFormat_24。
参数 RTC_AsynchPrediv 用来设置 RTC 的异步预分频系数,也就是设置 RTC_PRER 寄存器 的 PREDIV_A 相关位。同时,因为异步预分频系数是 7 位,所以最大值为 0x7F,不能超过这个值。
参数 RTC_SynchPrediv 用来设置 RTC 的同步预分频系数,也就是设置 RTC_PRER 寄存器的 PREDIV_S 相关位。同时,因为同步预分频系数也是 15 位,所以最大值为 0x7FFF,不能超 过这个值。
最后关于 RTC_Init 函数还要指出,在设置 RTC 相关参数之前,会先取消 RTC 写保护, 这个操作通过向寄存器 RTC_WPR 写入 0XCA 和 0X53 两个数据实现。所以 RTC_Init 函数体开头会有下面两行代码用来取消 RTC 写保护:
RTC->WPR = 0xCA;
RTC->WPR = 0x53;
在取消写保护之后,要对 RTC_PRER、RTC_TR 和 RTC_DR 等寄存器的写操作,必须先进入 RTC 初始化模式,才可以进行,库函数中进入初始化模式的函数为:
ErrorStatus RTC_EnterInitMode(void);
进入初始化模式之后,RTC_init 函数才去设置 RTC->CR 以及 RTC->PRER 寄存器的值。在设置完值之后,我们还要退出初始化模式,函数为:
void RTC_ExitInitMode(void)
最后再开启 RTC 写保护,往 RTC_WPR 寄存器写入值 0xFF 即可。
4)设置 RTC 的时间。
库函数中,设置 RTC 时间的函数为:
ErrorStatus RTC_SetTime(uint32_t RTC_Format, RTC_TimeTypeDef* RTC_TimeStruct);
实际上,RTC_SetTime 函数是用来设置时间寄存器 RTC_TR 的相关位的值。 RTC_SetTime 函数的第一个参数 RTC_Format,用来设置输入的时间格式为 BIN 格式还是 BCD 格式,可选值为 RTC_Format_BIN 和 RTC_Format_BCD。因为 RTC_DR 的数据必须是 BCD 格式,所以如果设置为 RTC_Format_BIN,那么在函数体内部会调用函数 RTC_ByteToBcd2 将参数转换为 BCD 格式。 接下来看看第二个初始化参数结构体 RTC_TimeTypeDef 的定义:
typedef struct
{
uint8_t RTC_Hours;
uint8_t RTC_Minutes;
uint8_t RTC_Seconds;
uint8_t RTC_H12;
}RTC_TimeTypeDef;
这四个的参数分别用来设置 RTC 时间参数的小时,分钟,秒钟,以及 AM/PM 符号。
5)设置 RTC 的日期。
设置 RTC 的日期函数为:
ErrorStatus RTC_SetDate(uint32_t RTC_Format, RTC_DateTypeDef* RTC_DateStruct);
RTC_SetDate 设置日期函数是用来设置日期寄存器 RTC_DR 的相关位的值。 第一个参数 RTC_Format,跟函数 RTC_SetTime 的第一个入口参数是一样的,用来设置输入 日期格式。 接下来我们看看第二个日期初始化参数结构体 RTC_DateTypeDef 的定义:
typedef struct
{
uint8_t RTC_WeekDay;
uint8_t RTC_Month;
uint8_t RTC_Date;
uint8_t RTC_Year;
}RTC_DateTypeDef;
这四个参数分别用来设置日期的星期几,月份,日期,年份。
6) 获取 RTC 当前日期和时间。
获取当前 RTC 时间的函数为:
void RTC_GetTime(uint32_t RTC_Format, RTC_TimeTypeDef* RTC_TimeStruct);
获取当前 RTC 日期的函数为:
void RTC_GetDate(uint32_t RTC_Format, RTC_DateTypeDef* RTC_DateStruct);
这两个函数实际就是读取RTC_TR寄存器和RTC_DR寄存器的时间和日期的值, 然后将值存放到相应的结构体中。
②使能后备寄存器访问: PWR_BackupAccessCmd();
③配置RTC时钟源,使能RTC时钟:
RCC_RTCCLKConfig();
RCC_RTCCLKCmd();
如果使用LSE,要打开LSE:RCC_LSEConfig(RCC_LSE_ON);
④初始化RTC(同步/异步分频系数和时钟格式):RTC_Init ();
⑤设置时间:RTC_SetTime ();
⑥设置日期:RTC_SetDate();
RTC闹钟配置一般步骤
⑥编写中断服务函数:RTC_Alarm_IRQHandler();
RTC周期性自动唤醒配置一般步骤
RTC_ITConfig();
EXTI_Init();
NVIC_Init();
编写中断服务函数: RTC_WKUP_IRQHandler();
RTC闹钟和周期性自动唤醒配置与中断服务函数
void RTC_Set_AlarmA(u8 week,u8 hour,u8 min,u8 sec)
{
EXTI_InitTypeDef EXTI_InitStructure;
RTC_AlarmTypeDef RTC_AlarmTypeInitStructure;
RTC_TimeTypeDef RTC_TimeTypeInitStructure;
RTC_AlarmCmd(RTC_Alarm_A,DISABLE);//关闭闹钟A
RTC_TimeTypeInitStructure.RTC_Hours=hour;//小时
RTC_TimeTypeInitStructure.RTC_Minutes=min;//分钟
RTC_TimeTypeInitStructure.RTC_Seconds=sec;//秒
RTC_TimeTypeInitStructure.RTC_H12=RTC_H12_AM;
RTC_AlarmTypeInitStructure.RTC_AlarmDateWeekDay=week;//星期
RTC_AlarmTypeInitStructure.RTC_AlarmDateWeekDaySel=RTC_AlarmDateWeekDaySel_WeekDay;//按星期闹
RTC_AlarmTypeInitStructure.RTC_AlarmMask=RTC_AlarmMask_None;//精确匹配星期,时分秒
RTC_AlarmTypeInitStructure.RTC_AlarmTime=RTC_TimeTypeInitStructure;
RTC_SetAlarm(RTC_Format_BIN,RTC_Alarm_A,&RTC_AlarmTypeInitStructure);
RTC_ClearITPendingBit(RTC_IT_ALRA);//清除RTC闹钟A的标志
EXTI_ClearITPendingBit(EXTI_Line17);//清除LINE17上的中断标志位
RTC_ITConfig(RTC_IT_ALRA,ENABLE);//开启闹钟A中断
RTC_AlarmCmd(RTC_Alarm_A,ENABLE);//开启闹钟A
EXTI_InitStructure.EXTI_Line = EXTI_Line17;//LINE17
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;//中断事件
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising; //上升沿触发
EXTI_InitStructure.EXTI_LineCmd = ENABLE;//使能LINE17
EXTI_Init(&EXTI_InitStructure);//配置
NVIC_InitStructure.NVIC_IRQChannel = RTC_Alarm_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02;//抢占优先级1
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x02;//子优先级2
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能外部中断通道
NVIC_Init(&NVIC_InitStructure);//配置
}
/*RTC_AlarmTypeDef 结构体的定义:
typedef struct
{
RTC_TimeTypeDef RTC_AlarmTime;
uint32_t RTC_AlarmMask;
uint32_t RTC_AlarmDateWeekDaySel;
uint8_t RTC_AlarmDateWeekDay;
}RTC_AlarmTypeDef;
结构体的第一个成员变量为 RTC_TimeTypeDef 类型的成员变量 RTC_AlarmTime,这个是
用来设置闹钟时间的。
第二个参数 RTC_AlarmMask,使用来设置闹钟时间掩码,也就是在我们第一个参数设置的
时间中(包括后面参数 RTC_AlarmDateWeekDay 设置的星期几/哪一天),哪些是无关的。比
如我们设置闹钟时间为每天的10点10分10秒,那么我们可以选择值RTC_AlarmMask_DateWeekDay,也就是我们不关心是星期几/每月哪一天。这里我们选择为RTC_AlarmMask_None,也就是精确匹配时间,所有的时分秒以及星期几/(或者每月哪一天)都要精确匹配。
第三个参数 RTC_AlarmDateWeekDaySel,用来选择是闹钟是按日期还是按星期。比如我们
选择RTC_AlarmDateWeekDaySel_WeekDay那么闹钟就是按星期。如果我们选择RTC_AlarmDateWeekDaySel_Date 那么闹钟就是按日期。
第四个参数 RTC_AlarmDateWeekDay 用来设置闹钟的日期或者星期几。比如我们第三个参
数 RTC_AlarmDateWeekDaySel 设置了值为 RTC_AlarmDateWeekDaySel_WeekDay,也就是按星期,那么参数RTC_AlarmDateWeekDay的取值范围就为星期一~星期天,也就是
RTC_Weekday_Monday~RTC_Weekday_Sunday。如果第三个参数 RTC_AlarmDateWeekDaySel
设置值为 RTC_AlarmDateWeekDaySel_Date,那么它的取值范围就为日期值,0~31。
*/
void RTC_Set_WakeUp(u32 wksel,u16 cnt)
{
EXTI_InitTypeDef EXTI_InitStructure;
RTC_WakeUpCmd(DISABLE);//关闭WAKE UP
RTC_WakeUpClockConfig(wksel);//唤醒时钟选择
RTC_SetWakeUpCounter(cnt);//设置WAKE UP自动重装载寄存器
RTC_ClearITPendingBit(RTC_IT_WUT); //清除RTC WAKE UP的标志
EXTI_ClearITPendingBit(EXTI_Line22);//清除LINE22上的中断标志位
RTC_ITConfig(RTC_IT_WUT,ENABLE);//开启WAKE UP 定时器中断
RTC_WakeUpCmd( ENABLE);//开启WAKE UP 定时器
EXTI_InitStructure.EXTI_Line = EXTI_Line22;//LINE22
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;//中断事件
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising; //上升沿触发
EXTI_InitStructure.EXTI_LineCmd = ENABLE;//使能LINE22
EXTI_Init(&EXTI_InitStructure);//配置
NVIC_InitStructure.NVIC_IRQChannel = RTC_WKUP_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02;//抢占优先级1
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x02;//子优先级2
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能外部中断通道
NVIC_Init(&NVIC_InitStructure);//配置
}
//RTC闹钟中断服务函数
void RTC_Alarm_IRQHandler(void)
{
if(RTC_GetFlagStatus(RTC_FLAG_ALRAF)==SET)//ALARM A中断?
{
RTC_ClearFlag(RTC_FLAG_ALRAF);//清除中断标志
printf("ALARM A!\r\n");
}
EXTI_ClearITPendingBit(EXTI_Line17); //清除中断线17的中断标志
}
//RTC WAKE UP中断服务函数
void RTC_WKUP_IRQHandler(void)
{
if(RTC_GetFlagStatus(RTC_FLAG_WUTF)==SET)//WK_UP中断?
{
RTC_ClearFlag(RTC_FLAG_WUTF); //清除中断标志
LED1=!LED1;
}
EXTI_ClearITPendingBit(EXTI_Line22);//清除中断线22的中断标志
}
main函数
int main(void)
{
u8 k=0;
RTC_TimeTypeDef RTC_TimeStruct;
RTC_DateTypeDef RTC_DateStruct;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2
delay_init(168); //初始化延时函数
uart_init(115200); //初始化串口波特率为115200
usmart_dev.init(84); //初始化USMART
LED_Init(); //初始化LED
My_RTC_Init(); //初始化RTC
RTC_Set_WakeUp(RTC_WakeUpClock_CK_SPRE_16bits,0); //配置WAKE UP中断,1秒钟中断一次
delay_ms(10000);
RTC_GetTime(RTC_Format_BIN,&RTC_TimeStruct);
printf("Time:%02d:%02d:%02d\r\n",RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds);
RTC_GetDate(RTC_Format_BIN, &RTC_DateStruct);
printf("Date:20%02d-%02d-%02d\r\n",RTC_DateStruct.RTC_Year,RTC_DateStruct.RTC_Month,RTC_DateStruct.RTC_Date);
printf("Week:%d\r\n",RTC_DateStruct.RTC_WeekDay);
while(1)
{
if(k%2==0)
{
RTC_GetTime(RTC_Format_BIN,&RTC_TimeStruct);
printf("Time:%02d:%02d:%02d\r\n",RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds);
}
LED0=!LED0; 4
delay_ms(500);
k++;
}
}
运行结果
RTC实时时钟实验