STM32L中的系統(tǒng)時間——硬件RTC的使用
鼓搗了將近一天。。。因為之前用過STM32F103芯片,而這次是STM32L151,這個L系列和F系列的RTC使用方式不同。廢話少說,上代碼:
RTC初始化:
//硬件RTC時鐘初始化
voidRTC_Configuration()
{
/*AllowaccesstotheRTC*/
PWR_RTCAccessCmd(ENABLE);
/*ResetRTCBackupDomain*/
RCC_RTCResetCmd(ENABLE);
RCC_RTCResetCmd(DISABLE);
/*LSEEnable*/
RCC_LSEConfig(RCC_LSE_ON);
/*WaituntilLSEisready*/
while(RCC_GetFlagStatus(RCC_FLAG_LSERDY)==RESET);
/*RTCClockSourceSelection*/
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);
/*EnabletheRTC*/
RCC_RTCCLKCmd(ENABLE);
}
讀寫RTC時間:
//設(shè)置本地硬件RTC時間
voidSetRTC(longhhmmss,longddmmyy)
{
RTC_TimeTypeDefRTC_TimeStruct;
RTC_DateTypeDefRTC_DateStruct;
RTC_TimeStruct.RTC_Hours=hhmmss/10000;
RTC_TimeStruct.RTC_Minutes=hhmmss/100%100;
RTC_TimeStruct.RTC_Seconds=hhmmss%100;
RTC_TimeStruct.RTC_H12=0;//程序中不使用上午/下午
RTC_SetTime(RTC_Format_BIN,&RTC_TimeStruct);
RTC_DateStruct.RTC_Year=ddmmyy%100;
RTC_DateStruct.RTC_Date=ddmmyy/10000;
RTC_DateStruct.RTC_Month=ddmmyy/100%100;
RTC_DateStruct.RTC_WeekDay=1;//程序中不使用星期
RTC_SetDate(RTC_Format_BIN,&RTC_DateStruct);
}
//獲取本地硬件RTC時間
voidGetRTC(long*hhmmss,long*yymmdd)
{
RTC_TimeTypeDeftime;
RTC_DateTypeDefdate;
RTC_GetTime(RTC_Format_BIN,&time);
RTC_GetDate(RTC_Format_BIN,&date);
*hhmmss=time.RTC_Hours*10000+time.RTC_Minutes*100+time.RTC_Seconds;
*yymmdd=date.RTC_Year*10000+date.RTC_Month*100+date.RTC_Date;
}
主函數(shù):
charcoord[64];//當前經(jīng)緯度
longdate,time;//當前UTC時間
intmain(void)
{
Init();
SetRTC(115921,140203);
while(1){
GetRTC(&time,&date);
sprintf(coord,"%06d,%06d",date,time);
delay_s(1);
}
}
另外我使用的是8Mhz(8000000Hz)的外部晶振作為系統(tǒng)時間。因此把系統(tǒng)時鐘RCC的初始化也貼上來,有需要的讀者自行取用:)
/**********************************************************************
*名稱:RCC_Configuration()
*功能:配置時鐘
*入口參數(shù):
*出口參數(shù):
-----------------------------------------------------------------------
*說明:使用庫函數(shù)
***********************************************************************/
voidRCC_Configuration(void)
{
ErrorStatusHSEStartUpStatus;
RCC_DeInit();
//使能外部晶振
RCC_HSEConfig(RCC_HSE_ON);
//等待外部晶振穩(wěn)定
HSEStartUpStatus=RCC_WaitForHSEStartUp();
//如果外部晶振啟動成功,則進行下一步操作
if(HSEStartUpStatus==SUCCESS)
{
//設(shè)置HCLK(AHB時鐘)=SYSCLK
RCC_HCLKConfig(RCC_SYSCLK_Div1);
//PCLK2(APB2)=HCLK
RCC_PCLK2Config(RCC_HCLK_Div1);
//PCLK1(APB1)=HCLK/2
RCC_PCLK1Config(RCC_HCLK_Div2);
FLASH_SetLatency(FLASH_Latency_1);
FLASH_PrefetchBufferCmd(ENABLE);
//等待PLL穩(wěn)定
//while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY)==RESET);
while(RCC_GetFlagStatus(RCC_FLAG_HSERDY)==RESET);
//系統(tǒng)時鐘SYSCLK來
//RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
RCC_SYSCLKConfig(RCC_SYSCLKSource_HSE);
//切換時鐘后等待系統(tǒng)時鐘穩(wěn)定
//while(RCC_GetSYSCLKSource()!=0x0C);
while(RCC_GetSYSCLKSource()!=0x08);
}
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA,ENABLE);
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOB,ENABLE);
}
運行效果就是,使用Keil uVision4.70的debug模式,可以看到Watch窗口中的coord變量實時顯示出當前的時間。
之前我還搜到將SysTick作為系統(tǒng)時間的方法,其實就是借助SysTick中斷來自行計時,雖然也能滿足項目需求(比較簡單),但后來經(jīng)同事提醒才發(fā)覺STM32L里面有內(nèi)置的硬件RTC時鐘,包含了年月日時分秒,才開始了這一天的鼓搗。
今天看了半天stm32l1xx_rtc.h和.c文件,這是在STM32L-Discovery工程里面內(nèi)置的,包含了對當前芯片的RTC功能的API接口定義和實現(xiàn),還有充足的注釋,個人感覺這個比文檔更清晰。不過這沒能解決問題。
最后還是用谷歌搜STM32L和RTC搜到的這個網(wǎng)頁,才是解決問題的關(guān)鍵。里面包含了STM32L中RTC初始化的代碼,測試可用,算是突破。之后就順利了。