(2)創建rtc.c文件并輸入以下代碼。
#include "rtc.h"
/***************************************************
Name :RTC_Init
Fuction :RTC初始化
Parameter :None
Return :None
***************************************************/
void RTC_Init()
{
if( BKP->DR1!=0x5050 )
{
RCC->APB1ENR |= 1<<28 ; //使能PWR時鐘
RCC->APB1ENR |= 1<<27 ; //使能BKP時鐘,RTC校準在BKP相關寄存器中
PWR->CR |= 1<<8 ; //取消BKP相關寄存器寫保護
RCC->BDCR |= 1<<16 ; //備份區域軟復位
RCC->BDCR &= ~( 1<<16 ) ; //備份區域軟復位結束
RCC->BDCR |= 1<<0 ; //開啟外部低速振蕩器
while( ( RCC->BDCR&0x02 )!=0x02 ) ; //等待外部時鐘就緒
RCC->BDCR |= 1<<8 ; //LSI作為RTC時鐘
RCC->BDCR |= 1<<15 ; //RTC時鐘使能
while( !( RTC->CRL&( 1<<5 ) ) ) ; //等待RTC寄存器最后一次操作完成
while( !( RTC->CRL&( 1<<3 ) ) ) ; //等待RTC寄存器同步完成
RTC->CRH &= ~( 7<<0 ) ; //不允許中斷,CRH寄存器低三位有效
while( !( RTC->CRL&( 1<<5 ) ) ) ; //等待RTC寄存器最后一次操作完成
RTC->CRL |= 1<<4 ; //進入配置模式
RTC->PRLH = 0 ;
RTC->PRLL = 32767 ; //設定分頻值
RTC->CRL &= ~( 1<<4 ) ; //退出配置模式
while( !( RTC->CRL&( 1<<5 ) ) ) ; //等待RTC寄存器最后一次操作完成
BKP->DR1 = 0x5050 ;
}
else
{
while( !( RTC->CRL&( 1<<3 ) ) ) ; //等待RTC寄存器同步
while( !( RTC->CRL&( 1<<5 ) ) ) ; //等待RTC寄存器操作完成
}
}
/***************************************************
Name :Is_Leap_Year
Function :閏年判定
Parameter :
year:年份
Return :閏年
***************************************************/
u8 Is_Leap_Year( u16 year )
{
//必須能被4整除
if( year%4==0 )
{
if( year%100==0 )
{
if( year%400==0 )
return 1 ;
else
return 0 ;
}
else
return 1 ;
}
else
return 0;
}
/***************************************************
Name :RTC_Set_Time
Fuction :設置時間
Parameter :
year:年
month:月
date:日
hour:時
minute:分
second:秒
Return :None
***************************************************/
u8 mon_table[] = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 } ;
void RTC_Set_Time( u8 year, u8 month, u8 date, u8 hour, u8 minute, u8 second )
{
u16 t ;
u32 seccount = 0 ;
//把所有年份的秒鐘相加
for( t=1970; t2000; t++ )
{
if( Is_Leap_Year( t ) )
seccount += 31622400 ; //閏年的秒鐘數
else
seccount += 31536000 ; //平年的秒鐘數
}
month -= 1 ;
for( t=0; t//把前面月份的秒鐘數相加
{
seccount += ( u32 )mon_table[ t ]*86400 ; //月份秒鐘數相加
if( Is_Leap_Year( year+2000 )&&( t==1 ) )
seccount += 86400 ; //閏年2月份增加一天的秒鐘數
}
seccount += ( date-1 )*86400 ; //把前面日期的秒鐘數相加
seccount += hour*3600 ; //小時秒鐘數
seccount += minute*60 ; //分鐘秒鐘數
seccount += second ; //最后的秒鐘加上去
//設置時鐘
RCC->APB1ENR |= 1<<28 ; //使能電源時鐘
RCC->APB1ENR |= 1<<27 ; //使能備份時鐘
PWR->CR |= 1<<8 ; //取消備份區寫保護
RTC->CRL |= 1<<4 ; //允許配置
RTC->CNTL = seccount&0xFFFF ;
RTC->CNTH = seccount>>16 ;
RTC->CRL &= ~( 1<<4 ) ; //配置更新
while( ( RTC->CRL&0x20 )!=0x20 ) ; //等待RTC寄存器操作完成
RTC_Get_Time() ; //設置完之后更新一下數據
}
/***************************************************
Name :RTC_Get_Time
Fuction :獲取時間
Parameter :None
Return :None
***************************************************/
RTC_Data RTC_Time;
void RTC_Get_Time()
{
u16 daycnt=0;
u32 timecount=0;
u32 temp=0;
u16 temp1=0;
//得到計數器中的值
timecount = RTC->CNTH ;
timecount <<= 16 ;
timecount += RTC->CNTL ;
//得到天數
temp = timecount/86400 ;
//超過一天了
if( daycnt!=temp )
{
daycnt = temp ;
temp1 = 1970 ; //從1970年開始
while( temp>=365 )
{
//閏年
if( Is_Leap_Year( temp1 ) )
{
if( temp>=366 )
temp -= 366 ; //閏年的秒鐘數
else
break ;
}
else
temp -= 365; //平年
temp1 ++ ;
}
RTC_Time.year = temp1-2000 ; //得到年份
temp1 = 0 ;
//超過了一個月
while( temp>=28 )
{
if( Is_Leap_Year( RTC_Time.year+2000 )&&( temp1==1 ) )//當年是不是閏年/2月份
{
if( temp>=29 )
temp -= 29 ;//閏年的秒鐘數
else
break;
}
else
{
if( temp>=mon_table[ temp1 ] )
temp -= mon_table[ temp1 ] ; //平年
else
break ;
}
temp1 ++ ;
}
RTC_Time.month = temp1+1 ; //得到月份
RTC_Time.date = temp+1 ; //得到日期
}
temp = timecount%86400 ; //得到秒鐘數
RTC_Time.hour = temp/3600 ; //小時
RTC_Time.minute = ( temp%3600 )/60 ; //分鐘
RTC_Time.second = ( temp%3600 )%60 ; //秒鐘
}
(3)創建1.c文件并輸入以下代碼。
#include "sys.h"
#include "delay.h"
#include "usart1.h"
#include "lcd.h"
#include "rtc.h"
int main()
{
u8 Str[ 50 ] ;
STM32_Clock_Init( 9 ) ; //STM32時鐘初始化
SysTick_Init( 72 ) ; //SysTick初始化
USART1_Init( 72, 115200 ) ; //初始化串口1波特率115200
LCD_Init() ; //LCD初始化
RTC_Init() ;
RTC_Set_Time( 20, 12, 10, 10, 8, 0 ) ;
while( 1 )
{
RTC_Get_Time() ;
sprintf( ( char * )Str, "20%02d-%02d-%02d %02d:%02d:%02d", RTC_Time.year, RTC_Time.month, RTC_Time.date, RTC_Time.hour, RTC_Time.minute, RTC_Time.second ) ;
LCD_ShowString( 10, 10, Str ) ;
delay_ms( 500 ) ;
}
}
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。
舉報投訴
-
電源
+關注
關注
184文章
17841瀏覽量
251815 -
晶體振蕩器
+關注
關注
9文章
631瀏覽量
29253 -
時鐘芯片
+關注
關注
2文章
253瀏覽量
39975 -
RTC
+關注
關注
2文章
544瀏覽量
67032
發布評論請先 登錄
相關推薦
CubeMX入門STM32 HAL庫的學習筆記
STM32CubeMX是ST官方開發的一款很好用的軟件,主要用于STM32系列單片機的硬件驅動層的配置。網上也有一些關于STM32CubeMX的相關教程,對它的相關操作進行說明。從今天起,小柯希望能將自己從CubeMX
發表于 08-17 06:55
STM32各模塊學習筆記
STM32個模塊學習筆記 目錄 STM32筆記之一 中斷優先級.......................................
發表于 11-30 03:32
?3177次閱讀
STM32學習之STM32cubeMX軟件安裝與使用(附STM32cubeMX5.2.1下載地址)
STM32學習之STM32cubeMX軟件安裝與使用(附STM32cubeMX5.2.1下載地址
發表于 12-07 11:51
?28次下載
STM32入門學習筆記之RTC實驗(上)
實時時鐘Real TimeClock(簡稱RTC),實時時鐘芯片是日常生活中應用最為廣泛的消費類電子產品之一。它為人們提供精確的實時時間,或者為電子系統提供精確的時間基準,目前實時時鐘芯片大多采用精度較高的晶體振蕩器作為時鐘源。有些時鐘芯片為了在主電源掉電時,還可以工作,需要外加電池供電。
評論