summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/arch/amd64/amd64/lock_machdep.c95
-rw-r--r--sys/arch/amd64/include/mplock.h12
2 files changed, 51 insertions, 56 deletions
diff --git a/sys/arch/amd64/amd64/lock_machdep.c b/sys/arch/amd64/amd64/lock_machdep.c
index 641b973a83f..3afdf49e8f5 100644
--- a/sys/arch/amd64/amd64/lock_machdep.c
+++ b/sys/arch/amd64/amd64/lock_machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: lock_machdep.c,v 1.4 2013/12/05 01:28:45 uebayasi Exp $ */
+/* $OpenBSD: lock_machdep.c,v 1.5 2014/03/14 02:08:57 dlg Exp $ */
/*
* Copyright (c) 2007 Artur Grabowski <art@openbsd.org>
@@ -28,10 +28,11 @@
#include <ddb/db_output.h>
void
-__mp_lock_init(struct __mp_lock *lock)
+__mp_lock_init(struct __mp_lock *mpl)
{
- lock->mpl_cpu = NULL;
- lock->mpl_count = 0;
+ memset(mpl->mpl_cpus, 0, sizeof(mpl->mpl_cpus));
+ mpl->mpl_users = 0;
+ mpl->mpl_ticket = 0;
}
#if defined(MP_LOCKDEBUG)
@@ -44,15 +45,15 @@ extern int __mp_lock_spinout;
#endif
static __inline void
-__mp_lock_spin(struct __mp_lock *mpl)
+__mp_lock_spin(struct __mp_lock *mpl, u_int me)
{
#ifndef MP_LOCKDEBUG
- while (mpl->mpl_count != 0)
+ while (mpl->mpl_ticket != me)
SPINLOCK_SPIN_HOOK;
#else
int ticks = __mp_lock_spinout;
- while (mpl->mpl_count != 0 && --ticks > 0)
+ while (mpl->mpl_ticket != me && --ticks > 0)
SPINLOCK_SPIN_HOOK;
if (ticks == 0) {
@@ -62,43 +63,35 @@ __mp_lock_spin(struct __mp_lock *mpl)
#endif
}
+static inline u_int
+fetch_and_add(u_int *var, u_int value)
+{
+ asm volatile("lock; xaddl %%eax, %2;"
+ : "=a" (value)
+ : "a" (value), "m" (*var)
+ : "memory");
+
+ return (value);
+}
+
void
__mp_lock(struct __mp_lock *mpl)
{
- /*
- * Please notice that mpl_count gets incremented twice for the
- * first lock. This is on purpose. The way we release the lock
- * in mp_unlock is to decrement the mpl_count and then check if
- * the lock should be released. Since mpl_count is what we're
- * spinning on, decrementing it in mpl_unlock to 0 means that
- * we can't clear mpl_cpu, because we're no longer holding the
- * lock. In theory mpl_cpu doesn't need to be cleared, but it's
- * safer to clear it and besides, setting mpl_count to 2 on the
- * first lock makes most of this code much simpler.
- */
-
- while (1) {
- long rf = read_rflags();
-
- disable_intr();
- if (x86_atomic_cas_ul(&mpl->mpl_count, 0, 1) == 0) {
- mpl->mpl_cpu = curcpu();
- }
-
- if (mpl->mpl_cpu == curcpu()) {
- mpl->mpl_count++;
- write_rflags(rf);
- break;
- }
- write_rflags(rf);
-
- __mp_lock_spin(mpl);
- }
+ struct __mp_lock_cpu *cpu = &mpl->mpl_cpus[cpu_number()];
+ long rf = read_rflags();
+
+ disable_intr();
+ if (cpu->mplc_depth++ == 0)
+ cpu->mplc_ticket = fetch_and_add(&mpl->mpl_users, 1);
+ write_rflags(rf);
+
+ __mp_lock_spin(mpl, cpu->mplc_ticket);
}
void
__mp_unlock(struct __mp_lock *mpl)
{
+ struct __mp_lock_cpu *cpu = &mpl->mpl_cpus[cpu_number()];
long rf = read_rflags();
#ifdef MP_LOCKDEBUG
@@ -109,29 +102,22 @@ __mp_unlock(struct __mp_lock *mpl)
#endif
disable_intr();
- if (--mpl->mpl_count == 1) {
- mpl->mpl_cpu = NULL;
- mpl->mpl_count = 0;
- }
+ if (--cpu->mplc_depth == 0)
+ mpl->mpl_ticket++;
write_rflags(rf);
}
int
__mp_release_all(struct __mp_lock *mpl)
{
- int rv = mpl->mpl_count - 1;
+ struct __mp_lock_cpu *cpu = &mpl->mpl_cpus[cpu_number()];
long rf = read_rflags();
-
-#ifdef MP_LOCKDEBUG
- if (mpl->mpl_cpu != curcpu()) {
- db_printf("__mp_release_all(%p): not held lock\n", mpl);
- Debugger();
- }
-#endif
+ int rv;
disable_intr();
- mpl->mpl_cpu = NULL;
- mpl->mpl_count = 0;
+ rv = cpu->mplc_depth;
+ cpu->mplc_depth = 0;
+ mpl->mpl_ticket++;
write_rflags(rf);
return (rv);
@@ -140,7 +126,8 @@ __mp_release_all(struct __mp_lock *mpl)
int
__mp_release_all_but_one(struct __mp_lock *mpl)
{
- int rv = mpl->mpl_count - 2;
+ struct __mp_lock_cpu *cpu = &mpl->mpl_cpus[cpu_number()];
+ int rv = cpu->mplc_depth - 1;
#ifdef MP_LOCKDEBUG
if (mpl->mpl_cpu != curcpu()) {
@@ -149,7 +136,7 @@ __mp_release_all_but_one(struct __mp_lock *mpl)
}
#endif
- mpl->mpl_count = 2;
+ cpu->mplc_depth = 1;
return (rv);
}
@@ -164,6 +151,8 @@ __mp_acquire_count(struct __mp_lock *mpl, int count)
int
__mp_lock_held(struct __mp_lock *mpl)
{
- return mpl->mpl_cpu == curcpu();
+ struct __mp_lock_cpu *cpu = &mpl->mpl_cpus[cpu_number()];
+
+ return (cpu->mplc_ticket == mpl->mpl_ticket && cpu->mplc_depth > 0);
}
diff --git a/sys/arch/amd64/include/mplock.h b/sys/arch/amd64/include/mplock.h
index 82b0d3212f9..118fabe0a2a 100644
--- a/sys/arch/amd64/include/mplock.h
+++ b/sys/arch/amd64/include/mplock.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: mplock.h,v 1.2 2007/11/26 23:50:03 art Exp $ */
+/* $OpenBSD: mplock.h,v 1.3 2014/03/14 02:08:57 dlg Exp $ */
/*
* Copyright (c) 2004 Niklas Hallqvist. All rights reserved.
@@ -27,9 +27,15 @@
#ifndef _MACHINE_MPLOCK_H_
#define _MACHINE_MPLOCK_H_
+struct __mp_lock_cpu {
+ u_int mplc_ticket;
+ u_int mplc_depth;
+};
+
struct __mp_lock {
- volatile struct cpu_info *mpl_cpu;
- volatile long mpl_count;
+ struct __mp_lock_cpu mpl_cpus[MAXCPUS];
+ volatile u_int mpl_ticket;
+ u_int mpl_users;
};
#ifndef _LOCORE