STM32G070KBT6的RTC HAL库使用

*配置问题

    首先使能时钟源,这里在时钟配置中选择LSI,为什么后面会说,然后使能Calender结构体,保证可以对RTC的年月日时分秒等进行写入和读取;alarmA和alarmB是闹钟,这里不用就Disable;      Tamper用于检测外部篡改事件(通常用于增强系统的安全性,尤其是在需要保护敏感数据或防止恶意操作的应用中。),使能后会有一个专用引脚,Timestamp是时间戳,使能后会出现一个引脚,当该引脚检测到篡改事件时,可以触发中断或其他响应机制(例如自动清除 RTC 的备份寄存器,以防止敏感数据被读取或篡改。)同时RTC 备份域中的特定寄存器可以记录篡改事件的发生,以便后续分析和处理。

    TimeStamp功能允许在特定事件发生时,记录当前的日期和时间。这对于需要精确记录事件发生时间的应用非常有用。使能后同样会有一个专用引脚,可以记录按钮按下、外部中断、篡改检测等事件的时间,在数据记录系统(RTC 备份域)中,可以为每条记录添加时间戳,以便后续分析。

    Calibration使能后也有一个专用引脚,用于对外输出特定频率的脉冲,外部可以通过该引脚输出的脉冲判断RTC的精度;

    根据我们选择的功能,我们需要配置上述参数,首先Asynchronous Predivider value和Synchronous Predivider value用于将RTC的时钟分频,我们一般进行秒级计数,所以将其分频到1HZ,计算方法为:时钟源频率/(Asynchronous Predivider value+1)/(Synchronous Predivider value),这里为32.768KHZ/128/256 = 1HZ;  

    Day Light Saving和Store Operation分别代表夏令时和重置存储操作,夏令时是一种为了节约能源而对时钟进行调整的制度。在夏令时期间,时钟会提前一小时,以便在白天更长的时间内利用自然光。夏令时的设置可以帮助设备自动调整时间以适应这种变化。我们一般不开启夏令时以保证RTC的时间是标准时间;重置存储操作:

  • RTC_STOREOPERATION_RESET:在低功耗模式下不存储当前时间设置,退出低功耗模式时时间会被重置。
  • RTC_STOREOPERATION_SET:在低功耗模式下保存当前时间设置,退出低功耗模式时恢复到之前的时间。

    我们一般选择RTC_STOREOPERATION_SET,这样就算进入和退出低功耗模式,RTC时间仍然会保持计数;

    Wake UP用于设置RTC唤醒中断,其中的Wake UP Clock是唤醒中断周期,设置为1HZ唤醒中断周期就是1秒,Wake Up Counter就是经过n+1次唤醒中断周期后触发中断,这里n为0,表示经过1次唤醒中断周期,也就是1秒后触发中断;

    Data Format分为BCD格式和BIN格式,BIN格式十进制数12表示0x0C,和我们正常的二进制-十进制转换相同,BCD格式是一种将每个十进制数的每一位用四位二进制表示的方法,十进制数12表示为0x12;一般我们用BIN格式更符合常识;

HAL_RTC_GetTime()和HAL_RTC_SetTime()用于获取和设置RTC的时分秒;

HAL_RTC_GetDate()和HAL_RTC_SetDate()用于获取和设置RTC的年月日星期;

注意上面的函数执行时要有顺序!读取时间时先执行HAL_RTC_GetTime()再执行HAL_RTC_GetDate(),不然就会导致读取时间无效的情况;同样设置时间先HAL_RTC_SetDate()再HAL_RTC_SetTime();代码例子如下:

RTC_Time.c:

#include "RTC_Time.h"

/**
  * @brief	时间设置
  * @param   
  *		@arg 	分别输入 年 月 日 星期 时 分 秒
  * @retval 无
  */
void RTC_SetTime(uint16_t yea,uint8_t mon,uint8_t da, uint8_t hou,uint8_t min,uint8_t sec)
{
	  RTC_TimeTypeDef RTC_Time;
    RTC_DateTypeDef RTC_Date;
	
		system_config.calendar.year = yea;
		system_config.calendar.century = yea/100;
		system_config.calendar.month = mon;
		system_config.calendar.date = da;
		system_config.calendar.hour = hou;
		system_config.calendar.min = min;
		system_config.calendar.sec = sec;
		system_config.calendar.week = GregorianDay(system_config.calendar);
	  Flash_WriteStruct(FLASH_USER_START_ADDR, &system_config);            //将系统配置保存到FLASH用户数据区
	
	
	  RTC_Date.Year = (uint8_t)(system_config.calendar.year%100);    //RTC中year格式为uint8_t,我们这里用其保存年份后两位
	  RTC_Date.Month = system_config.calendar.month;
	  RTC_Date.Date = system_config.calendar.date;
	  RTC_Date.WeekDay = system_config.calendar.week;
	
	  RTC_Time.Hours = system_config.calendar.hour;
		RTC_Time.Minutes = system_config.calendar.min;
	  RTC_Time.Seconds = system_config.calendar.sec;
	  RTC_Time.DayLightSaving = DayLightSaving_status;
	  RTC_Time.StoreOperation = StoreOperation_status;
	
	
	  HAL_RTC_SetDate(&hrtc, &RTC_Date, RTC_FORMAT_BIN);      //将配置写入到RTC
	  HAL_RTC_SetTime(&hrtc, &RTC_Time, RTC_FORMAT_BIN);
}

/**
  * @brief	获取时间
  * 
  *		
  */
void RTC_GetTime(void)
{
	  RTC_TimeTypeDef RTC_Time;
    RTC_DateTypeDef RTC_Date;
	
	  HAL_RTC_GetTime(&hrtc, &RTC_Time, RTC_FORMAT_BIN);   //先运行GetTime再运行GetDate
	  HAL_RTC_GetDate(&hrtc, &RTC_Date, RTC_FORMAT_BIN);
   
	
	  system_config.calendar.year = system_config.calendar.century*100 + RTC_Date.Year;  //将Flash中的世纪和RTC中的年份相加获得完整年份
		system_config.calendar.month = RTC_Date.Month;
		system_config.calendar.date = RTC_Date.Date;
	  system_config.calendar.week = RTC_Date.WeekDay;
	
		system_config.calendar.hour = RTC_Time.Hours;
		system_config.calendar.min = RTC_Time.Minutes;
		system_config.calendar.sec = RTC_Time.Seconds;
	  //Flash_WriteStruct(FLASH_USER_START_ADDR, &system_config);            //将系统配置保存到FLASH用户数据区
}




void HAL_RTCEx_WakeUpTimerEventCallback(RTC_HandleTypeDef *hrtc)
{
  /* Prevent unused argument(s) compilation warning */
   	
	
	//HAL_IWDG_Refresh(&hiwdg);   //喂狗,5s内必须喂一次
		RTC_GetTime();   	//更新时间   
		Write_Num_Time();							  //显示时间
  /* NOTE : This function should not be modified, when the callback is needed,
            the HAL_RTCEx_WakeUpTimerEventCallback could be implemented in the user file
   */
}



/*计算公历天数得出星期*/
static int GregorianDay(_calendar_obj tm)
{
	int leapsToDate;
	int lastYear;
	int day;
	int MonthOffset[] = { 0,31,59,90,120,151,181,212,243,273,304,334 };
 
	lastYear=tm.year-1;
 
	/*计算从公元元年到计数的前一年之中一共经历了多少个闰年*/
	leapsToDate = lastYear/4 - lastYear/100 + lastYear/400;      
 
     /*如若计数的这一年为闰年,且计数的月份在2月之后,则日数加1,否则不加1*/
	if((tm.year%4==0) &&
	   ((tm.year%100!=0) || (tm.year%400==0)) &&
	   (tm.month>2)) {
		/*
		 * We are past Feb. 29 in a leap year
		 */
		day=1;
	} else {
		day=0;
	}
 
	day += lastYear*365 + leapsToDate + MonthOffset[tm.month-1] + tm.date; /*计算从公元元年元旦到计数日期一共有多少天*/
 
	tm.week=day%7; //算出星期
	if(tm.week == 0) 
		tm.week = 7;
	return tm.week;
}
 

RTC_Time.h:

#ifndef __RTC_TIME_H__
#define __RTC_TIME_H__


#include "main.h"
#include "User_Flash.h"
#include "rtc.h"
#include "74HC595.h"

#define DayLightSaving_status     RTC_DAYLIGHTSAVING_NONE   //配置RTC夏令时状态
#define StoreOperation_status     RTC_STOREOPERATION_SET    //配置RTC 是否在低功耗模式保持运行

void RTC_SetTime(uint16_t yea,uint8_t mon,uint8_t da, uint8_t hou,uint8_t min,uint8_t sec);
void RTC_GetTime(void);
static int GregorianDay(_calendar_obj tm); 
#endif


*电源引脚问题:

    从上图我们可以看出LQFP32封装的G070KBT6是没有VBAT引脚的,只有VDD引脚,因此当主电源VDD断电后采用电池给VDD供电,因此RTC也只能由VDD保持供电,可以用一路ADC采样主电源电压,当检测到主电源电压接近0V时表示此时已切换到电池供电,然后进行相应操作。电路如下:

    巧思:要做低功耗,可以进入低功耗就把检测电源电压的ADC转化为外部中断,当再次上电时触发外部中断退出低功耗模式,这样就不用电池供电的时候还要用ADC检测电源是否重新上电,节省功耗。(未验证)

*时钟源问题 :

    只有一个外部高速时钟,可以选择LSI或者HSE/32(一般RTC外部时钟源可以选择LSE或者HSE,通常使用LSE,不过为了节省晶振的情况下也可以用HSE分频替代,这样也比LSI更稳定。),由于主电源断电之后采用外部高速时钟HSE为有源晶振,会停止起振,因此这里使用LSI内部低速32.768KHZ晶振给RTC提供时钟源。由前面关于RTC的手册描述我们可以知道当LSI计时时,VBAT模式下RTC是不工作的,因为vbat只能给LSE供电而不能给LSI供电。vdd断了以后LSI也相当于断电了,rtc自然就不走了。但是由于G070KBT6没有VBAT引脚,因此也就没有VBAT模式,就算用刚刚的电池供电也是给VDD供电,因此这里不用担心电池供电时RTC会停止工作。

引用中的代码是USB插入后主机进行枚举并复位USB的处理过程。在这段代码中,if语句检查是否发生了复位中断,如果发生了复位中断,则会进行一系列操作,包括清除复位标志、调用复位回调函数和设置地址为0。 引用中的代码是在USBD_CDC_Init函数中动态分配内存给pdev->pClassData。 引用中的代码是HAL_PCD_MspInit函数的定义部分,其中会根据pcdHandle->Instance的值进行相应的处理。 根据提供的引用信息,HAL_PCD_IRQHandler(&hpcd_USB_FS)是用于处理USB FS(全速)的中断请求的函数调用。具体的实现细节需要查看HAL库的文档和源代码。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [STM32 USB复合设备编写](https://blog.csdn.net/aifuxun2845/article/details/102026435)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *3* [记录stm32g431kbt6 编写SPI以及USB HID通信](https://blog.csdn.net/s_jk6652/article/details/126348126)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值