aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichel Lespinasse <walken@google.com>2011-03-06 21:07:50 -0500
committerThomas Gleixner <tglx@linutronix.de>2011-03-10 13:56:18 -0500
commit8fe8f545c6d753ead15e1f4919d39e8f9bb49629 (patch)
treeb188ce42c3b18f481e6f727004499171aa5462c1
parenta5abba989deceb731047425812d268daf7536575 (diff)
futex: Update futex_wait_setup comments about locking
Reviving a cleanup I had done about a year ago as part of a larger futex_set_wait proposal. Over the years, the locking of the hashed futex queue got improved, so that some of the "rare but normal" race conditions described in comments can't actually happen anymore. Signed-off-by: Michel Lespinasse <walken@google.com> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Darren Hart <dvhltc@us.ibm.com> Cc: Peter Zijlstra <peterz@infradead.org> LKML-Reference: <20110307020750.GA31188@google.com> Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
-rw-r--r--kernel/futex.c11
1 files changed, 6 insertions, 5 deletions
diff --git a/kernel/futex.c b/kernel/futex.c
index b766d28accd6..3184d3b9cadf 100644
--- a/kernel/futex.c
+++ b/kernel/futex.c
@@ -1781,13 +1781,14 @@ static int futex_wait_setup(u32 __user *uaddr, u32 val, unsigned int flags,
1781 * 1781 *
1782 * The basic logical guarantee of a futex is that it blocks ONLY 1782 * The basic logical guarantee of a futex is that it blocks ONLY
1783 * if cond(var) is known to be true at the time of blocking, for 1783 * if cond(var) is known to be true at the time of blocking, for
1784 * any cond. If we queued after testing *uaddr, that would open 1784 * any cond. If we locked the hash-bucket after testing *uaddr, that
1785 * a race condition where we could block indefinitely with 1785 * would open a race condition where we could block indefinitely with
1786 * cond(var) false, which would violate the guarantee. 1786 * cond(var) false, which would violate the guarantee.
1787 * 1787 *
1788 * A consequence is that futex_wait() can return zero and absorb 1788 * On the other hand, we insert q and release the hash-bucket only
1789 * a wakeup when *uaddr != val on entry to the syscall. This is 1789 * after testing *uaddr. This guarantees that futex_wait() will NOT
1790 * rare, but normal. 1790 * absorb a wakeup if *uaddr does not match the desired values
1791 * while the syscall executes.
1791 */ 1792 */
1792retry: 1793retry:
1793 ret = get_futex_key(uaddr, flags & FLAGS_SHARED, &q->key); 1794 ret = get_futex_key(uaddr, flags & FLAGS_SHARED, &q->key);