Received: by 2002:a5d:925a:0:0:0:0:0 with SMTP id e26csp1497401iol; Fri, 10 Jun 2022 08:39:58 -0700 (PDT) X-Google-Smtp-Source: ABdhPJyyDuy0vNFJea4i1davbzKImo4mrVXNE80pvsV9Csi8pd0B1byToNgd/MFbrN9eeRgTi9Ep X-Received: by 2002:a05:6402:390b:b0:42a:acb3:bb5 with SMTP id fe11-20020a056402390b00b0042aacb30bb5mr50648047edb.236.1654875598157; Fri, 10 Jun 2022 08:39:58 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1654875598; cv=none; d=google.com; s=arc-20160816; b=b9hFvWpdpmuvy1KrIeYXwgHIh5wX2uQlLvBS/T8F7KfF7F5hGyHbnrGP7sDBJP2nTV wULhvXL6yCO62cpxHdlFlgQBWXT56XHq7zkR1Peopa0dT1f83vA5TaFjyxYCN51k/+cc EYce2y7epiA0iBDFM17GLmFx/9RgSce9t+33POgKbWp44SGQP4NI8KnslXyKIqnZjBA6 yXIuuTAEiqiiQAtr9KsB1hS3jIdsKMgoubhLINWbUOAono3IRSG3gSIjYmmh21lTh5ay J2xGoZPoeW/ls4X40XlJM2b5+dSISIo0XmXK6Vgor2CiuUR26YGl0a6Gq9PDlgo1X8eo vo+A== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:content-language :in-reply-to:mime-version:user-agent:date:message-id:from:references :cc:to:subject; bh=MjMsKsrcpsn4Y0i64GeoJPaxeUQN9L3+Obw+EJfQe10=; b=Jvb+jWCgyhCp4ZqnGr0fSeMl2dBgw7RgGveuhz749f8G3epYTggewamUDUW2Z1mjV7 O8PpaWp89dHBz3K0ZP38pjz+qMBUcu6OC7M6dECLTdPYa2CgWbrbb7dVdIjEFKzhPCa2 0fj30cThicBq6JhsYMHak8lcDuSTe/ubJcfiv+YjGZgTAi1ob3p9H6dr6+UH14P1ZEfc bfTRetW1k7zH2zOZHvUkumEoP3o0in3pItGxNIXlSuKnWsuVmYmkGUGLARM76nSiW9Qb +nmfk1ELOZgHdrrjD6XtCXS1J0jjt+oLwkp2E6MXwHHIno1S28cBYWoGDzGS0RCDiq0s SplQ== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id rv7-20020a17090710c700b006f39e0cbe8esi7381843ejb.227.2022.06.10.08.39.27; Fri, 10 Jun 2022 08:39:58 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S239371AbiFJOty (ORCPT + 99 others); Fri, 10 Jun 2022 10:49:54 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:56420 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1344919AbiFJOtU (ORCPT ); Fri, 10 Jun 2022 10:49:20 -0400 Received: from www62.your-server.de (www62.your-server.de [213.133.104.62]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id C2193DE93; Fri, 10 Jun 2022 07:48:48 -0700 (PDT) Received: from sslproxy05.your-server.de ([78.46.172.2]) by www62.your-server.de with esmtpsa (TLSv1.3:TLS_AES_256_GCM_SHA384:256) (Exim 4.92.3) (envelope-from ) id 1nzfwF-0000ZJ-IP; Fri, 10 Jun 2022 16:48:39 +0200 Received: from [85.1.206.226] (helo=linux.home) by sslproxy05.your-server.de with esmtpsa (TLSv1.3:TLS_AES_256_GCM_SHA384:256) (Exim 4.92) (envelope-from ) id 1nzfwF-000IXa-9T; Fri, 10 Jun 2022 16:48:39 +0200 Subject: Re: [PATCH v3 1/2] bpf: Add bpf_verify_signature() helper To: Roberto Sassu , ast@kernel.org, andrii@kernel.org, kpsingh@kernel.org Cc: bpf@vger.kernel.org, netdev@vger.kernel.org, linux-kselftest@vger.kernel.org, linux-kernel@vger.kernel.org, kernel test robot , john.fastabend@gmail.com References: <20220610135916.1285509-1-roberto.sassu@huawei.com> <20220610135916.1285509-2-roberto.sassu@huawei.com> From: Daniel Borkmann Message-ID: Date: Fri, 10 Jun 2022 16:48:38 +0200 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:60.0) Gecko/20100101 Thunderbird/60.7.2 MIME-Version: 1.0 In-Reply-To: <20220610135916.1285509-2-roberto.sassu@huawei.com> Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-US Content-Transfer-Encoding: 7bit X-Authenticated-Sender: daniel@iogearbox.net X-Virus-Scanned: Clear (ClamAV 0.103.6/26568/Fri Jun 10 10:06:23 2022) X-Spam-Status: No, score=-3.1 required=5.0 tests=BAYES_00,NICE_REPLY_A, RCVD_IN_DNSWL_NONE,SPF_HELO_NONE,SPF_PASS,T_SCC_BODY_TEXT_LINE autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 6/10/22 3:59 PM, Roberto Sassu wrote: > Add the bpf_verify_signature() helper, to give eBPF security modules the > ability to check the validity of a signature against supplied data, by > using system-provided keys as trust anchor. > > The new helper makes it possible to enforce mandatory policies, as eBPF > programs might be allowed to make security decisions only based on data > sources the system administrator approves. > > The caller should specify the identifier of the keyring containing the keys > for signature verification: 0 for the primary keyring (immutable keyring of > system keys); 1 for both the primary and secondary keyring (where keys can > be added only if they are vouched for by existing keys in those keyrings); > 2 for the platform keyring (primarily used by the integrity subsystem to > verify a kexec'ed kerned image and, possibly, the initramfs signature); > 0xffff for the session keyring (for testing purposes). > > The caller should also specify the type of signature. Currently only PKCS#7 > is supported. > > Since the maximum number of parameters of an eBPF helper is 5, the keyring > and signature types share one (keyring ID: low 16 bits, signature type: > high 16 bits). > > Signed-off-by: Roberto Sassu > Reported-by: kernel test robot (cast warning) > --- > include/uapi/linux/bpf.h | 17 +++++++++++++ > kernel/bpf/bpf_lsm.c | 46 ++++++++++++++++++++++++++++++++++ > tools/include/uapi/linux/bpf.h | 17 +++++++++++++ > 3 files changed, 80 insertions(+) > > diff --git a/include/uapi/linux/bpf.h b/include/uapi/linux/bpf.h > index f4009dbdf62d..97521857e44a 100644 > --- a/include/uapi/linux/bpf.h > +++ b/include/uapi/linux/bpf.h > @@ -5249,6 +5249,22 @@ union bpf_attr { > * Pointer to the underlying dynptr data, NULL if the dynptr is > * read-only, if the dynptr is invalid, or if the offset and length > * is out of bounds. > + * > + * long bpf_verify_signature(u8 *data, u32 datalen, u8 *sig, u32 siglen, u32 info) > + * Description > + * Verify a signature of length *siglen* against the supplied data > + * with length *datalen*. *info* contains the keyring identifier > + * (low 16 bits) and the signature type (high 16 bits). The keyring > + * identifier can have the following values (some defined in > + * verification.h): 0 for the primary keyring (immutable keyring of > + * system keys); 1 for both the primary and secondary keyring > + * (where keys can be added only if they are vouched for by > + * existing keys in those keyrings); 2 for the platform keyring > + * (primarily used by the integrity subsystem to verify a kexec'ed > + * kerned image and, possibly, the initramfs signature); 0xffff for > + * the session keyring (for testing purposes). > + * Return > + * 0 on success, a negative value on error. > */ > #define __BPF_FUNC_MAPPER(FN) \ > FN(unspec), \ > @@ -5455,6 +5471,7 @@ union bpf_attr { > FN(dynptr_read), \ > FN(dynptr_write), \ > FN(dynptr_data), \ > + FN(verify_signature), \ > /* */ > > /* integer value in 'imm' field of BPF_CALL instruction selects which helper > diff --git a/kernel/bpf/bpf_lsm.c b/kernel/bpf/bpf_lsm.c > index c1351df9f7ee..20bd850ea3ee 100644 > --- a/kernel/bpf/bpf_lsm.c > +++ b/kernel/bpf/bpf_lsm.c > @@ -16,6 +16,8 @@ > #include > #include > #include > +#include > +#include > > /* For every LSM hook that allows attachment of BPF programs, declare a nop > * function where a BPF program can be attached. > @@ -132,6 +134,46 @@ static const struct bpf_func_proto bpf_get_attach_cookie_proto = { > .arg1_type = ARG_PTR_TO_CTX, > }; > > +#ifdef CONFIG_SYSTEM_DATA_VERIFICATION > +BPF_CALL_5(bpf_verify_signature, u8 *, data, u32, datalen, u8 *, sig, > + u32, siglen, u32, info) > +{ > + unsigned long keyring_id = info & U16_MAX; > + enum pkey_id_type id_type = info >> 16; > + const struct cred *cred = current_cred(); > + struct key *keyring; > + > + if (keyring_id > (unsigned long)VERIFY_USE_PLATFORM_KEYRING && > + keyring_id != U16_MAX) > + return -EINVAL; > + > + keyring = (keyring_id == U16_MAX) ? > + cred->session_keyring : (struct key *)keyring_id; > + > + switch (id_type) { > + case PKEY_ID_PKCS7: > + return verify_pkcs7_signature(data, datalen, sig, siglen, > + keyring, > + VERIFYING_UNSPECIFIED_SIGNATURE, > + NULL, NULL); > + default: > + return -EOPNOTSUPP; Question to you & KP: > Can we keep the helper generic so that it can be extended to more types of > signatures and pass the signature type as an enum? How many different signature types do we expect, say, in the next 6mo, to land here? Just thinking out loud whether it is better to keep it simple as with the last iteration where we have a helper specific to pkcs7, and if needed in future we add others. We only have the last reg as auxillary arg where we need to squeeze all info into it now. What if for other, future signature types this won't suffice? > + } > +} > + > +static const struct bpf_func_proto bpf_verify_signature_proto = { > + .func = bpf_verify_signature, > + .gpl_only = false, > + .ret_type = RET_INTEGER, > + .arg1_type = ARG_PTR_TO_MEM, > + .arg2_type = ARG_CONST_SIZE_OR_ZERO, Can verify_pkcs7_signature() handle null/0 len for data* args? > + .arg3_type = ARG_PTR_TO_MEM, > + .arg4_type = ARG_CONST_SIZE_OR_ZERO, Ditto for sig* args? > + .arg5_type = ARG_ANYTHING, > + .allowed = bpf_ima_inode_hash_allowed, > +}; > +#endif > + > static const struct bpf_func_proto * > bpf_lsm_func_proto(enum bpf_func_id func_id, const struct bpf_prog *prog) > { > @@ -158,6 +200,10 @@ bpf_lsm_func_proto(enum bpf_func_id func_id, const struct bpf_prog *prog) > return prog->aux->sleepable ? &bpf_ima_file_hash_proto : NULL; > case BPF_FUNC_get_attach_cookie: > return bpf_prog_has_trampoline(prog) ? &bpf_get_attach_cookie_proto : NULL; > +#ifdef CONFIG_SYSTEM_DATA_VERIFICATION > + case BPF_FUNC_verify_signature: > + return prog->aux->sleepable ? &bpf_verify_signature_proto : NULL; > +#endif > default: > return tracing_prog_func_proto(func_id, prog); > }