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

STM32单片机-BKP和RTC

STM32单片机-BKP和RTC

  • 一、Unix时间戳
    • 1.1 时间戳转换
  • 二、BKP(备份寄存器)
  • 三、RTC(实时时钟)
    • 3.1 RTC工作原理
  • 四、代码部分
    • 4.1 BKP备份寄存器
    • 4.2 RTC实时时钟

一、Unix时间戳

  • Unix时间戳定义为从伦敦时间1970年1月1日0时0分0秒开始所经过的秒数,不考虑闰秒
  • 时间戳存储在一个秒计数器中,秒计数器为32位/64位整形变量
  • 世界上所有时区秒计数器相同,不同时区通过添加偏移来得到当地时间

在这里插入图片描述

1.1 时间戳转换

  • C语言的time.h模块提供了时间获取时间戳转换的相关函数,可以方便地进行秒计数器、时器时间和字符串之间的转换

在这里插入图片描述
  下图为时间戳转换图
  time_t是int32或者int64数据类型,存储时间戳中自增的秒数
  struct tm是一个封装的结构体类型,结构体成员是秒,分,时等
  char*指向一个表示时间的字符串

在这里插入图片描述

二、BKP(备份寄存器)

  • BKP用于存储用户应用程序数据,当VDD主电源被切断,它们仍然由VBRT备用电源维持供电,当系统在待机模式下被唤醒,或系统复位或电源复位时,它们也不会被复位
  • TAMPER引脚产生的侵入事件将所有备份寄存器内容清除
  • RTC引脚输出RTC校准时钟RTC闹钟脉冲或者秒脉冲
  • 存储RTC时钟校准寄存器
  • 用户数据存储容量20字节/84字节

  下图为BKP基本结构
  BKP由数据寄存器-存储数据、控制寄存器、状态寄存器和RTC时钟校准寄存器组成,TAMPER有上升沿或者下降沿出现时,清除寄存器内容,保证安全

在这里插入图片描述

三、RTC(实时时钟)

  • RTC是一个独立的定时器,可以为系统提供时钟日历的功能
  • RTC时钟配置处于系统的后备区域系统复位时数据不清零,VDD断电后可以借助VBRT供电继续走时
  • 32位的可编程计数器,可对应Unix时间戳秒计数器
  • 20位的可编程预分频器,可适配不同频率的输入时钟
  • 可选择三种RTC时钟源:HSE时钟除以128(8MHz/128)、LSE振荡器时钟(VBRT)(32.768KHz)和LSI振荡器时钟(40KHz)

3.1 RTC工作原理

  下图为RTC框图
  灰色填充部分属于后备区域,在主电源掉电后,可以使用备用电源维持工作。RTCCLK是时钟源,一般选择LSE振荡器时钟,经过分频器分频,重装载的值加1就是分频系数。32位可编程计数器RTC_CNT存放时间数据,RTC_ALR是闹钟寄存器
  进入中断有3个中断源,RTC_Second秒中断,每秒进一次中断。RTC_Overflow溢出中断,当32位计数器溢出进入中断。RTC_Alarm闹钟中断

在这里插入图片描述

  下图为RTC基本结构
  选择时钟,RTCCKL先通过预分频器对时钟进行预分频,重装寄存器是计数目标,决定分频值余数寄存器是一个自减计数器,存储当前的计数值,最终得到1Hz的秒计数信号通向32位计数器,1秒自增一次,同时可以设定闹钟,最终也可以进入中断

在这里插入图片描述

四、代码部分

  注意事项

  • RTC的时钟开启没有单独选项,需要开启PWR和BKP时钟,设置PWR_CR的DBP,使能对BKP和RTC的访问
  • 若在读取RTC寄存器时,RTC的APB1接口曾经处在禁止状态,则软件首先必须等待RTC_CRL寄存器中的RSF位(寄存器同步标志)被硬件置1
  • 必须设置RTC_CRL寄存器中的CNF位,使RTC进入配置模式后,才能写入RTC_PRL(重装值)、RTC_CNT(计数器)、RTC_ALR(闹钟值)寄存器
  • 对RTC任何寄存器的写操作,都必须在前一次写操作结束后进行。可以通过查询RTC_CR寄存器中的RTOFF状态位,判断RTC寄存器是否处于更新中。仅当RTOFF状态位是1时,才可以写入RTC寄存器

4.1 BKP备份寄存器

  STM32的VBRT引脚接在ST-Link的3.3V,PB1接一个按键
  步骤:开启RCC(PWR和BKP) — 使能BKP-PWR_BackupAccessCmd(ENABLE) — 写入和读取
  下面为main.c

#include "stm32f10x.h"                  // Device header
#include "OLED.h"
#include "Key.h"uint8_t KeyNum;
uint16_t ArrayWrite[] = {0x1234,0x5678};
uint16_t ArrayRead[2];int main(void)
{OLED_Init();Key_Init();//开启RCC-PWR和BKPRCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP,ENABLE);//使能BKPOLED_ShowString(1,1,"W:");OLED_ShowString(2,1,"R:");while(1){KeyNum = Key_GetNum();if(KeyNum == 1){ArrayWrite[0] ++;ArrayWrite[1] ++;BKP_WriteBackupRegister(BKP_DR1,ArrayWrite[0]);BKP_WriteBackupRegister(BKP_DR2,ArrayWrite[1]);//写入OLED_ShowHexNum(1,3,ArrayWrite[0],4);OLED_ShowHexNum(1,8,ArrayWrite[1],4);}ArrayRead[0] = BKP_ReadBackupRegister(BKP_DR1);ArrayRead[1] = BKP_ReadBackupRegister(BKP_DR2);//读取OLED_ShowHexNum(2,3,ArrayRead[0],4);OLED_ShowHexNum(2,8,ArrayRead[1],4);}
}

4.2 RTC实时时钟

  下面是RTC相关的库函数

void RCC_LSEConfig(uint8_t RCC_LSE);//启动LSE时钟
void RCC_LSICmd(FunctionalState NewState);//配置LSI时钟
void RCC_RTCCLKConfig(uint32_t RCC_RTCCLKSource);//选择RTCCLK的时钟源
void RCC_RTCCLKCmd(FunctionalState NewState);//RTCCLK使能
FlagStatus RCC_GetFlagStatus(uint8_t RCC_FLAG);//等待RCC标志位,时钟启动完成void RTC_EnterConfigMode(void);//设置CNF位,使RTC进入配置模式
void RTC_ExitConfigMode(void);//退出配置模式
uint32_t  RTC_GetCounter(void);//获取CNT计数器的值
void RTC_SetCounter(uint32_t CounterValue);//写入计数器的值
void RTC_SetPrescaler(uint32_t PrescalerValue);//写入预分频器
void RTC_SetAlarm(uint32_t AlarmValue);//写入闹钟值
uint32_t  RTC_GetDivider(void);//读取余数寄存器值
void RTC_WaitForLastTask(void);//等待上一次操作完成
void RTC_WaitForSynchro(void);//等待同步
FlagStatus RTC_GetFlagStatus(uint16_t RTC_FLAG);//等待标志位

  步骤:开启PWRBKP时钟,使能BKP和RTC的访问 — 开启LSE时钟并等待LSE时钟启动完成 — 选择RCCCLK时钟源-LSE — 等待同步和等待上一次写入操作完成 — 设置预分频器初始时间写入数据读取数据
  其中想要实现复位不重置时间数据和主电源断开时不切断时间,需要使用BKP判断
  利用c语言库time.h中的函数,实现写入和读取时间,即计数器值和时间数据的相互转换
  下面为MyRTC.c和MyRTC.h

#include "stm32f10x.h"                  // Device header
#include "MyRTC.h"                  // Device header
#include <time.h>uint16_t MyRTC_Time[] = {2024,1,1,1,1,1};//时间数据void MyRTC_Init()
{//开启RCC-BKP、PWR和使能访问RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP,ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE);PWR_BackupAccessCmd(ENABLE);//第一次上电或者完全断电,BKPDR1默认为0时,才会初始化时间if(BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5){//开启LSE时钟,等待LSE时钟启动完成(标志位)RCC_LSEConfig(RCC_LSE_ON);while(RCC_GetFlagStatus(RCC_FLAG_LSERDY) == RESET);//选择RCCCLK时钟源-LSERCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);RCC_RTCCLKCmd(ENABLE);//等待同步和等待上一次写入操作完成RTC_WaitForSynchro();RTC_WaitForLastTask();//配置预分频器 =  32768MHz /32768 = 1HzRTC_SetPrescaler(32768-1);RTC_WaitForLastTask();//等待写入操作//设置初始时间MyRTC_SetTime();BKP_WriteBackupRegister(BKP_DR1,0xA5A5);//DR1写入A5A5}else{//等待同步和等待上一次写入操作完成RTC_WaitForSynchro();RTC_WaitForLastTask();}
}
/*
@brief:写入时间数据
*/
void MyRTC_SetTime()
{time_t time_cnt;//计数器值struct tm time_data;//时间数据time_data.tm_year = MyRTC_Time[0] - 1900;time_data.tm_mon = MyRTC_Time[1] - 1;time_data.tm_mday = MyRTC_Time[2];time_data.tm_hour = MyRTC_Time[3];time_data.tm_min = MyRTC_Time[4];time_data.tm_sec = MyRTC_Time[5];//写入数据time_cnt = mktime(&time_data)- 8*60*60;//时间数据--计数器(北京时间)RTC_SetCounter(time_cnt);RTC_WaitForLastTask();//等待写入操作
}
/*
@brief:读取时间数据
*/
void MyRTC_ReadTime()
{time_t time_cnt;//计数器值struct tm time_data;//时间数据time_cnt = RTC_GetCounter()+ 8*60*60;//读取时间存到计数器(北京时间)time_data = *localtime(&time_cnt);//将时间数据转到结构体MyRTC_Time[0] = time_data.tm_year + 1900;MyRTC_Time[1] = time_data.tm_mon + 1;MyRTC_Time[2] = time_data.tm_mday;MyRTC_Time[3] = time_data.tm_hour;MyRTC_Time[4] = time_data.tm_min;MyRTC_Time[5] = time_data.tm_sec;//读取数据
}
#ifndef __MYRTC_H
#define __MYRTC_Hextern uint16_t MyRTC_Time[];void MyRTC_Init();
void MyRTC_SetTime();
void MyRTC_ReadTime();#endif

  下面为main.c

#include "stm32f10x.h"                  // Device header
#include "OLED.h"
#include "MyRTC.h"int main(void)
{MyRTC_Init();OLED_Init();OLED_ShowString(1,1,"Data:XXXX-XX-XX");OLED_ShowString(2,1,"Time:XX:XX:XX");OLED_ShowString(3,1,"CNT:");while(1){MyRTC_ReadTime();OLED_ShowNum(1,6,MyRTC_Time[0],4);OLED_ShowNum(1,11,MyRTC_Time[1],2);OLED_ShowNum(1,14,MyRTC_Time[2],2);OLED_ShowNum(2,6,MyRTC_Time[3],2);OLED_ShowNum(2,9,MyRTC_Time[4],2);OLED_ShowNum(2,12,MyRTC_Time[5],2);OLED_ShowNum(3,6,RTC_GetCounter(),10);}
}

相关文章:

  • 如何级联移位寄存器(74HC595)
  • 【Linux】基础IO——文件描述符,重定向,FILE
  • WordPress 技巧:如何限制或取消自动清空回收站功能
  • 怎样去掉卷子上的答案并打印
  • mac下Xcode在iphone真机上测试运行iOS软件
  • [信号与系统]有关时域信号与频域信号的转换
  • 红队内网攻防渗透:内网渗透之内网对抗:隧道技术篇防火墙组策略FRPNPSChiselSocks代理端口映射C2上线
  • 力扣85.最大矩形
  • 【深度学习驱动流体力学】VTK创建、处理和可视化流体数据
  • 路由的params参数,命名路由,路由的params参数,命名路由
  • 架构师指南:现代 Datalake 参考架构
  • 深入理解Java虚拟机(JVM)中的垃圾回收器
  • VUE3 使用 vite-plugin-svg-icons加载SVG
  • 浅谈请求中数据转换
  • 程序猿成长之路之数据挖掘篇——决策树分类算法(1)——信息熵和信息增益
  • 【跃迁之路】【444天】程序员高效学习方法论探索系列(实验阶段201-2018.04.25)...
  • Android开发 - 掌握ConstraintLayout(四)创建基本约束
  • ES6之路之模块详解
  • GitUp, 你不可错过的秀外慧中的git工具
  • iOS筛选菜单、分段选择器、导航栏、悬浮窗、转场动画、启动视频等源码
  • LeetCode29.两数相除 JavaScript
  • mysql外键的使用
  • PAT A1120
  • React中的“虫洞”——Context
  • Spring Cloud中负载均衡器概览
  • VuePress 静态网站生成
  • 阿里云爬虫风险管理产品商业化,为云端流量保驾护航
  • 阿里云前端周刊 - 第 26 期
  • 不用申请服务号就可以开发微信支付/支付宝/QQ钱包支付!附:直接可用的代码+demo...
  • 从@property说起(二)当我们写下@property (nonatomic, weak) id obj时,我们究竟写了什么...
  • 我的面试准备过程--容器(更新中)
  • 写给高年级小学生看的《Bash 指南》
  • 译米田引理
  • 在Mac OS X上安装 Ruby运行环境
  • 追踪解析 FutureTask 源码
  • ​无人机石油管道巡检方案新亮点:灵活准确又高效
  • ‌U盘闪一下就没了?‌如何有效恢复数据
  • # 安徽锐锋科技IDMS系统简介
  • (13)[Xamarin.Android] 不同分辨率下的图片使用概论
  • (4) openssl rsa/pkey(查看私钥、从私钥中提取公钥、查看公钥)
  • (NSDate) 时间 (time )比较
  • (编程语言界的丐帮 C#).NET MD5 HASH 哈希 加密 与JAVA 互通
  • (附源码)spring boot网络空间安全实验教学示范中心网站 毕业设计 111454
  • (附源码)springboot美食分享系统 毕业设计 612231
  • (黑马点评)二、短信登录功能实现
  • (每日一问)设计模式:设计模式的原则与分类——如何提升代码质量?
  • (贪心) LeetCode 45. 跳跃游戏 II
  • (已解决)vue+element-ui实现个人中心,仿照原神
  • (转)Android学习笔记 --- android任务栈和启动模式
  • ****Linux下Mysql的安装和配置
  • *p++,*(p++),*++p,(*p)++区别?
  • .Net 中Partitioner static与dynamic的性能对比
  • .NET6 开发一个检查某些状态持续多长时间的类
  • .NET中GET与SET的用法
  • //解决validator验证插件多个name相同只验证第一的问题