2009-05-27 22:48:50 +00:00
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
/* seconds per day */
|
|
|
|
#define SPD 24*60*60
|
|
|
|
|
|
|
|
/* days per month -- nonleap! */
|
2015-01-11 20:40:51 +00:00
|
|
|
static const short __spm[13] =
|
2009-05-27 22:48:50 +00:00
|
|
|
{ 0,
|
|
|
|
(31),
|
|
|
|
(31+28),
|
|
|
|
(31+28+31),
|
|
|
|
(31+28+31+30),
|
|
|
|
(31+28+31+30+31),
|
|
|
|
(31+28+31+30+31+30),
|
|
|
|
(31+28+31+30+31+30+31),
|
|
|
|
(31+28+31+30+31+30+31+31),
|
|
|
|
(31+28+31+30+31+30+31+31+30),
|
|
|
|
(31+28+31+30+31+30+31+31+30+31),
|
|
|
|
(31+28+31+30+31+30+31+31+30+31+30),
|
|
|
|
(31+28+31+30+31+30+31+31+30+31+30+31),
|
|
|
|
};
|
|
|
|
|
2010-05-06 21:04:40 +00:00
|
|
|
static inline int isleap(int year) {
|
2009-05-27 22:48:50 +00:00
|
|
|
/* every fourth year is a leap year except for century years that are
|
|
|
|
* not divisible by 400. */
|
|
|
|
/* return (year % 4 == 0 && (year % 100 != 0 || year % 400 == 0)); */
|
|
|
|
return (!(year%4) && ((year%100) || !(year%400)));
|
|
|
|
}
|
|
|
|
|
|
|
|
struct tm *gmtime(const time_t *timep) {
|
|
|
|
static struct tm r;
|
|
|
|
time_t i;
|
|
|
|
register time_t work=*timep%(SPD);
|
|
|
|
r.tm_sec=work%60; work/=60;
|
|
|
|
r.tm_min=work%60; r.tm_hour=work/60;
|
|
|
|
work=*timep/(SPD);
|
|
|
|
r.tm_wday=(4+work)%7;
|
|
|
|
for (i=1970; ; ++i) {
|
2010-05-06 21:04:40 +00:00
|
|
|
register time_t k=isleap(i)?366:365;
|
2009-05-27 22:48:50 +00:00
|
|
|
if (work>=k)
|
|
|
|
work-=k;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
r.tm_year=i-1900;
|
|
|
|
r.tm_yday=work;
|
|
|
|
|
|
|
|
r.tm_mday=1;
|
2010-05-06 21:04:40 +00:00
|
|
|
if (isleap(i) && (work>58)) {
|
2009-05-27 22:48:50 +00:00
|
|
|
if (work==59) r.tm_mday=2; /* 29.2. */
|
|
|
|
work-=1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i=11; i && (__spm[i]>work); --i) ;
|
|
|
|
r.tm_mon=i;
|
|
|
|
r.tm_mday+=work-__spm[i];
|
|
|
|
return &r;
|
|
|
|
}
|