summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
Diffstat (limited to 'sys')
-rw-r--r--sys/kern/kern_sig.c45
1 files changed, 30 insertions, 15 deletions
diff --git a/sys/kern/kern_sig.c b/sys/kern/kern_sig.c
index 859e1ff75aa..de6f59f7188 100644
--- a/sys/kern/kern_sig.c
+++ b/sys/kern/kern_sig.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: kern_sig.c,v 1.132 2012/01/17 02:34:18 guenther Exp $ */
+/* $OpenBSD: kern_sig.c,v 1.133 2012/01/25 06:12:13 guenther Exp $ */
/* $NetBSD: kern_sig.c,v 1.54 1996/04/22 01:38:32 christos Exp $ */
/*
@@ -755,6 +755,7 @@ ptsignal(struct proc *p, int signum, enum signal_type type)
int s, prop;
sig_t action;
int mask;
+ struct process *pr;
struct proc *q;
int wakeparent = 0;
@@ -769,31 +770,45 @@ ptsignal(struct proc *p, int signum, enum signal_type type)
mask = sigmask(signum);
+ pr = p->p_p;
if (type == SPROCESS) {
- TAILQ_FOREACH(q, &p->p_p->ps_threads, p_thr_link) {
+ /*
+ * A process-wide signal can be diverted to a different
+ * thread that's in sigwait() for this signal. If there
+ * isn't such a thread, then pick a thread that doesn't
+ * have it blocked so that the stop/kill consideration
+ * isn't delayed. Otherwise, mark it pending on the
+ * main thread.
+ */
+ TAILQ_FOREACH(q, &pr->ps_threads, p_thr_link) {
/* ignore exiting threads */
if (q->p_flag & P_WEXIT)
continue;
+
+ /* sigwait: definitely go to this thread */
if (q->p_sigdivert & mask) {
- /* sigwait: convert to thread-specific */
- type = STHREAD;
p = q;
break;
}
+
+ /* unblocked: possibly go to this thread */
+ if ((q->p_sigmask & mask) == 0)
+ p = q;
}
}
if (type != SPROPAGATED)
- KNOTE(&p->p_p->ps_klist, NOTE_SIGNAL | signum);
+ KNOTE(&pr->ps_klist, NOTE_SIGNAL | signum);
prop = sigprop[signum];
/*
* If proc is traced, always give parent a chance.
*/
- if (p->p_flag & P_TRACED)
+ if (p->p_flag & P_TRACED) {
action = SIG_DFL;
- else if (p->p_sigdivert & mask) {
+ atomic_setbits_int(&p->p_siglist, mask);
+ } else if (p->p_sigdivert & mask) {
p->p_sigwait = signum;
atomic_clearbits_int(&p->p_sigdivert, ~0);
action = SIG_CATCH;
@@ -815,8 +830,8 @@ ptsignal(struct proc *p, int signum, enum signal_type type)
else {
action = SIG_DFL;
- if (prop & SA_KILL && p->p_p->ps_nice > NZERO)
- p->p_p->ps_nice = NZERO;
+ if (prop & SA_KILL && pr->ps_nice > NZERO)
+ pr->ps_nice = NZERO;
/*
* If sending a tty stop signal to a member of an
@@ -824,9 +839,11 @@ ptsignal(struct proc *p, int signum, enum signal_type type)
* the action is default; don't stop the process below
* if sleeping, and don't clear any pending SIGCONT.
*/
- if (prop & SA_TTYSTOP && p->p_p->ps_pgrp->pg_jobc == 0)
+ if (prop & SA_TTYSTOP && pr->ps_pgrp->pg_jobc == 0)
return;
}
+
+ atomic_setbits_int(&p->p_siglist, mask);
}
if (prop & SA_CONT) {
@@ -838,13 +855,11 @@ ptsignal(struct proc *p, int signum, enum signal_type type)
atomic_clearbits_int(&p->p_flag, P_CONTINUED);
}
- atomic_setbits_int(&p->p_siglist, mask);
-
/*
* XXX delay processing of SA_STOP signals unless action == SIG_DFL?
*/
if (prop & (SA_CONT | SA_STOP) && type != SPROPAGATED) {
- TAILQ_FOREACH(q, &p->p_p->ps_threads, p_thr_link) {
+ TAILQ_FOREACH(q, &pr->ps_threads, p_thr_link) {
if (q != p)
ptsignal(q, signum, SPROPAGATED);
}
@@ -895,7 +910,7 @@ ptsignal(struct proc *p, int signum, enum signal_type type)
* If a child holding parent blocked,
* stopping could cause deadlock.
*/
- if (p->p_p->ps_flags & PS_PPWAIT)
+ if (pr->ps_flags & PS_PPWAIT)
goto out;
atomic_clearbits_int(&p->p_siglist, mask);
p->p_xstat = signum;
@@ -989,7 +1004,7 @@ run:
out:
SCHED_UNLOCK(s);
if (wakeparent)
- wakeup(p->p_p->ps_pptr);
+ wakeup(pr->ps_pptr);
}
/*