-
Notifications
You must be signed in to change notification settings - Fork 34
/
Copy pathtest_keyload.c
178 lines (153 loc) · 6.82 KB
/
test_keyload.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
/*
* This test runs a fairly exhaustive test on the key loading functionality
*
* It runs 3000 parallel working keys of the same private key; working key i
* will be loaded at step i, and then generate signatures from that point;
* for example, at step 3, we'll have three working keys; one which was loaded
* at index 0 (and has produced 3 signatures so far); one which was loaded
* at index 1 (and has produced 2 signatures so far); one which was loaded
* at index 2 (and has only produced one signature). We check that every
* working key generates the same signature at each step (as everything in the
* signatures is based on the private key, the index, and the message, we
* should expect this).
*
* The idea is that we perform a key load from every possible offset in the
* tree; if any of the working key loads didn't perform properly (either
* getting the current auth path wrong, or not setting up the hashes for a
* future path), that key will generate a wrong signature.
*
* It also does a pretty decent test on whether our Merkle tree traversal
* logic is always traversing properly.
*
* It takes 20 minutes to do a full test on my test machine
*/
#include "hss.h"
#include "hash.h"
#include "test_hss.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define PARM_SET LMS_SHA256_N32_H5
#define PARM_SET_2 LMOTS_SHA256_N32_W2
#define LEVELS 3
#define MAX_ITER 3000
#define FAST_ITER 300 /* Number of iterations to run in fast more */
static bool generate_random(void *output, size_t length) {
unsigned char *p = output;
while (length--) {
*p++ = rand() % 256;
}
return true;
}
bool test_key_load(bool fast_flag, bool quiet_flag) {
bool success_flag = false;
param_set_t parm_set[3] = { PARM_SET, PARM_SET, PARM_SET };
param_set_t ots_parm_set[3] = { PARM_SET_2, PARM_SET_2, PARM_SET_2 };
size_t len_private_key = hss_get_private_key_len(LEVELS, parm_set,
ots_parm_set);
if (len_private_key == 0) return false;
unsigned char *private_key = malloc( len_private_key );
unsigned char *orig_private_key = malloc( len_private_key );
unsigned char *copy_private_key = malloc( len_private_key );
if (!private_key || !orig_private_key || !copy_private_key) return false;
unsigned len_public_key = hss_get_public_key_len(LEVELS, parm_set,
ots_parm_set);
if (len_public_key == 0 || len_public_key > HSS_MAX_PUBLIC_KEY_LEN) return false;
unsigned char public_key[ HSS_MAX_PUBLIC_KEY_LEN ];
unsigned char aux_data[2000];
/* Generate the master private key that we'll use for everyone */
if (!hss_generate_private_key( generate_random,
LEVELS, parm_set, ots_parm_set,
NULL, private_key,
public_key, len_public_key,
aux_data, sizeof aux_data, 0)) {
printf( "Public/private key gen failed\n" );
return false;
}
int i;
struct hss_working_key *w[ MAX_ITER+1 ];
for (i = 0; i <= MAX_ITER; i++) w[i] = 0;
unsigned iter;
if (fast_flag) iter = FAST_ITER; else iter = MAX_ITER;
unsigned len_signature = hss_get_signature_len(LEVELS, parm_set, ots_parm_set);
if (len_signature == 0) return false;
unsigned char *signature = malloc( len_signature );
unsigned char *copy_signature = malloc( len_signature );
if (!signature || !copy_signature) return false;
int percent = 0;
for (i=0; i<iter; i++) {
if (!quiet_flag) {
/* Display the running percentage. This function actually has */
/* quadratic behavior, as it checks every working set that's */
/* been created so far; we create a fresh one every iteration. */
/* Out percentage accounts for that. A pet peeve of mine is */
/* progress bars that aren't accurate; make sure this one is */
float new_percent = (100.0 * i * i) / iter / iter;
if (new_percent >= percent+1) {
percent = (int)new_percent;
printf( " %d%% (iter %d)\r", percent, i );
fflush(stdout);
}
}
/* Allocate the next working set */
/* Use more memory for most of the working keys */
/* By varying the memory irregularly (mod 7 is irregular for */
/* a binary-based tree), we have a better change at tickling */
/* a tree-walking bug */
size_t memory_target = (i % 7 == 5) ? 0 : 30000;
/* Create a fresh working set at the current index*/
w[i] = hss_load_private_key( NULL, private_key,
memory_target,
(i % 3 == 1) ? NULL : aux_data, sizeof aux_data, 0 );
if (!w[i]) { printf( "Out of memory\n" ); goto failed; }
memcpy( orig_private_key, private_key, len_private_key );
/* Generate a bunch of signatures of the same text */
char text[ 100 ];
unsigned len_text = sprintf( text, "Message #%d", i );
if (!hss_generate_signature( w[0], NULL, private_key,
text, len_text,
signature, len_signature, 0)) {
printf( "\nMaster generate signature failed\n" );
goto failed;
}
/* If we're doing a regression test, we really have to actually */
/* check the signatures, even if it's not the point of the test */
if (!hss_validate_signature( public_key, text,
len_text, signature, len_signature, 0 )) {
printf( "\nVerify signature failed\n" );
goto failed;
}
/* Now, go through and see if all the other working keys generate */
/* the same signature */
int j;
for (j=1; j<=i; j++) {
memcpy( copy_private_key, orig_private_key, len_private_key );
if (!hss_generate_signature( w[j], NULL, copy_private_key,
text, len_text,
copy_signature, len_signature, 0)) {
printf( "\nGenerate signature %d failed\n", j );
goto failed;
}
/* The signature and the private key should be the same as */
/* the master */
if (0 != memcmp( signature, copy_signature, len_signature )) {
printf( "\nError: signature %d not identical\n", j );
goto failed;
}
if (0 != memcmp( private_key, copy_private_key, len_private_key )) {
printf( "\nError: private_key %d not identical\n", j );
goto failed;
}
}
}
success_flag = true;
failed:
if (!quiet_flag) printf( "\n" );
free(private_key);
free(orig_private_key);
free(copy_private_key);
free(signature);
free(copy_signature);
for (i = 0; i <= MAX_ITER; i++) hss_free_working_key(w[i]);
return success_flag;
}