Commit 9cf2961b authored by Al Viro's avatar Al Viro Committed by Greg Kroah-Hartman

deal with deadlock in d_walk()

commit ca5358ef upstream.

... by not hitting rename_retry for reasons other than rename having
happened.  In other words, do _not_ restart when finding that
between unlocking the child and locking the parent the former got
into __dentry_kill().  Skip the killed siblings instead...
Signed-off-by: default avatarAl Viro <viro@zeniv.linux.org.uk>
Cc: Ben Hutchings <ben@decadent.org.uk>
[hujianyang: Backported to 3.14 refer to the work of Ben Hutchings in 3.2:
 - Adjust context to make __dentry_kill() apply to d_kill()]
Signed-off-by: default avatarhujianyang <hujianyang@huawei.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent d74ebe01
...@@ -435,7 +435,7 @@ static struct dentry *d_kill(struct dentry *dentry, struct dentry *parent) ...@@ -435,7 +435,7 @@ static struct dentry *d_kill(struct dentry *dentry, struct dentry *parent)
__releases(parent->d_lock) __releases(parent->d_lock)
__releases(dentry->d_inode->i_lock) __releases(dentry->d_inode->i_lock)
{ {
list_del(&dentry->d_child); __list_del_entry(&dentry->d_child);
/* /*
* Inform d_walk() that we are no longer attached to the * Inform d_walk() that we are no longer attached to the
* dentry tree * dentry tree
...@@ -1123,33 +1123,31 @@ static void d_walk(struct dentry *parent, void *data, ...@@ -1123,33 +1123,31 @@ static void d_walk(struct dentry *parent, void *data,
/* /*
* All done at this level ... ascend and resume the search. * All done at this level ... ascend and resume the search.
*/ */
rcu_read_lock();
ascend:
if (this_parent != parent) { if (this_parent != parent) {
struct dentry *child = this_parent; struct dentry *child = this_parent;
this_parent = child->d_parent; this_parent = child->d_parent;
rcu_read_lock();
spin_unlock(&child->d_lock); spin_unlock(&child->d_lock);
spin_lock(&this_parent->d_lock); spin_lock(&this_parent->d_lock);
/* /* might go back up the wrong parent if we have had a rename. */
* might go back up the wrong parent if we have had a rename if (need_seqretry(&rename_lock, seq))
* or deletion
*/
if (this_parent != child->d_parent ||
(child->d_flags & DCACHE_DENTRY_KILLED) ||
need_seqretry(&rename_lock, seq)) {
spin_unlock(&this_parent->d_lock);
rcu_read_unlock();
goto rename_retry; goto rename_retry;
next = child->d_child.next;
while (unlikely(child->d_flags & DCACHE_DENTRY_KILLED)) {
if (next == &this_parent->d_subdirs)
goto ascend;
child = list_entry(next, struct dentry, d_child);
next = next->next;
} }
rcu_read_unlock(); rcu_read_unlock();
next = child->d_child.next;
goto resume; goto resume;
} }
if (need_seqretry(&rename_lock, seq)) { if (need_seqretry(&rename_lock, seq))
spin_unlock(&this_parent->d_lock);
goto rename_retry; goto rename_retry;
} rcu_read_unlock();
if (finish) if (finish)
finish(data); finish(data);
...@@ -1159,6 +1157,9 @@ static void d_walk(struct dentry *parent, void *data, ...@@ -1159,6 +1157,9 @@ static void d_walk(struct dentry *parent, void *data,
return; return;
rename_retry: rename_retry:
spin_unlock(&this_parent->d_lock);
rcu_read_unlock();
BUG_ON(seq & 1);
if (!retry) if (!retry)
return; return;
seq = 1; seq = 1;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment