mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 18:54:53 +00:00 
			
		
		
		
	 b376a4ad18
			
		
	
	
		b376a4ad18
		
	
	
	
	
		
			
			* posixmodule.c: don't prototype getcwd() -- it's not portable...
* mappingobject.c: double-check validity of last_name_char in
  dict{lookup,insert,remove}.
* arraymodule.c: need memmove only for non-STDC Suns.
* Makefile: comment out HTML_LIBS and XT_USE by default
* pythonmain.c: don't prototype getopt() -- it's not standardized
* socketmodule.c: cast flags arg to {get,set}sockopt() and addrbuf arg to
  recvfrom() to (ANY*).
* pythonrun.c (initsigs): fix prototype, make it static
* intobject.c (LONG_BIT): only #define it if not already defined
* classobject.[ch]: remove all references to unused instance_convert()
* mappingobject.c (getmappingsize): Don't return NULL in int function.
		
	
			
		
			
				
	
	
		
			509 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			509 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***********************************************************
 | |
| Copyright 1991, 1992, 1993 by Stichting Mathematisch Centrum,
 | |
| Amsterdam, The Netherlands.
 | |
| 
 | |
|                         All Rights Reserved
 | |
| 
 | |
| Permission to use, copy, modify, and distribute this software and its 
 | |
| documentation for any purpose and without fee is hereby granted, 
 | |
| provided that the above copyright notice appear in all copies and that
 | |
| both that copyright notice and this permission notice appear in 
 | |
| supporting documentation, and that the names of Stichting Mathematisch
 | |
| Centrum or CWI not be used in advertising or publicity pertaining to
 | |
| distribution of the software without specific, written prior permission.
 | |
| 
 | |
| STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO
 | |
| THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
 | |
| FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE
 | |
| FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 | |
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 | |
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
 | |
| OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 | |
| 
 | |
| ******************************************************************/
 | |
| 
 | |
| /* Time module */
 | |
| 
 | |
| #include "allobjects.h"
 | |
| #include "modsupport.h"
 | |
| #include "ceval.h"
 | |
| 
 | |
| #include "sigtype.h"
 | |
| 
 | |
| #include <signal.h>
 | |
| #include <setjmp.h>
 | |
| 
 | |
| #ifdef BSD_TIME
 | |
| #define HAVE_GETTIMEOFDAY
 | |
| #include "myselect.h" /* Implies <sys/types.h>, <sys/time.h>, <sys/param.h> */
 | |
| #endif
 | |
| 
 | |
| #ifdef macintosh
 | |
| #define NO_UNISTD
 | |
| #endif
 | |
| 
 | |
| #ifndef NO_UNISTD
 | |
| #include <unistd.h>
 | |
| #endif
 | |
| 
 | |
| /* What happens here is not trivial.
 | |
|    The BSD_TIME code needs <sys/time.h> (for struct timeval).
 | |
|    The rest of the code needs only time_t, except some MS-DOS
 | |
|    code which needs clock_t as well.
 | |
|    Standard C says that time_t is defined in <time.h>, and
 | |
|    does not have <sys/types.h>; THINK C agrees (MS-DOS too?).
 | |
|    What's worse, in pure 4.3 BSD, older SunOS versions, and
 | |
|    probably everything derived from BSD, you can't #include
 | |
|    both <time.h> and <sys/time.h> in the same file, since
 | |
|    <sys/time.h> includes <time.h> without any protection,
 | |
|    and <time.h> contains a typedef, which can't be parsed twice!
 | |
|    So on traditional UNIX systems we include <sys/types.h>
 | |
|    and <sys/time.h> and hope this implies <time.h> and time_t,
 | |
|    while on other systems, including conforming Standard C
 | |
|    systems (where 'unix' can't be defined), we rely on <time.h>.
 | |
|    Still one problem: BSD_TIME won't work with strict Standard C...
 | |
| */
 | |
| 
 | |
| #ifdef unix
 | |
| #include <sys/types.h>
 | |
| #include <sys/time.h> /* Implies <time.h> everywhere, as far as I know */
 | |
| #else /* !unix */
 | |
| #include <time.h>
 | |
| #endif /* !unix */
 | |
| 
 | |
| #ifdef SYSV
 | |
| #if defined(sun) && defined(__STDC__)
 | |
| /* Temporary hack for Solaris 2. */
 | |
| #define _timezone timezone
 | |
| #define _altzone altzone
 | |
| #define _daylight daylight
 | |
| #define _tzname tzname
 | |
| #endif
 | |
| /* Access timezone stuff */
 | |
| #ifdef OLDTZ				/* ANSI prepends underscore to these */
 | |
| #define _timezone	timezone	/* seconds to be added to GMT */
 | |
| #define _altzone	0		/* _timezone if daylight saving time */
 | |
| #define _daylight	0		/* if zero, _altzone is not available*/
 | |
| #define _tzname		tzname		/* Name of timezone and altzone */
 | |
| #endif
 | |
| #ifdef NOALTTZ				/* if system doesn't support alt tz */
 | |
| #undef _daylight
 | |
| #undef _altzone
 | |
| #define _daylight	0
 | |
| #define _altzone 	0
 | |
| #endif
 | |
| #endif /* SYSV */
 | |
| 
 | |
| /* Forward declarations */
 | |
| static void floatsleep PROTO((double));
 | |
| static long millitimer PROTO((void)); 
 | |
| 
 | |
| /* Time methods */
 | |
| 
 | |
| static object *
 | |
| time_time(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| #ifdef HAVE_GETTIMEOFDAY
 | |
| 	struct timeval t;
 | |
| 	struct timezone tz;
 | |
| 	if (!getnoarg(args))
 | |
| 		return NULL;
 | |
| 	if (gettimeofday(&t, &tz) != 0) {
 | |
| 		err_errno(IOError);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	return newfloatobject(t.tv_sec*1.0 + t.tv_usec*0.000001);
 | |
| #else /* !HAVE_GETTIMEOFDAY */
 | |
| 	time_t secs;
 | |
| 	if (!getnoarg(args))
 | |
| 		return NULL;
 | |
| 	time(&secs);
 | |
| #ifdef macintosh
 | |
| /* The Mac epoch is 1904, while UNIX uses 1970; Python prefers 1970 */
 | |
| /* Moreover, the Mac returns local time.  This we cannot fix... */
 | |
| #define TIMEDIFF ((time_t) \
 | |
| 	(((1970-1904)*365L + (1970-1904)/4) * 24 * 3600))
 | |
| 	secs -= TIMEDIFF;
 | |
| #endif
 | |
| 	return newfloatobject((double)secs);
 | |
| #endif /* !HAVE_GETTIMEOFDAY */
 | |
| }
 | |
| 
 | |
| static jmp_buf sleep_intr;
 | |
| 
 | |
| /* ARGSUSED */
 | |
| static void
 | |
| sleep_catcher(sig)
 | |
| 	int sig; /* Not used but required by interface */
 | |
| {
 | |
| 	longjmp(sleep_intr, 1);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| time_sleep(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| 	double secs;
 | |
| 	SIGTYPE (*sigsave)() = 0; /* Initialized to shut lint up */
 | |
| 	if (!getargs(args, "d", &secs))
 | |
| 		return NULL;
 | |
| 	BGN_SAVE
 | |
| 	if (setjmp(sleep_intr)) {
 | |
| 		RET_SAVE
 | |
| 		signal(SIGINT, sigsave);
 | |
| 		err_set(KeyboardInterrupt);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	sigsave = signal(SIGINT, SIG_IGN);
 | |
| 	if (sigsave != (SIGTYPE (*)()) SIG_IGN)
 | |
| 		signal(SIGINT, sleep_catcher);
 | |
| 	floatsleep(secs);
 | |
| 	END_SAVE
 | |
| 	signal(SIGINT, sigsave);
 | |
| 	INCREF(None);
 | |
| 	return None;
 | |
| }
 | |
| 
 | |
| #ifdef macintosh
 | |
| #define DO_MILLI
 | |
| #endif
 | |
| 
 | |
| #ifdef AMOEBA
 | |
| #define DO_MILLI
 | |
| extern long sys_milli();
 | |
| #define millitimer sys_milli
 | |
| #endif /* AMOEBA */
 | |
| 
 | |
| #ifdef BSD_TIME
 | |
| #define DO_MILLI
 | |
| #endif /* BSD_TIME */
 | |
| 
 | |
| #ifdef MSDOS
 | |
| #define DO_MILLI
 | |
| #endif
 | |
| 
 | |
| #ifdef DO_MILLI
 | |
| 
 | |
| static object *
 | |
| time_millisleep(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| 	long msecs;
 | |
| 	SIGTYPE (*sigsave)();
 | |
| 	if (!getlongarg(args, &msecs))
 | |
| 		return NULL;
 | |
| 	BGN_SAVE
 | |
| 	if (setjmp(sleep_intr)) {
 | |
| 		RET_SAVE
 | |
| 		signal(SIGINT, sigsave);
 | |
| 		err_set(KeyboardInterrupt);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	sigsave = signal(SIGINT, SIG_IGN);
 | |
| 	if (sigsave != (SIGTYPE (*)()) SIG_IGN)
 | |
| 		signal(SIGINT, sleep_catcher);
 | |
| 	floatsleep(msecs / 1000.0);
 | |
| 	END_SAVE
 | |
| 	signal(SIGINT, sigsave);
 | |
| 	INCREF(None);
 | |
| 	return None;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| time_millitimer(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| 	long msecs;
 | |
| 	if (!getnoarg(args))
 | |
| 		return NULL;
 | |
| 	msecs = millitimer();
 | |
| 	return newintobject(msecs);
 | |
| }
 | |
| 
 | |
| #endif /* DO_MILLI */
 | |
| 
 | |
| 
 | |
| static object *
 | |
| time_convert(when, function)
 | |
| 	time_t when;
 | |
| 	struct tm * (*function) PROTO((time_t *));
 | |
| {
 | |
| 	struct tm *p = function(&when);
 | |
| 	return mkvalue("(iiiiiiiii)",
 | |
| 		       p->tm_year + 1900,
 | |
| 		       p->tm_mon + 1, /* Want January == 1 */
 | |
| 		       p->tm_mday,
 | |
| 		       p->tm_hour,
 | |
| 		       p->tm_min,
 | |
| 		       p->tm_sec,
 | |
| 		       (p->tm_wday + 6) % 7, /* Want Monday == 0 */
 | |
| 		       p->tm_yday + 1, /* Want January, 1 == 1 */
 | |
| 		       p->tm_isdst);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| time_gmtime(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| 	double when;
 | |
| 	if (!getargs(args, "d", &when))
 | |
| 		return NULL;
 | |
| 	return time_convert((time_t)when, gmtime);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| time_localtime(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| 	double when;
 | |
| 	if (!getargs(args, "d", &when))
 | |
| 		return NULL;
 | |
| 	return time_convert((time_t)when, localtime);
 | |
| }
 | |
| 
 | |
| static int
 | |
| gettmarg(args, p)
 | |
| 	object *args;
 | |
| 	struct tm *p;
 | |
| {
 | |
| 	if (!getargs(args, "(iiiiiiiii)",
 | |
| 		     &p->tm_year,
 | |
| 		     &p->tm_mon,
 | |
| 		     &p->tm_mday,
 | |
| 		     &p->tm_hour,
 | |
| 		     &p->tm_min,
 | |
| 		     &p->tm_sec,
 | |
| 		     &p->tm_wday,
 | |
| 		     &p->tm_yday,
 | |
| 		     &p->tm_isdst))
 | |
| 		return 0;
 | |
| 	if (p->tm_year >= 1900)
 | |
| 		p->tm_year -= 1900;
 | |
| 	p->tm_mon--;
 | |
| 	p->tm_wday = (p->tm_wday + 1) % 7;
 | |
| 	p->tm_yday--;
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static object *
 | |
| time_asctime(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| 	struct tm buf;
 | |
| 	char *p;
 | |
| 	if (!gettmarg(args, &buf))
 | |
| 		return NULL;
 | |
| 	p = asctime(&buf);
 | |
| 	if (p[24] == '\n')
 | |
| 		p[24] = '\0';
 | |
| 	return newstringobject(p);
 | |
| }
 | |
| 
 | |
| static object *
 | |
| time_ctime(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| 	double dt;
 | |
| 	time_t tt;
 | |
| 	char *p;
 | |
| 	if (!getargs(args, "d", &dt))
 | |
| 		return NULL;
 | |
| 	tt = dt;
 | |
| 	p = ctime(&tt);
 | |
| 	if (p[24] == '\n')
 | |
| 		p[24] = '\0';
 | |
| 	return newstringobject(p);
 | |
| }
 | |
| 
 | |
| /* Some very old systems may not have mktime().  Comment it out then! */
 | |
| 
 | |
| static object *
 | |
| time_mktime(self, args)
 | |
| 	object *self;
 | |
| 	object *args;
 | |
| {
 | |
| 	struct tm buf;
 | |
| 	if (!gettmarg(args, &buf))
 | |
| 		return NULL;
 | |
| 	return newintobject((long)mktime(&buf));
 | |
| }
 | |
| 
 | |
| static struct methodlist time_methods[] = {
 | |
| #ifdef DO_MILLI
 | |
| 	{"millisleep",	time_millisleep},
 | |
| 	{"millitimer",	time_millitimer},
 | |
| #endif /* DO_MILLI */
 | |
| 	{"sleep",	time_sleep},
 | |
| 	{"time",	time_time},
 | |
| 	{"gmtime",	time_gmtime},
 | |
| 	{"localtime",	time_localtime},
 | |
| 	{"asctime",	time_asctime},
 | |
| 	{"ctime",	time_ctime},
 | |
| 	{"mktime",	time_mktime},
 | |
| 	{NULL,		NULL}		/* sentinel */
 | |
| };
 | |
| 
 | |
| 
 | |
| void
 | |
| inittime()
 | |
| {
 | |
| 	object *m, *d;
 | |
| 	m = initmodule("time", time_methods);
 | |
| 	d = getmoduledict(m);
 | |
| #ifdef SYSV
 | |
| 	tzset();
 | |
| 	dictinsert(d, "timezone", newintobject((long)_timezone));
 | |
| 	dictinsert(d, "altzone", newintobject((long)_altzone));
 | |
| 	dictinsert(d, "daylight", newintobject((long)_daylight));
 | |
| 	dictinsert(d, "tzname", mkvalue("(zz)", _tzname[0], _tzname[1]));
 | |
| #else /* !SYSV */
 | |
| 	{
 | |
| #define YEAR ((time_t)((365 * 24 + 6) * 3600))
 | |
| 		time_t t;
 | |
| 		struct tm *p;
 | |
| 		long winterzone, summerzone;
 | |
| 		char wintername[10], summername[10];
 | |
| 		t = (time((time_t *)0) / YEAR) * YEAR;
 | |
| 		p = localtime(&t);
 | |
| 		winterzone = -p->tm_gmtoff;
 | |
| 		strncpy(wintername, p->tm_zone ? p->tm_zone : "   ", 9);
 | |
| 		wintername[9] = '\0';
 | |
| 		t += YEAR/2;
 | |
| 		p = localtime(&t);
 | |
| 		summerzone = -p->tm_gmtoff;
 | |
| 		strncpy(summername, p->tm_zone ? p->tm_zone : "   ", 9);
 | |
| 		summername[9] = '\0';
 | |
| 		dictinsert(d, "timezone", newintobject(winterzone));
 | |
| 		dictinsert(d, "altzone", newintobject(summerzone));
 | |
| 		dictinsert(d, "daylight",
 | |
| 			   newintobject((long)(winterzone != summerzone)));
 | |
| 		dictinsert(d, "tzname",
 | |
| 			   mkvalue("(zz)", wintername, summername));
 | |
| 	}
 | |
| #endif /* !SYSV */
 | |
| }
 | |
| 
 | |
| 
 | |
| #ifdef macintosh
 | |
| 
 | |
| #define MacTicks	(* (long *)0x16A)
 | |
| 
 | |
| #ifdef THINK_C_3_0
 | |
| sleep(secs)
 | |
| 	int secs;
 | |
| {
 | |
| 	register long deadline;
 | |
| 	
 | |
| 	deadline = MacTicks + mecs * 60;
 | |
| 	while (MacTicks < deadline) {
 | |
| 		if (intrcheck())
 | |
| 			sleep_catcher(SIGINT);
 | |
| 	}
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static void
 | |
| floatsleep(secs)
 | |
| 	double secs;
 | |
| {
 | |
| 	register long deadline;
 | |
| 	
 | |
| 	deadline = MacTicks + (long)(secs * 60.0);
 | |
| 	while (MacTicks < deadline) {
 | |
| 		if (intrcheck())
 | |
| 			sleep_catcher(SIGINT);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static long
 | |
| millitimer()
 | |
| {
 | |
| 	return MacTicks * 50 / 3; /* MacTicks * 1000 / 60 */
 | |
| }
 | |
| 
 | |
| #endif /* macintosh */
 | |
| 
 | |
| 
 | |
| #ifdef unix
 | |
| 
 | |
| #ifdef BSD_TIME
 | |
| 
 | |
| static long
 | |
| millitimer()
 | |
| {
 | |
| 	struct timeval t;
 | |
| 	struct timezone tz;
 | |
| 	if (gettimeofday(&t, &tz) != 0)
 | |
| 		return -1;
 | |
| 	return t.tv_sec*1000 + t.tv_usec/1000;
 | |
| }
 | |
| 
 | |
| static void
 | |
| floatsleep(secs)
 | |
| 	double secs;
 | |
| {
 | |
| 	struct timeval t;
 | |
| 	double frac;
 | |
| 	extern double fmod PROTO((double, double));
 | |
| 	extern double floor PROTO((double));
 | |
| 	frac = fmod(secs, 1.0);
 | |
| 	secs = floor(secs);
 | |
| 	t.tv_sec = (long)secs;
 | |
| 	t.tv_usec = (long)(frac*1000000.0);
 | |
| 	(void) select(0, (fd_set *)0, (fd_set *)0, (fd_set *)0, &t);
 | |
| }
 | |
| 
 | |
| #else /* !BSD_TIME */
 | |
| 
 | |
| static void
 | |
| floatsleep(secs)
 | |
| 	double secs;
 | |
| {
 | |
| 	sleep((int)secs);
 | |
| }
 | |
| 
 | |
| #endif /* !BSD_TIME */
 | |
| 
 | |
| #endif /* unix */
 | |
| 
 | |
| 
 | |
| #ifdef MSDOS
 | |
| 
 | |
| #ifndef CLOCKS_PER_SEC
 | |
| #define CLOCKS_PER_SEC 55	/* 54.945 msec per tick (18.2 HZ clock) */
 | |
| #endif
 | |
| 
 | |
| static void
 | |
| floatsleep(secs)
 | |
| 	double secs;
 | |
| {
 | |
| 	delay(long(secs/1000.0));
 | |
| }
 | |
| 
 | |
| static long
 | |
| millitimer()
 | |
| {
 | |
| 	clock_t ticks;
 | |
| 
 | |
| 	ticks = clock();	/* ticks since program start */
 | |
| 	return ticks * CLOCKS_PER_SEC;/* XXX shouldn't this be different? */
 | |
| }
 | |
| 
 | |
| floatsleep(secs)
 | |
|       double secs;
 | |
| {
 | |
|       clock_t t= clock( );
 | |
|       while( (clock()-t)/CLOCKS_PER_SEC<secs )
 | |
|               ;
 | |
| }
 | |
| 
 | |
| #endif /* MSDOS */
 |