[PATCH] futex: fix key reference counter in case of requeue.

From: Louis Rilling
Date: Thu Sep 16 2010 - 04:58:24 EST


From: Matthieu Fertrà <matthieu.fertre@xxxxxxxxxxx>

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@xxxxxxxxxxx>
Signed-off-by: Louis Rilling <louis.rilling@xxxxxxxxxxx>
---
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@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/