fs: add ksys_mount() helper; remove in-kernel calls to sys_mount()
authorDominik Brodowski <linux@dominikbrodowski.net>
Sun, 11 Mar 2018 10:34:39 +0000 (11:34 +0100)
committerDominik Brodowski <linux@dominikbrodowski.net>
Mon, 2 Apr 2018 18:15:48 +0000 (20:15 +0200)
Using this helper allows us to avoid the in-kernel calls to the sys_mount()
syscall. The ksys_ prefix denotes that this function is meant as a drop-in
replacement for the syscall. In particular, it uses the same calling
convention as sys_mount().

In the near future, all callers of ksys_mount() should be converted to call
do_mount() directly.

This patch is part of a series which removes in-kernel calls to syscalls.
On this basis, the syscall entry path can be streamlined. For details, see
http://lkml.kernel.org/r/20180325162527.GA17492@light.dominikbrodowski.net

Cc: Alexander Viro <viro@zeniv.linux.org.uk>
Signed-off-by: Dominik Brodowski <linux@dominikbrodowski.net>
drivers/base/devtmpfs.c
fs/namespace.c
include/linux/syscalls.h
init/do_mounts.c
init/do_mounts_initrd.c

index 50025d7959cbd36698ab05168a76b411c7f5ad7f..4afb04686c8e317ef4f24b41e50fa9d73f7f9722 100644 (file)
@@ -356,7 +356,8 @@ int devtmpfs_mount(const char *mntdir)
        if (!thread)
                return 0;
 
-       err = sys_mount("devtmpfs", (char *)mntdir, "devtmpfs", MS_SILENT, NULL);
+       err = ksys_mount("devtmpfs", (char *)mntdir, "devtmpfs", MS_SILENT,
+                        NULL);
        if (err)
                printk(KERN_INFO "devtmpfs: error mounting %i\n", err);
        else
@@ -382,7 +383,7 @@ static int devtmpfsd(void *p)
        *err = sys_unshare(CLONE_NEWNS);
        if (*err)
                goto out;
-       *err = sys_mount("devtmpfs", "/", "devtmpfs", MS_SILENT, options);
+       *err = ksys_mount("devtmpfs", "/", "devtmpfs", MS_SILENT, options);
        if (*err)
                goto out;
        sys_chdir("/.."); /* will traverse into overmounted root */
index 9d1374ab6e06f2cd7b57aedf196c53a745a1a683..642b8b2299443da0738a202b78cfc921e274928e 100644 (file)
@@ -3032,8 +3032,8 @@ struct dentry *mount_subtree(struct vfsmount *mnt, const char *name)
 }
 EXPORT_SYMBOL(mount_subtree);
 
-SYSCALL_DEFINE5(mount, char __user *, dev_name, char __user *, dir_name,
-               char __user *, type, unsigned long, flags, void __user *, data)
+int ksys_mount(char __user *dev_name, char __user *dir_name, char __user *type,
+              unsigned long flags, void __user *data)
 {
        int ret;
        char *kernel_type;
@@ -3066,6 +3066,12 @@ out_type:
        return ret;
 }
 
+SYSCALL_DEFINE5(mount, char __user *, dev_name, char __user *, dir_name,
+               char __user *, type, unsigned long, flags, void __user *, data)
+{
+       return ksys_mount(dev_name, dir_name, type, flags, data);
+}
+
 /*
  * Return true if path is reachable from root
  *
index 6ab7ed71a8b602e7dceb0d0e154e0749d30b26e5..3a9f9c534624b9eb644071c7b75240859322168d 100644 (file)
@@ -946,4 +946,7 @@ asmlinkage long sys_statx(int dfd, const char __user *path, unsigned flags,
  * the ksys_xyzyyz() functions prototyped below.
  */
 
+int ksys_mount(char __user *dev_name, char __user *dir_name, char __user *type,
+              unsigned long flags, void __user *data);
+
 #endif
index 7cf4f6dafd5f32031db0a1d28072fc93c0474f5a..eb768de43d847046597d53740167c6a2af6098dc 100644 (file)
@@ -363,7 +363,7 @@ static void __init get_fs_names(char *page)
 static int __init do_mount_root(char *name, char *fs, int flags, void *data)
 {
        struct super_block *s;
-       int err = sys_mount(name, "/root", fs, flags, data);
+       int err = ksys_mount(name, "/root", fs, flags, data);
        if (err)
                return err;
 
@@ -599,7 +599,7 @@ void __init prepare_namespace(void)
        mount_root();
 out:
        devtmpfs_mount("dev");
-       sys_mount(".", "/", NULL, MS_MOVE, NULL);
+       ksys_mount(".", "/", NULL, MS_MOVE, NULL);
        sys_chroot(".");
 }
 
index 53d4f0f326e74efb1167b6582624b32e74347c02..7868a6039fb4519c8a3dc175d7f8454113e2fc01 100644 (file)
@@ -43,7 +43,7 @@ static int init_linuxrc(struct subprocess_info *info, struct cred *new)
        sys_dup(0);
        /* move initrd over / and chdir/chroot in initrd root */
        sys_chdir("/root");
-       sys_mount(".", "/", NULL, MS_MOVE, NULL);
+       ksys_mount(".", "/", NULL, MS_MOVE, NULL);
        sys_chroot(".");
        sys_setsid();
        return 0;
@@ -81,7 +81,7 @@ static void __init handle_initrd(void)
        current->flags &= ~PF_FREEZER_SKIP;
 
        /* move initrd to rootfs' /old */
-       sys_mount("..", ".", NULL, MS_MOVE, NULL);
+       ksys_mount("..", ".", NULL, MS_MOVE, NULL);
        /* switch root and cwd back to / of rootfs */
        sys_chroot("..");
 
@@ -95,7 +95,7 @@ static void __init handle_initrd(void)
        mount_root();
 
        printk(KERN_NOTICE "Trying to move old root to /initrd ... ");
-       error = sys_mount("/old", "/root/initrd", NULL, MS_MOVE, NULL);
+       error = ksys_mount("/old", "/root/initrd", NULL, MS_MOVE, NULL);
        if (!error)
                printk("okay\n");
        else {