Mercurial > dropbear
comparison src/pk/ecc/ecc_sys.c @ 280:59400faa4b44 libtomcrypt-orig libtomcrypt-1.05
Re-import libtomcrypt 1.05 for cleaner propagating.
From crypt-1.05.tar.bz2, SHA1 of 88250202bb51570dc64f7e8f1c943cda9479258f
author | Matt Johnston <matt@ucc.asn.au> |
---|---|
date | Wed, 08 Mar 2006 12:58:00 +0000 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 280:59400faa4b44 |
---|---|
1 /* LibTomCrypt, modular cryptographic library -- Tom St Denis | |
2 * | |
3 * LibTomCrypt is a library that provides various cryptographic | |
4 * algorithms in a highly modular and flexible manner. | |
5 * | |
6 * The library is free for all purposes without any express | |
7 * guarantee it works. | |
8 * | |
9 * Tom St Denis, [email protected], http://libtomcrypt.org | |
10 */ | |
11 | |
12 /** | |
13 @file ecc_sys.c | |
14 ECC Crypto, Tom St Denis | |
15 */ | |
16 | |
17 /** | |
18 Encrypt a symmetric key with ECC | |
19 @param in The symmetric key you want to encrypt | |
20 @param inlen The length of the key to encrypt (octets) | |
21 @param out [out] The destination for the ciphertext | |
22 @param outlen [in/out] The max size and resulting size of the ciphertext | |
23 @param prng An active PRNG state | |
24 @param wprng The index of the PRNG you wish to use | |
25 @param hash The index of the hash you want to use | |
26 @param key The ECC key you want to encrypt to | |
27 @return CRYPT_OK if successful | |
28 */ | |
29 int ecc_encrypt_key(const unsigned char *in, unsigned long inlen, | |
30 unsigned char *out, unsigned long *outlen, | |
31 prng_state *prng, int wprng, int hash, | |
32 ecc_key *key) | |
33 { | |
34 unsigned char *pub_expt, *ecc_shared, *skey; | |
35 ecc_key pubkey; | |
36 unsigned long x, y, pubkeysize; | |
37 int err; | |
38 | |
39 LTC_ARGCHK(in != NULL); | |
40 LTC_ARGCHK(out != NULL); | |
41 LTC_ARGCHK(outlen != NULL); | |
42 LTC_ARGCHK(key != NULL); | |
43 | |
44 /* check that wprng/cipher/hash are not invalid */ | |
45 if ((err = prng_is_valid(wprng)) != CRYPT_OK) { | |
46 return err; | |
47 } | |
48 | |
49 if ((err = hash_is_valid(hash)) != CRYPT_OK) { | |
50 return err; | |
51 } | |
52 | |
53 if (inlen > hash_descriptor[hash].hashsize) { | |
54 return CRYPT_INVALID_HASH; | |
55 } | |
56 | |
57 /* make a random key and export the public copy */ | |
58 if ((err = ecc_make_key(prng, wprng, ecc_get_size(key), &pubkey)) != CRYPT_OK) { | |
59 return err; | |
60 } | |
61 | |
62 pub_expt = XMALLOC(ECC_BUF_SIZE); | |
63 ecc_shared = XMALLOC(ECC_BUF_SIZE); | |
64 skey = XMALLOC(MAXBLOCKSIZE); | |
65 if (pub_expt == NULL || ecc_shared == NULL || skey == NULL) { | |
66 if (pub_expt != NULL) { | |
67 XFREE(pub_expt); | |
68 } | |
69 if (ecc_shared != NULL) { | |
70 XFREE(ecc_shared); | |
71 } | |
72 if (skey != NULL) { | |
73 XFREE(skey); | |
74 } | |
75 ecc_free(&pubkey); | |
76 return CRYPT_MEM; | |
77 } | |
78 | |
79 pubkeysize = ECC_BUF_SIZE; | |
80 if ((err = ecc_export(pub_expt, &pubkeysize, PK_PUBLIC, &pubkey)) != CRYPT_OK) { | |
81 ecc_free(&pubkey); | |
82 goto LBL_ERR; | |
83 } | |
84 | |
85 /* make random key */ | |
86 x = ECC_BUF_SIZE; | |
87 if ((err = ecc_shared_secret(&pubkey, key, ecc_shared, &x)) != CRYPT_OK) { | |
88 ecc_free(&pubkey); | |
89 goto LBL_ERR; | |
90 } | |
91 ecc_free(&pubkey); | |
92 y = MAXBLOCKSIZE; | |
93 if ((err = hash_memory(hash, ecc_shared, x, skey, &y)) != CRYPT_OK) { | |
94 goto LBL_ERR; | |
95 } | |
96 | |
97 /* Encrypt key */ | |
98 for (x = 0; x < inlen; x++) { | |
99 skey[x] ^= in[x]; | |
100 } | |
101 | |
102 err = der_encode_sequence_multi(out, outlen, | |
103 LTC_ASN1_OBJECT_IDENTIFIER, hash_descriptor[hash].OIDlen, hash_descriptor[hash].OID, | |
104 LTC_ASN1_OCTET_STRING, pubkeysize, pub_expt, | |
105 LTC_ASN1_OCTET_STRING, inlen, skey, | |
106 LTC_ASN1_EOL, 0UL, NULL); | |
107 | |
108 LBL_ERR: | |
109 #ifdef LTC_CLEAN_STACK | |
110 /* clean up */ | |
111 zeromem(pub_expt, ECC_BUF_SIZE); | |
112 zeromem(ecc_shared, ECC_BUF_SIZE); | |
113 zeromem(skey, MAXBLOCKSIZE); | |
114 #endif | |
115 | |
116 XFREE(skey); | |
117 XFREE(ecc_shared); | |
118 XFREE(pub_expt); | |
119 | |
120 return err; | |
121 } | |
122 | |
123 /** | |
124 Decrypt an ECC encrypted key | |
125 @param in The ciphertext | |
126 @param inlen The length of the ciphertext (octets) | |
127 @param out [out] The plaintext | |
128 @param outlen [in/out] The max size and resulting size of the plaintext | |
129 @param key The corresponding private ECC key | |
130 @return CRYPT_OK if successful | |
131 */ | |
132 int ecc_decrypt_key(const unsigned char *in, unsigned long inlen, | |
133 unsigned char *out, unsigned long *outlen, | |
134 ecc_key *key) | |
135 { | |
136 unsigned char *ecc_shared, *skey, *pub_expt; | |
137 unsigned long x, y, hashOID[32]; | |
138 int hash, err; | |
139 ecc_key pubkey; | |
140 ltc_asn1_list decode[3]; | |
141 | |
142 LTC_ARGCHK(in != NULL); | |
143 LTC_ARGCHK(out != NULL); | |
144 LTC_ARGCHK(outlen != NULL); | |
145 LTC_ARGCHK(key != NULL); | |
146 | |
147 /* right key type? */ | |
148 if (key->type != PK_PRIVATE) { | |
149 return CRYPT_PK_NOT_PRIVATE; | |
150 } | |
151 | |
152 /* decode to find out hash */ | |
153 LTC_SET_ASN1(decode, 0, LTC_ASN1_OBJECT_IDENTIFIER, hashOID, sizeof(hashOID)/sizeof(hashOID[0])); | |
154 | |
155 if ((err = der_decode_sequence(in, inlen, decode, 1)) != CRYPT_OK) { | |
156 return err; | |
157 } | |
158 for (hash = 0; hash_descriptor[hash].name != NULL && | |
159 (hash_descriptor[hash].OIDlen != decode[0].size || | |
160 memcmp(hash_descriptor[hash].OID, hashOID, sizeof(unsigned long)*decode[0].size)); hash++); | |
161 | |
162 if (hash_descriptor[hash].name == NULL) { | |
163 return CRYPT_INVALID_PACKET; | |
164 } | |
165 | |
166 /* we now have the hash! */ | |
167 | |
168 /* allocate memory */ | |
169 pub_expt = XMALLOC(ECC_BUF_SIZE); | |
170 ecc_shared = XMALLOC(ECC_BUF_SIZE); | |
171 skey = XMALLOC(MAXBLOCKSIZE); | |
172 if (pub_expt == NULL || ecc_shared == NULL || skey == NULL) { | |
173 if (pub_expt != NULL) { | |
174 XFREE(pub_expt); | |
175 } | |
176 if (ecc_shared != NULL) { | |
177 XFREE(ecc_shared); | |
178 } | |
179 if (skey != NULL) { | |
180 XFREE(skey); | |
181 } | |
182 return CRYPT_MEM; | |
183 } | |
184 LTC_SET_ASN1(decode, 1, LTC_ASN1_OCTET_STRING, pub_expt, ECC_BUF_SIZE); | |
185 LTC_SET_ASN1(decode, 2, LTC_ASN1_OCTET_STRING, skey, MAXBLOCKSIZE); | |
186 | |
187 /* read the structure in now */ | |
188 if ((err = der_decode_sequence(in, inlen, decode, 3)) != CRYPT_OK) { | |
189 goto LBL_ERR; | |
190 } | |
191 | |
192 /* import ECC key from packet */ | |
193 if ((err = ecc_import(decode[1].data, decode[1].size, &pubkey)) != CRYPT_OK) { | |
194 goto LBL_ERR; | |
195 } | |
196 | |
197 /* make shared key */ | |
198 x = ECC_BUF_SIZE; | |
199 if ((err = ecc_shared_secret(key, &pubkey, ecc_shared, &x)) != CRYPT_OK) { | |
200 ecc_free(&pubkey); | |
201 goto LBL_ERR; | |
202 } | |
203 ecc_free(&pubkey); | |
204 | |
205 y = MAXBLOCKSIZE; | |
206 if ((err = hash_memory(hash, ecc_shared, x, ecc_shared, &y)) != CRYPT_OK) { | |
207 goto LBL_ERR; | |
208 } | |
209 | |
210 /* ensure the hash of the shared secret is at least as big as the encrypt itself */ | |
211 if (decode[2].size > y) { | |
212 err = CRYPT_INVALID_PACKET; | |
213 goto LBL_ERR; | |
214 } | |
215 | |
216 /* avoid buffer overflow */ | |
217 if (*outlen < decode[2].size) { | |
218 err = CRYPT_BUFFER_OVERFLOW; | |
219 goto LBL_ERR; | |
220 } | |
221 | |
222 /* Decrypt the key */ | |
223 for (x = 0; x < decode[2].size; x++) { | |
224 out[x] = skey[x] ^ ecc_shared[x]; | |
225 } | |
226 *outlen = x; | |
227 | |
228 err = CRYPT_OK; | |
229 LBL_ERR: | |
230 #ifdef LTC_CLEAN_STACK | |
231 zeromem(pub_expt, ECC_BUF_SIZE); | |
232 zeromem(ecc_shared, ECC_BUF_SIZE); | |
233 zeromem(skey, MAXBLOCKSIZE); | |
234 #endif | |
235 | |
236 XFREE(pub_expt); | |
237 XFREE(ecc_shared); | |
238 XFREE(skey); | |
239 | |
240 return err; | |
241 } | |
242 | |
243 /** | |
244 Sign a message digest | |
245 @param in The message digest to sign | |
246 @param inlen The length of the digest | |
247 @param out [out] The destination for the signature | |
248 @param outlen [in/out] The max size and resulting size of the signature | |
249 @param prng An active PRNG state | |
250 @param wprng The index of the PRNG you wish to use | |
251 @param key A private ECC key | |
252 @return CRYPT_OK if successful | |
253 */ | |
254 int ecc_sign_hash(const unsigned char *in, unsigned long inlen, | |
255 unsigned char *out, unsigned long *outlen, | |
256 prng_state *prng, int wprng, ecc_key *key) | |
257 { | |
258 ecc_key pubkey; | |
259 mp_int r, s, e, p; | |
260 int err; | |
261 | |
262 LTC_ARGCHK(in != NULL); | |
263 LTC_ARGCHK(out != NULL); | |
264 LTC_ARGCHK(outlen != NULL); | |
265 LTC_ARGCHK(key != NULL); | |
266 | |
267 /* is this a private key? */ | |
268 if (key->type != PK_PRIVATE) { | |
269 return CRYPT_PK_NOT_PRIVATE; | |
270 } | |
271 | |
272 /* is the IDX valid ? */ | |
273 if (is_valid_idx(key->idx) != 1) { | |
274 return CRYPT_PK_INVALID_TYPE; | |
275 } | |
276 | |
277 if ((err = prng_is_valid(wprng)) != CRYPT_OK) { | |
278 return err; | |
279 } | |
280 | |
281 /* get the hash and load it as a bignum into 'e' */ | |
282 /* init the bignums */ | |
283 if ((err = mp_init_multi(&r, &s, &p, &e, NULL)) != MP_OKAY) { | |
284 ecc_free(&pubkey); | |
285 err = mpi_to_ltc_error(err); | |
286 goto LBL_ERR; | |
287 } | |
288 if ((err = mp_read_radix(&p, (char *)sets[key->idx].order, 64)) != MP_OKAY) { goto error; } | |
289 if ((err = mp_read_unsigned_bin(&e, (unsigned char *)in, (int)inlen)) != MP_OKAY) { goto error; } | |
290 | |
291 /* make up a key and export the public copy */ | |
292 for (;;) { | |
293 if ((err = ecc_make_key(prng, wprng, ecc_get_size(key), &pubkey)) != CRYPT_OK) { | |
294 return err; | |
295 } | |
296 | |
297 /* find r = x1 mod n */ | |
298 if ((err = mp_mod(&pubkey.pubkey.x, &p, &r)) != MP_OKAY) { goto error; } | |
299 | |
300 if (mp_iszero(&r)) { | |
301 ecc_free(&pubkey); | |
302 } else { | |
303 /* find s = (e + xr)/k */ | |
304 if ((err = mp_invmod(&pubkey.k, &p, &pubkey.k)) != MP_OKAY) { goto error; } /* k = 1/k */ | |
305 if ((err = mp_mulmod(&key->k, &r, &p, &s)) != MP_OKAY) { goto error; } /* s = xr */ | |
306 if ((err = mp_addmod(&e, &s, &p, &s)) != MP_OKAY) { goto error; } /* s = e + xr */ | |
307 if ((err = mp_mulmod(&s, &pubkey.k, &p, &s)) != MP_OKAY) { goto error; } /* s = (e + xr)/k */ | |
308 | |
309 if (mp_iszero(&s)) { | |
310 ecc_free(&pubkey); | |
311 } else { | |
312 break; | |
313 } | |
314 } | |
315 } | |
316 | |
317 /* store as SEQUENCE { r, s -- integer } */ | |
318 err = der_encode_sequence_multi(out, outlen, | |
319 LTC_ASN1_INTEGER, 1UL, &r, | |
320 LTC_ASN1_INTEGER, 1UL, &s, | |
321 LTC_ASN1_EOL, 0UL, NULL); | |
322 goto LBL_ERR; | |
323 error: | |
324 err = mpi_to_ltc_error(err); | |
325 LBL_ERR: | |
326 mp_clear_multi(&r, &s, &p, &e, NULL); | |
327 ecc_free(&pubkey); | |
328 | |
329 return err; | |
330 } | |
331 | |
332 /* verify | |
333 * | |
334 * w = s^-1 mod n | |
335 * u1 = xw | |
336 * u2 = rw | |
337 * X = u1*G + u2*Q | |
338 * v = X_x1 mod n | |
339 * accept if v == r | |
340 */ | |
341 | |
342 /** | |
343 Verify an ECC signature | |
344 @param sig The signature to verify | |
345 @param siglen The length of the signature (octets) | |
346 @param hash The hash (message digest) that was signed | |
347 @param hashlen The length of the hash (octets) | |
348 @param stat Result of signature, 1==valid, 0==invalid | |
349 @param key The corresponding public ECC key | |
350 @return CRYPT_OK if successful (even if the signature is not valid) | |
351 */ | |
352 int ecc_verify_hash(const unsigned char *sig, unsigned long siglen, | |
353 const unsigned char *hash, unsigned long hashlen, | |
354 int *stat, ecc_key *key) | |
355 { | |
356 ecc_point *mG, *mQ; | |
357 mp_int r, s, v, w, u1, u2, e, p, m; | |
358 mp_digit mp; | |
359 int err; | |
360 | |
361 LTC_ARGCHK(sig != NULL); | |
362 LTC_ARGCHK(hash != NULL); | |
363 LTC_ARGCHK(stat != NULL); | |
364 LTC_ARGCHK(key != NULL); | |
365 | |
366 /* default to invalid signature */ | |
367 *stat = 0; | |
368 | |
369 /* is the IDX valid ? */ | |
370 if (is_valid_idx(key->idx) != 1) { | |
371 return CRYPT_PK_INVALID_TYPE; | |
372 } | |
373 | |
374 /* allocate ints */ | |
375 if ((err = mp_init_multi(&r, &s, &v, &w, &u1, &u2, &p, &e, &m, NULL)) != MP_OKAY) { | |
376 return CRYPT_MEM; | |
377 } | |
378 | |
379 /* allocate points */ | |
380 mG = new_point(); | |
381 mQ = new_point(); | |
382 if (mQ == NULL || mG == NULL) { | |
383 err = CRYPT_MEM; | |
384 goto done; | |
385 } | |
386 | |
387 /* parse header */ | |
388 if ((err = der_decode_sequence_multi(sig, siglen, | |
389 LTC_ASN1_INTEGER, 1UL, &r, | |
390 LTC_ASN1_INTEGER, 1UL, &s, | |
391 LTC_ASN1_EOL, 0UL, NULL)) != CRYPT_OK) { | |
392 goto done; | |
393 } | |
394 | |
395 /* get the order */ | |
396 if ((err = mp_read_radix(&p, (char *)sets[key->idx].order, 64)) != MP_OKAY) { goto error; } | |
397 | |
398 /* get the modulus */ | |
399 if ((err = mp_read_radix(&m, (char *)sets[key->idx].prime, 64)) != MP_OKAY) { goto error; } | |
400 | |
401 /* check for zero */ | |
402 if (mp_iszero(&r) || mp_iszero(&s) || mp_cmp(&r, &p) != MP_LT || mp_cmp(&s, &p) != MP_LT) { | |
403 err = CRYPT_INVALID_PACKET; | |
404 goto done; | |
405 } | |
406 | |
407 /* read hash */ | |
408 if ((err = mp_read_unsigned_bin(&e, (unsigned char *)hash, (int)hashlen)) != MP_OKAY) { goto error; } | |
409 | |
410 /* w = s^-1 mod n */ | |
411 if ((err = mp_invmod(&s, &p, &w)) != MP_OKAY) { goto error; } | |
412 | |
413 /* u1 = ew */ | |
414 if ((err = mp_mulmod(&e, &w, &p, &u1)) != MP_OKAY) { goto error; } | |
415 | |
416 /* u2 = rw */ | |
417 if ((err = mp_mulmod(&r, &w, &p, &u2)) != MP_OKAY) { goto error; } | |
418 | |
419 /* find mG = u1*G */ | |
420 if ((err = mp_read_radix(&mG->x, (char *)sets[key->idx].Gx, 64)) != MP_OKAY) { goto error; } | |
421 if ((err = mp_read_radix(&mG->y, (char *)sets[key->idx].Gy, 64)) != MP_OKAY) { goto error; } | |
422 mp_set(&mG->z, 1); | |
423 if ((err = ecc_mulmod(&u1, mG, mG, &m, 0)) != CRYPT_OK) { goto done; } | |
424 | |
425 /* find mQ = u2*Q */ | |
426 if ((err = mp_copy(&key->pubkey.x, &mQ->x)) != MP_OKAY) { goto error; } | |
427 if ((err = mp_copy(&key->pubkey.y, &mQ->y)) != MP_OKAY) { goto error; } | |
428 if ((err = mp_copy(&key->pubkey.z, &mQ->z)) != MP_OKAY) { goto error; } | |
429 if ((err = ecc_mulmod(&u2, mQ, mQ, &m, 0)) != CRYPT_OK) { goto done; } | |
430 | |
431 /* find the montgomery mp */ | |
432 if ((err = mp_montgomery_setup(&m, &mp)) != MP_OKAY) { goto error; } | |
433 /* add them */ | |
434 if ((err = add_point(mQ, mG, mG, &m, mp)) != CRYPT_OK) { goto done; } | |
435 | |
436 /* reduce */ | |
437 if ((err = ecc_map(mG, &m, mp)) != CRYPT_OK) { goto done; } | |
438 | |
439 /* v = X_x1 mod n */ | |
440 if ((err = mp_mod(&mG->x, &p, &v)) != CRYPT_OK) { goto done; } | |
441 | |
442 /* does v == r */ | |
443 if (mp_cmp(&v, &r) == MP_EQ) { | |
444 *stat = 1; | |
445 } | |
446 | |
447 /* clear up and return */ | |
448 err = CRYPT_OK; | |
449 goto done; | |
450 error: | |
451 err = mpi_to_ltc_error(err); | |
452 done: | |
453 del_point(mG); | |
454 del_point(mQ); | |
455 mp_clear_multi(&r, &s, &v, &w, &u1, &u2, &p, &e, &m, NULL); | |
456 return err; | |
457 } | |
458 | |
459 | |
460 /* $Source: /cvs/libtom/libtomcrypt/src/pk/ecc/ecc_sys.c,v $ */ | |
461 /* $Revision: 1.18 $ */ | |
462 /* $Date: 2005/06/14 20:47:55 $ */ |