X-Git-Url: http://git.megacz.com/?a=blobdiff_plain;f=ghc%2Frts%2FItimer.c;h=50be88dd799a68605bb14085b69b9d21db44e2c2;hb=d061776c476ef6f161749b27c3b4bda551c42c38;hp=f1dd8237117ded1155af3e3acfdd7745b65095d0;hpb=557947d3f93e11285e36423ddb08d859af60ab47;p=ghc-hetmet.git diff --git a/ghc/rts/Itimer.c b/ghc/rts/Itimer.c index f1dd823..50be88d 100644 --- a/ghc/rts/Itimer.c +++ b/ghc/rts/Itimer.c @@ -1,5 +1,4 @@ /* ----------------------------------------------------------------------------- - * $Id: Itimer.c,v 1.32 2003/02/22 04:51:50 sof Exp $ * * (c) The GHC Team, 1995-1999 * @@ -18,7 +17,6 @@ * to support. So much for standards. */ #include "Rts.h" -#if !defined(mingw32_TARGET_OS) /* to the end */ #include "RtsFlags.h" #include "Timer.h" #include "Itimer.h" @@ -41,9 +39,43 @@ # include #endif +/* Major bogosity: + * + * In the threaded RTS, we can't set the virtual timer because the + * thread which has the virtual timer might be sitting waiting for a + * capability, and the virtual timer only ticks in CPU time. + * + * So, possible solutions: + * + * (1) tick in realtime. Not very good, because this ticker is used for + * profiling, and this will give us unreliable time profiling + * results. Furthermore, this requires picking a single OS thread + * to be the timekeeper, which is a bad idea because the thread in + * question might just be making a temporary call into Haskell land. + * + * (2) save/restore the virtual timer around excursions into STG land. + * Sounds great, but I tried it and the resolution of the virtual timer + * isn't good enough (on Linux) - most of our excursions fall + * within the timer's resolution and we never make any progress. + * + * (3) have a virtual timer in every OS thread. Might be reasonable, + * because most of the time there is only ever one of these + * threads running, so it approximates a single virtual timer. + * But still quite bogus (and I got crashes when I tried this). + * + * For now, we're using (1), but this needs a better solution. --SDM + */ +#ifdef RTS_SUPPORTS_THREADS +#define ITIMER_FLAVOUR ITIMER_REAL +#define ITIMER_SIGNAL SIGALRM +#else +#define ITIMER_FLAVOUR ITIMER_VIRTUAL +#define ITIMER_SIGNAL SIGVTALRM +#endif + static int -install_vtalrm_handler(void) +install_vtalrm_handler(TickProc handle_tick) { struct sigaction action; @@ -52,26 +84,26 @@ install_vtalrm_handler(void) sigemptyset(&action.sa_mask); action.sa_flags = 0; - return sigaction(SIGVTALRM, &action, NULL); + return sigaction(ITIMER_SIGNAL, &action, NULL); } int -startTicker(nat ms) +startTicker(nat ms, TickProc handle_tick) { # ifndef HAVE_SETITIMER - /* fprintf(stderr, "No virtual timer on this system\n"); */ + /* debugBelch("No virtual timer on this system\n"); */ return -1; # else struct itimerval it; - install_vtalrm_handler(); + install_vtalrm_handler(handle_tick); timestamp = getourtimeofday(); it.it_value.tv_sec = ms / 1000; it.it_value.tv_usec = 1000 * (ms - (1000 * it.it_value.tv_sec)); it.it_interval = it.it_value; - return (setitimer(ITIMER_VIRTUAL, &it, NULL)); + return (setitimer(ITIMER_FLAVOUR, &it, NULL)); # endif } @@ -79,7 +111,7 @@ int stopTicker() { # ifndef HAVE_SETITIMER - /* fprintf(stderr, "No virtual timer on this system\n"); */ + /* debugBelch("No virtual timer on this system\n"); */ return -1; # else struct itimerval it; @@ -87,7 +119,7 @@ stopTicker() it.it_value.tv_sec = 0; it.it_value.tv_usec = 0; it.it_interval = it.it_value; - return (setitimer(ITIMER_VIRTUAL, &it, NULL)); + return (setitimer(ITIMER_FLAVOUR, &it, NULL)); # endif } @@ -103,8 +135,8 @@ startTicker(nat ms) timestamp = getourtimeofday(); se.sigev_notify = SIGEV_SIGNAL; - se.sigev_signo = SIGVTALRM; - se.sigev_value.sival_int = SIGVTALRM; + se.sigev_signo = ITIMER_SIGNAL; + se.sigev_value.sival_int = ITIMER_SIGNAL; if (timer_create(CLOCK_VIRTUAL, &se, &tid)) { barf("can't create virtual timer"); } @@ -124,8 +156,8 @@ stopTicker() timestamp = getourtimeofday(); se.sigev_notify = SIGEV_SIGNAL; - se.sigev_signo = SIGVTALRM; - se.sigev_value.sival_int = SIGVTALRM; + se.sigev_signo = ITIMER_SIGNAL; + se.sigev_value.sival_int = ITIMER_SIGNAL; if (timer_create(CLOCK_VIRTUAL, &se, &tid)) { barf("can't create virtual timer"); } @@ -136,13 +168,15 @@ stopTicker() } # endif +#if 0 +/* Currently unused */ void block_vtalrm_signal(void) { sigset_t signals; sigemptyset(&signals); - sigaddset(&signals, SIGVTALRM); + sigaddset(&signals, ITIMER_SIGNAL); (void) sigprocmask(SIG_BLOCK, &signals, NULL); } @@ -153,21 +187,22 @@ unblock_vtalrm_signal(void) sigset_t signals; sigemptyset(&signals); - sigaddset(&signals, SIGVTALRM); + sigaddset(&signals, ITIMER_SIGNAL); (void) sigprocmask(SIG_UNBLOCK, &signals, NULL); } +#endif /* gettimeofday() takes around 1us on our 500MHz PIII. Since we're * only calling it 50 times/s, it shouldn't have any great impact. */ -unsigned int +nat getourtimeofday(void) { struct timeval tv; gettimeofday(&tv, (struct timezone *) NULL); - return (tv.tv_sec * TICK_FREQUENCY + - tv.tv_usec * TICK_FREQUENCY / 1000000); + // cast to nat because nat may be 64 bit when int is only 32 bit + return ((nat)tv.tv_sec * TICK_FREQUENCY + + (nat)tv.tv_usec * TICK_FREQUENCY / 1000000); } -#endif /* !mingw32_TARGET_OS */