[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20080817191034.GA5258@Krystal>
Date: Sun, 17 Aug 2008 15:10:35 -0400
From: Mathieu Desnoyers <mathieu.desnoyers@...ymtl.ca>
To: Linus Torvalds <torvalds@...ux-foundation.org>
Cc: "H. Peter Anvin" <hpa@...or.com>,
Jeremy Fitzhardinge <jeremy@...p.org>,
Andrew Morton <akpm@...ux-foundation.org>,
Ingo Molnar <mingo@...e.hu>, Joe Perches <joe@...ches.com>,
"Paul E. McKenney" <paulmck@...ux.vnet.ibm.com>,
linux-kernel@...r.kernel.org
Subject: [RFC PATCH] Fair low-latency rwlock v5
* Linus Torvalds (torvalds@...ux-foundation.org) wrote:
>
>
> On Sun, 17 Aug 2008, Mathieu Desnoyers wrote:
> > +/*
> > + * Uncontended fastpath.
> > + */
> > +static int fair_write_lock_irq_fast(struct fair_rwlock *rwlock)
>
> So first off, you should call this "trylock", since it doesn't necessarily
> get the lock at all. It has nothing to do with fast.
>
> Secondly:
>
> > + value = atomic_long_read(&rwlock->value);
> > + if (likely(!value)) {
> > + /* no other reader nor writer present, try to take the lock */
> > + local_bh_disable();
> > + local_irq_disable();
> > + if (likely(atomic_long_cmpxchg(&rwlock->value, value,
>
> This is actually potentially very slow.
>
> Why? If the lock is uncontended, but is not in the current CPU's caches,
> the read -> rmw operation generates multiple cache coherency protocol
> events. First it gets the line in shared mode (for the read), and then
> later it turns it into exclusive mode.
>
> So if it's likely that the value is zero (or even if it's just the only
> case we really care about), then you really should do the
>
> atomic_long_cmpxchg(&rwlock->value, 0, newvalue);
>
> thing as the _first_ access to the lock.
>
> Yeah, yeah, that means that you need to do the local_bh_disable etc first
> too, and undo it if it fails, but the failure path should be the unusual
> one.
>
> Linus
Ah, you are right. This new version implements a "single cmpxchg"
uncontended code path, changes the _fast semantic for "trylock
uncontended" and also adds the trylock primitives.
Thanks for the input,
Mathieu
Fair low-latency rwlock v5
Fair low-latency rwlock provides fairness for writers against reader threads,
but lets softirq and irq readers in even when there are subscribed writers
waiting for the lock. Writers could starve reader threads.
Changelog since v4 :
rename _fast -> trylock uncontended
Remove the read from the uncontended read and write cases : just do a cmpxchg
expecting "sub_expect", 0 for "lock", 1 subscriber for a "trylock".
Use the value returned by the first cmpxchg as following value instead of doing
an unnecessary read.
Here is why read + rmw is wrong (quoting Linus) :
"This is actually potentially very slow.
Why? If the lock is uncontended, but is not in the current CPU's caches,
the read -> rmw operation generates multiple cache coherency protocol
events. First it gets the line in shared mode (for the read), and then
later it turns it into exclusive mode."
Changelog since v3 :
Added writer subscription and fair trylock. The writer subscription keeps the
reader threads out of the lock.
Changelog since v2 :
Add writer fairness in addition to fairness wrt interrupts and softirqs.
Added contention delays performance tests for thread and interrupt contexts to
changelog.
Changelog since v1 :
- No more spinlock to protect against concurrent writes, it is done within
the existing atomic variable.
- Write fastpath with a single atomic op.
* Linus Torvalds (torvalds@...ux-foundation.org) wrote:
>
>
> On Sat, 16 Aug 2008, Mathieu Desnoyers wrote:
> >
> > I have hit this problem when tying to implement a better rwlock design
> > than is currently in the mainline kernel (I know the RT kernel has a
> > hard time with rwlocks)
>
> Have you looked at my sleping rwlock trial thing?
>
[...]
> - and because it is designed for sleeping, I'm pretty sure that you can
> easily drop interrupts in the contention path, to make
> write_lock_irq[save]() be reasonable.
>
> In particular, the third bullet is the important one: because it's
> designed to have a "contention" path that has _extra_ information for the
> contended case, you could literally make the extra information have things
> like a list of pending writers, so that you can drop interrupts on one
> CPU, while you adding information to let the reader side know that if the
> read-lock happens on that CPU, it needs to be able to continue in order to
> not deadlock.
>
> Linus
No, I just had a look at it, thanks for the pointer!
Tweakable contention behavior seems interesting, but I don't think it
deals with the fact that on a mainline kernel, when an interrupt handler
comes in and asks for a read lock, it has to get it on the spot. (RT
kernels can get away with that using threaded threaded interrupts, but
that's a completely different scheme). Therefore, the impact is that
interrupts must be disabled around write lock usage, and we end up in
the situation where this interrupt disable section can last for a long
time, given it waits for every readers (including ones which does not
disable interrupts nor softirqs) to complete.
Actually, I just used LTTng traces and eventually made a small patch to
lockdep to detect whenever a spinlock or a rwlock is used both with
interrupts enabled and disabled. Those sites are likely to produce very
high latencies and should IMHO be considered as bogus. The basic bogus
scenario is to have a spinlock held on CPU A with interrupts enabled
being interrupted and then a softirq runs. On CPU B, the same lock is
acquired with interrupts off. We therefore disable interrupts on CPU B
for the duration of the softirq currently running on the CPU A, which is
really not something that helps keeping short latencies. My preliminary
results shows that there are a lot of inconsistent spinlock/rwlock irq
on/off uses in the kernel.
This kind of scenario is pretty easy to fix for spinlocks (either move
the interrupt disable within the spinlock section if the spinlock is
never used by an interrupt handler or make sure that every users has
interrupts disabled).
The problem comes with rwlocks : it is correct to have readers both with
and without irq disable, even when interrupt handlers use the read lock.
However, the write lock has to disable interrupt in that case, and we
suffer from the high latency I pointed out. The tasklist_lock is the
perfect example of this. In the following patch, I try to address this
issue.
The core idea is this :
This "fair" rwlock writer subscribes to the lock, which locks out the
reader threads. Then, it takes waits until all reader threads exited their
critical section and takes the mutex (1 bit within the "long"). Then it
disables softirqs, locks out the softirqs, waits for all softirqs to exit their
critical section, disables irqs and locks out irqs. It then waits for irqs to
exit their critical section. Only then is the writer allowed to modify the data
structure.
The writer fast path checks for a non-contended lock (all bits set to 0) and
does an atomic cmpxchg to set the subscription, mutex, softirq exclusion and irq
exclusion bits.
The reader does an atomic cmpxchg to check if there is a subscribed writer. If
not, it increments the reader count for its context (thread, softirq, irq).
The test module is available at :
http://ltt.polymtl.ca/svn/trunk/tests/kernel/test-fair-rwlock.c
** Performance tests
Dual quad-core Xeon 2.0GHz E5405
* Lock contention delays, per context, 60s test
6 thread readers (no delay loop)
3 thread writers (10us period)
2 periodical interrupt readers on 7/8 cpus (IPIs).
writer_thread/0 iterations : 2980374, max contention 1007442 cycles
writer_thread/1 iterations : 2992675, max contention 917364 cycles
trylock_writer_thread/0 iterations : 13533386, successful iterations : 3357015
trylock_writer_thread/1 iterations : 13310980, successful iterations : 3278394
reader_thread/0 iterations : 13624664, max contention 1296450 cycles
reader_thread/1 iterations : 12183042, max contention 1524810 cycles
reader_thread/2 iterations : 13302985, max contention 1017834 cycles
reader_thread/3 iterations : 13398307, max contention 1016700 cycles
trylock_reader_thread/0 iterations : 60849620, successful iterations : 14708736
trylock_reader_thread/1 iterations : 63830891, successful iterations : 15827695
interrupt_reader_thread/0 iterations : 559
interrupt readers on CPU 0, max contention : 13914 cycles
interrupt readers on CPU 1, max contention : 16446 cycles
interrupt readers on CPU 2, max contention : 11604 cycles
interrupt readers on CPU 3, max contention : 14124 cycles
interrupt readers on CPU 4, max contention : 16494 cycles
interrupt readers on CPU 5, max contention : 12930 cycles
interrupt readers on CPU 6, max contention : 9432 cycles
interrupt readers on CPU 7, max contention : 14514 cycles
trylock_interrupt_reader_thread/0 iterations : 573
trylock interrupt readers on CPU 0, iterations 1249, successful iterations : 1135
trylock interrupt readers on CPU 1, iterations 1152, successful iterations : 1064
trylock interrupt readers on CPU 2, iterations 1239, successful iterations : 1135
trylock interrupt readers on CPU 3, iterations 1227, successful iterations : 1130
trylock interrupt readers on CPU 4, iterations 1029, successful iterations : 908
trylock interrupt readers on CPU 5, iterations 1161, successful iterations : 1063
trylock interrupt readers on CPU 6, iterations 612, successful iterations : 578
trylock interrupt readers on CPU 7, iterations 1007, successful iterations : 932
Signed-off-by: Mathieu Desnoyers <mathieu.desnoyers@...ymtl.ca>
CC: Linus Torvalds <torvalds@...ux-foundation.org>
Cc: "H. Peter Anvin" <hpa@...or.com>
CC: Jeremy Fitzhardinge <jeremy@...p.org>
CC: Andrew Morton <akpm@...ux-foundation.org>
CC: Ingo Molnar <mingo@...e.hu>
CC: "Paul E. McKenney" <paulmck@...ux.vnet.ibm.com>
---
include/linux/fair-rwlock.h | 61 ++++
lib/Makefile | 2
lib/fair-rwlock.c | 557 ++++++++++++++++++++++++++++++++++++++++++++
3 files changed, 620 insertions(+)
Index: linux-2.6-lttng/include/linux/fair-rwlock.h
===================================================================
--- /dev/null 1970-01-01 00:00:00.000000000 +0000
+++ linux-2.6-lttng/include/linux/fair-rwlock.h 2008-08-17 11:48:42.000000000 -0400
@@ -0,0 +1,61 @@
+#ifndef _LINUX_FAIR_RWLOCK_H
+#define _LINUX_FAIR_RWLOCK_H
+
+/*
+ * Fair low-latency rwlock
+ *
+ * Allows writer fairness wrt readers and also minimally impact the irq latency
+ * of the system.
+ *
+ * Mathieu Desnoyers <mathieu.desnoyers@...ymtl.ca>
+ * August 2008
+ */
+
+#include <asm/atomic.h>
+
+struct fair_rwlock {
+ atomic_long_t value;
+};
+
+/* Reader lock */
+
+/*
+ * many readers, from irq/softirq/thread context.
+ * protects against writers.
+ */
+void fair_read_lock(struct fair_rwlock *rwlock);
+int fair_read_trylock(struct fair_rwlock *rwlock);
+void fair_read_unlock(struct fair_rwlock *rwlock);
+
+/* Writer Lock */
+
+/*
+ * Subscription for write lock. Must be used for write trylock only.
+ * Insures fairness wrt reader threads.
+ */
+void fair_write_subscribe(struct fair_rwlock *rwlock);
+
+/*
+ * Safe against other writers in thread context.
+ * Safe against irq/softirq/thread readers.
+ */
+void fair_write_lock_irq(struct fair_rwlock *rwlock);
+int fair_write_trylock_subscribed_irq(struct fair_rwlock *rwlock);
+void fair_write_unlock_irq(struct fair_rwlock *rwlock);
+
+/*
+ * Safe against other writers in thread context.
+ * Safe against softirq/thread readers.
+ */
+void fair_write_lock_bh(struct fair_rwlock *rwlock);
+int fair_write_trylock_subscribed_bh(struct fair_rwlock *rwlock);
+void fair_write_unlock_bh(struct fair_rwlock *rwlock);
+
+/*
+ * Safe against other writers in thread context.
+ * Safe against thread readers.
+ */
+void fair_write_lock(struct fair_rwlock *rwlock);
+int fair_write_trylock_subscribed(struct fair_rwlock *rwlock);
+void fair_write_unlock(struct fair_rwlock *rwlock);
+#endif /* _LINUX_FAIR_RWLOCK_H */
Index: linux-2.6-lttng/lib/Makefile
===================================================================
--- linux-2.6-lttng.orig/lib/Makefile 2008-08-16 03:22:14.000000000 -0400
+++ linux-2.6-lttng/lib/Makefile 2008-08-16 03:29:12.000000000 -0400
@@ -43,6 +43,8 @@ obj-$(CONFIG_DEBUG_PREEMPT) += smp_proce
obj-$(CONFIG_DEBUG_LIST) += list_debug.o
obj-$(CONFIG_DEBUG_OBJECTS) += debugobjects.o
+obj-y += fair-rwlock.o
+
ifneq ($(CONFIG_HAVE_DEC_LOCK),y)
lib-y += dec_and_lock.o
endif
Index: linux-2.6-lttng/lib/fair-rwlock.c
===================================================================
--- /dev/null 1970-01-01 00:00:00.000000000 +0000
+++ linux-2.6-lttng/lib/fair-rwlock.c 2008-08-17 14:20:38.000000000 -0400
@@ -0,0 +1,557 @@
+/*
+ * Fair low-latency rwlock
+ *
+ * Allows writer fairness wrt readers and also minimally impact the irq latency
+ * of the system.
+ *
+ * A typical case leading to long interrupt latencies :
+ *
+ * - rwlock shared between
+ * - Rare update in thread context
+ * - Frequent slow read in thread context (task list iteration)
+ * - Fast interrupt handler read
+ *
+ * The slow write must therefore disable interrupts around the write lock,
+ * but will therefore add up to the global interrupt latency; worse case being
+ * the duration of the slow read.
+ *
+ * This "fair" rwlock writer subscribes to the lock, which locks out the reader
+ * threads. Then, it takes waits until all reader threads exited their critical
+ * section and takes the mutex (1 bit within the "long"). Then it disables
+ * softirqs, locks out the softirqs, waits for all softirqs to exit their
+ * critical section, disables irqs and locks out irqs. It then waits for irqs to
+ * exit their critical section. Only then is the writer allowed to modify the
+ * data structure.
+ *
+ * The writer fast path checks for a non-contended lock (all bits set to 0) and
+ * does an atomic cmpxchg to set the subscription, mutex, softirq exclusion and
+ * irq exclusion bits.
+ *
+ * The reader does an atomic cmpxchg to check if there is a subscribed writer.
+ * If not, it increments the reader count for its context (thread, softirq,
+ * irq).
+ *
+ * rwlock bits :
+ *
+ * - bits 0 .. log_2(NR_CPUS)-1) are the thread readers count
+ * (max number of threads : NR_CPUS)
+ * - bits log_2(NR_CPUS) .. 2*log_2(NR_CPUS)-1 are the softirq readers count
+ * (max # of softirqs: NR_CPUS)
+ * - bits 2*log_2(NR_CPUS) .. 3*log_2(NR_CPUS)-1 are the hardirq readers count
+ * (max # of hardirqs: NR_CPUS)
+ *
+ * - bits 3*log_2(NR_CPUS) .. 4*log_2(NR_CPUS)-1 are the writer subscribers
+ * count. Locks against reader threads if non zero.
+ * (max # of writer subscribers : NR_CPUS)
+ * - bit 4*log_2(NR_CPUS) is the write mutex
+ * - bit 4*log_2(NR_CPUS)+1 is the writer lock against softirqs
+ * - bit 4*log_2(NR_CPUS)+2 is the writer lock against hardirqs
+ *
+ * e.g. : NR_CPUS = 16
+ *
+ * THREAD_RMASK: 0x0000000f
+ * SOFTIRQ_RMASK: 0x000000f0
+ * HARDIRQ_RMASK: 0x00000f00
+ * SUBSCRIBERS_WMASK: 0x0000f000
+ * WRITER_MUTEX: 0x00010000
+ * SOFTIRQ_WMASK: 0x00020000
+ * HARDIRQ_WMASK: 0x00040000
+ *
+ * Bits usage :
+ *
+ * nr cpus for thread read
+ * nr cpus for softirq read
+ * nr cpus for hardirq read
+ *
+ * IRQ-safe write lock :
+ * nr cpus for write subscribers, disables new thread readers if non zero
+ * if (nr thread read == 0 && write mutex == 0)
+ * 1 bit for write mutex
+ * (softirq off)
+ * 1 bit for softirq exclusion
+ * if (nr softirq read == 0)
+ * (hardirq off)
+ * 1 bit for hardirq exclusion
+ * if (nr hardirq read == 0)
+ * -> locked
+ *
+ * Copyright 2008 Mathieu Desnoyers <mathieu.desnoyers@...ymtl.ca>
+ */
+
+#include <linux/fair-rwlock.h>
+#include <linux/hardirq.h>
+#include <linux/module.h>
+
+#if (NR_CPUS > 64 && (BITS_PER_LONG == 32 || NR_CPUS > 32768))
+#error "fair rwlock needs more bits per long to deal with that many CPUs"
+#endif
+
+#define THREAD_ROFFSET 1UL
+#define THREAD_RMASK ((NR_CPUS - 1) * THREAD_ROFFSET)
+#define SOFTIRQ_ROFFSET (THREAD_RMASK + 1)
+#define SOFTIRQ_RMASK ((NR_CPUS - 1) * SOFTIRQ_ROFFSET)
+#define HARDIRQ_ROFFSET ((SOFTIRQ_RMASK | THREAD_RMASK) + 1)
+#define HARDIRQ_RMASK ((NR_CPUS - 1) * HARDIRQ_ROFFSET)
+
+#define SUBSCRIBERS_WOFFSET \
+ ((HARDIRQ_RMASK | SOFTIRQ_RMASK | THREAD_RMASK) + 1)
+#define SUBSCRIBERS_WMASK \
+ ((NR_CPUS - 1) * SUBSCRIBERS_WOFFSET)
+#define WRITER_MUTEX \
+ ((SUBSCRIBERS_WMASK | HARDIRQ_RMASK | SOFTIRQ_RMASK | THREAD_RMASK) + 1)
+#define SOFTIRQ_WMASK (WRITER_MUTEX << 1)
+#define SOFTIRQ_WOFFSET SOFTIRQ_WMASK
+#define HARDIRQ_WMASK (SOFTIRQ_WMASK << 1)
+#define HARDIRQ_WOFFSET HARDIRQ_WMASK
+
+#ifdef FAIR_RWLOCK_DEBUG
+#define printk_dbg printk
+#else
+#define printk_dbg(fmt, args...)
+#endif
+
+
+/*
+ * Uncontended fastpath.
+ */
+static long fair_read_trylock_uncontended(struct fair_rwlock *rwlock,
+ long expect_readers, long roffset)
+{
+ /* no other reader nor writer present, try to take the lock */
+ return atomic_long_cmpxchg(&rwlock->value, expect_readers,
+ (expect_readers + roffset));
+}
+
+/*
+ * Reader lock
+ *
+ * First try to get the uncontended lock. If it is non-zero (can be common,
+ * since we allow multiple readers), pass the returned cmpxchg value to the loop
+ * to try to get the reader lock.
+ *
+ * trylock will fail if a writer is subscribed or holds the lock, but will
+ * spin if there is concurency to win the cmpxchg. It could happen if, for
+ * instance, other concurrent reads need to update the roffset or if a
+ * writer updated the lock bits which does not contend us. Since many
+ * concurrent readers is a common case, it makes sense not to fail is it
+ * happens.
+ *
+ * the non-trylock case will spin for both situations.
+ */
+
+static int _fair_read_lock_ctx(struct fair_rwlock *rwlock,
+ long roffset, long wmask, int trylock)
+{
+ long value, newvalue;
+
+ /*
+ * Try uncontended lock.
+ */
+ value = fair_read_trylock_uncontended(rwlock, 0, roffset);
+ if (likely(value == 0))
+ return 1;
+
+ for (;;) {
+ if (value & wmask) {
+ if (!trylock) {
+ /* Order value reads */
+ smp_rmb();
+ value = atomic_long_read(&rwlock->value);
+ continue;
+ } else
+ return 0;
+ }
+
+ newvalue = atomic_long_cmpxchg(&rwlock->value,
+ value, value + roffset);
+ if (likely(newvalue == value))
+ break;
+ else
+ value = newvalue;
+ }
+
+ printk_dbg("lib reader got in with value %lX, wmask %lX\n",
+ value, wmask);
+ return 1;
+}
+
+/*
+ * many readers, from irq/softirq/thread context.
+ * protects against writers.
+ */
+void fair_read_lock(struct fair_rwlock *rwlock)
+{
+ if (in_irq())
+ _fair_read_lock_ctx(rwlock, HARDIRQ_ROFFSET, HARDIRQ_WMASK, 0);
+ else if (in_softirq())
+ _fair_read_lock_ctx(rwlock, SOFTIRQ_ROFFSET, SOFTIRQ_WMASK, 0);
+ else {
+ preempt_disable();
+ _fair_read_lock_ctx(rwlock, THREAD_ROFFSET, SUBSCRIBERS_WMASK,
+ 0);
+ }
+}
+EXPORT_SYMBOL_GPL(fair_read_lock);
+
+int fair_read_trylock(struct fair_rwlock *rwlock)
+{
+ int ret;
+
+ if (in_irq())
+ return _fair_read_lock_ctx(rwlock,
+ HARDIRQ_ROFFSET, HARDIRQ_WMASK, 1);
+ else if (in_softirq())
+ return _fair_read_lock_ctx(rwlock,
+ SOFTIRQ_ROFFSET, SOFTIRQ_WMASK, 1);
+ else {
+ preempt_disable();
+ ret = _fair_read_lock_ctx(rwlock,
+ THREAD_ROFFSET, SUBSCRIBERS_WMASK, 1);
+ if (!ret)
+ preempt_enable();
+ return ret;
+ }
+}
+EXPORT_SYMBOL_GPL(fair_read_trylock);
+
+void fair_read_unlock(struct fair_rwlock *rwlock)
+{
+ /* atomic_long_sub orders reads */
+ if (in_irq())
+ atomic_long_sub(HARDIRQ_ROFFSET, &rwlock->value);
+ else if (in_softirq())
+ atomic_long_sub(SOFTIRQ_ROFFSET, &rwlock->value);
+ else {
+ atomic_long_sub(THREAD_ROFFSET, &rwlock->value);
+ preempt_enable();
+ }
+}
+EXPORT_SYMBOL_GPL(fair_read_unlock);
+
+/* Writer lock */
+
+/*
+ * Lock out a specific execution context from the read lock. Wait for both the
+ * rmask and the wmask to be empty before proceeding to take the lock.
+ */
+static void _fair_write_lock_ctx_wait(struct fair_rwlock *rwlock,
+ long value, long rmask, long wmask)
+{
+ long newvalue;
+
+ for (;;) {
+ if (value & (rmask | wmask)) {
+ /* Order value reads */
+ smp_rmb();
+ value = atomic_long_read(&rwlock->value);
+ continue;
+ }
+ newvalue = atomic_long_cmpxchg(&rwlock->value,
+ value, value | wmask);
+ if (likely(newvalue == value))
+ break;
+ else
+ value = newvalue;
+ }
+ printk_dbg("lib writer got in with value %lX, new %lX, rmask %lX\n",
+ value, value | wmask, rmask);
+}
+
+/*
+ * Lock out a specific execution context from the read lock. First lock the read
+ * context out of the lock, then wait for every readers to exit their critical
+ * section.
+ */
+static void _fair_write_lock_ctx_force(struct fair_rwlock *rwlock,
+ long rmask, long woffset)
+{
+ long value;
+
+ atomic_long_add(woffset, &rwlock->value);
+ do {
+ value = atomic_long_read(&rwlock->value);
+ /* Order rwlock->value read wrt following reads */
+ smp_rmb();
+ } while (value & rmask);
+ printk_dbg("lib writer got in with value %lX, woffset %lX, rmask %lX\n",
+ value, woffset, rmask);
+}
+
+void fair_write_subscribe(struct fair_rwlock *rwlock)
+{
+ preempt_disable();
+
+ /* lock out threads */
+ atomic_long_add(SUBSCRIBERS_WOFFSET, &rwlock->value);
+}
+EXPORT_SYMBOL_GPL(fair_write_subscribe);
+
+/*
+ * Uncontended fastpath.
+ */
+static long
+fair_write_trylock_irq_uncontended(struct fair_rwlock *rwlock,
+ long expect_sub, long sub_woffset)
+{
+ long value;
+
+ /* no other reader nor writer present, try to take the lock */
+ local_bh_disable();
+ local_irq_disable();
+ value = atomic_long_cmpxchg(&rwlock->value, expect_sub,
+ expect_sub + (sub_woffset | SOFTIRQ_WOFFSET
+ | HARDIRQ_WOFFSET | WRITER_MUTEX));
+ if (unlikely(value != expect_sub)) {
+ local_irq_enable();
+ local_bh_enable();
+ }
+ return value;
+}
+
+/*
+ * Safe against other writers in thread context.
+ * Safe against irq/softirq/thread readers.
+ */
+void fair_write_lock_irq(struct fair_rwlock *rwlock)
+{
+ long value;
+
+ preempt_disable();
+ value = fair_write_trylock_irq_uncontended(rwlock,
+ 0, SUBSCRIBERS_WOFFSET);
+ if (likely(!value))
+ return;
+
+ /* lock out threads */
+ atomic_long_add(SUBSCRIBERS_WOFFSET, &rwlock->value);
+
+ /* lock out other writers when no reader threads left */
+ _fair_write_lock_ctx_wait(rwlock, value + SUBSCRIBERS_WOFFSET,
+ THREAD_RMASK, WRITER_MUTEX);
+
+ /* lock out softirqs */
+ local_bh_disable();
+ _fair_write_lock_ctx_force(rwlock, SOFTIRQ_RMASK, SOFTIRQ_WOFFSET);
+
+ /* lock out hardirqs */
+ local_irq_disable();
+ _fair_write_lock_ctx_force(rwlock, HARDIRQ_RMASK, HARDIRQ_WOFFSET);
+
+ /* atomic_long_cmpxchg orders writes */
+}
+EXPORT_SYMBOL_GPL(fair_write_lock_irq);
+
+/*
+ * Must already be subscribed. Will fail if there is contention caused by
+ * interrupt readers, softirq readers, other writers. Will also fail if there is
+ * concurrent cmpxchg update, even if it is only a writer subscription. It is
+ * not considered as a common case and therefore does not jusfity looping.
+ *
+ * Second cmpxchg deals with other write subscribers.
+ */
+int fair_write_trylock_subscribed_irq(struct fair_rwlock *rwlock)
+{
+ long value;
+
+ value = fair_write_trylock_irq_uncontended(rwlock,
+ SUBSCRIBERS_WOFFSET, 0);
+ if (likely(value == SUBSCRIBERS_WOFFSET))
+ return 1;
+
+ if (likely(!(value & ~SUBSCRIBERS_WMASK))) {
+ /* no other reader nor writer present, try to take the lock */
+ local_bh_disable();
+ local_irq_disable();
+ if (likely(atomic_long_cmpxchg(&rwlock->value, value,
+ value + (SOFTIRQ_WOFFSET
+ | HARDIRQ_WOFFSET | WRITER_MUTEX))
+ == value))
+ return 1;
+ local_irq_enable();
+ local_bh_enable();
+ }
+ return 0;
+
+}
+EXPORT_SYMBOL_GPL(fair_write_trylock_subscribed_irq);
+
+void fair_write_unlock_irq(struct fair_rwlock *rwlock)
+{
+ /*
+ * atomic_long_sub makes sure we commit the data before reenabling
+ * the lock.
+ */
+ atomic_long_sub(HARDIRQ_WOFFSET | SOFTIRQ_WOFFSET
+ | WRITER_MUTEX | SUBSCRIBERS_WOFFSET,
+ &rwlock->value);
+ local_irq_enable();
+ local_bh_enable();
+ preempt_enable();
+}
+EXPORT_SYMBOL_GPL(fair_write_unlock_irq);
+
+/*
+ * Uncontended fastpath.
+ */
+static long fair_write_trylock_bh_uncontended(struct fair_rwlock *rwlock,
+ long expect_sub, long sub_woffset)
+{
+ long value;
+
+ /* no other reader nor writer present, try to take the lock */
+ local_bh_disable();
+ value = atomic_long_cmpxchg(&rwlock->value, expect_sub,
+ expect_sub + (sub_woffset | SOFTIRQ_WOFFSET | WRITER_MUTEX));
+ if (unlikely(value != expect_sub))
+ local_bh_enable();
+ return value;
+}
+
+/*
+ * Safe against other writers in thread context.
+ * Safe against softirq/thread readers.
+ */
+void fair_write_lock_bh(struct fair_rwlock *rwlock)
+{
+ long value;
+
+ preempt_disable();
+ value = fair_write_trylock_bh_uncontended(rwlock,
+ 0, SUBSCRIBERS_WOFFSET);
+ if (likely(!value))
+ return;
+
+ /* lock out threads */
+ atomic_long_add(SUBSCRIBERS_WOFFSET, &rwlock->value);
+
+ /* lock out other writers when no reader threads left */
+ _fair_write_lock_ctx_wait(rwlock, value + SUBSCRIBERS_WOFFSET,
+ THREAD_RMASK, WRITER_MUTEX);
+
+ /* lock out softirqs */
+ local_bh_disable();
+ _fair_write_lock_ctx_force(rwlock, SOFTIRQ_RMASK, SOFTIRQ_WOFFSET);
+
+ /* atomic_long_cmpxchg orders writes */
+}
+EXPORT_SYMBOL_GPL(fair_write_lock_bh);
+
+/*
+ * Must already be subscribed. Will fail if there is contention caused by
+ * softirq readers, other writers. Will also fail if there is concurrent cmpxchg
+ * update, even if it is only a writer subscription. It is not considered as a
+ * common case and therefore does not jusfity looping.
+ */
+int fair_write_trylock_subscribed_bh(struct fair_rwlock *rwlock)
+{
+ long value;
+
+ value = fair_write_trylock_bh_uncontended(rwlock,
+ SUBSCRIBERS_WOFFSET, 0);
+ if (likely(value == SUBSCRIBERS_WOFFSET))
+ return 1;
+
+ if (likely(!(value & ~SUBSCRIBERS_WMASK))) {
+ /* no other reader nor writer present, try to take the lock */
+ local_bh_disable();
+ local_irq_disable();
+ if (likely(atomic_long_cmpxchg(&rwlock->value, value,
+ value + (SOFTIRQ_WOFFSET | WRITER_MUTEX))
+ == value))
+ return 1;
+ local_irq_enable();
+ local_bh_enable();
+ }
+ return 0;
+
+}
+EXPORT_SYMBOL_GPL(fair_write_trylock_subscribed_bh);
+
+void fair_write_unlock_bh(struct fair_rwlock *rwlock)
+{
+ /*
+ * atomic_long_sub makes sure we commit the data before reenabling
+ * the lock.
+ */
+ atomic_long_sub(SOFTIRQ_WOFFSET | WRITER_MUTEX | SUBSCRIBERS_WOFFSET,
+ &rwlock->value);
+ local_bh_enable();
+ preempt_enable();
+}
+EXPORT_SYMBOL_GPL(fair_write_unlock_bh);
+
+/*
+ * Uncontended fastpath.
+ */
+static long fair_write_trylock_uncontended(struct fair_rwlock *rwlock,
+ long expect_sub, long sub_woffset)
+{
+ /* no other reader nor writer present, try to take the lock */
+ return atomic_long_cmpxchg(&rwlock->value, expect_sub,
+ (expect_sub + sub_woffset)
+ | WRITER_MUTEX);
+}
+
+/*
+ * Safe against other writers in thread context.
+ * Safe against thread readers.
+ */
+void fair_write_lock(struct fair_rwlock *rwlock)
+{
+ long value;
+
+ preempt_disable();
+ value = fair_write_trylock_uncontended(rwlock, 0, SUBSCRIBERS_WOFFSET);
+ if (likely(!value))
+ return;
+
+ /* lock out threads */
+ atomic_long_add(SUBSCRIBERS_WOFFSET, &rwlock->value);
+
+ /* lock out other writers when no reader threads left */
+ _fair_write_lock_ctx_wait(rwlock, value + SUBSCRIBERS_WOFFSET,
+ THREAD_RMASK, WRITER_MUTEX);
+
+ /* atomic_long_cmpxchg orders writes */
+}
+EXPORT_SYMBOL_GPL(fair_write_lock);
+
+/*
+ * Must already be subscribed. Will fail if there is contention caused by
+ * other writers. Will also fail if there is concurrent cmpxchg update, even if
+ * it is only a writer subscription. It is not considered as a common case and
+ * therefore does not jusfity looping.
+ */
+int fair_write_trylock_subscribed(struct fair_rwlock *rwlock)
+{
+ long value;
+
+ value = fair_write_trylock_uncontended(rwlock, SUBSCRIBERS_WOFFSET, 0);
+ if (likely(value == SUBSCRIBERS_WOFFSET))
+ return 1;
+
+ if (likely(!(value & ~SUBSCRIBERS_WMASK))) {
+ /* no other reader nor writer present, try to take the lock */
+ local_bh_disable();
+ local_irq_disable();
+ if (likely(atomic_long_cmpxchg(&rwlock->value, value,
+ value | WRITER_MUTEX) == value))
+ return 1;
+ local_irq_enable();
+ local_bh_enable();
+ }
+ return 0;
+
+}
+EXPORT_SYMBOL_GPL(fair_write_trylock_subscribed);
+
+void fair_write_unlock(struct fair_rwlock *rwlock)
+{
+ /*
+ * atomic_long_sub makes sure we commit the data before reenabling
+ * the lock.
+ */
+ atomic_long_sub(WRITER_MUTEX | SUBSCRIBERS_WOFFSET, &rwlock->value);
+ preempt_enable();
+}
+EXPORT_SYMBOL_GPL(fair_write_unlock);
--
Mathieu Desnoyers
OpenPGP key fingerprint: 8CD5 52C3 8E3C 4140 715F BA06 3F25 A8FE 3BAE 9A68
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists