Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/lib/libpthread - Try to eliminate a hang in "parked" I've be...
details: https://anonhg.NetBSD.org/src/rev/a9267507c999
branches: trunk
changeset: 972133:a9267507c999
user: ad <ad%NetBSD.org@localhost>
date: Sat May 16 22:53:37 2020 +0000
description:
- Try to eliminate a hang in "parked" I've been seeing while stress testing.
Centralise wakeup of deferred waiters in pthread__clear_waiters() and use
throughout libpthread. Make fewer assumptions. Be more conservative in
pthread_mutex when dealing with pending waiters.
- Remove the "hint" argument everywhere since the kernel doesn't use it any
more.
diffstat:
lib/libpthread/pthread.c | 107 ++++++++----
lib/libpthread/pthread_barrier.c | 8 +-
lib/libpthread/pthread_cond.c | 24 +--
lib/libpthread/pthread_int.h | 8 +-
lib/libpthread/pthread_mutex.c | 323 ++++++++++++++------------------------
lib/libpthread/pthread_rwlock.c | 8 +-
6 files changed, 207 insertions(+), 271 deletions(-)
diffs (truncated from 854 to 300 lines):
diff -r c5885a3cb4f6 -r a9267507c999 lib/libpthread/pthread.c
--- a/lib/libpthread/pthread.c Sat May 16 22:07:06 2020 +0000
+++ b/lib/libpthread/pthread.c Sat May 16 22:53:37 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: pthread.c,v 1.169 2020/05/15 14:30:23 joerg Exp $ */
+/* $NetBSD: pthread.c,v 1.170 2020/05/16 22:53:37 ad Exp $ */
/*-
* Copyright (c) 2001, 2002, 2003, 2006, 2007, 2008, 2020
@@ -31,7 +31,7 @@
*/
#include <sys/cdefs.h>
-__RCSID("$NetBSD: pthread.c,v 1.169 2020/05/15 14:30:23 joerg Exp $");
+__RCSID("$NetBSD: pthread.c,v 1.170 2020/05/16 22:53:37 ad Exp $");
#define __EXPOSE_STACK 1
@@ -105,7 +105,7 @@
int pthread__concurrency;
int pthread__nspins;
-int pthread__unpark_max = PTHREAD__UNPARK_MAX;
+size_t pthread__unpark_max = PTHREAD__UNPARK_MAX;
int pthread__dbg; /* set by libpthread_dbg if active */
/*
@@ -191,9 +191,9 @@
{
pthread_t first;
char *p;
- int i;
int mib[2];
unsigned int value;
+ ssize_t slen;
size_t len;
extern int __isthreaded;
@@ -223,16 +223,16 @@
/* Initialize locks first; they're needed elsewhere. */
pthread__lockprim_init();
- for (i = 0; i < NHASHLOCK; i++) {
+ for (int i = 0; i < NHASHLOCK; i++) {
pthread_mutex_init(&hashlocks[i].mutex, NULL);
}
/* Fetch parameters. */
- i = (int)_lwp_unpark_all(NULL, 0, NULL);
- if (i == -1)
+ slen = _lwp_unpark_all(NULL, 0, NULL);
+ if (slen < 0)
err(EXIT_FAILURE, "_lwp_unpark_all");
- if (i < pthread__unpark_max)
- pthread__unpark_max = i;
+ if ((size_t)slen < pthread__unpark_max)
+ pthread__unpark_max = slen;
/* Basic data structure setup */
pthread_attr_init(&pthread_default_attr);
@@ -604,16 +604,57 @@
* awoken (because cancelled, for instance) make sure we have no waiters
* left.
*/
-static void
+void
pthread__clear_waiters(pthread_t self)
{
+ int rv;
- if (self->pt_nwaiters != 0) {
- (void)_lwp_unpark_all(self->pt_waiters, self->pt_nwaiters,
- NULL);
- self->pt_nwaiters = 0;
+ /* Zero waiters or one waiter in error case (pthread_exit()). */
+ if (self->pt_nwaiters == 0) {
+ if (self->pt_unpark != 0 && self->pt_willpark == 0) {
+ rv = (ssize_t)_lwp_unpark(self->pt_unpark, NULL);
+ self->pt_unpark = 0;
+ if (rv != 0 && errno != EALREADY && errno != EINTR &&
+ errno != ESRCH) {
+ pthread__errorfunc(__FILE__, __LINE__, __func__,
+ "_lwp_unpark failed");
+ }
+ }
+ return;
}
- self->pt_willpark = 0;
+
+ /* One waiter or two waiters (the second being a deferred wakeup). */
+ if (self->pt_nwaiters == 1) {
+ if (self->pt_unpark != 0) {
+ /* Fall through to multiple waiters case. */
+ self->pt_waiters[1] = self->pt_unpark;
+ self->pt_nwaiters = 2;
+ self->pt_unpark = 0;
+ } else if (self->pt_willpark) {
+ /* Defer to _lwp_park(). */
+ self->pt_unpark = self->pt_waiters[0];
+ self->pt_nwaiters = 0;
+ return;
+ } else {
+ /* Wake one now. */
+ rv = (ssize_t)_lwp_unpark(self->pt_waiters[0], NULL);
+ self->pt_nwaiters = 0;
+ if (rv != 0 && errno != EALREADY && errno != EINTR &&
+ errno != ESRCH) {
+ pthread__errorfunc(__FILE__, __LINE__, __func__,
+ "_lwp_unpark failed");
+ }
+ return;
+ }
+ }
+
+ /* Multiple waiters. */
+ rv = _lwp_unpark_all(self->pt_waiters, self->pt_nwaiters, NULL);
+ self->pt_nwaiters = 0;
+ if (rv != 0 && errno != EINTR) {
+ pthread__errorfunc(__FILE__, __LINE__, __func__,
+ "_lwp_unpark_all failed");
+ }
}
void
@@ -630,6 +671,7 @@
self = pthread__self();
/* Disable cancellability. */
+ self->pt_willpark = 0;
pthread_mutex_lock(&self->pt_lock);
self->pt_flags |= PT_FLAG_CS_DISABLED;
self->pt_cancel = 0;
@@ -660,10 +702,12 @@
if (self->pt_flags & PT_FLAG_DETACHED) {
/* pthread__reap() will drop the lock. */
pthread__reap(self);
+ pthread__assert(!self->pt_willpark);
pthread__clear_waiters(self);
_lwp_exit();
} else {
self->pt_state = PT_STATE_ZOMBIE;
+ pthread__assert(!self->pt_willpark);
pthread_mutex_unlock(&self->pt_lock);
pthread__clear_waiters(self);
/* Note: name will be freed by the joiner. */
@@ -1125,7 +1169,7 @@
int
pthread__park(pthread_t self, pthread_mutex_t *lock,
pthread_queue_t *queue, const struct timespec *abstime,
- int cancelpt, const void *hint)
+ int cancelpt)
{
int rv, error;
void *obj;
@@ -1170,7 +1214,7 @@
* have _lwp_park() restart it before blocking.
*/
error = _lwp_park(CLOCK_REALTIME, TIMER_ABSTIME,
- __UNCONST(abstime), self->pt_unpark, hint, hint);
+ __UNCONST(abstime), self->pt_unpark, NULL, NULL);
self->pt_unpark = 0;
if (error != 0) {
switch (rv = errno) {
@@ -1215,22 +1259,14 @@
pthread_mutex_t *interlock)
{
pthread_t target;
- u_int max;
- size_t nwaiters;
- max = pthread__unpark_max;
- nwaiters = self->pt_nwaiters;
target = PTQ_FIRST(queue);
- if (nwaiters == max) {
- /* Overflow. */
- (void)_lwp_unpark_all(self->pt_waiters, nwaiters,
- __UNVOLATILE(&interlock->ptm_waiters));
- nwaiters = 0;
+ if (self->pt_nwaiters == pthread__unpark_max) {
+ pthread__clear_waiters(self);
}
target->pt_sleepobj = NULL;
- self->pt_waiters[nwaiters++] = target->pt_lid;
+ self->pt_waiters[self->pt_nwaiters++] = target->pt_lid;
PTQ_REMOVE(queue, target, pt_sleep);
- self->pt_nwaiters = nwaiters;
pthread__mutex_deferwake(self, interlock);
}
@@ -1238,23 +1274,16 @@
pthread__unpark_all(pthread_queue_t *queue, pthread_t self,
pthread_mutex_t *interlock)
{
+ const size_t max = pthread__unpark_max;
pthread_t target;
- u_int max;
- size_t nwaiters;
- max = pthread__unpark_max;
- nwaiters = self->pt_nwaiters;
PTQ_FOREACH(target, queue, pt_sleep) {
- if (nwaiters == max) {
- /* Overflow. */
- (void)_lwp_unpark_all(self->pt_waiters, nwaiters,
- __UNVOLATILE(&interlock->ptm_waiters));
- nwaiters = 0;
+ if (self->pt_nwaiters == max) {
+ pthread__clear_waiters(self);
}
target->pt_sleepobj = NULL;
- self->pt_waiters[nwaiters++] = target->pt_lid;
+ self->pt_waiters[self->pt_nwaiters++] = target->pt_lid;
}
- self->pt_nwaiters = nwaiters;
PTQ_INIT(queue);
pthread__mutex_deferwake(self, interlock);
}
diff -r c5885a3cb4f6 -r a9267507c999 lib/libpthread/pthread_barrier.c
--- a/lib/libpthread/pthread_barrier.c Sat May 16 22:07:06 2020 +0000
+++ b/lib/libpthread/pthread_barrier.c Sat May 16 22:53:37 2020 +0000
@@ -1,7 +1,7 @@
-/* $NetBSD: pthread_barrier.c,v 1.21 2020/01/29 14:41:57 kamil Exp $ */
+/* $NetBSD: pthread_barrier.c,v 1.22 2020/05/16 22:53:37 ad Exp $ */
/*-
- * Copyright (c) 2001, 2003, 2006, 2007, 2009 The NetBSD Foundation, Inc.
+ * Copyright (c) 2001, 2003, 2006, 2007, 2009, 2020 The NetBSD Foundation, Inc.
* All rights reserved.
*
* This code is derived from software contributed to The NetBSD Foundation
@@ -30,7 +30,7 @@
*/
#include <sys/cdefs.h>
-__RCSID("$NetBSD: pthread_barrier.c,v 1.21 2020/01/29 14:41:57 kamil Exp $");
+__RCSID("$NetBSD: pthread_barrier.c,v 1.22 2020/05/16 22:53:37 ad Exp $");
#include <errno.h>
@@ -106,7 +106,7 @@
PTQ_INSERT_TAIL(&barrier->ptb_waiters, self, pt_sleep);
self->pt_sleepobj = &barrier->ptb_waiters;
(void)pthread__park(self, interlock, &barrier->ptb_waiters,
- NULL, 0, __UNVOLATILE(&interlock->ptm_waiters));
+ NULL, 0);
if (__predict_true(gen != barrier->ptb_generation)) {
break;
}
diff -r c5885a3cb4f6 -r a9267507c999 lib/libpthread/pthread_cond.c
--- a/lib/libpthread/pthread_cond.c Sat May 16 22:07:06 2020 +0000
+++ b/lib/libpthread/pthread_cond.c Sat May 16 22:53:37 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: pthread_cond.c,v 1.68 2020/04/14 23:35:07 joerg Exp $ */
+/* $NetBSD: pthread_cond.c,v 1.69 2020/05/16 22:53:37 ad Exp $ */
/*-
* Copyright (c) 2001, 2006, 2007, 2008 The NetBSD Foundation, Inc.
@@ -46,7 +46,7 @@
*/
#include <sys/cdefs.h>
-__RCSID("$NetBSD: pthread_cond.c,v 1.68 2020/04/14 23:35:07 joerg Exp $");
+__RCSID("$NetBSD: pthread_cond.c,v 1.69 2020/05/16 22:53:37 ad Exp $");
#include <stdlib.h>
#include <errno.h>
@@ -161,9 +161,7 @@
self->pt_willpark = 0;
do {
retval = _lwp_park(clkid, TIMER_ABSTIME,
- __UNCONST(abstime), self->pt_unpark,
- __UNVOLATILE(&mutex->ptm_waiters),
- __UNVOLATILE(&mutex->ptm_waiters));
+ __UNCONST(abstime), self->pt_unpark, NULL, NULL);
self->pt_unpark = 0;
} while (retval == -1 && errno == ESRCH);
pthread_mutex_lock(mutex);
@@ -254,9 +252,7 @@
* caller (this thread) releases the mutex.
*/
if (__predict_false(self->pt_nwaiters == (size_t)pthread__unpark_max)) {
- (void)_lwp_unpark_all(self->pt_waiters, self->pt_nwaiters,
- __UNVOLATILE(&mutex->ptm_waiters));
- self->pt_nwaiters = 0;
+ pthread__clear_waiters(self);
}
self->pt_waiters[self->pt_nwaiters++] = lid;
pthread__mutex_deferwake(self, mutex);
@@ -284,7 +280,6 @@
pthread_t self, signaled;
pthread_mutex_t *mutex;
u_int max;
- size_t nwaiters;
/*
* Try to defer waking threads (see pthread_cond_signal()).
@@ -294,19 +289,14 @@
pthread__spinlock(self, &cond->ptc_lock);
max = pthread__unpark_max;
mutex = cond->ptc_mutex;
Home |
Main Index |
Thread Index |
Old Index