dsa_gen.c 16.0 KB
Newer Older
1
/* crypto/dsa/dsa_gen.c */
2
/* Copyright (C) 1995-1998 Eric Young (eay@cryptsoft.com)
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
 * All rights reserved.
 *
 * This package is an SSL implementation written
 * by Eric Young (eay@cryptsoft.com).
 * The implementation was written so as to conform with Netscapes SSL.
 * 
 * This library is free for commercial and non-commercial use as long as
 * the following conditions are aheared to.  The following conditions
 * apply to all code found in this distribution, be it the RC4, RSA,
 * lhash, DES, etc., code; not just the SSL code.  The SSL documentation
 * included with this distribution is covered by the same copyright terms
 * except that the holder is Tim Hudson (tjh@cryptsoft.com).
 * 
 * Copyright remains Eric Young's, and as such any Copyright notices in
 * the code are not to be removed.
 * If this package is used in a product, Eric Young should be given attribution
 * as the author of the parts of the library used.
 * This can be in the form of a textual message at program startup or
 * in documentation (online or textual) provided with the package.
 * 
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. All advertising materials mentioning features or use of this software
 *    must display the following acknowledgement:
 *    "This product includes cryptographic software written by
 *     Eric Young (eay@cryptsoft.com)"
 *    The word 'cryptographic' can be left out if the rouines from the library
 *    being used are not cryptographic related :-).
 * 4. If you include any Windows specific code (or a derivative thereof) from 
 *    the apps directory (application code) you must include an acknowledgement:
 *    "This product includes software written by Tim Hudson (tjh@cryptsoft.com)"
 * 
 * THIS SOFTWARE IS PROVIDED BY ERIC YOUNG ``AS IS'' AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 * 
 * The licence and distribution terms for any publically available version or
 * derivative of this code cannot be changed.  i.e. this code cannot simply be
 * copied and put under another distribution licence
 * [including the GNU Public Licence.]
 */

#undef GENUINE_DSA

#ifdef GENUINE_DSA
62 63
/* Parameter generation follows the original release of FIPS PUB 186,
 * Appendix 2.2 (i.e. use SHA as defined in FIPS PUB 180) */
D
 
Dr. Stephen Henson 已提交
64
#define HASH    EVP_sha()
65
#else
66 67 68
/* Parameter generation follows the updated Appendix 2.2 for FIPS PUB 186,
 * also Appendix 2.2 of FIPS PUB 186-1 (i.e. use SHA as defined in
 * FIPS PUB 180-1) */
D
 
Dr. Stephen Henson 已提交
69
#define HASH    EVP_sha1()
70 71
#endif 

72 73
#include <openssl/opensslconf.h> /* To see if OPENSSL_NO_SHA is defined */

74
#ifndef OPENSSL_NO_SHA
75

D
Dr. Stephen Henson 已提交
76

D
Dr. Stephen Henson 已提交
77

78 79
#include <stdio.h>
#include "cryptlib.h"
D
 
Dr. Stephen Henson 已提交
80
#include <openssl/evp.h>
81 82
#include <openssl/bn.h>
#include <openssl/rand.h>
83
#include <openssl/sha.h>
D
Dr. Stephen Henson 已提交
84

N
Nils Larsch 已提交
85
#include "dsa_locl.h"
86

87
int DSA_generate_parameters_ex(DSA *ret, int bits,
88
		const unsigned char *seed_in, int seed_len,
89
		int *counter_ret, unsigned long *h_ret, BN_GENCB *cb)
90
	{
91 92 93
	if(ret->meth->dsa_paramgen)
		return ret->meth->dsa_paramgen(ret, bits, seed_in, seed_len,
				counter_ret, h_ret, cb);
N
Nils Larsch 已提交
94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
	else
		{
		const EVP_MD *evpmd;
		size_t qbits = bits >= 2048 ? 256 : 160;

		if (bits >= 2048)
			{
			qbits = 256;
			evpmd = EVP_sha256();
			}
		else
			{
			qbits = 160;
			evpmd = EVP_sha1();
			}

		return dsa_builtin_paramgen(ret, bits, qbits, evpmd,
111
			seed_in, seed_len, NULL, counter_ret, h_ret, cb);
N
Nils Larsch 已提交
112
		}
113 114
	}

N
Nils Larsch 已提交
115
int dsa_builtin_paramgen(DSA *ret, size_t bits, size_t qbits,
116
	const EVP_MD *evpmd, const unsigned char *seed_in, size_t seed_len,
117
	unsigned char *seed_out,
N
Nils Larsch 已提交
118
	int *counter_ret, unsigned long *h_ret, BN_GENCB *cb)
119
	{
120
	int ok=0;
N
Nils Larsch 已提交
121 122 123
	unsigned char seed[SHA256_DIGEST_LENGTH];
	unsigned char md[SHA256_DIGEST_LENGTH];
	unsigned char buf[SHA256_DIGEST_LENGTH],buf2[SHA256_DIGEST_LENGTH];
124 125
	BIGNUM *r0,*W,*X,*c,*test;
	BIGNUM *g=NULL,*q=NULL,*p=NULL;
126
	BN_MONT_CTX *mont=NULL;
B
Ben Laurie 已提交
127
	int i, k, n=0, m=0, qsize = qbits >> 3;
128
	int counter=0;
129
	int r=0;
130
	BN_CTX *ctx=NULL;
131 132
	unsigned int h=2;

N
Nils Larsch 已提交
133 134 135 136
	if (qsize != SHA_DIGEST_LENGTH && qsize != SHA224_DIGEST_LENGTH &&
	    qsize != SHA256_DIGEST_LENGTH)
		/* invalid q size */
		return 0;
137

N
Nils Larsch 已提交
138 139 140
	if (evpmd == NULL)
		/* use SHA1 as default */
		evpmd = EVP_sha1();
141

N
Nils Larsch 已提交
142 143
	if (bits < 512)
		bits = 512;
144

N
Nils Larsch 已提交
145 146
	bits = (bits+63)/64*64;

D
Dr. Stephen Henson 已提交
147 148 149 150
	/* NB: seed_len == 0 is special case: copy generated seed to
 	 * seed_in if it is not NULL.
 	 */
	if (seed_len && (seed_len < (size_t)qsize))
N
Nils Larsch 已提交
151
		seed_in = NULL;		/* seed buffer too small -- ignore */
152
	if (seed_len > (size_t)qsize) 
N
Nils Larsch 已提交
153 154 155 156 157 158 159 160 161 162
		seed_len = qsize;	/* App. 2.2 of FIPS PUB 186 allows larger SEED,
					 * but our internal buffers are restricted to 160 bits*/
	if (seed_in != NULL)
		memcpy(seed, seed_in, seed_len);

	if ((ctx=BN_CTX_new()) == NULL)
		goto err;

	if ((mont=BN_MONT_CTX_new()) == NULL)
		goto err;
163

164 165 166 167 168 169 170 171 172
	BN_CTX_start(ctx);
	r0 = BN_CTX_get(ctx);
	g = BN_CTX_get(ctx);
	W = BN_CTX_get(ctx);
	q = BN_CTX_get(ctx);
	X = BN_CTX_get(ctx);
	c = BN_CTX_get(ctx);
	p = BN_CTX_get(ctx);
	test = BN_CTX_get(ctx);
173

174 175
	if (!BN_lshift(test,BN_value_one(),bits-1))
		goto err;
176 177 178

	for (;;)
		{
179
		for (;;) /* find q */
180
			{
181
			int seed_is_random;
182

183
			/* step 1 */
184 185
			if(!BN_GENCB_call(cb, 0, m++))
				goto err;
186 187

			if (!seed_len)
188
				{
189 190
				if (RAND_pseudo_bytes(seed, qsize) < 0)
					goto err;
191 192
				seed_is_random = 1;
				}
193
			else
194 195 196 197
				{
				seed_is_random = 0;
				seed_len=0; /* use random seed if 'seed_in' turns out to be bad*/
				}
N
Nils Larsch 已提交
198 199
			memcpy(buf , seed, qsize);
			memcpy(buf2, seed, qsize);
200
			/* precompute "SEED + 1" for step 7: */
N
Nils Larsch 已提交
201
			for (i = qsize-1; i >= 0; i--)
202 203
				{
				buf[i]++;
N
Nils Larsch 已提交
204 205
				if (buf[i] != 0)
					break;
206 207 208
				}

			/* step 2 */
209 210 211 212
			if (!EVP_Digest(seed, qsize, md,   NULL, evpmd, NULL))
				goto err;
			if (!EVP_Digest(buf,  qsize, buf2, NULL, evpmd, NULL))
				goto err;
N
Nils Larsch 已提交
213
			for (i = 0; i < qsize; i++)
214 215 216
				md[i]^=buf2[i];

			/* step 3 */
N
Nils Larsch 已提交
217 218 219 220
			md[0] |= 0x80;
			md[qsize-1] |= 0x01;
			if (!BN_bin2bn(md, qsize, q))
				goto err;
221 222

			/* step 4 */
223
			r = BN_is_prime_fasttest_ex(q, DSS_prime_checks, ctx,
224
					seed_is_random, cb);
225
			if (r > 0)
226
				break;
227
			if (r != 0)
228
				goto err;
229

230 231 232 233
			/* do a callback call */
			/* step 5 */
			}

234 235
		if(!BN_GENCB_call(cb, 2, 0)) goto err;
		if(!BN_GENCB_call(cb, 3, 0)) goto err;
236 237 238

		/* step 6 */
		counter=0;
239
		/* "offset = 2" */
240 241 242 243 244

		n=(bits-1)/160;

		for (;;)
			{
245 246
			if ((counter != 0) && !BN_GENCB_call(cb, 0, counter))
				goto err;
247

248
			/* step 7 */
249
			BN_zero(W);
250
			/* now 'buf' contains "SEED + offset - 1" */
251 252
			for (k=0; k<=n; k++)
				{
253
				/* obtain "SEED + offset + k" by incrementing: */
N
Nils Larsch 已提交
254
				for (i = qsize-1; i >= 0; i--)
255 256
					{
					buf[i]++;
N
Nils Larsch 已提交
257 258
					if (buf[i] != 0)
						break;
259 260
					}

261 262 263
				if (!EVP_Digest(buf, qsize, md ,NULL, evpmd,
									NULL))
					goto err;
264 265

				/* step 8 */
N
Nils Larsch 已提交
266
				if (!BN_bin2bn(md, qsize, r0))
U
Ulf Möller 已提交
267
					goto err;
N
Nils Larsch 已提交
268
				if (!BN_lshift(r0,r0,(qsize << 3)*k)) goto err;
269
				if (!BN_add(W,W,r0)) goto err;
270 271 272
				}

			/* more of step 8 */
273 274 275
			if (!BN_mask_bits(W,bits-1)) goto err;
			if (!BN_copy(X,W)) goto err;
			if (!BN_add(X,X,test)) goto err;
276 277

			/* step 9 */
278 279 280 281
			if (!BN_lshift1(r0,q)) goto err;
			if (!BN_mod(c,X,r0,ctx)) goto err;
			if (!BN_sub(r0,c,BN_value_one())) goto err;
			if (!BN_sub(p,X,r0)) goto err;
282 283 284 285 286

			/* step 10 */
			if (BN_cmp(p,test) >= 0)
				{
				/* step 11 */
287
				r = BN_is_prime_fasttest_ex(p, DSS_prime_checks,
288
						ctx, 1, cb);
289 290 291 292
				if (r > 0)
						goto end; /* found it */
				if (r != 0)
					goto err;
293 294 295 296
				}

			/* step 13 */
			counter++;
297
			/* "offset = offset + n + 1" */
298 299 300 301 302 303

			/* step 14 */
			if (counter >= 4096) break;
			}
		}
end:
304 305
	if(!BN_GENCB_call(cb, 2, 1))
		goto err;
306

U
Ulf Möller 已提交
307
	/* We now need to generate g */
308
	/* Set r0=(p-1)/q */
309 310
	if (!BN_sub(test,p,BN_value_one())) goto err;
	if (!BN_div(r0,NULL,test,q,ctx)) goto err;
311

312 313
	if (!BN_set_word(test,h)) goto err;
	if (!BN_MONT_CTX_set(mont,p,ctx)) goto err;
314

315 316 317
	for (;;)
		{
		/* g=test^r0%p */
318
		if (!BN_mod_exp_mont(g,test,r0,p,ctx,mont)) goto err;
319
		if (!BN_is_one(g)) break;
320
		if (!BN_add(test,test,BN_value_one())) goto err;
321 322 323
		h++;
		}

324 325
	if(!BN_GENCB_call(cb, 3, 1))
		goto err;
326 327 328

	ok=1;
err:
329
	if (ok)
330
		{
331 332 333
		if(ret->p) BN_free(ret->p);
		if(ret->q) BN_free(ret->q);
		if(ret->g) BN_free(ret->g);
334 335 336
		ret->p=BN_dup(p);
		ret->q=BN_dup(q);
		ret->g=BN_dup(g);
337 338 339 340 341
		if (ret->p == NULL || ret->q == NULL || ret->g == NULL)
			{
			ok=0;
			goto err;
			}
342 343
		if (counter_ret != NULL) *counter_ret=counter;
		if (h_ret != NULL) *h_ret=h;
344 345
		if (seed_out)
			memcpy(seed_out, seed, qsize);
346
		}
347
	if(ctx)
348
		{
349 350
		BN_CTX_end(ctx);
		BN_CTX_free(ctx);
351
		}
352
	if (mont != NULL) BN_MONT_CTX_free(mont);
353
	return ok;
354
	}
355 356 357 358 359 360 361

/* This is a parameter generation algorithm for the DSA2 algorithm as
 * described in FIPS 186-3.
 */

int dsa_builtin_paramgen2(DSA *ret, size_t L, size_t N,
	const EVP_MD *evpmd, const unsigned char *seed_in, size_t seed_len,
362
	int idx, unsigned char *seed_out,
363 364 365
	int *counter_ret, unsigned long *h_ret, BN_GENCB *cb)
	{
	int ok=-1;
366
	unsigned char *seed = NULL, *seed_tmp = NULL;
367 368 369 370 371 372 373 374 375
	unsigned char md[EVP_MAX_MD_SIZE];
	int mdsize;
	BIGNUM *r0,*W,*X,*c,*test;
	BIGNUM *g=NULL,*q=NULL,*p=NULL;
	BN_MONT_CTX *mont=NULL;
	int i, k, n=0, m=0, qsize = N >> 3;
	int counter=0;
	int r=0;
	BN_CTX *ctx=NULL;
376
	EVP_MD_CTX mctx;
377 378
	unsigned int h=2;

379 380
	EVP_MD_CTX_init(&mctx);

381 382 383 384 385 386 387 388 389 390 391
	if (evpmd == NULL)
		{
		if (N == 160)
			evpmd = EVP_sha1();
		else if (N == 224)
			evpmd = EVP_sha224();
		else
			evpmd = EVP_sha256();
		}

	mdsize = M_EVP_MD_size(evpmd);
392 393 394 395 396
	/* If unverificable g generation only don't need seed */
	if (!ret->p || !ret->q || idx >= 0)
		{
		if (seed_len == 0)
			seed_len = mdsize;
397

398
		seed = OPENSSL_malloc(seed_len);
399

400 401 402 403
		if (seed_out)
			seed_tmp = seed_out;
		else
			seed_tmp = OPENSSL_malloc(seed_len);
404

405 406
		if (!seed || !seed_tmp)
			goto err;
407

408 409
		if (seed_in)
			memcpy(seed, seed_in, seed_len);
410

411
		}
412 413 414 415 416 417 418 419 420 421 422 423 424 425 426

	if ((ctx=BN_CTX_new()) == NULL)
		goto err;

	if ((mont=BN_MONT_CTX_new()) == NULL)
		goto err;

	BN_CTX_start(ctx);
	r0 = BN_CTX_get(ctx);
	g = BN_CTX_get(ctx);
	W = BN_CTX_get(ctx);
	X = BN_CTX_get(ctx);
	c = BN_CTX_get(ctx);
	test = BN_CTX_get(ctx);

427 428 429 430 431
	/* if p, q already supplied generate g only */
	if (ret->p && ret->q)
		{
		p = ret->p;
		q = ret->q;
432 433
		if (idx >= 0)
			memcpy(seed_tmp, seed, seed_len);
434 435 436 437 438 439 440 441
		goto g_only;
		}
	else
		{
		p = BN_CTX_get(ctx);
		q = BN_CTX_get(ctx);
		}

442 443 444 445 446 447 448 449 450 451 452 453
	if (!BN_lshift(test,BN_value_one(),L-1))
		goto err;
	for (;;)
		{
		for (;;) /* find q */
			{
			unsigned char *pmd;
			/* step 1 */
			if(!BN_GENCB_call(cb, 0, m++))
				goto err;

			if (!seed_in)
454 455 456 457
				{
				if (RAND_pseudo_bytes(seed, seed_len) < 0)
					goto err;
				}
458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493
			/* step 2 */
			if (!EVP_Digest(seed, seed_len, md, NULL, evpmd, NULL))
				goto err;
			/* Take least significant bits of md */
			if (mdsize > qsize)
				pmd = md + mdsize - qsize;
			else
				pmd = md;

			if (mdsize < qsize)
				memset(md + mdsize, 0, qsize - mdsize);

			/* step 3 */
			pmd[0] |= 0x80;
			pmd[qsize-1] |= 0x01;
			if (!BN_bin2bn(pmd, qsize, q))
				goto err;

			/* step 4 */
			r = BN_is_prime_fasttest_ex(q, DSS_prime_checks, ctx,
					seed_in ? 1 : 0, cb);
			if (r > 0)
				break;
			if (r != 0)
				goto err;
			/* Provided seed didn't produce a prime: error */
			if (seed_in)
				{
				ok = 0;
				DSAerr(DSA_F_DSA_BUILTIN_PARAMGEN2, DSA_R_Q_NOT_PRIME);
				goto err;
				}

			/* do a callback call */
			/* step 5 */
			}
494 495 496
		/* Copy seed to seed_out before we mess with it */
		if (seed_out)
			memcpy(seed_out, seed, seed_len);
497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563

		if(!BN_GENCB_call(cb, 2, 0)) goto err;
		if(!BN_GENCB_call(cb, 3, 0)) goto err;

		/* step 6 */
		counter=0;
		/* "offset = 1" */

		n=(L-1)/(mdsize << 3);

		for (;;)
			{
			if ((counter != 0) && !BN_GENCB_call(cb, 0, counter))
				goto err;

			/* step 7 */
			BN_zero(W);
			/* now 'buf' contains "SEED + offset - 1" */
			for (k=0; k<=n; k++)
				{
				/* obtain "SEED + offset + k" by incrementing: */
				for (i = seed_len-1; i >= 0; i--)
					{
					seed[i]++;
					if (seed[i] != 0)
						break;
					}

				if (!EVP_Digest(seed, seed_len, md ,NULL, evpmd,
									NULL))
					goto err;

				/* step 8 */
				if (!BN_bin2bn(md, mdsize, r0))
					goto err;
				if (!BN_lshift(r0,r0,(mdsize << 3)*k)) goto err;
				if (!BN_add(W,W,r0)) goto err;
				}

			/* more of step 8 */
			if (!BN_mask_bits(W,L-1)) goto err;
			if (!BN_copy(X,W)) goto err;
			if (!BN_add(X,X,test)) goto err;

			/* step 9 */
			if (!BN_lshift1(r0,q)) goto err;
			if (!BN_mod(c,X,r0,ctx)) goto err;
			if (!BN_sub(r0,c,BN_value_one())) goto err;
			if (!BN_sub(p,X,r0)) goto err;

			/* step 10 */
			if (BN_cmp(p,test) >= 0)
				{
				/* step 11 */
				r = BN_is_prime_fasttest_ex(p, DSS_prime_checks,
						ctx, 1, cb);
				if (r > 0)
						goto end; /* found it */
				if (r != 0)
					goto err;
				}

			/* step 13 */
			counter++;
			/* "offset = offset + n + 1" */

			/* step 14 */
564
			if (counter >= (int)(4 * L)) break;
565
			}
566 567 568 569 570 571
		if (seed_in)
			{
			ok = 0;
			DSAerr(DSA_F_DSA_BUILTIN_PARAMGEN2, DSA_R_INVALID_PARAMETERS);
			goto err;
			}
572 573 574 575 576
		}
end:
	if(!BN_GENCB_call(cb, 2, 1))
		goto err;

577 578
	g_only:

579 580 581 582 583
	/* We now need to generate g */
	/* Set r0=(p-1)/q */
	if (!BN_sub(test,p,BN_value_one())) goto err;
	if (!BN_div(r0,NULL,test,q,ctx)) goto err;

584 585 586 587 588 589 590
	if (idx < 0)
		{
		if (!BN_set_word(test,h))
			goto err;
		}
	else
		h = 1;
591 592 593 594
	if (!BN_MONT_CTX_set(mont,p,ctx)) goto err;

	for (;;)
		{
595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613
		static const unsigned char ggen[4] = {0x67,0x67,0x65,0x6e};
		if (idx >= 0)
			{
			md[0] = idx & 0xff;
			md[1] = (h >> 8) & 0xff;
			md[2] = h & 0xff;
			if (!EVP_DigestInit_ex(&mctx, evpmd, NULL))
				goto err;
			if (!EVP_DigestUpdate(&mctx, seed_tmp, seed_len))
				goto err;
			if (!EVP_DigestUpdate(&mctx, ggen, sizeof(ggen)))
				goto err;
			if (!EVP_DigestUpdate(&mctx, md, 3))
				goto err;
			if (!EVP_DigestFinal_ex(&mctx, md, NULL))
				goto err;
			if (!BN_bin2bn(md, mdsize, test))
				goto err;
			}
614 615 616
		/* g=test^r0%p */
		if (!BN_mod_exp_mont(g,test,r0,p,ctx,mont)) goto err;
		if (!BN_is_one(g)) break;
617
		if (idx < 0 && !BN_add(test,test,BN_value_one())) goto err;
618
		h++;
619 620
		if ( idx >= 0 && h > 0xffff)
			goto err;
621 622 623 624 625 626 627
		}

	if(!BN_GENCB_call(cb, 3, 1))
		goto err;

	ok=1;
err:
628
	if (ok == 1)
629
		{
630 631 632 633 634 635 636 637 638 639
		if (p != ret->p)
			{
			if(ret->p) BN_free(ret->p);
			ret->p=BN_dup(p);
			}
		if (q != ret->q)
			{
			if(ret->q) BN_free(ret->q);
			ret->q=BN_dup(q);
			}
640 641 642 643 644 645 646 647 648 649 650 651
		if(ret->g) BN_free(ret->g);
		ret->g=BN_dup(g);
		if (ret->p == NULL || ret->q == NULL || ret->g == NULL)
			{
			ok=-1;
			goto err;
			}
		if (counter_ret != NULL) *counter_ret=counter;
		if (h_ret != NULL) *h_ret=h;
		}
	if (seed)
		OPENSSL_free(seed);
652 653
	if (seed_out != seed_tmp)
		OPENSSL_free(seed_tmp);
654 655 656 657 658 659
	if(ctx)
		{
		BN_CTX_end(ctx);
		BN_CTX_free(ctx);
		}
	if (mont != NULL) BN_MONT_CTX_free(mont);
660
	EVP_MD_CTX_cleanup(&mctx);
661 662
	return ok;
	}
663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700

int dsa_paramgen_check_g(DSA *dsa)
	{
	BN_CTX *ctx;
	BIGNUM *tmp;
	BN_MONT_CTX *mont = NULL;
	int rv = -1;
	ctx = BN_CTX_new();
	if (!ctx)
		return -1;
	BN_CTX_start(ctx);
	if (BN_cmp(dsa->g, BN_value_one()) <= 0)
		return 0;
	if (BN_cmp(dsa->g, dsa->p) >= 0)
		return 0;
	tmp = BN_CTX_get(ctx);
	if (!tmp)
		goto err;
	if ((mont=BN_MONT_CTX_new()) == NULL)
		goto err;
	if (!BN_MONT_CTX_set(mont,dsa->p,ctx))
		goto err;
	/* Work out g^q mod p */
	if (!BN_mod_exp_mont(tmp,dsa->g,dsa->q, dsa->p, ctx, mont))
		goto err;
	if (!BN_cmp(tmp, BN_value_one()))
		rv = 1;
	else
		rv = 0;
	err:
	BN_CTX_end(ctx);
	if (mont)
		BN_MONT_CTX_free(mont);
	BN_CTX_free(ctx);
	return rv;

	}

U
Ulf Möller 已提交
701
#endif