STM32系列第16篇--RTC實(shí)時時鐘
RCT特征:
可編程的預(yù)分頻系數(shù),分頻系數(shù)最高2的20次方。
32位可編程計數(shù),用于較長時間段的測量。
2個分離的時鐘。
可以選擇三種RTC時鐘源:HSE/128;LSE振蕩器;LSI振蕩器。
2個獨(dú)立的復(fù)位類型:APB1由系統(tǒng)復(fù)位;RTC由后備域復(fù)位。
三個專門的可屏蔽中斷:鬧鐘中斷;秒中斷(一個可編程周期,最長可達(dá)1s);溢出中斷。
RTC工作原理框圖RTCCLK經(jīng)過RTC_DIV預(yù)分頻,RTC_PRL設(shè)置預(yù)分頻系數(shù),然后得到TR_CLK時鐘信號,我們一般設(shè)置其周期為1s,RTC_CNT計數(shù)器計數(shù),假如1970設(shè)置為時間起點(diǎn)為0s,通過當(dāng)前時間的秒數(shù)計算得到當(dāng)前的時間。RTC_ALR是設(shè)置鬧鐘時間,RTC_CNT計數(shù)到RTC_ALR就會產(chǎn)生計數(shù)中斷,RTC_Second為秒中斷,用于刷新時間,RTC_Overflow是溢出中斷。
實(shí)驗例程:#include"sys.h"#include"delay.h"#include"usart.h"#include"rtc.h"_calendar_objcalendar;//時鐘結(jié)構(gòu)體staticvoidRTC_NVIC_Config(void){NVIC_InitTypeDefNVIC_InitStructure;NVIC_InitStructure.NVIC_IRQChannel=RTC_IRQn;//RTC全局中斷NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;NVIC_InitStructure.NVIC_IRQChannelSubPriority=0;NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;//使能該通道中斷NVIC_Init(&NVIC_InitStructure);//根據(jù)NVIC_InitStruct中指定的參數(shù)初始化外設(shè)NVIC寄存器}//實(shí)時時鐘配置//初始化RTC時鐘,同時檢測時鐘是否工作正常//BKP->DR1用于保存是否第一次配置的設(shè)置//返回0:正常//其他:錯誤代碼u8 RTC_Init(void){ //檢查是不是第一次配置時鐘 u8 temp=0; RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE); //使能PWR和BKP外設(shè)時鐘 PWR_BackupAccessCmd(ENABLE); //使能后備寄存器訪問 if (BKP_ReadBackupRegister(BKP_DR1) != 0x5050) //從指定的后備寄存器中讀出數(shù)據(jù):讀出了與寫入的指定數(shù)據(jù)不相乎 { BKP_DeInit(); //復(fù)位備份區(qū)域 RCC_LSEConfig(RCC_LSE_ON); //設(shè)置外部低速晶振(LSE),使用外設(shè)低速晶振 while (RCC_GetFlagStatus(RCC_FLAG_LSERDY)==RESET&&temp<250) //檢查指定的RCC標(biāo)志位設(shè)置與否,等待低速晶振就緒 { temp++; delay_ms(10); } if(temp>=250)return 1;//初始化時鐘失敗,晶振有問題 RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE); //設(shè)置RTC時鐘(RTCCLK),選擇LSE作為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(32767); //設(shè)置RTC預(yù)分頻的值 RTC_WaitForLastTask(); //等待最近一次對RTC寄存器的寫操作完成 RTC_Set(2015,1,14,17,42,55); //設(shè)置時間 RTC_ExitConfigMode(); //退出配置模式 BKP_WriteBackupRegister(BKP_DR1, 0X5050); //向指定的后備寄存器中寫入用戶程序數(shù)據(jù) } else//系統(tǒng)繼續(xù)計時 { RTC_WaitForSynchro(); //等待最近一次對RTC寄存器的寫操作完成 RTC_ITConfig(RTC_IT_SEC, ENABLE); //使能RTC秒中斷 RTC_WaitForLastTask(); //等待最近一次對RTC寄存器的寫操作完成 } RTC_NVIC_Config();//RCT中斷分組設(shè)置 RTC_Get();//更新時間 return 0; //ok} //RTC時鐘中斷//每秒觸發(fā)一次 //extern u16 tcnt; void RTC_IRQHandler(void){ if (RTC_GetITStatus(RTC_IT_SEC) != RESET)//秒鐘中斷 { RTC_Get();//更新時間 } if(RTC_GetITStatus(RTC_IT_ALR)!= RESET)//鬧鐘中斷 { RTC_ClearITPendingBit(RTC_IT_ALR); //清鬧鐘中斷 RTC_Get(); //更新時間 printf("Alarm Time:%d-%d-%d %d:%d:%dn",calendar.w_year,calendar.w_month,calendar.w_date,calendar.hour,calendar.min,calendar.sec);//輸出鬧鈴時間 } RTC_ClearITPendingBit(RTC_IT_SEC|RTC_IT_OW); //清鬧鐘中斷 RTC_WaitForLastTask(); }//判斷是否是閏年函數(shù)//月份 1 2 3 4 5 6 7 8 9 10 11 12//閏年 31 29 31 30 31 30 31 31 30 31 30 31//非閏年 31 28 31 30 31 30 31 31 30 31 30 31//輸入:年份//輸出:該年份是不是閏年.1,是.0,不是u8 Is_Leap_Year(u16 year){ if(year%4==0) //必須能被4整除 { if(year%100==0) { if(year%400==0)return 1;//如果以00結(jié)尾,還要能被400整除 else return 0; }else return 1; }else return 0; } //設(shè)置時鐘//把輸入的時鐘轉(zhuǎn)換為秒鐘//以1970年1月1日為基準(zhǔn)//1970~2099年為合法年份//返回值:0,成功;其他:錯誤代碼.//月份數(shù)據(jù)表 u8 const table_week[12]={0,3,3,6,1,4,6,2,5,0,3,5}; //月修正數(shù)據(jù)表 //平年的月份日期表const u8 mon_table[12]={31,28,31,30,31,30,31,31,30,31,30,31};u8 RTC_Set(u16 syear,u8 smon,u8 sday,u8 hour,u8 min,u8 sec){ u16 t; u32 seccount=0; if(syear<1970||syear>2099)return 1; for(t=1970;t2099)return 1; for(t=1970;t