基於STM32 RTC的萬年曆

2021-01-11 電子產品世界
例子基本是照抄官方的 萬年曆算法也沒深入研究 主要是大賽 都要求會用DS1302 若我用STM32來做 肯定不用那個片子了。

這個用的是 LSE (片外低速時鐘)配合 掉電寄存器來確定是否配置時鐘。

本文引用地址:http://www.eepw.com.cn/article/201612/325211.htm

注釋很全 話不多說了。

u8 TimeDisplay;
int main(void)
{
SystemInit();

stm32_Init ();//GPIO PA8 Init
USART_Configuration();//USART2 9600-8-N-1
NVIC_Configuration();//Enable the RTC Interrupt
RTC_Configuration();//RTC的啟動

start_rct();//檢測是否配置時鐘
Time_Show();//不斷地時鐘串口輸出

}


void LEDToggle(void)
{
GPIOA->ODR=GPIOA->ODR^GPIO_Pin_8 ;
}

RTC.C///////////////////////////////////////////////////////////////////////////////

#include "stm32f10x.h"
#include //用到printf函數的串口的輸出函數 注意勾選MicroLIB

u32 Time_Regulate(void);
void Time_Adjust(void);
void Time_Show(void);
void Time_Display(u32 TimeVar);
u32 USART_Scanf(u32 value);
extern u8 TimeDisplay;

void RTC_Configuration(void)
{
/* Enable PWR and BKP clocks */
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE);

/* Allow access to BKP Domain */
PWR_BackupAccessCmd(ENABLE);

/* Reset Backup Domain */
//BKP_DeInit(); //記錄0XA5A5 來確定是否重置時間

/* Enable LSE */
RCC_LSEConfig(RCC_LSE_ON);
/* Wait till LSE is ready */
while (RCC_GetFlagStatus(RCC_FLAG_LSERDY) == RESET)
{}

/* Select LSE as RTC Clock Source */
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);

/* Enable RTC Clock */
RCC_RTCCLKCmd(ENABLE);

/* Wait for RTC registers synchronization */
RTC_WaitForSynchro();

/* Wait until last write operation on RTC registers has finished */
RTC_WaitForLastTask();

/* Enable the RTC Second */
RTC_ITConfig(RTC_IT_SEC, ENABLE);

/* Wait until last write operation on RTC registers has finished */
RTC_WaitForLastTask();

/* Set RTC prescaler: set RTC period to 1sec */
RTC_SetPrescaler(32767); /* RTC period = RTCCLK/RTC_PR = (32.768 KHz)/(32767+1) */

/* Wait until last write operation on RTC registers has finished */
RTC_WaitForLastTask();
}

/*******************************************************************************
* Function Name : Time_Regulate
* Description : Returns the time entered by user, using Hyperterminal.
* Input : None
* Output : None
* Return : Current time RTC counter value
*******************************************************************************/
//u32 Month_Days[13] = {0,31,28,31,30, 31, 30, 31, 31, 30, 31, 30, 31};

u32 Month_Days_Accu_C[13] = {0,31,59,90,120,151,181,212,243,273,304,334,365};
u32 Month_Days_Accu_L[13] = {0,31,60,91,121,152,182,213,244,274,305,335,366};
#define SecsPerDay (3600*24)

u32 Time_Regulate(void)
{
#if 1
u32 Tmp_Year=0xFFFF, Tmp_Month=0xFF, Tmp_Date=0xFF;
u32 LeapY, ComY, TotSeconds, TotDays;
#endif
u32 Tmp_HH = 0xFF, Tmp_MM = 0xFF, Tmp_SS = 0xFF;

printf("\r\n==============Time Settings=====================================");

#if 1
printf("\r\n Please Set Year");
while(Tmp_Year == 0xFFFF)
{
/*32-bit counter at Second Unit--> 4*1024*1024(s) --> 49710(day) --> 136(year)*/
Tmp_Year = USART_Scanf(2136);
}
printf(": %d", Tmp_Year);

printf("\r\n Please Set Month");
while(Tmp_Month == 0xFF)
{
Tmp_Month = USART_Scanf(12);
}
printf(": %d", Tmp_Month);

printf("\r\n Please Set Date");
while(Tmp_Date == 0xFF)
{
Tmp_Date = USART_Scanf(31);
}
printf(": %d", Tmp_Date);
#endif

printf("\r\n Please Set Hours");
while(Tmp_HH == 0xFF)
{
Tmp_HH = USART_Scanf(23);
}
printf(": %d", Tmp_HH);

printf("\r\n Please Set Minutes");
while(Tmp_MM == 0xFF)
{
Tmp_MM = USART_Scanf(59);
}
printf(": %d", Tmp_MM);;

printf("\r\n Please Set Seconds");
while(Tmp_SS == 0xFF)
{
Tmp_SS = USART_Scanf(59);
}
printf(": %d", Tmp_SS);

#if 1
{
/* change Year-Month-Data-Hour-Minute-Seconds into X(Second) to set RTC->CNTR */
if(Tmp_Year==2000)
LeapY = 0;
else
LeapY = (Tmp_Year - 2000 -1)/4 +1;

ComY = (Tmp_Year - 2000)-(LeapY);

if (Tmp_Year%4)
//common year
TotDays = LeapY*366 + ComY*365 + Month_Days_Accu_C[Tmp_Month-1] + (Tmp_Date-1);
else
//leap year
TotDays = LeapY*366 + ComY*365 + Month_Days_Accu_L[Tmp_Month-1] + (Tmp_Date-1);

TotSeconds = TotDays*SecsPerDay + (Tmp_HH*3600 + Tmp_MM*60 + Tmp_SS);
}
#endif

/* Return the value to store in RTC counter register */
//return((Tmp_HH*3600 + Tmp_MM*60 + Tmp_SS));
return TotSeconds;
}

/*******************************************************************************
* Function Name : Time_Adjust
* Description : Adjusts time.
* Input : None
* Output : None
* Return : None
*******************************************************************************/
void Time_Adjust(void)
{
/* Wait until last write operation on RTC registers has finished */
RTC_WaitForLastTask();
/* Change the current time */
RTC_SetCounter(Time_Regulate());
/* Wait until last write operation on RTC registers has finished */
RTC_WaitForLastTask();
}


/*******************************************************************************
* Function Name : Time_Display
* Description : Displays the current time.
* Input : - TimeVar: RTC counter value.
* Output : None
* Return : None
*******************************************************************************/
#define SecsPerComYear 3153600//(365*3600*24)
#define SecsPerLeapYear 31622400//(366*3600*24)
#define SecsPerFourYear 126230400//((365*3600*24)*3+(366*3600*24))
#define SecsPerDay (3600*24)

s32 Year_Secs_Accu[5]={0,
31622400,
63158400,
94694400,
126230400};

s32 Month_Secs_Accu_C[13] = { 0,
2678400,
5097600,
7776000,
10368000,
13046400,
15638400,
18316800,
20995200,
23587200,
26265600,
28857600,
31536000};
s32 Month_Secs_Accu_L[13] = {0,
2678400,
5184000,
7862400,
10454400,
13132800,
15724800,
18403200,
21081600,
23673600,
26352000,
28944000,
31622400};

void Time_Display(u32 TimeVar)
{
#if 1
u32 TY = 0, TM = 1, TD = 0;
s32 Num4Y,NumY, OffSec, Off4Y = 0;
u32 i;
s32 NumDay; //OffDay;
#endif
u32 THH = 0, TMM = 0, TSS = 0;
#if 0
/* Compute hours */
THH = TimeVar/3600;
/* Compute minutes */
TMM = (TimeVar % 3600)/60;
/* Compute seconds */
TSS = (TimeVar % 3600)% 60;
#endif
#if 1
{
Num4Y = TimeVar/SecsPerFourYear;
OffSec = TimeVar%SecsPerFourYear;

i=1;
while(OffSec > Year_Secs_Accu[i++])
Off4Y++;

/* Numer of Complete Year */
NumY = Num4Y*4 + Off4Y;
/* 2000,2001,...~2000+NumY-1 complete year before, so this year is 2000+NumY*/
TY = 2000+NumY;

OffSec = OffSec - Year_Secs_Accu[i-2];

/* Month (TBD with OffSec)*/
i=0;
if(TY%4)
{// common year
while(OffSec > Month_Secs_Accu_C[i++]);
TM = i-1;
OffSec = OffSec - Month_Secs_Accu_C[i-2];
}
else
{// leap year
while(OffSec > Month_Secs_Accu_L[i++]);
TM = i-1;
OffSec = OffSec - Month_Secs_Accu_L[i-2];
}

/* Date (TBD with OffSec) */
NumDay = OffSec/SecsPerDay;
OffSec = OffSec%SecsPerDay;
TD = NumDay+1;

/* Compute hours */
THH = OffSec/3600;
/* Compute minutes */
TMM = (OffSec % 3600)/60;
/* Compute seconds */
TSS = (OffSec % 3600)% 60;
}
#endif

printf("Date: %0.4d-%0.2d-%0.2d Time: %0.2d:%0.2d:%0.2d\r",TY, TM, TD,THH, TMM, TSS);
}

/*******************************************************************************
* Function Name : Time_Show
* Description : Shows the current time (HH:MM:SS) on the Hyperterminal.
* Input : None
* Output : None
* Return : None
******************************************************************************/
void Time_Show(void)
{
printf("\n\r");

/* Infinite loop */
while (1)
{
/* If 1s has paased */
if(TimeDisplay == 1)
{
/* Display current time */
Time_Display(RTC_GetCounter());
TimeDisplay = 0;
}
}
}

/*******************************************************************************
* Function Name : USART_Scanf
* Description : Gets numeric values from the hyperterminal.
* Input : None
* Output : None
* Return : None
*******************************************************************************/
u32 USART_Scanf(u32 value)
{
u32 index = 0;
u32 tmp[4] = {0, 0};
u32 Num;

if (value==2136)
Num = 4;
else
Num = 2;

while(index < Num)
{
/* Loop until RXNE = 1 */
while(USART_GetFlagStatus(USART2, USART_FLAG_RXNE) == RESET)
{
}
tmp[index++] = (USART_ReceiveData(USART2));
if((tmp[index - 1] < 0x30) || (tmp[index - 1] > 0x39))
{
printf("\n\rPlease enter valid number between 0 and 9");
index--;
}
}
/* Calculate the Corresponding value */
if (value!=2136)
index = ((tmp[0] - 0x30) * 10) + (tmp[1] - 0x30);
else
index = ((tmp[0] - 0x30) * 1000) + ((tmp[1] - 0x30) * 100) + ((tmp[2] - 0x30) * 10) + (tmp[3] - 0x30);
/* Checks */
if(index > value)
{
printf("\n\rPlease enter valid number between 0 and %d", value);
return 0xFF;
}
return index;
}

void start_rct(void)
{
if(BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5)
{
/* Backup data register value is not correct or not yet programmed (when
the first time the program is executed) */

printf("\r\n\n RTC not yet configured....");

/* RTC Configuration */
RTC_Configuration();

printf("\r\n RTC configured....");

/* Adjust time by values entred by the user on the hyperterminal */
Time_Adjust();

BKP_WriteBackupRegister(BKP_DR1, 0xA5A5);
}
else
{
/* Check if the Power On Reset flag is set */
if(RCC_GetFlagStatus(RCC_FLAG_PORRST) != RESET)
{
printf("\r\n\n Power On Reset occurred....");
}
/* Check if the Pin Reset flag is set */
else if(RCC_GetFlagStatus(RCC_FLAG_PINRST) != RESET)
{
printf("\r\n\n External Reset occurred....");
}

printf("\r\n No need to configure RTC....");
/*一下都是可以省略的 RTC_Configuration 已有啟用 RTC_IT_SEC */
/* Wait for RTC registers synchronization */
RTC_WaitForSynchro();

/* Enable the RTC Second */
RTC_ITConfig(RTC_IT_SEC, ENABLE);
/* Wait until last write operation on RTC registers has finished */
RTC_WaitForLastTask();
}

}

其中這個函數決定了 printf 函數的輸出目標 一定要有的。
int fputc(int ch)
{
USART_SendData(USART2, (u8) ch);
while(USART_GetFlagStatus(USART2, USART_FLAG_TXE) == RESET);
return ch;
}

秒中斷;

/******************************************************************************/
/* STM32F10x Peripherals Interrupt Handlers */
/******************************************************************************/

/**
* @brief This function handles RTC global interrupt request.
* @param None
* @retval None
*/
extern u8 TimeDisplay;
void LEDToggle(void);
void RTC_IRQHandler(void)
{
if (RTC_GetITStatus(RTC_IT_SEC) != RESET)
{
/* Clear the RTC Second interrupt */
RTC_ClearITPendingBit(RTC_IT_SEC);

/* Toggle LED1 閃燈*/
LEDToggle();

/* Enable time update */
TimeDisplay = 1;
}
}

不要使用串口助手 推薦使用超級終端。

相關焦點

  • 13個基於STM32的經典項目設計實例,全套資料~
    尤其對於新手,是一個學習stm32單片機的&ldquo;活生生&rdquo;的範例。本文引用地址:http://www.eepw.com.cn/article/201706/346863.htm1.
  • 一文搞懂Cortex-A9 RTC
    void rtc_init(void) {  RTCCON = 1;//使能RTC控制寫功能  RTC.BCDYEAR = 0x20;// 2020年11月11日, 15:24:50.以BCD碼格式寫入  RTC.BCDMON = 0x11;  RTC.BCDDAY = 0x11;  RTC.BCDHOUR = 0x15;  RTC.BCDMIN = 0x24;  RTC.BCDSEC
  • 13.一文搞懂Cortex-A9 RTC
    void rtc_init(void){ RTCCON = 1;//使能RTC控制寫功能 RTC.BCDYEAR = 0x20;// 2020年11月11日, 15:24:50.以BCD碼格式寫入 RTC.BCDMON = 0x11; RTC.BCDDAY = 0x11; RTC.BCDHOUR = 0x15; RTC.BCDMIN
  • 萬年曆查詢
    萬年曆農曆查詢說明萬年曆是記錄具體陽曆和陰曆的年曆,可以快速查詢更多年的陰曆與陽曆對照表及相關節日等信息。萬年曆農曆查詢 在線實用萬年曆查詢表關於萬年曆我國古代傳說中最古老的一部太陽曆,為紀念曆法編撰者萬年功績,便將這部曆法命名為「萬年曆」。
  • 個性化訂閱,中華萬年曆頭條日曆更懂你
    作為市面上一款最常見也是最受歡迎的日曆應用,中華萬年曆App到底有什麼樣的「迷之魅力」,能獲得2億多用戶的支持呢?自從問世以來,中華萬年曆日曆便在「時間」上做足了功課。比如中華萬年曆日曆的頭條資訊功能,賦予工具內容屬性,也賦予用戶更具體的時間服務。
  • 萬年曆老黃曆免費下載
    下載萬年曆老黃曆免費下載後,你就會被它深深吸引住。萬年曆老黃曆免費下載官方介紹:頭疼找不到有趣的軟體?下載萬年曆老黃曆免費下載吧,享受別樣的樂趣。
  • 萬能手錶調教工具——萬年曆
    首先,萬年曆調校一點也不複雜。對於鐘錶愛好者來說,學會調萬年曆僅僅相當於本科畢業。萬年曆與年曆的區別就在於:萬年曆不僅可以自動區分大、小月的天數,還能區分閏年或平年時,2月的天數。   將時間調至安全時間段之後,萬年曆的各項功能便可以隨意調校了,無需擔心會損壞萬年曆,但若想準確便捷的調好萬年曆,推薦您按照【日期月份/年份星期】的順序調校。   我們還是以今天的日期:2016年5月19日(周四)早上9點為例。
  • 永恆時間守望者 萬年曆腕錶
    新年伊始,腕錶的萬年曆將會自動邁入新的一年,滴答滴答地運行下去,從月份顯示到每一天的精密計算,對於不同的人群而言,萬年曆腕錶有著截然不同的特質,或華麗,或繁複,或簡潔,陪著人們繼續新的時光旅程。而幾代的鐘表大師們也致力於打造一條通往永恆的時間隧道,它的名字便是萬年曆腕錶。1615年,日內瓦的達布創造出歷史上第一隻具有萬年曆功能的鐘。自此數個世紀中,萬年曆腕錶成了極複雜計時的標誌,除了具備日期、星期、月份、月相、閏年指示等功能外,它最大的特性在於超過百年以上無須調校日期的「超能力」。但,令表迷們困惑的是,萬年曆腕錶真的是「萬年」不變的永恆之表嗎?這一次,便為大家揭開它的面紗,一探究竟。
  • iPhone日期顯示補充:"萬年曆(黃曆)"!
    iPhone日期顯示補充:"萬年曆(黃曆)"!2010年10月14日 14:27作者:劉志鵬編輯:劉志鵬文章出處:泡泡網原創     泡泡網iPhone頻道10月14日 「萬年曆
  • avr單片機和stm32區別與優缺點分析
    摘要:avr單片機和stm32單片機是目前使用較廣泛的單片機,那麼avr單片機和stm32單片機有什麼區別呢?有什麼優劣勢呢?   二、stm32單片機介紹   由ST廠商推出的stm32系列單片機,行業的朋友都知道,這是一款性價比超高的系列單片機,應該沒有之一,功能及其強大。
  • 【輕鬆玩藍牙】序列之13:RTC和比較器
    2.主要包括下面3個文件:main.c、rtc.c和rtc.h。
  • 基於STM32的OV7670攝像頭總結
    的工作原理:      圖1:不帶FIFO  圖2:帶FIFO  下面就講解這兩種方式的適用範圍:  不帶FIFO:這種方法最簡單,最直接,但是最不好實現的方法,原因是多數的CMOS晶片(如OV7670)的時鐘速度可以高達24M,一般單片機的IO口速度根本達不到(stm32
  • 日月壽星是誰,【萬年曆】又是因何而命名
    萬年曆是中國古代傳說中最古老的一部太陽曆。
  • 一文解析stm32產生spwm原理及程序
    打開APP 一文解析stm32產生spwm原理及程序 發表於 2018-05-18 09:11:49 本文主要詳解stm32產生spwm原理及程序,首先來了解一下生成SPWM波的基理是什麼,具體得跟隨小編一起來了解一下。
  • 基於STM32的多功能數字鐘(中文版)
    /**************************************************************************************程序功能:基於STM32的多功能數字鐘(中文版)
  • 基於線性倒立擺跟蹤的行走控制器開原始碼
    >sch-core: collision detectionTasks: inverse kinematicsmc_rbdyn_urdf: robot model loadercopra: linear model predictive control以下源碼項未公開發布,但可應Pierre Gergondet的要求提供:mc_rtc
  • stm32的微秒延時程序
    我的stm32系統時鐘為72MHZ,用軟體延時方法實現微秒級的延時,既空循環:延時函數為:本文引用地址:http://www.eepw.com.cn
  • 使用STM32 的DSP庫進行FFT變換
    * 使用三角函數生成採樣點,供FFT計算* 進行FFT測試時,按下面順序調用函數即可:* dsp_asm_init();* dsp_asm_test();*/#include "stm32f10x.h"#include "dsp_asm.h"#include "stm32_dsp.h"#include "table_fft.h"
  • Stm32的TFT LCD顯示器控制學習筆記
    學習stm32,TFT LCD顯示屏控制是很重要的一章,本人在初步學習STM32遇到了很多困難,所以把學習中積累的部分感覺重要的知識點羅列出來