diff options
Diffstat (limited to 'libc/misc')
| -rw-r--r-- | libc/misc/time/time.c | 231 | 
1 files changed, 145 insertions, 86 deletions
| diff --git a/libc/misc/time/time.c b/libc/misc/time/time.c index 8176e071a..ac2fe5926 100644 --- a/libc/misc/time/time.c +++ b/libc/misc/time/time.c @@ -1,31 +1,10 @@ -/*  Copyright (C) 2002     Manuel Novoa III +/* Copyright (C) 2002-2004   Manuel Novoa III    <mjn3@codepoet.org>   * - *  This library is free software; you can redistribute it and/or - *  modify it under the terms of the GNU Library General Public - *  License as published by the Free Software Foundation; either - *  version 2 of the License, or (at your option) any later version. + * GNU Library General Public License (LGPL) version 2 or later.   * - *  This library is distributed in the hope that it will be useful, - *  but WITHOUT ANY WARRANTY; without even the implied warranty of - *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU - *  Library General Public License for more details. - * - *  You should have received a copy of the GNU Library General Public - *  License along with this library; if not, write to the Free - *  Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + * Dedicated to Toni.  See uClibc/DEDICATION.mjn3 for details.   */ -/*  ATTENTION!   ATTENTION!   ATTENTION!   ATTENTION!   ATTENTION! - * - *  Besides uClibc, I'm using this code in my libc for elks, which is - *  a 16-bit environment with a fairly limited compiler.  It would make - *  things much easier for me if this file isn't modified unnecessarily. - *  In particular, please put any new or replacement functions somewhere - *  else, and modify the makefile to use your version instead. - *  Thanks.  Manuel - * - *  ATTENTION!   ATTENTION!   ATTENTION!   ATTENTION!   ATTENTION! */ -  /* June 15, 2002     Initial Notes:   *   * Note: It is assumed throught that time_t is either long or unsigned long. @@ -132,6 +111,13 @@   * Dec 14, 2003 Fix some dst issues in _time_mktime().   *   Normalize the tm_isdst value to -1, 0, or 1.   *   If no dst for this timezone, then reset tm_isdst to 0. + * + * May 7, 2004 + *   Change clock() to allow wrapping. + *   Add timegm() function. + *   Make lookup_tzname() static (as it should have been). + *   Have strftime() get timezone information from the passed struct + *     for the %z and %Z conversions when using struct tm extensions.   */  #define _GNU_SOURCE @@ -219,6 +205,13 @@ extern struct tm *_time_t2tm(const time_t *__restrict timer,  extern time_t _time_mktime(struct tm *timeptr, int store_on_success); +extern struct tm *__time_localtime_tzi(const time_t *__restrict timer, +									   struct tm *__restrict result, +									   rule_struct *tzi); + +extern time_t _time_mktime_tzi(struct tm *timeptr, int store_on_success, +							   rule_struct *tzi); +  /**********************************************************************/  #ifdef L_asctime @@ -376,51 +369,63 @@ char *asctime_r(register const struct tm *__restrict ptm,  #include <sys/times.h> -/* Note: According to glibc... - *    CAE XSH, Issue 4, Version 2: <time.h> - *    The value of CLOCKS_PER_SEC is required to be 1 million on all - *    XSI-conformant systems. - */ -  #ifndef __BCC__  #if CLOCKS_PER_SEC != 1000000L  #error unexpected value for CLOCKS_PER_SEC!  #endif  #endif +#ifdef __UCLIBC_CLK_TCK_CONST +# if __UCLIBC_CLK_TCK_CONST > CLOCKS_PER_SEC +#  error __UCLIBC_CLK_TCK_CONST > CLOCKS_PER_SEC! +# elif __UCLIBC_CLK_TCK_CONST < 1 +#  error __UCLIBC_CLK_TCK_CONST < 1! +# endif +#endif + +/* Note: SUSv3 notes + * + *   On XSI-conformant systems, CLOCKS_PER_SEC is defined to be one million. + * + *   The value returned by clock() may wrap around on some implementations. + *   For example, on a machine with 32-bit values for clock_t, it wraps + *   after 2147 seconds. + * + * This implies that we should bitwise and with LONG_MAX. + */ +  clock_t clock(void)  {  	struct tms xtms;  	unsigned long t;  	times(&xtms); +  	t = ((unsigned long) xtms.tms_utime) + xtms.tms_stime;  #ifndef __UCLIBC_CLK_TCK_CONST -#error __UCLIBC_CLK_TCK_CONST not defined! -#endif -#undef CLK_TCK -#define CLK_TCK __UCLIBC_CLK_TCK_CONST +# error __UCLIBC_CLK_TCK_CONST not defined! -#if CLK_TCK > CLOCKS_PER_SEC -#error __UCLIBC_CLK_TCK_CONST > CLOCKS_PER_SEC! -#elif CLK_TCK < 1 -#error __UCLIBC_CLK_TCK_CONST < 1! -#endif +#elif ((CLOCKS_PER_SEC % __UCLIBC_CLK_TCK_CONST) == 0) + +	/* CLOCKS_PER_SEC == k * __UCLIBC_CLK_TCK_CONST for some integer k >= 1. */ +	return ((t * (CLOCKS_PER_SEC/__UCLIBC_CLK_TCK_CONST)) & LONG_MAX); -#if (CLK_TCK == CLOCKS_PER_SEC) -	return (t <= LONG_MAX) ? t : -1; -#elif (CLOCKS_PER_SEC % CLK_TCK) == 0 -	return (t <= (LONG_MAX / (CLOCKS_PER_SEC/CLK_TCK))) -		? t * (CLOCKS_PER_SEC/CLK_TCK) -		: -1;  #else -	return (t <= ((LONG_MAX / CLOCKS_PER_SEC) * CLK_TCK -				  + ((LONG_MAX % CLOCKS_PER_SEC) * CLK_TCK) / CLOCKS_PER_SEC)) -		? (((t / CLK_TCK) * CLOCKS_PER_SEC) -		   + (((t % CLK_TCK) * CLOCKS_PER_SEC) / CLK_TCK)) -		: -1; + +	/* Unlike the previous case, the scaling factor is not an integer. +	 * So when tms_utime, tms_stime, or their sum wraps, some of the +	 * "visible" bits in the return value are affected.  Nothing we +	 * can really do about this though other than handle tms_utime and +	 * tms_stime seperately and then sum.  But since that doesn't really +	 * buy us much, we don't bother. */ + +	return ((((t / __UCLIBC_CLK_TCK_CONST) * CLOCKS_PER_SEC) +			 + ((((t % __UCLIBC_CLK_TCK_CONST) * CLOCKS_PER_SEC) +				 / __UCLIBC_CLK_TCK_CONST)) +			 ) & LONG_MAX); +  #endif  } @@ -525,6 +530,24 @@ struct tm *localtime(const time_t *timer)  /**********************************************************************/  #ifdef L_localtime_r +struct tm *localtime_r(register const time_t *__restrict timer, +					   register struct tm *__restrict result) +{ +	TZLOCK; + +	tzset(); + +	__time_localtime_tzi(timer, result, _time_tzinfo); + +	TZUNLOCK; + +	return result; +} + +#endif +/**********************************************************************/ +#ifdef L__time_localtime_tzi +  #ifdef __UCLIBC_HAS_TM_EXTENSIONS__  struct ll_tzname_item; @@ -539,7 +562,7 @@ static ll_tzname_item_t ll_tzname[] = {  	{ NULL, "???" }		  /* Always 2nd. (invalid or out-of-memory) */  }; -const char *lookup_tzname(const char *key) +static const char *lookup_tzname(const char *key)  {  	ll_tzname_item_t *p; @@ -574,9 +597,9 @@ static const unsigned char day_cor[] = { /* non-leap */  /* Note: timezone locking is done by localtime_r. */ -static int tm_isdst(register const struct tm *__restrict ptm) +static int tm_isdst(register const struct tm *__restrict ptm, +					register rule_struct *r)  { -	register rule_struct *r = _time_tzinfo;  	long sec;  	int i, isdst, isleap, day, day0, monlen, mday;  	int oday;					/* Note: oday can be uninitialized. */ @@ -647,21 +670,18 @@ static int tm_isdst(register const struct tm *__restrict ptm)  	return (isdst & 1);  } -struct tm *localtime_r(register const time_t *__restrict timer, -					   register struct tm *__restrict result) +struct tm *__time_localtime_tzi(register const time_t *__restrict timer, +								register struct tm *__restrict result, +								rule_struct *tzi)  {  	time_t x[1];  	long offset;  	int days, dst; -	TZLOCK; - -	tzset(); -  	dst = 0;  	do {  		days = -7; -		offset = 604800L - _time_tzinfo[dst].gmt_offset; +		offset = 604800L - tzi[dst].gmt_offset;  		if (*timer > (LONG_MAX - 604800L)) {  			days = -days;  			offset = -offset; @@ -671,12 +691,11 @@ struct tm *localtime_r(register const time_t *__restrict timer,  		_time_t2tm(x, days, result);  		result->tm_isdst = dst;  #ifdef __UCLIBC_HAS_TM_EXTENSIONS__ -		result->tm_gmtoff = - _time_tzinfo[dst].gmt_offset; -		result->tm_zone = lookup_tzname(_time_tzinfo[dst].tzname); +		result->tm_gmtoff = - tzi[dst].gmt_offset; +		result->tm_zone = lookup_tzname(tzi[dst].tzname);  #endif /* __UCLIBC_HAS_TM_EXTENSIONS__ */ -	} while ((++dst < 2) && ((result->tm_isdst = tm_isdst(result)) != 0)); - -	TZUNLOCK; +	} while ((++dst < 2) +			 && ((result->tm_isdst = tm_isdst(result, tzi)) != 0));  	return result;  } @@ -694,6 +713,20 @@ time_t mktime(struct tm *timeptr)  	return  _time_mktime(timeptr, 1);  } +#endif +/**********************************************************************/ +#ifdef L_timegm +/* Like `mktime' but timeptr represents Universal Time, not local time. */ + +time_t timegm(struct tm *timeptr) +{ +	rule_struct gmt_tzinfo[2]; + +	memset(gmt_tzinfo, 0, sizeof(gmt_tzinfo)); +	strcpy(gmt_tzinfo[0].tzname, "GMT"); /* Match glibc behavior here. */ + +	return  _time_mktime_tzi(timeptr, 1, gmt_tzinfo); +}  #endif  /**********************************************************************/ @@ -913,7 +946,9 @@ size_t __XL(strftime)(char *__restrict s, size_t maxsize,  	long tzo;  	register const char *p;  	register const char *o; +#ifndef __UCLIBC_HAS_TM_EXTENSIONS__  	const rule_struct *rsp; +#endif  	const char *stack[MAX_PUSH];  	size_t count;  	size_t o_count; @@ -1027,15 +1062,28 @@ size_t __XL(strftime)(char *__restrict s, size_t maxsize,  					goto OUTPUT;  				} +#ifdef __UCLIBC_HAS_TM_EXTENSIONS__ + +#define RSP_TZUNLOCK	((void) 0) +#define RSP_TZNAME		timeptr->tm_zone +#define RSP_GMT_OFFSET	timeptr->tm_gmtoff + +#else + +#define RSP_TZUNLOCK	TZUNLOCK +#define RSP_TZNAME		rsp->tzname +#define RSP_GMT_OFFSET	rsp->gmt_offset +  				TZLOCK;  				rsp = _time_tzinfo;  				if (timeptr->tm_isdst > 0) {  					++rsp;  				} +#endif  				if (*p == 'Z') { -					o = rsp->tzname; +					o = RSP_TZNAME;  					assert(o != NULL);  #if 0  					if (!o) {	/* PARANOIA */ @@ -1043,15 +1091,15 @@ size_t __XL(strftime)(char *__restrict s, size_t maxsize,  					}  #endif  					o_count = SIZE_MAX; -					TZUNLOCK; +					RSP_TZUNLOCK;  					goto OUTPUT;  				} else {		/* z */  					*s = '+'; -					if ((tzo = -rsp->gmt_offset) < 0) { +					if ((tzo = -RSP_GMT_OFFSET) < 0) {  						tzo = -tzo;  						*s = '-';  					} -					TZUNLOCK; +					RSP_TZUNLOCK;  					++s;  					--count; @@ -1060,6 +1108,7 @@ size_t __XL(strftime)(char *__restrict s, size_t maxsize,  					i = 16 + 6;	/* 0-fill, width = 4 */  				} +  			} else {  				/* TODO: don't need year for U, W */  				for (i=0 ; i < 3 ; i++) { @@ -2087,12 +2136,32 @@ struct tm __time_tm;	/* Global shared by gmtime() and localtime(). */  /**********************************************************************/  #ifdef L__time_mktime +time_t _time_mktime(struct tm *timeptr, int store_on_success) +{ +	time_t t; + +	TZLOCK; + +	tzset(); + +	t = _time_mktime_tzi(timeptr, store_on_success, _time_tzinfo); + +	TZUNLOCK; + +	return t; +} + +#endif +/**********************************************************************/ +#ifdef L__time_mktime_tzi +  static const unsigned char vals[] = {  	31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31, /* non-leap */  	    29,  }; -time_t _time_mktime(struct tm *timeptr, int store_on_success) +time_t _time_mktime_tzi(struct tm *timeptr, int store_on_success, +						rule_struct *tzi)  {  #ifdef __BCC__  	long days, secs; @@ -2106,13 +2175,9 @@ time_t _time_mktime(struct tm *timeptr, int store_on_success)  	register const unsigned char *s;  	int d, default_dst; -	TZLOCK; - -	tzset(); -  	memcpy(p, timeptr, sizeof(struct tm)); -	if (!_time_tzinfo[1].tzname[0]) { /* No dst in this timezone, */ +	if (!tzi[1].tzname[0]) { /* No dst in this timezone, */  		p[8] = 0;				/* so set tm_isdst to 0. */  	} @@ -2150,7 +2215,7 @@ time_t _time_mktime(struct tm *timeptr, int store_on_success)  	d = p[5] - 1;  	days = -719163L + ((long)d)*365 + ((d/4) - (d/100) + (d/400) + p[3] + p[7]);  	secs = p[0] + 60*( p[1] + 60*((long)(p[2])) ) -		+ _time_tzinfo[default_dst].gmt_offset; +		+ tzi[default_dst].gmt_offset;   DST_CORRECT:  	if (secs < 0) {  		secs += 120009600L; @@ -2165,7 +2230,7 @@ time_t _time_mktime(struct tm *timeptr, int store_on_success)  	d = p[5] - 1;  	d = -719163L + d*365 + (d/4) - (d/100) + (d/400);  	secs = p[0] -		+ _time_tzinfo[default_dst].gmt_offset +		+ tzi[default_dst].gmt_offset  		+ 60*( p[1]  			   + 60*(p[2]  					 + 24*(((146073L * ((long long)(p[6])) + d) @@ -2183,7 +2248,7 @@ time_t _time_mktime(struct tm *timeptr, int store_on_success)  	d = ((struct tm *)p)->tm_isdst;  	t = secs; -	localtime_r(&t, (struct tm *)p); +	__time_localtime_tzi(&t, (struct tm *)p, tzi);  	if (t == ((time_t)(-1))) {	/* Remember, time_t can be unsigned. */  	    goto DONE; @@ -2193,8 +2258,8 @@ time_t _time_mktime(struct tm *timeptr, int store_on_success)  #ifdef __BCC__  		secs -= (days * 86400L);  #endif -		secs += (_time_tzinfo[1-default_dst].gmt_offset -				 - _time_tzinfo[default_dst].gmt_offset); +		secs += (tzi[1-default_dst].gmt_offset +				 - tzi[default_dst].gmt_offset);  		goto DST_CORRECT;  	} @@ -2205,8 +2270,6 @@ time_t _time_mktime(struct tm *timeptr, int store_on_success)   DONE: -	TZUNLOCK; -  	return t;  } @@ -2254,7 +2317,3 @@ int dysize(int year)  #endif  /**********************************************************************/ -/* Like `mktime', but for TP represents Universal Time, not local time.  */ -/* time_t timegm(struct tm *tp) */ - - | 
