STM32基础篇:RTC × Unix时间戳 × BKP

embedded/2024/11/13 9:10:33/

Unix时间戳

最早是在Unix系统使用的,之后很多由Unix演变而来的系统也都继承了Unix时间戳的规定。目前,Linux、Windows、安卓这些系统,其底层的计时系统都是使用Unix时间戳。

Uinx时间戳(Unix Timestamp)定义为从UTC/GMT的1970年1月1日0时0分0秒开始所经过的秒数,不考虑闰秒。也就是说时间戳是一个秒计数器,且只记秒,不会向分钟、小时进位。

其年月日小时分钟均通过计算秒得出。

世界上所有时区共用一个时间戳的秒计数器,不同时区通过添加偏移来得到当地时间。如上图所示,相比于伦敦,北京时间偏移了8个小时。


UTC/GMT

GMT格林尼治标准时间,可理解为伦敦时间,是一种以地球自转为基础的时间计量系统。它将地球自转一周的时间间隔等分为24小时,以此确定计时标准。

但GMT是以前的时间标准,因为地球自转一周的时间是不固定的,由于潮汐力、地球活动等原因,地球目前是越转越慢的,因此时间基准也是不固定的(即1秒到底是多长)。

为了时间的定义更标准,科学家又提出来新的计时系统,叫做UTC,即协调世界时。是一种以原子钟为基础的时间计量系统。它规定铯133原子基态的两个超精细能级间在零磁场下跃迁辐射9192631770周所持续的时间为1秒。

当原子钟计时一天的时间与地球自转一周的时间相差超过0,9秒时,UTC会执行闰秒来保证其计时与地球自转的协调一致。

闰秒:计时标准是恒定不变的,但是地球越转越慢,误差超过0.9秒时,计时系统就多走一秒来等一下地球的自转。


时间戳转换

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

函数

作用
time_t time(time_t *seconds)获取系统时钟

struct tm *gmtime(const time_t *timer)

秒计数器转换为日期时间(格林尼治时间)

struct tm *localtime(const time_t *timer)

秒计数器转换为日期时间(当地时间)

time_t mktime(struct tm *timeptr)

日期时间转换为秒计数器(当地时间)

char *ctime(const time_t *timer)

秒计数器转换为字符串(默认格式)

char *asctime(const struct tm *timeptr)

日期时间转换为字符串(默认格式)

size_t strftime(char *str, size_t maxsize, const char *format, const struct tm *timeptr)

日期时间转换为字符串(自定义格式)

数据类型:

1、秒计数器时间类型:time_t,64位有符号的整型数据,用于存储时间戳中一直自增的秒数。

2、日期时间类型:struct tm,是一个结构体,具体如下:

3、字符串时间类型:char*,就是char类型数据的指针。用来指向一个表示时间的字符串。


BKP简介

BKP备份寄存器,可用于存储用户应用程序数据,当VDD(2.0~3.6V)电源被切断,他们仍然由VBAT(1.8~3.6V)维持供电。当系统在待机模式下被唤醒,或系统复位或电源复位时,他们也不会被复位(除非VBAT掉电)。

用户数据存储容量:

20字节中容量和小容量
84字节大容量和互联型

对于C8T6芯片,为中容量,对应20字节。可以看出BKP的容量其实非常小,一般只能存储少量的参数。


BKP基本结构

  • 后备区域

图中的橙色部分,我们可以叫做后备区域。注意,是BKP处于后备区域,但后备区域不只有BKP,还有RTC的相关电路。

STM32后备区域的特性就是:当VDD主电源掉电时,后备区域仍然可以由VBAT的备用电池供电;当VDD上电时,后备区域供电会由VBAT切换为VDD,可以节省电池电量。

  • 内部寄存器

BKP里主要有状态寄存器、控制寄存器、数据寄存器、RTC时钟校准寄存器。其中数据寄存器是主要部分,用来存储数据的。每个数据寄存器都是16位的(一个数据寄存器可以存2个字节),对于中容量和小容量的设备,里面有DR1~DR10总共10个数据寄存器,共20个字节。

  • 侵入检测

BKP可以从PC13位置的TAMPER引脚引入一个检测信号,当TAMPER产生一个上升沿或者下降沿时,会清除BKP所有的内容,以保证安全。

  • 时钟输出

将RTC的相关时钟从PC13位置的RTC引脚输出出去,供外部使用。其中,输出校准时钟时,再配合RTC时钟校准寄存器,可以对RTC的误差进行校准。


RTC简介

RTC即实时时钟,STM32中RTC是一个独立的定时器,可为系统提供时钟和日历(年月日时分秒)的功能。挂在APB1总线上。

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

  • 秒计数器与分频器

在RTC中,负责计数的装置只有一个32位的秒计数器。且1秒就要自增一次,所以驱动秒计数器的时钟---TR_CLK是一个1Hz的信号。但实际提供给RTC的时钟,也就是RTCCLK,一般频率都比较高。所以显然需要在这之间加一个分频器,保证输出给计数器的频率为1Hz。

RTC中的预分频器,是一个20位的分频器(可编程),可以选择对输入时钟进行1~2^20分频,这样可以适配不同频率的输入时钟。

  • RTC时钟来源

RTCCLK的来源主要有以下3种:

HSE时钟除以1288MHz/128
LSE振荡器时钟37.768KHz
LSI振荡器时钟40KHz

其中HSE是高速外部时钟信号(通常为高速晶振)、LSE低速外部时钟信号、LSI低速内部时钟信号。具体可参考RCC章节知识。

【注意】为了更好使用RTC模块,外部硬件电路也需要配置好。即备用电池模块、外部低速晶振等。


RTC框图

左边阴影区为核心的分频和计数区域,右边为中断输出使能和NVIC部分,上面为APB1总线读写部分,下面阴影区是与PWR关联的部分。(阴影区均为后备区域)

【注意】对于RTC和BKP的寄存器,其实都是16位的。例如左边的RTC_CR控制寄存器,实际上是两个16位的寄存器(RTC_CRH和RTC_CRL)。

  • 分频器

由两个寄存器组成,即重装载寄存器RTC_PRL余数寄存器RTC_DIV,与定时器时基单元的计数器CNT和重装寄存器ARR是一样作用。(且计几个数,溢出一次,就是几分频)

RTC_DIV是自减的计数器,每来一个输入时钟,DIV的值自减一次,自减到0时,再来一个时钟,DIV输出一个脉冲(溢出信号),同时从PRL获取重装值,回到重装值继续自减。

  • 闹钟寄存器

RTC内部除了32位的秒寄存器(RTC_CNT)外,还有一个32位的闹钟寄存器(RTC_ALR),其作用为设置闹钟。

我们可以在ALR里写一个秒数,设定闹钟,当CNT值与ALR设定的闹钟值一样时,就会产生RTC_Alarm闹钟信号,通向右边的中断系统。在中断函数里,可执行需要的操作。

【注意】RTC_Alarm闹钟信号还可使主机退出待机模式。

  • 中断

除了上述的RTC_Alarm闹钟信号能触发中断外,还有RTC_SecondRTC_Overflow能够触发中断。

其中RTC_Second是秒中断,开启后每秒触发一个中断。RTC_Overflow是指当32位的秒计数器计满后溢出时触发的中断,这个中断得到2106年才会触发。


RTC操作注意事项

  • 1、使能

对于一般的片上外设,只需要开启时钟使能即可使用。但对于RTC模块却有些复杂,首先要设置RCC_APB1ENR(开启APB1上外设的时钟),要同时开启PWR和BKP的时钟。然后设置PWR_CR的DBP位,来使能对BKP和RTC的访问。

  • 2、关于读取

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

可观察RTC内部框图,可以看出APB1的时钟为PCLK1,在主电源掉电时会停止;而RTC里的寄存器时钟为RTCCLK。当我们用PCLK1驱动的总线去读取RTCCLK驱动的寄存器时,就会有一个时钟不同步的问题。PCLK1的频率为36MHz,远大于RTCCLK的36KHz,如果在APB1刚开启时就立刻读取RTC寄存器,有可能RTC寄存器还没有更新到APB1总线上,这样读取到的值为错误的。

所以APB1总线刚开启时,要等一下RTCCLK,只要RTCCLK来一个上升沿,RTC把它的寄存器值同步到APB1总线上。

  • 3、关于写入

必须设置RTC_CRL寄存器中的CNF位,使RTC进入配置模式后,才能写入RTC_PRL、RTC_CNT、RTC_ALR寄存器。

在STM32的标准库函数中,写寄存器的函数都自动帮我们加上了这个操作,不需要我们再配置了。

  • 4、写等待机制

对于RTC任何寄存器的写操作,都必须在前一次写操作结束后进行。可以通过查询RTC_CR寄存器中的RTOFF状态位,判断RTC寄存器是否处于更新中。仅当RTOFF为1时,才可以写入RTC寄存器。

因为RCLK1与RTCCLK时钟频率不同,用PCLK的频率写入后,这个值还不能立刻更新到RTC寄存器里面,要等RTCCLK时钟来一个上升沿,值更新到RTC寄存器里,写入才算完成。



http://www.ppmy.cn/embedded/107189.html

相关文章

扑捉一只耿鬼(HTML文件)

图例&#xff1a; 代码&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><title>耿鬼</title><style>body {background: #fff;font-family: Comfortaa, sans-serif;}* {box-sizing:…

滑动窗口序列(单序列双指针)9/5

一、不间断子数组(滑动窗口哈希表) 题意&#xff1a; 给你一个数组nums,现在求子数组中都有 0 < |nums[i1] - nums[i2]| < 2 。这样称一个不间断子数组。&#xff08;简而言之&#xff1a;子数组中最大值和最小值的差距必须<2&#xff09;。求不间断子数组的数量 输…

Android 14(API 级别 34)中,DexClassLoader 不再支持可写 dex/jar 文件

Android 14&#xff08;API 级别 34&#xff09;中&#xff0c;DexClassLoader 不再支持从可写文件加载 dex/jar 文件。这意味着从Android 14开始&#xff0c;你不能再使用 DexClassLoader 来动态加载位于内部存储中的dex/jar文件&#xff0c;除非这些文件被设置为只读。 解决…

2024国赛数学建模A题思路模型代码

2024国赛数学建模思路资料&#xff0c;思路获取见文末名片 数学建模感想 纪念逝去的大学数学建模&#xff1a;两次校赛&#xff0c;两次国赛&#xff0c;两次美赛&#xff0c;一次电工杯。从大一下学期组队到现在&#xff0c;大三下学期&#xff0c;时间飞逝&#xff0c;我的…

Unity数据持久化 之 二进制存储法

本文仅作笔记学习和分享&#xff0c;不用做任何商业用途 本文包括但不限于unity官方手册&#xff0c;unity唐老狮等教程知识&#xff0c;如有不足还请斧正​​ 前置知识&#xff1a;1 Byte 8 bit &#xff0c;所以0000 00001 就是一个字节&#xff0c; 该串数字转为十进制代表1…

2024.8.29 Python,排序算法,列表的append规则

1.append和 res[] nums1[1,2,3] res.append(nums1[1]) print(res)#输出[2] res.append([nums1[1]]) print(res)#输出[[2]] res.append(nums1[1:2]) print(res)#输出[[2]] res.append(nums1[1:3]) print(res)#输出[[2,3]] resnums1[1:3] print(res)#输出[2,3]也就是说&#xff…

【MATLAB源码-第164期】基于matlab的轴承故障三种谱图:细化谱,功率谱,倒谱对比分析仿真。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 轴承故障分析是一种重要的维护和监控手段&#xff0c;能够帮助工程师及时发现和解决轴承在运行中可能遇到的各种问题。在轴承故障诊断中&#xff0c;通常会使用到三种谱图分析方法&#xff1a;细化谱&#xff08;Fine Spectr…

Django国际化和本地化

【图书介绍】《Django 5企业级Web应用开发实战&#xff08;视频教学版&#xff09;》_django 5企业级web应用开发实战(视频教学版)-CSDN博客 《Django 5企业级Web应用开发实战&#xff08;视频教学版&#xff09;》(王金柱)【摘要 书评 试读】- 京东图书 (jd.com) 本节主要介…