497 lines
12 KiB
C
497 lines
12 KiB
C
/*!
|
|
* \file stm32_systime.c
|
|
*
|
|
* \brief System time functions implementation
|
|
*
|
|
* \copyright Revised BSD License, see section \ref LICENSE.
|
|
*
|
|
* \code
|
|
* ______ _
|
|
* / _____) _ | |
|
|
* ( (____ _____ ____ _| |_ _____ ____| |__
|
|
* \____ \| ___ | (_ _) ___ |/ ___) _ \
|
|
* _____) ) ____| | | || |_| ____( (___| | | |
|
|
* (______/|_____)_|_|_| \__)_____)\____)_| |_|
|
|
* (C)2013-2018 Semtech - STMicroelectronics
|
|
*
|
|
* \endcode
|
|
*
|
|
* \author Miguel Luis ( Semtech )
|
|
*
|
|
* \author Gregory Cristian ( Semtech )
|
|
*
|
|
* \author MCD Application Team ( STMicroelectronics International )
|
|
*/
|
|
/**
|
|
******************************************************************************
|
|
* @file stm32_systime.c
|
|
* @author MCD Application Team
|
|
* @brief System time functions implementation
|
|
******************************************************************************
|
|
* @attention
|
|
*
|
|
* Copyright (c) 2019 STMicroelectronics.
|
|
* All rights reserved.
|
|
*
|
|
* This software is licensed under terms that can be found in the LICENSE file
|
|
* in the root directory of this software component.
|
|
* If no LICENSE file comes with this software, it is provided AS-IS.
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
|
|
/* Includes ------------------------------------------------------------------*/
|
|
#include <stdio.h>
|
|
#include "stm32_systime.h"
|
|
|
|
/** @addtogroup SYS_TIME
|
|
* @{
|
|
*/
|
|
|
|
/* Private defines -----------------------------------------------------------*/
|
|
/**
|
|
* @defgroup SYS_TIME_private_defines SYS_TIME private defines
|
|
* @{
|
|
*/
|
|
|
|
/**
|
|
* @brief number of day in leap year up to the end of February
|
|
*
|
|
*/
|
|
#define END_OF_FEBRUARY_LEAP 60 //31+29
|
|
|
|
/**
|
|
* @brief number of day in leap year up to the end of july
|
|
*
|
|
*/
|
|
#define END_OF_JULY_LEAP 213 //31+29+...
|
|
|
|
/**
|
|
* @brief number of day in normal year up to the end of February
|
|
*
|
|
*/
|
|
#define END_OF_FEBRUARY_NORM 59 //31+28
|
|
|
|
/**
|
|
* @brief number of day in normal year up to the end of july
|
|
*
|
|
*/
|
|
#define END_OF_JULY_NORM 212 //31+28+...
|
|
|
|
/**
|
|
* @brief delta is referenced to Unix time
|
|
* @note UNIX time 0 = starts at 01:00:00, 01/01/1970
|
|
* but calculation is easier from Monday 1st January 1968
|
|
*
|
|
*/
|
|
#define CALC_REF_YEAR 68
|
|
|
|
/**
|
|
* @brief delta is referenced to Unix time
|
|
* @note UNIX time 0 = starts at 01:00:00, 01/01/1970
|
|
* but calculation is easier from Monday 1st January 1968
|
|
*
|
|
*/
|
|
#define CALC_REF_YEAR_TO_UNIX_REF_YEAR_COMPENSATION_IN_SECONDS ( ( TM_DAYS_IN_LEAP_YEAR + TM_DAYS_IN_YEAR ) * TM_SECONDS_IN_1DAY )
|
|
|
|
/**
|
|
* @brief month correction table of a normal year. To calculate the day number within a year
|
|
* @note error compensation is between 0 and 2 days. 2 bits per month
|
|
*
|
|
*/
|
|
#define DAYS_IN_MONTH_CORRECTION_NORM ( (uint32_t )0x99AAA0 )
|
|
|
|
/**
|
|
* @brief month correction table of a leap year. To calculate the day number within a year
|
|
* @note error compensation is between 0 and 2 days. 2 bits per month
|
|
*
|
|
*/
|
|
#define DAYS_IN_MONTH_CORRECTION_LEAP ( (uint32_t )0x445550 )
|
|
|
|
/**
|
|
* @brief find X/365.25
|
|
*
|
|
*/
|
|
/* 365.25 = (366 + 365 + 365 + 365)/4 */
|
|
#define DIV_365_25( X ) ( ( ( X ) * 91867 + 22750 ) >> 25 )
|
|
|
|
/**
|
|
* @brief find the nearest quotient of X/86400 (8640 number of seconds in one week)
|
|
*
|
|
*/
|
|
#define DIV_APPROX_86400( X ) ( ( ( X ) >> 18 ) + ( ( X ) >> 17 ) )
|
|
|
|
/**
|
|
* @brief find the nearest quotient of X/1000
|
|
*
|
|
*/
|
|
#define DIV_APPROX_1000( X ) ( ( ( X ) >> 10 ) +( ( X ) >> 16 ) + ( ( X ) >> 17 ) )
|
|
|
|
/**
|
|
* @brief find the nearest quotient of X/60
|
|
*
|
|
*/
|
|
#define DIV_APPROX_60( X ) ( ( ( X ) * 17476 ) >> 20 )
|
|
|
|
/**
|
|
* @brief find the nearest quotient of X/61
|
|
*
|
|
*/
|
|
#define DIV_APPROX_61( X ) ( ( ( X ) * 68759 ) >> 22 )
|
|
|
|
/**
|
|
* @brief Calculates mod(x,7)
|
|
*
|
|
*/
|
|
#define MODULO_7( X ) ( ( X ) -( ( ( ( ( X ) + 1 ) * 299593 ) >> 21 ) * 7 ) )
|
|
|
|
/**
|
|
* @brief Calculates ceiling( X / N )
|
|
*
|
|
*/
|
|
#define DIVC( X, N ) ( ( ( X ) + ( N ) -1 ) / ( N ) )
|
|
|
|
/**
|
|
* @brief Calculates ceiling( X / 4 )
|
|
*
|
|
*/
|
|
#define DIVC_BY_4( X ) ( ( ( X ) + 3 ) >>2 )
|
|
|
|
/**
|
|
* @brief Calculates ceiling( X / 2 )
|
|
*
|
|
*/
|
|
#define DIVC_BY_2( X ) ( ( ( X ) + 1 ) >> 1 )
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/* Private constants -----------------------------------------------------------*/
|
|
/**
|
|
* @defgroup SYSTIME_private_variable SYSTIME private constants
|
|
* @{
|
|
*/
|
|
const char *WeekDayString[]={ "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" };
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/* Private function prototypes -----------------------------------------------*/
|
|
|
|
/**
|
|
* @defgroup SYSTIME_private_function_prototypes SYSTIME private function prototypes
|
|
* @{
|
|
*/
|
|
static uint32_t CalendarGetMonth( uint32_t days, uint32_t year );
|
|
static void CalendarDiv86400( uint32_t in, uint32_t* out, uint32_t* remainder );
|
|
static uint32_t CalendarDiv61( uint32_t in );
|
|
static void CalendarDiv60( uint32_t in, uint32_t* out, uint32_t* remainder );
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/* Functions Definition ------------------------------------------------------*/
|
|
/**
|
|
* @addtogroup SYSTIME_exported_function
|
|
* @{
|
|
*/
|
|
|
|
SysTime_t SysTimeAdd( SysTime_t a, SysTime_t b )
|
|
{
|
|
SysTime_t c = { .Seconds = 0, .SubSeconds = 0 };
|
|
|
|
c.Seconds = a.Seconds + b.Seconds;
|
|
c.SubSeconds = a.SubSeconds + b.SubSeconds;
|
|
if( c.SubSeconds >= 1000 )
|
|
{
|
|
c.Seconds++;
|
|
c.SubSeconds -= 1000;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
SysTime_t SysTimeSub( SysTime_t a, SysTime_t b )
|
|
{
|
|
SysTime_t c = { .Seconds = 0, .SubSeconds = 0 };
|
|
|
|
c.Seconds = a.Seconds - b.Seconds;
|
|
c.SubSeconds = a.SubSeconds - b.SubSeconds;
|
|
if( c.SubSeconds < 0 )
|
|
{
|
|
c.Seconds--;
|
|
c.SubSeconds += 1000;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
void SysTimeSet( SysTime_t sysTime )
|
|
{
|
|
SysTime_t DeltaTime;
|
|
|
|
SysTime_t calendarTime = { .Seconds = 0, .SubSeconds = 0 };
|
|
|
|
calendarTime.Seconds = UTIL_SYSTIMDriver.GetCalendarTime( ( uint16_t* )&calendarTime.SubSeconds );
|
|
|
|
// sysTime is UNIX epoch
|
|
DeltaTime = SysTimeSub( sysTime, calendarTime );
|
|
|
|
UTIL_SYSTIMDriver.BKUPWrite_Seconds( DeltaTime.Seconds );
|
|
UTIL_SYSTIMDriver.BKUPWrite_SubSeconds( ( uint32_t ) DeltaTime.SubSeconds );
|
|
}
|
|
|
|
SysTime_t SysTimeGet( void )
|
|
{
|
|
SysTime_t calendarTime = { .Seconds = 0, .SubSeconds = 0 };
|
|
SysTime_t sysTime = { .Seconds = 0, .SubSeconds = 0 };
|
|
SysTime_t DeltaTime;
|
|
|
|
calendarTime.Seconds = UTIL_SYSTIMDriver.GetCalendarTime( ( uint16_t* )&calendarTime.SubSeconds );
|
|
|
|
DeltaTime.SubSeconds = (int16_t)UTIL_SYSTIMDriver.BKUPRead_SubSeconds();
|
|
DeltaTime.Seconds = UTIL_SYSTIMDriver.BKUPRead_Seconds();
|
|
|
|
sysTime = SysTimeAdd( DeltaTime, calendarTime );
|
|
|
|
return sysTime;
|
|
}
|
|
|
|
|
|
SysTime_t SysTimeGetMcuTime( void )
|
|
{
|
|
SysTime_t calendarTime = { .Seconds = 0, .SubSeconds = 0 };
|
|
|
|
calendarTime.Seconds = UTIL_SYSTIMDriver.GetCalendarTime( ( uint16_t* )&calendarTime.SubSeconds );
|
|
|
|
return calendarTime;
|
|
}
|
|
|
|
uint32_t SysTimeToMs( SysTime_t sysTime )
|
|
{
|
|
SysTime_t DeltaTime;
|
|
DeltaTime.SubSeconds = (int16_t)UTIL_SYSTIMDriver.BKUPRead_SubSeconds();
|
|
DeltaTime.Seconds = UTIL_SYSTIMDriver.BKUPRead_Seconds();
|
|
|
|
SysTime_t calendarTime = SysTimeSub( sysTime, DeltaTime );
|
|
return calendarTime.Seconds * 1000 + calendarTime.SubSeconds;
|
|
}
|
|
|
|
SysTime_t SysTimeFromMs( uint32_t timeMs )
|
|
{
|
|
uint32_t seconds = timeMs / 1000;
|
|
SysTime_t sysTime = { .Seconds = seconds, .SubSeconds = timeMs - seconds * 1000 };
|
|
SysTime_t DeltaTime = { 0 };
|
|
|
|
DeltaTime.SubSeconds = (int16_t)UTIL_SYSTIMDriver.BKUPRead_SubSeconds();
|
|
DeltaTime.Seconds = UTIL_SYSTIMDriver.BKUPRead_Seconds();
|
|
return SysTimeAdd( sysTime, DeltaTime );
|
|
}
|
|
|
|
uint32_t SysTimeMkTime( const struct tm* localtime )
|
|
{
|
|
uint32_t nbdays;
|
|
uint32_t nbsecs;
|
|
uint32_t year = localtime->tm_year - CALC_REF_YEAR;
|
|
uint32_t correctionMonth[4] =
|
|
{
|
|
DAYS_IN_MONTH_CORRECTION_LEAP,
|
|
DAYS_IN_MONTH_CORRECTION_NORM,
|
|
DAYS_IN_MONTH_CORRECTION_NORM,
|
|
DAYS_IN_MONTH_CORRECTION_NORM
|
|
};
|
|
|
|
nbdays = DIVC( ( TM_DAYS_IN_YEAR * 3 + TM_DAYS_IN_LEAP_YEAR ) * year, 4 );
|
|
|
|
nbdays += ( DIVC_BY_2( ( localtime->tm_mon ) * ( 30 + 31 ) ) -
|
|
( ( ( correctionMonth[year % 4] >> ( ( localtime->tm_mon ) * 2 ) ) & 0x03 ) ) );
|
|
|
|
nbdays += ( localtime->tm_mday - 1 );
|
|
|
|
// Convert from days to seconds
|
|
nbsecs = nbdays * TM_SECONDS_IN_1DAY;
|
|
|
|
nbsecs += ( ( uint32_t )localtime->tm_sec +
|
|
( ( uint32_t )localtime->tm_min * TM_SECONDS_IN_1MINUTE ) +
|
|
( ( uint32_t )localtime->tm_hour * TM_SECONDS_IN_1HOUR ) );
|
|
return nbsecs - CALC_REF_YEAR_TO_UNIX_REF_YEAR_COMPENSATION_IN_SECONDS;
|
|
}
|
|
|
|
void SysTimeLocalTime( const uint32_t timestamp, struct tm *localtime )
|
|
{
|
|
uint32_t correctionMonth[4] =
|
|
{
|
|
DAYS_IN_MONTH_CORRECTION_LEAP,
|
|
DAYS_IN_MONTH_CORRECTION_NORM,
|
|
DAYS_IN_MONTH_CORRECTION_NORM,
|
|
DAYS_IN_MONTH_CORRECTION_NORM
|
|
};
|
|
uint32_t weekDays = 1; // Monday 1st January 1968
|
|
uint32_t seconds;
|
|
uint32_t minutes;
|
|
uint32_t days;
|
|
uint32_t divOut;
|
|
uint32_t divReminder;
|
|
|
|
CalendarDiv86400( timestamp + CALC_REF_YEAR_TO_UNIX_REF_YEAR_COMPENSATION_IN_SECONDS, &days, &seconds );
|
|
|
|
// Calculates seconds
|
|
CalendarDiv60( seconds, &minutes, &divReminder );
|
|
localtime->tm_sec = ( uint8_t )divReminder;
|
|
|
|
// Calculates minutes and hours
|
|
CalendarDiv60( minutes, &divOut, &divReminder);
|
|
localtime->tm_min = ( uint8_t )divReminder;
|
|
localtime->tm_hour = ( uint8_t )divOut;
|
|
|
|
// Calculates year
|
|
localtime->tm_year = DIV_365_25( days );
|
|
days-= DIVC_BY_4( ( TM_DAYS_IN_YEAR * 3 + TM_DAYS_IN_LEAP_YEAR ) * localtime->tm_year );
|
|
|
|
localtime->tm_yday = days;
|
|
|
|
// Calculates month
|
|
localtime->tm_mon = CalendarGetMonth( days, localtime->tm_year );
|
|
|
|
// calculates weekdays
|
|
weekDays += DIVC_BY_4( ( localtime->tm_year * 5 ) );
|
|
weekDays += days;
|
|
localtime->tm_wday = MODULO_7( weekDays );
|
|
|
|
days -= ( DIVC_BY_2( ( localtime->tm_mon ) * ( 30 + 31 ) ) -
|
|
( ( ( correctionMonth[localtime->tm_year % 4] >> ( ( localtime->tm_mon ) * 2 ) ) & 0x03 ) ) );
|
|
|
|
// Convert 0 to 1 indexed.
|
|
localtime->tm_mday = days + 1;
|
|
|
|
localtime->tm_year += CALC_REF_YEAR;
|
|
|
|
localtime->tm_isdst = -1;
|
|
}
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**************************** Private functions *******************************/
|
|
|
|
/**
|
|
* @addtogroup SYSTIME_private_function
|
|
*
|
|
* @{
|
|
*/
|
|
static uint32_t CalendarGetMonth( uint32_t days, uint32_t year )
|
|
{
|
|
uint32_t month;
|
|
if( ( year % 4 ) == 0 )
|
|
{ /*leap year*/
|
|
if( days < END_OF_FEBRUARY_LEAP )
|
|
{ // January or February
|
|
// month = days * 2 / ( 30 + 31 );
|
|
month = CalendarDiv61( days * 2 );
|
|
}
|
|
else if( days < END_OF_JULY_LEAP )
|
|
{
|
|
month = CalendarDiv61( ( days - END_OF_FEBRUARY_LEAP ) * 2 ) + 2;
|
|
}
|
|
else
|
|
{
|
|
month = CalendarDiv61( ( days - END_OF_JULY_LEAP ) * 2 ) + 7;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if( days < END_OF_FEBRUARY_NORM )
|
|
{ // January or February
|
|
month = CalendarDiv61( days * 2 );
|
|
}
|
|
else if( days < END_OF_JULY_NORM )
|
|
{
|
|
month = CalendarDiv61( ( days - END_OF_FEBRUARY_NORM ) * 2 ) + 2;
|
|
}
|
|
else
|
|
{
|
|
month = CalendarDiv61( ( days - END_OF_JULY_NORM ) * 2 ) + 7;
|
|
}
|
|
}
|
|
return month;
|
|
}
|
|
|
|
static void CalendarDiv86400( uint32_t in, uint32_t* out, uint32_t* remainder )
|
|
{
|
|
#if 0
|
|
*remainder = in % SECONDS_IN_1DAY;
|
|
*out = in / SECONDS_IN_1DAY;
|
|
#else
|
|
uint32_t outTemp = 0;
|
|
uint32_t divResult = DIV_APPROX_86400( in );
|
|
|
|
while( divResult >=1 )
|
|
{
|
|
outTemp += divResult;
|
|
in -= divResult * 86400;
|
|
divResult= DIV_APPROX_86400( in );
|
|
}
|
|
if( in >= 86400 )
|
|
{
|
|
outTemp += 1;
|
|
in -= 86400;
|
|
}
|
|
|
|
*remainder = in;
|
|
*out = outTemp;
|
|
#endif
|
|
}
|
|
|
|
static uint32_t CalendarDiv61( uint32_t in )
|
|
{
|
|
#if 0
|
|
return( in / 61 );
|
|
#else
|
|
uint32_t outTemp = 0;
|
|
uint32_t divResult = DIV_APPROX_61( in );
|
|
while( divResult >=1 )
|
|
{
|
|
outTemp += divResult;
|
|
in -= divResult * 61;
|
|
divResult = DIV_APPROX_61( in );
|
|
}
|
|
if( in >= 61 )
|
|
{
|
|
outTemp += 1;
|
|
in -= 61;
|
|
}
|
|
return outTemp;
|
|
#endif
|
|
}
|
|
|
|
static void CalendarDiv60( uint32_t in, uint32_t* out, uint32_t* remainder )
|
|
{
|
|
#if 0
|
|
*remainder = in % 60;
|
|
*out = in / 60;
|
|
#else
|
|
uint32_t outTemp = 0;
|
|
uint32_t divResult = DIV_APPROX_60( in );
|
|
|
|
while( divResult >=1 )
|
|
{
|
|
outTemp += divResult;
|
|
in -= divResult * 60;
|
|
divResult = DIV_APPROX_60( in );
|
|
}
|
|
if( in >= 60 )
|
|
{
|
|
outTemp += 1;
|
|
in -= 60;
|
|
}
|
|
*remainder = in;
|
|
*out = outTemp;
|
|
#endif
|
|
}
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|