STM32之RTC实时时钟

news2025/1/24 14:45:54

一、实时时钟概述

1、实时时钟介绍

英文缩写:RTC。显示年、月、日、时、分、秒、星期,自动计算闰年,能够区分每个月的天数。

RTC特点:能从RTC获取到具体的日期时间,断掉后再开机时间仍然准确(需要纽扣电池)。

RTC模块分为两种,一种集成在芯片内部,另外一种是外接RTC芯片。

2、常用的实时时钟芯片

常见的实时时钟芯片:

常见实时时钟芯片:DS1302、DS1307、PCF8563等。

显示年、月、日、时、分、秒、星期,自动计算闰年,能够区分每个月的天数。

二、STM32内部实时时钟介绍

1、STM32内部实时时钟特点

实时时钟 (RTC) 是一个独立的 BCD 定时器 /计数器。 RTC 提供一个日历时钟两个可编程闹钟中断,以及一个具有中断功能周期性可编程唤醒标志。 RTC 还包含用于管理低功耗式的自动唤醒单元。两个 32 位寄存器包含二进码十进数格式 (BCD) 的秒、分钟、小时( 12 24 小时制)、星期几、日期、月份和年份。此外,还可提供二进制格式的亚秒值。系统可以自动将月份的天数补偿为 28、 29(闰年)、 30 和 31 天。并且还可以进行夏令时补偿。其它 32 位寄存器还包含可编程的闹钟亚秒、 秒、分钟、小时、星期几和日期。
此外,还可以使用数字校准功能对晶振精度的偏差进行补偿。上电复位后,所有 RTC 寄存器都会受到保护以防止可能的非正常写访问。无论器件状态如何(运行模式、低功耗模式或处于复位状态),只要电源电压保持在工作范围内,RTC 便不会停止工作。

21 -- 0001 0101(平常的十进制转为二进制) -- 0x15

十进制转为BCD码的二进制格式(BCD能表示的范围是 0 - 9)

21 -- 0010 0001 -- 0x21

2、RTC电源部分

RTC断掉主电源以后就会由VBAT供电,所以可以做到断电数据不丢失的效果。

3、STM32内部实时时钟的功能介绍

RTC 单元的主要特性如下(参见图 222: RTC 框图):
● 包含亚秒、秒、分钟、小时( 12/24 小时制)、星期几、日期、月份和年份的日历。
● 软件可编程的夏令时补偿。
两个具有中断功能的可编程闹钟。可通过任意日历字段的组合驱动闹钟。
自动唤醒单元,可周期性地生成标志以触发自动唤醒中断。
● 参考时钟检测:可使用更加精确的第二时钟源(50 Hz 或 60 Hz)来提高日历的精确度。
● 利用亚秒级移位特性与外部时钟实现精确同步。
● 可屏蔽中断 /事件:— 闹钟 A
— 闹钟 B
— 唤醒中断
— 时间戳
 — 入侵检测
● 数字校准电路(周期性计数器调整)
— 精度为 5 ppm
— 精度为 0.95 ppm,在数秒钟的校准窗口中获得
● 用于事件保存的时间戳功能( 1 个事件)
● 入侵检测:
 — 2 个带可配置过滤器和内部上拉的入侵事件
● 20 个备份寄存器( 80 字节)。发生入侵检测事件时,将复位备份寄存器。

三、STM32内部实时时钟框架

四、RTC基本日历功能框架分析

1、RTC寄存器写保护

系统复位后,可通过 PWR 电源控制寄存器 (PWR_CR) 的 DBP 位保护 RTC 寄存器以防止非正常的写访问。必须将 DBP 位置 1 才能使能 RTC 寄存器的写访问。上电复位后,所有 RTC 寄存器均受到写保护。通过向写保护寄存器 (RTC_WPR) 写入一个密钥来使能对 RTC 寄存器的写操作。要解锁所有 RTC 寄存器(RTC_ISR[13:8]、 RTC_TAFCR 和 RTC_BKPxR 除外)的写保护,

需要执行以下步骤:
1. 将“0xCA”写入 RTC_WPR 寄存器。
2. 将“0x53”写入  RTC_WPR 寄存器。
写入一个错误的关键字会再次激活写保护。
保护机制不受系统复位影响。

2、RTC进入初始化模式(在设置RTC时间和日期要注意的)

要编程包括时间格式和预分频器配置在内的初始时间和日期日历值,需按照以下顺序操作:
1. 将 RTC_ISR 寄存器中的 INIT 位置 1 以进入初始化模式。在此模式下,日历计数器将停止工作并且其值可更新。
2. 轮询 RTC_ISR 寄存器中的 INITF 位。当 INITF 置 1 时进入初始化阶段模式。大约需要2 个 RTCCLK 时钟周期(由于时钟同步)。
3. 要为日历计数器生成 1 Hz 时钟,应首先编程 RTC_PRER 寄存器中的同步预分频系数,然后编程异步分频系数。即使只需要更改这两个字段中之一,也必须对 RTC_PRER寄存器执行两次单独的写访问。
4. 在影子寄存器( RTC_TR 和 RTC_DR)中加载初始时间和日期值,然后通过 RTC_CR寄存器中的 FMT 位配置时间格式( 12 或 24 小时制)。
5. 通过清零 INIT 位退出初始化模式。随后,自动加载实际日历计数器值,在 4 个 RTCCLK时钟周期后重新开始计数。
当初始化序列完成之后,日历开始计数。
系统复位后,应用可读取 RTC_ISR 寄存器中的 INITS 标志,以检查日历是否已初始化。如果该标志为 0,表明日历尚未初始化,因为年份字段设置为其上电复位时的默认值 (0x00)。要在初始化之后读取日历,必须首先用软件检查 RTC_ISR 寄存器的 RSF

3、RTC同步(读取RTC时间和日期时要注意的)

要正确读取 RTC 日历寄存器(RTC_SSR、 RTC_TR 和 RTC_DR), APB1 时钟频率 (fPCLK1 )必须等于或大于 fRTCCLK RTC 时钟频率的七倍。这可以确保同步机制行为的安全性。如果 APB1 时钟频率低于 RTC 时钟频率的七倍,则软件必须分两次读取日历时间寄存器和 日期寄存器。这样,当两次读取的 RTC_TR 结果相同时,才能确保数据正确。否则必须执行第三次读访问。任何情况下, APB1 的时钟频率都不能低于 RTC 的时钟频率。每次将日历寄存器中的值复制到 RTC_SSR、 RTC_TR 和 RTC_DR 影子寄存器时, RTC_ISR
寄存器中的 RSF 位都会置 1 。每两个 TRCCLK 周期执行一次复制。为确保这 3 个值来自同 一时刻点,读取 RTC_SSR 或 RTC_TR 时会锁定高阶日历影子寄存器中的值,直到读取 RTC_DR。为避免软件对日历执行读访问的时间间隔小于 2 个 RTCCLK 周期:第一次读取 日历之后必须通过软件将 RSF 清零,并且软件必等待到 RSF 1 之后才可再次读取 RTC_SSR RTC_TR RTC_DR 寄存器。

4、STM32内部实时时钟寄存器说明

RTC 时间寄存器 (RTC_TR)

位 22 PM: AM/PM 符号 (AM/PM notation)
0: AM 或 24 小时制
1: PM
位 21:20 HT[1:0]:小时的十位(BCD 格式) (Hour tens in BCD format)
位 16:16 HU[3:0]:小时的个位(BCD 格式) (Hour units in BCD format)

位 15 保留,必须保持复位值。
位 14:12 MNT[2:0]:分钟的十位(BCD 格式) (Minute tens in BCD format)
位 11:8 MNU[3:0]:分钟的个位(BCD 格式) (Minute units in BCD format)

位 7 保留,必须保持复位值。
位 6:4 ST[2:0]:秒的十位(BCD 格式) (Second tens in BCD format)
位 3:0 SU[3:0]:秒的个位(BCD 格式) (Second units in BCD format)

RTC 日期寄存器 (RTC_DR)

RTC_DR 是日历日期影子寄存器。只能在初始化模式下对该寄存器执行写操作

位 23:20 YT[3:0]:年份的十位(BCD 格式) (Year tens in BCD format)
位 19:16 YU[3:0]:年份的个位(BCD 格式) (Year units in BCD format)
位 15:13 WDU[2:0]:星期几的个位 (Week day units)
000:禁止
001:星期一
...
111:星期日
位 12 MT:月份的十位(BCD 格式) (Month tens in BCD format)
位 11:8 MU:月份的个位(BCD 格式) (Month units in BCD format)
位 7:6 保留,必须保持复位值。
位 5:4 DT[1:0]:日期的十位(BCD 格式) (Date tens in BCD format)
位 3:0 DU[3:0]:日期的个位(BCD 格式) (Date units in BCD format)

RTC 控制寄存器 (RTC_CR)

位 6 FMT:小时格式 (Hour format)
0 24 小时/天格式
1: AM/PM 小时格式
位 5 BYPSHAD:旁路影子寄存器 (Bypass the shadow registers)
0:日历值(从 RTC_SSR RTC_TR RTC_DR 读取时)取自影子寄存器,该影子寄存器
每两个 RTCCLK 周期更新一次。
1:日历值(从 RTC_SSR、 RTC_TR 和 RTC_DR 读取时)直接取自日历计数器。
注意:如果 APB1 时钟的频率低于 7 倍的 RTCCLK 频率,则必须将 BYPSHAD 置“1”。

RTC 初始化和状态寄存器 (RTC_ISR)

位 7 INIT:初始化模式 (Initialization mode)
0:自由运行模式。
1:初始化模式,用于编程时间和日期寄存器(RTC_TR RTC_DR)以及预分频器寄存器
(RTC_PRER)。计数器停止计数,当 INIT 被复位后,计数器从新值开始计数。
位 6 INITF:初始化标志 (Initialization flag)
当此位置 1 时, RTC 处于初始化状态,此时可更新事件、日期和预分频器寄存器。
0:不允许更新日历寄存器。
1:允许更新日历寄存器。

位 5 RSF:寄存器同步标志 (Registers synchronization flag)
每次将日历寄存器的值复制到影子寄存器(RTC_SSRx、 RTC_TRx 和 RTC_DRx)时,都
会由硬件将此位置 1。在初始化模式下、平移操作挂起时 (SHPF=1) 或在旁路影子寄存器模
式 (BYPSHAD=1) 下,该位由硬件清零。该位还可由软件清零。
0:日历影子寄存器尚未同步
1:日历影子寄存器已同步

RTC 预分频器寄存器 (RTC_PRER)

位 22:16 PREDIV_A[6:0]:异步预分频系数 (Asynchronous prescaler factor)
下面是异步分频系数的公式:
ck_apre 频率 = RTCCLK 频率/(PREDIV_A+1)
注意: PREDIV_A [6:0]= 000000 为禁用值。
位 15 保留,必须保持复位值。
位 14:0 PREDIV_S[14:0]:同步预分频系数 (Synchronous prescaler factor)
下面是同步分频系数的公式:
ck_spre 频率 = ck_apre 频率/(PREDIV_S+1)

RTC 写保护寄存器 (RTC_WPR)

位 7:0 KEY:写保护关键字 (Write protection key)
可通过软件对该字节执行写操作。
读取该字节时,始终返回 0x00。
有关如何解锁 RTC 寄存器写保护的介绍,请参见RTC 寄存器写保护。

五、RTC自动唤醒功能

通过设定一个时间周期,当时间到了的时候,就会产生一些标志/中断,通过IO口将当前标志输出出去,产生外部中断。一般自动唤醒都是设定一秒产生一次中断,在中断中获取RTC时间/日期。

1、RTC自动唤醒功能相关寄存器

RTC 控制寄存器 (RTC_CR)

14 WUTIE:使能唤醒定时器使能 (Wakeup timer interrupt enable)
0
:禁止唤醒定时器中断
1:使能唤醒定时器中断

10 WUTE:唤醒定时器使能 (Wakeup timer enable)
0
:禁止唤醒定时器
1:使能唤醒定时器

2:0 WUCKSEL[2:0]:唤醒时钟选择 (Wakeup clock selection)
000
:选择 RTC/16 时钟
001:选择 RTC/8 时钟
010:选择 RTC/4 时钟
011:选择 RTC/2 时钟
10x:选择 ck_spre 时钟(通常为 1 Hz
11x:选择 ck_spre 时钟(通常为 1 Hz)并将 WUT 计数器值增加 216(见下面的注释)

RTC 初始化和状态寄存器 (RTC_ISR)

10 WUTF:唤醒定时器标志 (Wakeup timer flag)
当唤醒自动重载计数器计数到 0 时,由硬件将此标志置 1
该标志由软件写零清除。
软件必须在
WUTF 再次置 1 1.5 RTCCLK 周期之前将该标志清零。

2 WUTWF:唤醒定时器写标志 (Wakeup timer write flag)
RTC_CR 寄存器中的 WUTE 位置 0 后,当唤醒定时器值可更改时,由硬件将该位置 1
0:不允许更新唤醒定时器配置
1:允许更新唤醒定时器配置

RTC 唤醒定时器寄存器 (RTC_WUTR)

六、​​​​​​​RTC闹钟功能

1、RTC闹钟功能框图分析

2、RTC闹钟功能相关寄存器

RTC 控制寄存器 (RTC_CR)

位 13 ALRBIE闹钟 B 中断使能 (Alarm B interrupt enable)
0:闹钟 B 中断禁止
1:闹钟 B 中断使能
位 12 ALRAIE:闹钟 A 中断使能 (Alarm A interrupt enable)
0:禁止闹钟 A 中断
1:使能闹钟 A 中断

位 9 ALRBE闹钟 B 使能 (Alarm B enable)
0:禁止闹钟 B
1:使能闹钟 B

位 8 ALRAE闹钟 A 使能 (Alarm A enable)
0:禁止闹钟 A
1:使能闹钟 A

RTC 初始化和状态寄存器 (RTC_ISR)

位 9 ALRBF:闹钟 B 标志 (Alarm B flag)
当时间/日期寄存器(RTC_TR 和 RTC_DR)与闹钟 B 寄存器 (RTC_ALRMBR) 匹配时,由
硬件将该标志置 1。
该标志由软件写零清除。
位 8 ALRAF:闹钟 A 标志 (Alarm A flag)
当时间/日期寄存器(RTC_TR 和 RTC_DR)与闹钟 A 寄存器 (RTC_ALRMAR) 匹配时,由
硬件将该标志置 1。
该标志由软件写零清除。

位 1 ALRBWF:闹钟 B 写标志 (Alarm B write flag)
在 RTC_CR 寄存器中的 ALRBIE 位置 0 之后,当闹钟 B 的值可更改时,由硬件将该位置 1。
该位在初始化模式下由硬件清零。
0:不允许更新闹钟 B
1:允许更新闹钟 B
位 0 ALRAWF:闹钟 A 写标志 (Alarm A write flag)
在 RTC_CR 寄存器中的 ALRAE 位置 0 后,当闹钟 A 的值可更改时,由硬件将该位置 1。
该位在初始化模式下由硬件清零。
0:不允许更新闹钟 A
1:允许更新闹钟 A

RTC 闹钟 A 寄存器 (RTC_ALRMAR)

RTC 闹钟 B 寄存器 (RTC_ALRMBR)

RTC基本日历功能
 软件设计
1. 打开PWR的时钟
2. 选择PWR寄存器中的CR寄存器的DBP 位置 1
3. 选择时钟源
4.开启相应的时钟源
5.判断开启成功没有
6. 选择相应的时钟源到RTC里
7.使能RTC的时钟
8.解除写保护
将“0xCA”写入 RTC_WPR 寄存器。
将“0x53”写入  RTC_WPR 寄存器。
9.将 RTC_ISR 寄存器中的 INIT 位置 1 以进入初始化模式。在此模式下,日历计数器将停止工作并且其值可更新。
10.轮询 RTC_ISR 寄存器中的 INITF 位。当 INITF 置 1 时进入初始化阶段模式。大约需要2 个 RTCCLK 时钟周期(由于时钟同步)。
11. 要为日历计数器生成 1 Hz 时钟,应首先编程 RTC_PRER 寄存器中的同步预分频系数,然后编程异步分频系数。即使只需要更改这两个字段中之一,也必须对 RTC_PRER寄存器执行两次单独的写访问。
12.在影子寄存器( RTC_TR 和 RTC_DR)中加载初始时间和日期值,然后通过 RTC_CR寄存器中的 FMT 位配置时间格式( 12 或 24 小时制)。
13.通过清零 INIT 位退出初始化模式。随后,自动加载实际日历计数器值,在 4 个 RTCCLK时钟周期后重新开始计数。
当初始化序列完成之后,日历开始计数。
14.读取相应的时间出来

#include "rtc.h"

const char *pt = __TIME__;
const char *pd = __DATE__;
TIME_DATA time_data;
u8 month[12][5] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
/***********************************************************
函数功能:判断闰年函数
函数形参:年
函数返回值:0平年  1闰年
************************************************************/
u8 Pd_Rn(u16 year)
{
	if( (year%4==0 && year%100!= 0) || (year%400 == 0) )
	{
		return 1;
	}
	return 0;
}

/***********************************************************
函数功能:1990年1月1日到今天的天数函数
函数形参:年 月 日
函数返回值:星期几
************************************************************/
u8 Statistics_Days(u16 year, u8 mon, u8 day)
{
	u32 buf = 0;//存储天数
	u16 i;//0-66535
	u8 week = 0;
	
	for(i = 1990; i < year; i++)
	{
		if( Pd_Rn(i) )
		{
			buf += 366;
		}else
		{
			buf += 365;
		}
	}
	
	switch(mon)//7
	{
		case 12: buf += 30; 
		case 11: buf += 31; 
		case 10: buf += 30; 
		case  9: buf += 31; 
		case  8: buf += 31; 
		case  7: buf += 30; 
		case  6: buf += 31; 
		case  5: buf += 30; 
		case  4: buf += 31; 
		case  3: buf += 28; buf += Pd_Rn(year); 
		case  2: buf += 31; 
		case  1: buf += 0; 
	}
	
	//统计从这个1日 到 今天的天数
	buf += day;//1990年1月1日  到  今天的天数
	
	switch(buf % 7)
	{
		case 1: week = 1; break;
		case 2: week = 2; break;
		case 3: week = 3; break;
		case 4: week = 4; break;
		case 5: week = 5; break;
		case 6: week = 6; break;
		case 0: week = 7; break;
	}
	
	return week;//将星期几返回出去了
}

//设置时间
ErrorStatus RTC_Set_Time(void)
{
	/*****************解析时间**********************/
	time_data.hour = (pt[0]-'0')*10 + (pt[1]-'0');//、得到小时
	time_data.minute = (pt[3]-'0')*10 + (pt[4]-'0');//得到分钟
	time_data.second = (pt[6]-'0')*10 + (pt[7]-'0');//得到秒
	RTC_TimeTypeDef RTC_TimeTypeInitStructure;
	RTC_TimeTypeInitStructure.RTC_Hours = time_data.hour;
	RTC_TimeTypeInitStructure.RTC_Minutes = time_data.minute;
	RTC_TimeTypeInitStructure.RTC_Seconds = time_data.second;
	RTC_TimeTypeInitStructure.RTC_H12 = RTC_H12_AM;
	return RTC_SetTime(RTC_Format_BIN,&RTC_TimeTypeInitStructure);
}
	
//设置日期
ErrorStatus RTC_Setime_dataate()
{
	u8 str[4] = {0};
	u8 i;
	/*****************解析日期**********************/
	for(i = 0; i < 3; i++)
	{
		str[i] = pd[i];//May
	}
	str[i] = '\0';
	
	for(i = 0; i < 12; i++)
	{
		if(strcmp((char *)str, (char *)month[i]) == 0 )
		{
			i += 1;
			break;//找到月份了
		}
	}
	time_data.month = i;//得到月
	if( pd[4] == ' ' )
	{
		time_data.day = pd[5]-'0';//得到日
	}else
	{
		time_data.day = (pd[4]-'0')*10 + (pd[5]-'0');//得到日
	}
	
	time_data.year = (pd[9]-'0')*10 + (pd[10]-'0');//得到年
	
	/*****************解析星期**********************/
	time_data.week = Statistics_Days(time_data.year+2000, time_data.month, time_data.day);//得到星期几
	
	RTC_DateTypeDef RTC_DateTypeInitStructure;
	RTC_DateTypeInitStructure.RTC_Year = time_data.year;
	RTC_DateTypeInitStructure.RTC_Month = time_data.month;
	RTC_DateTypeInitStructure.RTC_Date = time_data.day;
	RTC_DateTypeInitStructure.RTC_WeekDay = time_data.week;
	return RTC_SetDate(RTC_Format_BIN,&RTC_DateTypeInitStructure);
}

/************************
函数功能:RTC初始化
形参:无
返回值:成功返回0,失败返回1
说明:24小时制
************************/
u8 My_Rtc_Init(void)
{
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);//使能PWR时钟
	PWR_BackupAccessCmd(ENABLE);	//使能后备寄存器访问 
	RTC_WriteProtectionCmd(DISABLE); 
	u16 retry= 0; 
	//RCC_LSEConfig(RCC_LSE_ON);//LSE 开启 
	RCC_LSICmd(ENABLE);
	while(RCC_GetFlagStatus(RCC_FLAG_LSIRDY) == RESET)	//检查指定的RCC标志位设置与否,等待低速晶振就绪
	{
		retry++;
		delay_ms(10);
		if(retry == 200)
		{	
			return 1;		//LSE 开启失败. 
		}
	}

	RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);		//设置RTC时钟(RTCCLK),选择LSE作为RTC时钟    
	RCC_RTCCLKCmd(ENABLE);	//使能RTC时钟 
	
	RTC_InitTypeDef RTC_InitStructure;
	RTC_InitStructure.RTC_AsynchPrediv = 0x7F;//RTC异步分频系数(1~0X7F)
	RTC_InitStructure.RTC_SynchPrediv = 0xF9;//RTC同步分频系数(0~7FFF)
	RTC_InitStructure.RTC_HourFormat = RTC_HourFormat_24;//RTC设置为,24小时格式
	RTC_Init(&RTC_InitStructure);
	if(RTC_ReadBackupRegister(RTC_BKP_DR1) != 0xbbbb)
	{
		RTC_Set_Time();	//设置时间
		RTC_Setime_dataate();		//设置日期
		RTC_WriteBackupRegister(RTC_BKP_DR1, 0xbbbb); 
	}
	
	 
	return 0;
}

//RTC唤醒功能初始化
void Rtc_Wakeup_Init(void)
{
	EXTI_InitTypeDef EXTI_InitStructure;
	EXTI_InitStructure.EXTI_Line = EXTI_Line22;
	EXTI_InitStructure.EXTI_LineCmd = ENABLE;
	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;
	EXTI_Init(&EXTI_InitStructure);//配置
	
	RTC_WakeUpCmd(DISABLE);//关闭WAKE UP
	RTC_WakeUpClockConfig(RTC_WakeUpClock_CK_SPRE_16bits);//唤醒时钟选择
	RTC_SetWakeUpCounter(0);//设置WAKE UP自动重装载寄存器
	RTC_ClearITPendingBit(RTC_IT_WUT); //清除RTC WAKE UP的标志
	EXTI_ClearITPendingBit(EXTI_Line22);//清除LINE22上的中断标志位
	RTC_ITConfig(RTC_IT_WUT,ENABLE);//开启WAKE UP 定时器中断
	//设置中断
	NVIC_SetPriority(RTC_WKUP_IRQn,NVIC_EncodePriority(7-2,1,1));
	NVIC_EnableIRQ(RTC_WKUP_IRQn);
	RTC_WakeUpCmd(ENABLE);//开启WAKE UP 定时器 
}



//WAKE UP中断函数
void RTC_WKUP_IRQHandler(void)
{
	u8 data[256];
	u8 time[256];
	RTC_TimeTypeDef RTC_TimeStruct;
	RTC_DateTypeDef RTC_DateStruct;
	RTC_ClearFlag(RTC_FLAG_WUTF);	//清除中断标志
	EXTI_ClearITPendingBit(EXTI_Line22);//清除中断线22的中断标志 
	RTC_GetTime(RTC_Format_BIN,&RTC_TimeStruct);
	RTC_GetDate(RTC_Format_BIN,&RTC_DateStruct);
	sprintf((char*)data,"20%02d-%02d-%02d-%01d",RTC_DateStruct.RTC_Year,RTC_DateStruct.RTC_Month,RTC_DateStruct.RTC_Date,RTC_DateStruct.RTC_WeekDay);
	sprintf((char*)time,"%02d:%02d:%02d",RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds);
}

//RTC闹钟功能初始化,周几的闹钟
void Rtc_Alarm(u8 week,u8 hour, u8 minute, u8 second)
{
	
	EXTI_InitTypeDef EXTI_InitStructure;
	EXTI_InitStructure.EXTI_Line = EXTI_Line17;
	EXTI_InitStructure.EXTI_LineCmd = ENABLE;
	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;
	EXTI_Init(&EXTI_InitStructure);//配置外部中断线
	
	RTC_AlarmTypeDef   RTC_AlarmAStruct;
	RTC_AlarmCmd(RTC_Alarm_A,DISABLE); //先关闭闹钟A

	RTC_TimeTypeDef RTC_TimeTypeInitStructure;
	RTC_TimeTypeInitStructure.RTC_Hours = hour;
	RTC_TimeTypeInitStructure.RTC_Minutes = minute;
	RTC_TimeTypeInitStructure.RTC_Seconds = second;
	RTC_TimeTypeInitStructure.RTC_H12 = RTC_H12_AM;
	
	RTC_AlarmAStruct.RTC_AlarmDateWeekDay = week;
	RTC_AlarmAStruct.RTC_AlarmDateWeekDaySel = RTC_AlarmDateWeekDaySel_WeekDay;//按星期闹钟
	RTC_AlarmAStruct.RTC_AlarmMask = RTC_AlarmMask_None;//不屏蔽 
	RTC_AlarmAStruct.RTC_AlarmTime = RTC_TimeTypeInitStructure; 
	RTC_SetAlarm(RTC_Format_BIN,RTC_Alarm_A,&RTC_AlarmAStruct);
	
	//设置中断
	NVIC_SetPriority(RTC_Alarm_IRQn,NVIC_EncodePriority(7-2,1,1));
	NVIC_EnableIRQ(RTC_Alarm_IRQn);	
	//使能闹钟A的中断
	RTC_ITConfig(RTC_IT_ALRA,ENABLE); 
	//开启闹钟A
	RTC_AlarmCmd(RTC_Alarm_A,ENABLE); 

}

//闹钟A中断服务函数
void RTC_Alarm_IRQHandler()
{
	//判断中断是否发生
	if(RTC_GetITStatus(RTC_IT_ALRA)==SET)
	{
		RTC_ClearITPendingBit(RTC_IT_ALRA);//清中断标志位
		
	}
	EXTI_ClearITPendingBit(EXTI_Line17);
}	

TIME_DATA dateAndTime;

//获取当前时间
TIME_DATA *RTC_getDateAndTime(void)
{
	RTC_DateTypeDef RTC_Date;//定义结构体,用于保存获取的日期和时间
	RTC_TimeTypeDef RTC_Time;
	RTC_GetDate(RTC_Format_BIN,&RTC_Date);
	RTC_GetTime(RTC_Format_BIN,&RTC_Time); 
	
	dateAndTime.year = RTC_Date.RTC_Year;
	dateAndTime.month = RTC_Date.RTC_Month;
	dateAndTime.day = RTC_Date.RTC_Date;
	dateAndTime.week = RTC_Date.RTC_WeekDay;
	
	dateAndTime.hour = RTC_Time.RTC_Hours;
	dateAndTime.minute = RTC_Time.RTC_Minutes;
	dateAndTime.second = RTC_Time.RTC_Seconds;
	dateAndTime.ampm = RTC_Time.RTC_H12;
	
	return &dateAndTime;
}

#ifndef RTC_H_
#define RTC_H_
#include "stm32f4xx.h"
#include "stdio.h"
#include "string.h"
typedef struct
{
	u8 hour;
	u8 minute;
	u8 second;
	u8 year;
	u8 month;
	u8 day;
	u8 week;
	u8 ampm;
}TIME_DATA;

typedef	struct
{
	u8 twentyMsCount;
	u8 hour;
	u8 minute;
	u8 second;	
}timeStamp_t;

u8 My_Rtc_Init(void);
void Rtc_Wakeup_Init(void);
void Rtc_Alarm(u8 week,u8 hour, u8 minute, u8 second);
TIME_DATA *RTC_getDateAndTime(void);
#endif

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

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

相关文章

如何控制项目管理中的日程冲突?

《全球公司生产力报告》发现&#xff0c;62% 的公司领导表示&#xff0c;资源调度是他们在项目管理方面面临的最大挑战。其中&#xff0c;日程冲突是利用共享资源池管理多个项目的典型挑战。例如&#xff0c;团队成员参与的活动可能会重叠&#xff0c;也可能是任务分配给了无法…

漏洞复现-SpringBlade export-user SQL 注入漏洞(附漏洞检测脚本)

免责声明 文章中涉及的漏洞均已修复&#xff0c;敏感信息均已做打码处理&#xff0c;文章仅做经验分享用途&#xff0c;切勿当真&#xff0c;未授权的攻击属于非法行为&#xff01;文章中敏感信息均已做多层打马处理。传播、利用本文章所提供的信息而造成的任何直接或者间接的…

【MySQL】NET HELPMSG 3534 报错解决

问题&#xff1a; 解决方案&#xff1a; 注意&#xff1a;确保自己是否使用了带管理员权限的命令窗口。 1.清空你的 MySQL 下的 data 文件夹&#xff1b; 2.确保系统环境变量中已经配置了 mysql 的 bin 目录到Path中&#xff1b; 3.执行以下命令&#xff1a; mysql8为我自己…

【linux】 查看 Linux 重启历史记录(reboot)

了解 Linux 重启日志 /var/log 目录隐藏着 Linux 日志机制的核心信息&#xff0c;它是记录系统活动的宝贵仓库。然而&#xff0c;仅仅有日志还不够&#xff0c;真正的难题在于&#xff0c;如何从大量数据中提炼出与系统重启相关的关键信息。 在 /var/log 目录中&#xff0c;可…

力扣746. 使用最小花费爬楼梯

动态规划 思路&#xff1a; 定义 dp[i] 为到达下标 i 层的最小花费&#xff1b;则状态转移方程为&#xff1a; 第 i 层可以从第 i - 1 层爬一层或者第 i - 2 层爬两层到达&#xff1b;则 dp[i] std::min(dp[i - 1] cost[i - 1], dp[i - 2] cost[i - 2])初始状态&#xff1a…

【操作系统】实验一 Linux操作系统安装

&#x1f57a;作者&#xff1a; 主页 我的专栏C语言从0到1探秘C数据结构从0到1探秘Linux &#x1f618;欢迎关注&#xff1a;&#x1f44d;点赞&#x1f64c;收藏✍️留言 &#x1f3c7;码字不易&#xff0c;你的&#x1f44d;点赞&#x1f64c;收藏❤️关注对我真的很重要&…

Tofu5m目标识别跟踪器

Tofu5m 是高性价比目标识别跟踪模块&#xff0c;支持可见光视频或红外网络视频的输入&#xff0c;支持视频下的多类型物体检测、识别、跟踪等功能。 产品支持视频编码、设备管理、目标检测、深度学习识别、跟踪等功能&#xff0c;提供多机版与触控版管理软件&#xff0c;为二次…

Ant Design Vue 的a-tree-select中的show-search不生效问题

解决方案&#xff1a; 1、配置show-search 2、配置tree-node-filter-prop tree-node-filter-prop的值为接口返回的显示文本&#xff0c;就是你下拉框显示的那个本文 演示&#xff1a;

Web03--CSS进阶

1、CSS常用属性 1.1 文本字体相关属性设置 样式名 描述 text-align 设置内容位置 text-decoration 控制下划线 none没有 underline有 line-hight 行高 font-size 设置字体大小 font-weight 设置字体粗细的 font-famliy 设置字体样式 letter-spacing 设置中文字…

2023年度AI盘点 AIGC|AGI|ChatGPT|人工智能大模型

前言 「作者主页」&#xff1a;雪碧有白泡泡 「个人网站」&#xff1a;雪碧的个人网站 2023年是人工智能大语言模型大爆发的一年&#xff0c;一些概念和英文缩写也在这一年里集中出现&#xff0c;很容易混淆&#xff0c;甚至把人搞懵。 文章目录 前言01 《ChatGPT 驱动软件开…

C++11手撕线程池 call_once 单例模式 Singleton / condition_variable 与其使用场景

一、call_once 单例模式 Singleton 大家可以先看这篇文章&#xff1a;https://zh.cppreference.com/w/cpp/thread/call_once /*std::call_oncevoid call_once( std::once_flag& flag, Callable&& f, Args&&... args ); */ #include <iostream> #i…

龙湖集团2023年销售额减少约14%:股价屡创新低,股东难言满意

近日&#xff0c;龙湖集团控股有限公司&#xff08;HK:00960&#xff0c;下称“龙湖集团”或“龙湖”&#xff09;发布了2023年未经审核运营数据。2023年&#xff0c;龙湖集团实现经营性收入248.7亿元&#xff0c;同比增加6.28%&#xff1b;其中运营收入约129.4亿元&#xff0c…

CM11 链表分割

链表分割_牛客题霸_牛客网 (nowcoder.com) 一、思路分析 二、源码 一、思路分析 创建两个链表small、big 遍历原来链表 比X小的节点尾插到small 比X大的节点尾插到big 最后来链接起来 这样不会改变各个节点的相对顺序 二、源码 ListNode* partition(ListNode* pHead,…

Java21 + SpringBoot3集成easy-captcha实现验证码显示和登录校验

文章目录 前言相关技术简介easy-captcha 实现步骤引入maven依赖定义实体类定义登录服务类定义登录控制器前端登录页面实现测试和验证 总结附录使用Session缓存验证码前端登录页面实现代码 前言 近日心血来潮想做一个开源项目&#xff0c;目标是做一款可以适配多端、功能完备的…

【Linux C | 进程】进程环境 | 什么是进程?进程的开始、终止、存储空间布局、命令行参数、环境变量

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; &#x1f923;本文内容&#x1f923;&a…

如何无公网ip实现SSH远程访问本地局域网openEuler系统?

文章目录 1. 本地SSH连接测试2. openEuler安装Cpolar3. 配置 SSH公网地址4. 公网远程SSH连接5. 固定连接SSH公网地址6. SSH固定地址连接测试 欧拉操作系统(openEuler, 简称“欧拉”)是面向数字基础设施的操作系统,支持服务器、云计算、边缘openEuler是面向数字基础设施的操作系…

2023.1.17 关于 Redis 持久化 AOF 策略详解

目录 引言 AOF 策略 实例演示一 缓冲区 重写机制 手动触发 自动触发 AOF 重写流程 实例演示二 引言 Redis 实现持久化的两大策略 RDB ——> Redis DataBase&#xff08;定期备份&#xff09;AOF ——> Append Only File&#xff08;实时备份&#xff09; 注意&…

Linux中的软件包管理器yum

目录 1.什么是软件包 2.关于 rzsz 3.查看软件包 4.如何安装软件 5.如何卸载软件 1.什么是软件包 ● 在Linux下安装软件, 一个通常的办法是下载到程序的源代码, 并进行编译, 得到可执行程序. ● 但是这样太麻烦了, 于是有些人把一些常用的软件提前编译好, 做成软件包(可以理…

EasyExcelFactory 导入导出功能的实战使用

EasyExcelFactory 导入导出功能的实战使用分享&#xff1a; 1、jar包引入 <!-- 阿里巴巴Excel处理--><dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>3.0.6</version></dependen…

华为AC+FIT AP组网配置

AC配置 vlan batch 100 to 101dhcp enableip pool apgateway-list 192.168.100.254 network 192.168.100.0 mask 255.255.255.0 interface Vlanif100ip address 192.168.100.254 255.255.255.0dhcp select globalinterface GigabitEthernet0/0/1port link-type trunkport trun…