diff options
Diffstat (limited to 'fs/namespace.c')
| -rw-r--r-- | fs/namespace.c | 14 |
1 files changed, 11 insertions, 3 deletions
diff --git a/fs/namespace.c b/fs/namespace.c index 7d70d63ceb29..c768f733c8d6 100644 --- a/fs/namespace.c +++ b/fs/namespace.c | |||
| @@ -965,10 +965,12 @@ EXPORT_SYMBOL(may_umount_tree); | |||
| 965 | int may_umount(struct vfsmount *mnt) | 965 | int may_umount(struct vfsmount *mnt) |
| 966 | { | 966 | { |
| 967 | int ret = 1; | 967 | int ret = 1; |
| 968 | down_read(&namespace_sem); | ||
| 968 | spin_lock(&vfsmount_lock); | 969 | spin_lock(&vfsmount_lock); |
| 969 | if (propagate_mount_busy(mnt, 2)) | 970 | if (propagate_mount_busy(mnt, 2)) |
| 970 | ret = 0; | 971 | ret = 0; |
| 971 | spin_unlock(&vfsmount_lock); | 972 | spin_unlock(&vfsmount_lock); |
| 973 | up_read(&namespace_sem); | ||
| 972 | return ret; | 974 | return ret; |
| 973 | } | 975 | } |
| 974 | 976 | ||
| @@ -1352,12 +1354,12 @@ static int attach_recursive_mnt(struct vfsmount *source_mnt, | |||
| 1352 | if (err) | 1354 | if (err) |
| 1353 | goto out_cleanup_ids; | 1355 | goto out_cleanup_ids; |
| 1354 | 1356 | ||
| 1357 | spin_lock(&vfsmount_lock); | ||
| 1358 | |||
| 1355 | if (IS_MNT_SHARED(dest_mnt)) { | 1359 | if (IS_MNT_SHARED(dest_mnt)) { |
| 1356 | for (p = source_mnt; p; p = next_mnt(p, source_mnt)) | 1360 | for (p = source_mnt; p; p = next_mnt(p, source_mnt)) |
| 1357 | set_mnt_shared(p); | 1361 | set_mnt_shared(p); |
| 1358 | } | 1362 | } |
| 1359 | |||
| 1360 | spin_lock(&vfsmount_lock); | ||
| 1361 | if (parent_path) { | 1363 | if (parent_path) { |
| 1362 | detach_mnt(source_mnt, parent_path); | 1364 | detach_mnt(source_mnt, parent_path); |
| 1363 | attach_mnt(source_mnt, path); | 1365 | attach_mnt(source_mnt, path); |
| @@ -1534,8 +1536,12 @@ static int do_remount(struct path *path, int flags, int mnt_flags, | |||
| 1534 | err = change_mount_flags(path->mnt, flags); | 1536 | err = change_mount_flags(path->mnt, flags); |
| 1535 | else | 1537 | else |
| 1536 | err = do_remount_sb(sb, flags, data, 0); | 1538 | err = do_remount_sb(sb, flags, data, 0); |
| 1537 | if (!err) | 1539 | if (!err) { |
| 1540 | spin_lock(&vfsmount_lock); | ||
| 1541 | mnt_flags |= path->mnt->mnt_flags & MNT_PNODE_MASK; | ||
| 1538 | path->mnt->mnt_flags = mnt_flags; | 1542 | path->mnt->mnt_flags = mnt_flags; |
| 1543 | spin_unlock(&vfsmount_lock); | ||
| 1544 | } | ||
| 1539 | up_write(&sb->s_umount); | 1545 | up_write(&sb->s_umount); |
| 1540 | if (!err) { | 1546 | if (!err) { |
| 1541 | security_sb_post_remount(path->mnt, flags, data); | 1547 | security_sb_post_remount(path->mnt, flags, data); |
| @@ -1665,6 +1671,8 @@ int do_add_mount(struct vfsmount *newmnt, struct path *path, | |||
| 1665 | { | 1671 | { |
| 1666 | int err; | 1672 | int err; |
| 1667 | 1673 | ||
| 1674 | mnt_flags &= ~(MNT_SHARED | MNT_WRITE_HOLD); | ||
| 1675 | |||
| 1668 | down_write(&namespace_sem); | 1676 | down_write(&namespace_sem); |
| 1669 | /* Something was mounted here while we slept */ | 1677 | /* Something was mounted here while we slept */ |
| 1670 | while (d_mountpoint(path->dentry) && | 1678 | while (d_mountpoint(path->dentry) && |
