aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--fs/btrfs/super.c35
-rw-r--r--fs/namespace.c28
-rw-r--r--fs/nfs/super.c30
-rw-r--r--include/linux/fs.h1
4 files changed, 41 insertions, 53 deletions
diff --git a/fs/btrfs/super.c b/fs/btrfs/super.c
index cfbedd7755b..17ee7fc5e64 100644
--- a/fs/btrfs/super.c
+++ b/fs/btrfs/super.c
@@ -825,13 +825,9 @@ static char *setup_root_args(char *args)
825static struct dentry *mount_subvol(const char *subvol_name, int flags, 825static struct dentry *mount_subvol(const char *subvol_name, int flags,
826 const char *device_name, char *data) 826 const char *device_name, char *data)
827{ 827{
828 struct super_block *s;
829 struct dentry *root; 828 struct dentry *root;
830 struct vfsmount *mnt; 829 struct vfsmount *mnt;
831 struct mnt_namespace *ns_private;
832 char *newargs; 830 char *newargs;
833 struct path path;
834 int error;
835 831
836 newargs = setup_root_args(data); 832 newargs = setup_root_args(data);
837 if (!newargs) 833 if (!newargs)
@@ -842,36 +838,17 @@ static struct dentry *mount_subvol(const char *subvol_name, int flags,
842 if (IS_ERR(mnt)) 838 if (IS_ERR(mnt))
843 return ERR_CAST(mnt); 839 return ERR_CAST(mnt);
844 840
845 ns_private = create_mnt_ns(mnt); 841 root = mount_subtree(mnt, subvol_name);
846 if (IS_ERR(ns_private))
847 return ERR_CAST(ns_private);
848
849 /*
850 * This will trigger the automount of the subvol so we can just
851 * drop the mnt we have here and return the dentry that we
852 * found.
853 */
854 error = vfs_path_lookup(mnt->mnt_root, mnt, subvol_name,
855 LOOKUP_FOLLOW, &path);
856 put_mnt_ns(ns_private);
857 if (error)
858 return ERR_PTR(error);
859 842
860 if (!is_subvolume_inode(path.dentry->d_inode)) { 843 if (!IS_ERR(root) && !is_subvolume_inode(root->d_inode)) {
861 path_put(&path); 844 struct super_block *s = root->d_sb;
862 error = -EINVAL; 845 dput(root);
846 root = ERR_PTR(-EINVAL);
847 deactivate_locked_super(s);
863 printk(KERN_ERR "btrfs: '%s' is not a valid subvolume\n", 848 printk(KERN_ERR "btrfs: '%s' is not a valid subvolume\n",
864 subvol_name); 849 subvol_name);
865 return ERR_PTR(-EINVAL);
866 } 850 }
867 851
868 /* Get a ref to the sb and the dentry we found and return it */
869 s = path.mnt->mnt_sb;
870 atomic_inc(&s->s_active);
871 root = dget(path.dentry);
872 path_put(&path);
873 down_write(&s->s_umount);
874
875 return root; 852 return root;
876} 853}
877 854
diff --git a/fs/namespace.c b/fs/namespace.c
index aea4b768984..50ee30345b4 100644
--- a/fs/namespace.c
+++ b/fs/namespace.c
@@ -2490,6 +2490,34 @@ struct mnt_namespace *create_mnt_ns(struct vfsmount *mnt)
2490} 2490}
2491EXPORT_SYMBOL(create_mnt_ns); 2491EXPORT_SYMBOL(create_mnt_ns);
2492 2492
2493struct dentry *mount_subtree(struct vfsmount *mnt, const char *name)
2494{
2495 struct mnt_namespace *ns;
2496 struct path path;
2497 int err;
2498
2499 ns = create_mnt_ns(mnt);
2500 if (IS_ERR(ns))
2501 return ERR_CAST(ns);
2502
2503 err = vfs_path_lookup(mnt->mnt_root, mnt,
2504 name, LOOKUP_FOLLOW|LOOKUP_AUTOMOUNT, &path);
2505
2506 put_mnt_ns(ns);
2507
2508 if (err)
2509 return ERR_PTR(err);
2510
2511 /* trade a vfsmount reference for active sb one */
2512 atomic_inc(&path.mnt->mnt_sb->s_active);
2513 mntput(path.mnt);
2514 /* lock the sucker */
2515 down_write(&path.mnt->mnt_sb->s_umount);
2516 /* ... and return the root of (sub)tree on it */
2517 return path.dentry;
2518}
2519EXPORT_SYMBOL(mount_subtree);
2520
2493SYSCALL_DEFINE5(mount, char __user *, dev_name, char __user *, dir_name, 2521SYSCALL_DEFINE5(mount, char __user *, dev_name, char __user *, dir_name,
2494 char __user *, type, unsigned long, flags, void __user *, data) 2522 char __user *, type, unsigned long, flags, void __user *, data)
2495{ 2523{
diff --git a/fs/nfs/super.c b/fs/nfs/super.c
index 46d69f38fd5..134777406ee 100644
--- a/fs/nfs/super.c
+++ b/fs/nfs/super.c
@@ -2787,35 +2787,17 @@ static void nfs_referral_loop_unprotect(void)
2787static struct dentry *nfs_follow_remote_path(struct vfsmount *root_mnt, 2787static struct dentry *nfs_follow_remote_path(struct vfsmount *root_mnt,
2788 const char *export_path) 2788 const char *export_path)
2789{ 2789{
2790 struct mnt_namespace *ns_private;
2791 struct super_block *s;
2792 struct dentry *dentry; 2790 struct dentry *dentry;
2793 struct path path; 2791 int ret = nfs_referral_loop_protect();
2794 int ret;
2795 2792
2796 ns_private = create_mnt_ns(root_mnt); 2793 if (ret) {
2797 if (IS_ERR(ns_private)) 2794 mntput(root_mnt);
2798 return ERR_CAST(ns_private);
2799
2800 ret = nfs_referral_loop_protect();
2801 if (ret == 0) {
2802 ret = vfs_path_lookup(root_mnt->mnt_root, root_mnt,
2803 export_path, LOOKUP_FOLLOW|LOOKUP_AUTOMOUNT,
2804 &path);
2805 nfs_referral_loop_unprotect();
2806 }
2807
2808 put_mnt_ns(ns_private);
2809
2810 if (ret != 0)
2811 return ERR_PTR(ret); 2795 return ERR_PTR(ret);
2796 }
2812 2797
2813 s = path.mnt->mnt_sb; 2798 dentry = mount_subtree(root_mnt, export_path);
2814 atomic_inc(&s->s_active); 2799 nfs_referral_loop_unprotect();
2815 dentry = dget(path.dentry);
2816 2800
2817 path_put(&path);
2818 down_write(&s->s_umount);
2819 return dentry; 2801 return dentry;
2820} 2802}
2821 2803
diff --git a/include/linux/fs.h b/include/linux/fs.h
index 0c4df261af7..e3130220ce3 100644
--- a/include/linux/fs.h
+++ b/include/linux/fs.h
@@ -1886,6 +1886,7 @@ extern struct dentry *mount_single(struct file_system_type *fs_type,
1886extern struct dentry *mount_nodev(struct file_system_type *fs_type, 1886extern struct dentry *mount_nodev(struct file_system_type *fs_type,
1887 int flags, void *data, 1887 int flags, void *data,
1888 int (*fill_super)(struct super_block *, void *, int)); 1888 int (*fill_super)(struct super_block *, void *, int));
1889extern struct dentry *mount_subtree(struct vfsmount *mnt, const char *path);
1889void generic_shutdown_super(struct super_block *sb); 1890void generic_shutdown_super(struct super_block *sb);
1890void kill_block_super(struct super_block *sb); 1891void kill_block_super(struct super_block *sb);
1891void kill_anon_super(struct super_block *sb); 1892void kill_anon_super(struct super_block *sb);