BKP(备份寄存器)和 RTC(实时时钟)

news2025/6/2 7:35:24

什么是BKP?

        备份寄存器(BackupRegister)42个16位的寄存器(不同设备存在差异:20字节(中容量和小容量)/84字节(大容量和互联型)),可用来存储 最多84个字节的用户应用程序数据。他们处在备份域里。(VDD电 源被切断,他们仍然由VBAT维持供电。

特点:

  • 系统待机模式下被唤醒,或系统复位或电源复位时,他们也不会被复位
  • 当断电后,后备寄存器中写入的数据会丢失

作用:

 管理侵入检测 和 RTC校准功能

        复位后,备份寄存器 和 RTC的访问 被禁止,并且 备份域 被保护以防止可能存在的意外的写操作。执行以下操作可以使能对备份寄存器和RTC的访问

  • 通过设置寄存器RCC_APB1ENR的 PWREN 和 BKPEN 位来 打开电源 和 后备接口的时钟
  • 电源控制寄存器(PWR_CR)的 DBP位 来使能对 后备寄存器和 RTC的访问。

BKP框图 

小实验:读写BKP后备寄存器 

bkp.c文件代码:

#include "rtc.h"

RTC_HandleTypeDef rtc_handle = {0};
void rtc_init(void){
    //要想读写BKP  
    //1.要开始电源和后背寄存器的时钟
    __HAL_RCC_PWR_CLK_ENABLE();   
    __HAL_RCC_BKP_CLK_ENABLE();
    //使能后备寄存器
    HAL_PWR_EnableBkUpAccess();
    
    rtc_handle.Instance = RTC;
    rtc_handle.Init.AsynchPrediv = 32767;
    rtc_handle.Init.OutPut = RTC_OUTPUTSOURCE_NONE;    
    HAL_RTC_Init(&rtc_handle);
}

/*读后备寄存器中的数据,并返回*/
uint16_t rtc_read_bkr(uint8_t bkrx){
    
    uint32_t data = 0;
    
    data = HAL_RTCEx_BKUPRead(&rtc_handle, bkrx);
    
    return (uint16_t)data;
}

/*对后备寄存器中写数据,指定参数:哪个寄存器,写入的数据*/
void rtc_write_bkr(uint8_t bkrx,uint16_t data){
    
    HAL_RTCEx_BKUPWrite(&rtc_handle,bkrx,data);
}
  • bkp.h文件代码
#ifndef __RTC_H__
#define __RTC_H__
#include "stm32f1xx.h"

void rtc_init(void);
uint16_t rtc_read_bkr(uint8_t bkrx);
void rtc_write_bkr(uint8_t bkrx,uint16_t data);

#endif

  •  mian.c文件代码
#include "sys.h"
#include "led.h"
#include "delay.h"
#include "uart1.h"
#include "rtc.h"


int main(void)
{
    HAL_Init();                         /* 初始化HAL库 */
    stm32_clock_init(RCC_PLL_MUL9);     /* 设置时钟, 72Mhz */
    led_init();                         /* LED初始化 */
    uart1_init(115200);
    printf("hello,world\r\n");
    rtc_init();
    
    rtc_write_bkr(1,0xABCD);
    printf("读出来的十六进制的数是:%X\r\n",rtc_read_bkr(1));
    
    while(1)
    { 
    }
}

实验现象:

  • 现象1:当系统复位后,之前写入的的数据依旧被打印出来。
  • 现象2:断电后,后备寄存器中的数据会丢失。 

什么是RTC? 

        实时时钟是一个独立的定时器。 RTC模块拥有一组连续计数的计数器,在相应软件配置下,可提供时钟 和 日历的功能(F1芯片是没有这个功能的)。修改计数器的值可以重新设置系统当前的时间和日期。

和上面的BKP原理相同:        

        RTC模块 和 时钟配置系统 (RCC_BDCR寄存器处于后备区域,即在系统复位或从待机模式唤醒后, RTC的设置和时间维持不变。

        复位后,对备份寄存器和RTC的访问被禁止,并且备份域被保护以防止可能存在的意外的写操作。执行以下操作可以使能对备份寄存器和RTC的访问:

  • 通过设置寄存器RCC_APB1ENRPWRENBKPEN位来 打开电源 和 后备接口 的时钟
  • 电源控制寄存器(PWR_CR)DBP位来使能对 后备寄存器和 RTC的访问。

RTC框图 

 简图:

可选择三种RTC时钟源:

  • HSE时钟除以128(通常为8MHz/128
  • LSE振荡器时钟(通常为32.768KHz
  • LSI振荡器时钟(40KHz

32位的可编程计数器(CNT),可对应Unix时间戳的秒计数器。

Unix时间戳(格林威治 -- 北京时间)是从197011日的 00:00:00(UTC/GMT的午夜)开始所经过的秒数,不考虑闰秒。

20位的可编程预分频器,可适配不同频率的输入时钟。

(参考手册)图: 

RTC寄存器 

  • 备份域控制寄存器 (RCC_BDCR)

  • RTC控制寄存器高位(RTC_CRH) 

  • RTC控制寄存器低位(RTC_CRL) 

  •  RTC预分频装载寄存器(RTC_PRLH/RTC_PRLL) 

  • RTC预分频器余数寄存器(RTC_DIVH / RTC_DIVL) 

RTC库函数 

  • 获取时间、日期、闹钟的函数: 

  •  设置时间、日期、闹钟的函数:

  • 获取 RTC控制寄存器中的RTOFF位是否置1:可以进行写操作

RTC驱动步骤

 小实验1:读写RTC时间实验

实验目的

配置RTC来实现,设置时钟和日期 

硬件清单

 stm32f103c8t6、USB转TTL、ST-Link

实验代码 

  • rtc.c文件代码:
#include "rtc.h"
#include "stdio.h"

RTC_HandleTypeDef rtc_handle = {0};

void rtc_init(void){
    //要想读写BKP  
    //1.要开始电源和后背寄存器的时钟
    __HAL_RCC_PWR_CLK_ENABLE();   
    __HAL_RCC_BKP_CLK_ENABLE();
    //使能后备寄存器
    HAL_PWR_EnableBkUpAccess();
    
    rtc_handle.Instance = RTC;
    rtc_handle.Init.AsynchPrediv = 32767;
    rtc_handle.Init.OutPut = RTC_OUTPUTSOURCE_NONE;    
    HAL_RTC_Init(&rtc_handle);
}

void HAL_RTC_MspInit(RTC_HandleTypeDef *hrtc){
    if(hrtc->Instance == RTC){
       __HAL_RCC_RTC_ENABLE();
        //配置外部时钟源(3种):LSE,HSE,LSI
        RCC_OscInitTypeDef osc_initstruct = {0};
        
        osc_initstruct.OscillatorType = RCC_OSCILLATORTYPE_LSE;
        osc_initstruct.LSEState = RCC_LSE_ON;
        osc_initstruct.PLL.PLLState = RCC_PLL_NONE;
        HAL_RCC_OscConfig(&osc_initstruct);
        
        //配置RTC选择的外部时钟源
        RCC_PeriphCLKInitTypeDef periph_initstruct = {0};
        periph_initstruct.PeriphClockSelection = RCC_PERIPHCLK_RTC;
        periph_initstruct.RTCClockSelection = RCC_RTCCLKSOURCE_LSE;
        
        HAL_RCCEx_PeriphCLKConfig(&periph_initstruct);
        
        HAL_NVIC_SetPriority(RTC_Alarm_IRQn,2,2);
        HAL_NVIC_EnableIRQ(RTC_Alarm_IRQn);
    }
}

/*读后备寄存器中的数据,并返回*/
uint16_t rtc_read_bkr(uint8_t bkrx){
    
    uint32_t data = 0;
    data = HAL_RTCEx_BKUPRead(&rtc_handle, bkrx);
    return (uint16_t)data;
}

/*对后备寄存器中写数据,指定参数:哪个寄存器,写入的数据*/
void rtc_write_bkr(uint8_t bkrx,uint16_t data){
    HAL_RTCEx_BKUPWrite(&rtc_handle,bkrx,data);
}

/*获取时间的函数*/
void rtc_get_time(void){
    RTC_TimeTypeDef rtc_time = {0};
    RTC_DateTypeDef rtc_date = {0};
    
    HAL_RTC_GetTime(&rtc_handle,&rtc_time,RTC_FORMAT_BIN);
    HAL_RTC_GetDate(&rtc_handle,&rtc_date,RTC_FORMAT_BIN);
    
    printf("rtc time: %d-%02d-%02d;%02d:%02d:%02d\r\n",rtc_date.Year + 2000,rtc_date.Month,rtc_date.Date,rtc_time.Hours,rtc_time.Minutes,rtc_time.Seconds);
}

/*设置时间的函数*/
void rtc_set_time(struct tm time_data){
    RTC_TimeTypeDef rtc_time = {0};
    RTC_DateTypeDef rtc_date = {0};
    
    rtc_time.Hours = time_data.tm_hour;
    rtc_time.Minutes = time_data.tm_min;
    rtc_time.Seconds = time_data.tm_sec;
    
    HAL_RTC_SetTime(&rtc_handle,&rtc_time,RTC_FORMAT_BIN);
    
    rtc_date.Year = time_data.tm_year - 2000;
    rtc_date.Month = time_data.tm_mon;
    rtc_date.Date = time_data.tm_mday;
    HAL_RTC_SetDate(&rtc_handle,&rtc_date,RTC_FORMAT_BIN);
    
    while(!__HAL_RTC_ALARM_GET_FLAG(&rtc_handle,RTC_FLAG_RTOFF));
}

注意事项: 在MspInit()函数中,初始化:三个时钟源和RTC的时钟源:

  •  HAL_RCC_OscConfig( ); //三个时钟源的选择
  •  HAL_RCCEx_PeriphCLKConfig( ); //RTC时钟源的选择

在设置时间的函数中,要判断RCC控制寄存器中的RTOFF的标志位是否置1:可以读写。

上面代码的意思是:若RTOFF位没有置1,进行等待。

  • rtc.h文件代码:
#ifndef __RTC_H__
#define __RTC_H__
#include "stm32f1xx.h"
#include "time.h"

void rtc_init(void);
uint16_t rtc_read_bkr(uint8_t bkrx);
void rtc_write_bkr(uint8_t bkrx,uint16_t data);

void rtc_get_time(void);
void rtc_set_time(struct tm time_data);
void rtc_set_alarm(struct tm time_date);
#endif

  • mian.c文件代码
#include "sys.h"
#include "led.h"
#include "delay.h"
#include "uart1.h"
#include "rtc.h"

int main(void)
{
    HAL_Init();                         /* 初始化HAL库 */
    stm32_clock_init(RCC_PLL_MUL9);     /* 设置时钟, 72Mhz */
    led_init();                         /* LED初始化 */
    uart1_init(115200);
    printf("hello,world\r\n");
    rtc_init();
    
    if(rtc_read_bkr(1) != 0xABED){
        rtc_write_bkr(1,0xABED);
        printf("读出来的十六进制的数是:%X\r\n",rtc_read_bkr(1));
        struct tm time_data;
    
        time_data.tm_year = 2025;
        time_data.tm_mon = 5;
        time_data.tm_mday = 30;
        time_data.tm_hour = 11;
        time_data.tm_min  = 31;
        time_data.tm_sec = 0;
        rtc_set_time(time_data);  
    }
    while(1)
    { 
        rtc_get_time();
        delay_ms(1000);
    }
}

总结:要复习函数中的形参传入结构体知识点。 

小实验2:RTC闹钟实验

     实验目的

     设置闹钟

    实验代码

    • MspInit()函数中添加:

    •  中断服务函数

    •  设置闹钟的实验

    •  rtc.h文件代码
    #ifndef __RTC_H__
    #define __RTC_H__
    #include "stm32f1xx.h"
    #include "time.h"
    
    void rtc_init(void);
    uint16_t rtc_read_bkr(uint8_t bkrx);
    void rtc_write_bkr(uint8_t bkrx,uint16_t data);
    
    void rtc_get_time(void);
    void rtc_set_time(struct tm time_data);
    void rtc_set_alarm(struct tm time_date);
    #endif
    
    
    • mian.c文件代码 
    #include "sys.h"
    #include "led.h"
    #include "delay.h"
    #include "uart1.h"
    #include "rtc.h"
    
    
    int main(void)
    {
        HAL_Init();                         /* 初始化HAL库 */
        stm32_clock_init(RCC_PLL_MUL9);     /* 设置时钟, 72Mhz */
        led_init();                         /* LED初始化 */
        uart1_init(115200);
        printf("hello,world\r\n");
        rtc_init();
        
        if(rtc_read_bkr(1) != 0xABED){
            rtc_write_bkr(1,0xABED);
            printf("读出来的十六进制的数是:%X\r\n",rtc_read_bkr(1));
            struct tm time_data;
        
            time_data.tm_year = 2025;
            time_data.tm_mon = 5;
            time_data.tm_mday = 30;
            time_data.tm_hour = 11;
            time_data.tm_min  = 31;
            time_data.tm_sec = 0;
            rtc_set_time(time_data);  
            
            struct tm alarm_date;
            
            alarm_date.tm_hour = 11;
            alarm_date.tm_min = 31;
            alarm_date.tm_sec = 20;
            
            rtc_set_alarm(alarm_date);
        }
        while(1)
        { 
            rtc_get_time();
            delay_ms(1000);
        }
    }
    
    

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

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

    相关文章

    【EdgeYOLO】《EdgeYOLO: An Edge-Real-Time Object Detector》

    Liu S, Zha J, Sun J, et al. EdgeYOLO: An edge-real-time object detector[C]//2023 42nd Chinese Control Conference (CCC). IEEE, 2023: 7507-7512. CCC-2023 源码:https://github.com/LSH9832/edgeyolo 论文:https://arxiv.org/pdf/2302.07483 …

    调试技巧总结

    目录 一.调试1.什么是调试2.调试语义的分类2.1 静态语义2.2 动态语义 二.实用的调试技巧1.屏蔽代码2.借助打印3.查看汇编代码4.调试技巧总结 一.调试 1.什么是调试 调试,通俗易懂地说就是不断排查代码的错误,进行修正的过程,在写代码的时候…

    ubuntu安装blender并配置应用程序图标

    ubuntu安装blender并配置应用程序图标 下载blender安装包解压缩并安装启动blender添加应用程序启动图标 下载blender安装包 blender中文服务站的下载网址 这里选择Linux 64位的Blender 4.2.4 LTS。下载速度很快。下载得到 解压缩并安装 将下载的压缩包放在/opt目录下&#…

    基于LBS的上门代厨APP开发全流程解析

    上门做饭将会取代外卖行业成为新一轮的创业风口吗?杭州一位女士的3菜一汤88元套餐引爆社交网络,这个包含做饭、洗碗、收拾厨房的全套服务,正在重新定义"到家经济"的边界。当25岁的研究生系着围裙出现在客户厨房,当年轻姑…

    Redisson学习专栏(三):高级特性与实战(Spring/Spring Boot 集成,响应式编程,分布式服务,性能优化)

    文章目录 前言一、Spring Boot深度整合实战1.1 分布式缓存管理1.2 声明式缓存1.3 响应式编程 二、分布式服务治理2.1 服务端实现2.2 客户端调用2.3 高级特性2.4 服务治理功能 三、分布式任务调度引擎四、连接池配置与网络参数调优4.1 连接池配置4.2 网络参数调优4.3 集群模式特…

    华为欧拉系统中部署FTP服务与Filestash应用:实现高效文件管理和共享

    华为欧拉系统中部署FTP服务与Filestash应用:实现高效文件管理和共享 前言一、相关服务介绍1.1 Huawei Cloud EulerOS介绍1.2 Filestash介绍1.3 华为云Flexus应用服务器L实例介绍二、本次实践介绍2.1 本次实践介绍2.2 本次环境规划三、检查云服务器环境3.1 登录华为云3.2 SSH远…

    基于Docker和YARN的大数据环境部署实践最新版

    基于Docker和YARN的大数据环境部署实践 目的 本操作手册旨在指导用户通过Docker容器技术,快速搭建一个完整的大数据环境。该环境包含以下核心组件: Hadoop HDFS/YARN(分布式存储与资源调度)Spark on YARN(分布式计算…

    【大模型】Bert

    一、背景与起源 上下文建模的局限:在 BERT 之前,诸如 Word2Vec、GloVe 等词向量方法只能给出静态的词表示;而基于单向或浅层双向 LSTM/Transformer 的语言模型(如 OpenAI GPT)只能捕捉文本从左到右(或右到…

    3 分钟学会使用 Puppeteer 将 HTML 转 PDF

    需求背景 1、网页存档与文档管理 需要将网页内容长期保存或归档为PDF,确保内容不被篡改或丢失,适用于法律文档、合同、技术文档等场景。PDF格式便于存储和检索。 2、电子报告生成 动态生成的HTML内容(如数据分析报告、仪表盘)需导出为PDF供下载或打印。PDF保留排版和样…

    速通《Sklearn 与 TensorFlow 机器学习实用指南》

    1.机器学习概览 1.1 什么是机器学习 机器学习是通过编程让计算机从数据中进行学习的科学。 1.2 为什么使用机器学习? 使用机器学习,是为了让计算机通过数据自动学习规律并进行预测或决策,无需显式编程规则。 1.3 机器学习系统的类型 1.…

    Ubuntu 下搭建ESP32 ESP-IDF开发环境,并在windows下用VSCode通过SSH登录Ubuntu开发ESP32应用

    Ubuntu 下搭建ESP32 ESP-IDF开发环境,网上操作指南很多,本来一直也没有想过要写这么一篇文章。因为我其实不太习惯在linux下开发应用,平时更习惯windows的软件操作,只是因为windows下开发ESP32的应用编译时太慢,让人受…

    NodeMediaEdge接入NodeMediaServer

    如何使用NME接入NMS 简介 NodeMediaEdge是一款部署在监控摄像机网络前端中,拉取Onvif或者rtsp/rtmp/http视频流并使用rtmp/kmp推送到公网流媒体服务器的工具。 通过云平台协议注册到NodeMediaServer后,可以同NodeMediaServer结合使用。使用图形化的管理…

    【Java基础-环境搭建-创建项目】IntelliJ IDEA创建Java项目的详细步骤

    在Java开发的世界里,选择一个强大的集成开发环境(IDE)是迈向高效编程的第一步。而IntelliJ IDEA无疑是Java开发者中最受欢迎的选择之一。它以其强大的功能、智能的代码辅助和简洁的用户界面,帮助无数开发者快速构建和部署Java项目…

    PHP7+MySQL5.6 查立得源码授权系统DNS验证版

    # PHP7MySQL5.6 查立得源码授权系统DNS验证版 ## 一、系统概述 本系统是一个基于PHP7和MySQL5.6的源码授权系统,使用DNS TXT记录验证域名所有权,实现对软件源码的授权保护。 系统支持多版本管理,可以灵活配置不同版本的价格和下载路径&#…

    【QQ音乐】sign签名| data参数加密 | AES-GCM加密 | webpack (下)

    1.目标 网址&#xff1a;https://y.qq.com/n/ryqq/toplist/26 我们知道了 sign P(n.data)&#xff0c;其中n.data是明文的请求参数 2.webpack生成data加密参数 那么 L(n.data)就是密文的请求参数。返回一个Promise {<pending>}&#xff0c;所以L(n.data) 是一个异步函数…

    3D虚拟工厂

    1、在线体验 3D虚拟工厂在线体验 vue3three.jsblender 2、功能介绍 1. 全屏显示功能2. 镜头重置功能3. 企业概况信息模块4. 标签隐藏/显示功能5. 模型自动旋转功能6. 办公楼分层分解展示7. 白天/夜晚 切换8. 场景资源预加载功能9. 晴天/雨天/雾天10. 无人机视角模式11. 行人…

    http传输协议的加密

    创建目录存放签证 [rootserver100 ~]# mkdir /etc/nginx/certs [rootserver100 ~]# openssl req -newkey rsa:2048 -nodes -sha256 -keyout /etc/nginx/certs/timinglee.org.key -x509 -days 365 -out /etc/nginx/certs/timinglee.org.crt ..................................…

    半导体晶圆制造洁净厂房的微振控制方案-江苏泊苏系统集成有限公司

    半导体晶圆制造洁净厂房的微振控制方案-江苏泊苏系统集成有限公司 微振控制在现行国家标准《电子工业洁净厂房设计规范》GB50472中有关微振控制的规定主要有&#xff1a;洁净厂房的微振控制设施的设计分阶段进行&#xff0c;应包括设计、施工和投产等各阶段的微振测试、厂房建…

    常见压缩算法性能和压缩率对比 LZ4 LZO ZSTD SNAPPY

    网传压缩算法对比表 算法压缩率压缩速度解压速度支持流式压缩适用场景LZ4低极快极快是实时数据压缩、日志压缩、内存缓存等Zstandard高快快是文件压缩、网络传输、数据库备份等Brotli很高中等快是静态资源压缩&#xff08;HTML、CSS、JS&#xff09;等LZO低极快快是嵌入式系统…

    Spring Boot 应用中实现配置文件敏感信息加密解密方案

    Spring Boot 应用中实现配置文件敏感信息加密解密方案 背景与挑战 &#x1f6a9;一、设计目标 &#x1f3af;二、整体启动流程 &#x1f504;三、方案实现详解 ⚙️3.1 配置解密入口&#xff1a;EnvironmentPostProcessor3.2 通用解密工具类&#xff1a;EncryptionTool 四、快速…