On 18-Feb-02 Matthew Dillon wrote:
>     While testing some Giant removal stuff I noticed that my current
>     system sometimes got into an extremely non-optimal flip-flop situation
>     between two processes contesting Giant on an SMP system which halved the
>     syscall performance in the test.
> 
>     In my getuid test, for example, with Giant in place I was getting
>     683Kcalls/sec with one process and 427Kcalls/sec with two.  Giant
>     was being obtained in two places: in userret and in getuid().
> 
>     When I turned off Giant in getuid() the syscall performance actually
>     went DOWN, to 250Kcalls/sec with two processes.  This was a totally
>     unexpected result.
> 
>     It turns out that the two processes got into an extremely non-optimal 
>     contested/sleep/wakeup situation, even though they do not actually have
>     to sleep on Giant in this situation.
> 
>     The solution is to allow _mtx_lock_sleep() to spin instead of sleep in
>     the situation where:  (1) there are no runnable processes other then
>     the ones already running on a cpu, (2) interrupts are enabled, and 
>     (3) the mutex in question is not contested (to avoid starving the thread
>     contesting the mutex).  In this case we can spin.
> 
>     This will go in tonight if no problems arise.

I would rather make the locks adaptive like so:  (untested)

--- //depot/projects/smpng/sys/conf/options     2002/02/08 13:19:07
+++ //depot/user/jhb/lock/conf/options  2002/02/08 13:50:54
@@ -56,6 +56,7 @@
                                        # mapped I/O
 
 # Miscellaneous options.
+ADAPTIVE_MUTEXES
 BLEED
 COMPAT_43      opt_compat.h
 COMPAT_SUNOS   opt_compat.h
--- //depot/projects/smpng/sys/i386/include/cpufunc.h   2001/12/18 13:07:32
+++ //depot/user/jhb/lock/i386/include/cpufunc.h        2001/12/20 15:54:32
@@ -550,6 +550,12 @@
        __asm __volatile("movl %0,%%dr7" : : "r" (sel));
 }
 
+static __inline void
+cpu_pause(void)
+{
+       __asm __volatile("pause");
+}
+
 static __inline critical_t
 cpu_critical_enter(void)
 {
--- //depot/projects/smpng/sys/kern/kern_mutex.c        2002/02/08 14:19:21
+++ //depot/user/jhb/lock/kern/kern_mutex.c     2002/02/08 13:50:54
@@ -34,6 +34,7 @@
  * Machine independent bits of mutex implementation.
  */
 
+#include "opt_adaptive_mutexes.h"
 #include "opt_ddb.h"
 
 #include <sys/param.h>
@@ -345,7 +354,22 @@
                        continue;
                }
 
+#if defined(SMP) && defined(ADAPTIVE_MUTEXES)
                /*
+                * If the current owner of the lock is executing on another
+                * CPU, spin instead of blocking.
+                */
+               if (((struct thread *)(v & MTX_FLAGMASK)->td_kse->kse_oncpu !=
+                   NOCPU) {
+                       mtx_unlock_spin(&sched_lock);
+#ifdef __i386__
+                       cpu_pause();
+#endif
+                       continue;
+               }
+#endif /* SMP && ADAPTIVE_MUTEXES */
+
+               /*
                 * We deffinately must sleep for this lock.
                 */
                mtx_assert(m, MA_NOTOWNED);
@@ -433,6 +457,9 @@
                /* Give interrupts a chance while we spin. */
                critical_exit();
                while (m->mtx_lock != MTX_UNOWNED) {
+#ifdef __i386__
+                       cpu_pause();
+#endif
                        if (i++ < 10000000)
                                continue;
                        if (i++ < 60000000)

This is more a specific problem with Giant and I don't think it will be a
problem with other mutexes, so I'd prefer a solution not quite so tailored to
this particular behavior of Giant.

-- 

John Baldwin <[EMAIL PROTECTED]>  <><  http://www.FreeBSD.org/~jhb/
"Power Users Use the Power to Serve!"  -  http://www.FreeBSD.org/

To Unsubscribe: send mail to [EMAIL PROTECTED]
with "unsubscribe freebsd-current" in the body of the message

Reply via email to