32单片机RTC时间接续,掉电时间保存

这篇具有很好参考价值的文章主要介绍了32单片机RTC时间接续,掉电时间保存。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1、实现思路

前提:首先要实现RTC掉电之后时间还能继续走,RTC电池是必要的

说明:设备第一次启动需要初始化配置RTC,但当二次启动再重新配置RTC会导致RTC计数器置零,所以传统的程序流程是不行的,我们需要知道设备是第一次启动还是二次启动,来判断是否需要重新初始化配置RTC。另外RTC电池会给RTC功能部分供电不代表会给MCU供电,即使是二次启动不需要再初始化RTC,并不代表就不需要初始化MCU了,MCU的部分RTC相关的功能还是需要进行配置才能获取到RTC时间。

32单片机RTC时间接续,掉电时间保存,32位单片机开发,单片机,嵌入式硬件,stm32,c语言

2、实现思路

方式一:

要区分是否是第一次启动,可能很多人第一个想法就是使用Flash,第一次启动在flash保存一个标志位,第二次启动就可以读取到标志位。

说明:实际并不推荐这种方式,如果将来RTC电池没电了,设备上电如果不初始化RTC,那么时间就不会走了,与时间相关的功能都将失效,甚至是设备死机。

方式二:

通过读取rtc计数器值判断,如果rtc没有被配置过那么rtc计数器值就是0,然后再判断要如何配置RTC及相关功能。

说明:推荐的方式,如果配置过rtc计数器开始工作,那么计数器值就不会是0,即使rtc电池没电也不过是时间无法保存,每次上电时间都会重置罢了。

方式三:

使用方式一和方式二结合的方式,通过在flash设置标志位和rtc计数器相结合的方式,第一次启动设置标志位,适用于设备带恢复出厂设置功能的需求,如果标志位未设置就初始化RTC,如果设置了再判断rtc计数器的值,这样如果我们需要重置设备,只需要清除标志位,就可以重置时间,当然也可以选择不使用flash保存标志位,直接将rtc计数器置零或其他默认值的方式也可以,根据自己需求进行选择。

代码样例:

以GD32单片机为例

RTC配置代码:

//MCU配置
void rtc_reconfig(void)
{
	/* enable PMU and BKPI clocks */
    rcu_periph_clock_enable(RCU_BKPI);
    rcu_periph_clock_enable(RCU_PMU);
    /* allow access to BKP domain */
    pmu_backup_write_enable();
    /* enable the RTC second interrupt*/
    rtc_interrupt_enable(RTC_INT_SECOND);
    rtc_interrupt_enable(RTC_INT_ALARM);
    /* wait until last write operation on RTC registers has finished */
    rtc_lwoff_wait();
	nvic_int_enable();
}
//RTC+MCU初始化
void rtc_configuration(void)
{
	time_t ltime;
    /* enable PMU and BKPI clocks */
    rcu_periph_clock_enable(RCU_BKPI);
    rcu_periph_clock_enable(RCU_PMU);
    /* allow access to BKP domain */
    pmu_backup_write_enable();

    /* reset backup domain */
    bkp_deinit();

    /* enable LXTAL */
    rcu_osci_on(RCU_LXTAL);
    /* wait till LXTAL is ready */
    rcu_osci_stab_wait(RCU_LXTAL);
    
    /* select RCU_LXTAL as RTC clock source */
    rcu_rtc_clock_config(RCU_RTCSRC_LXTAL);

    /* enable RTC Clock */
    rcu_periph_clock_enable(RCU_RTC);

    /* wait for RTC registers synchronization */
    rtc_register_sync_wait();

    /* wait until last write operation on RTC registers has finished */
    rtc_lwoff_wait();

    /* enable the RTC second interrupt*/
    rtc_interrupt_enable(RTC_INT_SECOND);
    rtc_interrupt_enable(RTC_INT_ALARM);
    /* wait until last write operation on RTC registers has finished */
    rtc_lwoff_wait();

    /* set RTC prescaler: set RTC period to 1s */
    rtc_prescaler_set(32767);
	rtc_str2tm("2024/01/16 10:21:00",&ltime);
    /* wait until last write operation on RTC registers has finished */
    rtc_lwoff_wait();
    /* change the current time */
    rtc_counter_set(ltime);
    rtc_lwoff_wait();
    /* set the alarm time = currenttime + 10 second*/
    //rtc_alarm_config((time_time2val(global_datetime.time)+10)%0x00015180);
    /* wait until last write operation on RTC registers has finished */
    //rtc_lwoff_wait();
	nvic_int_enable();
}

实用的时间转换函数:

避免大家重复造轮子,直接贴出来供大家使用

//获取某年多少天
uint16_t rtc_getyday(uint16_t year)
{
	
	uint16_t max_day = 0;
	if(year%400==0){
		max_day = 366;
	}else if(year%4==0 && year%100!=0){
		max_day = 366;
	}else{
		max_day = 365;
	}
	return max_day;
}
//获取某年的某月有多少天
uint16_t rtc_getmday(uint16_t year,uint16_t mon)
{
	uint16_t max_day = 0;
	switch(mon){
		case 1:
		case 3:
		case 5:
		case 7:
		case 8:
		case 10:
		case 12:
			max_day = 31;
			break;
		case 4:
		case 6:
		case 9:
		case 11:
			max_day = 30;
			break;
		case 2:
			if(rtc_getyday(year)==366){
				max_day = 29;
			}else{
				max_day = 28;
			}
			break;
		default:
			return 0;
			break;
	}
}
//将字符时间转换为时间戳
int rtc_str2tm(const char * ctime,time_t * ltm)
{
	uint32_t totalday;
	time_t timeval=0;
	int i;
	struct tm tmptm = {0};
	*ltm = 0;
	sscanf(ctime,"%04u/%02u/%02u %02u:%02u:%02u",&tmptm.tm_year,&tmptm.tm_mon,&tmptm.tm_mday,&tmptm.tm_hour,&tmptm.tm_min,&tmptm.tm_sec);
	if(tmptm.tm_year>2100){
		return -1;
	}
	if(tmptm.tm_mon>12){
		return -1;
	}
	if(tmptm.tm_mday>rtc_getmday(tmptm.tm_year,tmptm.tm_mon)){
		return -1;
	}
	if(tmptm.tm_hour>=24){
		return -1;
	}
	if(tmptm.tm_min>=60){
		return -1;
	}
	if(tmptm.tm_sec>=60){
		return -1;
	}
	for(i=1970;i<tmptm.tm_year;i++){
		totalday = rtc_getyday(i);
		timeval += totalday*60*60*24;
	}
	for(i=1;i<tmptm.tm_mon;i++){
		totalday = rtc_getmday(tmptm.tm_year,i);
		timeval += totalday*60*60*24;
	}
	tmptm.tm_mday--;
	timeval += tmptm.tm_mday*60*60*24;
	timeval += tmptm.tm_hour*60*60;
	timeval += tmptm.tm_min*60;
	timeval += tmptm.tm_sec;
	
	*ltm = timeval;
	return 0;
}

C语言常用时间函数库说明:

需要包含头文件time.h文章来源地址https://www.toymoban.com/news/detail-798766.html

/*   time.h   */

typedef unsigned int time_t;     /* 时间戳 date/time in unix secs past 1-Jan-70 */

struct tm {
    int tm_sec;   /* 秒 seconds after the minute, 0 to 60
                     (0 - 60 allows for the occasional leap second) */
    int tm_min;   /* 分 minutes after the hour, 0 to 59 */
    int tm_hour;  /* 时 hours since midnight, 0 to 23 */
    int tm_mday;  /* 日 day of the month, 1 to 31 */
    int tm_mon;   /* 月 months since January, 0 to 11 */
    int tm_year;  /* 年 years since 1900 */
    int tm_wday;  /* 周几 days since Sunday, 0 to 6 */
    int tm_yday;  /* 一年的第几天 days since January 1, 0 to 365 */
    int tm_isdst; /* 夏令时 Daylight Savings Time flag */
};

time_t mktime(struct tm * /*timeptr*/) /* 时间结构体转时间戳  */

struct tm *localtime(const time_t * /*timer*/)  /* 时间戳转时间结构体  */

到了这里,关于32单片机RTC时间接续,掉电时间保存的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • STC单片机掉电(停机)模式介绍和使用

    ✨在STC8/12/15单片机中都包含有掉电(省电模式)。 📑掉电模式/停机模式 🌿STC12/15/8通用。 📋将寄存器PCON的.1Bit位置为1( PCON |=0x02 ),单片机将进入掉电模式,(掉电模式也叫停机模式)进入掉电模式后,内部时钟停振,由于无时钟源,CPU、定时器、看门狗、AD转换、串行口

    2024年02月12日
    浏览(28)
  • APM32F072单片机进入STOP模式,并通过RTC Wakeup Timer和USART1串口接收事件唤醒

    串口初始化(注意USART1时钟源要选择HSI): 使用power_init函数初始化RTC,然后调用power_enter_stop_mode(n)函数进入STOP模式,n秒后自动唤醒,或由USART1接收唤醒:

    2024年02月13日
    浏览(36)
  • STM32F4单片机内部FLASH编程时间

    单片机内部的flash除了存储固件以外,经常将其分为多个区域,用来存储一些参数或存储OTA升级等待更新的固件,这时就会涉及单片机内部flash的编程和擦除操作。STM32同系列的单片机内部flash特性和扇区大小都不太一样,以下基于STM32F407VET6此型号进行简单介绍。 STM32F4xx中文参

    2024年02月03日
    浏览(43)
  • STM32单片机示例:64位全局时间戳发生器

    STM32H743 / H750 系列的芯片有一个64位的全局时间戳发生器( Global timestamp generator ),这篇文章将对它的使用做个记录。 全局时间戳发生器相关的内容可以参考官方参考手册: TGS时钟来源与APB总线时钟,这就是TGS计数器时钟了,并且用于TGS计数时没法对其进行分频操作。我们使

    2024年02月10日
    浏览(31)
  • STM32系列单片机“中断触发时间、最小中断周期、指令周期、平均执行速度、和单条指令执行时间”的问题研究

    查阅相关资料书籍和博客总结了一下知识点,以便学习巩固复习。 在学习《ARM Cortex-M3与Cortex-M4权威指南(第3版)》 这本书具体说明了触发中断需要多长时间。 在权威指南的第74页提到, Cortex-M3和Cortex-M4的中断等待非常小,只有12个周期 。也就是说触发中断后,需要12个时钟

    2024年04月10日
    浏览(32)
  • 学习笔记|定时器|STC中断|定时器时间计算|STC32G单片机视频开发教程(冲哥)|第十一集:定时器的作用和意义

    什么是定时器:定时器-与非网 上节课的一段代码: TimeCount++然后一个延时1毫秒,每运行1ms,变量就会加一。 系统已经运行了多少个毫秒。 实际使用时的代码如下, 判断按键有沿有按下的时候,我们等待按键松开,还有一个while循环。 如果没有松开,会一直死在这一行。所以,

    2024年02月09日
    浏览(44)
  • C语言+单片机-内存分布详解,全网最全,值得收藏保存

    目录 一、C语言内存分区 1. 代码区 2. 常量区 3. 全局(静态)区 4. 堆区(heap) 5. 栈区(stack) 二、STM32存储器分配 1. 随机存储器—RAM 2. 只读存储器—ROM 三、基于STM32代码验证 1. 详细代码如下 2. 运行结果如下 四、单片机中的内存分布 1.含义解释 2. 程序存储分布 3.程序占用Flash和SRA

    2024年02月09日
    浏览(30)
  • 单片机如何判断高电平时间

    单片机判断高电平时间的方法通常涉及到定时器的使用。以下是一些常见的方法: 使用定时器捕获高电平 :首先,你需要初始化定时器,并选择合适的预分频。例如,如果你的系统时钟频率是12MHz,你可能会选择预分频为1,这样定时器的工作频率就是12MHz。然后,你需要设置

    2024年01月21日
    浏览(28)
  • GD32单片机和STM32单片机的对比分析

    GD32单片机和STM32单片机都是基于Arm Cortex-M3/M4内核的32位通用微控制器,广泛应用于各种嵌入式系统和物联网领域。两者之间有很多相似之处,但也有一些不同之处,本文将从以下几个方面对比分析两者的特点、优势和开发成本。 GD32单片机采用的是二代的M3/M4内核,而STM32单片

    2024年02月16日
    浏览(46)
  • 51单片机延时程序的延时时间计算

            最近在上单片机原理及应用课程,做实验的时候遇到了软件延时程序如何计算延时时间的问题,经过一阵摸索终于领悟到其中奥秘......耶(比耶)。 延时函数是使用STC-ISP生成的,晶振为12MHz,指令集是STC-Y1(即89系列),使用do while()循环实现软件精确延时。下面讲

    2024年02月11日
    浏览(29)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包