|
@@ -2718,7 +2718,7 @@ struct dentry *d_ancestor(struct dentry *p1, struct dentry *p2)
|
|
* This helper attempts to cope with remotely renamed directories
|
|
* This helper attempts to cope with remotely renamed directories
|
|
*
|
|
*
|
|
* It assumes that the caller is already holding
|
|
* It assumes that the caller is already holding
|
|
- * dentry->d_parent->d_inode->i_mutex, inode->i_lock and rename_lock
|
|
|
|
|
|
+ * dentry->d_parent->d_inode->i_mutex, and rename_lock
|
|
*
|
|
*
|
|
* Note: If ever the locking in lock_rename() changes, then please
|
|
* Note: If ever the locking in lock_rename() changes, then please
|
|
* remember to update this too...
|
|
* remember to update this too...
|
|
@@ -2744,7 +2744,6 @@ out_unalias:
|
|
__d_move(alias, dentry, false);
|
|
__d_move(alias, dentry, false);
|
|
ret = 0;
|
|
ret = 0;
|
|
out_err:
|
|
out_err:
|
|
- spin_unlock(&inode->i_lock);
|
|
|
|
if (m2)
|
|
if (m2)
|
|
mutex_unlock(m2);
|
|
mutex_unlock(m2);
|
|
if (m1)
|
|
if (m1)
|
|
@@ -2790,10 +2789,11 @@ struct dentry *d_splice_alias(struct inode *inode, struct dentry *dentry)
|
|
if (S_ISDIR(inode->i_mode)) {
|
|
if (S_ISDIR(inode->i_mode)) {
|
|
struct dentry *new = __d_find_any_alias(inode);
|
|
struct dentry *new = __d_find_any_alias(inode);
|
|
if (unlikely(new)) {
|
|
if (unlikely(new)) {
|
|
|
|
+ /* The reference to new ensures it remains an alias */
|
|
|
|
+ spin_unlock(&inode->i_lock);
|
|
write_seqlock(&rename_lock);
|
|
write_seqlock(&rename_lock);
|
|
if (unlikely(d_ancestor(new, dentry))) {
|
|
if (unlikely(d_ancestor(new, dentry))) {
|
|
write_sequnlock(&rename_lock);
|
|
write_sequnlock(&rename_lock);
|
|
- spin_unlock(&inode->i_lock);
|
|
|
|
dput(new);
|
|
dput(new);
|
|
new = ERR_PTR(-ELOOP);
|
|
new = ERR_PTR(-ELOOP);
|
|
pr_warn_ratelimited(
|
|
pr_warn_ratelimited(
|
|
@@ -2812,7 +2812,6 @@ struct dentry *d_splice_alias(struct inode *inode, struct dentry *dentry)
|
|
} else {
|
|
} else {
|
|
__d_move(new, dentry, false);
|
|
__d_move(new, dentry, false);
|
|
write_sequnlock(&rename_lock);
|
|
write_sequnlock(&rename_lock);
|
|
- spin_unlock(&inode->i_lock);
|
|
|
|
security_d_instantiate(new, inode);
|
|
security_d_instantiate(new, inode);
|
|
}
|
|
}
|
|
iput(inode);
|
|
iput(inode);
|