[PATCH] KEYS: Fix RCU handling in key_gc_keyring()

Previous thread: [RFC][PATCH 0/12] KVM, x86, ppc, asm-generic: moving dirty bitmaps to user space by Takuya Yoshikawa on Tuesday, May 4, 2010 - 5:56 am. (26 messages)

Next thread: [PATCH 2/2] hwmon: applesmc: Add temperature sensor labels to sysfs interface (rev2) by Henrik Rydberg on Tuesday, May 4, 2010 - 6:39 am. (2 messages)
From: David Howells
Date: Tuesday, May 4, 2010 - 6:16 am

key_gc_keyring() needs to either hold the RCU read lock or hold the keyring
semaphore if it's going to scan the keyring's list.  Given that it only needs
to read the key list, and it's doing so under a spinlock, the RCU read lock is
the thing to use.

Furthermore, the RCU check added in e7b0a61b7929632d36cf052d9e2820ef0a9c1bfe is
incorrect as holding the spinlock on key_serial_lock is not grounds for
assuming a keyring's pointer list can be read safely.  Instead, a simple
rcu_dereference() inside of the previously mentioned RCU read lock is what we
want.

Reported-by: Serge E. Hallyn <serue@us.ibm.com>
Signed-off-by: David Howells <dhowells@redhat.com>
---

 security/keys/gc.c |    9 ++++++---
 1 files changed, 6 insertions(+), 3 deletions(-)

diff --git a/security/keys/gc.c b/security/keys/gc.c
index 1990231..a46e825 100644
--- a/security/keys/gc.c
+++ b/security/keys/gc.c
@@ -77,10 +77,10 @@ static bool key_gc_keyring(struct key *keyring, time_t limit)
 		goto dont_gc;
 
 	/* scan the keyring looking for dead keys */
-	klist = rcu_dereference_check(keyring->payload.subscriptions,
-				      lockdep_is_held(&key_serial_lock));
+	rcu_read_lock();
+	klist = rcu_dereference(keyring->payload.subscriptions);
 	if (!klist)
-		goto dont_gc;
+		goto unlock_dont_gc;
 
 	for (loop = klist->nkeys - 1; loop >= 0; loop--) {
 		key = klist->keys[loop];
@@ -89,11 +89,14 @@ static bool key_gc_keyring(struct key *keyring, time_t limit)
 			goto do_gc;
 	}
 
+unlock_dont_gc:
+	rcu_read_unlock();
 dont_gc:
 	kleave(" = false");
 	return false;
 
 do_gc:
+	rcu_read_unlock();
 	key_gc_cursor = keyring->serial;
 	key_get(keyring);
 	spin_unlock(&key_serial_lock);

--

From: Serge E. Hallyn
Date: Tuesday, May 4, 2010 - 8:46 am

You're obviously being far too kind.  In apparent trend for last night

Acked-by: Serge Hallyn <serue@us.ibm.com>

thanks,
-serge
--

From: David Howells
Date: Tuesday, May 4, 2010 - 8:58 am

You pointed out the wrongness of it - it just wasn't in the way you thought.

David
--

From: Paul E. McKenney
Date: Tuesday, May 4, 2010 - 5:59 pm

As the guilty party in e7b0a61b7929632d36cf052d9e2820ef0a9c1bfe...  ;-)

--

From: David Howells
Date: Wednesday, May 5, 2010 - 2:31 am

Only partially guilty;-)

David
--

Previous thread: [RFC][PATCH 0/12] KVM, x86, ppc, asm-generic: moving dirty bitmaps to user space by Takuya Yoshikawa on Tuesday, May 4, 2010 - 5:56 am. (26 messages)

Next thread: [PATCH 2/2] hwmon: applesmc: Add temperature sensor labels to sysfs interface (rev2) by Henrik Rydberg on Tuesday, May 4, 2010 - 6:39 am. (2 messages)