If I have a program like this (in pseudocode):
mutex_lock;
func() {
lock(mutex_lock);
// Some code (long enough to make a
// race condition if no proper synchronisation
// is available). We also going to call a signal,
// say, SIGINT, through (ctrl-c), while we are in
// the range of locking and unlocking the lock.
unlock(mutex_lock);
}
sig_handler_func(sig) {
// Say, we are handling SIGINT (ctrl-c) signal
// And we need to call func from here too.
if (sig == SIGINT) {
func();
}
}
main() {
// Calling func from main
func();
}
then there would be a deadlock when trying to acquire the lock in func, while it is already acquired from the 'main' caller. I was wondering if there is a way to suspend the signal handling in the C language, specifically, when, in this example, entering func and acquiring the lock, and resume the signal handling AND call the handlers upon exiting func.
You need two locks. The one used inside your
func()
, and one to protect the process's signal mask.You have to make masking and unmasking the signal atomic also:
With no lock around the
pthread_sigmask()
, threads are likely to corrupt the process sigmask as execution overlaps.You may use the test mutex functions (trylock) in this kind of undefined situations, to be safe. And with this, you don't necessarily need to block too.
You want
pthread_sigmask
, the multithread version ofsigprocmask
Here's some sample pseudo code:
I'm not totally sure, but, you may need to use
pthread_sigmask
to block signals in all but one thread and do the above from that thread only.Also, I'd be remiss if I didn't say that I'd refactor your code. The number of things you can do in a signal handler [aside from this] is limited (e.g. no
malloc
, noprintf
, etc.)Dedicating one thread for signal handling and having it do
sigsetjmp
and the signal handler doessiglongjmp
.Or have the signal handler set a volatile global (e.g.
signal_occurred
) that is monitored at base level.Thus, all the "heavy lifting" that you'd be doing in the signal handler can be done from base task level where you can do anything.