summaryrefslogtreecommitdiffstats
path: root/lib/libkse/thread/thr_kill.c
diff options
context:
space:
mode:
authorjb <jb@FreeBSD.org>1998-09-30 06:27:31 +0000
committerjb <jb@FreeBSD.org>1998-09-30 06:27:31 +0000
commit55e005ea963d7d451f7e92a1301813d8a96846d5 (patch)
tree54683b42cb8ed10c5db1bfa67fab9d23dcbc2d8f /lib/libkse/thread/thr_kill.c
parent1326d7916eb67796a6dbb8e51347e5a20af03959 (diff)
downloadFreeBSD-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.c44
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: */
OpenPOWER on IntegriCloud