[<prev] [next>] [thread-next>] [day] [month] [year] [list]
Message-Id: <1287055805-18233-1-git-send-email-louis.rilling@kerlabs.com>
Date: Thu, 14 Oct 2010 13:30:05 +0200
From: Louis Rilling <louis.rilling@...labs.com>
To: linux-kernel@...r.kernel.org
Cc: Rusty Russell <rusty@...tcorp.com.au>, Ingo Molnar <mingo@...e.hu>,
Thomas Gleixner <tglx@...utronix.de>,
Matthieu Fertré <matthieu.fertre@...labs.com>,
Louis Rilling <louis.rilling@...labs.com>
Subject: [RESEND PATCH] futex: fix key reference counter in case of requeue.
From: Matthieu Fertré <matthieu.fertre@...labs.com>
This patch ensures that we are referring to the right key when dropping
reference for the futex_wait operation.
The following scenario explains a typical case where the bug was
happening:
Process P calls futex_wait() on futex identified by 'key1'. 2 references
are taken on this key: one for the struct futex_q itself, and one for the
futex_wait operation.
If now, process P is requeued on a futex identified by 'key2', its
futex_q->key is updated from 'key1' to 'key2' and a reference is got
to 'key2' and one is dropped to 'key1'.
Later, another process calls futex_wake(): it gets a reference to
'key2', wakes process P, and drops reference to 'key2'.
Once process P is woken up, it should unqueue, drop reference to 'key2'
(the one referring to the futex_q, this is done in unqueue_me())
and to 'key1' (the one referring to futex_wait operation). Without this
patch it drops reference to 'key2' instead of 'key1'.
Signed-off-by: Matthieu Fertré <matthieu.fertre@...labs.com>
Signed-off-by: Louis Rilling <louis.rilling@...labs.com>
---
kernel/futex.c | 8 ++++++--
1 files changed, 6 insertions(+), 2 deletions(-)
diff --git a/kernel/futex.c b/kernel/futex.c
index 6a3a5fa..bed6717 100644
--- a/kernel/futex.c
+++ b/kernel/futex.c
@@ -1791,6 +1791,7 @@ static int futex_wait(u32 __user *uaddr, int fshared,
struct restart_block *restart;
struct futex_hash_bucket *hb;
struct futex_q q;
+ union futex_key key;
int ret;
if (!bitset)
@@ -1817,6 +1818,9 @@ retry:
if (ret)
goto out;
+ /* save the key in case of requeue. */
+ key = q.key;
+
/* queue_me and wait for wakeup, timeout, or a signal. */
futex_wait_queue_me(hb, &q, to);
@@ -1833,7 +1837,7 @@ retry:
* victim of a spurious wakeup as well.
*/
if (!signal_pending(current)) {
- put_futex_key(fshared, &q.key);
+ put_futex_key(fshared, &key);
goto retry;
}
@@ -1857,7 +1861,7 @@ retry:
ret = -ERESTART_RESTARTBLOCK;
out_put_key:
- put_futex_key(fshared, &q.key);
+ put_futex_key(fshared, &key);
out:
if (to) {
hrtimer_cancel(&to->timer);
--
1.5.6.5
--
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