diff options
author | Miklos Szeredi <mszeredi@suse.cz> | 2012-05-14 13:06:13 +0200 |
---|---|---|
committer | John Rigby <john.rigby@linaro.org> | 2012-06-25 15:02:25 -0600 |
commit | b6b7eb678036d05af00f957b7f190fe576c49100 (patch) | |
tree | c3e940c6e0e751ce213030c41f98b351df54f3ac | |
parent | 336d839d58edb7dde1d5db4d03ec1e19ab545ee9 (diff) |
UBUNTU: ubuntu: overlayfs -- vfs: introduce clone_private_mount()
Overlayfs needs a private clone of the mount, so create a function for
this and export to modules.
Signed-off-by: Miklos Szeredi <mszeredi@suse.cz>
-rw-r--r-- | fs/namespace.c | 18 | ||||
-rw-r--r-- | include/linux/mount.h | 3 |
2 files changed, 21 insertions, 0 deletions
diff --git a/fs/namespace.c b/fs/namespace.c index 7db5f2b1527..6731fc258b3 100644 --- a/fs/namespace.c +++ b/fs/namespace.c @@ -1327,6 +1327,24 @@ void drop_collected_mounts(struct vfsmount *mnt) release_mounts(&umount_list); } +struct vfsmount *clone_private_mount(struct path *path) +{ + struct mount *old_mnt = real_mount(path->mnt); + struct mount *new_mnt; + + if (IS_MNT_UNBINDABLE(old_mnt)) + return ERR_PTR(-EINVAL); + + down_read(&namespace_sem); + new_mnt = clone_mnt(old_mnt, path->dentry, CL_PRIVATE); + up_read(&namespace_sem); + if (!new_mnt) + return ERR_PTR(-ENOMEM); + + return &new_mnt->mnt; +} +EXPORT_SYMBOL_GPL(clone_private_mount); + int iterate_mounts(int (*f)(struct vfsmount *, void *), void *arg, struct vfsmount *root) { diff --git a/include/linux/mount.h b/include/linux/mount.h index d7029f4a191..344a2623eb2 100644 --- a/include/linux/mount.h +++ b/include/linux/mount.h @@ -66,6 +66,9 @@ extern void mnt_pin(struct vfsmount *mnt); extern void mnt_unpin(struct vfsmount *mnt); extern int __mnt_is_readonly(struct vfsmount *mnt); +struct path; +extern struct vfsmount *clone_private_mount(struct path *path); + struct file_system_type; extern struct vfsmount *vfs_kern_mount(struct file_system_type *type, int flags, const char *name, |