diff options
Diffstat (limited to 'fs/namespace.c')
-rw-r--r-- | fs/namespace.c | 36 |
1 files changed, 26 insertions, 10 deletions
diff --git a/fs/namespace.c b/fs/namespace.c index 2019899f2ab8..ce97becff461 100644 --- a/fs/namespace.c +++ b/fs/namespace.c | |||
@@ -16,6 +16,7 @@ | |||
16 | #include <linux/init.h> | 16 | #include <linux/init.h> |
17 | #include <linux/quotaops.h> | 17 | #include <linux/quotaops.h> |
18 | #include <linux/acct.h> | 18 | #include <linux/acct.h> |
19 | #include <linux/capability.h> | ||
19 | #include <linux/module.h> | 20 | #include <linux/module.h> |
20 | #include <linux/seq_file.h> | 21 | #include <linux/seq_file.h> |
21 | #include <linux/namespace.h> | 22 | #include <linux/namespace.h> |
@@ -47,6 +48,10 @@ static int hash_mask __read_mostly, hash_bits __read_mostly; | |||
47 | static kmem_cache_t *mnt_cache; | 48 | static kmem_cache_t *mnt_cache; |
48 | static struct rw_semaphore namespace_sem; | 49 | static struct rw_semaphore namespace_sem; |
49 | 50 | ||
51 | /* /sys/fs */ | ||
52 | decl_subsys(fs, NULL, NULL); | ||
53 | EXPORT_SYMBOL_GPL(fs_subsys); | ||
54 | |||
50 | static inline unsigned long hash(struct vfsmount *mnt, struct dentry *dentry) | 55 | static inline unsigned long hash(struct vfsmount *mnt, struct dentry *dentry) |
51 | { | 56 | { |
52 | unsigned long tmp = ((unsigned long)mnt / L1_CACHE_BYTES); | 57 | unsigned long tmp = ((unsigned long)mnt / L1_CACHE_BYTES); |
@@ -355,14 +360,14 @@ static int show_vfsmnt(struct seq_file *m, void *v) | |||
355 | { MS_SYNCHRONOUS, ",sync" }, | 360 | { MS_SYNCHRONOUS, ",sync" }, |
356 | { MS_DIRSYNC, ",dirsync" }, | 361 | { MS_DIRSYNC, ",dirsync" }, |
357 | { MS_MANDLOCK, ",mand" }, | 362 | { MS_MANDLOCK, ",mand" }, |
358 | { MS_NOATIME, ",noatime" }, | ||
359 | { MS_NODIRATIME, ",nodiratime" }, | ||
360 | { 0, NULL } | 363 | { 0, NULL } |
361 | }; | 364 | }; |
362 | static struct proc_fs_info mnt_info[] = { | 365 | static struct proc_fs_info mnt_info[] = { |
363 | { MNT_NOSUID, ",nosuid" }, | 366 | { MNT_NOSUID, ",nosuid" }, |
364 | { MNT_NODEV, ",nodev" }, | 367 | { MNT_NODEV, ",nodev" }, |
365 | { MNT_NOEXEC, ",noexec" }, | 368 | { MNT_NOEXEC, ",noexec" }, |
369 | { MNT_NOATIME, ",noatime" }, | ||
370 | { MNT_NODIRATIME, ",nodiratime" }, | ||
366 | { 0, NULL } | 371 | { 0, NULL } |
367 | }; | 372 | }; |
368 | struct proc_fs_info *fs_infop; | 373 | struct proc_fs_info *fs_infop; |
@@ -451,7 +456,7 @@ EXPORT_SYMBOL(may_umount); | |||
451 | void release_mounts(struct list_head *head) | 456 | void release_mounts(struct list_head *head) |
452 | { | 457 | { |
453 | struct vfsmount *mnt; | 458 | struct vfsmount *mnt; |
454 | while(!list_empty(head)) { | 459 | while (!list_empty(head)) { |
455 | mnt = list_entry(head->next, struct vfsmount, mnt_hash); | 460 | mnt = list_entry(head->next, struct vfsmount, mnt_hash); |
456 | list_del_init(&mnt->mnt_hash); | 461 | list_del_init(&mnt->mnt_hash); |
457 | if (mnt->mnt_parent != mnt) { | 462 | if (mnt->mnt_parent != mnt) { |
@@ -814,7 +819,7 @@ static int graft_tree(struct vfsmount *mnt, struct nameidata *nd) | |||
814 | return -ENOTDIR; | 819 | return -ENOTDIR; |
815 | 820 | ||
816 | err = -ENOENT; | 821 | err = -ENOENT; |
817 | down(&nd->dentry->d_inode->i_sem); | 822 | mutex_lock(&nd->dentry->d_inode->i_mutex); |
818 | if (IS_DEADDIR(nd->dentry->d_inode)) | 823 | if (IS_DEADDIR(nd->dentry->d_inode)) |
819 | goto out_unlock; | 824 | goto out_unlock; |
820 | 825 | ||
@@ -826,7 +831,7 @@ static int graft_tree(struct vfsmount *mnt, struct nameidata *nd) | |||
826 | if (IS_ROOT(nd->dentry) || !d_unhashed(nd->dentry)) | 831 | if (IS_ROOT(nd->dentry) || !d_unhashed(nd->dentry)) |
827 | err = attach_recursive_mnt(mnt, nd, NULL); | 832 | err = attach_recursive_mnt(mnt, nd, NULL); |
828 | out_unlock: | 833 | out_unlock: |
829 | up(&nd->dentry->d_inode->i_sem); | 834 | mutex_unlock(&nd->dentry->d_inode->i_mutex); |
830 | if (!err) | 835 | if (!err) |
831 | security_sb_post_addmount(mnt, nd); | 836 | security_sb_post_addmount(mnt, nd); |
832 | return err; | 837 | return err; |
@@ -962,7 +967,7 @@ static int do_move_mount(struct nameidata *nd, char *old_name) | |||
962 | goto out; | 967 | goto out; |
963 | 968 | ||
964 | err = -ENOENT; | 969 | err = -ENOENT; |
965 | down(&nd->dentry->d_inode->i_sem); | 970 | mutex_lock(&nd->dentry->d_inode->i_mutex); |
966 | if (IS_DEADDIR(nd->dentry->d_inode)) | 971 | if (IS_DEADDIR(nd->dentry->d_inode)) |
967 | goto out1; | 972 | goto out1; |
968 | 973 | ||
@@ -1004,7 +1009,7 @@ static int do_move_mount(struct nameidata *nd, char *old_name) | |||
1004 | list_del_init(&old_nd.mnt->mnt_expire); | 1009 | list_del_init(&old_nd.mnt->mnt_expire); |
1005 | spin_unlock(&vfsmount_lock); | 1010 | spin_unlock(&vfsmount_lock); |
1006 | out1: | 1011 | out1: |
1007 | up(&nd->dentry->d_inode->i_sem); | 1012 | mutex_unlock(&nd->dentry->d_inode->i_mutex); |
1008 | out: | 1013 | out: |
1009 | up_write(&namespace_sem); | 1014 | up_write(&namespace_sem); |
1010 | if (!err) | 1015 | if (!err) |
@@ -1286,7 +1291,13 @@ long do_mount(char *dev_name, char *dir_name, char *type_page, | |||
1286 | mnt_flags |= MNT_NODEV; | 1291 | mnt_flags |= MNT_NODEV; |
1287 | if (flags & MS_NOEXEC) | 1292 | if (flags & MS_NOEXEC) |
1288 | mnt_flags |= MNT_NOEXEC; | 1293 | mnt_flags |= MNT_NOEXEC; |
1289 | flags &= ~(MS_NOSUID | MS_NOEXEC | MS_NODEV | MS_ACTIVE); | 1294 | if (flags & MS_NOATIME) |
1295 | mnt_flags |= MNT_NOATIME; | ||
1296 | if (flags & MS_NODIRATIME) | ||
1297 | mnt_flags |= MNT_NODIRATIME; | ||
1298 | |||
1299 | flags &= ~(MS_NOSUID | MS_NOEXEC | MS_NODEV | MS_ACTIVE | | ||
1300 | MS_NOATIME | MS_NODIRATIME); | ||
1290 | 1301 | ||
1291 | /* ... and get the mountpoint */ | 1302 | /* ... and get the mountpoint */ |
1292 | retval = path_lookup(dir_name, LOOKUP_FOLLOW, &nd); | 1303 | retval = path_lookup(dir_name, LOOKUP_FOLLOW, &nd); |
@@ -1526,6 +1537,10 @@ static void chroot_fs_refs(struct nameidata *old_nd, struct nameidata *new_nd) | |||
1526 | * pointed to by put_old must yield the same directory as new_root. No other | 1537 | * pointed to by put_old must yield the same directory as new_root. No other |
1527 | * file system may be mounted on put_old. After all, new_root is a mountpoint. | 1538 | * file system may be mounted on put_old. After all, new_root is a mountpoint. |
1528 | * | 1539 | * |
1540 | * Also, the current root cannot be on the 'rootfs' (initial ramfs) filesystem. | ||
1541 | * See Documentation/filesystems/ramfs-rootfs-initramfs.txt for alternatives | ||
1542 | * in this situation. | ||
1543 | * | ||
1529 | * Notes: | 1544 | * Notes: |
1530 | * - we don't move root/cwd if they are not at the root (reason: if something | 1545 | * - we don't move root/cwd if they are not at the root (reason: if something |
1531 | * cared enough to change them, it's probably wrong to force them elsewhere) | 1546 | * cared enough to change them, it's probably wrong to force them elsewhere) |
@@ -1569,7 +1584,7 @@ asmlinkage long sys_pivot_root(const char __user * new_root, | |||
1569 | user_nd.dentry = dget(current->fs->root); | 1584 | user_nd.dentry = dget(current->fs->root); |
1570 | read_unlock(¤t->fs->lock); | 1585 | read_unlock(¤t->fs->lock); |
1571 | down_write(&namespace_sem); | 1586 | down_write(&namespace_sem); |
1572 | down(&old_nd.dentry->d_inode->i_sem); | 1587 | mutex_lock(&old_nd.dentry->d_inode->i_mutex); |
1573 | error = -EINVAL; | 1588 | error = -EINVAL; |
1574 | if (IS_MNT_SHARED(old_nd.mnt) || | 1589 | if (IS_MNT_SHARED(old_nd.mnt) || |
1575 | IS_MNT_SHARED(new_nd.mnt->mnt_parent) || | 1590 | IS_MNT_SHARED(new_nd.mnt->mnt_parent) || |
@@ -1622,7 +1637,7 @@ asmlinkage long sys_pivot_root(const char __user * new_root, | |||
1622 | path_release(&root_parent); | 1637 | path_release(&root_parent); |
1623 | path_release(&parent_nd); | 1638 | path_release(&parent_nd); |
1624 | out2: | 1639 | out2: |
1625 | up(&old_nd.dentry->d_inode->i_sem); | 1640 | mutex_unlock(&old_nd.dentry->d_inode->i_mutex); |
1626 | up_write(&namespace_sem); | 1641 | up_write(&namespace_sem); |
1627 | path_release(&user_nd); | 1642 | path_release(&user_nd); |
1628 | path_release(&old_nd); | 1643 | path_release(&old_nd); |
@@ -1714,6 +1729,7 @@ void __init mnt_init(unsigned long mempages) | |||
1714 | i--; | 1729 | i--; |
1715 | } while (i); | 1730 | } while (i); |
1716 | sysfs_init(); | 1731 | sysfs_init(); |
1732 | subsystem_register(&fs_subsys); | ||
1717 | init_rootfs(); | 1733 | init_rootfs(); |
1718 | init_mount_tree(); | 1734 | init_mount_tree(); |
1719 | } | 1735 | } |