#include "code.h" #include "gf2x.h" #include "hqc.h" #include "parameters.h" #include "parsing.h" #include "randombytes.h" #include "shake_prng.h" #include "vector.h" #include /** * @file hqc.c * @brief Implementation of hqc.h */ /** * @brief Keygen of the HQC_PKE IND_CPA scheme * * The public key is composed of the syndrome s as well as the seed used to generate the vector h. * * The secret key is composed of the seed used to generate vectors x and y. * As a technicality, the public key is appended to the secret key in order to respect NIST API. * * @param[out] pk String containing the public key * @param[out] sk String containing the secret key */ void PQCLEAN_HQC128_CLEAN_hqc_pke_keygen(uint8_t *pk, uint8_t *sk) { seedexpander_state sk_seedexpander; seedexpander_state pk_seedexpander; uint8_t sk_seed[SEED_BYTES] = {0}; uint8_t sigma[VEC_K_SIZE_BYTES] = {0}; uint8_t pk_seed[SEED_BYTES] = {0}; uint64_t x[VEC_N_SIZE_64] = {0}; uint64_t y[VEC_N_SIZE_64] = {0}; uint64_t h[VEC_N_SIZE_64] = {0}; uint64_t s[VEC_N_SIZE_64] = {0}; // Create seed_expanders for public key and secret key randombytes(sk_seed, SEED_BYTES); randombytes(sigma, VEC_K_SIZE_BYTES); PQCLEAN_HQC128_CLEAN_seedexpander_init(&sk_seedexpander, sk_seed, SEED_BYTES); randombytes(pk_seed, SEED_BYTES); PQCLEAN_HQC128_CLEAN_seedexpander_init(&pk_seedexpander, pk_seed, SEED_BYTES); // Compute secret key PQCLEAN_HQC128_CLEAN_vect_set_random_fixed_weight(&sk_seedexpander, x, PARAM_OMEGA); PQCLEAN_HQC128_CLEAN_vect_set_random_fixed_weight(&sk_seedexpander, y, PARAM_OMEGA); // Compute public key PQCLEAN_HQC128_CLEAN_vect_set_random(&pk_seedexpander, h); PQCLEAN_HQC128_CLEAN_vect_mul(s, y, h); PQCLEAN_HQC128_CLEAN_vect_add(s, x, s, VEC_N_SIZE_64); // Parse keys to string PQCLEAN_HQC128_CLEAN_hqc_public_key_to_string(pk, pk_seed, s); PQCLEAN_HQC128_CLEAN_hqc_secret_key_to_string(sk, sk_seed, sigma, pk); PQCLEAN_HQC128_CLEAN_seedexpander_release(&pk_seedexpander); PQCLEAN_HQC128_CLEAN_seedexpander_release(&sk_seedexpander); } /** * @brief Encryption of the HQC_PKE IND_CPA scheme * * The cihertext is composed of vectors u and v. * * @param[out] u Vector u (first part of the ciphertext) * @param[out] v Vector v (second part of the ciphertext) * @param[in] m Vector representing the message to encrypt * @param[in] theta Seed used to derive randomness required for encryption * @param[in] pk String containing the public key */ void PQCLEAN_HQC128_CLEAN_hqc_pke_encrypt(uint64_t *u, uint64_t *v, uint8_t *m, uint8_t *theta, const uint8_t *pk) { seedexpander_state vec_seedexpander; uint64_t h[VEC_N_SIZE_64] = {0}; uint64_t s[VEC_N_SIZE_64] = {0}; uint64_t r1[VEC_N_SIZE_64] = {0}; uint64_t r2[VEC_N_SIZE_64] = {0}; uint64_t e[VEC_N_SIZE_64] = {0}; uint64_t tmp1[VEC_N_SIZE_64] = {0}; uint64_t tmp2[VEC_N_SIZE_64] = {0}; // Create seed_expander from theta PQCLEAN_HQC128_CLEAN_seedexpander_init(&vec_seedexpander, theta, SEED_BYTES); // Retrieve h and s from public key PQCLEAN_HQC128_CLEAN_hqc_public_key_from_string(h, s, pk); // Generate r1, r2 and e PQCLEAN_HQC128_CLEAN_vect_set_random_fixed_weight(&vec_seedexpander, r1, PARAM_OMEGA_R); PQCLEAN_HQC128_CLEAN_vect_set_random_fixed_weight(&vec_seedexpander, r2, PARAM_OMEGA_R); PQCLEAN_HQC128_CLEAN_vect_set_random_fixed_weight(&vec_seedexpander, e, PARAM_OMEGA_E); // Compute u = r1 + r2.h PQCLEAN_HQC128_CLEAN_vect_mul(u, r2, h); PQCLEAN_HQC128_CLEAN_vect_add(u, r1, u, VEC_N_SIZE_64); // Compute v = m.G by encoding the message PQCLEAN_HQC128_CLEAN_code_encode(v, m); PQCLEAN_HQC128_CLEAN_vect_resize(tmp1, PARAM_N, v, PARAM_N1N2); // Compute v = m.G + s.r2 + e PQCLEAN_HQC128_CLEAN_vect_mul(tmp2, r2, s); PQCLEAN_HQC128_CLEAN_vect_add(tmp2, e, tmp2, VEC_N_SIZE_64); PQCLEAN_HQC128_CLEAN_vect_add(tmp2, tmp1, tmp2, VEC_N_SIZE_64); PQCLEAN_HQC128_CLEAN_vect_resize(v, PARAM_N1N2, tmp2, PARAM_N); PQCLEAN_HQC128_CLEAN_seedexpander_release(&vec_seedexpander); } /** * @brief Decryption of the HQC_PKE IND_CPA scheme * * @param[out] m Vector representing the decrypted message * @param[in] u Vector u (first part of the ciphertext) * @param[in] v Vector v (second part of the ciphertext) * @param[in] sk String containing the secret key * @returns 0 */ uint8_t PQCLEAN_HQC128_CLEAN_hqc_pke_decrypt(uint8_t *m, uint8_t *sigma, const uint64_t *u, const uint64_t *v, const uint8_t *sk) { uint64_t x[VEC_N_SIZE_64] = {0}; uint64_t y[VEC_N_SIZE_64] = {0}; uint8_t pk[PUBLIC_KEY_BYTES] = {0}; uint64_t tmp1[VEC_N_SIZE_64] = {0}; uint64_t tmp2[VEC_N_SIZE_64] = {0}; // Retrieve x, y, pk from secret key PQCLEAN_HQC128_CLEAN_hqc_secret_key_from_string(x, y, sigma, pk, sk); // Compute v - u.y PQCLEAN_HQC128_CLEAN_vect_resize(tmp1, PARAM_N, v, PARAM_N1N2); PQCLEAN_HQC128_CLEAN_vect_mul(tmp2, y, u); PQCLEAN_HQC128_CLEAN_vect_add(tmp2, tmp1, tmp2, VEC_N_SIZE_64); // Compute m by decoding v - u.y PQCLEAN_HQC128_CLEAN_code_decode(m, tmp2); return 0; }