STM32 RTC日历程序

发布者:钱币之歌最新更新时间:2016-10-10 来源: eefocus关键字:STM32  RTC  日历程序 手机看文章 扫描二维码
随时随地手机看文章
STM32的RTC只是一个32bit 计数器,没有年月日星期等信息,比起专用RTC芯片那差很远。要实现时间日期功能,要程序实现。记得linux下面的时间也是一个32bit的计数器。一查,原来网上牛人多的是,早已实现。而且用此法实现,非常简单。完全不用考虑什么闰年,大小月等。关键函数mktime,标准函数库函数。
 
以下程序作者为“ jjldc (九九)“
 
rtc_time.h

#ifndef _RTC_TIME_H_
#define _RTC_TIME_H_
#include

extern struct tm Time_ConvUnixToCalendar(time_t t);
extern time_t Time_ConvCalendarToUnix(struct tm t);
extern time_t Time_GetUnixTime(void);
extern struct tm Time_GetCalendarTime(void);
extern void Time_SetUnixTime(time_t);
extern void Time_SetCalendarTime(struct tm t);

#endif

 
rtc_time.c

/*******************************************************************************
* 本文件实现基于RTC的日期功能,提供年月日的读写。(基于ANSI-C的time.h)
*
* 作者:jjldc (九九)
* QQ: 77058617
*
* RTC中保存的时间格式,是UNIX时间戳格式的。即一个32bit的time_t变量(实为u32)
*
* ANSI-C的标准库中,提供了两种表示时间的数据 型:
* time_t: UNIX时间戳(从1970-1-1起到某时间经过的秒数)
* typedef unsigned int time_t;
*
* struct tm: Calendar格式(年月日形式)
* tm结构如下:
* 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
* ...
* }
* 其中wday,yday可以自动产生,软件直接读取
* mon的取值为0-11
* ***注意***:
* tm_year:在time.h库中定义为1900年起的年份,即2008年应表示为2008-1900=108
* 这种表示方法对用户来说不是十分友好,与现实有较大差异。
* 所以在本文件中,屏蔽了这种差异。
* 即外部调用本文件的函数时,tm结构体类型的日期,tm_year即为2008
* 注意:若要调用系统库time.c中的函数,需要自行将tm_year-=1900
*
* 成员函数说明:
* struct tm Time_ConvUnixToCalendar(time_t t);
* 输入一个Unix时间戳(time_t),返回Calendar格式日期
* time_t Time_ConvCalendarToUnix(struct tm t);
* 输入一个Calendar格式日期,返回Unix时间戳(time_t)
* time_t Time_GetUnixTime(void);
* 从RTC取当前时间的Unix时间戳值
* struct tm Time_GetCalendarTime(void);
* 从RTC取当前时间的日历时间
* void Time_SetUnixTime(time_t);
* 输入UNIX时间戳格式时间,设置为当前RTC时间
* void Time_SetCalendarTime(struct tm t);
* 输入Calendar格式时间,设置为当前RTC时间
*
* 外部调用实例:
* 定义一个Calendar格式的日期变量:
* struct tm now;
* now.tm_year = 2008;
* now.tm_mon = 11; //12月
* now.tm_mday = 20;
* now.tm_hour = 20;
* now.tm_min = 12;
* now.tm_sec = 30;
*
* 获取当前日期时间:
* tm_now = Time_GetCalendarTime();
* 然后可以直接读tm_now.tm_wday获取星期数
*
* 设置时间:
* Step1. tm_now.xxx = xxxxxxxxx;
* Step2. Time_SetCalendarTime(tm_now);
*
* 计算两个时间的差
* struct tm t1,t2;
* t1_t = Time_ConvCalendarToUnix(t1);
* t2_t = Time_ConvCalendarToUnix(t2);
* dt = t1_t - t2_t;
* dt就是两个时间差的秒数
* dt_tm = mktime(dt); //注意dt的年份匹配,ansi库中函数为相对年份,注意超限
* 另可以参考相关资料,调用ansi-c库的格式化输出等功能,ctime,strftime等
*
*******************************************************************************/

/* Includes ------------------------------------------------------------------*/
#include "stm32f10x_lib.h"
#include "RTC_Time.h"
/* Private typedef -----------------------------------------------------------*/
/* Private define ------------------------------------------------------------*/
/* Private macro -------------------------------------------------------------*/
/* Private variables ---------------------------------------------------------*/
/* Private function prototypes -----------------------------------------------*/
void Time_Set(u32 t);
/* Private functions ---------------------------------------------------------*/


/*******************************************************************************
* Function Name : Time_ConvUnixToCalendar(time_t t)
* Description : 转换UNIX时间戳为日历时间
* Input : u32 t 当前时间的UNIX时间戳
* Output : None
* Return : struct tm
*******************************************************************************/
struct tm Time_ConvUnixToCalendar(time_t t)
{
struct tm *t_tm;
t_tm = localtime(&t);
t_tm->tm_year += 1900; //localtime转换结果的tm_year是相对值,需要转成绝对值
return *t_tm;
}

/*******************************************************************************
* Function Name : Time_ConvCalendarToUnix(struct tm t)
* Description : 写入RTC时钟当前时间
* Input : struct tm t
* Output : None
* Return : time_t
*******************************************************************************/
time_t Time_ConvCalendarToUnix(struct tm t)
{
t.tm_year -= 1900; //外部tm结构体存储的年份为2008格式
//而time.h中定义的年份格式为1900年开始的年份
//所以,在日期转换时要考虑到这个因素。
return mktime(&t);
}

/*******************************************************************************
* Function Name : Time_GetUnixTime()
* Description : 从RTC取当前时间的Unix时间戳值
* Input : None
* Output : None
* Return : time_t t
*******************************************************************************/
time_t Time_GetUnixTime(void)
{
return (time_t)RTC_GetCounter();
}

/*******************************************************************************
* Function Name : Time_GetCalendarTime()
* Description : 从RTC取当前时间的日历时间(struct tm)
* Input : None
* Output : None
* Return : time_t t
*******************************************************************************/
struct tm Time_GetCalendarTime(void)
{
time_t t_t;
struct tm t_tm;

t_t = (time_t)RTC_GetCounter();
t_tm = Time_ConvUnixToCalendar(t_t);
return t_tm;
}

/*******************************************************************************
* Function Name : Time_SetUnixTime()
* Description : 将给定的Unix时间戳写入RTC
* Input : time_t t
* Output : None
* Return : None
*******************************************************************************/
void Time_SetUnixTime(time_t t)
{
PWR_BackupAccessCmd(ENABLE);
RTC_WaitForLastTask();
RTC_SetCounter((u32)t);
RTC_WaitForLastTask();
PWR_BackupAccessCmd(DISABLE);
return;
}

/*******************************************************************************
* Function Name : Time_SetCalendarTime()
* Description : 将给定的Calendar格式时间转换成UNIX时间戳写入RTC
* Input : struct tm t
* Output : None
* Return : None
*******************************************************************************/
void Time_SetCalendarTime(struct tm t)
{
Time_SetUnixTime(Time_ConvCalendarToUnix(t));
return;
}

 

关键字:STM32  RTC  日历程序 引用地址:STM32 RTC日历程序

上一篇:USART发送与接收无意识中断嵌套
下一篇:stm32 stack 设置大小造成错误

推荐阅读最新更新时间:2024-03-16 15:14

STM32小白入门(第13天)-------RTC实时时钟和闹钟事件
一、RTC的概述 RTC就是实时时钟,详细英文 Real Time Clock。 二、详细描述 1.使用 2. 中断配置注意事项 三、程序设计 一)RTC唤醒事件 1. RTC的初始化 void rtc_init(void) { /* Enable the PWR clock ,使能电源时钟*/ RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE); /* Allow access to RTC,允许访问RTC */ PWR_BackupAccessCmd(ENABLE); /* Enable the LS
[单片机]
<font color='red'>STM32</font>小白入门(第13天)-------<font color='red'>RTC</font>实时时钟和闹钟事件
STM32待机模式Standby Mode与ADC DMA模式测试备忘
1)STM32运行十秒后自动进入待机,使用WakeUp引脚唤醒CPU后,重复以上动作。 int main(void) { time_cnt = 0; HAL_Init(); SystemClock_Config(); HAL_GPIO_WritePin(GPIOG, GPIO_PIN_13, GPIO_PIN_RESET); HAL_PWR_DisableWakeUpPin(PWR_CSR_EWUP); //禁止wakeup PA0引脚,可以用来做普通引脚功能 while (1) { HAL_Delay(1000); HAL_GPIO_TogglePin(GPIOG, GPIO_P
[单片机]
<font color='red'>STM32</font>待机模式Standby Mode与ADC DMA模式测试备忘
STM32】使用HAL库进行电机测速,原理、代码、滤波
STM32是一款非常强大的微处理器,广泛应用于各种电机控制系统中。对于电机控制系统来说,测速是非常重要的,因为只有知道电机的转速,才能控制电机的转动。在STM32中,我们可以使用HAL库来实现电机测速。 测速原理: 电机测速的原理是通过测量电机转子的转速来得到电机的转速,一般有两种测速方式,一种是使用霍尔传感器,另一种是使用反电动势法。使用霍尔传感器的测速方法是在电机转子上安装三个霍尔传感器,通过测量这三个传感器的输出电压来确定电机转子的位置和转速。使用反电动势法的测速方法是通过测量电机绕组上的反电动势来确定电机的转速。 代码实现: 使用HAL库的电机测速需要先初始化定时器,然后在中断函数中计算电机转速。具体代码实现如下: 1
[单片机]
【<font color='red'>STM32</font>】使用HAL库进行电机测速,原理、代码、滤波
stm32固件库中GPIO的工作模式
固件库中GPIO的工作模式 总结一下可分为三种工作模式 输入模式、输出模式、复用模式 1、输入模式 1.1上拉输入:默认电平上拉 1.2下拉输入:默认电平下拉 1.3浮空输入:电平不确定完全由外部输入决定,多用于按键开关 1.4模拟输入:用于ADC采集(模数转换Analog-to-digital conversion) 在输入模式的时候,输出模式是被禁止的 2、输出模式 2.1推挽模式:双mos管轮流工作通过GPIOx_ODR可控制I/O输出高低电平 2.2开漏输出:只有N-MOS管工作PMOS管保持关闭,GPIOx_ODR可控制I/O输出高电平(N-MOS导通)和高阻态(N-MOS不导通);一般只用于IIC ,SMB
[单片机]
<font color='red'>stm32</font>固件库中GPIO的工作模式
STM32 I2C 硬件中断方式实现方法
流程图如下: I2C 中断处理函数如下: /** * @brief This function handles I2C1 Event interrupt request, tx, rx * buffer and number of bytes will be changed. * @param None * @retval None */ void I2C1_EV_IRQHandler(void) { #ifdef ARC_I2C_IRQ uint32_t i2cEvent; I2C_param_struct __IO *pI2C_param; pI2C_param = ARC_get_
[单片机]
<font color='red'>STM32</font> I2C 硬件中断方式实现方法
ARM学习《十》—关于STM32RTC调试
这两天一直在调试STM32的RTC部分,本来打算弄一个万年历的,但是现在看来是暂时实现不了了。为什么这样说,因为RTC对晶振的要求非常高,必须是6p负载电容的32768晶振,这种晶振很难买,而且还很贵。下面是摘自一位网友的话: 今天到电子市场找了一下,几乎都是12.5p负载电容的32768晶振,只有一家有少量,负载电容是6p,20ppm的晶振要价是12.5p晶振的5倍,而且从外观上也看不出来,也没有测试方法能测出负载电容是6p还是12.5p。卖晶振的老板在这行干了10几年,一说到6p的32768晶振就笑了。这个要求以前就有多个公司中过招,特别是DALLAS的片子,让一家公司吃尽了苦头,焊上的许多高精度12.5p晶振被迫全部换掉,订
[单片机]
STM32+SD卡 利用FATFS文件系统创建文件夹并新建txt文件
这几天在移植FATFS文件系统,在移植过程中需要新建文件夹下创建新的cfg文件,查找资料后发现需要用到以下函数: f_mkdir( 0:/2017110223 );//新建文件夹,其中2017110223是文件夹名称 f_open(filescr1, 0:2017110223/201711011.txt , FA_OPEN_NEW_COVER);//在新建的文件夹下创建新的txt文本文件 新建完成之后要记得关闭该文件,否则就会出错。 f_res = f_close(filescr1); 在移植FATFS文件系统的时候我们可以用随时查看文件操作函数的返回值,针对这些返回值都有特定的意义,针对这些返回值我们可以很快锁定问题。 t
[单片机]
STM32直接驱动RGB接口的TFT数字彩屏设计
引言     随着工业技术的不断发展,人机界面的开发及应用空前火热,为了具有比较友好的人机界面,TFT数字彩屏被广泛的应用,但是TFT彩屏通常都不带有控制器,所以现在驱动彩屏的方案大致有2种:     ①采用ARM9或者更高级别的平台,芯片上带有TFT控制器,可以直接挂接TFT数字屏。     ②采用低端CPU处理器平台,外加TFT控制器模块,再挂接TFT数字屏。     对于方案①来说,系统的复杂度会莫名地增加,再加上该类的平台中主MCU多为BGA封装,对于需求很多小量多样化产品的客户来说,较难以接受这样的方案;而 ②方案平白无故添加了一个LCD控制器。这两种方案无论哪一种都增加了硬件成本,本文提出了一种由STM32的FSM
[电源管理]
<font color='red'>STM32</font>直接驱动RGB接口的TFT数字彩屏设计
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
更多每日新闻
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved