Commit 8b19e341 authored by Linus Torvalds's avatar Linus Torvalds

vfs: make getcwd() get the root and pwd path under rcu

This allows us to skip all the crazy spinlocks and reference count
updates, and instead use the fs sequence read-lock to get an atomic
snapshot of the root and cwd information.

We might want to make the rule that "prepend_path()" is always called
with the RCU lock held, but the RCU lock nests fine and this is the
minimal fix.
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 5762482f
...@@ -3015,15 +3015,16 @@ char *dentry_path(struct dentry *dentry, char *buf, int buflen) ...@@ -3015,15 +3015,16 @@ char *dentry_path(struct dentry *dentry, char *buf, int buflen)
return ERR_PTR(-ENAMETOOLONG); return ERR_PTR(-ENAMETOOLONG);
} }
static inline void get_fs_root_and_pwd(struct fs_struct *fs, struct path *root, static void get_fs_root_and_pwd_rcu(struct fs_struct *fs, struct path *root,
struct path *pwd) struct path *pwd)
{ {
spin_lock(&fs->lock); unsigned seq;
do {
seq = read_seqcount_begin(&fs->seq);
*root = fs->root; *root = fs->root;
path_get(root);
*pwd = fs->pwd; *pwd = fs->pwd;
path_get(pwd); } while (read_seqcount_retry(&fs->seq, seq));
spin_unlock(&fs->lock);
} }
/* /*
...@@ -3053,7 +3054,8 @@ SYSCALL_DEFINE2(getcwd, char __user *, buf, unsigned long, size) ...@@ -3053,7 +3054,8 @@ SYSCALL_DEFINE2(getcwd, char __user *, buf, unsigned long, size)
if (!page) if (!page)
return -ENOMEM; return -ENOMEM;
get_fs_root_and_pwd(current->fs, &root, &pwd); rcu_read_lock();
get_fs_root_and_pwd_rcu(current->fs, &root, &pwd);
error = -ENOENT; error = -ENOENT;
br_read_lock(&vfsmount_lock); br_read_lock(&vfsmount_lock);
...@@ -3088,8 +3090,7 @@ SYSCALL_DEFINE2(getcwd, char __user *, buf, unsigned long, size) ...@@ -3088,8 +3090,7 @@ SYSCALL_DEFINE2(getcwd, char __user *, buf, unsigned long, size)
} }
out: out:
path_put(&pwd); rcu_read_unlock();
path_put(&root);
free_page((unsigned long) page); free_page((unsigned long) page);
return error; return error;
} }
......
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