Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/kern lwp_unpark(): no need to acquire LWP refs or drop t...



details:   https://anonhg.NetBSD.org/src/rev/1ad659d3c999
branches:  trunk
changeset: 1009870:1ad659d3c999
user:      ad <ad%NetBSD.org@localhost>
date:      Tue May 05 22:12:06 2020 +0000

description:
lwp_unpark(): no need to acquire LWP refs or drop the proc lock.

On the hacky benchmarks I have, held over from the transition to 1:1
threading, this restores pthread_cond_signal() perf to radixtree/sleepq
levels, and semes much better than either with pthread_cond_broadcast() and
10 threads.  It would be interesting to see what might be achieved with a
lockless lookup, which is within grasp now thanks to pid_table being used
for lookup.

diffstat:

 sys/kern/sys_lwp.c |  33 +++++----------------------------
 1 files changed, 5 insertions(+), 28 deletions(-)

diffs (74 lines):

diff -r 3b36c7a071ab -r 1ad659d3c999 sys/kern/sys_lwp.c
--- a/sys/kern/sys_lwp.c        Tue May 05 21:22:48 2020 +0000
+++ b/sys/kern/sys_lwp.c        Tue May 05 22:12:06 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: sys_lwp.c,v 1.79 2020/04/24 03:22:06 thorpej Exp $     */
+/*     $NetBSD: sys_lwp.c,v 1.80 2020/05/05 22:12:06 ad Exp $  */
 
 /*-
  * Copyright (c) 2001, 2006, 2007, 2008, 2019, 2020 The NetBSD Foundation, Inc.
@@ -35,7 +35,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: sys_lwp.c,v 1.79 2020/04/24 03:22:06 thorpej Exp $");
+__KERNEL_RCSID(0, "$NetBSD: sys_lwp.c,v 1.80 2020/05/05 22:12:06 ad Exp $");
 
 #include <sys/param.h>
 #include <sys/systm.h>
@@ -467,36 +467,12 @@
 
        mutex_enter(p->p_lock);
        for (target = 0; target < ntargets; target++) {
-               /*
-                * We don't bother excluding idle LWPs here, as
-                * setting LW_UNPARKED on them won't do any harm.
-                */
                t = proc_find_lwp(p, tp[target]);
                if (__predict_false(t == NULL)) {
                        error = ESRCH;
                        continue;
                }
 
-               /*
-                * The locking order is p::p_lock -> l::l_mutex,
-                * but it may not be unsafe to release p::p_lock
-                * while l::l_mutex is held because l::l_mutex is
-                * a scheduler lock and we don't want to get tied
-                * in knots while unwinding priority inheritance.
-                * So, get a reference count on the LWP and then
-                * unlock p::p_lock before acquiring l::l_mutex.
-                */
-               if (__predict_false(t->l_stat == LSZOMB)) {
-                       continue;
-               }
-               lwp_addref(t);
-               mutex_exit(p->p_lock);
-
-               /*
-                * Note the LWP cannot become a zombie while we
-                * hold a reference.
-                */
-
                lwp_lock(t);
                if (__predict_true(t->l_syncobj == &lwp_park_syncobj)) {
                        /*
@@ -504,6 +480,9 @@
                         * lwp_unsleep() will release the LWP lock.
                         */
                        lwp_unsleep(t, true);
+               } else if (__predict_false(t->l_stat == LSZOMB)) {
+                       lwp_unlock(t);
+                       error = ESRCH;
                } else {
                        /*
                         * It hasn't parked yet because the wakeup side won
@@ -516,8 +495,6 @@
                        t->l_flag |= LW_UNPARKED;
                        lwp_unlock(t);
                }
-               mutex_enter(p->p_lock);
-               lwp_delref2(t);
        }
        mutex_exit(p->p_lock);
 



Home | Main Index | Thread Index | Old Index