|
@@ -316,19 +316,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);
|
|
@@ -446,7 +444,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);
|
|
@@ -455,7 +453,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);
|
|
@@ -465,7 +463,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);
|
|
@@ -475,17 +473,8 @@ struct dentry *autofs4_expire_indirect(struct super_block *sb,
|
|
found:
|
|
found:
|
|
pr_debug("returning %p %pd\n", expired, expired);
|
|
pr_debug("returning %p %pd\n", 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;
|
|
}
|
|
}
|
|
|
|
|
|
@@ -496,7 +485,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;
|
|
@@ -554,7 +543,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);
|
|
|
|
|
|
@@ -583,7 +572,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);
|