#include "owcpa.h" #include "poly.h" #include "sample.h" static int owcpa_check_ciphertext(const unsigned char *ciphertext) { /* A ciphertext is log2(q)*(n-1) bits packed into bytes. */ /* Check that any unused bits of the final byte are zero. */ uint16_t t = 0; t = ciphertext[NTRU_CIPHERTEXTBYTES - 1]; t &= 0xff << (8 - (7 & (NTRU_LOGQ * NTRU_PACK_DEG))); /* We have 0 <= t < 256 */ /* Return 0 on success (t=0), 1 on failure */ return (int) (1 & ((~t + 1) >> 15)); } static int owcpa_check_r(const poly *r) { /* A valid r has coefficients in {0,1,q-1} and has r[N-1] = 0 */ /* Note: We may assume that 0 <= r[i] <= q-1 for all i */ int i; uint32_t t = 0; uint16_t c; for (i = 0; i < NTRU_N - 1; i++) { c = r->coeffs[i]; t |= (c + 1) & (NTRU_Q - 4); /* 0 iff c is in {-1,0,1,2} */ t |= (c + 2) & 4; /* 1 if c = 2, 0 if c is in {-1,0,1} */ } t |= r->coeffs[NTRU_N - 1]; /* Coefficient n-1 must be zero */ /* We have 0 <= t < 2^16. */ /* Return 0 on success (t=0), 1 on failure */ return (int) (1 & ((~t + 1) >> 31)); } static int owcpa_check_m(const poly *m) { /* Check that m is in message space, i.e. */ /* (1) |{i : m[i] = 1}| = |{i : m[i] = 2}|, and */ /* (2) |{i : m[i] != 0}| = NTRU_WEIGHT. */ /* Note: We may assume that m has coefficients in {0,1,2}. */ int i; uint32_t t = 0; uint16_t ps = 0; uint16_t ms = 0; for (i = 0; i < NTRU_N; i++) { ps += m->coeffs[i] & 1; ms += m->coeffs[i] & 2; } t |= ps ^ (ms >> 1); /* 0 if (1) holds */ t |= ms ^ NTRU_WEIGHT; /* 0 if (1) and (2) hold */ /* We have 0 <= t < 2^16. */ /* Return 0 on success (t=0), 1 on failure */ return (int) (1 & ((~t + 1) >> 31)); } void PQCLEAN_NTRUHPS2048509_AVX2_owcpa_keypair(unsigned char *pk, unsigned char *sk, const unsigned char seed[NTRU_SAMPLE_FG_BYTES]) { int i; poly x1, x2, x3, x4, x5; poly *f = &x1, *g = &x2, *invf_mod3 = &x3; poly *gf = &x3, *invgf = &x4, *tmp = &x5; poly *invh = &x3, *h = &x3; PQCLEAN_NTRUHPS2048509_AVX2_sample_fg(f, g, seed); PQCLEAN_NTRUHPS2048509_AVX2_poly_S3_inv(invf_mod3, f); PQCLEAN_NTRUHPS2048509_AVX2_poly_S3_tobytes(sk, f); PQCLEAN_NTRUHPS2048509_AVX2_poly_S3_tobytes(sk + NTRU_PACK_TRINARY_BYTES, invf_mod3); /* Lift coeffs of f and g from Z_p to Z_q */ PQCLEAN_NTRUHPS2048509_AVX2_poly_Z3_to_Zq(f); PQCLEAN_NTRUHPS2048509_AVX2_poly_Z3_to_Zq(g); /* g = 3*g */ for (i = 0; i < NTRU_N; i++) { g->coeffs[i] = 3 * g->coeffs[i]; } PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_mul(gf, g, f); PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_inv(invgf, gf); PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_mul(tmp, invgf, f); PQCLEAN_NTRUHPS2048509_AVX2_poly_Sq_mul(invh, tmp, f); PQCLEAN_NTRUHPS2048509_AVX2_poly_Sq_tobytes(sk + 2 * NTRU_PACK_TRINARY_BYTES, invh); PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_mul(tmp, invgf, g); PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_mul(h, tmp, g); PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_sum_zero_tobytes(pk, h); } void PQCLEAN_NTRUHPS2048509_AVX2_owcpa_enc(unsigned char *c, const poly *r, const poly *m, const unsigned char *pk) { int i; poly x1, x2; poly *h = &x1, *liftm = &x1; poly *ct = &x2; PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_sum_zero_frombytes(h, pk); PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_mul(ct, r, h); PQCLEAN_NTRUHPS2048509_AVX2_poly_lift(liftm, m); for (i = 0; i < NTRU_N; i++) { ct->coeffs[i] = ct->coeffs[i] + liftm->coeffs[i]; } PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_sum_zero_tobytes(c, ct); } int PQCLEAN_NTRUHPS2048509_AVX2_owcpa_dec(unsigned char *rm, const unsigned char *ciphertext, const unsigned char *secretkey) { int i; int fail; poly x1, x2, x3, x4; poly *c = &x1, *f = &x2, *cf = &x3; poly *mf = &x2, *finv3 = &x3, *m = &x4; poly *liftm = &x2, *invh = &x3, *r = &x4; poly *b = &x1; PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_sum_zero_frombytes(c, ciphertext); PQCLEAN_NTRUHPS2048509_AVX2_poly_S3_frombytes(f, secretkey); PQCLEAN_NTRUHPS2048509_AVX2_poly_Z3_to_Zq(f); PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_mul(cf, c, f); PQCLEAN_NTRUHPS2048509_AVX2_poly_Rq_to_S3(mf, cf); PQCLEAN_NTRUHPS2048509_AVX2_poly_S3_frombytes(finv3, secretkey + NTRU_PACK_TRINARY_BYTES); PQCLEAN_NTRUHPS2048509_AVX2_poly_S3_mul(m, mf, finv3); PQCLEAN_NTRUHPS2048509_AVX2_poly_S3_tobytes(rm + NTRU_PACK_TRINARY_BYTES, m); fail = 0; /* Check that the unused bits of the last byte of the ciphertext are zero */ fail |= owcpa_check_ciphertext(ciphertext); /* For the IND-CCA2 KEM we must ensure that c = Enc(h, (r,m)). */ /* We can avoid re-computing r*h + Lift(m) as long as we check that */ /* r (defined as b/h mod (q, Phi_n)) and m are in the message space. */ /* (m can take any value in S3 in NTRU_HRSS) */ fail |= owcpa_check_m(m); /* b = c - Lift(m) mod (q, x^n - 1) */ PQCLEAN_NTRUHPS2048509_AVX2_poly_lift(liftm, m); for (i = 0; i < NTRU_N; i++) { b->coeffs[i] = c->coeffs[i] - liftm->coeffs[i]; } /* r = b / h mod (q, Phi_n) */ PQCLEAN_NTRUHPS2048509_AVX2_poly_Sq_frombytes(invh, secretkey + 2 * NTRU_PACK_TRINARY_BYTES); PQCLEAN_NTRUHPS2048509_AVX2_poly_Sq_mul(r, b, invh); /* NOTE: Our definition of r as b/h mod (q, Phi_n) follows Figure 4 of */ /* [Sch18] https://eprint.iacr.org/2018/1174/20181203:032458. */ /* This differs from Figure 10 of Saito--Xagawa--Yamakawa */ /* [SXY17] https://eprint.iacr.org/2017/1005/20180516:055500 */ /* where r gets a final reduction modulo p. */ /* We need this change to use Proposition 1 of [Sch18]. */ /* Proposition 1 of [Sch18] shows that re-encryption with (r,m) yields c. */ /* if and only if fail==0 after the following call to owcpa_check_r */ /* The procedure given in Fig. 8 of [Sch18] can be skipped because we have */ /* c(1) = 0 due to the use of poly_Rq_sum_zero_{to,from}bytes. */ fail |= owcpa_check_r(r); PQCLEAN_NTRUHPS2048509_AVX2_poly_trinary_Zq_to_Z3(r); PQCLEAN_NTRUHPS2048509_AVX2_poly_S3_tobytes(rm, r); return fail; }