提交 99ba9fd0 编写于 作者: M Matt Caswell

Fix BN_hex2bn/BN_dec2bn NULL ptr/heap corruption

In the BN_hex2bn function the number of hex digits is calculated using
an int value |i|. Later |bn_expand| is called with a value of |i * 4|.
For large values of |i| this can result in |bn_expand| not allocating any
memory because |i * 4| is negative. This leaves ret->d as NULL leading
to a subsequent NULL ptr deref. For very large values of |i|, the
calculation |i * 4| could be a positive value smaller than |i|. In this
case memory is allocated to ret->d, but it is insufficiently sized
leading to heap corruption. A similar issue exists in BN_dec2bn.

This could have security consequences if BN_hex2bn/BN_dec2bn is ever
called by user applications with very large untrusted hex/dec data. This is
anticipated to be a rare occurrence.

All OpenSSL internal usage of this function uses data that is not expected
to be untrusted, e.g. config file data or application command line
arguments. If user developed applications generate config file data based
on untrusted data then it is possible that this could also lead to security
consequences. This is also anticipated to be a rare.

Issue reported by Guido Vranken.

CVE-2016-0797
Reviewed-by: NAndy Polyakov <appro@openssl.org>
上级 15e58273
...@@ -774,6 +774,17 @@ int bn_probable_prime_dh(BIGNUM *rnd, int bits, ...@@ -774,6 +774,17 @@ int bn_probable_prime_dh(BIGNUM *rnd, int bits,
int bn_probable_prime_dh_retry(BIGNUM *rnd, int bits, BN_CTX *ctx); int bn_probable_prime_dh_retry(BIGNUM *rnd, int bits, BN_CTX *ctx);
int bn_probable_prime_dh_coprime(BIGNUM *rnd, int bits, BN_CTX *ctx); int bn_probable_prime_dh_coprime(BIGNUM *rnd, int bits, BN_CTX *ctx);
static ossl_inline BIGNUM *bn_expand(BIGNUM *a, int bits)
{
if (bits > (INT_MAX - BN_BITS2 + 1))
return NULL;
if(((bits+BN_BITS2-1)/BN_BITS2) <= (a)->dmax)
return a;
return bn_expand2((a),(bits+BN_BITS2-1)/BN_BITS2);
}
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif #endif
......
...@@ -57,6 +57,7 @@ ...@@ -57,6 +57,7 @@
#include <stdio.h> #include <stdio.h>
#include <ctype.h> #include <ctype.h>
#include <limits.h>
#include "internal/cryptlib.h" #include "internal/cryptlib.h"
#include <openssl/buffer.h> #include <openssl/buffer.h>
#include "bn_lcl.h" #include "bn_lcl.h"
...@@ -183,7 +184,11 @@ int BN_hex2bn(BIGNUM **bn, const char *a) ...@@ -183,7 +184,11 @@ int BN_hex2bn(BIGNUM **bn, const char *a)
a++; a++;
} }
for (i = 0; isxdigit((unsigned char)a[i]); i++) ; for (i = 0; i <= (INT_MAX/4) && isxdigit((unsigned char)a[i]); i++)
continue;
if (i > INT_MAX/4)
goto err;
num = i + neg; num = i + neg;
if (bn == NULL) if (bn == NULL)
...@@ -198,7 +203,7 @@ int BN_hex2bn(BIGNUM **bn, const char *a) ...@@ -198,7 +203,7 @@ int BN_hex2bn(BIGNUM **bn, const char *a)
BN_zero(ret); BN_zero(ret);
} }
/* i is the number of hex digests; */ /* i is the number of hex digits */
if (bn_expand(ret, i * 4) == NULL) if (bn_expand(ret, i * 4) == NULL)
goto err; goto err;
...@@ -254,7 +259,11 @@ int BN_dec2bn(BIGNUM **bn, const char *a) ...@@ -254,7 +259,11 @@ int BN_dec2bn(BIGNUM **bn, const char *a)
a++; a++;
} }
for (i = 0; isdigit((unsigned char)a[i]); i++) ; for (i = 0; i <= (INT_MAX/4) && isdigit((unsigned char)a[i]); i++)
continue;
if (i > INT_MAX/4)
goto err;
num = i + neg; num = i + neg;
if (bn == NULL) if (bn == NULL)
...@@ -272,7 +281,7 @@ int BN_dec2bn(BIGNUM **bn, const char *a) ...@@ -272,7 +281,7 @@ int BN_dec2bn(BIGNUM **bn, const char *a)
BN_zero(ret); BN_zero(ret);
} }
/* i is the number of digests, a bit of an over expand; */ /* i is the number of digits, a bit of an over expand */
if (bn_expand(ret, i * 4) == NULL) if (bn_expand(ret, i * 4) == NULL)
goto err; goto err;
......
...@@ -56,13 +56,12 @@ ...@@ -56,13 +56,12 @@
# define HEADER_BN_INT_H # define HEADER_BN_INT_H
# include <openssl/bn.h> # include <openssl/bn.h>
# include <limits.h>
#ifdef __cplusplus #ifdef __cplusplus
extern "C" { extern "C" {
#endif #endif
# define bn_expand(a,bits) ((((((bits+BN_BITS2-1))/BN_BITS2)) <= (a)->dmax)?\
(a):bn_expand2((a),(bits+BN_BITS2-1)/BN_BITS2))
BIGNUM *bn_wexpand(BIGNUM *a, int words); BIGNUM *bn_wexpand(BIGNUM *a, int words);
BIGNUM *bn_expand2(BIGNUM *a, int words); BIGNUM *bn_expand2(BIGNUM *a, int words);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册