743 lines
20 KiB
743 lines
20 KiB
#ifdef __MINGW32__
|
|
|
|
#include "mingw.h"
|
|
|
|
#undef socket
|
|
#undef connect
|
|
#undef accept
|
|
#undef shutdown
|
|
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
|
|
int win32_poll(struct pollfd *fds, unsigned int nfds, int timo)
|
|
{
|
|
struct timeval timeout, *toptr;
|
|
fd_set ifds, ofds, efds, *ip, *op;
|
|
unsigned int i, rc;
|
|
|
|
/* Set up the file-descriptor sets in ifds, ofds and efds. */
|
|
FD_ZERO(&ifds);
|
|
FD_ZERO(&ofds);
|
|
FD_ZERO(&efds);
|
|
for (i = 0, op = ip = 0; i < nfds; ++i) {
|
|
fds[i].revents = 0;
|
|
if(fds[i].events & (POLLIN|POLLPRI)) {
|
|
ip = &ifds;
|
|
FD_SET(fds[i].fd, ip);
|
|
}
|
|
if(fds[i].events & POLLOUT) {
|
|
op = &ofds;
|
|
FD_SET(fds[i].fd, op);
|
|
}
|
|
FD_SET(fds[i].fd, &efds);
|
|
}
|
|
|
|
/* Set up the timeval structure for the timeout parameter */
|
|
if(timo < 0) {
|
|
toptr = 0;
|
|
} else {
|
|
toptr = &timeout;
|
|
timeout.tv_sec = timo / 1000;
|
|
timeout.tv_usec = (timo - timeout.tv_sec * 1000) * 1000;
|
|
}
|
|
|
|
#ifdef DEBUG_POLL
|
|
printf("Entering select() sec=%ld usec=%ld ip=%lx op=%lx\n",
|
|
(long)timeout.tv_sec, (long)timeout.tv_usec, (long)ip, (long)op);
|
|
#endif
|
|
rc = select(0, ip, op, &efds, toptr);
|
|
#ifdef DEBUG_POLL
|
|
printf("Exiting select rc=%d\n", rc);
|
|
#endif
|
|
|
|
if(rc <= 0)
|
|
return rc;
|
|
|
|
if(rc > 0) {
|
|
for ( i = 0; i < nfds; ++i) {
|
|
int fd = fds[i].fd;
|
|
if(fds[i].events & (POLLIN|POLLPRI) && FD_ISSET(fd, &ifds))
|
|
fds[i].revents |= POLLIN;
|
|
if(fds[i].events & POLLOUT && FD_ISSET(fd, &ofds))
|
|
fds[i].revents |= POLLOUT;
|
|
if(FD_ISSET(fd, &efds))
|
|
/* Some error was detected ... should be some way to know. */
|
|
fds[i].revents |= POLLHUP;
|
|
#ifdef DEBUG_POLL
|
|
printf("%d %d %d revent = %x\n",
|
|
FD_ISSET(fd, &ifds), FD_ISSET(fd, &ofds), FD_ISSET(fd, &efds),
|
|
fds[i].revents
|
|
);
|
|
#endif
|
|
}
|
|
}
|
|
return rc;
|
|
}
|
|
static void
|
|
set_connect_errno(int winsock_err)
|
|
{
|
|
switch(winsock_err) {
|
|
case WSAEINVAL:
|
|
case WSAEALREADY:
|
|
case WSAEWOULDBLOCK:
|
|
errno = EINPROGRESS;
|
|
break;
|
|
default:
|
|
errno = winsock_err;
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
set_socket_errno(int winsock_err)
|
|
{
|
|
switch(winsock_err) {
|
|
case WSAEWOULDBLOCK:
|
|
errno = EAGAIN;
|
|
break;
|
|
default:
|
|
errno = winsock_err;
|
|
break;
|
|
}
|
|
}
|
|
/*
|
|
* A wrapper around the socket() function. The purpose of this wrapper
|
|
* is to ensure that the global errno symbol is set if an error occurs,
|
|
* even if we are using winsock.
|
|
*/
|
|
SOCKET
|
|
win32_socket(int domain, int type, int protocol)
|
|
{
|
|
SOCKET fd = socket(domain, type, protocol);
|
|
if(fd == INVALID_SOCKET) {
|
|
set_socket_errno(WSAGetLastError());
|
|
}
|
|
return fd;
|
|
}
|
|
/*
|
|
* A wrapper around the connect() function. The purpose of this wrapper
|
|
* is to ensure that the global errno symbol is set if an error occurs,
|
|
* even if we are using winsock.
|
|
*/
|
|
int
|
|
win32_connect(SOCKET fd, struct sockaddr *addr, socklen_t addr_len)
|
|
{
|
|
int rc = connect(fd, addr, addr_len);
|
|
assert(rc == 0 || rc == SOCKET_ERROR);
|
|
if(rc == SOCKET_ERROR) {
|
|
set_connect_errno(WSAGetLastError());
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* A wrapper around the accept() function. The purpose of this wrapper
|
|
* is to ensure that the global errno symbol is set if an error occurs,
|
|
* even if we are using winsock.
|
|
*/
|
|
SOCKET
|
|
win32_accept(SOCKET fd, struct sockaddr *addr, socklen_t *addr_len)
|
|
{
|
|
SOCKET newfd = accept(fd, addr, addr_len);
|
|
if(newfd == INVALID_SOCKET) {
|
|
set_socket_errno(WSAGetLastError());
|
|
newfd = -1;
|
|
}
|
|
return newfd;
|
|
}
|
|
|
|
/*
|
|
* A wrapper around the shutdown() function. The purpose of this wrapper
|
|
* is to ensure that the global errno symbol is set if an error occurs,
|
|
* even if we are using winsock.
|
|
*/
|
|
int
|
|
win32_shutdown(SOCKET fd, int mode)
|
|
{
|
|
int rc = shutdown(fd, mode);
|
|
assert(rc == 0 || rc == SOCKET_ERROR);
|
|
if(rc == SOCKET_ERROR) {
|
|
set_socket_errno(WSAGetLastError());
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
int win32_close_socket(SOCKET fd)
|
|
{
|
|
int rc = closesocket(fd);
|
|
if(rc == SOCKET_ERROR) {
|
|
set_socket_errno(WSAGetLastError());
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
ssize_t win32_write_socket(SOCKET fd, void *buf, int n)
|
|
{
|
|
int rc = send(fd, buf, n, 0);
|
|
if(rc == SOCKET_ERROR) {
|
|
set_socket_errno(WSAGetLastError());
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
ssize_t win32_read_socket(SOCKET fd, void *buf, int n)
|
|
{
|
|
int rc = recv(fd, buf, n, 0);
|
|
if(rc == SOCKET_ERROR) {
|
|
set_socket_errno(WSAGetLastError());
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
|
|
char * win32_strtok_r(char *s, const char *delim, char **lasts)
|
|
{
|
|
register char *spanp;
|
|
register int c, sc;
|
|
char *tok;
|
|
|
|
|
|
if (s == NULL && (s = *lasts) == NULL)
|
|
return (NULL);
|
|
|
|
/*
|
|
* Skip (span) leading delimiters (s += strspn(s, delim), sort of).
|
|
*/
|
|
cont:
|
|
c = *s++;
|
|
for (spanp = (char *)delim; (sc = *spanp++) != 0;) {
|
|
if (c == sc)
|
|
goto cont;
|
|
}
|
|
|
|
if (c == 0) { /* no non-delimiter characters */
|
|
*lasts = NULL;
|
|
return (NULL);
|
|
}
|
|
tok = s - 1;
|
|
|
|
/*
|
|
* Scan token (scan for delimiters: s += strcspn(s, delim), sort of).
|
|
* Note that delim must have one NUL; we stop if we see that, too.
|
|
*/
|
|
for (;;) {
|
|
c = *s++;
|
|
spanp = (char *)delim;
|
|
do {
|
|
if ((sc = *spanp++) == c) {
|
|
if (c == 0)
|
|
s = NULL;
|
|
else
|
|
s[-1] = 0;
|
|
*lasts = s;
|
|
return (tok);
|
|
}
|
|
} while (sc != 0);
|
|
}
|
|
/* NOTREACHED */
|
|
}
|
|
|
|
char *win32_strsep (char **stringp, const char *delim)
|
|
{
|
|
register char *s;
|
|
register const char *spanp;
|
|
register int c, sc;
|
|
char *tok;
|
|
|
|
if ((s = *stringp) == NULL)
|
|
return (NULL);
|
|
for (tok = s;;) {
|
|
c = *s++;
|
|
spanp = delim;
|
|
do {
|
|
if ((sc = *spanp++) == c) {
|
|
if (c == 0)
|
|
s = NULL;
|
|
else
|
|
s[-1] = 0;
|
|
*stringp = s;
|
|
return (tok);
|
|
}
|
|
} while (sc != 0);
|
|
}
|
|
/* NOTREACHED */
|
|
}
|
|
|
|
#ifdef noneed
|
|
/******************************************************************************
|
|
time32.c -- extended time functions that use all 32 bits
|
|
|
|
Author: Mark Baranowski
|
|
Email: requestXXX@els-software.org (remove XXX)
|
|
Download: http://els-software.org
|
|
|
|
Last significant change: May 6, 2015
|
|
|
|
These functions are provided "as is" in the hopes that they might serve some
|
|
higher purpose. If you want these functions to serve some lower purpose,
|
|
then that too is all right. So as to keep these hopes alive you may
|
|
freely distribute these functions so long as this header remains intact.
|
|
You may also freely distribute modified versions of these functions so long
|
|
as you indicate that such versions are modified and so long as you
|
|
provide access to the unmodified original copy.
|
|
|
|
Note: The most recent version of these functions may be obtained from
|
|
http://els-software.org
|
|
|
|
The following functions support Unix time beyond Jan 19 03:14:08 2038 GMT,
|
|
assuming that the future standard will treat the 32-bits used by Unix's
|
|
present-day file system as unsigned.
|
|
|
|
These functions work by mapping years within the region 2038-2106 down
|
|
into the region of 2010-2037. All fields of the "tm" structure,
|
|
including those fields dealing with day-of-week and daylight savings time
|
|
are correct! Bear in mind, however, that the definition of daylight
|
|
savings time changes with the whims of man, thus the notion of daylight
|
|
savings held during 2010-2037 may not be the same as the notion held
|
|
thereafter.
|
|
|
|
See also: time(3)
|
|
|
|
****************************************************************************/
|
|
|
|
#include "sysdefs.h"
|
|
|
|
#include "defs.h"
|
|
#include "time32.h"
|
|
#include "sysInfo.h"
|
|
|
|
/*****************************************************************************/
|
|
#if defined(HAVE_LONG_LONG_TIME)
|
|
|
|
struct tm *localtime32_r(const time_t *clock, struct tm *res)
|
|
{return localtime_r(clock, res);}
|
|
|
|
struct tm *gmtime32_r(const time_t *clock, struct tm *res)
|
|
{return gmtime_r(clock, res);}
|
|
|
|
#ifdef USE_POSIX_TIME_R
|
|
char *asctime32_r(const struct tm *tm, char *buf)
|
|
{return asctime_r(tm, buf);}
|
|
|
|
char *ctime32_r(const time_t *clock, char *buf)
|
|
{return ctime_r(clock, buf);}
|
|
#else
|
|
char *asctime32_r(const struct tm *tm, char *buf, int buflen)
|
|
{return asctime_r(tm, buf, buflen);}
|
|
|
|
char *ctime32_r(const time_t *clock, char *buf, int buflen)
|
|
{return ctime_r(clock, buf, buflen);}
|
|
#endif
|
|
|
|
struct tm *localtime32(const time_t *clock)
|
|
{return localtime(clock);}
|
|
|
|
struct tm *gmtime32(const time_t *clock)
|
|
{return gmtime(clock);}
|
|
|
|
size_t strftime32(char *str, size_t max,
|
|
const char *format, const struct tm *tm)
|
|
{return strftime(str, max, format, tm);}
|
|
|
|
char *asctime32(const struct tm *tm)
|
|
{return asctime(tm);}
|
|
|
|
char *ctime32(const time_t *clock)
|
|
{return ctime(clock);}
|
|
|
|
#ifdef HAVE_MKTIME
|
|
time_t mktime32(const struct tm *tm)
|
|
{return mktime(tm);}
|
|
#endif
|
|
|
|
#ifdef HAVE_TIMELOCAL
|
|
/* FreeBSD/Darwin do NOT declare these args as "const": */
|
|
time_t timelocal32(struct tm *tm)
|
|
{return timelocal(tm);}
|
|
|
|
time_t timegm32(struct tm *tm)
|
|
{return timegm(tm);}
|
|
#endif
|
|
|
|
/*****************************************************************************/
|
|
#else
|
|
|
|
Local void mapclock32(time_t *clock, int *years);
|
|
Local void mapyears32(int *years, time_t *clock);
|
|
|
|
#if !defined(HAVE_TIME_R)
|
|
struct tm *localtime_r(const time_t *clock, struct tm *xtime);
|
|
struct tm *gmtime_r(const time_t *clock, struct tm *xtime);
|
|
# ifdef USE_POSIX_TIME_R
|
|
char *asctime_r(const struct tm *tm, char *buf);
|
|
char *ctime_r(const time_t *clock, char *buf);
|
|
# else
|
|
char *asctime_r(const struct tm *tm, char *buf, int buflen);
|
|
char *ctime_r(const time_t *clock, char *buf, int buflen);
|
|
# endif
|
|
#endif /* !defined(HAVE_TIME_R) */
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
struct tm *localtime32_r(const time_t *clock, struct tm *xtime)
|
|
{
|
|
time_t xclock = *clock;
|
|
int xyears;
|
|
mapclock32(&xclock, &xyears);
|
|
localtime_r(&xclock, xtime);
|
|
xtime->tm_year += xyears;
|
|
return(xtime);
|
|
}
|
|
|
|
|
|
struct tm *localtime32(const time_t *clock)
|
|
{
|
|
static struct tm xtime; /* return value must be static */
|
|
return(localtime32_r(clock, &xtime));
|
|
}
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
struct tm *gmtime32_r(const time_t *clock, struct tm *xtime)
|
|
{
|
|
time_t xclock = *clock;
|
|
int xyears;
|
|
mapclock32(&xclock, &xyears);
|
|
gmtime_r(&xclock, xtime);
|
|
xtime->tm_year += xyears;
|
|
return(xtime);
|
|
}
|
|
|
|
|
|
struct tm *gmtime32(const time_t *clock)
|
|
{
|
|
static struct tm xtime; /* return value must be static */
|
|
return(gmtime32_r(clock, &xtime));
|
|
}
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
/* NB: strftime32 will work correctly for all years > 2038 IF and only
|
|
IF the "tm" parameter given it was generated using one of the
|
|
time32 functions: localtime32(), localtime32_r(), gmtime32(), or
|
|
gmtime32_r(). */
|
|
/* NB: strftime() is known to be in SunOS4/5, HPUX10/11, Linux2.2/4/6;
|
|
So far Only SunOS5 is known to have cftime() and ascftime() */
|
|
size_t strftime32(char *str, size_t max,
|
|
const char *format, const struct tm *tm)
|
|
{
|
|
#if defined(SUNOS)
|
|
/* SunOS 5.8, 5.9, 5.10 has a quirk where "strftime(..., ..., "%a", tm)"
|
|
corrupts tzname[0] and tzname[1] for certain values of tm_year, e.g.
|
|
"edate -C 0xd0700000" currupts tzname[], but "edate -C 0xd0800000"
|
|
doesn't. Setting tm_year within spec fixes this problem, but creates
|
|
a different problem if asked to print the year. All other OSes
|
|
including SunOS5.7 appear to take tm_year at face value. */
|
|
if (osVersion == 0) osVersion = get_osVersion();
|
|
if (osVersion >= 50800)
|
|
{
|
|
struct tm xtime = *tm;
|
|
time_t xclock = 0;
|
|
mapyears32(&xtime.tm_year, &xclock);
|
|
return(strftime(str, max, format, &xtime));
|
|
}
|
|
#endif
|
|
/* Fall through for OS versions that take tm_year at face value: */
|
|
return(strftime(str, max, format, tm));
|
|
}
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
/* NB: asctime32() and asctime32_r() will work correctly for all years > 2038
|
|
IF and only IF the "tm" parameter given it was generated using one of the
|
|
time32 functions: localtime32(), localtime32_r(), gmtime32(), or
|
|
gmtime32_r(). */
|
|
|
|
#ifdef USE_POSIX_TIME_R
|
|
char *asctime32_r(const struct tm *tm, char *buf)
|
|
{
|
|
return(asctime_r(tm, buf));
|
|
}
|
|
#else
|
|
char *asctime32_r(const struct tm *tm, char *buf, int buflen)
|
|
{
|
|
return(asctime_r(tm, buf, buflen));
|
|
}
|
|
#endif
|
|
|
|
|
|
char *asctime32(const struct tm *tm)
|
|
{
|
|
# define BUF_SIZE (26+8) /* 8 bytes of slack */
|
|
static char buf[BUF_SIZE]; /* return value must be static */
|
|
#ifdef USE_POSIX_TIME_R
|
|
return(asctime32_r(tm, buf));
|
|
#else
|
|
return(asctime32_r(tm, buf, BUF_SIZE));
|
|
#endif
|
|
}
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
#ifdef USE_POSIX_TIME_R
|
|
char *ctime32_r(const time_t *clock, char *buf)
|
|
{
|
|
struct tm xtime;
|
|
return(asctime32_r(localtime32_r(clock, &xtime), buf));
|
|
}
|
|
#else
|
|
char *ctime32_r(const time_t *clock, char *buf, int buflen)
|
|
{
|
|
struct tm xtime;
|
|
return(asctime32_r(localtime32_r(clock, &xtime), buf, buflen));
|
|
}
|
|
#endif
|
|
|
|
|
|
char *ctime32(const time_t *clock)
|
|
{
|
|
# define BUF_SIZE (26+8) /* 8 bytes of slack */
|
|
static char buf[BUF_SIZE]; /* return value must be static */
|
|
#ifdef USE_POSIX_TIME_R
|
|
return(ctime32_r(clock, buf));
|
|
#else
|
|
return(ctime32_r(clock, buf, BUF_SIZE));
|
|
#endif
|
|
}
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
/* mktime is known to be supported by: HPUX 10+, SunOS5+, Linux2.2+, OSF1
|
|
mktime is known to NOT be supported by: SunOS4 */
|
|
#ifdef HAVE_MKTIME
|
|
time_t mktime32(const struct tm *tm)
|
|
{
|
|
struct tm xtime = *tm;
|
|
time_t xclock;
|
|
mapyears32(&xtime.tm_year, &xclock);
|
|
return(mktime(&xtime) + xclock);
|
|
}
|
|
#endif /*HAVE_MKTIME*/
|
|
|
|
|
|
/* timelocal and timegm are only known to be supported by SunOS4. Perhaps
|
|
older BSD-based OSes also support them, but POSIX based UNIXes do not. */
|
|
#ifdef HAVE_TIMELOCAL
|
|
time_t timelocal32(struct tm *tm)
|
|
{
|
|
struct tm xtime = *tm;
|
|
time_t xclock;
|
|
mapyears32(&xtime.tm_year, &xclock);
|
|
return(timelocal(&xtime) + xclock);
|
|
}
|
|
|
|
|
|
time_t timegm32(struct tm *tm)
|
|
{
|
|
struct tm xtime = *tm;
|
|
time_t xclock;
|
|
mapyears32(&xtime.tm_year, &xclock);
|
|
return(timegm(&xtime) + xclock);
|
|
}
|
|
#endif /*HAVE_TIMELOCAL*/
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
#define _28_YEARS ( 28 * SECS_PER_YEAR)
|
|
#define _68_YEARS ( 68 * SECS_PER_YEAR)
|
|
#define _90_YEARS ( 88 * SECS_PER_YEAR + 2*365*SECS_PER_DAY)
|
|
#define _96_YEARS ( 96 * SECS_PER_YEAR)
|
|
#define JAN_1_2100 (128 * SECS_PER_YEAR + 2*365*SECS_PER_DAY)
|
|
#define JAN_1_2102 (JAN_1_2100 + 2*365*SECS_PER_DAY) /* leap year missing */
|
|
#define JAN_1_2106 (JAN_1_2102 + 4*SECS_PER_YEAR)
|
|
#define TIME32 ((Ulong)0x80000000)
|
|
|
|
Local void mapclock32(time_t *clock, int *years)
|
|
{
|
|
Ulong xclock = (Ulong)*clock;
|
|
int xyears = 0;
|
|
|
|
/* Prevent certain processors (e.g. DEC_ALPHA) from sign extending: */
|
|
if (sizeof(Ulong) > 4) xclock &= 0xffffffff;
|
|
|
|
/* Years from 1970 up until Jan 19 03:14:08 2038 GMT need no mapping. */
|
|
|
|
if (xclock >= TIME32)
|
|
{
|
|
/* Map years beyond Jan 19 03:14:08 2038 GMT: */
|
|
if (xclock >= JAN_1_2100)
|
|
{
|
|
if (xclock < JAN_1_2102)
|
|
{
|
|
/* Map years 2100 and 2101:
|
|
(These two years must be contiguously mapped in order for
|
|
localtime(JAN_1_2101) to return the correct "tm_yday" value.
|
|
Hint: think about how Jan 1 2101, 00:00 GMT maps into
|
|
Dec 31 2100 LOCAL TIME). */
|
|
|
|
xclock -= _90_YEARS;
|
|
xyears = 90;
|
|
}
|
|
else if (xclock < JAN_1_2106)
|
|
{
|
|
/* Map years from 2102 up until 2106: */
|
|
xclock -= _68_YEARS;
|
|
xclock += SECS_PER_DAY; /* Compensate for missing leap year in 2100! */
|
|
xyears = 68;
|
|
}
|
|
else
|
|
{
|
|
/* Map years from Jan 1 2106 up until Feb 7 04:28:14 2106 GMT: */
|
|
xclock -= _96_YEARS;
|
|
xclock += SECS_PER_DAY; /* Compensate for missing leap year in 2100! */
|
|
xyears = 96;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Map years from 2038 up until 2100: */
|
|
while (xclock >= TIME32)
|
|
{
|
|
xclock -= _28_YEARS;
|
|
xyears += 28;
|
|
}
|
|
}
|
|
}
|
|
|
|
*clock = xclock;
|
|
*years = xyears;
|
|
return;
|
|
}
|
|
|
|
|
|
#define _1970 ( 70)
|
|
#define _2038 ( 68 + _1970)
|
|
#define _2100 (130 + _1970)
|
|
#define _2102 (132 + _1970)
|
|
#define _2106 (136 + _1970)
|
|
|
|
Local void mapyears32(int *years, time_t *clock)
|
|
{
|
|
Ulong xclock = 0;
|
|
int xyears = *years;
|
|
|
|
/* Years from 1970 up until Jan 19 03:14:08 2038 GMT need no mapping. */
|
|
|
|
if (xyears >= _2038)
|
|
{
|
|
/* Map years beyond Jan 19 03:14:08 2038 GMT: */
|
|
if (xyears >= _2100)
|
|
{
|
|
if (xyears < _2102)
|
|
{
|
|
/* Map years 2100 and 2101:
|
|
(These two years must be contiguously mapped in order for
|
|
localtime(JAN_1_2101) to return the correct "tm_yday" value.
|
|
Hint: think about how Jan 1 2101, 00:00 GMT maps into
|
|
Dec 31 2100 LOCAL TIME). */
|
|
|
|
xyears -= 90;
|
|
xclock = _90_YEARS;
|
|
}
|
|
else if (xyears < _2106)
|
|
{
|
|
/* Map years from 2102 up until 2106: */
|
|
xyears -= 68;
|
|
xclock = _68_YEARS;
|
|
xclock -= SECS_PER_DAY; /* Compensate for missing leap year in 2100! */
|
|
}
|
|
else
|
|
{
|
|
/* Map years from Jan 1 2106 up until Feb 7 04:28:14 2106 GMT: */
|
|
xyears -= 96;
|
|
xclock = _96_YEARS;
|
|
xclock -= SECS_PER_DAY; /* Compensate for missing leap year in 2100! */
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Map years from 2038 up until 2100: */
|
|
while (xyears >= _2038)
|
|
{
|
|
xyears -= 28;
|
|
xclock += _28_YEARS;
|
|
}
|
|
}
|
|
}
|
|
|
|
*years = xyears;
|
|
*clock = xclock;
|
|
return;
|
|
}
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
/* The following ERSATZ *time_r() routines localtime_r(), gmtime_r(),
|
|
asctime_r(), and ctime_r() are provided here as cheap substitutes to be
|
|
used ONLY by those Unixes that are lacking these functions. Moreover,
|
|
when building on a Unix that has NATIVE support for the following ERSATZ
|
|
functions be sure to define HAVE_TIME_R so that the compiler will instead
|
|
make use of its NATIVE *time_r() routines.
|
|
|
|
NOTE: If you must resort to using the following ERSATZ *time_r() routines
|
|
then as a consequence the *time32_r() functions in the previous section
|
|
will not be fully reentrant. HOWEVER, if you can AVOID using the
|
|
following routines and instead make use of your Unix's NATIVE *time_r()
|
|
routines then as a result the *time32_r() in the previous section will
|
|
also be reentrant! */
|
|
|
|
#if !defined(HAVE_TIME_R)
|
|
|
|
struct tm *localtime_r(const time_t *clock, struct tm *xtime)
|
|
{
|
|
*xtime = *localtime(clock);
|
|
return(xtime);
|
|
}
|
|
struct tm *gmtime_r(const time_t *clock, struct tm *xtime)
|
|
{
|
|
*xtime = *gmtime(clock);
|
|
return(xtime);
|
|
}
|
|
|
|
#include <string.h>
|
|
# ifdef USE_POSIX_TIME_R
|
|
char *asctime_r(const struct tm *tm, char *buf)
|
|
{
|
|
strcpy(buf, asctime(tm));
|
|
return(buf);
|
|
}
|
|
char *ctime_r(const time_t *clock, char *buf)
|
|
{
|
|
strcpy(buf, ctime(clock));
|
|
return(buf);
|
|
}
|
|
# else
|
|
char *asctime_r(const struct tm *tm, char *buf, int buflen)
|
|
{
|
|
strncpy(buf, asctime(tm), buflen);
|
|
return(buf);
|
|
}
|
|
char *ctime_r(const time_t *clock, char *buf, int buflen)
|
|
{
|
|
strncpy(buf, ctime(clock), buflen);
|
|
return(buf);
|
|
}
|
|
# endif
|
|
|
|
#endif /* !defined(HAVE_TIME_R) */
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
/*****************************************************************************/
|
|
#endif /* !defined(HAVE_LONG_LONG_TIME) */
|
|
|
|
#endif
|
|
#endif
|
|
|
|
|
|
|