RTC時鐘
- 一、RTC時鐘的簡介
- 二、RTC的具體配置
- 1.對應的時鐘圖
- 2.RTC配置一般步驟
- 3.RTC特征
- ①鬧鐘中斷
- ②秒中斷(常用)
- ③溢位中斷(應該不常用)
- 3.主函式中的使用
- 未解決的問題
- 其他學習總結
- 1.將OLED從mini板移到C8T6上
- 2.AD20復制問題
- 總結
- 寫在最后
一、RTC時鐘的簡介
RTC時鐘不同于STM32單片機上其他的時鐘,RTC時鐘在STM32mini板子上經一個紐扣電池供電,所以它不需要我們再供給電源就可以進行計數,這樣的特殊性使得它可以讓我們利用它制作一個類似萬年歷的小鐘表,
二、RTC的具體配置
1.對應的時鐘圖

這里的TR_CLK是由RTCCLK分頻后進行裝載,實際我們將它裝載的周期控制到一秒,再用一個終端服務函式使得我們每一秒都將時間更新,而RTC_DIV是經沒有分頻的RTCCLK進行裝載的 ,所以它的裝載周期就會小很多,并且TR_CLK和RTC_DIV的數量關系恰好是分頻系數,這就使得我們可以精確到秒后面更小的單位,但是對于萬年歷這樣的功能,對于時間的精準性不需要太高所以就不需要對RTC_DIV這個引數進行操作,
2.RTC配置一般步驟
①使能PWR和BKP時鐘
②使能后備暫存器訪問
③配置RTC時鐘源,使能RTC時鐘
④設定RTC預分頻系數
⑤設定時間
⑥開啟相關中斷(在需要的時候)
⑦撰寫終端服務函式
⑧一些操作要等待寫操作完成和同步
代碼如下:
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)//通過判斷暫存器標志位判斷低速晶振是否準備就緒
{
temp++;
delay_ms(10);
}
if(temp>=250)return 1;
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE); //選擇外部低速晶振作為時鐘來源
RCC_RTCCLKCmd(ENABLE);
RTC_WaitForLastTask(); //等待寫操作完成
RTC_WaitForSynchro(); //等待同步完成
RTC_ITConfig(RTC_IT_SEC, ENABLE); //使能RTC秒中斷
RTC_WaitForLastTask();
RTC_EnterConfigMode(); //允許配置
RTC_SetPrescaler(32767); //設定預分頻系數
RTC_WaitForLastTask();
RTC_Set(2020,10,18,9,40,20); //設定初始時間
RTC_ExitConfigMode(); //退出配置模式
3.RTC特征
RTC時鐘可以有三種中斷函式,分別是鬧鐘中斷,秒中斷和溢位中斷,
①鬧鐘中斷
用來產生一個軟體可編程的鬧鐘中斷
②秒中斷(常用)
用來產生一個可編程的周期性中斷信號(最長為1秒)
③溢位中斷(應該不常用)
指示內部可編程計數器溢位并回轉為0的狀態
3.主函式中的使用
u8 RTC_Get(void)
{
static u16 daycnt=0;
u32 timecount=0;
u32 temp=0;
u16 temp1=0;
timecount=RTC_GetCounter();
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 {temp1++;break;}
}
else temp-=365; //平年
temp1++;
}
calendar.w_year=temp1;//得到年份
temp1=0;
while(temp>=28)//超過了一個月
{
if(Is_Leap_Year(calendar.w_year)&&temp1==1)//今年是不是閏年
{
if(temp>=29)temp-=29;//閏年的秒針數
else break;
}
else
{
if(temp>=mon_table[temp1])temp-=mon_table[temp1];
else break;
}
temp1++;
}
calendar.w_month=temp1+1; //月
calendar.w_date=temp+1; //日
}
temp=timecount%86400; //得到今天總共過了多少秒
calendar.hour=temp/3600; //小時
calendar.min=(temp%3600)/60; //分鐘
calendar.sec=(temp%3600)%60; //秒
calendar.week=RTC_Get_Week(calendar.w_year,calendar.w_month,calendar.w_date);//得到第幾周
return 0;
}
這個函式是在RTC函式中,是一個顯示當前時間的函式,之所以很復雜是因為在RTC的暫存器中只記錄秒數,也就是我們需要根據現在的秒數進行計算當前的日期和時間,我們可以直接進行利用,
在主函式中我們可以配合顯示模塊,將時間進行顯示,上面展示的函式將年月日時分秒星期寫如一個結構體,在參考時我們需要用一個結構體指標進行使用,
未解決的問題
商家給的例程是用LCD螢屏對時間進行的顯示,燒錄后效果達到,
我想要將時間顯示在OLED螢屏上時,打算像往常一樣 直接復制一個RTC工程檔案然后在HARDWARE檔案里新建一個OLED的檔案夾,之后將對應的OLED檔案復制過來,再在MDK5中添加.c檔案然后添加路徑,但是編譯后一直會顯示OLED的某一個頭檔案找不到或者已損壞,排查后確定不是損壞問題,而是檔案路徑問題,但是路徑已經添加過了還是不知道為何報錯,下面是問題截圖


問題還未解決,如果以后解決會在這里進行修改,
盡管這樣,使用正點原子的OLED程式就不會產生這樣的報錯,
其他學習總結
1.將OLED從mini板移到C8T6上
具體操作就是將程式在C8T6上沒有的引腳轉換為C8T6上的引腳
#define LED_ON GPIO_ResetBits(GPIOD,GPIO_Pin_2)
#define LED_OFF GPIO_SetBits(GPIOD,GPIO_Pin_2)
在C8T6上沒有D引腳,所以我們進行轉換,這兩行代碼在OLED.h中找到
我選擇換到了PA1和PA2,不僅是這里需要修改還有就是對IO 口的使能需要進行修改,
最后還需要將工程檔案中的芯片選擇為C8T6,由于OLED沒有使用定時器,所以我們更換IO只需要選擇兩個無特殊作用的就可以,
2.AD20復制問題
在畫板子程序中,經常會出現鋪銅后導線也被鋪銅,所以常常直接復制所有的元件到新PCB中就可以解決,但是這里有個細節沒有發現,就是如果我們直接使用Ctrl+c,Ctrl+v就會讓我們復制后的 PCB沒有網路,解決方法就是我們需要進行特殊粘貼,具體操作如下
復制整個原理圖時,我們能可以全部框選后Ctrl+c進行復制,之后
選擇特殊粘貼

之后我們一定不要勾選粘貼到當前層,否則他會強制將兩層板壓縮到一層板,產生很多報錯,還有就是保持網路名稱需要勾選上,這樣我們復制后的PCB就有網路了,

總結
這周的學習比較倉促,所以還有一些問題沒有解決,希望可以快點解決OLED添加工程的問題,
RTC的使用情況比較特殊,必須使用一塊電池進行供電,所以C8T6不能使用,但是以后可以自己畫板在其上添加,就可以擁有一個獨立的時鐘模塊,可能會為以后的低功耗有幫助,
AD20還是有一些不太熟練,有時候對板子進行修改還是會有一些問題,還需要多總結,
寫在最后
本人小白一枚,這些僅僅是我個人的經驗與認知,也許含有多處錯誤,希望讀者給我指正,
轉載請註明出處,本文鏈接:https://www.uj5u.com/qianduan/182897.html
標籤:其他
