|
最近在用STM32 RTC時(shí)鐘部分,要實(shí)現(xiàn)的功能是通過(guò)RTC模塊實(shí)現(xiàn)日歷功能,首先配置RTC模塊
int RTC_Configuration(void)
{
u32count=0x200000;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP,ENABLE);
PWR_BackupAccessCmd(ENABLE);
BKP_DeInit();
RCC_LSEConfig(RCC_LSE_ON);
while ((RCC_GetFlagStatus(RCC_FLAG_LSERDY) == RESET)&& (--count) );
if ( count == 0)//防止沒(méi)有外部
{
return -1;
}
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);
RCC_RTCCLKCmd(ENABLE);
RTC_WaitForSynchro();
RTC_WaitForLastTask();
RTC_SetPrescaler(32767);
RTC_WaitForLastTask();
return 0;
}
RTC_Configuration函數(shù)主要是初始化RTC硬件部分,打開(kāi)對(duì)應(yīng)模塊的時(shí)鐘源,設(shè)置時(shí)鐘頻率為1HZ及1S中斷一次,系統(tǒng)已這個(gè)時(shí)鐘源為基礎(chǔ)進(jìn)行處理!
讀取RTC時(shí)鐘寄存器:時(shí)鐘寄存器是32位格式分為高16和低16
uint32_t RTC_GetCounter(void)
{
uint16_t tmp =0;
tmp =RTC->CNTL;
return(((uint32_t)RTC->CNTH<< 16 ) | tmp) ;
}
設(shè)置秒寄存器
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR |RCC_APB1Periph_BKP, ENABLE);
PWR_BackupAccessCmd(ENABLE);
RTC_WaitForLastTask();
RTC_SetCounter(*time);
RTC_WaitForLastTask();
BKP_WriteBackupRegister(BKP_DR1, 0xA5A5);
其中RTC_SetCounter是設(shè)置秒累加器的
void RTC_SetCounter(uint32_t CounterValue)
{
RTC_EnterConfigMode();
RTC->CNTH = CounterValue>> 16;
RTC->CNTL = (CounterValue& RTC_LSB_MASK);
RTC_ExitConfigMode();
}
也是通過(guò)設(shè)置RTC高16位RTC->CNTH和RTC低16位寄存器實(shí)現(xiàn)的,可見(jiàn)RTC部分也是通過(guò)控制RTC->CNTL和 RTC->CNTH 寄存器實(shí)現(xiàn)的,寫(xiě)寄存器就是設(shè)置,然后讀寄存器就是讀秒,它是按照設(shè)置的1HZ及1s的頻率不斷的累加,掉電也是可以工作的,因?yàn)閂BAT管腳外接3V紐扣電池作為后備電池,
設(shè)置日期的函數(shù)
void set_date(rt_uint32_t year, rt_uint32_t month, rt_uint32_tday)
{
time_t now;
struct tm* ti;
rt_device_tdevice;
ti = RT_NULL;
time(&now);
ti =localtime(&now);
if (ti != RT_NULL)
{
ti->tm_year = year -1900;//UNIX下時(shí)間處理
ti->tm_mon = month - 1;
ti->tm_mday = day;
}
now = mktime(ti);
device =rt_device_find("rtc");
if (device !=RT_NULL)
{
rt_rtc_control(device,RT_DEVICE_CTRL_RTC_SET_TIME, &now);
}
}
FINSH_FUNCTION_EXPORT(set_date, set date. e.g:set_date(2010,2,28))
void set_time(rt_uint32_t hour, rt_uint32_t minute,rt_uint32_t second)
{
time_t now;
struct tm* ti;
rt_device_tdevice;
ti = RT_NULL;
time(&now);
ti =localtime(&now);
if (ti != RT_NULL)
{
ti->tm_hour = hour;
ti->tm_min = minute;
ti->tm_sec = second;
}
now = mktime(ti);
device =rt_device_find("rtc");
if (device !=RT_NULL)
{
rt_rtc_control(device,RT_DEVICE_CTRL_RTC_SET_TIME, &now);
}
}
顯示時(shí)間和日期函數(shù)
void list_date(void)
{
time_t now;
struct tm *pt;
time(&now);//讀取當(dāng)前秒數(shù)
pt =localtime(&now); //把當(dāng)前時(shí)間戳轉(zhuǎn)換成對(duì)應(yīng)的日期
if(pt != NULL)
{
rt_kprintf(" Y = %d M = %d D = %d\n",(1900+pt->tm_year),(1+pt->tm_mon),pt->tm_mday);
rt_kprintf(" %d \n", pt->tm_wday);
rt_kprintf(" %d:%d:%d \n", pt->tm_hour,pt->tm_min, pt->tm_sec);
}
rt_kprintf("%d\n",now);//把秒轉(zhuǎn)換成對(duì)應(yīng)的時(shí)間日期
rt_kprintf("%s\n",ctime(&now));//把秒轉(zhuǎn)換成對(duì)應(yīng)的時(shí)間日期
}
上面還有一個(gè)需要注意的地方
gmtime在MDK下面是無(wú)法正常使用的,嗲有g(shù)mtime函數(shù)返回的一直是NULL,后來(lái)找到庫(kù)函數(shù)使用說(shuō)明:
gmtime語(yǔ)法:
#include <time.h> struct tm *gmtime( const time_t *time ); | 功能:函數(shù)返回給定的統(tǒng)一世界時(shí)間(通常是格林威治時(shí)間),如果系統(tǒng)不支持統(tǒng)一世界時(shí)間系統(tǒng)返回NULL。警告!
相關(guān)主題:
localtime(), time(), andasctime().
gmtime()函數(shù)是分時(shí)區(qū)的世界時(shí)間,而MDK環(huán)境下面是不支持的,所以使用gmtime()函數(shù)無(wú)法得到想要的結(jié)果,解決方法很簡(jiǎn)單直接localtime()即可
#include <time.h> struct tm *localtime( const time_t *time ); | 功能:函數(shù)返回本地日歷時(shí)間。警告!
相關(guān)主題:
gmtime(), time(), andasctime().
localtime函數(shù)此處不再去介紹它是標(biāo)準(zhǔn)的POSIX接口函數(shù),所以可以去查POSIX時(shí)間相關(guān)的API,此處不再詳細(xì)介紹,總體來(lái)說(shuō)RTC使用還是挺簡(jiǎn)單的,希望此篇文章能夠給像我一樣使用RTC的同學(xué)一些幫助,
|
|