autofs races
commit ea01a18494b3d7a91b2f1f2a6a5aaef4741bc294 upstream. * make autofs4_expire_indirect() skip the dentries being in process of expiry * do *not* mess with list_move(); making sure that dentry with AUTOFS_INF_EXPIRING are not picked for expiry is enough. * do not remove NO_RCU when we set EXPIRING, don't bother with smp_mb() there. Clear it at the same time we clear EXPIRING. Makes a bunch of tests simpler. * rename NO_RCU to WANT_EXPIRE, which is what it really is. Signed-off-by: Al Viro <viro@zeniv.linux.org.uk> Cc: Ian Kent <raven@themaw.net> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
9aea5e0ded
commit
30b54a26af
@ -79,9 +79,13 @@ struct autofs_info {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define AUTOFS_INF_EXPIRING (1<<0) /* dentry is in the process of expiring */
|
#define AUTOFS_INF_EXPIRING (1<<0) /* dentry is in the process of expiring */
|
||||||
#define AUTOFS_INF_NO_RCU (1<<1) /* the dentry is being considered
|
#define AUTOFS_INF_WANT_EXPIRE (1<<1) /* the dentry is being considered
|
||||||
* for expiry, so RCU_walk is
|
* for expiry, so RCU_walk is
|
||||||
* not permitted
|
* not permitted. If it progresses to
|
||||||
|
* actual expiry attempt, the flag is
|
||||||
|
* not cleared when EXPIRING is set -
|
||||||
|
* in that case it gets cleared only
|
||||||
|
* when it comes to clearing EXPIRING.
|
||||||
*/
|
*/
|
||||||
#define AUTOFS_INF_PENDING (1<<2) /* dentry pending mount */
|
#define AUTOFS_INF_PENDING (1<<2) /* dentry pending mount */
|
||||||
|
|
||||||
|
@ -315,19 +315,17 @@ struct dentry *autofs4_expire_direct(struct super_block *sb,
|
|||||||
if (ino->flags & AUTOFS_INF_PENDING)
|
if (ino->flags & AUTOFS_INF_PENDING)
|
||||||
goto out;
|
goto out;
|
||||||
if (!autofs4_direct_busy(mnt, root, timeout, do_now)) {
|
if (!autofs4_direct_busy(mnt, root, timeout, do_now)) {
|
||||||
ino->flags |= AUTOFS_INF_NO_RCU;
|
ino->flags |= AUTOFS_INF_WANT_EXPIRE;
|
||||||
spin_unlock(&sbi->fs_lock);
|
spin_unlock(&sbi->fs_lock);
|
||||||
synchronize_rcu();
|
synchronize_rcu();
|
||||||
spin_lock(&sbi->fs_lock);
|
spin_lock(&sbi->fs_lock);
|
||||||
if (!autofs4_direct_busy(mnt, root, timeout, do_now)) {
|
if (!autofs4_direct_busy(mnt, root, timeout, do_now)) {
|
||||||
ino->flags |= AUTOFS_INF_EXPIRING;
|
ino->flags |= AUTOFS_INF_EXPIRING;
|
||||||
smp_mb();
|
|
||||||
ino->flags &= ~AUTOFS_INF_NO_RCU;
|
|
||||||
init_completion(&ino->expire_complete);
|
init_completion(&ino->expire_complete);
|
||||||
spin_unlock(&sbi->fs_lock);
|
spin_unlock(&sbi->fs_lock);
|
||||||
return root;
|
return root;
|
||||||
}
|
}
|
||||||
ino->flags &= ~AUTOFS_INF_NO_RCU;
|
ino->flags &= ~AUTOFS_INF_WANT_EXPIRE;
|
||||||
}
|
}
|
||||||
out:
|
out:
|
||||||
spin_unlock(&sbi->fs_lock);
|
spin_unlock(&sbi->fs_lock);
|
||||||
@ -444,7 +442,7 @@ struct dentry *autofs4_expire_indirect(struct super_block *sb,
|
|||||||
while ((dentry = get_next_positive_subdir(dentry, root))) {
|
while ((dentry = get_next_positive_subdir(dentry, root))) {
|
||||||
spin_lock(&sbi->fs_lock);
|
spin_lock(&sbi->fs_lock);
|
||||||
ino = autofs4_dentry_ino(dentry);
|
ino = autofs4_dentry_ino(dentry);
|
||||||
if (ino->flags & AUTOFS_INF_NO_RCU)
|
if (ino->flags & AUTOFS_INF_WANT_EXPIRE)
|
||||||
expired = NULL;
|
expired = NULL;
|
||||||
else
|
else
|
||||||
expired = should_expire(dentry, mnt, timeout, how);
|
expired = should_expire(dentry, mnt, timeout, how);
|
||||||
@ -453,7 +451,7 @@ struct dentry *autofs4_expire_indirect(struct super_block *sb,
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
ino = autofs4_dentry_ino(expired);
|
ino = autofs4_dentry_ino(expired);
|
||||||
ino->flags |= AUTOFS_INF_NO_RCU;
|
ino->flags |= AUTOFS_INF_WANT_EXPIRE;
|
||||||
spin_unlock(&sbi->fs_lock);
|
spin_unlock(&sbi->fs_lock);
|
||||||
synchronize_rcu();
|
synchronize_rcu();
|
||||||
spin_lock(&sbi->fs_lock);
|
spin_lock(&sbi->fs_lock);
|
||||||
@ -463,7 +461,7 @@ struct dentry *autofs4_expire_indirect(struct super_block *sb,
|
|||||||
goto found;
|
goto found;
|
||||||
}
|
}
|
||||||
|
|
||||||
ino->flags &= ~AUTOFS_INF_NO_RCU;
|
ino->flags &= ~AUTOFS_INF_WANT_EXPIRE;
|
||||||
if (expired != dentry)
|
if (expired != dentry)
|
||||||
dput(expired);
|
dput(expired);
|
||||||
spin_unlock(&sbi->fs_lock);
|
spin_unlock(&sbi->fs_lock);
|
||||||
@ -473,17 +471,8 @@ struct dentry *autofs4_expire_indirect(struct super_block *sb,
|
|||||||
found:
|
found:
|
||||||
DPRINTK("returning %p %pd", expired, expired);
|
DPRINTK("returning %p %pd", expired, expired);
|
||||||
ino->flags |= AUTOFS_INF_EXPIRING;
|
ino->flags |= AUTOFS_INF_EXPIRING;
|
||||||
smp_mb();
|
|
||||||
ino->flags &= ~AUTOFS_INF_NO_RCU;
|
|
||||||
init_completion(&ino->expire_complete);
|
init_completion(&ino->expire_complete);
|
||||||
spin_unlock(&sbi->fs_lock);
|
spin_unlock(&sbi->fs_lock);
|
||||||
spin_lock(&sbi->lookup_lock);
|
|
||||||
spin_lock(&expired->d_parent->d_lock);
|
|
||||||
spin_lock_nested(&expired->d_lock, DENTRY_D_LOCK_NESTED);
|
|
||||||
list_move(&expired->d_parent->d_subdirs, &expired->d_child);
|
|
||||||
spin_unlock(&expired->d_lock);
|
|
||||||
spin_unlock(&expired->d_parent->d_lock);
|
|
||||||
spin_unlock(&sbi->lookup_lock);
|
|
||||||
return expired;
|
return expired;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -494,7 +483,7 @@ int autofs4_expire_wait(struct dentry *dentry, int rcu_walk)
|
|||||||
int status;
|
int status;
|
||||||
|
|
||||||
/* Block on any pending expire */
|
/* Block on any pending expire */
|
||||||
if (!(ino->flags & (AUTOFS_INF_EXPIRING | AUTOFS_INF_NO_RCU)))
|
if (!(ino->flags & AUTOFS_INF_WANT_EXPIRE))
|
||||||
return 0;
|
return 0;
|
||||||
if (rcu_walk)
|
if (rcu_walk)
|
||||||
return -ECHILD;
|
return -ECHILD;
|
||||||
@ -551,7 +540,7 @@ int autofs4_expire_run(struct super_block *sb,
|
|||||||
ino = autofs4_dentry_ino(dentry);
|
ino = autofs4_dentry_ino(dentry);
|
||||||
/* avoid rapid-fire expire attempts if expiry fails */
|
/* avoid rapid-fire expire attempts if expiry fails */
|
||||||
ino->last_used = now;
|
ino->last_used = now;
|
||||||
ino->flags &= ~AUTOFS_INF_EXPIRING;
|
ino->flags &= ~(AUTOFS_INF_EXPIRING|AUTOFS_INF_WANT_EXPIRE);
|
||||||
complete_all(&ino->expire_complete);
|
complete_all(&ino->expire_complete);
|
||||||
spin_unlock(&sbi->fs_lock);
|
spin_unlock(&sbi->fs_lock);
|
||||||
|
|
||||||
@ -579,7 +568,7 @@ int autofs4_do_expire_multi(struct super_block *sb, struct vfsmount *mnt,
|
|||||||
spin_lock(&sbi->fs_lock);
|
spin_lock(&sbi->fs_lock);
|
||||||
/* avoid rapid-fire expire attempts if expiry fails */
|
/* avoid rapid-fire expire attempts if expiry fails */
|
||||||
ino->last_used = now;
|
ino->last_used = now;
|
||||||
ino->flags &= ~AUTOFS_INF_EXPIRING;
|
ino->flags &= ~(AUTOFS_INF_EXPIRING|AUTOFS_INF_WANT_EXPIRE);
|
||||||
complete_all(&ino->expire_complete);
|
complete_all(&ino->expire_complete);
|
||||||
spin_unlock(&sbi->fs_lock);
|
spin_unlock(&sbi->fs_lock);
|
||||||
dput(dentry);
|
dput(dentry);
|
||||||
|
@ -455,7 +455,7 @@ static int autofs4_d_manage(struct dentry *dentry, bool rcu_walk)
|
|||||||
* a mount-trap.
|
* a mount-trap.
|
||||||
*/
|
*/
|
||||||
struct inode *inode;
|
struct inode *inode;
|
||||||
if (ino->flags & (AUTOFS_INF_EXPIRING | AUTOFS_INF_NO_RCU))
|
if (ino->flags & AUTOFS_INF_WANT_EXPIRE)
|
||||||
return 0;
|
return 0;
|
||||||
if (d_mountpoint(dentry))
|
if (d_mountpoint(dentry))
|
||||||
return 0;
|
return 0;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user