X.509: Introduce scope-based x509_certificate allocation

Add a DEFINE_FREE() clause for x509_certificate structs and use it in
x509_cert_parse() and x509_key_preparse().  These are the only functions
where scope-based x509_certificate allocation currently makes sense.
A third user will be introduced with the forthcoming SPDM library
(Security Protocol and Data Model) for PCI device authentication.

Unlike most other DEFINE_FREE() clauses, this one checks for IS_ERR()
instead of NULL before calling x509_free_certificate() at end of scope.
That's because the "constructor" of x509_certificate structs,
x509_cert_parse(), returns a valid pointer or an ERR_PTR(), but never
NULL.

Comparing the Assembler output before/after has shown they are identical,
save for the fact that gcc-12 always generates two return paths when
__cleanup() is used, one for the success case and one for the error case.

In x509_cert_parse(), add a hint for the compiler that kzalloc() never
returns an ERR_PTR().  Otherwise the compiler adds a gratuitous IS_ERR()
check on return.  Introduce an assume() macro for this which can be
re-used elsewhere in the kernel to provide hints for the compiler.

Suggested-by: Jonathan Cameron <Jonathan.Cameron@Huawei.com>
Link: https://lore.kernel.org/all/20231003153937.000034ca@Huawei.com/
Link: https://lwn.net/Articles/934679/
Signed-off-by: Lukas Wunner <lukas@wunner.de>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
This commit is contained in:
Lukas Wunner 2024-04-07 19:57:40 +02:00 committed by Herbert Xu
parent c9ccfd5e0f
commit 5c6ca9d936
4 changed files with 30 additions and 49 deletions

View File

@ -60,24 +60,24 @@ EXPORT_SYMBOL_GPL(x509_free_certificate);
*/ */
struct x509_certificate *x509_cert_parse(const void *data, size_t datalen) struct x509_certificate *x509_cert_parse(const void *data, size_t datalen)
{ {
struct x509_certificate *cert; struct x509_certificate *cert __free(x509_free_certificate);
struct x509_parse_context *ctx; struct x509_parse_context *ctx __free(kfree) = NULL;
struct asymmetric_key_id *kid; struct asymmetric_key_id *kid;
long ret; long ret;
ret = -ENOMEM;
cert = kzalloc(sizeof(struct x509_certificate), GFP_KERNEL); cert = kzalloc(sizeof(struct x509_certificate), GFP_KERNEL);
assume(!IS_ERR(cert)); /* Avoid gratuitous IS_ERR() check on return */
if (!cert) if (!cert)
goto error_no_cert; return ERR_PTR(-ENOMEM);
cert->pub = kzalloc(sizeof(struct public_key), GFP_KERNEL); cert->pub = kzalloc(sizeof(struct public_key), GFP_KERNEL);
if (!cert->pub) if (!cert->pub)
goto error_no_ctx; return ERR_PTR(-ENOMEM);
cert->sig = kzalloc(sizeof(struct public_key_signature), GFP_KERNEL); cert->sig = kzalloc(sizeof(struct public_key_signature), GFP_KERNEL);
if (!cert->sig) if (!cert->sig)
goto error_no_ctx; return ERR_PTR(-ENOMEM);
ctx = kzalloc(sizeof(struct x509_parse_context), GFP_KERNEL); ctx = kzalloc(sizeof(struct x509_parse_context), GFP_KERNEL);
if (!ctx) if (!ctx)
goto error_no_ctx; return ERR_PTR(-ENOMEM);
ctx->cert = cert; ctx->cert = cert;
ctx->data = (unsigned long)data; ctx->data = (unsigned long)data;
@ -85,7 +85,7 @@ struct x509_certificate *x509_cert_parse(const void *data, size_t datalen)
/* Attempt to decode the certificate */ /* Attempt to decode the certificate */
ret = asn1_ber_decoder(&x509_decoder, ctx, data, datalen); ret = asn1_ber_decoder(&x509_decoder, ctx, data, datalen);
if (ret < 0) if (ret < 0)
goto error_decode; return ERR_PTR(ret);
/* Decode the AuthorityKeyIdentifier */ /* Decode the AuthorityKeyIdentifier */
if (ctx->raw_akid) { if (ctx->raw_akid) {
@ -95,20 +95,19 @@ struct x509_certificate *x509_cert_parse(const void *data, size_t datalen)
ctx->raw_akid, ctx->raw_akid_size); ctx->raw_akid, ctx->raw_akid_size);
if (ret < 0) { if (ret < 0) {
pr_warn("Couldn't decode AuthKeyIdentifier\n"); pr_warn("Couldn't decode AuthKeyIdentifier\n");
goto error_decode; return ERR_PTR(ret);
} }
} }
ret = -ENOMEM;
cert->pub->key = kmemdup(ctx->key, ctx->key_size, GFP_KERNEL); cert->pub->key = kmemdup(ctx->key, ctx->key_size, GFP_KERNEL);
if (!cert->pub->key) if (!cert->pub->key)
goto error_decode; return ERR_PTR(-ENOMEM);
cert->pub->keylen = ctx->key_size; cert->pub->keylen = ctx->key_size;
cert->pub->params = kmemdup(ctx->params, ctx->params_size, GFP_KERNEL); cert->pub->params = kmemdup(ctx->params, ctx->params_size, GFP_KERNEL);
if (!cert->pub->params) if (!cert->pub->params)
goto error_decode; return ERR_PTR(-ENOMEM);
cert->pub->paramlen = ctx->params_size; cert->pub->paramlen = ctx->params_size;
cert->pub->algo = ctx->key_algo; cert->pub->algo = ctx->key_algo;
@ -116,33 +115,23 @@ struct x509_certificate *x509_cert_parse(const void *data, size_t datalen)
/* Grab the signature bits */ /* Grab the signature bits */
ret = x509_get_sig_params(cert); ret = x509_get_sig_params(cert);
if (ret < 0) if (ret < 0)
goto error_decode; return ERR_PTR(ret);
/* Generate cert issuer + serial number key ID */ /* Generate cert issuer + serial number key ID */
kid = asymmetric_key_generate_id(cert->raw_serial, kid = asymmetric_key_generate_id(cert->raw_serial,
cert->raw_serial_size, cert->raw_serial_size,
cert->raw_issuer, cert->raw_issuer,
cert->raw_issuer_size); cert->raw_issuer_size);
if (IS_ERR(kid)) { if (IS_ERR(kid))
ret = PTR_ERR(kid); return ERR_CAST(kid);
goto error_decode;
}
cert->id = kid; cert->id = kid;
/* Detect self-signed certificates */ /* Detect self-signed certificates */
ret = x509_check_for_self_signed(cert); ret = x509_check_for_self_signed(cert);
if (ret < 0) if (ret < 0)
goto error_decode; return ERR_PTR(ret);
kfree(ctx); return_ptr(cert);
return cert;
error_decode:
kfree(ctx);
error_no_ctx:
x509_free_certificate(cert);
error_no_cert:
return ERR_PTR(ret);
} }
EXPORT_SYMBOL_GPL(x509_cert_parse); EXPORT_SYMBOL_GPL(x509_cert_parse);

View File

@ -5,6 +5,7 @@
* Written by David Howells (dhowells@redhat.com) * Written by David Howells (dhowells@redhat.com)
*/ */
#include <linux/cleanup.h>
#include <linux/time.h> #include <linux/time.h>
#include <crypto/public_key.h> #include <crypto/public_key.h>
#include <keys/asymmetric-type.h> #include <keys/asymmetric-type.h>
@ -44,6 +45,8 @@ struct x509_certificate {
* x509_cert_parser.c * x509_cert_parser.c
*/ */
extern void x509_free_certificate(struct x509_certificate *cert); extern void x509_free_certificate(struct x509_certificate *cert);
DEFINE_FREE(x509_free_certificate, struct x509_certificate *,
if (!IS_ERR(_T)) x509_free_certificate(_T))
extern struct x509_certificate *x509_cert_parse(const void *data, size_t datalen); extern struct x509_certificate *x509_cert_parse(const void *data, size_t datalen);
extern int x509_decode_time(time64_t *_t, size_t hdrlen, extern int x509_decode_time(time64_t *_t, size_t hdrlen,
unsigned char tag, unsigned char tag,

View File

@ -161,12 +161,11 @@ int x509_check_for_self_signed(struct x509_certificate *cert)
*/ */
static int x509_key_preparse(struct key_preparsed_payload *prep) static int x509_key_preparse(struct key_preparsed_payload *prep)
{ {
struct asymmetric_key_ids *kids; struct x509_certificate *cert __free(x509_free_certificate);
struct x509_certificate *cert; struct asymmetric_key_ids *kids __free(kfree) = NULL;
char *p, *desc __free(kfree) = NULL;
const char *q; const char *q;
size_t srlen, sulen; size_t srlen, sulen;
char *desc = NULL, *p;
int ret;
cert = x509_cert_parse(prep->data, prep->datalen); cert = x509_cert_parse(prep->data, prep->datalen);
if (IS_ERR(cert)) if (IS_ERR(cert))
@ -188,9 +187,8 @@ static int x509_key_preparse(struct key_preparsed_payload *prep)
} }
/* Don't permit addition of blacklisted keys */ /* Don't permit addition of blacklisted keys */
ret = -EKEYREJECTED;
if (cert->blacklisted) if (cert->blacklisted)
goto error_free_cert; return -EKEYREJECTED;
/* Propose a description */ /* Propose a description */
sulen = strlen(cert->subject); sulen = strlen(cert->subject);
@ -202,10 +200,9 @@ static int x509_key_preparse(struct key_preparsed_payload *prep)
q = cert->raw_serial; q = cert->raw_serial;
} }
ret = -ENOMEM;
desc = kmalloc(sulen + 2 + srlen * 2 + 1, GFP_KERNEL); desc = kmalloc(sulen + 2 + srlen * 2 + 1, GFP_KERNEL);
if (!desc) if (!desc)
goto error_free_cert; return -ENOMEM;
p = memcpy(desc, cert->subject, sulen); p = memcpy(desc, cert->subject, sulen);
p += sulen; p += sulen;
*p++ = ':'; *p++ = ':';
@ -215,16 +212,14 @@ static int x509_key_preparse(struct key_preparsed_payload *prep)
kids = kmalloc(sizeof(struct asymmetric_key_ids), GFP_KERNEL); kids = kmalloc(sizeof(struct asymmetric_key_ids), GFP_KERNEL);
if (!kids) if (!kids)
goto error_free_desc; return -ENOMEM;
kids->id[0] = cert->id; kids->id[0] = cert->id;
kids->id[1] = cert->skid; kids->id[1] = cert->skid;
kids->id[2] = asymmetric_key_generate_id(cert->raw_subject, kids->id[2] = asymmetric_key_generate_id(cert->raw_subject,
cert->raw_subject_size, cert->raw_subject_size,
"", 0); "", 0);
if (IS_ERR(kids->id[2])) { if (IS_ERR(kids->id[2]))
ret = PTR_ERR(kids->id[2]); return PTR_ERR(kids->id[2]);
goto error_free_kids;
}
/* We're pinning the module by being linked against it */ /* We're pinning the module by being linked against it */
__module_get(public_key_subtype.owner); __module_get(public_key_subtype.owner);
@ -242,15 +237,7 @@ static int x509_key_preparse(struct key_preparsed_payload *prep)
cert->sig = NULL; cert->sig = NULL;
desc = NULL; desc = NULL;
kids = NULL; kids = NULL;
ret = 0; return 0;
error_free_kids:
kfree(kids);
error_free_desc:
kfree(desc);
error_free_cert:
x509_free_certificate(cert);
return ret;
} }
static struct asymmetric_key_parser x509_key_parser = { static struct asymmetric_key_parser x509_key_parser = {

View File

@ -148,6 +148,8 @@ void ftrace_likely_update(struct ftrace_likely_data *f, int val,
} while (0) } while (0)
#endif #endif
#define assume(cond) do { if (!(cond)) __builtin_unreachable(); } while (0)
/* /*
* KENTRY - kernel entry point * KENTRY - kernel entry point
* This can be used to annotate symbols (functions or data) that are used * This can be used to annotate symbols (functions or data) that are used