diff options
author | Thomas Gleixner <tglx@linutronix.de> | 2010-04-28 12:25:20 +0200 |
---|---|---|
committer | Thomas Gleixner <tglx@linutronix.de> | 2010-04-28 22:54:12 +0200 |
commit | 4b44047bffdaf134f84b683efed4d828c9d2f87b (patch) | |
tree | 10b207c414b15892b8ccb9d341658a0ada2d1f36 | |
parent | f008b2b9ac46014c18036a3ed6c5e73cf4c3477b (diff) | |
download | lwn-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.c | 25 |
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); |