forked from mirrors/linux
		
	vfs: syscall: Add move_mount(2) to move mounts around
Add a move_mount() system call that will move a mount from one place to another and, in the next commit, allow to attach an unattached mount tree. The new system call looks like the following: int move_mount(int from_dfd, const char *from_path, int to_dfd, const char *to_path, unsigned int flags); Signed-off-by: David Howells <dhowells@redhat.com> cc: linux-api@vger.kernel.org Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
		
							parent
							
								
									a07b200047
								
							
						
					
					
						commit
						2db154b3ea
					
				
					 8 changed files with 131 additions and 33 deletions
				
			
		| 
						 | 
				
			
			@ -399,7 +399,8 @@
 | 
			
		|||
385	i386	io_pgetevents		sys_io_pgetevents_time32	__ia32_compat_sys_io_pgetevents
 | 
			
		||||
386	i386	rseq			sys_rseq			__ia32_sys_rseq
 | 
			
		||||
387	i386	open_tree		sys_open_tree			__ia32_sys_open_tree
 | 
			
		||||
# don't use numbers 388 through 392, add new calls at the end
 | 
			
		||||
388	i386	move_mount		sys_move_mount			__ia32_sys_move_mount
 | 
			
		||||
# don't use numbers 389 through 392, add new calls at the end
 | 
			
		||||
393	i386	semget			sys_semget    			__ia32_sys_semget
 | 
			
		||||
394	i386	semctl			sys_semctl    			__ia32_compat_sys_semctl
 | 
			
		||||
395	i386	shmget			sys_shmget    			__ia32_sys_shmget
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -344,6 +344,7 @@
 | 
			
		|||
333	common	io_pgetevents		__x64_sys_io_pgetevents
 | 
			
		||||
334	common	rseq			__x64_sys_rseq
 | 
			
		||||
335	common	open_tree		__x64_sys_open_tree
 | 
			
		||||
336	common	move_mount		__x64_sys_move_mount
 | 
			
		||||
# don't use numbers 387 through 423, add new calls after the last
 | 
			
		||||
# 'common' entry
 | 
			
		||||
424	common	pidfd_send_signal	__x64_sys_pidfd_send_signal
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										128
									
								
								fs/namespace.c
									
									
									
									
									
								
							
							
						
						
									
										128
									
								
								fs/namespace.c
									
									
									
									
									
								
							| 
						 | 
				
			
			@ -2539,72 +2539,81 @@ static inline int tree_contains_unbindable(struct mount *mnt)
 | 
			
		|||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int do_move_mount(struct path *path, const char *old_name)
 | 
			
		||||
static int do_move_mount(struct path *old_path, struct path *new_path)
 | 
			
		||||
{
 | 
			
		||||
	struct path old_path, parent_path;
 | 
			
		||||
	struct path parent_path = {.mnt = NULL, .dentry = NULL};
 | 
			
		||||
	struct mount *p;
 | 
			
		||||
	struct mount *old;
 | 
			
		||||
	struct mountpoint *mp;
 | 
			
		||||
	int err;
 | 
			
		||||
	if (!old_name || !*old_name)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	err = kern_path(old_name, LOOKUP_FOLLOW, &old_path);
 | 
			
		||||
	if (err)
 | 
			
		||||
		return err;
 | 
			
		||||
 | 
			
		||||
	mp = lock_mount(path);
 | 
			
		||||
	err = PTR_ERR(mp);
 | 
			
		||||
	mp = lock_mount(new_path);
 | 
			
		||||
	if (IS_ERR(mp))
 | 
			
		||||
		goto out;
 | 
			
		||||
		return PTR_ERR(mp);
 | 
			
		||||
 | 
			
		||||
	old = real_mount(old_path.mnt);
 | 
			
		||||
	p = real_mount(path->mnt);
 | 
			
		||||
	old = real_mount(old_path->mnt);
 | 
			
		||||
	p = real_mount(new_path->mnt);
 | 
			
		||||
 | 
			
		||||
	err = -EINVAL;
 | 
			
		||||
	if (!check_mnt(p) || !check_mnt(old))
 | 
			
		||||
		goto out1;
 | 
			
		||||
 | 
			
		||||
	if (old->mnt.mnt_flags & MNT_LOCKED)
 | 
			
		||||
		goto out1;
 | 
			
		||||
 | 
			
		||||
	err = -EINVAL;
 | 
			
		||||
	if (old_path.dentry != old_path.mnt->mnt_root)
 | 
			
		||||
		goto out1;
 | 
			
		||||
		goto out;
 | 
			
		||||
 | 
			
		||||
	if (!mnt_has_parent(old))
 | 
			
		||||
		goto out1;
 | 
			
		||||
		goto out;
 | 
			
		||||
 | 
			
		||||
	if (d_is_dir(path->dentry) !=
 | 
			
		||||
	      d_is_dir(old_path.dentry))
 | 
			
		||||
		goto out1;
 | 
			
		||||
	if (old->mnt.mnt_flags & MNT_LOCKED)
 | 
			
		||||
		goto out;
 | 
			
		||||
 | 
			
		||||
	if (old_path->dentry != old_path->mnt->mnt_root)
 | 
			
		||||
		goto out;
 | 
			
		||||
 | 
			
		||||
	if (d_is_dir(new_path->dentry) !=
 | 
			
		||||
	    d_is_dir(old_path->dentry))
 | 
			
		||||
		goto out;
 | 
			
		||||
	/*
 | 
			
		||||
	 * Don't move a mount residing in a shared parent.
 | 
			
		||||
	 */
 | 
			
		||||
	if (IS_MNT_SHARED(old->mnt_parent))
 | 
			
		||||
		goto out1;
 | 
			
		||||
		goto out;
 | 
			
		||||
	/*
 | 
			
		||||
	 * Don't move a mount tree containing unbindable mounts to a destination
 | 
			
		||||
	 * mount which is shared.
 | 
			
		||||
	 */
 | 
			
		||||
	if (IS_MNT_SHARED(p) && tree_contains_unbindable(old))
 | 
			
		||||
		goto out1;
 | 
			
		||||
		goto out;
 | 
			
		||||
	err = -ELOOP;
 | 
			
		||||
	for (; mnt_has_parent(p); p = p->mnt_parent)
 | 
			
		||||
		if (p == old)
 | 
			
		||||
			goto out1;
 | 
			
		||||
			goto out;
 | 
			
		||||
 | 
			
		||||
	err = attach_recursive_mnt(old, real_mount(path->mnt), mp, &parent_path);
 | 
			
		||||
	err = attach_recursive_mnt(old, real_mount(new_path->mnt), mp,
 | 
			
		||||
				   &parent_path);
 | 
			
		||||
	if (err)
 | 
			
		||||
		goto out1;
 | 
			
		||||
		goto out;
 | 
			
		||||
 | 
			
		||||
	/* if the mount is moved, it should no longer be expire
 | 
			
		||||
	 * automatically */
 | 
			
		||||
	list_del_init(&old->mnt_expire);
 | 
			
		||||
out1:
 | 
			
		||||
	unlock_mount(mp);
 | 
			
		||||
out:
 | 
			
		||||
	unlock_mount(mp);
 | 
			
		||||
	if (!err)
 | 
			
		||||
		path_put(&parent_path);
 | 
			
		||||
	return err;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int do_move_mount_old(struct path *path, const char *old_name)
 | 
			
		||||
{
 | 
			
		||||
	struct path old_path;
 | 
			
		||||
	int err;
 | 
			
		||||
 | 
			
		||||
	if (!old_name || !*old_name)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	err = kern_path(old_name, LOOKUP_FOLLOW, &old_path);
 | 
			
		||||
	if (err)
 | 
			
		||||
		return err;
 | 
			
		||||
 | 
			
		||||
	err = do_move_mount(&old_path, path);
 | 
			
		||||
	path_put(&old_path);
 | 
			
		||||
	return err;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -3050,7 +3059,7 @@ long do_mount(const char *dev_name, const char __user *dir_name,
 | 
			
		|||
	else if (flags & (MS_SHARED | MS_PRIVATE | MS_SLAVE | MS_UNBINDABLE))
 | 
			
		||||
		retval = do_change_type(&path, flags);
 | 
			
		||||
	else if (flags & MS_MOVE)
 | 
			
		||||
		retval = do_move_mount(&path, dev_name);
 | 
			
		||||
		retval = do_move_mount_old(&path, dev_name);
 | 
			
		||||
	else
 | 
			
		||||
		retval = do_new_mount(&path, type_page, sb_flags, mnt_flags,
 | 
			
		||||
				      dev_name, data_page);
 | 
			
		||||
| 
						 | 
				
			
			@ -3278,6 +3287,61 @@ SYSCALL_DEFINE5(mount, char __user *, dev_name, char __user *, dir_name,
 | 
			
		|||
	return ksys_mount(dev_name, dir_name, type, flags, data);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Move a mount from one place to another.
 | 
			
		||||
 *
 | 
			
		||||
 * Note the flags value is a combination of MOVE_MOUNT_* flags.
 | 
			
		||||
 */
 | 
			
		||||
SYSCALL_DEFINE5(move_mount,
 | 
			
		||||
		int, from_dfd, const char *, from_pathname,
 | 
			
		||||
		int, to_dfd, const char *, to_pathname,
 | 
			
		||||
		unsigned int, flags)
 | 
			
		||||
{
 | 
			
		||||
	struct path from_path, to_path;
 | 
			
		||||
	unsigned int lflags;
 | 
			
		||||
	int ret = 0;
 | 
			
		||||
 | 
			
		||||
	if (!may_mount())
 | 
			
		||||
		return -EPERM;
 | 
			
		||||
 | 
			
		||||
	if (flags & ~MOVE_MOUNT__MASK)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	/* If someone gives a pathname, they aren't permitted to move
 | 
			
		||||
	 * from an fd that requires unmount as we can't get at the flag
 | 
			
		||||
	 * to clear it afterwards.
 | 
			
		||||
	 */
 | 
			
		||||
	lflags = 0;
 | 
			
		||||
	if (flags & MOVE_MOUNT_F_SYMLINKS)	lflags |= LOOKUP_FOLLOW;
 | 
			
		||||
	if (flags & MOVE_MOUNT_F_AUTOMOUNTS)	lflags |= LOOKUP_AUTOMOUNT;
 | 
			
		||||
	if (flags & MOVE_MOUNT_F_EMPTY_PATH)	lflags |= LOOKUP_EMPTY;
 | 
			
		||||
 | 
			
		||||
	ret = user_path_at(from_dfd, from_pathname, lflags, &from_path);
 | 
			
		||||
	if (ret < 0)
 | 
			
		||||
		return ret;
 | 
			
		||||
 | 
			
		||||
	lflags = 0;
 | 
			
		||||
	if (flags & MOVE_MOUNT_T_SYMLINKS)	lflags |= LOOKUP_FOLLOW;
 | 
			
		||||
	if (flags & MOVE_MOUNT_T_AUTOMOUNTS)	lflags |= LOOKUP_AUTOMOUNT;
 | 
			
		||||
	if (flags & MOVE_MOUNT_T_EMPTY_PATH)	lflags |= LOOKUP_EMPTY;
 | 
			
		||||
 | 
			
		||||
	ret = user_path_at(to_dfd, to_pathname, lflags, &to_path);
 | 
			
		||||
	if (ret < 0)
 | 
			
		||||
		goto out_from;
 | 
			
		||||
 | 
			
		||||
	ret = security_move_mount(&from_path, &to_path);
 | 
			
		||||
	if (ret < 0)
 | 
			
		||||
		goto out_to;
 | 
			
		||||
 | 
			
		||||
	ret = do_move_mount(&from_path, &to_path);
 | 
			
		||||
 | 
			
		||||
out_to:
 | 
			
		||||
	path_put(&to_path);
 | 
			
		||||
out_from:
 | 
			
		||||
	path_put(&from_path);
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Return true if path is reachable from root
 | 
			
		||||
 *
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -160,6 +160,10 @@
 | 
			
		|||
 *	Parse a string of security data filling in the opts structure
 | 
			
		||||
 *	@options string containing all mount options known by the LSM
 | 
			
		||||
 *	@opts binary data structure usable by the LSM
 | 
			
		||||
 * @move_mount:
 | 
			
		||||
 *	Check permission before a mount is moved.
 | 
			
		||||
 *	@from_path indicates the mount that is going to be moved.
 | 
			
		||||
 *	@to_path indicates the mountpoint that will be mounted upon.
 | 
			
		||||
 * @dentry_init_security:
 | 
			
		||||
 *	Compute a context for a dentry as the inode is not yet available
 | 
			
		||||
 *	since NFSv4 has no label backed by an EA anyway.
 | 
			
		||||
| 
						 | 
				
			
			@ -1501,6 +1505,7 @@ union security_list_options {
 | 
			
		|||
					unsigned long *set_kern_flags);
 | 
			
		||||
	int (*sb_add_mnt_opt)(const char *option, const char *val, int len,
 | 
			
		||||
			      void **mnt_opts);
 | 
			
		||||
	int (*move_mount)(const struct path *from_path, const struct path *to_path);
 | 
			
		||||
	int (*dentry_init_security)(struct dentry *dentry, int mode,
 | 
			
		||||
					const struct qstr *name, void **ctx,
 | 
			
		||||
					u32 *ctxlen);
 | 
			
		||||
| 
						 | 
				
			
			@ -1835,6 +1840,7 @@ struct security_hook_heads {
 | 
			
		|||
	struct hlist_head sb_set_mnt_opts;
 | 
			
		||||
	struct hlist_head sb_clone_mnt_opts;
 | 
			
		||||
	struct hlist_head sb_add_mnt_opt;
 | 
			
		||||
	struct hlist_head move_mount;
 | 
			
		||||
	struct hlist_head dentry_init_security;
 | 
			
		||||
	struct hlist_head dentry_create_files_as;
 | 
			
		||||
#ifdef CONFIG_SECURITY_PATH
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -250,6 +250,7 @@ int security_sb_clone_mnt_opts(const struct super_block *oldsb,
 | 
			
		|||
				unsigned long *set_kern_flags);
 | 
			
		||||
int security_add_mnt_opt(const char *option, const char *val,
 | 
			
		||||
				int len, void **mnt_opts);
 | 
			
		||||
int security_move_mount(const struct path *from_path, const struct path *to_path);
 | 
			
		||||
int security_dentry_init_security(struct dentry *dentry, int mode,
 | 
			
		||||
					const struct qstr *name, void **ctx,
 | 
			
		||||
					u32 *ctxlen);
 | 
			
		||||
| 
						 | 
				
			
			@ -611,6 +612,12 @@ static inline int security_add_mnt_opt(const char *option, const char *val,
 | 
			
		|||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline int security_move_mount(const struct path *from_path,
 | 
			
		||||
				      const struct path *to_path)
 | 
			
		||||
{
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline int security_inode_alloc(struct inode *inode)
 | 
			
		||||
{
 | 
			
		||||
	return 0;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -986,6 +986,9 @@ asmlinkage long sys_statx(int dfd, const char __user *path, unsigned flags,
 | 
			
		|||
asmlinkage long sys_rseq(struct rseq __user *rseq, uint32_t rseq_len,
 | 
			
		||||
			 int flags, uint32_t sig);
 | 
			
		||||
asmlinkage long sys_open_tree(int dfd, const char __user *path, unsigned flags);
 | 
			
		||||
asmlinkage long sys_move_mount(int from_dfd, const char __user *from_path,
 | 
			
		||||
			       int to_dfd, const char __user *to_path,
 | 
			
		||||
			       unsigned int ms_flags);
 | 
			
		||||
asmlinkage long sys_pidfd_send_signal(int pidfd, int sig,
 | 
			
		||||
				       siginfo_t __user *info,
 | 
			
		||||
				       unsigned int flags);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -61,4 +61,15 @@
 | 
			
		|||
#define OPEN_TREE_CLONE		1		/* Clone the target tree and attach the clone */
 | 
			
		||||
#define OPEN_TREE_CLOEXEC	O_CLOEXEC	/* Close the file on execve() */
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * move_mount() flags.
 | 
			
		||||
 */
 | 
			
		||||
#define MOVE_MOUNT_F_SYMLINKS		0x00000001 /* Follow symlinks on from path */
 | 
			
		||||
#define MOVE_MOUNT_F_AUTOMOUNTS		0x00000002 /* Follow automounts on from path */
 | 
			
		||||
#define MOVE_MOUNT_F_EMPTY_PATH		0x00000004 /* Empty from path permitted */
 | 
			
		||||
#define MOVE_MOUNT_T_SYMLINKS		0x00000010 /* Follow symlinks on to path */
 | 
			
		||||
#define MOVE_MOUNT_T_AUTOMOUNTS		0x00000020 /* Follow automounts on to path */
 | 
			
		||||
#define MOVE_MOUNT_T_EMPTY_PATH		0x00000040 /* Empty to path permitted */
 | 
			
		||||
#define MOVE_MOUNT__MASK		0x00000077
 | 
			
		||||
 | 
			
		||||
#endif /* _UAPI_LINUX_MOUNT_H */
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -866,6 +866,11 @@ int security_add_mnt_opt(const char *option, const char *val, int len,
 | 
			
		|||
}
 | 
			
		||||
EXPORT_SYMBOL(security_add_mnt_opt);
 | 
			
		||||
 | 
			
		||||
int security_move_mount(const struct path *from_path, const struct path *to_path)
 | 
			
		||||
{
 | 
			
		||||
	return call_int_hook(move_mount, 0, from_path, to_path);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int security_inode_alloc(struct inode *inode)
 | 
			
		||||
{
 | 
			
		||||
	int rc = lsm_inode_alloc(inode);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in a new issue