summaryrefslogtreecommitdiffstats
path: root/lib/libpthread
diff options
context:
space:
mode:
authordeischen <deischen@FreeBSD.org>2003-04-29 21:03:33 +0000
committerdeischen <deischen@FreeBSD.org>2003-04-29 21:03:33 +0000
commit6bd4376dc86f246ff07451d32eeeab0742747649 (patch)
tree91af310836f600ff861fcc771d66919156ee5215 /lib/libpthread
parenteeceed0838e543d6948276df87b878354ca10adb (diff)
downloadFreeBSD-src-6bd4376dc86f246ff07451d32eeeab0742747649.zip
FreeBSD-src-6bd4376dc86f246ff07451d32eeeab0742747649.tar.gz
Create the thread signal lock as a KSE lock (as opposed to
a thread lock). Better protect access to thread state while searching for threads to handle a signal. Better protect access to process pending signals while processing a thread in sigwait(). Submitted by: davidxu
Diffstat (limited to 'lib/libpthread')
-rw-r--r--lib/libpthread/thread/thr_init.c2
-rw-r--r--lib/libpthread/thread/thr_sig.c6
-rw-r--r--lib/libpthread/thread/thr_sigwait.c37
3 files changed, 23 insertions, 22 deletions
diff --git a/lib/libpthread/thread/thr_init.c b/lib/libpthread/thread/thr_init.c
index 804462c..d43adf5 100644
--- a/lib/libpthread/thread/thr_init.c
+++ b/lib/libpthread/thread/thr_init.c
@@ -499,7 +499,7 @@ init_private(void)
* process signal mask and pending signal sets.
*/
if (_lock_init(&_thread_signal_lock, LCK_ADAPTIVE,
- _thr_lock_wait, _thr_lock_wakeup) != 0)
+ _kse_lock_wait, _kse_lock_wakeup) != 0)
PANIC("Cannot initialize _thread_signal_lock");
if (_lock_init(&_mutex_static_lock, LCK_ADAPTIVE,
_thr_lock_wait, _thr_lock_wakeup) != 0)
diff --git a/lib/libpthread/thread/thr_sig.c b/lib/libpthread/thread/thr_sig.c
index 5ac4998..cd699db 100644
--- a/lib/libpthread/thread/thr_sig.c
+++ b/lib/libpthread/thread/thr_sig.c
@@ -286,11 +286,10 @@ thr_sig_find(struct kse *curkse, int sig, siginfo_t *info)
KSE_LOCK_ACQUIRE(curkse, &_thread_list_lock);
TAILQ_FOREACH(pthread, &_thread_list, tle) {
+ /* Take the scheduling lock. */
+ KSE_SCHED_LOCK(curkse, pthread->kseg);
if ((pthread->state == PS_SIGWAIT) &&
sigismember(pthread->data.sigwait, sig)) {
- /* Take the scheduling lock. */
- KSE_SCHED_LOCK(curkse, pthread->kseg);
-
/*
* Return the signal number and make the
* thread runnable.
@@ -328,6 +327,7 @@ thr_sig_find(struct kse *curkse, int sig, siginfo_t *info)
} else if (signaled_thread == NULL)
signaled_thread = pthread;
}
+ KSE_SCHED_UNLOCK(curkse, pthread->kseg);
}
KSE_LOCK_RELEASE(curkse, &_thread_list_lock);
diff --git a/lib/libpthread/thread/thr_sigwait.c b/lib/libpthread/thread/thr_sigwait.c
index 7b5a31f..9bb4285 100644
--- a/lib/libpthread/thread/thr_sigwait.c
+++ b/lib/libpthread/thread/thr_sigwait.c
@@ -49,7 +49,8 @@ _sigwait(const sigset_t *set, int *sig)
int i;
sigset_t tempset, waitset;
struct sigaction act;
-
+ kse_critical_t crit;
+
_thr_enter_cancellation_point(curthread);
/*
@@ -71,8 +72,11 @@ _sigwait(const sigset_t *set, int *sig)
/*
* Check to see if a pending signal is in the wait mask.
- * This has to be atomic. */
+ * This has to be atomic.
+ */
tempset = curthread->sigpend;
+ crit = _kse_critical_enter();
+ KSE_LOCK_ACQUIRE(curthread->kse, &_thread_signal_lock);
SIGSETOR(tempset, _thr_proc_sigpending);
SIGSETAND(tempset, waitset);
if (SIGNOTEMPTY(tempset)) {
@@ -83,24 +87,20 @@ _sigwait(const sigset_t *set, int *sig)
}
/* Clear the pending signal: */
- if (sigismember(&curthread->sigpend,i))
- sigdelset(&curthread->sigpend,i);
+ if (sigismember(&curthread->sigpend, i))
+ sigdelset(&curthread->sigpend, i);
else
- sigdelset(&_thr_proc_sigpending,i);
+ sigdelset(&_thr_proc_sigpending, i);
+ KSE_LOCK_RELEASE(curthread->kse, &_thread_signal_lock);
+ _kse_critical_leave(crit);
+ _thr_leave_cancellation_point(curthread);
/* Return the signal number to the caller: */
*sig = i;
-
- _thr_leave_cancellation_point(curthread);
return (0);
}
/*
- * Lock the array of SIG_DFL wait counts.
- */
- THR_LOCK_ACQUIRE(curthread, &_thread_signal_lock);
-
- /*
* Enter a loop to find the signals that are SIG_DFL. For
* these signals we must install a dummy signal handler in
* order for the kernel to pass them in to us. POSIX says
@@ -123,8 +123,8 @@ _sigwait(const sigset_t *set, int *sig)
}
}
/* Done accessing _thread_dfl_count for now. */
- THR_LOCK_RELEASE(curthread, &_thread_signal_lock);
-
+ KSE_LOCK_RELEASE(curthread->kse, &_thread_signal_lock);
+ _kse_critical_leave(crit);
if (ret == 0) {
/*
* Save the wait signal mask. The wait signal
@@ -151,7 +151,8 @@ _sigwait(const sigset_t *set, int *sig)
/*
* Relock the array of SIG_DFL wait counts.
*/
- THR_LOCK_ACQUIRE(curthread, &_thread_signal_lock);
+ crit = _kse_critical_enter();
+ KSE_LOCK_ACQUIRE(curthread->kse, &_thread_signal_lock);
/* Restore the sigactions: */
act.sa_handler = SIG_DFL;
@@ -166,10 +167,10 @@ _sigwait(const sigset_t *set, int *sig)
}
}
/* Done accessing _thread_dfl_count. */
- THR_LOCK_RELEASE(curthread, &_thread_signal_lock);
-
+ KSE_LOCK_RELEASE(curthread->kse, &_thread_signal_lock);
+ _kse_critical_leave(crit);
_thr_leave_cancellation_point(curthread);
-
+
/* Return the completion status: */
return (ret);
}
OpenPOWER on IntegriCloud