这次是RTC的笔记:)
RTC这东西晕晕的,由于一个模块触及到了RTC,BKP,RCC多个模块,之间的联系让人有点含糊
入门的常识请咱们看手册,我来总结:
总归,RTC仅仅个能靠电池保持运转的32位定时器over!
所以,使用时要注意以下问题:
1. 上电后要检查备份电池有没有断过电。怎么检查? 恩,RTC的示例代码中现已明示:
往备份域寄存器中写一个特别的字符,备份域寄存器是和RTC一同在断电下能保存数据的。
上电后检查下这个特别字符是否还存在,假如存在,ok,RTC的数据应该也没丢,不需求重新装备它
假如那个特别字符丢了,那RTC的定时器数据必定也丢了,那咱们要重新来装备RTC了
这个进程包含时钟使能、RTC时钟源切换、设置分频系数等等,这个能够参阅FWLibexampleRTCCalendar的代码
在我的这个实例里,检查备份域掉电在Init.c的RTC_Conig()中,函数内若检测到BKP掉电,则会调用RTC_Configuration()
2. 由于RTC的一些设置是保存在后备域中的,so,操作RTC的设置寄存器前,要翻开后备域模块中的写保护功用。
3. RTC设定值写入前后都要检查指令有没有完结,调用RTC_WaitForLastTask();
详细的RTC初始化代码如下:
////////////////////////////////////////////////////////////////////////////////
// RTC时钟初始化!
////////////////////////////////////////////////////////////////////////////////
void RTC_Configuration(void)
{
//启用PWR和BKP的时钟(from APB1)
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE);
//后备域解锁
PWR_BackupAccessCmd(ENABLE);
//备份寄存器模块复位
BKP_DeInit();
//外部32.768K其哟偶那个
RCC_LSEConfig(RCC_LSE_ON);
//等候安稳
while (RCC_GetFlagStatus(RCC_FLAG_LSERDY) == RESET);
//RTC时钟源装备成LSE(外部32.768K)
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);
//RTC敞开
RCC_RTCCLKCmd(ENABLE);
//敞开后需求等候APB1时钟与RTC时钟同步,才干读写寄存器
RTC_WaitForSynchro();
//读写寄存器前,要确认上一个操作现已完毕
RTC_WaitForLastTask();
//设置RTC分频器,使RTC时钟为1Hz
//RTC period = RTCCLK/RTC_PR = (32.768 KHz)/(32767+1)
RTC_SetPrescaler(32767);
//等候寄存器写入完结
RTC_WaitForLastTask();
//使能秒中止
RTC_ITConfig(RTC_IT_SEC, ENABLE);
//等候写入完结
RTC_WaitForLastTask();
return;
}
void RTC_Config(void)
{
//咱们在BKP的后备寄存器1中,存了一个特别字符0xA5A5
//第一次上电或后备电源掉电后,该寄存器数据丢掉,
//标明RTC数据丢掉,需求重新装备
if (BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5)
{
//重新装备RTC
RTC_Configuration();
//装备完结后,向后备寄存器中写特别字符0xA5A5
BKP_WriteBackupRegister(BKP_DR1, 0xA5A5);
}
else
{
//若后备寄存器没有掉电,则无需重新装备RTC
//这儿咱们能够使用RCC_GetFlagStatus()函数检查本次复位类型
if (RCC_GetFlagStatus(RCC_FLAG_PORRST) != RESET)
{
//这是上电复位
}
else if (RCC_GetFlagStatus(RCC_FLAG_PINRST) != RESET)
{
//这是外部RST管脚复位
}
//铲除RCC中复位标志
RCC_ClearFlag();
//尽管RTC模块不需求重新装备,且掉电后依托后备电池仍然运转
//可是每次上电后,仍是要使能RTCCLK???????
//RCC_RTCCLKCmd(ENABLE);
//等候RTC时钟与APB1时钟同步
//RTC_WaitForSynchro();
//使能秒中止
RTC_ITConfig(RTC_IT_SEC, ENABLE);
//等候操作完结
RTC_WaitForLastTask();
}
#ifdef RTCClockOutput_Enable
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE);
PWR_BackupAccessCmd(ENABLE);
BKP_TamperPinCmd(DISABLE);
BKP_RTCOutputConfig(BKP_RTCOutputSource_CalibClock);
#endif
return;
}