diff options
author | Eric W. Biederman <ebiederm@xmission.com> | 2014-12-05 19:01:11 -0500 |
---|---|---|
committer | Eric W. Biederman <ebiederm@xmission.com> | 2014-12-09 17:58:40 -0500 |
commit | 273d2c67c3e179adb1e74f403d1e9a06e3f841b5 (patch) | |
tree | 500bf14c930ea5c1db4c40dec54d95a00085552c /kernel/user_namespace.c | |
parent | 0542f17bf2c1f2430d368f44c8fcf2f82ec9e53e (diff) |
userns: Don't allow setgroups until a gid mapping has been setablished
setgroups is unique in not needing a valid mapping before it can be called,
in the case of setgroups(0, NULL) which drops all supplemental groups.
The design of the user namespace assumes that CAP_SETGID can not actually
be used until a gid mapping is established. Therefore add a helper function
to see if the user namespace gid mapping has been established and call
that function in the setgroups permission check.
This is part of the fix for CVE-2014-8989, being able to drop groups
without privilege using user namespaces.
Cc: stable@vger.kernel.org
Reviewed-by: Andy Lutomirski <luto@amacapital.net>
Signed-off-by: "Eric W. Biederman" <ebiederm@xmission.com>
Diffstat (limited to 'kernel/user_namespace.c')
-rw-r--r-- | kernel/user_namespace.c | 14 |
1 files changed, 14 insertions, 0 deletions
diff --git a/kernel/user_namespace.c b/kernel/user_namespace.c index b99c862a2e3f..27c8dab48c07 100644 --- a/kernel/user_namespace.c +++ b/kernel/user_namespace.c | |||
@@ -843,6 +843,20 @@ static bool new_idmap_permitted(const struct file *file, | |||
843 | return false; | 843 | return false; |
844 | } | 844 | } |
845 | 845 | ||
846 | bool userns_may_setgroups(const struct user_namespace *ns) | ||
847 | { | ||
848 | bool allowed; | ||
849 | |||
850 | mutex_lock(&id_map_mutex); | ||
851 | /* It is not safe to use setgroups until a gid mapping in | ||
852 | * the user namespace has been established. | ||
853 | */ | ||
854 | allowed = ns->gid_map.nr_extents != 0; | ||
855 | mutex_unlock(&id_map_mutex); | ||
856 | |||
857 | return allowed; | ||
858 | } | ||
859 | |||
846 | static void *userns_get(struct task_struct *task) | 860 | static void *userns_get(struct task_struct *task) |
847 | { | 861 | { |
848 | struct user_namespace *user_ns; | 862 | struct user_namespace *user_ns; |