blob: 87ea6c43c83ffd4a5ab05fcdc9d8c67250372882 [file]
/*
* Embedded Linux library
* Copyright (C) 2018 Intel Corporation
*
* SPDX-License-Identifier: LGPL-2.1-or-later
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <string.h>
#include <assert.h>
#include <ell/ell.h>
#include "ell/ecc-private.h"
static bool use_real_getrandom = true;
bool __wrap_l_getrandom(void *buf, size_t len);
bool __real_l_getrandom(void *buf, size_t len);
bool __wrap_l_getrandom(void *buf, size_t len)
{
static const uint8_t random_buf[] = { 0xba, 0xaf, 0x6d, 0x97, 0x71,
0xe5, 0xda, 0xc9, 0x89, 0x6e,
0x58, 0x18, 0x92, 0xf8, 0x55,
0x4f, 0x29, 0xf0, 0xbd, 0x10,
0xaf, 0x0e, 0x38, 0xb5, 0xe6,
0x44, 0x56, 0x9d, 0x99, 0x2a,
0x7f, 0xe2, 0x8d, 0x46, 0xb0,
0x73, 0xcd, 0xd3, 0x6a, 0x7b,
0xa6, 0xd3, 0xde, 0xbf, 0x38,
0x96, 0xb7, 0xc3 };
if (use_real_getrandom)
return __real_l_getrandom(buf, len);
if (len > sizeof(random_buf))
return false;
memcpy(buf, random_buf, len);
return true;
}
/*
* Tests the most basic case. Generate two full public keys and use to create
* two identical shared secrets.
*/
static void test_basic(const void *data)
{
const struct l_ecc_curve *curve = l_ecc_curve_from_ike_group(19);
struct l_ecc_scalar *private1;
struct l_ecc_scalar *private2;
struct l_ecc_point *public1;
struct l_ecc_point *public2;
struct l_ecc_scalar *secret1;
struct l_ecc_scalar *secret2;
assert(l_ecdh_generate_key_pair(curve, &private1, &public1));
assert(l_ecdh_generate_key_pair(curve, &private2, &public2));
assert(l_ecdh_generate_shared_secret(private1, public2, &secret1));
assert(l_ecdh_generate_shared_secret(private2, public1, &secret2));
assert(!memcmp(secret1->c, secret2->c, 32));
l_ecc_scalar_free(private1);
l_ecc_scalar_free(private2);
l_ecc_point_free(public1);
l_ecc_point_free(public2);
l_ecc_scalar_free(secret1);
l_ecc_scalar_free(secret2);
}
/*
* Test vector from RFC 5114 - 256-bit Random ECP Group
*/
static void test_vector_p256(const void *data)
{
const struct l_ecc_curve *curve = l_ecc_curve_from_ike_group(19);
uint64_t a_sec_buf[4] = { 0x867B7291D507A3AFull, 0x3FAF432A5ABCE59Eull,
0xE96A8E337A128499ull, 0x814264145F2F56F2ull };
uint64_t a_pub_buf[8] = { 0x5E8D3B4BA83AEB15ull, 0x7165BE50BC42AE4Aull,
0xC9B5A8D4160D09E9ull, 0x2AF502F3BE8952F2ull,
0xC0F5015ECE5EFD85ull, 0x6795BD4BFF6E6DE3ull,
0x8681A0F9872D79D5ull, 0xEB0FAF4CA986C4D3ull };
uint64_t b_sec_buf[4] = { 0xEE1B593761CF7F41ull, 0x19CE6BCCAD562B8Eull,
0xDB95A200CC0AB26Aull, 0x2CE1788EC197E096ull };
uint64_t b_pub_buf[8] = { 0xB3AB0715F6CE51B0ull, 0xAE06AAEA279FA775ull,
0x5346E8DE6C2C8646ull, 0xB120DE4AA3649279ull,
0x85C34DDE5708B2B6ull, 0x3727027092A84113ull,
0xD8EC685FA3F071D8ull, 0x9F1B7EECE20D7B5Eull };
uint64_t ss_buf[4] = { 0x7F80D21C820C2788ull,
0xF5811E9DC8EC8EEAull,
0x93310412D19A08F1ull,
0xDD0F5396219D1EA3ull };
struct l_ecc_scalar *a_shared;
struct l_ecc_scalar *b_shared;
struct l_ecc_scalar *a_secret = _ecc_constant_new(curve, a_sec_buf,
sizeof(a_sec_buf));
struct l_ecc_point *a_public = l_ecc_point_new(curve);
struct l_ecc_scalar *b_secret = _ecc_constant_new(curve, b_sec_buf,
sizeof(b_sec_buf));
struct l_ecc_point *b_public = l_ecc_point_new(curve);
memcpy(a_public->x, a_pub_buf, 32);
memcpy(a_public->y, a_pub_buf + 4, 32);
memcpy(b_public->x, b_pub_buf, 32);
memcpy(b_public->y, b_pub_buf + 4, 32);
use_real_getrandom = false;
assert(l_ecdh_generate_shared_secret(a_secret, b_public, &a_shared));
assert(l_ecdh_generate_shared_secret(b_secret, a_public, &b_shared));
assert(!memcmp(a_shared->c, ss_buf, 32));
assert(!memcmp(b_shared->c, ss_buf, 32));
use_real_getrandom = true;
l_ecc_scalar_free(a_secret);
l_ecc_scalar_free(b_secret);
l_ecc_point_free(a_public);
l_ecc_point_free(b_public);
l_ecc_scalar_free(a_shared);
l_ecc_scalar_free(b_shared);
}
/*
* Test vector from RFC 5114 - 384-bit Random ECP Group
*/
static void test_vector_p384(const void *data)
{
const struct l_ecc_curve *curve = l_ecc_curve_from_ike_group(20);
uint64_t a_sec_buf[6] = { 0x86F05FEADB9376F1ull, 0xD706A90CBCB5DF29ull,
0xD709EE7A7962A156ull, 0x5DFD8A7965571C48ull,
0x44DD14E9FD126071ull, 0xD27335EA71664AF2ull };
uint64_t a_pub_buf[12] = { 0x7D016FE27A8B8C66ull, 0x7E6A8EA9D1FD7742ull,
0x0EE6B0403D627954ull, 0xE057AB62F82054D1ull,
0xDA4C6D9074417D05ull, 0x793148F1787634D5ull,
0xBACED214A1A1D128ull, 0x8F7A685923DE3B67ull,
0x6B8F398BB29E4236ull, 0xC947392E94F4C3F0ull,
0xF480F4FB4CD40504ull, 0xC6C41294331D23E6ull };
uint64_t b_sec_buf[6] = { 0x2C4A6C768BCD94D2ull, 0x9BE52E00C194A413ull,
0x1F80231121CCE3D3ull, 0x3B6125262C36A7DFull,
0x9C0F00D456C2F702ull, 0x52D1791FDB4B70F8ull };
uint64_t b_pub_buf[12] = { 0x223F12B5A1ABC120ull, 0x789D72A84865AE2Full,
0x4ABC17647B6B9999ull, 0x5B36DB65915359B4ull,
0xF74B8D4EFB708B3Dull, 0x5CD42AB9C41B5347ull,
0xE035B0EDF36755DEull, 0x40BDE8723415A8ECull,
0x0CECA16356CA9332ull, 0x8F6D5B348C0FA4D8ull,
0xA3A8BFAC46B404BDull, 0xE171458FEAA939AAull };
uint64_t ss_buf[6] = {
0xDE159A58028ABC0Eull, 0x27AA8A4540884C37ull,
0x59D926EB1B8456E4ull, 0xCAE53160137D904Cull,
0x55981B110575E0A8ull, 0x5EA1FC4AF7256D20ull };
struct l_ecc_scalar *a_shared;
struct l_ecc_scalar *b_shared;
struct l_ecc_scalar *a_secret = _ecc_constant_new(curve, a_sec_buf,
sizeof(a_sec_buf));
struct l_ecc_point *a_public = l_ecc_point_new(curve);
struct l_ecc_scalar *b_secret = _ecc_constant_new(curve, b_sec_buf,
sizeof(b_sec_buf));
struct l_ecc_point *b_public = l_ecc_point_new(curve);
memcpy(a_public->x, a_pub_buf, 48);
memcpy(a_public->y, a_pub_buf + 6, 48);
memcpy(b_public->x, b_pub_buf, 48);
memcpy(b_public->y, b_pub_buf + 6, 48);
use_real_getrandom = false;
assert(l_ecdh_generate_shared_secret(a_secret, b_public, &a_shared));
assert(l_ecdh_generate_shared_secret(b_secret, a_public, &b_shared));
assert(!memcmp(a_shared->c, ss_buf, 48));
assert(!memcmp(b_shared->c, ss_buf, 48));
use_real_getrandom = true;
l_ecc_scalar_free(a_secret);
l_ecc_scalar_free(b_secret);
l_ecc_point_free(a_public);
l_ecc_point_free(b_public);
l_ecc_scalar_free(a_shared);
l_ecc_scalar_free(b_shared);
}
int main(int argc, char *argv[])
{
l_test_init(&argc, &argv);
if (l_getrandom_is_supported())
l_test_add("ECDH Basic", test_basic, NULL);
l_test_add("ECDH test vector P256", test_vector_p256, NULL);
l_test_add("ECDH test vector P384", test_vector_p384, NULL);
return l_test_run();
}