[Mono-dev] Need help with signal handlers.

Jonathan Pryor jonpryor at vt.edu
Fri Aug 19 22:22:38 EDT 2005


On Fri, 2005-08-19 at 15:43 -0700, Mike Hull wrote:
> I'm trying to pass signals received with signal.h into managed code.  
> After I mono_runtime_exec_managed_code I can not receive signals using 
> signal(sig,signal_handler).  If I don't mono_runtime_exec_managed_code I 
> receive the signals just fine.

You're in a world of hurt. :-)

Mono.Posix.Syscall.signal() is horribly, horribly broken.  How broken?
It takes a raw `int' as the signal number to catch.  You *cannot* use
Mono.Posix.Signals because those don't match *ANY* operating system's
values.  Consequently, it's inherently non-portable.

Mono.Unix.Stdlib.signal() is the replacement, but it isn't without the
next problem.

The larger problem is one of reentrancy: Mono is not reentrant, and
neither is the P/Invoke layer.

Reentrancy is similar to thread safety, but different: when a signal
occurs, a (potentially random) process thread is "hijacked", a new stack
frame is added (or worse, the current stack frame is re-used), and the
signal handler is executed.  (Since an existing thread is re-used, and
probably the most recently executing thread, normal thread-safety tricks
such as mutexes won't work -- you have to protect against the *same*
thread re-entering the currently executing function.  Fun.)

The only safe thing you can do is call other reentrant functions, and
*most* functions are NOT reentrant (malloc(3) isn't, neither is free(2),
function calls are "iffy" depending on how much stack space you
have...).

In short, signal handlers are a whole new world of pain.  Managed code
only makes things worse, since the entire P/Invoke layer isn't
reentrant, so you risk many unknowns just trying to use a managed
delegate as a signal handler, never mind doing anything complicated from
the handler.

Thus, the advice: assuming you risk using a signal handler in managed
code, all you can do from managed code is modify a variable.  That's it.
Anything else, *especially* a P/Invoke call, is suspect.

Alternatively, all your signal handlers should be implemented in C.  You
could hook them up from C#.  If you do this, take care that your handler
only calls reentrant functions.

 - Jon





More information about the Mono-devel-list mailing list