cred: Do not default to init_cred in prepare_kernel_cred()
authorKees Cook <keescook@chromium.org>
Wed, 26 Oct 2022 23:31:11 +0000 (16:31 -0700)
committerKees Cook <keescook@chromium.org>
Tue, 1 Nov 2022 17:04:52 +0000 (10:04 -0700)
A common exploit pattern for ROP attacks is to abuse prepare_kernel_cred()
in order to construct escalated privileges[1]. Instead of providing a
short-hand argument (NULL) to the "daemon" argument to indicate using
init_cred as the base cred, require that "daemon" is always set to
an actual task. Replace all existing callers that were passing NULL
with &init_task.

Future attacks will need to have sufficiently powerful read/write
primitives to have found an appropriately privileged task and written it
to the ROP stack as an argument to succeed, which is similarly difficult
to the prior effort needed to escalate privileges before struct cred
existed: locate the current cred and overwrite the uid member.

This has the added benefit of meaning that prepare_kernel_cred() can no
longer exceed the privileges of the init task, which may have changed from
the original init_cred (e.g. dropping capabilities from the bounding set).

[1] https://google.com/search?q=commit_creds(prepare_kernel_cred(0))

Cc: "Eric W. Biederman" <ebiederm@xmission.com>
Cc: David Howells <dhowells@redhat.com>
Cc: "Rafael J. Wysocki" <rafael@kernel.org>
Cc: Steve French <sfrench@samba.org>
Cc: Ronnie Sahlberg <lsahlber@redhat.com>
Cc: Shyam Prasad N <sprasad@microsoft.com>
Cc: Tom Talpey <tom@talpey.com>
Cc: Namjae Jeon <linkinjeon@kernel.org>
Cc: Trond Myklebust <trond.myklebust@hammerspace.com>
Cc: Anna Schumaker <anna@kernel.org>
Cc: Chuck Lever <chuck.lever@oracle.com>
Cc: Jeff Layton <jlayton@kernel.org>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Eric Dumazet <edumazet@google.com>
Cc: Jakub Kicinski <kuba@kernel.org>
Cc: Paolo Abeni <pabeni@redhat.com>
Cc: "Michal Koutný" <mkoutny@suse.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: linux-cifs@vger.kernel.org
Cc: samba-technical@lists.samba.org
Cc: linux-nfs@vger.kernel.org
Cc: netdev@vger.kernel.org
Signed-off-by: Kees Cook <keescook@chromium.org>
Acked-by: Luis Chamberlain <mcgrof@kernel.org>
Reviewed-by: Sergey Senozhatsky <senozhatsky@chromium.org>
Acked-by: Russ Weight <russell.h.weight@intel.com>
Acked-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Acked-by: Paulo Alcantara (SUSE) <pc@cjr.nz>
Link: https://lore.kernel.org/r/20221026232943.never.775-kees@kernel.org
drivers/base/firmware_loader/main.c
fs/cifs/cifs_spnego.c
fs/cifs/cifsacl.c
fs/ksmbd/smb_common.c
fs/nfs/flexfilelayout/flexfilelayout.c
fs/nfs/nfs4idmap.c
fs/nfsd/nfs4callback.c
kernel/cred.c
net/dns_resolver/dns_key.c

index 7c3590fd97c28d8977f2be0c6789264ff619c07a..017c4cdb219eb115e0bc5ce93269c24ff2ff4a70 100644 (file)
@@ -821,7 +821,7 @@ _request_firmware(const struct firmware **firmware_p, const char *name,
         * called by a driver when serving an unrelated request from userland, we use
         * the kernel credentials to read the file.
         */
-       kern_cred = prepare_kernel_cred(NULL);
+       kern_cred = prepare_kernel_cred(&init_task);
        if (!kern_cred) {
                ret = -ENOMEM;
                goto out;
index 342717bf1dc28dc885a44815b460726961fa3d16..6f3285f1dfee58390d157dc8223d269220a730e1 100644 (file)
@@ -189,7 +189,7 @@ init_cifs_spnego(void)
         * spnego upcalls.
         */
 
-       cred = prepare_kernel_cred(NULL);
+       cred = prepare_kernel_cred(&init_task);
        if (!cred)
                return -ENOMEM;
 
index fa480d62f31375df31d2c925643fad7903779874..574de2b225ae6ab6532491a04d0ca7ddd986895d 100644 (file)
@@ -465,7 +465,7 @@ init_cifs_idmap(void)
         * this is used to prevent malicious redirections from being installed
         * with add_key().
         */
-       cred = prepare_kernel_cred(NULL);
+       cred = prepare_kernel_cred(&init_task);
        if (!cred)
                return -ENOMEM;
 
index d96da872d70a1701b2e99b962f82ac6b83fd6351..2a4fbbd55b91fc763119abbbbf54726c80be02a9 100644 (file)
@@ -623,7 +623,7 @@ int ksmbd_override_fsids(struct ksmbd_work *work)
        if (share->force_gid != KSMBD_SHARE_INVALID_GID)
                gid = share->force_gid;
 
-       cred = prepare_kernel_cred(NULL);
+       cred = prepare_kernel_cred(&init_task);
        if (!cred)
                return -ENOMEM;
 
index 1ec79ccf89ad241d2d8ffca51550143eb81464ee..7deb3cd76abe4b623c7d79554d7c9d7506491d1b 100644 (file)
@@ -493,10 +493,10 @@ ff_layout_alloc_lseg(struct pnfs_layout_hdr *lh,
                gid = make_kgid(&init_user_ns, id);
 
                if (gfp_flags & __GFP_FS)
-                       kcred = prepare_kernel_cred(NULL);
+                       kcred = prepare_kernel_cred(&init_task);
                else {
                        unsigned int nofs_flags = memalloc_nofs_save();
-                       kcred = prepare_kernel_cred(NULL);
+                       kcred = prepare_kernel_cred(&init_task);
                        memalloc_nofs_restore(nofs_flags);
                }
                rc = -ENOMEM;
index e3fdd2f45b01f581d9ad165aea047e8d5a2fbf2b..25a7c771cfd89f3e6d494f26a78212d3d619c135 100644 (file)
@@ -203,7 +203,7 @@ int nfs_idmap_init(void)
        printk(KERN_NOTICE "NFS: Registering the %s key type\n",
                key_type_id_resolver.name);
 
-       cred = prepare_kernel_cred(NULL);
+       cred = prepare_kernel_cred(&init_task);
        if (!cred)
                return -ENOMEM;
 
index f0e69edf5f0f1ccabe2a703aca39e0d048bf8ba6..4a9e8d17e56aab3f14113fc2c1130983d71c5455 100644 (file)
@@ -870,7 +870,7 @@ static const struct cred *get_backchannel_cred(struct nfs4_client *clp, struct r
        } else {
                struct cred *kcred;
 
-               kcred = prepare_kernel_cred(NULL);
+               kcred = prepare_kernel_cred(&init_task);
                if (!kcred)
                        return NULL;
 
index e10c15f51c1feaf258b6c837fe812f329afca64b..811ad654abd18c4da2f9a1ef79dd7e404b9cd69b 100644 (file)
@@ -701,9 +701,9 @@ void __init cred_init(void)
  * override a task's own credentials so that work can be done on behalf of that
  * task that requires a different subjective context.
  *
- * @daemon is used to provide a base for the security record, but can be NULL.
- * If @daemon is supplied, then the security data will be derived from that;
- * otherwise they'll be set to 0 and no groups, full capabilities and no keys.
+ * @daemon is used to provide a base cred, with the security data derived from
+ * that; if this is "&init_task", they'll be set to 0, no groups, full
+ * capabilities, and no keys.
  *
  * The caller may change these controls afterwards if desired.
  *
@@ -714,17 +714,16 @@ struct cred *prepare_kernel_cred(struct task_struct *daemon)
        const struct cred *old;
        struct cred *new;
 
+       if (WARN_ON_ONCE(!daemon))
+               return NULL;
+
        new = kmem_cache_alloc(cred_jar, GFP_KERNEL);
        if (!new)
                return NULL;
 
        kdebug("prepare_kernel_cred() alloc %p", new);
 
-       if (daemon)
-               old = get_task_cred(daemon);
-       else
-               old = get_cred(&init_cred);
-
+       old = get_task_cred(daemon);
        validate_creds(old);
 
        *new = *old;
index 3aced951d5ab8b589f370edab51e3aba54f2174b..01e54b46ae0b9786730a79401cb58762896d0b50 100644 (file)
@@ -337,7 +337,7 @@ static int __init init_dns_resolver(void)
         * this is used to prevent malicious redirections from being installed
         * with add_key().
         */
-       cred = prepare_kernel_cred(NULL);
+       cred = prepare_kernel_cred(&init_task);
        if (!cred)
                return -ENOMEM;