STM32之RTC实时时钟

一、实时时钟概述

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

JSON简单了解

文章目录 1、JSON介绍2、ES6模版字符串3、JS对象转化为JSON字符串3.1、手动JS对象转化为JSON字符串3.2、自动JS对象转化为JSON字符串 4、JS对象和java互相转换 1、JSON介绍 JSON 概念&#xff1a;JavaScript Object Notation。JavaScript 对象表示法&#xff0c;简单理解JSON是…

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

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

设计模式篇---中介者模式

文章目录 概念结构实例总结 概念 中介者模式&#xff1a;用一个中介对象来封装一系列的对象交互。中介者使各对象不需要显示地相互引用&#xff0c;从而使其耦合松散&#xff0c;而且可以独立地改变它们之间的交互。 就好比世界各个国家之间可能会产生冲突&#xff0c;但是当产…

乡镇县城小市场的外卖服务需求,现在是属于谁的机遇?

目前&#xff0c;县域经济正面临着几大利好。“返乡就业、返乡创业和告老还乡”现象越发普遍&#xff0c;这不仅在小县城中有所体现&#xff0c;同样在乡镇中也呈现出同样的趋势。一些产业链和工厂纷纷下沉到乡镇&#xff0c;带来了更多的就业机会。这不仅能够吸引年轻人回乡就…

【C初阶——基础刷题】刷题8

本文由睡觉待开机原创&#xff0c;转载请注明出处。 本内容在csdn网站首发 欢迎各位点赞—评论—收藏 如果存在不足之处请评论留言&#xff0c;共同进步&#xff01; 这里写目录标题 题目链接T1&#xff1a;T2:T3&#xff1a;T4&#xff1a;T5:T6:T7&#xff1a;T8&#xff1a;…

行云部署前端架构解析-前言 | 京东云技术团队

一个简单的自我介绍 项目规模 截止目前上万次代码提交&#xff0c;总代码行数1超过21万行&#xff0c;其中人工维护的代码超过 13万行&#xff0c;近千个文件。 前端线上服务直接对接的后端服务&#xff0c;达十多个。 跟很多应用一样, 它有行云的入口, 也有独立的服务, 还…

【Linux】python版本控制和环境管理

文章目录 1.查看目前python的版本2.添加软件源并更新3.选择你想要下载的版本4.警示&#xff1a;没必要设置默认版本误区千万千万不要覆盖python3软链接解决办法 5.pip软件包管理最省心稍微麻烦换源 网上有很多教程都是教导小白去官方下载之后编译安装。但是&#xff0c;小白连c…

【数据结构】 循环单链表的基本操作 (C语言版)

目录 一、循环单链表 1、循环单链表的定义&#xff1a; 2、循环单链表的优缺点&#xff1a; 二、循环单链表的基本操作算法&#xff08;C语言&#xff09; 1、宏定义 2、创建结构体 3、循环单链表的初始化 4、循环单链表的插入 5、求单链表长度 6、循环单链表的清空…

<信息安全>《2 国内主要企业网络安全公司概览(二)》

4 北京天融信科技有限公司(简称天融信) 信息内容LOGO成立日期创始于1995年总部北京市海淀区上地东路1号院3号楼北侧301室背景民营企业是否上市天融信[002212]A股市值99亿主要产品网络安全大数据云服务员工规模6000多人简介天融信科技集团&#xff08;证券代码&#xff1a;0022…

Java多线程并发篇----第二十九篇

系列文章目录 文章目录 系列文章目录前言一、什么是不可变对象,它对写并发应用有什么帮助二、Java 中用到的线程调度算法是什么?三、什么是线程组,为什么在 Java 中不推荐使用?前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点…

功能最强大的在线考试、投票系统,支持一键部署,开源了

简介 功能最强大&搭建最简单&界面更美观的在线考试/调查问卷/公开查询/题库刷题/360度评估/投票系统&#xff0c;支持一键部署。 特性 支持 20 多种题型&#xff0c;如填空、选择、下拉、级联、矩阵、分页、签名、题组、上传、横向填空等 多种创建问卷方式&#xff…

应用层—HTTP详解(抓包工具、报文格式、构造http等……)

文章目录 HTTP1. 抓包工具的使用1.1 配置信息1.2 观察数据 2. 分析 https 抓包结果3. HTTP请求详解3.1 认识 URL3.1.1 URL 基本格式3.1.2 查询字符串 (query string)3.1.3 关于 URL Encode 3.2 认识 http 方法3.2.1 [经典问题] Get 和 Post 主要的区别是什么&#xff1f;&#…

【AI】小白入门笔记

前言 2024年&#xff0c;愿新年胜旧年&#xff01;作为AI世界的小白&#xff0c;今天先来从一些概念讲起&#xff0c;希望路过的朋友们多多指教&#xff01; 正文 AI (人工智能) 提起AI, 大家可能会想起各种机器人&#xff0c;移动手机的“Siri”,"小爱同学", 是语…

给创业者的一份忠告~创业就要选择低成本的轻资产创业项目

王健林继续出售资产&#xff0c;5天卖了4座万达广场&#xff0c;出售计划仍在继续..... 根据天眼查app最新消息&#xff0c;厦门殿前万达广场商业管理有限公司发生工商变更&#xff0c;2023至今已经转让10座万达广场。两年前万达旗下的核心子公司万达商管&#xff0c;向一群投资…

Java开发分析工具 JProfiler的详细使用方法解析(附 JProfiler for Mac许可证秘钥)

JProfiler 是一款功能强大的Java代码分析工具&#xff0c;JProfiler的直观UI可帮助您解决性能瓶颈&#xff0c;确定内存泄漏并了解线程问题且JProfiler Mac破解版配置会话非常简单&#xff0c;第三方集成使得入门变得轻而易举&#xff0c;并且以自然的方式呈现数据分析。 解…

【Linux】Linux开发工具 - vim的基本操作

IDE例子 Linux编辑器-vim使用 vi/vim的区别简单点来说&#xff0c;它们都是多模式编辑器&#xff0c;不同的是vim是vi的升级版本&#xff0c;它不仅兼容vi的所有指令&#xff0c;而且还有一些新的特性在里面。例如语法加亮&#xff0c;可视化操作不仅可以在终端运行&#xff…

NVM (Node Version Manager) 安装使用

博文目录 文章目录 管理工具安装使用 管理工具 GitHub, nvm-windows nvm-windows: Similar (not identical) to nvm, but for Windows 管理 Node.js 版本有多种工具可选择, 其中使用最广泛的是 nvm, 目前 72.3k Star, 不支持 Windows 系统, nvm-windows, 是其他大佬为 Windows…

若依管理系统搭建教程,ruoyi-vue环境搭建

环境部署 准备工作 JDK > 1.8 (推荐1.8版本) Mysql > 5.7.0 (推荐5.7版本) Maven > 3.0 运行系统 1、前往Gitee下载页面([https://gitee.com/y_project/RuoYi (opens new window)](https://gitee.com/y_project/RuoYi))下载解压到工作目录 2、导入到Eclipse&#…

SpringMVC第一天

简介 SpringMVC技术与Servlet技术功能等同&#xff0c;均属于web层开发技术 SpringMVC是一种基于java实现的MVC模型的轻量级Web框架 优点 使用简单,开发便捷(相比于Servlet) 灵活性强 入门案例 第一步、导入SpringMVC与Servlet坐标 <?xml version"1.0" encod…

【数据结构与算法】之字符串系列-20240121

这里写目录标题 一、344. 反转字符串二、125. 验证回文串三、205. 同构字符串四、242. 有效的字母异位词五、290. 单词规律 一、344. 反转字符串 简单 编写一个函数&#xff0c;其作用是将输入的字符串反转过来。输入字符串以字符数组 s 的形式给出。 不要给另外的数组分配额…