mirror of
https://github.com/henrydcase/pqc.git
synced 2024-12-04 21:34:01 +00:00
139 lines
2.8 KiB
C
139 lines
2.8 KiB
C
|
/*
|
||
|
This file is for Niederreiter encryption
|
||
|
*/
|
||
|
|
||
|
#include "encrypt.h"
|
||
|
|
||
|
#include "params.h"
|
||
|
#include "randombytes.h"
|
||
|
#include "util.h"
|
||
|
|
||
|
#include <stdint.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#include "gf.h"
|
||
|
|
||
|
static inline uint8_t same_mask(uint16_t x, uint16_t y) {
|
||
|
uint32_t mask;
|
||
|
|
||
|
mask = x ^ y;
|
||
|
mask -= 1;
|
||
|
mask >>= 31;
|
||
|
mask = -mask;
|
||
|
|
||
|
return (uint8_t)mask;
|
||
|
}
|
||
|
|
||
|
/* output: e, an error vector of weight t */
|
||
|
static void gen_e(unsigned char *e) {
|
||
|
size_t i, j;
|
||
|
int eq, count;
|
||
|
|
||
|
uint16_t ind_[ SYS_T * 2 ];
|
||
|
uint8_t *ind_8 = (uint8_t *)ind_;
|
||
|
uint16_t ind[ SYS_T * 2 ];
|
||
|
uint8_t mask;
|
||
|
unsigned char val[ SYS_T ];
|
||
|
|
||
|
while (1) {
|
||
|
randombytes(ind_8, sizeof(ind_));
|
||
|
// Copy to uint16_t ind_ in a little-endian way
|
||
|
for (i = 0; i < sizeof(ind_); i += 2) {
|
||
|
ind_[i / 2] = ((uint16_t)ind_8[i + 1]) << 8 | (uint16_t)ind_8[i];
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < SYS_T * 2; i++) {
|
||
|
ind_[i] &= GFMASK;
|
||
|
}
|
||
|
|
||
|
// moving and counting indices in the correct range
|
||
|
|
||
|
count = 0;
|
||
|
for (i = 0; i < SYS_T * 2; i++) {
|
||
|
if (ind_[i] < SYS_N) {
|
||
|
ind[ count++ ] = ind_[i];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (count < SYS_T) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// check for repetition
|
||
|
|
||
|
eq = 0;
|
||
|
|
||
|
for (i = 1; i < SYS_T; i++) {
|
||
|
for (j = 0; j < i; j++) {
|
||
|
if (ind[i] == ind[j]) {
|
||
|
eq = 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (eq == 0) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for (j = 0; j < SYS_T; j++) {
|
||
|
val[j] = 1 << (ind[j] & 7);
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < SYS_N / 8; i++) {
|
||
|
e[i] = 0;
|
||
|
|
||
|
for (j = 0; j < SYS_T; j++) {
|
||
|
mask = same_mask((uint16_t)i, (ind[j] >> 3));
|
||
|
|
||
|
e[i] |= val[j] & mask;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* input: public key pk, error vector e */
|
||
|
/* output: syndrome s */
|
||
|
static void syndrome(unsigned char *s, const unsigned char *pk, const unsigned char *e) {
|
||
|
unsigned char b, row[SYS_N / 8];
|
||
|
const unsigned char *pk_ptr = pk;
|
||
|
|
||
|
int i, j;
|
||
|
|
||
|
for (i = 0; i < SYND_BYTES; i++) {
|
||
|
s[i] = 0;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < PK_NROWS; i++) {
|
||
|
for (j = 0; j < SYS_N / 8; j++) {
|
||
|
row[j] = 0;
|
||
|
}
|
||
|
|
||
|
for (j = 0; j < PK_ROW_BYTES; j++) {
|
||
|
row[ SYS_N / 8 - PK_ROW_BYTES + j ] = pk_ptr[j];
|
||
|
}
|
||
|
|
||
|
row[i / 8] |= 1 << (i % 8);
|
||
|
|
||
|
b = 0;
|
||
|
for (j = 0; j < SYS_N / 8; j++) {
|
||
|
b ^= row[j] & e[j];
|
||
|
}
|
||
|
|
||
|
b ^= b >> 4;
|
||
|
b ^= b >> 2;
|
||
|
b ^= b >> 1;
|
||
|
b &= 1;
|
||
|
|
||
|
s[ i / 8 ] |= (b << (i % 8));
|
||
|
|
||
|
pk_ptr += PK_ROW_BYTES;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void PQCLEAN_MCELIECE460896_CLEAN_encrypt(unsigned char *s, unsigned char *e, const unsigned char *pk) {
|
||
|
gen_e(e);
|
||
|
|
||
|
syndrome(s, pk, e);
|
||
|
}
|
||
|
|