当前位置: 首页 > article >正文

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实时时钟实现


对应的头文件程序

#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 时钟初始化


// 全局数组,时间为 年 月 日 时 分 秒
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();
    }
}
*****************************************************************
*/

// 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函数


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

......


http://www.kler.cn/a/487396.html

相关文章:

  • uniapp:钉钉小程序需要录音权限及调用录音
  • 工作中Excel技巧整理
  • Android GSI (Generic System Image)
  • 2025年01月09日Github流行趋势
  • 在Rust中实现Drop trait的注意事项有哪些?
  • IP属地与IP地址:联系与区别的深度剖析
  • 【网络协议】IPv4 地址分配 - 第二部分
  • 分布式锁 Redis vs etcd
  • nmap命令详解:网络安全扫描的利器
  • CSS语言的数据库交互
  • Nginx | 解决 Spring Boot 与 Nginx 中的 “413 Request Entity Too Large“ 错误
  • WebRTC:构建实时通信应用的利器
  • 音视频入门基础:MPEG2-PS专题(6)——FFmpeg源码中,获取PS流的视频信息的实现
  • 超简单,使用Kube-Vip实现K8s高可用VIP详细教程
  • Attention计算中的各个矩阵的维度都是如何一步步变化的?
  • SQL UNION 操作符
  • 【Linux 之一 】Linux常用命令汇总
  • Redis数据库笔记——Cluster集群模式
  • 状态模式详解
  • 大模型搜索引擎增强问答demo-纯python实现