diff options
author | jb <jb@FreeBSD.org> | 1998-09-30 06:27:31 +0000 |
---|---|---|
committer | jb <jb@FreeBSD.org> | 1998-09-30 06:27:31 +0000 |
commit | 55e005ea963d7d451f7e92a1301813d8a96846d5 (patch) | |
tree | 54683b42cb8ed10c5db1bfa67fab9d23dcbc2d8f /lib/libkse/thread/thr_kill.c | |
parent | 1326d7916eb67796a6dbb8e51347e5a20af03959 (diff) | |
download | FreeBSD-src-55e005ea963d7d451f7e92a1301813d8a96846d5.zip FreeBSD-src-55e005ea963d7d451f7e92a1301813d8a96846d5.tar.gz |
Implementation of an additional state called SIGWAIT (with the previous
one renamed to SIGSUSPEND) to fix sigwait().
Submitted by: Daniel M. Eischen <eischen@vigrid.com>
Diffstat (limited to 'lib/libkse/thread/thr_kill.c')
-rw-r--r-- | lib/libkse/thread/thr_kill.c | 44 |
1 files changed, 37 insertions, 7 deletions
diff --git a/lib/libkse/thread/thr_kill.c b/lib/libkse/thread/thr_kill.c index 98b3a2d..292ba5c 100644 --- a/lib/libkse/thread/thr_kill.c +++ b/lib/libkse/thread/thr_kill.c @@ -46,18 +46,48 @@ pthread_kill(pthread_t pthread, int sig) /* Invalid signal: */ ret = EINVAL; + /* Ignored signals get dropped on the floor. */ + else if (_thread_sigact[sig - 1].sa_handler == SIG_IGN) + ret = 0; + /* Find the thread in the list of active threads: */ else if ((ret = _find_thread(pthread)) == 0) { - if ((pthread->state == PS_SIGWAIT) && - sigismember(&pthread->sigmask, sig)) { - /* Change the state of the thread to run: */ - PTHREAD_NEW_STATE(pthread,PS_RUNNING); + switch (pthread->state) { + case PS_SIGSUSPEND: + /* + * Only wake up the thread if the signal is unblocked + * and there is a handler installed for the signal. + */ + if (!sigismember(&pthread->sigmask, sig) && + _thread_sigact[sig - 1].sa_handler != SIG_DFL) { + /* Change the state of the thread to run: */ + PTHREAD_NEW_STATE(pthread,PS_RUNNING); + + /* Return the signal number: */ + pthread->signo = sig; + } + /* Increment the pending signal count: */ + sigaddset(&pthread->sigpend,sig); + break; + + case PS_SIGWAIT: + /* Wake up the thread if the signal is blocked. */ + if (sigismember(pthread->data.sigwait, sig)) { + /* Change the state of the thread to run: */ + PTHREAD_NEW_STATE(pthread,PS_RUNNING); + + /* Return the signal number: */ + pthread->signo = sig; + } else + /* Increment the pending signal count. */ + sigaddset(&pthread->sigpend,sig); + break; - /* Return the signal number: */ - pthread->signo = sig; - } else + default: /* Increment the pending signal count: */ sigaddset(&pthread->sigpend,sig); + break; + } } /* Return the completion status: */ |