ucounts: Move get_ucounts from cred_alloc_blank to key_change_session_keyring
authorEric W. Biederman <ebiederm@xmission.com>
Sat, 16 Oct 2021 17:17:30 +0000 (12:17 -0500)
committerEric W. Biederman <ebiederm@xmission.com>
Wed, 20 Oct 2021 15:34:20 +0000 (10:34 -0500)
Setting cred->ucounts in cred_alloc_blank does not make sense.  The
uid and user_ns are deliberately not set in cred_alloc_blank but
instead the setting is delayed until key_change_session_keyring.

So move dealing with ucounts into key_change_session_keyring as well.

Unfortunately that movement of get_ucounts adds a new failure mode to
key_change_session_keyring.  I do not see anything stopping the parent
process from calling setuid and changing the relevant part of it's
cred while keyctl_session_to_parent is running making it fundamentally
necessary to call get_ucounts in key_change_session_keyring.  Which
means that the new failure mode cannot be avoided.

A failure of key_change_session_keyring results in a single threaded
parent keeping it's existing credentials.  Which results in the parent
process not being able to access the session keyring and whichever
keys are in the new keyring.

Further get_ucounts is only expected to fail if the number of bits in
the refernece count for the structure is too few.

Since the code has no other way to report the failure of get_ucounts
and because such failures are not expected to be common add a WARN_ONCE
to report this problem to userspace.

Between the WARN_ONCE and the parent process not having access to
the keys in the new session keyring I expect any failure of get_ucounts
will be noticed and reported and we can find another way to handle this
condition.  (Possibly by just making ucounts->count an atomic_long_t).

Cc: stable@vger.kernel.org
Fixes: 905ae01c4ae2 ("Add a reference to ucounts for each cred")
Link: https://lkml.kernel.org/r/7k0ias0uf.fsf_-_@disp2133
Tested-by: Yu Zhao <yuzhao@google.com>
Reviewed-by: Alexey Gladkov <legion@kernel.org>
Signed-off-by: "Eric W. Biederman" <ebiederm@xmission.com>
kernel/cred.c
security/keys/process_keys.c

index 16c05dfbec4d8a97992fa284dba26715ce267204..1ae0b4948a5a820d4ea3f758c733418dcb3f286f 100644 (file)
@@ -225,8 +225,6 @@ struct cred *cred_alloc_blank(void)
 #ifdef CONFIG_DEBUG_CREDENTIALS
        new->magic = CRED_MAGIC;
 #endif
-       new->ucounts = get_ucounts(&init_ucounts);
-
        if (security_cred_alloc_blank(new, GFP_KERNEL_ACCOUNT) < 0)
                goto error;
 
index e3d79a7b6db661113eac26533d1182d8da5bf3fd..b5d5333ab3300e86862515082946552fcf4e6e1d 100644 (file)
@@ -918,6 +918,13 @@ void key_change_session_keyring(struct callback_head *twork)
                return;
        }
 
+       /* If get_ucounts fails more bits are needed in the refcount */
+       if (unlikely(!get_ucounts(old->ucounts))) {
+               WARN_ONCE(1, "In %s get_ucounts failed\n", __func__);
+               put_cred(new);
+               return;
+       }
+
        new->  uid      = old->  uid;
        new-> euid      = old-> euid;
        new-> suid      = old-> suid;
@@ -927,6 +934,7 @@ void key_change_session_keyring(struct callback_head *twork)
        new-> sgid      = old-> sgid;
        new->fsgid      = old->fsgid;
        new->user       = get_uid(old->user);
+       new->ucounts    = old->ucounts;
        new->user_ns    = get_user_ns(old->user_ns);
        new->group_info = get_group_info(old->group_info);