1.0 RTC简介
RTC(Real Time Clock)实时时钟 RTC是一个独立的定时器 ,可为系统提供时钟和日历的功能 RTC和时钟配置系统处于后备区域,系统复位时数据不清零 ,VDD(2.0~3.6V)断电后可借助VBAT(1.8~3.6V)供电继续走时 32位的可编程计数器,可对应Unix时间戳的秒计数器 20位的可编程预分频器,可适配不同频率的输入时钟 可选择三种RTC时钟源: HSE时钟除以128(通常为8MHz/128) LSE振荡器时钟(通常为32.768KHz) LSI振荡器时钟(40KHz)
--手册解读--
2.0 RTC框图
3.0 RTC基本结构
4.0 RTC库函数讲解
相关库函数对应的作用:
注:在C语言中在数据的前面加0要谨慎,在C语言中数据前面加0会被编译器默认为是8进制的数,而在8进制中,数字9是不成立的,数据中01表示的是8进制的 1。
5.0 RTC实时时钟实现
对应的头文件程序
cpp
#ifndef __MYRTC_H_
#define __MYRTC_H_
#include <stdint.h>
extern uint16_t MyRTC_Time[];
void MyRTC_Init(void);
void MyRTC_SetTime(void);
void MyRTC_ReadTime(void);
#endif
6.0 时钟初始化
cpp
// 全局数组,时间为 年 月 日 时 分 秒
uint16_t MyRTC_Time[] = {2023, 1, 1, 23, 59, 55};
// 初始化RTC时钟
void MyRTC_Init(void)
{
// 开启RCC时钟
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE); // 开启PWR的时钟
RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP, ENABLE); // 开启BKP的时钟
PWR_BackupAccessCmd(ENABLE); // 使用PWR开启对备份寄存器的访问
// 通过写入备份寄存器的标志位,判断RTC是否是第一次配置
if (BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5)
{
// 开启LSE时钟
RCC_LSEConfig(RCC_LSE_ON);
// LES准备就绪
while (RCC_GetFlagStatus(RCC_FLAG_LSERDY) != SET)
;
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE); // 选择RTCCLK来源为LSE,外部低速时钟
RCC_RTCCLKCmd(ENABLE); // RTCCLK使能
RTC_WaitForSynchro(); // 等待同步,等待RTC时钟同步更新完成
RTC_WaitForLastTask(); // 等待上一次操作完成
RTC_SetPrescaler(32768 - 1); // 设置RTC预分频器,预分频后的计数频率为1Hz
RTC_WaitForLastTask(); // 等待上一次操作完成
MyRTC_SetTime(); // 设置时间,调用此函数,全局数组里时间值刷新到RTC硬件电路
// 在备份寄存器写入自己规定的标志位,用于判断RTC是不是第一次执行配置
BKP_WriteBackupRegister(BKP_DR1, 0xA5A5);
}
else // RTC不是第一次配置
{
RTC_WaitForSynchro(); // 等待同步
RTC_WaitForLastTask(); // 等待上一次操作完成
}
}
// 如果LSE无法起振导致程序卡死在初始化函数中
// 可将初始化函数替换为下述代码,使用LSI当作RTCCLK
// LSI无法由备用电源供电,故主电源掉电时,RTC走时会暂停
/****************************************************************
* void MyRTC_Init(void)
{
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP, ENABLE);
PWR_BackupAccessCmd(ENABLE);
if (BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5)
{
RCC_LSICmd(ENABLE);
while (RCC_GetFlagStatus(RCC_FLAG_LSIRDY) != SET);
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);
RCC_RTCCLKCmd(ENABLE);
RTC_WaitForSynchro();
RTC_WaitForLastTask();
RTC_SetPrescaler(40000 - 1);
RTC_WaitForLastTask();
MyRTC_SetTime();
BKP_WriteBackupRegister(BKP_DR1, 0xA5A5);
}
else
{
RCC_LSICmd(ENABLE); //即使不是第一次配置,也需要再次开启LSI时钟
while (RCC_GetFlagStatus(RCC_FLAG_LSIRDY) != SET);
RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);
RCC_RTCCLKCmd(ENABLE);
RTC_WaitForSynchro();
RTC_WaitForLastTask();
}
}
*****************************************************************
*/
cpp
// RTC读取时间
void MyRTC_ReadTime(void)
{
// 定义秒计数器的数据类型
time_t time_cnt;
// 定义日期时间的数据类型
struct tm time_date;
// 读取RTC的CNT值,获取当前的秒计数器,调整时间为东八区时间
time_cnt = RTC_GetCounter() + 8 * 60 * 60;
// 将秒计数器转换为日期时间格式
time_date = *localtime(&time_cnt);
MyRTC_Time[0] = time_date.tm_year + 1900; // 将日期时间结构体赋值给数组的时间
MyRTC_Time[1] = time_date.tm_mon + 1;
MyRTC_Time[2] = time_date.tm_mday;
MyRTC_Time[3] = time_date.tm_hour;
MyRTC_Time[4] = time_date.tm_min;
MyRTC_Time[5] = time_date.tm_sec;
}
7.0 MAIN函数
cpp
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"
int main(void)
{
/*模块初始化*/
OLED_Init(); //OLED初始化
AD_Init(); //AD初始化
/*显示静态字符串*/
OLED_ShowString(1, 1, "AD0:");
OLED_ShowString(2, 1, "AD1:");
OLED_ShowString(3, 1, "AD2:");
OLED_ShowString(4, 1, "AD3:");
while (1)
{
OLED_ShowNum(1, 5, AD_Value[0], 4); //显示转换结果第0个数据
OLED_ShowNum(2, 5, AD_Value[1], 4); //显示转换结果第1个数据
OLED_ShowNum(3, 5, AD_Value[2], 4); //显示转换结果第2个数据
OLED_ShowNum(4, 5, AD_Value[3], 4); //显示转换结果第3个数据
Delay_ms(100); //延时100ms,手动增加一些转换的间隔时间
}
}
......