【江协科技STM32】读写备份寄存器&RTC实时时钟(学习笔记)

参考相关文章理解:

【江协科技STM32】Unix时间戳(学习笔记)-CSDN博客

【江协科技STM32】BKP备寄存器&RTC实时时钟(学习笔记)_stm32断电保存时钟-CSDN博客

读写备份寄存器

接线图:VBAT是从STLINK的3.3V引出来的,注意不要接到5V的

BKP初始化:

①设置RCC_APB1ENR的PWREN和BKPEN,开启PWR和BKP时钟

②设置PWR_CR的DBP,使能对BKP和RTC的访问

代码较少,就没有对BKP单独进行封装,下面直接写

cs 复制代码
uint8_t KeyNum;					//定义用于接收按键键码的变量

uint16_t ArrayWrite[] = {0x1234, 0x5678};	//定义要写入数据的测试数组
uint16_t ArrayRead[2];						//定义要读取数据的测试数组

int main(void)
{
	OLED_Init();				//OLED初始化
	Key_Init();					//按键初始化
	
	OLED_ShowString(1, 1, "W:");
	OLED_ShowString(2, 1, "R:");
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);		//开启PWR的时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP, ENABLE);		//开启BKP的时钟
	
	/*备份寄存器访问使能*/
	PWR_BackupAccessCmd(ENABLE);							//使用PWR开启对备份寄存器的访问
	
	while (1)
	{
		KeyNum = Key_GetNum();		//获取按键键码
		
		if (KeyNum == 1)			//按键1按下
		{
			ArrayWrite[0] ++;		//测试数据自增,先加加在写入
			ArrayWrite[1] ++;
			
			BKP_WriteBackupRegister(BKP_DR1, ArrayWrite[0]);	//写入测试数据到备份寄存器
			BKP_WriteBackupRegister(BKP_DR2, ArrayWrite[1]);

            //ArrayWrite[0] ++;		//测试数据自增,或者写入后再加加
			//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);
	}
}

相关函数:

void PWR_BackupAccessCmd(FunctionalState NewState)//启用或禁用对RTC和备份寄存器的访问

|----------|-------------------------------------|
| 参数 | 说明 |
| NewState | 访问RTC和备份寄存器的新状态;取值包括:ENABLE或DISABLE |

void BKP_WriteBackupRegister(uint16_t BKP_DR, uint16_t Data)//将用户数据写入指定的数据备份寄存器

|--------|-----------------------------------------------------------------------------|
| 参数 | 说明 |
| BKP_DR | 指定数据备份寄存器,该参数可以是BKP_DRx,其中x:[1,42],STM32是中容量型芯片,DR是范围1~10;大容量和互联型才有42各DR |
| Data | 写入的数据 |

uint16_t BKP_ReadBackupRegister(uint16_t BKP_DR)//从指定的数据备份寄存器读取数据

|--------|-----------------------------------------------------------------------------|
| 参数 | 说明 |
| BKP_DR | 指定数据备份寄存器,该参数可以是BKP_DRx,其中x:[1,42],STM32是中容量型芯片,DR是范围1~10;大容量和互联型才有42各DR |

返回值:指定的数据备份寄存器的内容。

实验结果:

RTC实时时钟

接线图:

RTC初始化

具体步骤:

①和BKP一样,开启PWR和BKP时钟,使能对BKP和RTC的访问

②启动RTC时钟,计划使用LSE作为系统时钟,所以要使用RCC模块里的函数,开启LSE时钟,注意LSE时钟为了省电,默认是关闭的,所以需要我们自己手动开启

③ 配置RTCCLK这个数据选择器,指定LSE为RTCCLK,函数也是在RCC模块

④ 等待RTC_CRL寄存器中的RSF位(寄存器同步标志)被硬件置1和查询RTC_CR寄存器中的RTOFF状态位,判断RTC寄存器是否处于更新中。仅当RTOFF状态位是1时,才可以写入RTC寄存器

⑤配置预分频器,给PRL预重装载器一个合适的分频值,确保输出频率为1Hz

⑥配置CNT的值,给RTC一个合适的初始时间,之后需要闹钟或者中断就配置就可以

cs 复制代码
void HerRTC_Init(void)
{
	/*开启时钟*/
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);		//开启PWR的时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP, ENABLE);		//开启BKP的时钟
	
	/*备份寄存器访问使能*/
	PWR_BackupAccessCmd(ENABLE);							//使用PWR开启对备份寄存器的访问
	
	if (BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5)			//通过写入备份寄存器的标志位,判断RTC是否是第一次配置
															//if成立则执行第一次的RTC配置
	{
		RCC_LSEConfig(RCC_LSE_ON);							//开启LSE时钟
		while (RCC_GetFlagStatus(RCC_FLAG_LSERDY) != SET);	//等待LSE准备就绪
		
		RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);				//选择RTCCLK来源为LSE
		RCC_RTCCLKCmd(ENABLE);								//RTCCLK使能
		
		RTC_WaitForSynchro();								//等待同步
		RTC_WaitForLastTask();								//等待上一次操作完成
		
		RTC_SetPrescaler(32768 - 1);						//设置RTC预分频器,预分频后的计数频率为1Hz
		RTC_WaitForLastTask();								//等待上一次操作完成
		
		HerRTC_SetTime();									//设置时间,调用此函数,全局数组里时间值刷新到RTC硬件电路
		
		BKP_WriteBackupRegister(BKP_DR1, 0xA5A5);			//在备份寄存器写入自己规定的标志位,用于判断RTC是不是第一次执行配置
	}
	else													//RTC不是第一次配置
	{
		RTC_WaitForSynchro();								//等待同步
		RTC_WaitForLastTask();								//等待上一次操作完成
	}
}

void RCC_LSEConfig(uint8_t RCC_LSE) //配置外部低速振荡器(LSE)

|---------|-----------|
| 参数 | 说明 |
| RCC_LSE | 指定LSE的新状态 |

FlagStatus RCC_GetFlagStatus(uint8_t RCC_FLAG)//检查是否设置了指定的RCC标志

|----------|----------|
| 参数 | 说明 |
| RCC_FLAG | 指定要检查的标志 |

返回值: RCC_FLAG的新状态(SET或RESET)

void RCC_RTCCLKConfig(uint32_t RCC_RTCCLKSource)//配置RTC时钟(RTCCLK)就是配置数据选择器

注意:一旦选择了RTC时钟,就不能更改它,除非重置备份域 。

|------------------|--------|
| 参数 | 说明 |
| RCC_RTCCLKSource | RTC时钟源 |

void RCC_RTCCLKCmd(FunctionalState NewState)//开启或关闭RTC时钟

注意:只有通过RCC_RTCCLKConfig函数选择了RTC时钟后,才能使用该功能。

|----------|-----------------------------|
| 参数 | 说明 |
| NewState | RTC时钟的新状态取值为:ENABLE或DISABLE |

void RTC_WaitForSynchro(void)//等待RTC寄存器(RTC_CNT, RTC_ALR和RTC_PRL),设置RSF标志1

void RTC_WaitForLastTask(void)//等待RTC寄存器上的最后一次写操作完成,循环直到设置RTOFF标志
void RTC_SetPrescaler(uint32_t PrescalerValue)//设置RTC预分频器值

|----------------|--------------|
| 参数 | 说明 |
| PrescalerValue | RTC预caler的新值 |

小细节:

对应上节RTC操作注意事项的第三点:

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

进入和退出RTC的配置模式的代码不用我们自己额外调用,相关函数已经帮我们写好,我们知道有这个东西就可以

如果RTC晶振不起振的解决方法:戳看解决方法

接下来还要写两个函数,一个是设置时间,一个是读取时间。计划是,读取时间,我们就把读取到的秒数转换为年月日时分秒放在一个全局数组里,设置时间,我们再把年月日时分秒转换为秒数,再写入RTC的CNT。

RTC设置时间

定义数组注意:小细节

cs 复制代码
uint16_t MyRTC_Time[] = {2023, 1, 1, 23, 59, 55};	//定义全局的时间数组,数组内容分别为年、月、日、时、分、秒

void MyRTC_SetTime(void);				//函数声明,调用函数不能在函数定义前,所以要声明


/**
  * 函    数:RTC设置时间
  * 参    数:无
  * 返 回 值:无
  * 说    明:调用此函数后,全局数组里时间值将刷新到RTC硬件电路
  */
void MyRTC_SetTime(void)
{
	time_t time_cnt;		//定义秒计数器数据类型
	struct tm time_date;	//定义日期时间数据类型
	
	time_date.tm_year = MyRTC_Time[0] - 1900;		//将数组的时间赋值给日期时间结构体
	time_date.tm_mon = MyRTC_Time[1] - 1;
	time_date.tm_mday = MyRTC_Time[2];
	time_date.tm_hour = MyRTC_Time[3];
	time_date.tm_min = MyRTC_Time[4];
	time_date.tm_sec = MyRTC_Time[5];
	
	time_cnt = mktime(&time_date) - 8 * 60 * 60;	//调用mktime函数,将日期时间转换为秒计数器格式
													//- 8 * 60 * 60为东八区的时区调整
	
	RTC_SetCounter(time_cnt);						//将秒计数器写入到RTC的CNT中
	RTC_WaitForLastTask();							//等待上一次操作完成
}

void RTC_SetCounter(uint32_t CounterValue)//设置RTC计数器值

|--------------|----------|
| 参数 | 说明 |
| CounterValue | RTC计数器新值 |

time_t mktime(struct tm*)函数文章开头unix时间戳篇有介绍

struct tm* localtime(const time_t*); 函数文章开头unix时间戳篇有介绍

time.h头文件在keil的小区别:time.h头文件

RTC读取时间

cs 复制代码
/**
  * 函    数:RTC读取时间
  * 参    数:无
  * 返 回 值:无
  * 说    明:调用此函数后,RTC硬件电路里时间值将刷新到全局数组
  */
void MyRTC_ReadTime(void)
{
	time_t time_cnt;		//定义秒计数器数据类型
	struct tm time_date;	//定义日期时间数据类型
	
	time_cnt = RTC_GetCounter() + 8 * 60 * 60;		//读取RTC的CNT,获取当前的秒计数器
													//+ 8 * 60 * 60为东八区的时区调整
	
	time_date = *localtime(&time_cnt);				//使用localtime函数,将秒计数器转换为日期时间格式
	
	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;
}

外部声明全局数组

main函数

cs 复制代码
int main(void)
{
	OLED_Init();		//OLED初始化
	MyRTC_Init();		//RTC初始化
	
	OLED_ShowString(1, 1, "Date:XXXX-XX-XX");
	OLED_ShowString(2, 1, "Time:XX:XX:XX");
	OLED_ShowString(3, 1, "CNT :");
	OLED_ShowString(4, 1, "DIV :");
	
	while (1)
	{
		MyRTC_ReadTime();							//RTC读取时间,最新的时间存储到MyRTC_Time数组中
		
		OLED_ShowNum(1, 6, MyRTC_Time[0], 4);		//显示MyRTC_Time数组中的时间值,年
		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);	//显示32位的秒计数器
		OLED_ShowNum(4, 6, RTC_GetDivider(), 10);	//显示余数寄存器
	}
}

uint32_t RTC_GetDivider(void)//获取RTC余数寄存器值

程序现象:

相关推荐
sealaugh3216 分钟前
aws(学习笔记第三十六课) apigw-http-api-lambda-dynamodb
笔记·学习·aws
*TQK*38 分钟前
CSS学习笔记5——渐变属性+盒子模型阶段案例
css·笔记·学习
Logintern091 小时前
分享一个Pyside6实现web数据展示界面的效果图
python·学习·web·数据·pyside6
尺度商业1 小时前
郭英成以茶会友对话李开复,探讨AI科技赋能
人工智能·科技
会挖坑的石头1 小时前
C语言术语
c语言
知识分享小能手1 小时前
CSS3学习教程,从入门到精通,CSS3 媒体查询实现响应式布局语法指南(21)
前端·css·学习·css3·html5·媒体
魑魅魍魉都是鬼1 小时前
音视频 一 看书的笔记 基础视频知识
笔记·音视频
viperrrrrrrrrr72 小时前
大数据学习(92)-spark详解
大数据·学习·spark
慵懒学者2 小时前
16 Junit单元测试框架、反射、注解、动态代理(黑马Java视频笔记)
java·笔记·junit·单元测试
2402_881319302 小时前
3.28学习总结
数据结构·学习·算法