Commit f69b6618 authored by Andrew Morton's avatar Andrew Morton Committed by Linus Torvalds

[PATCH] autofs4: fix handling of chdir and chroot

From: <raven@themaw.net>

Pushed changes in sys_chdir and sys_chroot into the revalidate/lookup by
using nameidata hint.
parent 7d1f193d
...@@ -77,24 +77,6 @@ static void autofs4_update_usage(struct dentry *dentry) ...@@ -77,24 +77,6 @@ static void autofs4_update_usage(struct dentry *dentry)
} }
} }
static void autofs4_check_pwd(struct file *file, struct file *fp)
{
struct dentry *pwd = file->f_dentry;
struct dentry *new_pwd = fp->f_dentry;
struct vfsmount *new_mnt = fp->f_vfsmnt;
/* dentry is a pwd of mountpoint so move to it */
if (current->fs->pwd == pwd)
set_fs_pwd(current->fs, new_mnt, new_pwd);
/* dentry is root of a chrooted mountpoint so move to it */
if (current->fs->root == pwd) {
set_fs_root(current->fs, new_mnt, new_pwd);
/* alternate os ABI not supported */
/* set_fs_altroot(); */
}
}
/* /*
* From 2.4 kernel readdir.c * From 2.4 kernel readdir.c
*/ */
...@@ -186,7 +168,7 @@ static int autofs4_dir_open(struct inode *inode, struct file *file) ...@@ -186,7 +168,7 @@ static int autofs4_dir_open(struct inode *inode, struct file *file)
if (!empty) if (!empty)
d_invalidate(dentry); d_invalidate(dentry);
nd.flags = LOOKUP_CONTINUE; nd.flags = LOOKUP_DIRECTORY;
status = (dentry->d_op->d_revalidate)(dentry, &nd); status = (dentry->d_op->d_revalidate)(dentry, &nd);
if (!status) if (!status)
...@@ -206,7 +188,6 @@ static int autofs4_dir_open(struct inode *inode, struct file *file) ...@@ -206,7 +188,6 @@ static int autofs4_dir_open(struct inode *inode, struct file *file)
file->private_data = NULL; file->private_data = NULL;
return status; return status;
} }
autofs4_check_pwd(file, fp);
file->private_data = fp; file->private_data = fp;
} }
out: out:
...@@ -326,7 +307,8 @@ static int try_to_fill_dentry(struct dentry *dentry, ...@@ -326,7 +307,8 @@ static int try_to_fill_dentry(struct dentry *dentry,
return 1; return 1;
} }
/* Trigger mount for path component or follow link */ /* Trigger mount for path component or follow link */
} else if (flags & LOOKUP_CONTINUE || current->link_count) { } else if (flags & (LOOKUP_CONTINUE | LOOKUP_DIRECTORY) ||
current->link_count) {
DPRINTK(("try_to_fill_entry: waiting for mount name=%.*s\n", DPRINTK(("try_to_fill_entry: waiting for mount name=%.*s\n",
dentry->d_name.len, dentry->d_name.name)); dentry->d_name.len, dentry->d_name.name));
...@@ -343,35 +325,6 @@ static int try_to_fill_dentry(struct dentry *dentry, ...@@ -343,35 +325,6 @@ static int try_to_fill_dentry(struct dentry *dentry,
spin_unlock(&dentry->d_lock); spin_unlock(&dentry->d_lock);
return 0; return 0;
} }
/* also for chdir or chroot so subsequent path walks work properly */
} else if (dentry == current->fs->pwd || dentry == current->fs->root) {
struct vfsmount *mnt;
DPRINTK(("try_to_fill_entry: waiting for mount name=%.*s\n",
dentry->d_name.len, dentry->d_name.name));
spin_lock(&dentry->d_lock);
dentry->d_flags |= DCACHE_AUTOFS_PENDING;
spin_unlock(&dentry->d_lock);
status = autofs4_wait(sbi, dentry, NFY_MOUNT);
DPRINTK(("try_to_fill_entry: mount done status=%d\n", status));
if ( status ) {
spin_lock(&dentry->d_lock);
dentry->d_flags &= ~DCACHE_AUTOFS_PENDING;
spin_unlock(&dentry->d_lock);
return 0;
}
if (dentry == current->fs->pwd) {
mnt = lookup_mnt(current->fs->pwdmnt, dentry);
set_fs_pwd(current->fs, mnt, mnt->mnt_root);
} else {
mnt = lookup_mnt(current->fs->rootmnt, dentry);
set_fs_root(current->fs, mnt, mnt->mnt_root);
}
mntput(mnt);
} }
/* We don't update the usages for the autofs daemon itself, this /* We don't update the usages for the autofs daemon itself, this
......
...@@ -517,43 +517,18 @@ asmlinkage long sys_chdir(const char __user * filename) ...@@ -517,43 +517,18 @@ asmlinkage long sys_chdir(const char __user * filename)
{ {
struct nameidata nd; struct nameidata nd;
int error; int error;
struct vfsmount *old_mnt;
struct dentry *old_dentry;
error = __user_walk(filename, LOOKUP_FOLLOW|LOOKUP_DIRECTORY, &nd); error = __user_walk(filename, LOOKUP_FOLLOW|LOOKUP_DIRECTORY, &nd);
if (error) if (error)
goto out; goto out;
old_mnt = mntget(current->fs->pwdmnt);
old_dentry = dget(current->fs->pwd);
error = permission(nd.dentry->d_inode,MAY_EXEC,&nd); error = permission(nd.dentry->d_inode,MAY_EXEC,&nd);
if (error) if (error)
goto dput_and_out; goto dput_and_out;
set_fs_pwd(current->fs, nd.mnt, nd.dentry); set_fs_pwd(current->fs, nd.mnt, nd.dentry);
/*
* if we chdir to an autofs4 mount point we must get in early
* for subsequent path_walks to work properly.
*/
if (nd.dentry->d_op && nd.dentry->d_op->d_revalidate) {
int res;
res = nd.dentry->d_op->d_revalidate(nd.dentry, &nd);
if (res) {
error = permission(current->fs->pwd->d_inode, MAY_EXEC, &nd);
if (!error)
goto dput_and_out;
} else
error = -ENOENT;
set_fs_pwd(current->fs, old_mnt, old_dentry);
}
dput_and_out: dput_and_out:
mntput(old_mnt);
dput(old_dentry);
path_release(&nd); path_release(&nd);
out: out:
return error; return error;
...@@ -593,16 +568,11 @@ asmlinkage long sys_chroot(const char __user * filename) ...@@ -593,16 +568,11 @@ asmlinkage long sys_chroot(const char __user * filename)
{ {
struct nameidata nd; struct nameidata nd;
int error; int error;
struct vfsmount *old_mnt;
struct dentry *old_dentry;
error = __user_walk(filename, LOOKUP_FOLLOW | LOOKUP_DIRECTORY | LOOKUP_NOALT, &nd); error = __user_walk(filename, LOOKUP_FOLLOW | LOOKUP_DIRECTORY | LOOKUP_NOALT, &nd);
if (error) if (error)
goto out; goto out;
old_mnt = mntget(current->fs->pwdmnt);
old_dentry = dget(current->fs->pwd);
error = permission(nd.dentry->d_inode,MAY_EXEC,&nd); error = permission(nd.dentry->d_inode,MAY_EXEC,&nd);
if (error) if (error)
goto dput_and_out; goto dput_and_out;
...@@ -612,31 +582,9 @@ asmlinkage long sys_chroot(const char __user * filename) ...@@ -612,31 +582,9 @@ asmlinkage long sys_chroot(const char __user * filename)
goto dput_and_out; goto dput_and_out;
set_fs_root(current->fs, nd.mnt, nd.dentry); set_fs_root(current->fs, nd.mnt, nd.dentry);
/*
* if we chroot to an autofs4 mount point we must get in early
* for subsequent path_walks to work properly.
*/
if (nd.dentry->d_op && nd.dentry->d_op->d_revalidate) {
int res;
res = nd.dentry->d_op->d_revalidate(nd.dentry, &nd);
if (res) {
error = permission(current->fs->pwd->d_inode, MAY_EXEC, &nd);
if (!error)
goto valid;
} else
error = -ENOENT;
set_fs_root(current->fs, old_mnt, old_dentry);
goto dput_and_out;
}
valid:
set_fs_altroot(); set_fs_altroot();
error = 0; error = 0;
dput_and_out: dput_and_out:
mntput(old_mnt);
dput(old_dentry);
path_release(&nd); path_release(&nd);
out: out:
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