Headline
CVE-2021-46880
x509/x509_verify.c in LibreSSL before 3.4.2, and OpenBSD before 7.0 errata 006, allows authentication bypass because an error for an unverified certificate chain is sometimes discarded.
untrusted comment: verify with openbsd-70-base.pub RWR3KL+gSr4QZ9IRG1dRMntMl/Gcj1+4+rRDN2QOAbtBsZiMJ4/eSTojHLY1LOTH9kAoJYaQ0NXK27XjAyLVdBOudgp3RY+e3QI= OpenBSD 7.0 errata 006, November 26, 2021: In some situations the X.509 verifier would discard an error on an unverified certificate chain, resulting in an authentication bypass. Apply by doing: signify -Vep /etc/signify/openbsd-70-base.pub -x 006_x509.patch.sig \ -m - | (cd /usr/src && patch -p0) And then rebuild and install libcrypto and unwind: cd /usr/src/lib/libcrypto make obj make includes make make install cd /usr/src/sbin/unwind make obj make make install Index: lib/libcrypto/x509/x509_internal.h =================================================================== RCS file: /cvs/src/lib/libcrypto/x509/x509_internal.h,v retrieving revision 1.12 diff -u -p -r1.12 x509_internal.h — lib/libcrypto/x509/x509_internal.h 3 Sep 2021 08:58:53 -0000 1.12 +++ lib/libcrypto/x509/x509_internal.h 24 Nov 2021 04:51:24 -0000 @@ -90,7 +90,7 @@ int x509_vfy_check_revocation(X509_STORE int x509_vfy_check_policy(X509_STORE_CTX *ctx); int x509_vfy_check_trust(X509_STORE_CTX *ctx); int x509_vfy_check_chain_extensions(X509_STORE_CTX *ctx); -int x509_vfy_callback_indicate_success(X509_STORE_CTX *ctx); +int x509_vfy_callback_indicate_completion(X509_STORE_CTX *ctx); void x509v3_cache_extensions(X509 *x); X509 *x509_vfy_lookup_cert_match(X509_STORE_CTX *ctx, X509 *x); Index: lib/libcrypto/x509/x509_verify.c =================================================================== RCS file: /cvs/src/lib/libcrypto/x509/x509_verify.c,v retrieving revision 1.49 diff -u -p -r1.49 x509_verify.c — lib/libcrypto/x509/x509_verify.c 9 Sep 2021 15:09:43 -0000 1.49 +++ lib/libcrypto/x509/x509_verify.c 24 Nov 2021 04:51:24 -0000 @@ -1171,62 +1171,99 @@ x509_verify(struct x509_verify_ctx *ctx, x509_verify_chain_free(current_chain); /* - * Bring back the failure case we wanted to the xsc if - * we saved one. + * Do the new verifier style return, where we don’t have an xsc + * that allows a crazy callback to turn invalid things into valid. */ - if (!x509_verify_ctx_restore_xsc_error(ctx)) - goto err; + if (ctx->xsc == NULL) { + /* + * Safety net: + * We could not find a validated chain, and for some reason do not + * have an error set. + */ + if (ctx->chains_count == 0 && ctx->error == X509_V_OK) + ctx->error = X509_V_ERR_UNSPECIFIED; + + /* + * If we are not using an xsc, and have no possibility for the + * crazy OpenSSL callback API changing the results of + * validation steps (because the callback can make validation + * proceed in the presence of invalid certs), any chains we + * have here are correctly built and verified. + */ + if (ctx->chains_count > 0) + ctx->error = X509_V_OK; + + return ctx->chains_count; + } /* - * Safety net: - * We could not find a validated chain, and for some reason do not - * have an error set. + * Otherwise we are doing compatibility with an xsc, which means that we + * will have one chain, which might actually be a bogus chain because + * the callback told us to ignore errors and proceed to build an invalid + * chain. Possible return values from this include returning 1 with an + * invalid chain and a value of xsc->error != X509_V_OK (It’s tradition + * that makes it ok). */ - if (ctx->chains_count == 0 && ctx->error == X509_V_OK) { - ctx->error = X509_V_ERR_UNSPECIFIED; - if (ctx->xsc != NULL && ctx->xsc->error != X509_V_OK) - ctx->error = ctx->xsc->error; - } - /* Clear whatever errors happened if we have any validated chain */ - if (ctx->chains_count > 0) - ctx->error = X509_V_OK; + if (ctx->chains_count > 0) { + /* + * The chain we have using an xsc might not be a verified chain + * if the callback perverted things while we built it to ignore + * failures and proceed with chain building. We put this chain + * and the error associated with it on the xsc. + */ + if (!x509_verify_ctx_set_xsc_chain(ctx, ctx->chains[0], 1, 1)) + goto err; - if (ctx->xsc != NULL) { - ctx->xsc->error = ctx->error; - if (ctx->chains_count > 0) { - /* Take the first chain we found. */ - if (!x509_verify_ctx_set_xsc_chain(ctx, ctx->chains[0], - 1, 1)) - goto err; - ctx->xsc->error = X509_V_OK; - /* - * Call the callback indicating success up our already - * verified chain. The callback could still tell us to - * fail. - */ - if(!x509_vfy_callback_indicate_success(ctx->xsc)) { - /* The callback can change the error code */ - ctx->error = ctx->xsc->error; - goto err; - } - } else { - /* - * We had a failure, indicate the failure, but - * allow the callback to override at depth 0 - */ - if (ctx->xsc->verify_cb(0, ctx->xsc)) { - ctx->xsc->error = X509_V_OK; - return 1; - } + /* + * Call the callback for completion up our built + * chain. The callback could still tell us to + * fail. Since this chain might exist as the result of + * callback doing perversions, we could still return + * “success” with something other than X509_V_OK set + * as the error. + */ + if (!x509_vfy_callback_indicate_completion(ctx->xsc)) + goto err; + } else { + /* + * We did not find a chain. Bring back the failure + * case we wanted to the xsc if we saved one. If we + * did not we should have just the leaf on the xsc. + */ + if (!x509_verify_ctx_restore_xsc_error(ctx)) + goto err; + + /* + * Safety net, ensure we have an error set in the + * failing case. + */ + if (ctx->xsc->error == X509_V_OK) { + if (ctx->error == X509_V_OK) + ctx->error = X509_V_ERR_UNSPECIFIED; + ctx->xsc->error = ctx->error; } + + /* + * Let the callback override the return value + * at depth 0 if it chooses to + */ + return ctx->xsc->verify_cb(0, ctx->xsc); } - return (ctx->chains_count); + + /* We only ever find one chain in compat mode with an xsc. */ + return 1; err: if (ctx->error == X509_V_OK) ctx->error = X509_V_ERR_UNSPECIFIED; - if (ctx->xsc != NULL) - ctx->xsc->error = ctx->error; + + if (ctx->xsc != NULL) { + if (ctx->xsc->error == X509_V_OK) + ctx->xsc->error = X509_V_ERR_UNSPECIFIED; + ctx->error = ctx->xsc->error; + } + return 0; } + Index: lib/libcrypto/x509/x509_vfy.c =================================================================== RCS file: /cvs/src/lib/libcrypto/x509/x509_vfy.c,v retrieving revision 1.89 diff -u -p -r1.89 x509_vfy.c — lib/libcrypto/x509/x509_vfy.c 3 Sep 2021 08:58:53 -0000 1.89 +++ lib/libcrypto/x509/x509_vfy.c 24 Nov 2021 04:51:24 -0000 @@ -1960,8 +1960,12 @@ internal_verify(X509_STORE_CTX *ctx) return x509_vfy_internal_verify(ctx, 0); } +/* + * Internal verify, but with a chain where the verification + * math has already been performed. + */ int -x509_vfy_callback_indicate_success(X509_STORE_CTX *ctx) +x509_vfy_callback_indicate_completion(X509_STORE_CTX *ctx) { return x509_vfy_internal_verify(ctx, 1); }