-
Notifications
You must be signed in to change notification settings - Fork 6
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add MLKEM reference implementation (#4)
* Add MLKEM reference implementation This adds the reference implementation from https://github.com/pq-crystals/kyber/tree/standard/ref as a starting point for the hackathon. Signed-off-by: Matthias J. Kannwischer <[email protected]> * MLKEM: remove Makefile Signed-off-by: Matthias J. Kannwischer <[email protected]> --------- Signed-off-by: Matthias J. Kannwischer <[email protected]>
- Loading branch information
1 parent
5e4c9ce
commit 235d16c
Showing
27 changed files
with
2,912 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
# SPDX-License-Identifier: Apache-2.0 | ||
# temporary Makefile to be replaced by Arm cross-compilation | ||
|
||
CC ?= /usr/bin/cc | ||
CFLAGS += -Wall -Wextra -Wpedantic -Wmissing-prototypes -Wredundant-decls \ | ||
-Wshadow -Wpointer-arith -O3 -fomit-frame-pointer -z noexecstack \ | ||
-Imlkem | ||
|
||
MLKEM_SOURCES = $(wildcard mlkem/*.c) | ||
MLKEM_HEADERS = $(wildcard mlkem/*.h) | ||
|
||
.PHONY: all clean | ||
|
||
all: mlkem-512-test mlkem-768-test mlkem-1024-test | ||
|
||
mlkem-512-test: test/test.c $(MLKEM_SOURCES) $(MLKEM_HEADERS) | ||
$(CC) $(CFLAGS) -DKYBER_K=2 $< $(MLKEM_SOURCES) -o $@ | ||
|
||
mlkem-768-test: test/test.c $(MLKEM_SOURCES) $(MLKEM_HEADERS) | ||
$(CC) $(CFLAGS) -DKYBER_K=3 $< $(MLKEM_SOURCES) -o $@ | ||
|
||
mlkem-1024-test: test/test.c $(MLKEM_SOURCES) $(MLKEM_HEADERS) | ||
$(CC) $(CFLAGS) -DKYBER_K=4 $< $(MLKEM_SOURCES) -o $@ | ||
|
||
clean: | ||
$(RM) mlkem-512-test mlkem-768-test mlkem-1024-test |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
Public Domain (https://creativecommons.org/share-your-work/public-domain/cc0/); | ||
or Apache 2.0 License (https://www.apache.org/licenses/LICENSE-2.0.html). | ||
|
||
For Keccak and AES we are using public-domain | ||
code from sources and by authors listed in | ||
comments on top of the respective files. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,66 @@ | ||
#ifndef API_H | ||
#define API_H | ||
|
||
#include <stdint.h> | ||
|
||
#define pqcrystals_kyber512_SECRETKEYBYTES 1632 | ||
#define pqcrystals_kyber512_PUBLICKEYBYTES 800 | ||
#define pqcrystals_kyber512_CIPHERTEXTBYTES 768 | ||
#define pqcrystals_kyber512_KEYPAIRCOINBYTES 64 | ||
#define pqcrystals_kyber512_ENCCOINBYTES 32 | ||
#define pqcrystals_kyber512_BYTES 32 | ||
|
||
#define pqcrystals_kyber512_ref_SECRETKEYBYTES pqcrystals_kyber512_SECRETKEYBYTES | ||
#define pqcrystals_kyber512_ref_PUBLICKEYBYTES pqcrystals_kyber512_PUBLICKEYBYTES | ||
#define pqcrystals_kyber512_ref_CIPHERTEXTBYTES pqcrystals_kyber512_CIPHERTEXTBYTES | ||
#define pqcrystals_kyber512_ref_KEYPAIRCOINBYTES pqcrystals_kyber512_KEYPAIRCOINBYTES | ||
#define pqcrystals_kyber512_ref_ENCCOINBYTES pqcrystals_kyber512_ENCCOINBYTES | ||
#define pqcrystals_kyber512_ref_BYTES pqcrystals_kyber512_BYTES | ||
|
||
int pqcrystals_kyber512_ref_keypair_derand(uint8_t *pk, uint8_t *sk, const uint8_t *coins); | ||
int pqcrystals_kyber512_ref_keypair(uint8_t *pk, uint8_t *sk); | ||
int pqcrystals_kyber512_ref_enc_derand(uint8_t *ct, uint8_t *ss, const uint8_t *pk, const uint8_t *coins); | ||
int pqcrystals_kyber512_ref_enc(uint8_t *ct, uint8_t *ss, const uint8_t *pk); | ||
int pqcrystals_kyber512_ref_dec(uint8_t *ss, const uint8_t *ct, const uint8_t *sk); | ||
|
||
#define pqcrystals_kyber768_SECRETKEYBYTES 2400 | ||
#define pqcrystals_kyber768_PUBLICKEYBYTES 1184 | ||
#define pqcrystals_kyber768_CIPHERTEXTBYTES 1088 | ||
#define pqcrystals_kyber768_KEYPAIRCOINBYTES 64 | ||
#define pqcrystals_kyber768_ENCCOINBYTES 32 | ||
#define pqcrystals_kyber768_BYTES 32 | ||
|
||
#define pqcrystals_kyber768_ref_SECRETKEYBYTES pqcrystals_kyber768_SECRETKEYBYTES | ||
#define pqcrystals_kyber768_ref_PUBLICKEYBYTES pqcrystals_kyber768_PUBLICKEYBYTES | ||
#define pqcrystals_kyber768_ref_CIPHERTEXTBYTES pqcrystals_kyber768_CIPHERTEXTBYTES | ||
#define pqcrystals_kyber768_ref_KEYPAIRCOINBYTES pqcrystals_kyber768_KEYPAIRCOINBYTES | ||
#define pqcrystals_kyber768_ref_ENCCOINBYTES pqcrystals_kyber768_ENCCOINBYTES | ||
#define pqcrystals_kyber768_ref_BYTES pqcrystals_kyber768_BYTES | ||
|
||
int pqcrystals_kyber768_ref_keypair_derand(uint8_t *pk, uint8_t *sk, const uint8_t *coins); | ||
int pqcrystals_kyber768_ref_keypair(uint8_t *pk, uint8_t *sk); | ||
int pqcrystals_kyber768_ref_enc_derand(uint8_t *ct, uint8_t *ss, const uint8_t *pk, const uint8_t *coins); | ||
int pqcrystals_kyber768_ref_enc(uint8_t *ct, uint8_t *ss, const uint8_t *pk); | ||
int pqcrystals_kyber768_ref_dec(uint8_t *ss, const uint8_t *ct, const uint8_t *sk); | ||
|
||
#define pqcrystals_kyber1024_SECRETKEYBYTES 3168 | ||
#define pqcrystals_kyber1024_PUBLICKEYBYTES 1568 | ||
#define pqcrystals_kyber1024_CIPHERTEXTBYTES 1568 | ||
#define pqcrystals_kyber1024_KEYPAIRCOINBYTES 64 | ||
#define pqcrystals_kyber1024_ENCCOINBYTES 32 | ||
#define pqcrystals_kyber1024_BYTES 32 | ||
|
||
#define pqcrystals_kyber1024_ref_SECRETKEYBYTES pqcrystals_kyber1024_SECRETKEYBYTES | ||
#define pqcrystals_kyber1024_ref_PUBLICKEYBYTES pqcrystals_kyber1024_PUBLICKEYBYTES | ||
#define pqcrystals_kyber1024_ref_CIPHERTEXTBYTES pqcrystals_kyber1024_CIPHERTEXTBYTES | ||
#define pqcrystals_kyber1024_ref_KEYPAIRCOINBYTES pqcrystals_kyber1024_KEYPAIRCOINBYTES | ||
#define pqcrystals_kyber1024_ref_ENCCOINBYTES pqcrystals_kyber1024_ENCCOINBYTES | ||
#define pqcrystals_kyber1024_ref_BYTES pqcrystals_kyber1024_BYTES | ||
|
||
int pqcrystals_kyber1024_ref_keypair_derand(uint8_t *pk, uint8_t *sk, const uint8_t *coins); | ||
int pqcrystals_kyber1024_ref_keypair(uint8_t *pk, uint8_t *sk); | ||
int pqcrystals_kyber1024_ref_enc_derand(uint8_t *ct, uint8_t *ss, const uint8_t *pk, const uint8_t *coins); | ||
int pqcrystals_kyber1024_ref_enc(uint8_t *ct, uint8_t *ss, const uint8_t *pk); | ||
int pqcrystals_kyber1024_ref_dec(uint8_t *ss, const uint8_t *ct, const uint8_t *sk); | ||
|
||
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,128 @@ | ||
#include <stdint.h> | ||
#include "params.h" | ||
#include "cbd.h" | ||
|
||
/************************************************* | ||
* Name: load32_littleendian | ||
* | ||
* Description: load 4 bytes into a 32-bit integer | ||
* in little-endian order | ||
* | ||
* Arguments: - const uint8_t *x: pointer to input byte array | ||
* | ||
* Returns 32-bit unsigned integer loaded from x | ||
**************************************************/ | ||
static uint32_t load32_littleendian(const uint8_t x[4]) | ||
{ | ||
uint32_t r; | ||
r = (uint32_t)x[0]; | ||
r |= (uint32_t)x[1] << 8; | ||
r |= (uint32_t)x[2] << 16; | ||
r |= (uint32_t)x[3] << 24; | ||
return r; | ||
} | ||
|
||
/************************************************* | ||
* Name: load24_littleendian | ||
* | ||
* Description: load 3 bytes into a 32-bit integer | ||
* in little-endian order. | ||
* This function is only needed for Kyber-512 | ||
* | ||
* Arguments: - const uint8_t *x: pointer to input byte array | ||
* | ||
* Returns 32-bit unsigned integer loaded from x (most significant byte is zero) | ||
**************************************************/ | ||
#if KYBER_ETA1 == 3 | ||
static uint32_t load24_littleendian(const uint8_t x[3]) | ||
{ | ||
uint32_t r; | ||
r = (uint32_t)x[0]; | ||
r |= (uint32_t)x[1] << 8; | ||
r |= (uint32_t)x[2] << 16; | ||
return r; | ||
} | ||
#endif | ||
|
||
|
||
/************************************************* | ||
* Name: cbd2 | ||
* | ||
* Description: Given an array of uniformly random bytes, compute | ||
* polynomial with coefficients distributed according to | ||
* a centered binomial distribution with parameter eta=2 | ||
* | ||
* Arguments: - poly *r: pointer to output polynomial | ||
* - const uint8_t *buf: pointer to input byte array | ||
**************************************************/ | ||
static void cbd2(poly *r, const uint8_t buf[2*KYBER_N/4]) | ||
{ | ||
unsigned int i,j; | ||
uint32_t t,d; | ||
int16_t a,b; | ||
|
||
for(i=0;i<KYBER_N/8;i++) { | ||
t = load32_littleendian(buf+4*i); | ||
d = t & 0x55555555; | ||
d += (t>>1) & 0x55555555; | ||
|
||
for(j=0;j<8;j++) { | ||
a = (d >> (4*j+0)) & 0x3; | ||
b = (d >> (4*j+2)) & 0x3; | ||
r->coeffs[8*i+j] = a - b; | ||
} | ||
} | ||
} | ||
|
||
/************************************************* | ||
* Name: cbd3 | ||
* | ||
* Description: Given an array of uniformly random bytes, compute | ||
* polynomial with coefficients distributed according to | ||
* a centered binomial distribution with parameter eta=3. | ||
* This function is only needed for Kyber-512 | ||
* | ||
* Arguments: - poly *r: pointer to output polynomial | ||
* - const uint8_t *buf: pointer to input byte array | ||
**************************************************/ | ||
#if KYBER_ETA1 == 3 | ||
static void cbd3(poly *r, const uint8_t buf[3*KYBER_N/4]) | ||
{ | ||
unsigned int i,j; | ||
uint32_t t,d; | ||
int16_t a,b; | ||
|
||
for(i=0;i<KYBER_N/4;i++) { | ||
t = load24_littleendian(buf+3*i); | ||
d = t & 0x00249249; | ||
d += (t>>1) & 0x00249249; | ||
d += (t>>2) & 0x00249249; | ||
|
||
for(j=0;j<4;j++) { | ||
a = (d >> (6*j+0)) & 0x7; | ||
b = (d >> (6*j+3)) & 0x7; | ||
r->coeffs[4*i+j] = a - b; | ||
} | ||
} | ||
} | ||
#endif | ||
|
||
void poly_cbd_eta1(poly *r, const uint8_t buf[KYBER_ETA1*KYBER_N/4]) | ||
{ | ||
#if KYBER_ETA1 == 2 | ||
cbd2(r, buf); | ||
#elif KYBER_ETA1 == 3 | ||
cbd3(r, buf); | ||
#else | ||
#error "This implementation requires eta1 in {2,3}" | ||
#endif | ||
} | ||
|
||
void poly_cbd_eta2(poly *r, const uint8_t buf[KYBER_ETA2*KYBER_N/4]) | ||
{ | ||
#if KYBER_ETA2 == 2 | ||
cbd2(r, buf); | ||
#else | ||
#error "This implementation requires eta2 = 2" | ||
#endif | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
#ifndef CBD_H | ||
#define CBD_H | ||
|
||
#include <stdint.h> | ||
#include "params.h" | ||
#include "poly.h" | ||
|
||
#define poly_cbd_eta1 KYBER_NAMESPACE(poly_cbd_eta1) | ||
void poly_cbd_eta1(poly *r, const uint8_t buf[KYBER_ETA1*KYBER_N/4]); | ||
|
||
#define poly_cbd_eta2 KYBER_NAMESPACE(poly_cbd_eta2) | ||
void poly_cbd_eta2(poly *r, const uint8_t buf[KYBER_ETA2*KYBER_N/4]); | ||
|
||
#endif |
Oops, something went wrong.