/* $OpenBSD: sntrup4591761.c,v 1.3 2019/01/30 19:51:15 markus Exp $ */ /* * Public Domain, Authors: * - Daniel J. Bernstein * - Chitchanok Chuengsatiansup * - Tanja Lange * - Christine van Vredendaal */ #include "includes.h" #include #include "crypto_api.h" /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/int32_sort.h */ #ifndef int32_sort_h #define int32_sort_h static void int32_sort(crypto_int32 *,int); #endif /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/int32_sort.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void minmax(crypto_int32 *x,crypto_int32 *y) { crypto_uint32 xi = *x; crypto_uint32 yi = *y; crypto_uint32 xy = xi ^ yi; crypto_uint32 c = yi - xi; c ^= xy & (c ^ yi); c >>= 31; c = -c; c &= xy; *x = xi ^ c; *y = yi ^ c; } static void int32_sort(crypto_int32 *x,int n) { int top,p,q,i; if (n < 2) return; top = 1; while (top < n - top) top += top; for (p = top;p > 0;p >>= 1) { for (i = 0;i < n - p;++i) if (!(i & p)) minmax(x + i,x + i + p); for (q = top;q > p;q >>= 1) for (i = 0;i < n - q;++i) if (!(i & p)) minmax(x + i + p,x + i + q); } } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/small.h */ #ifndef small_h #define small_h typedef crypto_int8 small; static void small_encode(unsigned char *,const small *); static void small_decode(small *,const unsigned char *); static void small_random(small *); static void small_random_weightw(small *); #endif /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/mod3.h */ #ifndef mod3_h #define mod3_h /* -1 if x is nonzero, 0 otherwise */ static inline int mod3_nonzero_mask(small x) { return -x*x; } /* input between -100000 and 100000 */ /* output between -1 and 1 */ static inline small mod3_freeze(crypto_int32 a) { a -= 3 * ((10923 * a) >> 15); a -= 3 * ((89478485 * a + 134217728) >> 28); return a; } static inline small mod3_minusproduct(small a,small b,small c) { crypto_int32 A = a; crypto_int32 B = b; crypto_int32 C = c; return mod3_freeze(A - B * C); } static inline small mod3_plusproduct(small a,small b,small c) { crypto_int32 A = a; crypto_int32 B = b; crypto_int32 C = c; return mod3_freeze(A + B * C); } static inline small mod3_product(small a,small b) { return a * b; } static inline small mod3_sum(small a,small b) { crypto_int32 A = a; crypto_int32 B = b; return mod3_freeze(A + B); } static inline small mod3_reciprocal(small a1) { return a1; } static inline small mod3_quotient(small num,small den) { return mod3_product(num,mod3_reciprocal(den)); } #endif /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/modq.h */ #ifndef modq_h #define modq_h typedef crypto_int16 modq; /* -1 if x is nonzero, 0 otherwise */ static inline int modq_nonzero_mask(modq x) { crypto_int32 r = (crypto_uint16) x; r = -r; r >>= 30; return r; } /* input between -9000000 and 9000000 */ /* output between -2295 and 2295 */ static inline modq modq_freeze(crypto_int32 a) { a -= 4591 * ((228 * a) >> 20); a -= 4591 * ((58470 * a + 134217728) >> 28); return a; } static inline modq modq_minusproduct(modq a,modq b,modq c) { crypto_int32 A = a; crypto_int32 B = b; crypto_int32 C = c; return modq_freeze(A - B * C); } static inline modq modq_plusproduct(modq a,modq b,modq c) { crypto_int32 A = a; crypto_int32 B = b; crypto_int32 C = c; return modq_freeze(A + B * C); } static inline modq modq_product(modq a,modq b) { crypto_int32 A = a; crypto_int32 B = b; return modq_freeze(A * B); } static inline modq modq_square(modq a) { crypto_int32 A = a; return modq_freeze(A * A); } static inline modq modq_sum(modq a,modq b) { crypto_int32 A = a; crypto_int32 B = b; return modq_freeze(A + B); } static inline modq modq_reciprocal(modq a1) { modq a2 = modq_square(a1); modq a3 = modq_product(a2,a1); modq a4 = modq_square(a2); modq a8 = modq_square(a4); modq a16 = modq_square(a8); modq a32 = modq_square(a16); modq a35 = modq_product(a32,a3); modq a70 = modq_square(a35); modq a140 = modq_square(a70); modq a143 = modq_product(a140,a3); modq a286 = modq_square(a143); modq a572 = modq_square(a286); modq a1144 = modq_square(a572); modq a1147 = modq_product(a1144,a3); modq a2294 = modq_square(a1147); modq a4588 = modq_square(a2294); modq a4589 = modq_product(a4588,a1); return a4589; } static inline modq modq_quotient(modq num,modq den) { return modq_product(num,modq_reciprocal(den)); } #endif /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/params.h */ #ifndef params_h #define params_h #define q 4591 /* XXX: also built into modq in various ways */ #define qshift 2295 #define p 761 #define w 286 #define rq_encode_len 1218 #define small_encode_len 191 #endif /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/r3.h */ #ifndef r3_h #define r3_h static void r3_mult(small *,const small *,const small *); extern int r3_recip(small *,const small *); #endif /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/rq.h */ #ifndef rq_h #define rq_h static void rq_encode(unsigned char *,const modq *); static void rq_decode(modq *,const unsigned char *); static void rq_encoderounded(unsigned char *,const modq *); static void rq_decoderounded(modq *,const unsigned char *); static void rq_round3(modq *,const modq *); static void rq_mult(modq *,const modq *,const small *); int rq_recip3(modq *,const small *); #endif /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/swap.h */ #ifndef swap_h #define swap_h static void swap(void *,void *,int,int); #endif /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/dec.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ #ifdef KAT #endif int crypto_kem_sntrup4591761_dec( unsigned char *k, const unsigned char *cstr, const unsigned char *sk ) { small f[p]; modq h[p]; small grecip[p]; modq c[p]; modq t[p]; small t3[p]; small r[p]; modq hr[p]; unsigned char rstr[small_encode_len]; unsigned char hash[64]; int i; int result = 0; int weight; small_decode(f,sk); small_decode(grecip,sk + small_encode_len); rq_decode(h,sk + 2 * small_encode_len); rq_decoderounded(c,cstr + 32); rq_mult(t,c,f); for (i = 0;i < p;++i) t3[i] = mod3_freeze(modq_freeze(3*t[i])); r3_mult(r,t3,grecip); #ifdef KAT { int j; printf("decrypt r:"); for (j = 0;j < p;++j) if (r[j] == 1) printf(" +%d",j); else if (r[j] == -1) printf(" -%d",j); printf("\n"); } #endif weight = 0; for (i = 0;i < p;++i) weight += (1 & r[i]); weight -= w; result |= modq_nonzero_mask(weight); /* XXX: puts limit on p */ rq_mult(hr,h,r); rq_round3(hr,hr); for (i = 0;i < p;++i) result |= modq_nonzero_mask(hr[i] - c[i]); small_encode(rstr,r); crypto_hash_sha512(hash,rstr,sizeof rstr); result |= crypto_verify_32(hash,cstr); for (i = 0;i < 32;++i) k[i] = (hash[32 + i] & ~result); return result; } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/enc.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ #ifdef KAT #endif int crypto_kem_sntrup4591761_enc( unsigned char *cstr, unsigned char *k, const unsigned char *pk ) { small r[p]; modq h[p]; modq c[p]; unsigned char rstr[small_encode_len]; unsigned char hash[64]; small_random_weightw(r); #ifdef KAT { int i; printf("encrypt r:"); for (i = 0;i < p;++i) if (r[i] == 1) printf(" +%d",i); else if (r[i] == -1) printf(" -%d",i); printf("\n"); } #endif small_encode(rstr,r); crypto_hash_sha512(hash,rstr,sizeof rstr); rq_decode(h,pk); rq_mult(c,h,r); rq_round3(c,c); memcpy(k,hash + 32,32); memcpy(cstr,hash,32); rq_encoderounded(cstr + 32,c); return 0; } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/keypair.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ #if crypto_kem_sntrup4591761_PUBLICKEYBYTES != rq_encode_len #error "crypto_kem_sntrup4591761_PUBLICKEYBYTES must match rq_encode_len" #endif #if crypto_kem_sntrup4591761_SECRETKEYBYTES != rq_encode_len + 2 * small_encode_len #error "crypto_kem_sntrup4591761_SECRETKEYBYTES must match rq_encode_len + 2 * small_encode_len" #endif int crypto_kem_sntrup4591761_keypair(unsigned char *pk,unsigned char *sk) { small g[p]; small grecip[p]; small f[p]; modq f3recip[p]; modq h[p]; do small_random(g); while (r3_recip(grecip,g) != 0); small_random_weightw(f); rq_recip3(f3recip,f); rq_mult(h,f3recip,g); rq_encode(pk,h); small_encode(sk,f); small_encode(sk + small_encode_len,grecip); memcpy(sk + 2 * small_encode_len,pk,rq_encode_len); return 0; } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/r3_mult.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void r3_mult(small *h,const small *f,const small *g) { small fg[p + p - 1]; small result; int i, j; for (i = 0;i < p;++i) { result = 0; for (j = 0;j <= i;++j) result = mod3_plusproduct(result,f[j],g[i - j]); fg[i] = result; } for (i = p;i < p + p - 1;++i) { result = 0; for (j = i - p + 1;j < p;++j) result = mod3_plusproduct(result,f[j],g[i - j]); fg[i] = result; } for (i = p + p - 2;i >= p;--i) { fg[i - p] = mod3_sum(fg[i - p],fg[i]); fg[i - p + 1] = mod3_sum(fg[i - p + 1],fg[i]); } for (i = 0;i < p;++i) h[i] = fg[i]; } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/r3_recip.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ /* caller must ensure that x-y does not overflow */ static int smaller_mask_r3_recip(int x,int y) { return (x - y) >> 31; } static void vectormod3_product(small *z,int len,const small *x,const small c) { int i; for (i = 0;i < len;++i) z[i] = mod3_product(x[i],c); } static void vectormod3_minusproduct(small *z,int len,const small *x,const small *y,const small c) { int i; for (i = 0;i < len;++i) z[i] = mod3_minusproduct(x[i],y[i],c); } static void vectormod3_shift(small *z,int len) { int i; for (i = len - 1;i > 0;--i) z[i] = z[i - 1]; z[0] = 0; } /* r = s^(-1) mod m, returning 0, if s is invertible mod m or returning -1 if s is not invertible mod m r,s are polys of degree

= loops) break; c = mod3_quotient(g[p],f[p]); vectormod3_minusproduct(g,p + 1,g,f,c); vectormod3_shift(g,p + 1); #ifdef SIMPLER vectormod3_minusproduct(v,loops + 1,v,u,c); vectormod3_shift(v,loops + 1); #else if (loop < p) { vectormod3_minusproduct(v,loop + 1,v,u,c); vectormod3_shift(v,loop + 2); } else { vectormod3_minusproduct(v + loop - p,p + 1,v + loop - p,u + loop - p,c); vectormod3_shift(v + loop - p,p + 2); } #endif e -= 1; ++loop; swapmask = smaller_mask_r3_recip(e,d) & mod3_nonzero_mask(g[p]); swap(&e,&d,sizeof e,swapmask); swap(f,g,(p + 1) * sizeof(small),swapmask); #ifdef SIMPLER swap(u,v,(loops + 1) * sizeof(small),swapmask); #else if (loop < p) { swap(u,v,(loop + 1) * sizeof(small),swapmask); } else { swap(u + loop - p,v + loop - p,(p + 1) * sizeof(small),swapmask); } #endif } c = mod3_reciprocal(f[p]); vectormod3_product(r,p,u + p,c); return smaller_mask_r3_recip(0,d); } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/randomsmall.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void small_random(small *g) { int i; for (i = 0;i < p;++i) { crypto_uint32 r = small_random32(); g[i] = (small) (((1073741823 & r) * 3) >> 30) - 1; } } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/randomweightw.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void small_random_weightw(small *f) { crypto_int32 r[p]; int i; for (i = 0;i < p;++i) r[i] = small_random32(); for (i = 0;i < w;++i) r[i] &= -2; for (i = w;i < p;++i) r[i] = (r[i] & -3) | 1; int32_sort(r,p); for (i = 0;i < p;++i) f[i] = ((small) (r[i] & 3)) - 1; } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/rq.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void rq_encode(unsigned char *c,const modq *f) { crypto_int32 f0, f1, f2, f3, f4; int i; for (i = 0;i < p/5;++i) { f0 = *f++ + qshift; f1 = *f++ + qshift; f2 = *f++ + qshift; f3 = *f++ + qshift; f4 = *f++ + qshift; /* now want f0 + 6144*f1 + ... as a 64-bit integer */ f1 *= 3; f2 *= 9; f3 *= 27; f4 *= 81; /* now want f0 + f1<<11 + f2<<22 + f3<<33 + f4<<44 */ f0 += f1 << 11; *c++ = f0; f0 >>= 8; *c++ = f0; f0 >>= 8; f0 += f2 << 6; *c++ = f0; f0 >>= 8; *c++ = f0; f0 >>= 8; f0 += f3 << 1; *c++ = f0; f0 >>= 8; f0 += f4 << 4; *c++ = f0; f0 >>= 8; *c++ = f0; f0 >>= 8; *c++ = f0; } /* XXX: using p mod 5 = 1 */ f0 = *f++ + qshift; *c++ = f0; f0 >>= 8; *c++ = f0; } static void rq_decode(modq *f,const unsigned char *c) { crypto_uint32 c0, c1, c2, c3, c4, c5, c6, c7; crypto_uint32 f0, f1, f2, f3, f4; int i; for (i = 0;i < p/5;++i) { c0 = *c++; c1 = *c++; c2 = *c++; c3 = *c++; c4 = *c++; c5 = *c++; c6 = *c++; c7 = *c++; /* f0 + f1*6144 + f2*6144^2 + f3*6144^3 + f4*6144^4 */ /* = c0 + c1*256 + ... + c6*256^6 + c7*256^7 */ /* with each f between 0 and 4590 */ c6 += c7 << 8; /* c6 <= 23241 = floor(4591*6144^4/2^48) */ /* f4 = (16/81)c6 + (1/1296)(c5+[0,1]) - [0,0.75] */ /* claim: 2^19 f4 < x < 2^19(f4+1) */ /* where x = 103564 c6 + 405(c5+1) */ /* proof: x - 2^19 f4 = (76/81)c6 + (37/81)c5 + 405 - (32768/81)[0,1] + 2^19[0,0.75] */ /* at least 405 - 32768/81 > 0 */ /* at most (76/81)23241 + (37/81)255 + 405 + 2^19 0.75 < 2^19 */ f4 = (103564*c6 + 405*(c5+1)) >> 19; c5 += c6 << 8; c5 -= (f4 * 81) << 4; c4 += c5 << 8; /* f0 + f1*6144 + f2*6144^2 + f3*6144^3 */ /* = c0 + c1*256 + c2*256^2 + c3*256^3 + c4*256^4 */ /* c4 <= 247914 = floor(4591*6144^3/2^32) */ /* f3 = (1/54)(c4+[0,1]) - [0,0.75] */ /* claim: 2^19 f3 < x < 2^19(f3+1) */ /* where x = 9709(c4+2) */ /* proof: x - 2^19 f3 = 19418 - (1/27)c4 - (262144/27)[0,1] + 2^19[0,0.75] */ /* at least 19418 - 247914/27 - 262144/27 > 0 */ /* at most 19418 + 2^19 0.75 < 2^19 */ f3 = (9709*(c4+2)) >> 19; c4 -= (f3 * 27) << 1; c3 += c4 << 8; /* f0 + f1*6144 + f2*6144^2 */ /* = c0 + c1*256 + c2*256^2 + c3*256^3 */ /* c3 <= 10329 = floor(4591*6144^2/2^24) */ /* f2 = (4/9)c3 + (1/576)c2 + (1/147456)c1 + (1/37748736)c0 - [0,0.75] */ /* claim: 2^19 f2 < x < 2^19(f2+1) */ /* where x = 233017 c3 + 910(c2+2) */ /* proof: x - 2^19 f2 = 1820 + (1/9)c3 - (2/9)c2 - (32/9)c1 - (1/72)c0 + 2^19[0,0.75] */ /* at least 1820 - (2/9)255 - (32/9)255 - (1/72)255 > 0 */ /* at most 1820 + (1/9)10329 + 2^19 0.75 < 2^19 */ f2 = (233017*c3 + 910*(c2+2)) >> 19; c2 += c3 << 8; c2 -= (f2 * 9) << 6; c1 += c2 << 8; /* f0 + f1*6144 */ /* = c0 + c1*256 */ /* c1 <= 110184 = floor(4591*6144/2^8) */ /* f1 = (1/24)c1 + (1/6144)c0 - (1/6144)f0 */ /* claim: 2^19 f1 < x < 2^19(f1+1) */ /* where x = 21845(c1+2) + 85 c0 */ /* proof: x - 2^19 f1 = 43690 - (1/3)c1 - (1/3)c0 + 2^19 [0,0.75] */ /* at least 43690 - (1/3)110184 - (1/3)255 > 0 */ /* at most 43690 + 2^19 0.75 < 2^19 */ f1 = (21845*(c1+2) + 85*c0) >> 19; c1 -= (f1 * 3) << 3; c0 += c1 << 8; f0 = c0; *f++ = modq_freeze(f0 + q - qshift); *f++ = modq_freeze(f1 + q - qshift); *f++ = modq_freeze(f2 + q - qshift); *f++ = modq_freeze(f3 + q - qshift); *f++ = modq_freeze(f4 + q - qshift); } c0 = *c++; c1 = *c++; c0 += c1 << 8; *f++ = modq_freeze(c0 + q - qshift); } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/rq_mult.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void rq_mult(modq *h,const modq *f,const small *g) { modq fg[p + p - 1]; modq result; int i, j; for (i = 0;i < p;++i) { result = 0; for (j = 0;j <= i;++j) result = modq_plusproduct(result,f[j],g[i - j]); fg[i] = result; } for (i = p;i < p + p - 1;++i) { result = 0; for (j = i - p + 1;j < p;++j) result = modq_plusproduct(result,f[j],g[i - j]); fg[i] = result; } for (i = p + p - 2;i >= p;--i) { fg[i - p] = modq_sum(fg[i - p],fg[i]); fg[i - p + 1] = modq_sum(fg[i - p + 1],fg[i]); } for (i = 0;i < p;++i) h[i] = fg[i]; } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/rq_recip3.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ /* caller must ensure that x-y does not overflow */ static int smaller_mask_rq_recip3(int x,int y) { return (x - y) >> 31; } static void vectormodq_product(modq *z,int len,const modq *x,const modq c) { int i; for (i = 0;i < len;++i) z[i] = modq_product(x[i],c); } static void vectormodq_minusproduct(modq *z,int len,const modq *x,const modq *y,const modq c) { int i; for (i = 0;i < len;++i) z[i] = modq_minusproduct(x[i],y[i],c); } static void vectormodq_shift(modq *z,int len) { int i; for (i = len - 1;i > 0;--i) z[i] = z[i - 1]; z[0] = 0; } /* r = (3s)^(-1) mod m, returning 0, if s is invertible mod m or returning -1 if s is not invertible mod m r,s are polys of degree

= loops) break; c = modq_quotient(g[p],f[p]); vectormodq_minusproduct(g,p + 1,g,f,c); vectormodq_shift(g,p + 1); #ifdef SIMPLER vectormodq_minusproduct(v,loops + 1,v,u,c); vectormodq_shift(v,loops + 1); #else if (loop < p) { vectormodq_minusproduct(v,loop + 1,v,u,c); vectormodq_shift(v,loop + 2); } else { vectormodq_minusproduct(v + loop - p,p + 1,v + loop - p,u + loop - p,c); vectormodq_shift(v + loop - p,p + 2); } #endif e -= 1; ++loop; swapmask = smaller_mask_rq_recip3(e,d) & modq_nonzero_mask(g[p]); swap(&e,&d,sizeof e,swapmask); swap(f,g,(p + 1) * sizeof(modq),swapmask); #ifdef SIMPLER swap(u,v,(loops + 1) * sizeof(modq),swapmask); #else if (loop < p) { swap(u,v,(loop + 1) * sizeof(modq),swapmask); } else { swap(u + loop - p,v + loop - p,(p + 1) * sizeof(modq),swapmask); } #endif } c = modq_reciprocal(f[p]); vectormodq_product(r,p,u + p,c); return smaller_mask_rq_recip3(0,d); } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/rq_round3.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void rq_round3(modq *h,const modq *f) { int i; for (i = 0;i < p;++i) h[i] = ((21846 * (f[i] + 2295) + 32768) >> 16) * 3 - 2295; } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/rq_rounded.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void rq_encoderounded(unsigned char *c,const modq *f) { crypto_int32 f0, f1, f2; int i; for (i = 0;i < p/3;++i) { f0 = *f++ + qshift; f1 = *f++ + qshift; f2 = *f++ + qshift; f0 = (21846 * f0) >> 16; f1 = (21846 * f1) >> 16; f2 = (21846 * f2) >> 16; /* now want f0 + f1*1536 + f2*1536^2 as a 32-bit integer */ f2 *= 3; f1 += f2 << 9; f1 *= 3; f0 += f1 << 9; *c++ = f0; f0 >>= 8; *c++ = f0; f0 >>= 8; *c++ = f0; f0 >>= 8; *c++ = f0; } /* XXX: using p mod 3 = 2 */ f0 = *f++ + qshift; f1 = *f++ + qshift; f0 = (21846 * f0) >> 16; f1 = (21846 * f1) >> 16; f1 *= 3; f0 += f1 << 9; *c++ = f0; f0 >>= 8; *c++ = f0; f0 >>= 8; *c++ = f0; } static void rq_decoderounded(modq *f,const unsigned char *c) { crypto_uint32 c0, c1, c2, c3; crypto_uint32 f0, f1, f2; int i; for (i = 0;i < p/3;++i) { c0 = *c++; c1 = *c++; c2 = *c++; c3 = *c++; /* f0 + f1*1536 + f2*1536^2 */ /* = c0 + c1*256 + c2*256^2 + c3*256^3 */ /* with each f between 0 and 1530 */ /* f2 = (64/9)c3 + (1/36)c2 + (1/9216)c1 + (1/2359296)c0 - [0,0.99675] */ /* claim: 2^21 f2 < x < 2^21(f2+1) */ /* where x = 14913081*c3 + 58254*c2 + 228*(c1+2) */ /* proof: x - 2^21 f2 = 456 - (8/9)c0 + (4/9)c1 - (2/9)c2 + (1/9)c3 + 2^21 [0,0.99675] */ /* at least 456 - (8/9)255 - (2/9)255 > 0 */ /* at most 456 + (4/9)255 + (1/9)255 + 2^21 0.99675 < 2^21 */ f2 = (14913081*c3 + 58254*c2 + 228*(c1+2)) >> 21; c2 += c3 << 8; c2 -= (f2 * 9) << 2; /* f0 + f1*1536 */ /* = c0 + c1*256 + c2*256^2 */ /* c2 <= 35 = floor((1530+1530*1536)/256^2) */ /* f1 = (128/3)c2 + (1/6)c1 + (1/1536)c0 - (1/1536)f0 */ /* claim: 2^21 f1 < x < 2^21(f1+1) */ /* where x = 89478485*c2 + 349525*c1 + 1365*(c0+1) */ /* proof: x - 2^21 f1 = 1365 - (1/3)c2 - (1/3)c1 - (1/3)c0 + (4096/3)f0 */ /* at least 1365 - (1/3)35 - (1/3)255 - (1/3)255 > 0 */ /* at most 1365 + (4096/3)1530 < 2^21 */ f1 = (89478485*c2 + 349525*c1 + 1365*(c0+1)) >> 21; c1 += c2 << 8; c1 -= (f1 * 3) << 1; c0 += c1 << 8; f0 = c0; *f++ = modq_freeze(f0 * 3 + q - qshift); *f++ = modq_freeze(f1 * 3 + q - qshift); *f++ = modq_freeze(f2 * 3 + q - qshift); } c0 = *c++; c1 = *c++; c2 = *c++; f1 = (89478485*c2 + 349525*c1 + 1365*(c0+1)) >> 21; c1 += c2 << 8; c1 -= (f1 * 3) << 1; c0 += c1 << 8; f0 = c0; *f++ = modq_freeze(f0 * 3 + q - qshift); *f++ = modq_freeze(f1 * 3 + q - qshift); } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/small.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ /* XXX: these functions rely on p mod 4 = 1 */ /* all coefficients in -1, 0, 1 */ static void small_encode(unsigned char *c,const small *f) { small c0; int i; for (i = 0;i < p/4;++i) { c0 = *f++ + 1; c0 += (*f++ + 1) << 2; c0 += (*f++ + 1) << 4; c0 += (*f++ + 1) << 6; *c++ = c0; } c0 = *f++ + 1; *c++ = c0; } static void small_decode(small *f,const unsigned char *c) { unsigned char c0; int i; for (i = 0;i < p/4;++i) { c0 = *c++; *f++ = ((small) (c0 & 3)) - 1; c0 >>= 2; *f++ = ((small) (c0 & 3)) - 1; c0 >>= 2; *f++ = ((small) (c0 & 3)) - 1; c0 >>= 2; *f++ = ((small) (c0 & 3)) - 1; } c0 = *c++; *f++ = ((small) (c0 & 3)) - 1; } /* from libpqcrypto-20180314/crypto_kem/sntrup4591761/ref/swap.c */ /* See https://ntruprime.cr.yp.to/software.html for detailed documentation. */ static void swap(void *x,void *y,int bytes,int mask) { int i; char xi, yi, c, t; c = mask; for (i = 0;i < bytes;++i) { xi = i[(char *) x]; yi = i[(char *) y]; t = c & (xi ^ yi); xi ^= t; yi ^= t; i[(char *) x] = xi; i[(char *) y] = yi; } }