diff options
Diffstat (limited to 'fs/super.c')
-rw-r--r-- | fs/super.c | 13 |
1 files changed, 10 insertions, 3 deletions
diff --git a/fs/super.c b/fs/super.c index ea662b0e5e78..b8b6a086c03b 100644 --- a/fs/super.c +++ b/fs/super.c | |||
@@ -469,7 +469,7 @@ struct super_block *sget_userns(struct file_system_type *type, | |||
469 | struct super_block *old; | 469 | struct super_block *old; |
470 | int err; | 470 | int err; |
471 | 471 | ||
472 | if (!(flags & MS_KERNMOUNT) && | 472 | if (!(flags & (MS_KERNMOUNT|MS_SUBMOUNT)) && |
473 | !(type->fs_flags & FS_USERNS_MOUNT) && | 473 | !(type->fs_flags & FS_USERNS_MOUNT) && |
474 | !capable(CAP_SYS_ADMIN)) | 474 | !capable(CAP_SYS_ADMIN)) |
475 | return ERR_PTR(-EPERM); | 475 | return ERR_PTR(-EPERM); |
@@ -499,7 +499,7 @@ retry: | |||
499 | } | 499 | } |
500 | if (!s) { | 500 | if (!s) { |
501 | spin_unlock(&sb_lock); | 501 | spin_unlock(&sb_lock); |
502 | s = alloc_super(type, flags, user_ns); | 502 | s = alloc_super(type, (flags & ~MS_SUBMOUNT), user_ns); |
503 | if (!s) | 503 | if (!s) |
504 | return ERR_PTR(-ENOMEM); | 504 | return ERR_PTR(-ENOMEM); |
505 | goto retry; | 505 | goto retry; |
@@ -540,8 +540,15 @@ struct super_block *sget(struct file_system_type *type, | |||
540 | { | 540 | { |
541 | struct user_namespace *user_ns = current_user_ns(); | 541 | struct user_namespace *user_ns = current_user_ns(); |
542 | 542 | ||
543 | /* We don't yet pass the user namespace of the parent | ||
544 | * mount through to here so always use &init_user_ns | ||
545 | * until that changes. | ||
546 | */ | ||
547 | if (flags & MS_SUBMOUNT) | ||
548 | user_ns = &init_user_ns; | ||
549 | |||
543 | /* Ensure the requestor has permissions over the target filesystem */ | 550 | /* Ensure the requestor has permissions over the target filesystem */ |
544 | if (!(flags & MS_KERNMOUNT) && !ns_capable(user_ns, CAP_SYS_ADMIN)) | 551 | if (!(flags & (MS_KERNMOUNT|MS_SUBMOUNT)) && !ns_capable(user_ns, CAP_SYS_ADMIN)) |
545 | return ERR_PTR(-EPERM); | 552 | return ERR_PTR(-EPERM); |
546 | 553 | ||
547 | return sget_userns(type, test, set, flags, user_ns, data); | 554 | return sget_userns(type, test, set, flags, user_ns, data); |