0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

【GD32F303紅楓派開(kāi)發(fā)板使用手冊(cè)】第九講 RTC-萬(wàn)年歷實(shí)驗(yàn)

聚沃科技 ? 2024-06-07 09:43 ? 次閱讀
wKgaomZVdiiAfR9BAB3mDFhHnZc972.png

9.1實(shí)驗(yàn)內(nèi)容

通過(guò)本實(shí)驗(yàn)主要學(xué)習(xí)以下內(nèi)容:

  • RTC簡(jiǎn)介
  • RTC復(fù)位
  • RTC實(shí)現(xiàn)萬(wàn)年歷
  • RTC使用注意事項(xiàng)

9.2實(shí)驗(yàn)原理

9.2.1RTC簡(jiǎn)介

RTC(Real Time Clock)——實(shí)時(shí)時(shí)鐘定時(shí)器,可以用作日歷。RTC電路分兩個(gè)電源域部分,其一位于備份域中,該部分包括一個(gè)32位的累加計(jì)數(shù)器、一個(gè)鬧鐘、一個(gè)預(yù)分頻器、一個(gè)分頻器以及RTC時(shí)鐘配置寄存器。備份域這部分電路不會(huì)因?yàn)橄到y(tǒng)復(fù)位或者MCU進(jìn)入低功耗而丟失數(shù)據(jù),所以在系統(tǒng)復(fù)位或MCU從低功耗下喚醒,RTC的設(shè)置和時(shí)間都可以保持不變。另一部分位于VDD電源域中,該部分只包括APB接口以及一組控制寄存器。

關(guān)于RTC的兩個(gè)電源域及分布在兩個(gè)電源域中的內(nèi)容,需要讀者牢記,否則會(huì)由于細(xì)節(jié)沒(méi)處理好導(dǎo)致RTC工作異常。

以下為GD32F303的RTC框圖:

wKgaomZiZNyAGxZyAAE4ZXdvRok560.png

圖中RTC_CNT為計(jì)數(shù)值,每個(gè)SC_CLK時(shí)鐘到來(lái)時(shí),這個(gè)計(jì)數(shù)值增加+1。SC_CLK時(shí)鐘源有三個(gè):LXTAL(外部低速晶振)、IRC40K(內(nèi)部40K晶振)和HXTAL/128,三個(gè)時(shí)鐘源經(jīng)過(guò)RTC_DIV產(chǎn)生SC_CLK。RTC_DIV的配置是通過(guò)RTC_PSC加載所得,RTC_PSC由寄存器RTC預(yù)分頻寄存器高位(RTC_PSCH)和RTC預(yù)分頻寄存器低位(RTC_PSCL)配置,有效位20bit,即RTC的時(shí)鐘分頻器最大值為2的20次方,完全足夠應(yīng)用了。

了解了上面的內(nèi)容,RTC的工作原理就很好理解了。舉個(gè)例子,RTC的時(shí)鐘源選擇LXTAL,頻率為32.768KHz,然后設(shè)置RTC_PSC為32768,那么SC_CLK即為1Hz,也就是說(shuō),每秒鐘RTC_CNT值+1,所以我們只要獲得一段時(shí)間內(nèi)RTC_CNT值增加了多少數(shù),也就知道經(jīng)過(guò)了多少秒。

我們從手冊(cè)中可以看到RTC_CNT由RTC計(jì)數(shù)寄存器高位(RTC_CNTH)和RTC計(jì)數(shù)寄存器低位(RTC_CNTL)設(shè)置,這兩個(gè)寄存器組合起來(lái)的有效位為32bit,即RTC_CNT可以記錄2的32次方,即4,294,967,296個(gè)數(shù),按照每秒增加一次的話,可以記錄136多年,

除了基礎(chǔ)的記時(shí)間的功能,RTC還有一個(gè)鬧鐘功能,RTC運(yùn)行時(shí),當(dāng)RTC_CNT的值增加到和RTC_ALRM(由RTC鬧鐘寄存器高位(RTC_ALRMH)和RTC鬧鐘寄存器低位(RTC_ALRML)設(shè)置)相等時(shí),則會(huì)產(chǎn)生ALRM中斷,當(dāng)然,程序中需要實(shí)現(xiàn)使能ALARM中斷(ALRMIE)。

RTC還有另外兩個(gè)中斷,一個(gè)是秒中斷,另一個(gè)是溢出中斷。秒中斷好理解,即每秒鐘進(jìn)入一個(gè)中斷;溢出中斷則是當(dāng)RTC_CNT溢出時(shí)產(chǎn)生中斷。

9.2.2RTC復(fù)位

這里把RTC的復(fù)位單獨(dú)作為一個(gè)章節(jié)來(lái)說(shuō),是因?yàn)檫@里很容易出錯(cuò)導(dǎo)致想要實(shí)現(xiàn)的功能無(wú)法實(shí)現(xiàn)。

前面說(shuō)過(guò),RTC分為兩個(gè)電源域——備份域和VDD電源域,而一般的復(fù)位比如NRST腳復(fù)位、軟件復(fù)位等只能復(fù)位VDD和VDDA電源域 ,而無(wú)法復(fù)位備份域;備份域復(fù)位需要VBAT掉電或者通過(guò)備份域控制寄存器(RCU_BDCTL)的BKPDRST來(lái)進(jìn)行備份域復(fù)位。

上節(jié)中的RTC框圖中被深色框住的即屬于備份域,這里提到一個(gè)很容易出錯(cuò)的地方,即RTC的時(shí)鐘源選擇。從框圖看到時(shí)鐘源由RTCSRC[1:0]來(lái)設(shè)置,這個(gè)位域?qū)儆趥浞萦蚩刂萍拇嫫鳎≧CU_BDCTL)。

備份域控制寄存器( RCU_BDCTL):

wKgaomZiZOyADpW2AACTeOREhls922.pngwKgZomZiZOyAO7mLAAEvIGqW0OQ864.png

寄存器描述中指出,一旦RTC的時(shí)鐘源選擇后,除了將備份域復(fù)位,否則時(shí)鐘不能被改變。舉個(gè)例子:一個(gè)產(chǎn)品選擇LXTAL作為RTC時(shí)鐘,但可能因?yàn)槟承┰騆XTAL停振了,需要將時(shí)鐘源切換到IRC40K,程序如何實(shí)現(xiàn)呢?沒(méi)錯(cuò),需要復(fù)位備份域(控制BKPDRST位)才能重新選擇時(shí)鐘源,但一旦備份域進(jìn)行了復(fù)位,包括RTC_CNT等數(shù)據(jù)都會(huì)丟失,所以在備份域復(fù)位前需要對(duì)RTC內(nèi)的各個(gè)數(shù)據(jù)進(jìn)行保存處理,待備份域復(fù)位后再重新寫入。

9.2.3RTC實(shí)現(xiàn)萬(wàn)年歷

本實(shí)驗(yàn)用RTC做一個(gè)萬(wàn)年歷,其中還需要考慮到閏年閏月的情況。實(shí)驗(yàn)設(shè)置的基準(zhǔn)時(shí)間是1970年,即當(dāng)RTC_CNT為0時(shí),為1970年,實(shí)驗(yàn)最高可記錄到2106年(1970+136)。

9.3硬件設(shè)計(jì)

本實(shí)驗(yàn)實(shí)現(xiàn)每秒鐘通過(guò)串口打印實(shí)時(shí)時(shí)間,即需要使用到開(kāi)發(fā)板USB串口模塊。另外RTC時(shí)鐘源使用的是外部低速晶振,外部晶振原理圖如下:

wKgZomZiZQ6AaL37AABaIpa9qvg566.png

9.4代碼解析

9.4.1RTC配置

在driver_rtc.c中定義了RTC配置函數(shù)rtc_configuration

C
void rtc_configuration(void)
{
/*使能備份域和PMU時(shí)鐘*/
rcu_periph_clock_enable(RCU_BKPI);
rcu_periph_clock_enable(RCU_PMU);
/*使能備份域?qū)懝δ?/
pmu_backup_write_enable();
/*備份域復(fù)位*/
bkp_deinit();
/*依據(jù)選擇的時(shí)鐘源進(jìn)行時(shí)鐘配置*/
/*使用LXTAL*/
#ifdef RTC_CLOCK_SOURCE_LXTAL
/* 使能 LXTAL */
rcu_osci_on(RCU_LXTAL);
/* 等待LXTAL Ready */
rcu_osci_stab_wait(RCU_LXTAL);
/*選擇LXTAL作為RTC時(shí)鐘*/
rcu_rtc_clock_config(RCU_RTCSRC_LXTAL);
#endif
/*使用IRC40K*/
#ifdef RTC_CLOCK_SOURCE_IRC40K
/* 使能 IRC40K*/
rcu_osci_on(RCU_IRC40K);
/* 等待IRC40K Ready */
rcu_osci_stab_wait(RCU_IRC40K);
/*選擇IRC40K作為RTC時(shí)鐘*/
rcu_rtc_clock_config(RCU_RTCSRC_IRC40K);
#endif

#ifdef RTC_CLOCK_SOURCE_HXTAL_DIV_128
/* 使能LXTAL */
rcu_osci_on(RCU_HXTAL);
/* 等待HXTAL Ready */
rcu_osci_stab_wait(RCU_HXTAL);

/*選擇HXTAL/128作為RTC時(shí)鐘*/
rcu_rtc_clock_config(RCU_RTCSRC_HXTAL_DIV_128);
#endif
/*使能RTC時(shí)鐘*/
rcu_periph_clock_enable(RCU_RTC);
/*等待RTC寄存器同步*/
rtc_register_sync_wait();
/*等待上一次操作完成*/
rtc_lwoff_wait();
/*使能RTC秒中斷*/
rtc_interrupt_enable(RTC_INT_SECOND);
/*等待上一次操作完成*/
rtc_lwoff_wait();
/*依據(jù)選擇的時(shí)鐘源來(lái)設(shè)置分頻,使RTC周期為1s*/
#ifdef RTC_CLOCK_SOURCE_LXTAL
rtc_prescaler_set(32767);
#endif
#ifdef RTC_CLOCK_SOURCE_IRC40K
rtc_prescaler_set(40000);
#endif
#ifdef RTC_CLOCK_SOURCE_HXTAL_DIV_128
rtc_prescaler_set(HXTAL_VALUE/128);
#endif
/*等待上一次操作完成*/
rtc_lwoff_wait();
}

時(shí)鐘源通過(guò)driver_rtc.h中的宏定義來(lái)選擇:

C
#define RTC_CLOCK_SOURCE_LXTAL
//#define RTC_CLOCK_SOURCE_IRC40K
//#define RTC_CLOCK_SOURCE_HXTAL_DIV_128

9.4.2萬(wàn)年歷實(shí)現(xiàn)

在bsp_rtc.c中定義了實(shí)現(xiàn)萬(wàn)年歷的幾個(gè)函數(shù):rtc_time_set、rtc_time_display、is_leap_year等。

rtc_time_set函數(shù)——第一次需要手動(dòng)設(shè)置當(dāng)前時(shí)間:

C
uint32_t rtc_time_set(uint16_t year, uint8_t month, uint8_t day, uint8_t hour, uint8_t minute, uint8_t second)
{
uint16_t t;
uint32_t seccount = 0;
if (bkp_read_data(BKP_DATA_0) != 0xA5A5)
{
rtc_configuration();

if(year < 1970 || year > 2099)
return 1;

for(t = 1970; t < year; t++){
if(is_leap_year(t)){
seccount += 31622400;
}else{
seccount += 31536000;
}
}
month -= 1;

for(t=0; t < month; t++){
seccount += (uint32_t)month_table[t] * 86400;
if(is_leap_year(year) && t==1){
seccount+=86400;
}
}
seccount += (uint32_t)(day-1) * 86400;
seccount += (uint32_t)hour * 3600;
seccount += (uint32_t)minute * 60;
seccount += second;
rtc_counter_set(seccount);
bkp_write_data(BKP_DATA_0, 0xA5A5);
return 0;
}
else
{
if (rcu_flag_get(RCU_FLAG_PORRST) != RESET){
printf("Power On Reset occurred....\r\n");
}else if (rcu_flag_get(RCU_FLAG_EPRST) != RESET){
printf("External Reset occurred....\r\n");
}
rcu_all_reset_flag_clear();
rcu_periph_clock_enable(RCU_PMU);
pmu_backup_write_enable();
rtc_register_sync_wait();
rtc_interrupt_enable(RTC_INT_SECOND);
rtc_lwoff_wait();
return 0;
}
}

該函數(shù)中在第一次上電運(yùn)行時(shí),會(huì)進(jìn)行初始時(shí)間設(shè)置,然后寫入特定數(shù)據(jù)到備份域數(shù)據(jù)寄存器中。當(dāng)發(fā)生系統(tǒng)復(fù)位但Vbat未掉電的情況下,則不會(huì)重新進(jìn)行時(shí)間設(shè)置,但需要重新開(kāi)啟秒時(shí)鐘中斷,因?yàn)镾CIE處于VDD電源域而不在備份域。

rtc_time_display函數(shù)——打印實(shí)時(shí)時(shí)間:

C
void rtc_time_display(uint32_t timevar)
{
static uint16_t daycnt = 0;
uint32_t temp = 0;
uint16_t temp1 = 0;
temp = timevar / 86400;

if(daycnt != temp) {
daycnt = temp;
temp1 = 1970;

while(temp >= 365){
if(is_leap_year(temp1)){
if(temp >= 366)
temp-=366;
else
break;
}else
temp -= 365;
temp1++;
}

calendar.years = temp1;
temp1=0;

while(temp >= 28)
{
if(is_leap_year(calendar.years) && temp1 == 1){
if(temp >= 29)
temp -= 29;
else
break;
}else{
if(temp >= month_table[temp1])
temp -= month_table[temp1];
else
break;
}
temp1++;
}
calendar.months = temp1 + 1;
calendar.days = temp + 1;
}

temp = timevar % 86400;
calendar.hours = temp / 3600;
calendar.minutes = (temp % 3600) / 60;
calendar.seconds = (temp % 3600) % 60;

printf("Time: %0.4d-%0.2d-%0.2d,%0.2d:%0.2d:%0.2d\r\n", calendar.years, calendar.months, calendar.days, calendar.hours, calendar.minutes, calendar.seconds);
}

is_leap_year函數(shù)——判斷當(dāng)前年是否為閏年:

C
uint8_t is_leap_year(uint16_t year)
{
if((year%4 == 0 && year % 100 != 0) || (year % 400 == 0)){
return 1;
}else{
return 0;
}
}

9.4.3main函數(shù)和中斷函數(shù)實(shí)現(xiàn)

以下為main函數(shù)代碼:

C
int main(void)
{
delay_init();//delay函數(shù)初始化
bsp_uart_init(&BOARD_UART);//BOARD_UART串口初始化
nvic_irq_enable(RTC_IRQn,1,0);//打開(kāi)RTC的NVIC
rtc_time_set(year_set,month_set,day_set,hour_set,minute_set,second_set); //設(shè)置當(dāng)前時(shí)間
while (1){
/* 判斷是否到1s了*/
if (timedisplay == 1){
/* 顯示實(shí)時(shí)時(shí)間*/
rtc_time_display(rtc_counter_get());
timedisplay = 0;
}
}
}

本例程main函數(shù)首先進(jìn)行了延時(shí)函數(shù)初始化,再初始化開(kāi)發(fā)板USB串口,開(kāi)啟RTC的NVIC后設(shè)置了當(dāng)前時(shí)間,在while(1)循環(huán)中等待RTC數(shù)據(jù)更新,然后將實(shí)時(shí)時(shí)間打印出來(lái)。

中斷函數(shù)代碼:

C
void RTC_IRQHandler(void)
{
if (rtc_flag_get(RTC_FLAG_SECOND) != RESET){
/* 清除RTC秒中斷標(biāo)志位*/
rtc_flag_clear(RTC_FLAG_SECOND);
timedisplay = 1;
/* 等待上一次操作完成 */
rtc_lwoff_wait();

}
}

9.5實(shí)驗(yàn)結(jié)果

為了驗(yàn)證萬(wàn)年歷是否工作正常,設(shè)定初始時(shí)間為2022年12月31日23時(shí)59分50s,當(dāng)程序開(kāi)始運(yùn)行時(shí),每秒鐘打印時(shí)間,經(jīng)過(guò)10s后,可以看到時(shí)間變?yōu)?023年1月1日0時(shí)0分0秒,說(shuō)明萬(wàn)年歷有效。

wKgZomZiZXaAM8nXAAAfiD_RaEg908.png

教程GD32 MCU方案商聚沃科技原創(chuàng)發(fā)布,了解更多GD32 MCU教程,關(guān)注聚沃科技官網(wǎng)

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 單片機(jī)
    +關(guān)注

    關(guān)注

    6037

    文章

    44558

    瀏覽量

    635354
  • 開(kāi)發(fā)板
    +關(guān)注

    關(guān)注

    25

    文章

    5050

    瀏覽量

    97483
  • 萬(wàn)年歷
    +關(guān)注

    關(guān)注

    3

    文章

    186

    瀏覽量

    23926
  • RTC
    RTC
    +關(guān)注

    關(guān)注

    2

    文章

    538

    瀏覽量

    66542
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    [分享]萬(wàn)年歷算法

    萬(wàn)年歷算法
    發(fā)表于 02-25 17:33

    電子萬(wàn)年歷設(shè)計(jì)

    電子萬(wàn)年歷設(shè)計(jì)
    發(fā)表于 08-20 22:46

    萬(wàn)年歷

    如何在萬(wàn)年歷加上鬧鐘
    發(fā)表于 09-24 15:29

    萬(wàn)年歷

    萬(wàn)年歷
    發(fā)表于 02-16 22:57

    萬(wàn)年歷電路圖

    萬(wàn)年歷電路圖萬(wàn)年歷電路圖
    發(fā)表于 08-05 14:59

    萬(wàn)年歷

    萬(wàn)年歷萬(wàn)年歷萬(wàn)年歷萬(wàn)年歷
    發(fā)表于 03-20 21:08

    萬(wàn)年歷 仿真

    游戲 萬(wàn)年歷
    發(fā)表于 07-08 11:19

    STM32RTC萬(wàn)年歷制作本設(shè)計(jì)

    STM32RTC萬(wàn)年歷制作本設(shè)計(jì)是用STM32F103c8t6制作的簡(jiǎn)單萬(wàn)年歷首先是配置RTC時(shí)鐘然后是配置時(shí)鐘,年月日等的處理頭文件本設(shè)計(jì)
    發(fā)表于 08-12 06:26

    如何去實(shí)現(xiàn)一種基于STM32F103的可調(diào)時(shí)鐘萬(wàn)年歷

    如何用STM32F103內(nèi)部RTC實(shí)現(xiàn)可調(diào)時(shí)鐘萬(wàn)年歷?如何去實(shí)現(xiàn)一種基于STM32F103的可調(diào)時(shí)鐘萬(wàn)年歷呢?
    發(fā)表于 11-22 06:44

    星空GD32F303開(kāi)發(fā)板的相關(guān)資料下載

    一、開(kāi)發(fā)板介紹星空GD開(kāi)發(fā)板是由旗點(diǎn)科技推出的一款GD32開(kāi)發(fā)板,板載
    發(fā)表于 12-10 08:27

    萬(wàn)年歷

    電子萬(wàn)年歷,可以運(yùn)行的哦,單片機(jī)相關(guān)知識(shí)。
    發(fā)表于 05-17 11:09 ?16次下載

    萬(wàn)年歷

    基于C51單片機(jī)的萬(wàn)年歷
    發(fā)表于 12-17 20:48 ?119次下載

    ds1302萬(wàn)年歷protues仿真 51單片機(jī)萬(wàn)年歷仿真 實(shí)時(shí)

    ds1302萬(wàn)年歷protues仿真 51單片機(jī)萬(wàn)年歷仿真 實(shí)時(shí)時(shí)鐘仿真程序
    發(fā)表于 01-14 22:32 ?123次下載

    萬(wàn)年歷protues仿真 實(shí)時(shí)時(shí)鐘仿真 12864萬(wàn)年歷仿真 5

    萬(wàn)年歷protues仿真 實(shí)時(shí)時(shí)鐘仿真 12864萬(wàn)年歷仿真 51萬(wàn)年歷設(shè)計(jì)
    發(fā)表于 01-14 22:32 ?175次下載

    GD32F303紅楓開(kāi)發(fā)板使用手冊(cè)】第二十 SPI-SPI NAND FLASH讀寫實(shí)驗(yàn)

    通過(guò)本實(shí)驗(yàn)主要學(xué)習(xí)以下內(nèi)容: ?SPI通信協(xié)議,參考19.2.1東方紅開(kāi)發(fā)板使用手冊(cè) ?GD32F303 SPI操作方式,參考19.2.2東方紅開(kāi)
    的頭像 發(fā)表于 06-20 09:50 ?906次閱讀
    【<b class='flag-5'>GD32F303</b><b class='flag-5'>紅楓</b><b class='flag-5'>派</b><b class='flag-5'>開(kāi)發(fā)板</b><b class='flag-5'>使用手冊(cè)</b>】第二十<b class='flag-5'>講</b> SPI-SPI NAND FLASH讀寫<b class='flag-5'>實(shí)驗(yàn)</b>