diff options
author | Simon Marlow <simonmar@microsoft.com> | 2007-07-24 15:30:57 +0000 |
---|---|---|
committer | Simon Marlow <simonmar@microsoft.com> | 2007-07-24 15:30:57 +0000 |
commit | 681aad99cb29ce54f72ec2a916fb1aab0fa0fabb (patch) | |
tree | 65e13d3bc1acad8b859e63886c5e5ea429f7236f /rts/RtsStartup.c | |
parent | 1bd1fb932375bc4b065cc8de23f0d251e8075395 (diff) | |
download | haskell-681aad99cb29ce54f72ec2a916fb1aab0fa0fabb.tar.gz |
hs_exit()/shutdownHaskell(): wait for outstanding foreign calls to complete before returning
This is pertinent to #1177. When shutting down a DLL, we need to be
sure that there are no OS threads that can return to the code that we
are about to unload, and previously the threaded RTS was unsafe in
this respect.
When exiting a standalone program we don't have to be quite so
paranoid: all the code will disappear at the same time as any running
threads. Happily exiting a program happens via a different path:
shutdownHaskellAndExit(). If we're about to exit(), then there's no
need to wait for foreign calls to complete.
Diffstat (limited to 'rts/RtsStartup.c')
-rw-r--r-- | rts/RtsStartup.c | 36 |
1 files changed, 29 insertions, 7 deletions
diff --git a/rts/RtsStartup.c b/rts/RtsStartup.c index a363c133f4..8be4044c05 100644 --- a/rts/RtsStartup.c +++ b/rts/RtsStartup.c @@ -355,12 +355,25 @@ hs_add_root(void (*init_root)(void)) initProfiling2(); } -/* ----------------------------------------------------------------------------- - Shutting down the RTS - -------------------------------------------------------------------------- */ +/* ---------------------------------------------------------------------------- + * Shutting down the RTS + * + * The wait_foreign parameter means: + * True ==> wait for any threads doing foreign calls now. + * False ==> threads doing foreign calls may return in the + * future, but will immediately block on a mutex. + * (capability->lock). + * + * If this RTS is a DLL that we're about to unload, then you want + * safe=True, otherwise the thread might return to code that has been + * unloaded. If this is a standalone program that is about to exit, + * then you can get away with safe=False, which is better because we + * won't hang on exit if there is a blocked foreign call outstanding. + * + ------------------------------------------------------------------------- */ -void -hs_exit(void) +static void +hs_exit_(rtsBool wait_foreign) { if (hs_init_count <= 0) { errorBelch("warning: too many hs_exit()s"); @@ -386,7 +399,7 @@ hs_exit(void) #endif /* stop all running tasks */ - exitScheduler(); + exitScheduler(wait_foreign); #if defined(GRAN) /* end_gr_simulation prints global stats if requested -- HWL */ @@ -497,6 +510,14 @@ hs_exit(void) } +// The real hs_exit(): +void +hs_exit(void) +{ + hs_exit_(rtsTrue); + // be safe; this might be a DLL +} + // Compatibility interfaces void shutdownHaskell(void) @@ -509,7 +530,8 @@ shutdownHaskellAndExit(int n) { if (hs_init_count == 1) { OnExitHook(); - hs_exit(); + hs_exit_(rtsFalse); + // we're about to exit(), no need to wait for foreign calls to return. #if defined(PAR) /* really exit (stg_exit() would call shutdownParallelSystem() again) */ exit(n); |