STM32-RTC实时时钟

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,手动增加一些转换的间隔时间
	}
}

......

相关推荐
lucy1530275107928 分钟前
刷式直流电机驱动芯片,适用于打印机、电器、工业设备以及其他小型机器中——GC8870
人工智能·stm32·单片机·嵌入式硬件·机器人
皮皮黄-机电工程师1 小时前
单片机控制步进电机 A4988 Proteus仿真
单片机·proteus·步进电机·a4988
一只搬砖的猹1 小时前
项目实战——使用python脚本完成指定OTA或者其他功能的自动化断电上电测试
linux·单片机·嵌入式硬件·python自动化·rtos·嵌入式软件·ota
一口一口吃成大V2 小时前
FPGA随记——时钟时序一些基本知识
单片机·嵌入式硬件
菜哥万岁万岁万万岁3 小时前
STM32 USB组合设备 MSC CDC
stm32·单片机·嵌入式硬件
wenchm3 小时前
细说STM32F407单片机以DMA方式读写外部SRAM的方法
stm32·单片机·嵌入式硬件
电气_空空3 小时前
基于单片机的无人值守输液监控系统软件设计
单片机·嵌入式硬件·毕业设计·毕设
半个番茄3 小时前
STM32: 输入捕获基本结构
stm32·单片机
2301_805962934 小时前
NRF24L01模块STM32调试心得-调试顺序
stm32·单片机·嵌入式硬件
云山工作室5 小时前
基于单片机的智能婴儿监控系统设计(论文+源码)
单片机·嵌入式硬件·毕业设计·毕设