2016-05-18 02:51:26 +08:00
|
|
|
/*
|
2017-08-18 11:52:46 +08:00
|
|
|
* Copyright 1995-2017 The OpenSSL Project Authors. All Rights Reserved.
|
1998-12-21 18:52:47 +08:00
|
|
|
*
|
2016-05-18 02:51:26 +08:00
|
|
|
* Licensed under the OpenSSL license (the "License"). You may not use
|
|
|
|
* this file except in compliance with the License. You can obtain a copy
|
|
|
|
* in the file LICENSE in the source distribution or at
|
|
|
|
* https://www.openssl.org/source/license.html
|
1998-12-21 18:52:47 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2015-05-14 22:56:48 +08:00
|
|
|
#include "internal/cryptlib.h"
|
2017-08-22 05:17:35 +08:00
|
|
|
#include "internal/refcount.h"
|
1999-04-24 06:13:45 +08:00
|
|
|
#include <openssl/bn.h>
|
2016-03-30 22:21:39 +08:00
|
|
|
#include "dsa_locl.h"
|
1999-04-24 06:13:45 +08:00
|
|
|
#include <openssl/asn1.h>
|
2016-03-19 02:30:20 +08:00
|
|
|
#include <openssl/engine.h>
|
|
|
|
#include <openssl/dh.h>
|
1998-12-21 18:52:47 +08:00
|
|
|
|
1999-04-20 05:31:43 +08:00
|
|
|
DSA *DSA_new(void)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
|
|
|
return DSA_new_method(NULL);
|
|
|
|
}
|
1999-08-23 01:57:38 +08:00
|
|
|
|
2001-09-26 04:23:40 +08:00
|
|
|
int DSA_set_method(DSA *dsa, const DSA_METHOD *meth)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* NB: The caller is specifically setting a method, so it's not up to us
|
|
|
|
* to deal with which ENGINE it comes from.
|
|
|
|
*/
|
|
|
|
const DSA_METHOD *mtmp;
|
|
|
|
mtmp = dsa->meth;
|
|
|
|
if (mtmp->finish)
|
|
|
|
mtmp->finish(dsa);
|
2003-01-31 01:39:26 +08:00
|
|
|
#ifndef OPENSSL_NO_ENGINE
|
2016-02-26 01:09:06 +08:00
|
|
|
ENGINE_finish(dsa->engine);
|
|
|
|
dsa->engine = NULL;
|
2003-01-31 01:39:26 +08:00
|
|
|
#endif
|
2015-01-22 11:40:55 +08:00
|
|
|
dsa->meth = meth;
|
|
|
|
if (meth->init)
|
|
|
|
meth->init(dsa);
|
|
|
|
return 1;
|
|
|
|
}
|
1999-08-23 01:57:38 +08:00
|
|
|
|
2016-03-31 00:18:55 +08:00
|
|
|
const DSA_METHOD *DSA_get_method(DSA *d)
|
|
|
|
{
|
|
|
|
return d->meth;
|
|
|
|
}
|
|
|
|
|
2000-10-27 05:07:28 +08:00
|
|
|
DSA *DSA_new_method(ENGINE *engine)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
2016-03-09 03:11:48 +08:00
|
|
|
DSA *ret = OPENSSL_zalloc(sizeof(*ret));
|
2015-01-22 11:40:55 +08:00
|
|
|
|
|
|
|
if (ret == NULL) {
|
|
|
|
DSAerr(DSA_F_DSA_NEW_METHOD, ERR_R_MALLOC_FAILURE);
|
2016-03-04 23:43:46 +08:00
|
|
|
return NULL;
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
2016-03-09 03:11:48 +08:00
|
|
|
|
|
|
|
ret->references = 1;
|
|
|
|
ret->lock = CRYPTO_THREAD_lock_new();
|
|
|
|
if (ret->lock == NULL) {
|
2016-04-30 22:23:33 +08:00
|
|
|
DSAerr(DSA_F_DSA_NEW_METHOD, ERR_R_MALLOC_FAILURE);
|
2016-03-09 03:11:48 +08:00
|
|
|
OPENSSL_free(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-22 11:40:55 +08:00
|
|
|
ret->meth = DSA_get_default_method();
|
2003-01-31 01:39:26 +08:00
|
|
|
#ifndef OPENSSL_NO_ENGINE
|
2016-03-09 03:11:48 +08:00
|
|
|
ret->flags = ret->meth->flags & ~DSA_FLAG_NON_FIPS_ALLOW; /* early default init */
|
2015-01-22 11:40:55 +08:00
|
|
|
if (engine) {
|
|
|
|
if (!ENGINE_init(engine)) {
|
|
|
|
DSAerr(DSA_F_DSA_NEW_METHOD, ERR_R_ENGINE_LIB);
|
2016-03-09 03:11:48 +08:00
|
|
|
goto err;
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
|
|
|
ret->engine = engine;
|
|
|
|
} else
|
|
|
|
ret->engine = ENGINE_get_default_DSA();
|
|
|
|
if (ret->engine) {
|
|
|
|
ret->meth = ENGINE_get_DSA(ret->engine);
|
2016-02-26 01:09:06 +08:00
|
|
|
if (ret->meth == NULL) {
|
2015-01-22 11:40:55 +08:00
|
|
|
DSAerr(DSA_F_DSA_NEW_METHOD, ERR_R_ENGINE_LIB);
|
2016-03-09 03:11:48 +08:00
|
|
|
goto err;
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
|
|
|
}
|
2003-01-31 01:39:26 +08:00
|
|
|
#endif
|
2001-06-24 07:07:34 +08:00
|
|
|
|
2015-01-22 11:40:55 +08:00
|
|
|
ret->flags = ret->meth->flags & ~DSA_FLAG_NON_FIPS_ALLOW;
|
2016-03-04 23:43:46 +08:00
|
|
|
|
2016-03-09 03:11:48 +08:00
|
|
|
if (!CRYPTO_new_ex_data(CRYPTO_EX_INDEX_DSA, ret, &ret->ex_data))
|
|
|
|
goto err;
|
2016-03-04 23:43:46 +08:00
|
|
|
|
|
|
|
if ((ret->meth->init != NULL) && !ret->meth->init(ret)) {
|
2016-03-09 03:11:48 +08:00
|
|
|
DSAerr(DSA_F_DSA_NEW_METHOD, ERR_R_INIT_FAIL);
|
|
|
|
err:
|
2016-03-04 23:43:46 +08:00
|
|
|
DSA_free(ret);
|
2015-01-22 11:40:55 +08:00
|
|
|
ret = NULL;
|
|
|
|
}
|
|
|
|
|
2016-03-04 23:43:46 +08:00
|
|
|
return ret;
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
1998-12-21 18:52:47 +08:00
|
|
|
|
1999-04-20 05:31:43 +08:00
|
|
|
void DSA_free(DSA *r)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
|
|
|
int i;
|
1998-12-21 18:52:47 +08:00
|
|
|
|
2015-01-22 11:40:55 +08:00
|
|
|
if (r == NULL)
|
|
|
|
return;
|
1998-12-21 18:52:47 +08:00
|
|
|
|
2016-08-27 22:01:08 +08:00
|
|
|
CRYPTO_DOWN_REF(&r->references, &i, r->lock);
|
2016-01-31 01:04:25 +08:00
|
|
|
REF_PRINT_COUNT("DSA", r);
|
2015-01-22 11:40:55 +08:00
|
|
|
if (i > 0)
|
|
|
|
return;
|
2016-01-31 01:04:25 +08:00
|
|
|
REF_ASSERT_ISNT(i < 0);
|
1998-12-21 18:52:47 +08:00
|
|
|
|
2015-01-22 11:40:55 +08:00
|
|
|
if (r->meth->finish)
|
|
|
|
r->meth->finish(r);
|
2003-01-31 01:39:26 +08:00
|
|
|
#ifndef OPENSSL_NO_ENGINE
|
2016-02-28 23:01:41 +08:00
|
|
|
ENGINE_finish(r->engine);
|
2003-01-31 01:39:26 +08:00
|
|
|
#endif
|
1999-08-23 01:57:38 +08:00
|
|
|
|
2015-01-22 11:40:55 +08:00
|
|
|
CRYPTO_free_ex_data(CRYPTO_EX_INDEX_DSA, r, &r->ex_data);
|
|
|
|
|
2016-03-04 23:43:46 +08:00
|
|
|
CRYPTO_THREAD_lock_free(r->lock);
|
|
|
|
|
2015-05-01 09:37:06 +08:00
|
|
|
BN_clear_free(r->p);
|
|
|
|
BN_clear_free(r->q);
|
|
|
|
BN_clear_free(r->g);
|
|
|
|
BN_clear_free(r->pub_key);
|
|
|
|
BN_clear_free(r->priv_key);
|
2015-01-22 11:40:55 +08:00
|
|
|
OPENSSL_free(r);
|
|
|
|
}
|
1998-12-21 18:52:47 +08:00
|
|
|
|
2001-09-03 21:40:07 +08:00
|
|
|
int DSA_up_ref(DSA *r)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
2016-03-04 23:43:46 +08:00
|
|
|
int i;
|
|
|
|
|
2016-08-27 22:01:08 +08:00
|
|
|
if (CRYPTO_UP_REF(&r->references, &i, r->lock) <= 0)
|
2016-03-04 23:43:46 +08:00
|
|
|
return 0;
|
2016-01-31 01:04:25 +08:00
|
|
|
|
|
|
|
REF_PRINT_COUNT("DSA", r);
|
|
|
|
REF_ASSERT_ISNT(i < 2);
|
2015-01-22 11:40:55 +08:00
|
|
|
return ((i > 1) ? 1 : 0);
|
|
|
|
}
|
2001-08-26 01:24:21 +08:00
|
|
|
|
2000-11-07 21:54:39 +08:00
|
|
|
int DSA_size(const DSA *r)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
|
|
|
int ret, i;
|
|
|
|
ASN1_INTEGER bs;
|
|
|
|
unsigned char buf[4]; /* 4 bytes looks really small. However,
|
|
|
|
* i2d_ASN1_INTEGER() will not look beyond
|
|
|
|
* the first byte, as long as the second
|
|
|
|
* parameter is NULL. */
|
|
|
|
|
|
|
|
i = BN_num_bits(r->q);
|
|
|
|
bs.length = (i + 7) / 8;
|
|
|
|
bs.data = buf;
|
|
|
|
bs.type = V_ASN1_INTEGER;
|
|
|
|
/* If the top bit is set the asn1 encoding is 1 larger. */
|
|
|
|
buf[0] = 0xff;
|
|
|
|
|
|
|
|
i = i2d_ASN1_INTEGER(&bs, NULL);
|
|
|
|
i += i; /* r and s */
|
|
|
|
ret = ASN1_object_size(1, i, V_ASN1_SEQUENCE);
|
2017-10-17 22:04:09 +08:00
|
|
|
return ret;
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
1998-12-21 18:52:47 +08:00
|
|
|
|
2000-01-24 07:41:49 +08:00
|
|
|
int DSA_set_ex_data(DSA *d, int idx, void *arg)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
2017-10-17 22:04:09 +08:00
|
|
|
return CRYPTO_set_ex_data(&d->ex_data, idx, arg);
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
1999-08-23 01:57:38 +08:00
|
|
|
|
2000-01-24 07:41:49 +08:00
|
|
|
void *DSA_get_ex_data(DSA *d, int idx)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
2017-10-17 22:04:09 +08:00
|
|
|
return CRYPTO_get_ex_data(&d->ex_data, idx);
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
1999-08-23 01:57:38 +08:00
|
|
|
|
2014-01-18 22:51:40 +08:00
|
|
|
int DSA_security_bits(const DSA *d)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
2015-12-30 21:34:53 +08:00
|
|
|
if (d->p && d->q)
|
|
|
|
return BN_security_bits(BN_num_bits(d->p), BN_num_bits(d->q));
|
|
|
|
return -1;
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
2014-01-18 22:51:40 +08:00
|
|
|
|
2001-02-20 00:06:34 +08:00
|
|
|
#ifndef OPENSSL_NO_DH
|
2000-11-07 21:54:39 +08:00
|
|
|
DH *DSA_dup_DH(const DSA *r)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* DSA has p, q, g, optional pub_key, optional priv_key. DH has p,
|
|
|
|
* optional length, g, optional pub_key, optional priv_key, optional q.
|
|
|
|
*/
|
|
|
|
|
|
|
|
DH *ret = NULL;
|
2016-04-07 00:49:48 +08:00
|
|
|
BIGNUM *p = NULL, *q = NULL, *g = NULL, *pub_key = NULL, *priv_key = NULL;
|
2015-01-22 11:40:55 +08:00
|
|
|
|
|
|
|
if (r == NULL)
|
|
|
|
goto err;
|
|
|
|
ret = DH_new();
|
|
|
|
if (ret == NULL)
|
|
|
|
goto err;
|
2016-04-07 00:49:48 +08:00
|
|
|
if (r->p != NULL || r->g != NULL || r->q != NULL) {
|
|
|
|
if (r->p == NULL || r->g == NULL || r->q == NULL) {
|
|
|
|
/* Shouldn't happen */
|
2015-01-22 11:40:55 +08:00
|
|
|
goto err;
|
2016-04-07 00:49:48 +08:00
|
|
|
}
|
|
|
|
p = BN_dup(r->p);
|
|
|
|
g = BN_dup(r->g);
|
|
|
|
q = BN_dup(r->q);
|
|
|
|
if (p == NULL || g == NULL || q == NULL || !DH_set0_pqg(ret, p, q, g))
|
2015-01-22 11:40:55 +08:00
|
|
|
goto err;
|
2016-04-07 21:08:52 +08:00
|
|
|
p = g = q = NULL;
|
2015-01-22 11:40:55 +08:00
|
|
|
}
|
2016-04-07 00:49:48 +08:00
|
|
|
|
|
|
|
if (r->pub_key != NULL) {
|
|
|
|
pub_key = BN_dup(r->pub_key);
|
|
|
|
if (pub_key == NULL)
|
2015-01-22 11:40:55 +08:00
|
|
|
goto err;
|
2016-04-07 00:49:48 +08:00
|
|
|
if (r->priv_key != NULL) {
|
|
|
|
priv_key = BN_dup(r->priv_key);
|
|
|
|
if (priv_key == NULL)
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
if (!DH_set0_key(ret, pub_key, priv_key))
|
2015-01-22 11:40:55 +08:00
|
|
|
goto err;
|
2016-04-07 00:49:48 +08:00
|
|
|
} else if (r->priv_key != NULL) {
|
|
|
|
/* Shouldn't happen */
|
|
|
|
goto err;
|
|
|
|
}
|
2015-01-22 11:40:55 +08:00
|
|
|
|
|
|
|
return ret;
|
1999-08-05 19:50:18 +08:00
|
|
|
|
|
|
|
err:
|
2016-04-07 00:49:48 +08:00
|
|
|
BN_free(p);
|
|
|
|
BN_free(g);
|
|
|
|
BN_free(q);
|
|
|
|
BN_free(pub_key);
|
|
|
|
BN_free(priv_key);
|
2015-03-24 22:17:37 +08:00
|
|
|
DH_free(ret);
|
2015-01-22 11:40:55 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
1999-08-05 19:50:18 +08:00
|
|
|
#endif
|
2016-03-30 22:21:39 +08:00
|
|
|
|
2016-06-14 21:48:16 +08:00
|
|
|
void DSA_get0_pqg(const DSA *d,
|
|
|
|
const BIGNUM **p, const BIGNUM **q, const BIGNUM **g)
|
2016-03-30 22:21:39 +08:00
|
|
|
{
|
2016-03-31 00:18:55 +08:00
|
|
|
if (p != NULL)
|
|
|
|
*p = d->p;
|
|
|
|
if (q != NULL)
|
|
|
|
*q = d->q;
|
|
|
|
if (g != NULL)
|
|
|
|
*g = d->g;
|
2016-03-30 22:21:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int DSA_set0_pqg(DSA *d, BIGNUM *p, BIGNUM *q, BIGNUM *g)
|
|
|
|
{
|
2016-06-14 21:48:16 +08:00
|
|
|
/* If the fields p, q and g in d are NULL, the corresponding input
|
RSA, DSA, DH: Allow some given input to be NULL on already initialised keys
The diverse {RSA,DSA,DH}_set0_* functions are made to allow some
parameters to be NULL IF the corresponding numbers in the given key
structure have already been previously initialised. Specifically,
this allows the addition of private components to be added to a key
that already has the public half, approximately like this:
RSA_get0_key(rsa, NULL, &e, NULL);
RSA_get0_factors(rsa, &p, &q);
/* calculate new d */
RSA_set0_key(rsa, NULL, NULL, d);
Reviewed-by: Matt Caswell <matt@openssl.org>
2016-04-26 02:28:54 +08:00
|
|
|
* parameters MUST be non-NULL.
|
|
|
|
*/
|
2016-06-16 17:07:32 +08:00
|
|
|
if ((d->p == NULL && p == NULL)
|
|
|
|
|| (d->q == NULL && q == NULL)
|
|
|
|
|| (d->g == NULL && g == NULL))
|
2016-03-30 22:21:39 +08:00
|
|
|
return 0;
|
RSA, DSA, DH: Allow some given input to be NULL on already initialised keys
The diverse {RSA,DSA,DH}_set0_* functions are made to allow some
parameters to be NULL IF the corresponding numbers in the given key
structure have already been previously initialised. Specifically,
this allows the addition of private components to be added to a key
that already has the public half, approximately like this:
RSA_get0_key(rsa, NULL, &e, NULL);
RSA_get0_factors(rsa, &p, &q);
/* calculate new d */
RSA_set0_key(rsa, NULL, NULL, d);
Reviewed-by: Matt Caswell <matt@openssl.org>
2016-04-26 02:28:54 +08:00
|
|
|
|
|
|
|
if (p != NULL) {
|
|
|
|
BN_free(d->p);
|
|
|
|
d->p = p;
|
|
|
|
}
|
|
|
|
if (q != NULL) {
|
|
|
|
BN_free(d->q);
|
|
|
|
d->q = q;
|
|
|
|
}
|
|
|
|
if (g != NULL) {
|
|
|
|
BN_free(d->g);
|
|
|
|
d->g = g;
|
|
|
|
}
|
2016-03-30 22:21:39 +08:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-06-14 21:48:16 +08:00
|
|
|
void DSA_get0_key(const DSA *d,
|
|
|
|
const BIGNUM **pub_key, const BIGNUM **priv_key)
|
2016-03-30 22:21:39 +08:00
|
|
|
{
|
2016-03-31 00:18:55 +08:00
|
|
|
if (pub_key != NULL)
|
|
|
|
*pub_key = d->pub_key;
|
|
|
|
if (priv_key != NULL)
|
|
|
|
*priv_key = d->priv_key;
|
2016-03-30 22:21:39 +08:00
|
|
|
}
|
|
|
|
|
2016-03-31 00:18:55 +08:00
|
|
|
int DSA_set0_key(DSA *d, BIGNUM *pub_key, BIGNUM *priv_key)
|
2016-03-30 22:21:39 +08:00
|
|
|
{
|
2016-06-14 21:48:16 +08:00
|
|
|
/* If the field pub_key in d is NULL, the corresponding input
|
RSA, DSA, DH: Allow some given input to be NULL on already initialised keys
The diverse {RSA,DSA,DH}_set0_* functions are made to allow some
parameters to be NULL IF the corresponding numbers in the given key
structure have already been previously initialised. Specifically,
this allows the addition of private components to be added to a key
that already has the public half, approximately like this:
RSA_get0_key(rsa, NULL, &e, NULL);
RSA_get0_factors(rsa, &p, &q);
/* calculate new d */
RSA_set0_key(rsa, NULL, NULL, d);
Reviewed-by: Matt Caswell <matt@openssl.org>
2016-04-26 02:28:54 +08:00
|
|
|
* parameters MUST be non-NULL. The priv_key field may
|
|
|
|
* be left NULL.
|
|
|
|
*/
|
2016-06-14 21:48:16 +08:00
|
|
|
if (d->pub_key == NULL && pub_key == NULL)
|
2016-03-30 22:21:39 +08:00
|
|
|
return 0;
|
|
|
|
|
RSA, DSA, DH: Allow some given input to be NULL on already initialised keys
The diverse {RSA,DSA,DH}_set0_* functions are made to allow some
parameters to be NULL IF the corresponding numbers in the given key
structure have already been previously initialised. Specifically,
this allows the addition of private components to be added to a key
that already has the public half, approximately like this:
RSA_get0_key(rsa, NULL, &e, NULL);
RSA_get0_factors(rsa, &p, &q);
/* calculate new d */
RSA_set0_key(rsa, NULL, NULL, d);
Reviewed-by: Matt Caswell <matt@openssl.org>
2016-04-26 02:28:54 +08:00
|
|
|
if (pub_key != NULL) {
|
|
|
|
BN_free(d->pub_key);
|
|
|
|
d->pub_key = pub_key;
|
|
|
|
}
|
|
|
|
if (priv_key != NULL) {
|
|
|
|
BN_free(d->priv_key);
|
|
|
|
d->priv_key = priv_key;
|
|
|
|
}
|
2016-03-30 22:21:39 +08:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DSA_clear_flags(DSA *d, int flags)
|
|
|
|
{
|
|
|
|
d->flags &= ~flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
int DSA_test_flags(const DSA *d, int flags)
|
|
|
|
{
|
|
|
|
return d->flags & flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DSA_set_flags(DSA *d, int flags)
|
|
|
|
{
|
|
|
|
d->flags |= flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
ENGINE *DSA_get0_engine(DSA *d)
|
|
|
|
{
|
|
|
|
return d->engine;
|
|
|
|
}
|
2016-08-02 01:37:03 +08:00
|
|
|
|
|
|
|
int DSA_bits(const DSA *dsa)
|
|
|
|
{
|
|
|
|
return BN_num_bits(dsa->p);
|
|
|
|
}
|