|
@@ -7,13 +7,21 @@
|
|
* Date Author Notes
|
|
* Date Author Notes
|
|
* 2019-08-21 zhangjun copy from minilibc
|
|
* 2019-08-21 zhangjun copy from minilibc
|
|
* 2020-09-07 Meco Man combine gcc armcc iccarm
|
|
* 2020-09-07 Meco Man combine gcc armcc iccarm
|
|
|
|
+ * 2021-02-05 Meco Man add timegm()
|
|
*/
|
|
*/
|
|
|
|
|
|
#include <sys/time.h>
|
|
#include <sys/time.h>
|
|
#include <rtthread.h>
|
|
#include <rtthread.h>
|
|
|
|
+#ifdef RT_USING_DEVICE
|
|
|
|
+#include <rtdevice.h>
|
|
|
|
+#endif
|
|
|
|
+
|
|
|
|
|
|
#if !defined (__IAR_SYSTEMS_ICC__)
|
|
#if !defined (__IAR_SYSTEMS_ICC__)
|
|
|
|
|
|
|
|
+/* seconds per day */
|
|
|
|
+#define SPD 24*60*60
|
|
|
|
+
|
|
/* days per month -- nonleap! */
|
|
/* days per month -- nonleap! */
|
|
const short __spm[13] =
|
|
const short __spm[13] =
|
|
{
|
|
{
|
|
@@ -34,10 +42,7 @@ const short __spm[13] =
|
|
static const char days[] = "Sun Mon Tue Wed Thu Fri Sat ";
|
|
static const char days[] = "Sun Mon Tue Wed Thu Fri Sat ";
|
|
static const char months[] = "Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec ";
|
|
static const char months[] = "Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec ";
|
|
|
|
|
|
-/* seconds per day */
|
|
|
|
-#define SPD 24*60*60
|
|
|
|
-
|
|
|
|
-int __isleap(int year)
|
|
|
|
|
|
+static int __isleap(int year)
|
|
{
|
|
{
|
|
/* every fourth year is a leap year except for century years that are
|
|
/* every fourth year is a leap year except for century years that are
|
|
* not divisible by 400. */
|
|
* not divisible by 400. */
|
|
@@ -45,6 +50,12 @@ int __isleap(int year)
|
|
return (!(year % 4) && ((year % 100) || !(year % 400)));
|
|
return (!(year % 4) && ((year % 100) || !(year % 400)));
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static void num2str(char *c, int i)
|
|
|
|
+{
|
|
|
|
+ c[0] = i / 10 + '0';
|
|
|
|
+ c[1] = i % 10 + '0';
|
|
|
|
+}
|
|
|
|
+
|
|
struct tm *gmtime_r(const time_t *timep, struct tm *r)
|
|
struct tm *gmtime_r(const time_t *timep, struct tm *r)
|
|
{
|
|
{
|
|
time_t i;
|
|
time_t i;
|
|
@@ -87,10 +98,15 @@ struct tm* gmtime(const time_t* t)
|
|
return gmtime_r(t, &tmp);
|
|
return gmtime_r(t, &tmp);
|
|
}
|
|
}
|
|
|
|
|
|
-/*TODO timezone is not supprt now */
|
|
|
|
|
|
+/*TODO: timezone */
|
|
struct tm* localtime_r(const time_t* t, struct tm* r)
|
|
struct tm* localtime_r(const time_t* t, struct tm* r)
|
|
{
|
|
{
|
|
- return gmtime_r(t, r);
|
|
|
|
|
|
+ time_t local_tz;
|
|
|
|
+ int timezone;
|
|
|
|
+
|
|
|
|
+ timezone = 0 * 3600 * 8; /* GTM: UTC+0 */
|
|
|
|
+ local_tz = *t + timezone;
|
|
|
|
+ return gmtime_r(&local_tz, r);
|
|
}
|
|
}
|
|
|
|
|
|
struct tm* localtime(const time_t* t)
|
|
struct tm* localtime(const time_t* t)
|
|
@@ -99,84 +115,10 @@ struct tm* localtime(const time_t* t)
|
|
return localtime_r(t, &tmp);
|
|
return localtime_r(t, &tmp);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+/* TODO: timezone */
|
|
time_t mktime(struct tm * const t)
|
|
time_t mktime(struct tm * const t)
|
|
{
|
|
{
|
|
- register time_t day;
|
|
|
|
- register time_t i;
|
|
|
|
- register time_t years = t->tm_year - 70;
|
|
|
|
-
|
|
|
|
- if (t->tm_sec > 60)
|
|
|
|
- {
|
|
|
|
- t->tm_min += t->tm_sec / 60;
|
|
|
|
- t->tm_sec %= 60;
|
|
|
|
- }
|
|
|
|
- if (t->tm_min > 60)
|
|
|
|
- {
|
|
|
|
- t->tm_hour += t->tm_min / 60;
|
|
|
|
- t->tm_min %= 60;
|
|
|
|
- }
|
|
|
|
- if (t->tm_hour > 24)
|
|
|
|
- {
|
|
|
|
- t->tm_mday += t->tm_hour / 24;
|
|
|
|
- t->tm_hour %= 24;
|
|
|
|
- }
|
|
|
|
- if (t->tm_mon > 12)
|
|
|
|
- {
|
|
|
|
- t->tm_year += t->tm_mon / 12;
|
|
|
|
- t->tm_mon %= 12;
|
|
|
|
- }
|
|
|
|
- while (t->tm_mday > __spm[1 + t->tm_mon])
|
|
|
|
- {
|
|
|
|
- if (t->tm_mon == 1 && __isleap(t->tm_year + 1900))
|
|
|
|
- {
|
|
|
|
- --t->tm_mday;
|
|
|
|
- }
|
|
|
|
- t->tm_mday -= __spm[t->tm_mon];
|
|
|
|
- ++t->tm_mon;
|
|
|
|
- if (t->tm_mon > 11)
|
|
|
|
- {
|
|
|
|
- t->tm_mon = 0;
|
|
|
|
- ++t->tm_year;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (t->tm_year < 70)
|
|
|
|
- return (time_t) - 1;
|
|
|
|
-
|
|
|
|
- /* Days since 1970 is 365 * number of years + number of leap years since 1970 */
|
|
|
|
- day = years * 365 + (years + 1) / 4;
|
|
|
|
-
|
|
|
|
- /* After 2100 we have to substract 3 leap years for every 400 years
|
|
|
|
- This is not intuitive. Most mktime implementations do not support
|
|
|
|
- dates after 2059, anyway, so we might leave this out for it's
|
|
|
|
- bloat. */
|
|
|
|
- if (years >= 131)
|
|
|
|
- {
|
|
|
|
- years -= 131;
|
|
|
|
- years /= 100;
|
|
|
|
- day -= (years >> 2) * 3 + 1;
|
|
|
|
- if ((years &= 3) == 3)
|
|
|
|
- years--;
|
|
|
|
- day -= years;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- day += t->tm_yday = __spm[t->tm_mon] + t->tm_mday - 1 +
|
|
|
|
- (__isleap(t->tm_year + 1900) & (t->tm_mon > 1));
|
|
|
|
-
|
|
|
|
- /* day is now the number of days since 'Jan 1 1970' */
|
|
|
|
- i = 7;
|
|
|
|
- t->tm_wday = (day + 4) % i; /* Sunday=0, Monday=1, ..., Saturday=6 */
|
|
|
|
-
|
|
|
|
- i = 24;
|
|
|
|
- day *= i;
|
|
|
|
- i = 60;
|
|
|
|
- return ((day + t->tm_hour) * i + t->tm_min) * i + t->tm_sec;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-static void num2str(char *c, int i)
|
|
|
|
-{
|
|
|
|
- c[0] = i / 10 + '0';
|
|
|
|
- c[1] = i % 10 + '0';
|
|
|
|
|
|
+ return timegm(t);
|
|
}
|
|
}
|
|
|
|
|
|
char* asctime_r(const struct tm *t, char *buf)
|
|
char* asctime_r(const struct tm *t, char *buf)
|
|
@@ -213,28 +155,6 @@ char* ctime(const time_t *timep)
|
|
|
|
|
|
#endif /* __IAR_SYSTEMS_ICC__ */
|
|
#endif /* __IAR_SYSTEMS_ICC__ */
|
|
|
|
|
|
-int gettimeofday(struct timeval *tp, void *ignore)
|
|
|
|
-{
|
|
|
|
- time_t time = 0;
|
|
|
|
-#ifdef RT_USING_DEVICE
|
|
|
|
- rt_device_t device;
|
|
|
|
- device = rt_device_find("rtc");
|
|
|
|
- RT_ASSERT(device != RT_NULL);
|
|
|
|
- rt_device_control(device, RT_DEVICE_CTRL_RTC_GET_TIME, &time);
|
|
|
|
- if (tp != RT_NULL)
|
|
|
|
- {
|
|
|
|
- tp->tv_sec = time;
|
|
|
|
- tp->tv_usec = 0;
|
|
|
|
- }
|
|
|
|
-#else
|
|
|
|
- tv->tv_sec = 0;
|
|
|
|
- tv->tv_usec = 0;
|
|
|
|
-#endif
|
|
|
|
-
|
|
|
|
- return time;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-
|
|
|
|
/**
|
|
/**
|
|
* Returns the current time.
|
|
* Returns the current time.
|
|
*
|
|
*
|
|
@@ -243,15 +163,14 @@ int gettimeofday(struct timeval *tp, void *ignore)
|
|
* @return time_t return timestamp current.
|
|
* @return time_t return timestamp current.
|
|
*
|
|
*
|
|
*/
|
|
*/
|
|
-/* for IAR 6.2 later Compiler */
|
|
|
|
-#if defined (__IAR_SYSTEMS_ICC__) && (__VER__) >= 6020000
|
|
|
|
|
|
+#if defined (__IAR_SYSTEMS_ICC__) && (__VER__) >= 6020000 /* for IAR 6.2 later Compiler */
|
|
#pragma module_name = "?time"
|
|
#pragma module_name = "?time"
|
|
#if _DLIB_TIME_USES_64
|
|
#if _DLIB_TIME_USES_64
|
|
time_t __time64(time_t *t)
|
|
time_t __time64(time_t *t)
|
|
#else
|
|
#else
|
|
time_t __time32(time_t *t)
|
|
time_t __time32(time_t *t)
|
|
#endif
|
|
#endif
|
|
-#else
|
|
|
|
|
|
+#else /* Keil & GCC */
|
|
time_t time(time_t *t)
|
|
time_t time(time_t *t)
|
|
#endif
|
|
#endif
|
|
{
|
|
{
|
|
@@ -290,3 +209,99 @@ RT_WEAK clock_t clock(void)
|
|
{
|
|
{
|
|
return rt_tick_get();
|
|
return rt_tick_get();
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+/* TODO: timezone */
|
|
|
|
+int gettimeofday(struct timeval *tp, struct timezone *tz)
|
|
|
|
+{
|
|
|
|
+ time_t time = 0;
|
|
|
|
+#ifdef RT_USING_DEVICE
|
|
|
|
+ rt_device_t device;
|
|
|
|
+ device = rt_device_find("rtc");
|
|
|
|
+ RT_ASSERT(device != RT_NULL);
|
|
|
|
+ rt_device_control(device, RT_DEVICE_CTRL_RTC_GET_TIME, &time);
|
|
|
|
+ if (tp != RT_NULL)
|
|
|
|
+ {
|
|
|
|
+ tp->tv_sec = time;
|
|
|
|
+ tp->tv_usec = 0;
|
|
|
|
+ }
|
|
|
|
+#else
|
|
|
|
+ tv->tv_sec = 0;
|
|
|
|
+ tv->tv_usec = 0;
|
|
|
|
+#endif
|
|
|
|
+
|
|
|
|
+ return time;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+time_t timegm(struct tm * const t)
|
|
|
|
+{
|
|
|
|
+ register time_t day;
|
|
|
|
+ register time_t i;
|
|
|
|
+ register time_t years = t->tm_year - 70;
|
|
|
|
+
|
|
|
|
+ if (t->tm_sec > 60)
|
|
|
|
+ {
|
|
|
|
+ t->tm_min += t->tm_sec / 60;
|
|
|
|
+ t->tm_sec %= 60;
|
|
|
|
+ }
|
|
|
|
+ if (t->tm_min > 60)
|
|
|
|
+ {
|
|
|
|
+ t->tm_hour += t->tm_min / 60;
|
|
|
|
+ t->tm_min %= 60;
|
|
|
|
+ }
|
|
|
|
+ if (t->tm_hour > 24)
|
|
|
|
+ {
|
|
|
|
+ t->tm_mday += t->tm_hour / 24;
|
|
|
|
+ t->tm_hour %= 24;
|
|
|
|
+ }
|
|
|
|
+ if (t->tm_mon > 12)
|
|
|
|
+ {
|
|
|
|
+ t->tm_year += t->tm_mon / 12;
|
|
|
|
+ t->tm_mon %= 12;
|
|
|
|
+ }
|
|
|
|
+ while (t->tm_mday > __spm[1 + t->tm_mon])
|
|
|
|
+ {
|
|
|
|
+ if (t->tm_mon == 1 && __isleap(t->tm_year + 1900))
|
|
|
|
+ {
|
|
|
|
+ --t->tm_mday;
|
|
|
|
+ }
|
|
|
|
+ t->tm_mday -= __spm[t->tm_mon];
|
|
|
|
+ ++t->tm_mon;
|
|
|
|
+ if (t->tm_mon > 11)
|
|
|
|
+ {
|
|
|
|
+ t->tm_mon = 0;
|
|
|
|
+ ++t->tm_year;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (t->tm_year < 70)
|
|
|
|
+ return (time_t) - 1;
|
|
|
|
+
|
|
|
|
+ /* Days since 1970 is 365 * number of years + number of leap years since 1970 */
|
|
|
|
+ day = years * 365 + (years + 1) / 4;
|
|
|
|
+
|
|
|
|
+ /* After 2100 we have to substract 3 leap years for every 400 years
|
|
|
|
+ This is not intuitive. Most mktime implementations do not support
|
|
|
|
+ dates after 2059, anyway, so we might leave this out for it's
|
|
|
|
+ bloat. */
|
|
|
|
+ if (years >= 131)
|
|
|
|
+ {
|
|
|
|
+ years -= 131;
|
|
|
|
+ years /= 100;
|
|
|
|
+ day -= (years >> 2) * 3 + 1;
|
|
|
|
+ if ((years &= 3) == 3)
|
|
|
|
+ years--;
|
|
|
|
+ day -= years;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ day += t->tm_yday = __spm[t->tm_mon] + t->tm_mday - 1 +
|
|
|
|
+ (__isleap(t->tm_year + 1900) & (t->tm_mon > 1));
|
|
|
|
+
|
|
|
|
+ /* day is now the number of days since 'Jan 1 1970' */
|
|
|
|
+ i = 7;
|
|
|
|
+ t->tm_wday = (day + 4) % i; /* Sunday=0, Monday=1, ..., Saturday=6 */
|
|
|
|
+
|
|
|
|
+ i = 24;
|
|
|
|
+ day *= i;
|
|
|
|
+ i = 60;
|
|
|
|
+ return ((day + t->tm_hour) * i + t->tm_min) * i + t->tm_sec;
|
|
|
|
+}
|