You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
lokinet/crypto/libntrup/src/ref/dec.c

76 lines
1.5 KiB
C

#ifdef KAT
#include <stdio.h>
#endif
#include <sodium/crypto_hash_sha512.h>
#include <sodium/crypto_verify_32.h>
#include <sodium/crypto_kem.h>
#include "params.h"
#include "small.h"
#include "mod3.h"
#include "rq.h"
#include "r3.h"
int
crypto_kem_dec_ref(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;
}