summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorThomas Gleixner <tglx@linutronix.de>2010-04-28 12:25:20 +0200
committerThomas Gleixner <tglx@linutronix.de>2010-04-28 22:54:12 +0200
commit4b44047bffdaf134f84b683efed4d828c9d2f87b (patch)
tree10b207c414b15892b8ccb9d341658a0ada2d1f36
parentf008b2b9ac46014c18036a3ed6c5e73cf4c3477b (diff)
downloadlwn-4b44047bffdaf134f84b683efed4d828c9d2f87b.tar.gz
lwn-4b44047bffdaf134f84b683efed4d828c9d2f87b.zip
fs: namespace: Make put_mnt_ns rt aware
On RT the lock() inside the preempt disabled region of get_cpu_var() results in a might sleep warning. Restructure the code and check the atomic transition to 0 open coded to avoid vfsmount_write_lock() in the case when ns->count is > 1. If ns->count == 1 then do the atomic decrement under full locking of namespace_sem and vfsmount_write_lock(). In most cases the atomic_dec_and_test() will have dropped ns->count to 0 so we need the full locking anyway. Based on a patch from John Stultz Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
-rw-r--r--fs/namespace.c25
1 files changed, 17 insertions, 8 deletions
diff --git a/fs/namespace.c b/fs/namespace.c
index 1a9232356ef9..d8924bfaa9db 100644
--- a/fs/namespace.c
+++ b/fs/namespace.c
@@ -2513,20 +2513,29 @@ void put_mnt_ns(struct mnt_namespace *ns)
{
struct vfsmount *root;
LIST_HEAD(umount_list);
- spinlock_t *lock;
- lock = &get_cpu_var(vfsmount_lock);
- if (!atomic_dec_and_lock(&ns->count, lock)) {
- put_cpu_var(vfsmount_lock);
+ /*
+ * We open code this to avoid vfsmount_write_lock() in case of
+ * ns->count > 1
+ */
+ if (atomic_add_unless(&ns->count, -1, 1))
+ return;
+
+ /*
+ * Do the full locking here as it's likely that ns->count will
+ * drop to zero and we have to take namespace_sem and all vfs
+ * mount locks anyway for umount_tree().
+ */
+ down_write(&namespace_sem);
+ vfsmount_write_lock();
+ if (!atomic_dec_and_test(&ns->count)) {
+ vfsmount_write_unlock();
+ up_write(&namespace_sem);
return;
}
root = ns->root;
ns->root = NULL;
- spin_unlock(lock);
- put_cpu_var(vfsmount_lock);
- down_write(&namespace_sem);
- vfsmount_write_lock();
umount_tree(root, 0, &umount_list);
vfsmount_write_unlock();
up_write(&namespace_sem);