【STM32-学习笔记-11-】RTC实时时钟

文章目录

  • RTC实时时钟
    • 一、RTC简介
    • 二、RTC框图
    • 三、RTC基本结构
    • 四、RTC操作注意事项
    • 五、RTC函数
    • 六、配置RTC
        • MyRTC.c
    • 七、示例:实时时钟
      • ①、main.c
      • ②、MyRTC.c
      • ③、MyRTC.h

RTC实时时钟

一、RTC简介

  • RTC(Real Time Clock)实时时钟

  • RTC是一个独立的定时器,可为系统提供时钟和日历的功能

  • RTC和时钟配置系统处于后备区域,系统复位时数据不清零,VDD(2.0~3.6V)断电后可借助VBAT(1.8~3.6V)供电继续走时

  • 32位的可编程计数器,可对应Unix时间戳的秒计数器

  • 20位的可编程预分频器,可适配不同频率的输入时钟(产生稳定的1Hz的时钟

  • 可选择三种RTC时钟源:

    • 高速外部时钟信号)HSE时钟除以128(通常为8MHz/128)
      • 主要作为系统时钟
    • 低速外部时钟信号)LSE振荡器时钟(通常为32.768KHz)
      • 主要用于RTC时钟,可由VBAT供电
      • 2 15 = 32768 2^{15}=32768 215=32768 则使用15位的计数器,计数值从0~32767,自然溢出即可产生1Hz的时钟信号
    • 低速内部时钟信号)LSI振荡器时钟(40KHz)
      • 主要作为看门狗时钟
    • image-20250115103618025

二、RTC框图

  • RTC寄存器由RTCCLK寄存器驱动

image-20250115104623366

  • 灰色区域在主电源掉电之后可由VBAT供电
  • RTC_PRL 预分频装载寄存器,用来保存RTC预分频器的周期计数值(写入R则是R+1分频
  • RTC_DIV 预分频器余数寄存器
    • 每来一个脉冲,计数值-1;自减到0时,再来一个脉冲,则会产生一个溢出信号(TR_CLK);同时DIV从PRL获取一个重装值,再继续自减
    • 该溢出信号即为所需的1Hz时钟
  • RTC_CNT (秒计数器)相当于Unix时间戳,再利用<time.h>库函数即可计算出时间
  • RTC_ALR 设置闹钟
    • RTC_ALR=RTC_CNT时,则会产生RTC_Alarm闹钟信号通往NVIC中断控制器

三、RTC基本结构

image-20250115110845391

image-20250115111150460

四、RTC操作注意事项

  1. 使能对BKP(备份寄存器)和RTC的访问

    • 需要设置RCC_APB1ENR寄存器的PWRENBKPEN位,以**

      •   RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE);
        
    • 需要设置PWR_CR寄存器的DBP位,以使能对BKPRTC的访问

      •   PWR_BackupAccessCmd(ENABLE);
        
  2. 等待寄存器同步标志位被置位

    • 如果在读取RTC寄存器时,RTC的APB1接口曾经处于禁止状态,则软件首先必须等待RTC_CRL寄存器中的RSF位(寄存器同步标志)被硬件置1

      •   while (!(RTC->CRL & RTC_CRL_RSF));  // 等待寄存器同步标志位被置位
        
    • 这是因为在APB1接口禁止状态下,寄存器的值可能不一致,需要等待同步标志位被置位以确保寄存器值的一致性

  3. 进入配置模式

    • 必须设置RTC_CRL寄存器中的CNF位,使RTC进入配置模式后,才能写入RTC_PRL(预分频器)、RTC_CNT(计数器)、RTC_ALR(闹钟寄存器)等寄存器

      •   RTC->CRL |= RTC_CRL_CNF;  // 设置配置模式位while (!(RTC->CRL & RTC_CRL_CNF));  // 等待配置模式位被硬件清除
        
  4. 写操作的时序要求:等待前一次写操作结束

    • 对RTC任何寄存器的写操作,都必须在前一次写操作结束后进行

    • 可以通过查询RTC_CR寄存器中的RTOFF状态位,判断RTC寄存器是否处于更新中

    • 仅当RTOFF状态位是1时,才可以写入RTC寄存器。这是因为RTC寄存器在更新过程中是不允许写入的,必须等待更新完成(即RTOFF位被置位)后才能进行下一次写操作

      •   while (!(RTC->CR & RTC_CR_RTOFF));  // 等待上一次写操作完成RTC->CNT = new_value;  // 写入新的值
        

五、RTC函数

// 配置RTC中断
void RTC_ITConfig(uint16_t RTC_IT, FunctionalState NewState);// 进入RTC配置模式
void RTC_EnterConfigMode(void);
// 退出RTC配置模式
void RTC_ExitConfigMode(void);// 获取RTC计数器CNT的值
uint32_t  RTC_GetCounter(void);
// 设置RTC计数器CNT的值
void RTC_SetCounter(uint32_t CounterValue);// 设置RTC预分频值(写入PRL重装寄存器中)
void RTC_SetPrescaler(uint32_t PrescalerValue);// 设置RTC闹钟值ALR
void RTC_SetAlarm(uint32_t AlarmValue);// 获取RTC分频值(获取余数寄存器RTC_DIV的值)
uint32_t RTC_GetDivider(void);// 等待RTC上一个任务完成
void RTC_WaitForLastTask(void);// 等待RTC同步(等待RSF【寄存器同步标志】置1)
void RTC_WaitForSynchro(void);// 获取RTC标志位状态
FlagStatus RTC_GetFlagStatus(uint16_t RTC_FLAG);
// 清除RTC标志位
void RTC_ClearFlag(uint16_t RTC_FLAG);// 获取RTC中断状态
ITStatus RTC_GetITStatus(uint16_t RTC_IT);
// 清除RTC中断待处理位
void RTC_ClearITPendingBit(uint16_t RTC_IT);

六、配置RTC

  1. 开启电源(PWR)和备份寄存器(BKP)时钟

    •   	//开启电源(PWR)和备份寄存器(BKP)时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP, ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);
      
  2. 使能对BKPRTC的访问

    •   	//使能对BKP和RTC的访问PWR_BackupAccessCmd(ENABLE);
      
  3. 开启LSE(外部低速时钟)时钟(RCC模块中)

  4. 配置RTCCLK数据选择器,指定LSERTCCLK的时钟源(RCC模块中)

  5. 等待:

    1. 前一次写操作结束
    2. 等待寄存器同步标志位被置位
  6. 配置PRL重装寄存器

  7. 配置CNT

  8. 选配闹钟和中断

MyRTC.c
#include "stm32f10x.h"                  // Device headervoid MyRTC_Init(void)
{//开启电源(PWR)和备份寄存器(BKP)时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP, ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);PWR_BackupAccessCmd(ENABLE);//使能对BKP和RTC的访问RCC_LSEConfig(RCC_LSE_ON);//开启LSE时钟while(RCC_GetFlagStatus(RCC_FLAG_LSERDY) == !SET);//等待LSE启动完成RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);//指定LSE为RTC的时钟源RCC_RTCCLKCmd(ENABLE);//使能RTCCLK的时钟RTC_WaitForLastTask();//等待RTC上一个任务完成RTC_WaitForSynchro();//等待RTC同步RTC_SetPrescaler(32768 - 1);//(0~32767所以要减1)配置预分频器的值(32.768kHz/32768 = 1)RTC_WaitForSynchro();//写入寄存器时都需要等待RTC同步RTC_SetCounter(1735660800);//配置计数器CNT的值(时间戳)【2025-01-1 0:0:00】RTC_WaitForSynchro();//等待RTC同步
}

七、示例:实时时钟

①、main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "MyRTC.h"int main(void)
{OLED_Init();MyRTC_Init();OLED_ShowString(1,1,"Date:    -  -");OLED_ShowString(2,1,"Time:  :  :");OLED_ShowString(3,1,"CNT :");OLED_ShowString(4,6,"DIV:");OLED_ShowHexNum(4,1, BKP_ReadBackupRegister(BKP_DR1), 4);struct tm* DATA;while(1){DATA = MyRTC_Get_Time();OLED_ShowNum(1,6, DATA->tm_year + 1900,4);//年OLED_ShowNum(1,11,DATA->tm_mon + 1,2);		//月OLED_ShowNum(1,14,DATA->tm_mday,2);				//日OLED_ShowNum(2,6, DATA->tm_hour,2);				//时OLED_ShowNum(2,9, DATA->tm_min ,2);				//分OLED_ShowNum(2,12,DATA->tm_sec ,2);				//秒OLED_ShowNum(3,6,RTC_GetCounter(),10);OLED_ShowNum(4,10,RTC_GetDivider(),6);}
}

②、MyRTC.c

可通过设置Timestamp时间戳变量来设置系统时间

#include "stm32f10x.h"                  // Device header
#include "MyRTC.h"#define TIME_ZONE_OFFSET (8 * 60 * 60)// 东八区时区偏移量(秒)time_t Timestamp = 1735660780;//所设置时间的时间戳void MyRTC_Init(void)
{//开启电源(PWR)和备份寄存器(BKP)时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP, ENABLE);PWR_BackupAccessCmd(ENABLE);//使能对BKP和RTC的访问//加上if_else是为了防止重复初始化和时间重置if(BKP_ReadBackupRegister(BKP_DR1) != 0x1212){RCC_LSEConfig(RCC_LSE_ON);//开启LSE时钟while(RCC_GetFlagStatus(RCC_FLAG_LSERDY) != SET);//等待LSE启动完成RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);//指定LSE为RTC的时钟源RCC_RTCCLKCmd(ENABLE);//使能RTCCLK的时钟RTC_WaitForSynchro();//等待RTC同步RTC_WaitForLastTask();//等待RTC上一个任务完成RTC_SetPrescaler(32768 - 1);//(0~32767所以要减1)配置预分频器的值(32.768kHz/32768 = 1)RTC_WaitForLastTask();//每一次写入操作都需要等待RTC上一个任务完成//		RTC_SetCounter(1735660800);//配置计数器CNT的值(时间戳)【2025-01-1 0:0:00】
//		RTC_WaitForLastTask();//等待RTC上一个任务完成MyRTC_Set_Time();//在备份寄存器写入自己规定的标志位,用于判断RTC是不是第一次执行配置BKP_WriteBackupRegister(BKP_DR1, 0x1212);}else{RTC_WaitForSynchro();//等待RTC同步RTC_WaitForLastTask();//等待RTC上一个任务完成}
}void MyRTC_Set_Time(void)//设置时间
{	RTC_SetCounter(Timestamp);//配置计数器CNT的值(时间戳)RTC_WaitForSynchro();//等待RTC同步
}struct tm* MyRTC_Get_Time(void)//获取时间
{time_t time_cnt;time_cnt = RTC_GetCounter() + TIME_ZONE_OFFSET;//东八区 加上8天(即8*60*60秒)struct tm* Temp;Temp = localtime(&time_cnt);return Temp;
}

③、MyRTC.h

#ifndef __MYRTC_H__
#define __MYRTC_H__
#include "stdint.h"
#include <time.h>extern time_t Timestamp;//所设置时间的时间戳void MyRTC_Init(void);
void MyRTC_Set_Time(void);//设置时间
struct tm* MyRTC_Get_Time(void);//获取时间#endif

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/67561.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

STL—stack与queue

目录 Stack stack的使用 stack的模拟实现 queue queue的使用 queue的模拟实现 priority_queue priority_queue的用法 priority_queue的模拟实现 容器适配器 种类 Stack http://www.cplusplus.com/reference/stack/stack/?kwstack stack是栈&#xff0c;后入先出 stack的…

【redis】redis-cli命令行工具的使用

redis-cli命令行工具是一个功能强大的Redis客户端&#xff0c;它允许用户与Redis数据库进行交互和管理。 以下是一些常用参数的使用说明&#xff1a; 基本连接参数 -h, --host <hostname>&#xff1a;指定要连接的Redis服务器的主机名或IP地址。如果未指定&#xff0c;…

wordpress的火车头商品发布接口

<?php require ../wp-load.php; ini_set(memory_limit, 1024M); set_time_limit(180);$top_cat ; # 图片链接域名替换 $image_host ;$start_time microtime(true);$counter 0; // 临时缓存 $products $skus $categories []; $var_sku_index 1;$rowData$_POST;// if…

GaussDB日常维护操作

GaussDB日常维护操作 日常维护检查操作系统参数数据库健康状态日志收集日志清理应用连接数表的例行维护索引重建慢SQL诊断 日常维护检查 实例状态检查&#xff1a; #检查集群实例状态 gs_check -U omm -i CheckClusterStatecm_ctl query -Cvipd#检查主备DN的角色和同步状态 g…

Asp .Net Core 实现微服务:集成 Ocelot+Nacos+Swagger+Cors实现网关、服务注册、服务发现

什么是 Ocelot ? Ocelot是一个开源的ASP.NET Core微服务网关&#xff0c;它提供了API网关所需的所有功能&#xff0c;如路由、认证、限流、监控等。 Ocelot是一个简单、灵活且功能强大的API网关&#xff0c;它可以与现有的服务集成&#xff0c;并帮助您保护、监控和扩展您的…

性价比1.2V电压基准替代

前言&#xff1a; 小于2V的电压基准比较少&#xff0c;且价格稍贵&#xff0c;对于要求不高的场合&#xff0c;1117可以替代使用&#xff0c;温度系数低于 100ppm/C, 价格便宜。 1117是线性稳压器的一种&#xff0c;一般情况下&#xff0c;输出电压可调。 如下述的1117&#xf…

使用 Java 和 FreeMarker 实现自动生成供货清单,动态生成 Word 文档,简化文档处理流程。

在上一篇博客中主要是使用SpringBootApache POI实现了BOM物料清单Excel表格导出&#xff0c;详见以下博客&#xff1a; Spring Boot Apache POI 实现 Exc&#xff08;&#xff09;el 导出&#xff1a;BOM物料清单生成器&#xff08;支持中文文件名、样式美化、数据合并&#…

探索与创作:2024年CSDN平台上的成长与突破

文章目录 我与CSDN的初次邂逅初学阶段的阅读CSDN&#xff1a;编程新手的避风港初学者的福音&#xff1a;细致入微的知识讲解考试复习神器&#xff1a;技术总结的“救命指南”曾经的自己&#xff1a;为何迟迟不迈出写博客的第一步兴趣萌芽&#xff1a;从“读”到“想写”的初体验…

鸿蒙(HarmonyOS)的开发

在当今数字化时代&#xff0c;软件开发不仅需要技术上的创新和突破&#xff0c;同时也必须严格遵守法律法规。作为中国自主研发的操作系统&#xff0c;鸿蒙&#xff08;HarmonyOS&#xff09;的开发者们更应该注重这一点。本文将围绕“鸿蒙开发需遵守法律法规”这一主题&#x…

SSM课设-学生管理系统

【课设者】SSM课设-学生管理系统 技术栈: 后端: SpringSpringMVCMybatisMySQLJSP 前端: HtmlCssJavaScriptEasyUIAjax 功能: 学生端: 登陆 学生信息管理 个人信息管理 老师端: 多了教师信息管理 管理员端: 多了班级信息管理 多了年级信息管理 多了系统用户管理

力扣 打家劫舍

动态规划&#xff0c;当前状态由前两个状态获得&#xff0c;滚动数组。 题目 从题可以看出要达到最高金额时&#xff0c;要从相邻的房屋拿。因此是当前房屋的金额隔一个做累加&#xff0c;当然还需要跟前一个相邻的房屋做比较&#xff0c;便于取到哪边金额更高&#xff0c;因此…

用公网服务器实现内网穿透

首先需要一个公网服务器 下载frp 搜索github下载到frp&#xff0c;服务端frps/客户端frpc。。下载的时候要注意自己本地内网机的cpu版本和服务端cpu架构 我的电脑是mac M1PRO版本 下载的是&#xff1a;darwinarm64 比如 服务端一般是Linux&#xff08;Intel 64位CPU&#xf…

clang、C++23标准库模块基本使用

1. 已经知道clang和LLVM的版本大于18的话&#xff0c;就支持C23标准的新特性&#xff1a;标准库模块啦&#xff01;那么该怎么使用呢&#xff1f; 2. 操作系统是Manjaro 3. 必须同时把clang 、LLVM 、 libc 、 libcabi这四个软件包同时安装好&#xff01;缺一不可…

【Django开发】django美多商城项目完整开发4.0第12篇:商品部分,表结构【附代码文档】

本教程的知识点为&#xff1a; 项目准备 项目准备 配置 1. 修改settings/dev.py 文件中的路径信息 2. INSTALLED_APPS 3. 数据库 用户部分 图片 1. 后端接口设计&#xff1a; 视图原型 2. 具体视图实现 用户部分 使用Celery完成发送 判断帐号是否存在 1. 判断用户名是否存在 后…

Redis的安装和使用--Windows系统

Redis下载地址&#xff1a; windows版本readis下载&#xff08;GitHub&#xff09;&#xff1a; https://github.com/tporadowski/redis/releases &#xff08;推荐使用&#xff09; https://github.com/MicrosoftArchive/redis/releases 官网下载&#xff08;无Windows版本…

Linux操作命令之云计算基础命令

一、图形化界面/文本模式 ctrlaltF2-6 图形切换到文本 ctrlalt 鼠标跳出虚拟机 ctrlaltF1 文本切换到图形 shift ctrl "" 扩大 ctrl "-" 缩小 shift ctrl "n" 新终端 shift ctrl "t" 新标签 alt 1,…

LabVIEW桥接传感器配置与数据采集

该LabVIEW程序主要用于配置桥接传感器并进行数据采集&#xff0c;涉及电压激励、桥接电阻、采样设置及错误处理。第一个VI&#xff08;"Auto Cleanup"&#xff09;用于自动清理资源&#xff0c;建议保留以确保系统稳定运行。 以下是对图像中各个组件的详细解释&#…

逐笔成交逐笔委托Level2高频数据下载和分析:20241230

逐笔委托逐笔成交下载 链接: https://pan.baidu.com/s/11Tdq06bbYX4ID9dEaiv_lQ?pwdcge6 提取码: cge6 Level2逐笔成交逐笔委托数据分享下载 利用Level2的逐笔交易和委托数据&#xff0c;这种以毫秒为单位的详细信息能揭露众多关键信息&#xff0c;如庄家意图、伪装行为&…

网络编程 | UDP广播通信

1、什么是广播 在上一篇博客文章中已经对UDP进行了详细的说明介绍及如何编程实现。本文将接着上一文的内容&#xff0c;在其基础上&#xff0c;对UDP的知识体系进一步深入的讲解。 网络编程 | UDP套接字通信及编程实现经验教程-CSDN博客 例子&#xff1a;在一些中小学的操场中&…

Count Sketch--计数草图

背景 Count Sketch 是一种空间高效的概率型数据结构&#xff0c;由 Moses Charikar、Kevin Chen 和 Martin Farach-Colton 在 2002 年提出&#xff0c;用于估计数据流中元素的频率&#xff0c;也可用于解决重击者问题。 原理 算法结构 参数设定&#xff1a;Count Sketch算法…