montana/iOS/liboqs/zephyr/samples/Signatures/src/main.c

191 lines
5.2 KiB
C
Raw Normal View History

// SPDX-License-Identifier: MIT
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
// Autogenerated header file from Zephyr containing the version number
#include <version.h>
#if KERNEL_VERSION_NUMBER >= 0x30500
#include <zephyr/random/random.h>
#else
#include <zephyr/random/rand32.h>
#endif
#include <oqs/oqs.h>
typedef struct magic_s {
uint8_t val[31];
} magic_t;
void zephyr_randombytes(uint8_t *random_array, size_t bytes_to_read)
{
// Obtain random bytes from the zephyr RNG
sys_rand_get(random_array, bytes_to_read);
}
static OQS_STATUS sig_test_correctness(const char *method_name) {
OQS_SIG *sig = NULL;
uint8_t *public_key = NULL;
uint8_t *secret_key = NULL;
uint8_t *message = NULL;
size_t message_len = 100;
uint8_t *signature = NULL;
size_t signature_len;
OQS_STATUS rc, ret = OQS_ERROR;
//The magic numbers are random values.
//The length of the magic number was chosen to be 31 to break alignment
magic_t magic;
OQS_randombytes(magic.val, sizeof(magic_t));
sig = OQS_SIG_new(method_name);
if (sig == NULL) {
fprintf(stderr, "ERROR: OQS_SIG_new failed\n");
goto err;
}
printf("================================================================================\n");
printf("Sample computation for signature %s\n", sig->method_name);
printf("================================================================================\n");
public_key = malloc(sig->length_public_key + 2 * sizeof(magic_t));
secret_key = malloc(sig->length_secret_key + 2 * sizeof(magic_t));
message = malloc(message_len + 2 * sizeof(magic_t));
signature = malloc(sig->length_signature + 2 * sizeof(magic_t));
if ((public_key == NULL) || (secret_key == NULL) || (message == NULL) || (signature == NULL)) {
fprintf(stderr, "ERROR: malloc failed\n");
goto err;
}
//Set the magic numbers before
memcpy(public_key, magic.val, sizeof(magic_t));
memcpy(secret_key, magic.val, sizeof(magic_t));
memcpy(message, magic.val, sizeof(magic_t));
memcpy(signature, magic.val, sizeof(magic_t));
public_key += sizeof(magic_t);
secret_key += sizeof(magic_t);
message += sizeof(magic_t);
signature += sizeof(magic_t);
// and after
memcpy(public_key + sig->length_public_key, magic.val, sizeof(magic_t));
memcpy(secret_key + sig->length_secret_key, magic.val, sizeof(magic_t));
memcpy(message + message_len, magic.val, sizeof(magic_t));
memcpy(signature + sig->length_signature, magic.val, sizeof(magic_t));
OQS_randombytes(message, message_len);
rc = OQS_SIG_keypair(sig, public_key, secret_key);
if (rc != OQS_SUCCESS) {
fprintf(stderr, "ERROR: OQS_SIG_keypair failed\n");
goto err;
}
rc = OQS_SIG_sign(sig, signature, &signature_len, message, message_len, secret_key);
if (rc != OQS_SUCCESS) {
fprintf(stderr, "ERROR: OQS_SIG_sign failed\n");
goto err;
}
rc = OQS_SIG_verify(sig, message, message_len, signature, signature_len, public_key);
if (rc != OQS_SUCCESS) {
fprintf(stderr, "ERROR: OQS_SIG_verify failed\n");
goto err;
}
/* modify the signature to invalidate it */
OQS_randombytes(signature, signature_len);
rc = OQS_SIG_verify(sig, message, message_len, signature, signature_len, public_key);
if (rc != OQS_ERROR) {
fprintf(stderr, "ERROR: OQS_SIG_verify should have failed!\n");
goto err;
}
/* check magic values */
int rv = memcmp(public_key + sig->length_public_key, magic.val, sizeof(magic_t));
rv |= memcmp(secret_key + sig->length_secret_key, magic.val, sizeof(magic_t));
rv |= memcmp(message + message_len, magic.val, sizeof(magic_t));
rv |= memcmp(signature + sig->length_signature, magic.val, sizeof(magic_t));
rv |= memcmp(public_key - sizeof(magic_t), magic.val, sizeof(magic_t));
rv |= memcmp(secret_key - sizeof(magic_t), magic.val, sizeof(magic_t));
rv |= memcmp(message - sizeof(magic_t), magic.val, sizeof(magic_t));
rv |= memcmp(signature - sizeof(magic_t), magic.val, sizeof(magic_t));
if (rv) {
fprintf(stderr, "ERROR: Magic numbers do not mtach\n");
goto err;
}
printf("verification passes as expected\n");
ret = OQS_SUCCESS;
goto cleanup;
err:
ret = OQS_ERROR;
cleanup:
if ((secret_key) && (sig != NULL)) {
OQS_MEM_secure_free(secret_key - sizeof(magic_t), sig->length_secret_key + 2 * sizeof(magic_t));
}
if (public_key) {
OQS_MEM_insecure_free(public_key - sizeof(magic_t));
}
if (message) {
OQS_MEM_insecure_free(message - sizeof(magic_t));
}
if (signature) {
OQS_MEM_insecure_free(signature - sizeof(magic_t));
}
OQS_SIG_free(sig);
return ret;
}
int main(void)
{
OQS_STATUS rc;
printf("Testing signature algorithms using liboqs version %s\n", OQS_version());
OQS_init();
/* Set a RNG callback for Zephyr */
OQS_randombytes_custom_algorithm(zephyr_randombytes);
if (OQS_SIG_alg_count() == 0) {
printf("No signature algorithms enabled!\n");
OQS_destroy();
return EXIT_FAILURE;
}
for (int i = 0; i < OQS_SIG_alg_count(); i++) {
const char *alg_name = OQS_SIG_alg_identifier(i);
if ((!OQS_SIG_alg_is_enabled(alg_name)) || !(strncmp("SLH_DSA", alg_name, 7))) {
printf("Signature algorithm %s not enabled!\n", alg_name);
}
else {
rc = sig_test_correctness(alg_name);
if (rc != OQS_SUCCESS) {
OQS_destroy();
return EXIT_FAILURE;
}
}
}
OQS_destroy();
printf("Test done\n");
return EXIT_SUCCESS;
}