u16 temp=BKP_ReadBackupRegister(BKP_DR1); //0x5050
//开启RTC的时钟
RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP|RCC_APB1Periph_PWR,ENABLE); //RTC是存在后备区域中的,所以是要开启后备区域和电源时钟的
PWR_BackupAccessCmd(ENABLE); //电源控制寄存器(PWR_CR)的BPB位使能对后备区域和RTC的访问
if(temp!=0x5050)
{
// //开启RTC的时钟
// RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP|RCC_APB1Periph_PWR,ENABLE); //RTC是存在后备区域中的,所以是要开启后备区域和电源时钟的
// PWR_BackupAccessCmd(ENABLE); //电源控制寄存器(PWR_CR)的BPB位使能对后备区域和RTC的访问
BKP_DeInit(); //复位备份区域
RCC_RTCCLKConfig(RCC_RTCCLKSource_HSE_Div128);//配置RTC的时钟来源,8M的128分频
RCC_RTCCLKCmd(ENABLE); //使能RTC时钟
RTC_WaitForLastTask(); //等待上一次对RTC寄存器的读写操作完成
RTC_WaitForSynchro(); //等待APB1和RTC等时钟同步
RTC_ITConfig(RTC_IT_SEC,ENABLE); //开启秒中断和闹钟中断
RTC_WaitForLastTask(); //等待上一次对RTC寄存器的读写操作完成
RTC_EnterConfigMode(); //进入RTC的配置,CNF=1;
RTC_SetPrescaler(62500-1); //设置时钟预分频为1Hz=8M/128=62.5KHz/62500=1Hz //内部还会自动加1
RTC_WaitForLastTask(); //等待最近一次对RTC寄存器的写操作完成
RTC_Set_Time(2014,3,9,20,25,5); //设置时间
RTC_ExitConfigMode(); //退出配置
BKP_WriteBackupRegister(BKP_DR1,0x5050); //向指定的后备寄存器中写入用户程序数据
}
else
{
// //开启RTC的时钟
// RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP|RCC_APB1Periph_PWR,ENABLE); //RTC是存在后备区域中的,所以是要开启后备区域和电源时钟的
// PWR_BackupAccessCmd(ENABLE); //电源控制寄存器(PWR_CR)的BPB位使能对后备区域和RTC的访问
RTC_WaitForSynchro(); //等待同步 ,为什么会在这里死循环呢?如何重新开启时钟的话,这个问题就会解决了。
RTC_ITConfig(RTC_IT_SEC,ENABLE); //使能RTC秒中断
RTC_WaitForLastTask(); //等待最近一次对RTC寄存器的写操作完成
}
//配置中断
RTC_NVIC_Init();
//获取时间
RTC_Get_Time();
}
最好是在if外面开启时钟,不然很容易在else中的RTC_WaitForSynchro();出现死循环。
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
---------------------------------
我的不用在外面,呵呵
---------------------------------
谢谢分享
RTC_WaitForSynchro(); //等待同步 ,为什么会在这里死循环呢?如何重新开启时钟的话,这个问题就会解决了。
就会出现BKP_0x5050,RTCEN=0,LSEON=0,RTCSEL=0,所以就会在RTC_WaitForSynchro()除死机,
但用战舰开发板上的RTC程序测试没有发现这个现象,不知道是不是跟程序的大小有关,
个人觉得,因上面几个位状态改变出现在下载后调试前,所以应该不是程序的问题。
刚才在论坛上查了一下,没发现有类似的问题,所以还是想和大家分享一下,一家之言,不足为据。
u8 RTC_InterOscInit(void)
{
//检查是不是第一次配置时钟
u8 temp=0;
// int delay;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE); //使能PWR和BKP外设时钟
PWR_BackupAccessCmd(ENABLE); //使能后备寄存器访问
// RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI); //设置RTC时钟(RTCCLK),选择LSI作为RTC时钟
// RCC_RTCCLKCmd(ENABLE); //使能RTC时钟
RCC_LSICmd(ENABLE);
if (BKP_ReadBackupRegister(BKP_DR1) != 0x5050) //从指定的后备寄存器中读出数据:读出了与写入的指定数据不相乎
{
BKP_DeInit(); //复位备份区域
RCC_LSICmd(ENABLE); //设置内部低速晶振(LSI),使用外设低速晶振
while (RCC_GetFlagStatus(RCC_FLAG_LSIRDY) == RESET) //检查指定的RCC标志位设置与否,等待低速晶振就绪
{
temp++;
delay_ms(10);
}
if(temp>=250)return 1;//初始化时钟失败,晶振有问题
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI); //设置RTC时钟(RTCCLK),选择LSI作为RTC时钟
RCC_RTCCLKCmd(ENABLE); //使能RTC时钟
RTC_WaitForLastTask(); //等待最近一次对RTC寄存器的写操作完成
RTC_WaitForSynchro(); //等待RTC寄存器同步
RTC_ITConfig(RTC_IT_SEC, ENABLE); //使能RTC秒中断
RTC_WaitForLastTask(); //等待最近一次对RTC寄存器的写操作完成
RTC_EnterConfigMode();/// 允许配置
RTC_SetPrescaler(40000); //设置RTC预分频的值
RTC_WaitForLastTask(); //等待最近一次对RTC寄存器的写操作完成
RTC_Set(2017,0,0,0,0,0); //设置时间
RTC_ExitConfigMode(); //退出配置模式
BKP_WriteBackupRegister(BKP_DR1, 0X5050); //向指定的后备寄存器中写入用户程序数据
}
else//系统继续计时
{
RTC_WaitForSynchro(); //等待最近一次对RTC寄存器的写操作完成
RTC_ITConfig(RTC_IT_SEC, ENABLE); //使能RTC秒中断
RTC_WaitForLastTask(); //等待最近一次对RTC寄存器的写操作完成
}
RTC_NVIC_Config();//RCT中断分组设置
RTC_Get();//更新时间
return 0; //ok
}
一周热门 更多>