/****************************************************************************** Copyright (c) 1992, 1995, 1996 Xerox Corporation. All rights reserved. Portions of this code were written by Stephen White, aka ghond. Use and copying of this software and preparation of derivative works based upon this software are permitted. Any distribution of this software or derivative works must comply with all applicable United States export control laws. This software is made available AS IS, and Xerox Corporation makes no warranty about the software, its performance or its conformity to any specification. Any person obtaining a copy of this software is requested to send their name and post office or electronic mail address to: Pavel Curtis Xerox PARC 3333 Coyote Hill Rd. Palo Alto, CA 94304 Pavel@Xerox.Com *****************************************************************************/ #include "my-signal.h" #include "my-stdlib.h" #include "my-sys-time.h" #include "my-time.h" #include "my-unistd.h" #include "config.h" #include "timers.h" #if (defined(MACH) && defined(CMU)) || !defined(SIGVTALRM) /* Virtual interval timers are broken on Mach 3.0 */ # undef ITIMER_VIRTUAL #endif typedef struct Timer_Entry Timer_Entry; struct Timer_Entry { Timer_Entry *next; time_t when; Timer_Proc proc; Timer_Data data; Timer_ID id; }; static Timer_Entry *active_timers = 0; static Timer_Entry *free_timers = 0; static Timer_Entry *virtual_timer = 0; static Timer_ID next_id = 0; static Timer_Entry * allocate_timer(void) { if (free_timers) { Timer_Entry *this = free_timers; free_timers = this->next; return this; } else return (Timer_Entry *) malloc(sizeof(Timer_Entry)); } static void free_timer(Timer_Entry *this) { this->next = free_timers; free_timers = this; } static void restart_timers(void); static void wakeup_call(int signo) { Timer_Entry *this = active_timers; Timer_Proc proc = this->proc; Timer_ID id = this->id; Timer_Data data = this->data; active_timers = active_timers->next; free_timer(this); restart_timers(); if (proc) (*proc)(id, data); } #ifdef ITIMER_VIRTUAL static void virtual_wakeup_call(int signo) { Timer_Entry *this = virtual_timer; Timer_Proc proc = this->proc; Timer_ID id = this->id; Timer_Data data = this->data; virtual_timer = 0; free_timer(this); if (proc) (*proc)(id, data); } #endif static void stop_timers() { alarm(0); signal(SIGALRM, SIG_IGN); signal(SIGALRM, wakeup_call); #ifdef ITIMER_VIRTUAL { struct itimerval itimer, oitimer; itimer.it_value.tv_sec = 0; itimer.it_value.tv_usec = 0; itimer.it_interval.tv_sec = 0; itimer.it_interval.tv_usec = 0; setitimer(ITIMER_VIRTUAL, &itimer, &oitimer); signal(SIGVTALRM, SIG_IGN); signal(SIGVTALRM, virtual_wakeup_call); if (virtual_timer) virtual_timer->when = oitimer.it_value.tv_sec; } #endif } static void restart_timers() { if (active_timers) { time_t now = time(0); signal(SIGALRM, wakeup_call); if (now < active_timers->when) /* first timer is in the future */ alarm(active_timers->when - now); else kill(getpid(), SIGALRM); /* we're already late... */ } #ifdef ITIMER_VIRTUAL if (virtual_timer) { signal(SIGVTALRM, virtual_wakeup_call); if (virtual_timer->when > 0) { struct itimerval itimer; itimer.it_value.tv_sec = virtual_timer->when; itimer.it_value.tv_usec = 0; itimer.it_interval.tv_sec = 0; itimer.it_interval.tv_usec = 0; setitimer(ITIMER_VIRTUAL, &itimer, 0); } else kill(getpid(), SIGVTALRM); } #endif } Timer_ID set_timer(unsigned seconds, Timer_Proc proc, Timer_Data data) { Timer_Entry *this = allocate_timer(); Timer_Entry **t; this->id = next_id++; this->when = time(0) + seconds; this->proc = proc; this->data = data; stop_timers(); t = &active_timers; while (*t && this->when >= (*t)->when) t = &((*t)->next); this->next = *t; *t = this; restart_timers(); return this->id; } Timer_ID set_virtual_timer(unsigned seconds, Timer_Proc proc, Timer_Data data) { #ifdef ITIMER_VIRTUAL if (virtual_timer) return -1; stop_timers(); virtual_timer = allocate_timer(); virtual_timer->id = next_id++; virtual_timer->when = seconds; virtual_timer->proc = proc; virtual_timer->data = data; restart_timers(); return virtual_timer->id; #else /* !ITIMER_VIRTUAL */ return set_timer(seconds, proc, data); #endif } int virtual_timer_available() { #ifdef ITIMER_VIRTUAL return 1; #else return 0; #endif } unsigned timer_wakeup_interval(Timer_ID id) { Timer_Entry *t; #ifdef ITIMER_VIRTUAL if (virtual_timer && virtual_timer->id == id) { struct itimerval itimer; getitimer(ITIMER_VIRTUAL, &itimer); return itimer.it_value.tv_sec; } #endif for (t = active_timers; t; t = t->next) if (t->id == id) return t->when - time(0);; return 0; } void timer_sleep(unsigned seconds) { set_timer(seconds, 0, 0); pause(); } int cancel_timer(Timer_ID id) { Timer_Entry **t = &active_timers; int found = 0; stop_timers(); if (virtual_timer && virtual_timer->id == id) { free(virtual_timer); virtual_timer = 0; found = 1; } else { while (*t) { if ((*t)->id == id) { Timer_Entry *tt = *t; *t = tt->next; found = 1; free(tt); break; } t = &((*t)->next); } } restart_timers(); return found; } void reenable_timers(void) { #if HAVE_SIGEMPTYSET sigset_t sigs; sigemptyset(&sigs); sigaddset(&sigs, SIGALRM); sigprocmask(SIG_UNBLOCK, &sigs, 0); #else #if HAVE_SIGSETMASK int old_mask = sigsetmask(-1); /* block everything, get old mask */ old_mask &= ~sigmask(SIGALRM); /* clear blocked bit for SIGALRM */ sigsetmask(old_mask); /* reset the signal mask */ #else #if HAVE_SIGRELSE sigrelse(SIGALRM); /* restore previous signal action */ #else #error I need some way to stop blocking SIGALRM! #endif #endif #endif } char rcsid_timers[] = "$Id: timers.c,v 2.1 1996/02/08 06:43:56 pavel Exp $"; /* $Log: timers.c,v $ * Revision 2.1 1996/02/08 06:43:56 pavel * Updated copyright notice for 1996. Release 1.8.0beta1. * * Revision 2.0 1995/11/30 04:41:20 pavel * New baseline version, corresponding to release 1.8.0alpha1. * * Revision 1.9 1992/10/23 23:03:47 pavel * Added copyright notice. * * Revision 1.8 1992/10/23 22:24:08 pavel * Eliminated all uses of the useless macro NULL. * * Revision 1.7 1992/10/21 03:02:35 pavel * Converted to use new automatic configuration system. * * Revision 1.6 1992/10/17 21:01:52 pavel * #undef'd ITIMER_VIRTUAL on CMU MACH because it apparently doesn't work on * that system. * Added an #ifdef of ITIMER_VIRTUAL around the virtual_wakeup_call() * signal-handler. * * Revision 1.5 1992/09/22 22:48:59 pavel * Added missing #include of "config.h". * * Revision 1.4 1992/08/21 00:45:29 pavel * Changed to conditionalize on the option AVOID_POSIX rather than USE_POSIX. * * Revision 1.3 1992/07/30 00:42:45 pavel * Add support for compiling on RISC/os 4.52 and NonStop-UX A22. * * Revision 1.2 1992/07/21 00:07:26 pavel * Added rcsid_<filename-root> declaration to hold the RCS ident. string. * * Revision 1.1 1992/07/20 23:23:12 pavel * Initial RCS-controlled version. */