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

......

相关推荐
sinat_3607048221 分钟前
STM32 看门狗
stm32·单片机·嵌入式硬件
亿道电子Emdoor23 分钟前
【ARM】MDK如何生成指定大小的bin文件,并指定空区域的填充数据
arm开发·stm32·单片机
mftang29 分钟前
STM32 CAN过滤器配置和应用方法介绍
stm32·单片机·嵌入式硬件
楼台的春风2 小时前
【详细讲解在STM32的UART通信中使用DMA机制】
stm32·单片机·嵌入式硬件·mcu·物联网·嵌入式·信息与通信
Chambor_mak3 小时前
stm32单片机个人学习笔记16(SPI通信协议)
stm32·单片机·学习
小白今天也很酷3 小时前
Python与MCU通信:串口数据采集及CSV/Excel存储方法
python·单片机·excel
海的预约4 小时前
51单片机-按键
单片机·嵌入式硬件·51单片机
【云轩】8 小时前
基于STM32与IFX007T的电机驱动全解析(无人机/机器人实战)
stm32·机器人·无人机
山东布谷科技官方11 小时前
AI大模型发展对语音直播交友系统源码开发搭建的影响
人工智能·实时音视频·交友
qq_75568224015 小时前
STM32使用NRF2401进行数据传送
stm32·单片机·嵌入式硬件