From df9c5647ceab8e274d0fd6703eefe90ef0953c22 Mon Sep 17 00:00:00 2001 From: dkostic <25055813+dkostic@users.noreply.github.com> Date: Tue, 5 Nov 2024 13:34:53 -0800 Subject: [PATCH] ML-KEM keygen Pairwise Consistency Test (#1964) This change implements the ML-KEM Pairwise Consistency Test (PCT) for key generation as required by FIPS 140-3 Implementation Guidance. This effectively triples the runtime of key generation. --- crypto/fipsmodule/ml_kem/ml_kem_ref/kem.c | 48 ++++++++++++++++++----- 1 file changed, 38 insertions(+), 10 deletions(-) diff --git a/crypto/fipsmodule/ml_kem/ml_kem_ref/kem.c b/crypto/fipsmodule/ml_kem/ml_kem_ref/kem.c index d0c357675c..02faae7b0f 100644 --- a/crypto/fipsmodule/ml_kem/ml_kem_ref/kem.c +++ b/crypto/fipsmodule/ml_kem/ml_kem_ref/kem.c @@ -1,14 +1,33 @@ #include #include #include -#include "params.h" -#include "kem.h" -#include "indcpa.h" -#include "verify.h" -#include "reduce.h" -#include "symmetric.h" +#include "./params.h" +#include "./kem.h" +#include "./indcpa.h" +#include "./verify.h" +#include "./reduce.h" +#include "./symmetric.h" +#include "../../../internal.h" + #include "openssl/rand.h" +#if defined(AWSLC_FIPS) +// FIPS 203. Pair-wise Consistency Test (PCT) required per [FIPS 140-3 IG](https://csrc.nist.gov/csrc/media/Projects/cryptographic-module-validation-program/documents/fips%20140-3/FIPS%20140-3%20IG.pdf): +// The PCT consists of applying the encapsulation key to encapsulate a shared +// secret leading to ciphertext, and then applying decapsulation key to +// retrieve the same shared secret. Returns 0 if the PCT passes, 1 otherwise. +static int keygen_pct(ml_kem_params *params, const uint8_t *ek, const uint8_t *dk) { + uint8_t ct[KYBER_CIPHERTEXTBYTES_MAX]; + uint8_t ss_enc[KYBER_SSBYTES]; + uint8_t ss_dec[KYBER_SSBYTES]; + + crypto_kem_enc(params, ct, ss_enc, ek); + crypto_kem_dec(params, ss_dec, ct, dk); + + return verify(ss_enc, ss_dec, KYBER_SSBYTES); +} +#endif + /************************************************* * Name: crypto_kem_keypair_derand * @@ -22,7 +41,7 @@ * - uint8_t *coins: pointer to input randomness * (an already allocated array filled with 2*KYBER_SYMBYTES random bytes) ** -* Returns 0 (success) +* Returns 0 on success, aborts on failure. **************************************************/ int crypto_kem_keypair_derand(ml_kem_params *params, uint8_t *pk, @@ -34,6 +53,13 @@ int crypto_kem_keypair_derand(ml_kem_params *params, hash_h(sk+params->secret_key_bytes-2*KYBER_SYMBYTES, pk, params->public_key_bytes); /* Value z for pseudo-random output on reject */ memcpy(sk+params->secret_key_bytes-KYBER_SYMBYTES, coins+KYBER_SYMBYTES, KYBER_SYMBYTES); + +#if defined(AWSLC_FIPS) + // Abort in case of PCT failure. + if (keygen_pct(params, pk, sk)) { + BORINGSSL_FIPS_abort(); + } +#endif return 0; } @@ -48,7 +74,7 @@ int crypto_kem_keypair_derand(ml_kem_params *params, * - uint8_t *sk: pointer to output private key * (an already allocated array of KYBER_SECRETKEYBYTES bytes) * -* Returns 0 (success) +* Returns 0 on success, aborts on failure. **************************************************/ int crypto_kem_keypair(ml_kem_params *params, uint8_t *pk, @@ -56,11 +82,13 @@ int crypto_kem_keypair(ml_kem_params *params, { uint8_t coins[2*KYBER_SYMBYTES]; RAND_bytes(coins, 2*KYBER_SYMBYTES); - crypto_kem_keypair_derand(params, pk, sk, coins); + int res = crypto_kem_keypair_derand(params, pk, sk, coins); + assert(res == 0); // FIPS 203. Section 3.3 Destruction of intermediate values. OPENSSL_cleanse(coins, sizeof(coins)); - return 0; + + return res; } // Converts a centered representative |in| which is an integer in