tomcrypt/testprof/ecc_test.c

248 lines
6.6 KiB
C
Raw Normal View History

2005-04-17 07:37:13 -04:00
#include <tomcrypt_test.h>
2004-05-30 22:36:47 -04:00
2007-07-20 13:48:02 -04:00
#ifdef LTC_MECC
2004-10-29 23:00:26 -04:00
2014-05-01 12:01:13 -04:00
static unsigned int sizes[] = {
#ifdef LTC_ECC112
2006-11-17 09:21:24 -05:00
14,
#endif
#ifdef LTC_ECC128
2006-11-17 09:21:24 -05:00
16,
#endif
#ifdef LTC_ECC160
2006-11-17 09:21:24 -05:00
20,
#endif
#ifdef LTC_ECC192
2005-06-08 20:08:13 -04:00
24,
#endif
#ifdef LTC_ECC224
2005-06-08 20:08:13 -04:00
28,
#endif
#ifdef LTC_ECC256
2005-06-08 20:08:13 -04:00
32,
#endif
#ifdef LTC_ECC384
2005-06-08 20:08:13 -04:00
48,
#endif
#ifdef LTC_ECC521
2005-06-08 20:08:13 -04:00
65
#endif
};
2006-12-16 13:10:04 -05:00
#ifdef LTC_ECC_SHAMIR
int ecc_test_shamir(void)
{
void *modulus, *mp, *kA, *kB, *rA, *rB;
ecc_point *G, *A, *B, *C1, *C2;
int x, y, z;
unsigned char buf[ECC_BUF_SIZE];
DO(mp_init_multi(&kA, &kB, &rA, &rB, &modulus, NULL));
LTC_ARGCHK((G = ltc_ecc_new_point()) != NULL);
LTC_ARGCHK((A = ltc_ecc_new_point()) != NULL);
LTC_ARGCHK((B = ltc_ecc_new_point()) != NULL);
LTC_ARGCHK((C1 = ltc_ecc_new_point()) != NULL);
LTC_ARGCHK((C2 = ltc_ecc_new_point()) != NULL);
for (x = 0; x < (int)(sizeof(sizes)/sizeof(sizes[0])); x++) {
/* get the base point */
for (z = 0; ltc_ecc_sets[z].name; z++) {
2014-05-01 12:01:13 -04:00
if (sizes[z] < (unsigned int)ltc_ecc_sets[z].size) break;
2006-12-16 13:10:04 -05:00
}
LTC_ARGCHK(ltc_ecc_sets[z].name != NULL);
/* load it */
DO(mp_read_radix(G->x, ltc_ecc_sets[z].Gx, 16));
DO(mp_read_radix(G->y, ltc_ecc_sets[z].Gy, 16));
DO(mp_set(G->z, 1));
DO(mp_read_radix(modulus, ltc_ecc_sets[z].prime, 16));
DO(mp_montgomery_setup(modulus, &mp));
/* do 100 random tests */
for (y = 0; y < 100; y++) {
/* pick a random r1, r2 */
LTC_ARGCHK(yarrow_read(buf, sizes[x], &yarrow_prng) == sizes[x]);
DO(mp_read_unsigned_bin(rA, buf, sizes[x]));
LTC_ARGCHK(yarrow_read(buf, sizes[x], &yarrow_prng) == sizes[x]);
DO(mp_read_unsigned_bin(rB, buf, sizes[x]));
/* compute rA * G = A */
DO(ltc_mp.ecc_ptmul(rA, G, A, modulus, 1));
2014-05-01 12:01:13 -04:00
2006-12-16 13:10:04 -05:00
/* compute rB * G = B */
DO(ltc_mp.ecc_ptmul(rB, G, B, modulus, 1));
/* pick a random kA, kB */
LTC_ARGCHK(yarrow_read(buf, sizes[x], &yarrow_prng) == sizes[x]);
DO(mp_read_unsigned_bin(kA, buf, sizes[x]));
LTC_ARGCHK(yarrow_read(buf, sizes[x], &yarrow_prng) == sizes[x]);
DO(mp_read_unsigned_bin(kB, buf, sizes[x]));
/* now, compute kA*A + kB*B = C1 using the older method */
DO(ltc_mp.ecc_ptmul(kA, A, C1, modulus, 0));
DO(ltc_mp.ecc_ptmul(kB, B, C2, modulus, 0));
DO(ltc_mp.ecc_ptadd(C1, C2, C1, modulus, mp));
DO(ltc_mp.ecc_map(C1, modulus, mp));
/* now compute using mul2add */
DO(ltc_mp.ecc_mul2add(A, kA, B, kB, C2, modulus));
/* is they the sames? */
if ((mp_cmp(C1->x, C2->x) != LTC_MP_EQ) || (mp_cmp(C1->y, C2->y) != LTC_MP_EQ) || (mp_cmp(C1->z, C2->z) != LTC_MP_EQ)) {
fprintf(stderr, "ECC failed shamir test: size=%d, testno=%d\n", sizes[x], y);
return 1;
}
}
mp_montgomery_free(mp);
}
ltc_ecc_del_point(C2);
ltc_ecc_del_point(C1);
ltc_ecc_del_point(B);
ltc_ecc_del_point(A);
ltc_ecc_del_point(G);
mp_clear_multi(kA, kB, rA, rB, modulus, NULL);
return 0;
}
#endif
2004-05-30 22:36:47 -04:00
int ecc_tests (void)
{
2017-05-09 16:05:30 -04:00
unsigned char buf[4][4096], ch;
2005-06-08 20:08:13 -04:00
unsigned long x, y, z, s;
2004-05-30 22:36:47 -04:00
int stat, stat2;
2004-12-30 18:55:53 -05:00
ecc_key usera, userb, pubKey, privKey;
2014-05-01 12:01:13 -04:00
2004-05-30 22:36:47 -04:00
DO(ecc_test ());
2005-11-18 00:15:37 -05:00
for (s = 0; s < (sizeof(sizes)/sizeof(sizes[0])); s++) {
2005-06-08 20:08:13 -04:00
/* make up two keys */
DO(ecc_make_key (&yarrow_prng, find_prng ("yarrow"), sizes[s], &usera));
DO(ecc_make_key (&yarrow_prng, find_prng ("yarrow"), sizes[s], &userb));
/* make the shared secret */
2006-11-17 09:21:24 -05:00
x = sizeof(buf[0]);
2005-06-08 20:08:13 -04:00
DO(ecc_shared_secret (&usera, &userb, buf[0], &x));
2006-11-17 09:21:24 -05:00
y = sizeof(buf[1]);
2005-06-08 20:08:13 -04:00
DO(ecc_shared_secret (&userb, &usera, buf[1], &y));
if (y != x) {
fprintf(stderr, "ecc Shared keys are not same size.");
return 1;
}
if (memcmp (buf[0], buf[1], x)) {
fprintf(stderr, "ecc Shared keys not same contents.");
return 1;
}
/* now export userb */
2006-11-17 09:21:24 -05:00
y = sizeof(buf[0]);
2005-06-08 20:08:13 -04:00
DO(ecc_export (buf[1], &y, PK_PUBLIC, &userb));
ecc_free (&userb);
/* import and make the shared secret again */
DO(ecc_import (buf[1], y, &userb));
2006-11-17 09:21:24 -05:00
z = sizeof(buf[0]);
2005-06-08 20:08:13 -04:00
DO(ecc_shared_secret (&usera, &userb, buf[2], &z));
if (z != x) {
fprintf(stderr, "failed. Size don't match?");
return 1;
}
if (memcmp (buf[0], buf[2], x)) {
2005-06-14 18:16:11 -04:00
fprintf(stderr, "Failed. Contents didn't match.");
2005-06-08 20:08:13 -04:00
return 1;
}
2006-11-17 09:21:24 -05:00
/* export with ANSI X9.63 */
y = sizeof(buf[1]);
DO(ecc_ansi_x963_export(&userb, buf[1], &y));
ecc_free (&userb);
/* now import the ANSI key */
DO(ecc_ansi_x963_import(buf[1], y, &userb));
/* shared secret */
z = sizeof(buf[0]);
DO(ecc_shared_secret (&usera, &userb, buf[2], &z));
if (z != x) {
fprintf(stderr, "failed. Size don't match?");
return 1;
}
if (memcmp (buf[0], buf[2], x)) {
fprintf(stderr, "Failed. Contents didn't match.");
return 1;
}
2005-06-08 20:08:13 -04:00
ecc_free (&usera);
ecc_free (&userb);
/* test encrypt_key */
DO(ecc_make_key (&yarrow_prng, find_prng ("yarrow"), sizes[s], &usera));
/* export key */
x = sizeof(buf[0]);
DO(ecc_export(buf[0], &x, PK_PUBLIC, &usera));
DO(ecc_import(buf[0], x, &pubKey));
x = sizeof(buf[0]);
DO(ecc_export(buf[0], &x, PK_PRIVATE, &usera));
DO(ecc_import(buf[0], x, &privKey));
2017-05-09 16:05:30 -04:00
for (ch = 0; ch < 32; ch++) {
buf[0][ch] = ch;
2005-06-08 20:08:13 -04:00
}
y = sizeof (buf[1]);
DO(ecc_encrypt_key (buf[0], 32, buf[1], &y, &yarrow_prng, find_prng ("yarrow"), find_hash ("sha256"), &pubKey));
zeromem (buf[0], sizeof (buf[0]));
x = sizeof (buf[0]);
DO(ecc_decrypt_key (buf[1], y, buf[0], &x, &privKey));
if (x != 32) {
fprintf(stderr, "Failed (length)");
return 1;
}
2017-05-09 16:05:30 -04:00
for (ch = 0; ch < 32; ch++) {
if (buf[0][ch] != ch) {
2005-06-08 20:08:13 -04:00
fprintf(stderr, "Failed (contents)");
return 1;
}
}
/* test sign_hash */
2017-05-09 16:05:30 -04:00
for (ch = 0; ch < 16; ch++) {
buf[0][ch] = ch;
2005-06-08 20:08:13 -04:00
}
x = sizeof (buf[1]);
DO(ecc_sign_hash (buf[0], 16, buf[1], &x, &yarrow_prng, find_prng ("yarrow"), &privKey));
DO(ecc_verify_hash (buf[1], x, buf[0], 16, &stat, &pubKey));
buf[0][0] ^= 1;
DO(ecc_verify_hash (buf[1], x, buf[0], 16, &stat2, &privKey));
2014-05-01 12:01:13 -04:00
if (!(stat == 1 && stat2 == 0)) {
2005-06-08 20:08:13 -04:00
fprintf(stderr, "ecc_verify_hash failed %d, %d, ", stat, stat2);
return 1;
}
2014-05-01 12:01:13 -04:00
ecc_free (&usera);
2005-06-08 20:08:13 -04:00
ecc_free (&pubKey);
ecc_free (&privKey);
2004-05-30 22:36:47 -04:00
}
2006-12-16 13:10:04 -05:00
#ifdef LTC_ECC_SHAMIR
return ecc_test_shamir();
#else
2004-05-30 22:36:47 -04:00
return 0;
2006-12-16 13:10:04 -05:00
#endif
2004-05-30 22:36:47 -04:00
}
2004-10-29 23:00:26 -04:00
#else
int ecc_tests(void)
{
return CRYPT_NOP;
2004-10-29 23:00:26 -04:00
}
#endif
2005-06-08 20:08:13 -04:00
/* $Source$ */
/* $Revision$ */
/* $Date$ */