diff options
Diffstat (limited to 'kernel/user_namespace.c')
| -rw-r--r-- | kernel/user_namespace.c | 17 |
1 files changed, 13 insertions, 4 deletions
diff --git a/kernel/user_namespace.c b/kernel/user_namespace.c index d8c30db06c5b..9064b919a406 100644 --- a/kernel/user_namespace.c +++ b/kernel/user_namespace.c | |||
| @@ -62,6 +62,9 @@ int create_user_ns(struct cred *new) | |||
| 62 | kgid_t group = new->egid; | 62 | kgid_t group = new->egid; |
| 63 | int ret; | 63 | int ret; |
| 64 | 64 | ||
| 65 | if (parent_ns->level > 32) | ||
| 66 | return -EUSERS; | ||
| 67 | |||
| 65 | /* | 68 | /* |
| 66 | * Verify that we can not violate the policy of which files | 69 | * Verify that we can not violate the policy of which files |
| 67 | * may be accessed that is specified by the root directory, | 70 | * may be accessed that is specified by the root directory, |
| @@ -92,6 +95,7 @@ int create_user_ns(struct cred *new) | |||
| 92 | atomic_set(&ns->count, 1); | 95 | atomic_set(&ns->count, 1); |
| 93 | /* Leave the new->user_ns reference with the new user namespace. */ | 96 | /* Leave the new->user_ns reference with the new user namespace. */ |
| 94 | ns->parent = parent_ns; | 97 | ns->parent = parent_ns; |
| 98 | ns->level = parent_ns->level + 1; | ||
| 95 | ns->owner = owner; | 99 | ns->owner = owner; |
| 96 | ns->group = group; | 100 | ns->group = group; |
| 97 | 101 | ||
| @@ -105,16 +109,21 @@ int create_user_ns(struct cred *new) | |||
| 105 | int unshare_userns(unsigned long unshare_flags, struct cred **new_cred) | 109 | int unshare_userns(unsigned long unshare_flags, struct cred **new_cred) |
| 106 | { | 110 | { |
| 107 | struct cred *cred; | 111 | struct cred *cred; |
| 112 | int err = -ENOMEM; | ||
| 108 | 113 | ||
| 109 | if (!(unshare_flags & CLONE_NEWUSER)) | 114 | if (!(unshare_flags & CLONE_NEWUSER)) |
| 110 | return 0; | 115 | return 0; |
| 111 | 116 | ||
| 112 | cred = prepare_creds(); | 117 | cred = prepare_creds(); |
| 113 | if (!cred) | 118 | if (cred) { |
| 114 | return -ENOMEM; | 119 | err = create_user_ns(cred); |
| 120 | if (err) | ||
| 121 | put_cred(cred); | ||
| 122 | else | ||
| 123 | *new_cred = cred; | ||
| 124 | } | ||
| 115 | 125 | ||
| 116 | *new_cred = cred; | 126 | return err; |
| 117 | return create_user_ns(cred); | ||
| 118 | } | 127 | } |
| 119 | 128 | ||
| 120 | void free_user_ns(struct user_namespace *ns) | 129 | void free_user_ns(struct user_namespace *ns) |
