2002-11-20 00:00:25 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Copyright (C) 2002 by Linus Nielsen Feltzing
|
|
|
|
*
|
2008-06-28 18:10:04 +00:00
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
2002-11-20 00:00:25 +00:00
|
|
|
*
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2005-02-18 13:47:17 +00:00
|
|
|
#include <stdio.h> /* get NULL */
|
2002-11-20 00:00:25 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2006-03-05 18:19:46 +00:00
|
|
|
#include "kernel.h"
|
2002-11-20 00:00:25 +00:00
|
|
|
#include "rtc.h"
|
2017-01-27 02:08:55 +00:00
|
|
|
#ifdef HAVE_RTC_IRQ
|
|
|
|
#include "rtc-target.h"
|
|
|
|
#endif
|
2002-11-20 22:46:35 +00:00
|
|
|
#include "timefuncs.h"
|
2002-11-20 00:00:25 +00:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
static struct tm tm;
|
2009-06-29 14:31:46 +00:00
|
|
|
|
2021-07-03 00:19:58 +00:00
|
|
|
time_t dostime_mktime(uint16_t dosdate, uint16_t dostime)
|
|
|
|
{
|
|
|
|
/* this knows our mktime() only uses these struct tm fields */
|
|
|
|
struct tm tm;
|
|
|
|
tm.tm_sec = ((dostime ) & 0x1f) * 2;
|
|
|
|
tm.tm_min = ((dostime >> 5) & 0x3f);
|
|
|
|
tm.tm_hour = ((dostime >> 11) );
|
|
|
|
tm.tm_mday = ((dosdate ) & 0x1f);
|
|
|
|
tm.tm_mon = ((dosdate >> 5) & 0x0f) - 1;
|
|
|
|
tm.tm_year = ((dosdate >> 9) ) + 80;
|
|
|
|
|
|
|
|
return mktime(&tm);
|
|
|
|
}
|
|
|
|
|
2021-07-07 17:26:24 +00:00
|
|
|
void dostime_localtime(time_t time, uint16_t* dosdate, uint16_t* dostime)
|
|
|
|
{
|
|
|
|
struct tm tm;
|
|
|
|
#if (CONFIG_PLATFORM & PLATFORM_NATIVE)
|
|
|
|
gmtime_r(&time, &tm);
|
|
|
|
#else
|
|
|
|
localtime_r(&time, &tm);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
*dostime = ((tm.tm_sec / 2) << 0)|
|
|
|
|
((tm.tm_min ) << 5)|
|
|
|
|
((tm.tm_hour ) << 11);
|
|
|
|
*dosdate = ((tm.tm_mday ) << 0)|
|
|
|
|
((tm.tm_mon + 1) << 5)|
|
|
|
|
((tm.tm_year - 80) << 9);
|
|
|
|
}
|
|
|
|
|
2009-06-29 14:58:02 +00:00
|
|
|
#if !CONFIG_RTC
|
2017-01-27 02:08:55 +00:00
|
|
|
static inline bool rtc_dirty(void)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int rtc_read_datetime(struct tm *tm)
|
2009-06-29 14:31:46 +00:00
|
|
|
{
|
|
|
|
tm->tm_sec = 0;
|
|
|
|
tm->tm_min = 0;
|
|
|
|
tm->tm_hour = 0;
|
|
|
|
tm->tm_mday = 1;
|
|
|
|
tm->tm_mon = 0;
|
|
|
|
tm->tm_year = 70;
|
2017-01-27 02:08:55 +00:00
|
|
|
tm->tm_wday = 4;
|
|
|
|
tm->tm_yday = 0;
|
|
|
|
return 1;
|
2009-06-29 14:31:46 +00:00
|
|
|
}
|
2009-06-29 14:58:02 +00:00
|
|
|
#endif /* !CONFIG_RTC */
|
2004-04-20 10:23:57 +00:00
|
|
|
|
2011-12-15 21:38:23 +00:00
|
|
|
#if CONFIG_RTC
|
2004-08-17 01:45:48 +00:00
|
|
|
bool valid_time(const struct tm *tm)
|
2004-04-20 10:23:57 +00:00
|
|
|
{
|
2021-07-20 04:04:36 +00:00
|
|
|
if (!tm || (tm->tm_hour < 0 || tm->tm_hour > 23 ||
|
2004-04-20 10:23:57 +00:00
|
|
|
tm->tm_sec < 0 || tm->tm_sec > 59 ||
|
|
|
|
tm->tm_min < 0 || tm->tm_min > 59 ||
|
|
|
|
tm->tm_year < 100 || tm->tm_year > 199 ||
|
|
|
|
tm->tm_mon < 0 || tm->tm_mon > 11 ||
|
2004-04-25 23:47:50 +00:00
|
|
|
tm->tm_wday < 0 || tm->tm_wday > 6 ||
|
2021-07-20 04:04:36 +00:00
|
|
|
tm->tm_mday < 1 || tm->tm_mday > 31))
|
2004-04-20 10:23:57 +00:00
|
|
|
return false;
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-01-27 02:08:55 +00:00
|
|
|
/* Don't read the RTC more than once per second
|
|
|
|
* returns true if the rtc needs to be read
|
|
|
|
* targets may override with their own implementation
|
|
|
|
*/
|
|
|
|
#ifndef HAVE_RTC_IRQ
|
|
|
|
static inline bool rtc_dirty(void)
|
2002-11-20 00:00:25 +00:00
|
|
|
{
|
2006-03-09 19:47:12 +00:00
|
|
|
static long timeout = 0;
|
2002-11-20 00:00:25 +00:00
|
|
|
|
2006-03-09 19:47:12 +00:00
|
|
|
/* Don't read the RTC more than once per second */
|
2009-10-18 15:50:30 +00:00
|
|
|
if (TIME_AFTER(current_tick, timeout))
|
2008-07-15 17:17:01 +00:00
|
|
|
{
|
2017-01-27 02:08:55 +00:00
|
|
|
/* Once per second, 1/5th of a second off */
|
|
|
|
timeout = current_tick / HZ * HZ + 6*HZ / 5;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif /* HAVE_RTC_IRQ */
|
|
|
|
#endif /* CONFIG_RTC */
|
|
|
|
|
|
|
|
struct tm *get_time(void)
|
|
|
|
{
|
|
|
|
if (rtc_dirty())
|
|
|
|
{
|
2009-09-26 14:58:32 +00:00
|
|
|
rtc_read_datetime(&tm);
|
2006-03-05 18:19:46 +00:00
|
|
|
tm.tm_isdst = -1; /* Not implemented for now */
|
2006-03-07 05:26:18 +00:00
|
|
|
}
|
2017-01-27 02:08:55 +00:00
|
|
|
|
2002-11-20 00:00:25 +00:00
|
|
|
return &tm;
|
2004-04-20 10:23:57 +00:00
|
|
|
}
|
|
|
|
|
2004-08-17 01:45:48 +00:00
|
|
|
int set_time(const struct tm *tm)
|
2004-04-20 10:23:57 +00:00
|
|
|
{
|
2007-03-16 23:47:03 +00:00
|
|
|
#if CONFIG_RTC
|
2004-04-20 10:23:57 +00:00
|
|
|
int rc;
|
2009-09-26 14:58:32 +00:00
|
|
|
|
2004-04-20 10:23:57 +00:00
|
|
|
if (valid_time(tm))
|
|
|
|
{
|
2009-09-26 14:58:32 +00:00
|
|
|
rc = rtc_write_datetime(tm);
|
2004-04-20 10:23:57 +00:00
|
|
|
|
2008-04-15 18:34:20 +00:00
|
|
|
if (rc < 0)
|
2004-04-20 10:23:57 +00:00
|
|
|
return -1;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return -2;
|
|
|
|
}
|
2009-09-26 14:58:32 +00:00
|
|
|
#else /* No RTC */
|
2004-04-20 10:23:57 +00:00
|
|
|
(void)tm;
|
2011-03-08 21:36:47 +00:00
|
|
|
return -1;
|
2009-09-26 14:58:32 +00:00
|
|
|
#endif /* RTC */
|
2002-11-20 00:00:25 +00:00
|
|
|
}
|
2006-10-19 09:42:58 +00:00
|
|
|
|
2011-12-16 00:09:28 +00:00
|
|
|
#if CONFIG_RTC
|
2009-08-11 19:30:19 +00:00
|
|
|
void set_day_of_week(struct tm *tm)
|
2009-08-11 17:54:47 +00:00
|
|
|
{
|
2009-09-26 14:58:32 +00:00
|
|
|
int y=tm->tm_year+1900;
|
|
|
|
int d=tm->tm_mday;
|
|
|
|
int m=tm->tm_mon;
|
|
|
|
static const char mo[] = { 0, 3, 2, 5, 0, 3, 5, 1, 4, 6, 2, 4 };
|
2009-08-11 17:54:47 +00:00
|
|
|
|
2009-09-26 14:58:32 +00:00
|
|
|
if(m == 0 || m == 1) y--;
|
|
|
|
tm->tm_wday = (d + mo[m] + y + y/4 - y/100 + y/400) % 7;
|
2009-08-11 17:54:47 +00:00
|
|
|
}
|
2021-05-05 16:52:19 +00:00
|
|
|
|
|
|
|
void set_day_of_year(struct tm *tm)
|
|
|
|
{
|
|
|
|
int y=tm->tm_year+1900;
|
|
|
|
int d=tm->tm_mday;
|
|
|
|
int m=tm->tm_mon;
|
|
|
|
d+=m*30;
|
|
|
|
if( ( (m>1) && !(y%4) ) && ( (y%100) || !(y%400) ) )
|
|
|
|
d++;
|
|
|
|
if(m>6)
|
|
|
|
{
|
|
|
|
d+=4;
|
|
|
|
m-=7;
|
|
|
|
}
|
|
|
|
tm->tm_yday = d + ((m+1) /2);
|
|
|
|
}
|
2011-12-16 00:09:28 +00:00
|
|
|
#endif /* CONFIG_RTC */
|
2009-09-26 14:58:32 +00:00
|
|
|
|