RAND_add()/RAND_seed(): fix failure on short input or low entropy

Commit 5b4cb385c1 (#7382) introduced a bug which had the effect
that RAND_add()/RAND_seed() failed for buffer sizes less than
32 bytes. The reason was that now the added random data was used
exlusively as entropy source for reseeding. When the random input
was too short or contained not enough entropy, the DRBG failed
without querying the available entropy sources.

This commit makes drbg_add() act smarter: it checks the entropy
requirements explicitely. If the random input fails this check,
it won't be added as entropy input, but only as additional data.
More precisely, the behaviour depends on whether an os entropy
source was configured (which is the default on most os):

- If an os entropy source is avaible then we declare the buffer
  content as additional data by setting randomness to zero and
  trigger a regular   reseeding.

- If no os entropy source is available, a reseeding will fail
  inevitably. So drbg_add() uses a trick to mix the buffer contents
  into the DRBG state without forcing a reseeding: it generates a
  dummy random byte, using the buffer content as additional data.

Related-to: #7449

Reviewed-by: Paul Dale <paul.dale@oracle.com>
(Merged from https://github.com/openssl/openssl/pull/7456)
This commit is contained in:
Dr. Matthias St. Pierre 2018-10-21 15:45:34 +02:00
parent 6ec6448b93
commit 8817215d5c
2 changed files with 158 additions and 50 deletions

View file

@ -1031,11 +1031,53 @@ static int drbg_bytes(unsigned char *out, int count)
return ret; return ret;
} }
/*
* Calculates the minimum length of a full entropy buffer
* which is necessary to seed (i.e. instantiate) the DRBG
* successfully.
*
* NOTE: There is a copy of this function in drbgtest.c.
* If you change anything here, you need to update
* the copy accordingly.
*/
static size_t rand_drbg_seedlen(RAND_DRBG *drbg)
{
/*
* If no os entropy source is available then RAND_seed(buffer, bufsize)
* is expected to succeed if and only if the buffer length satisfies
* the following requirements, which follow from the calculations
* in RAND_DRBG_instantiate().
*/
size_t min_entropy = drbg->strength;
size_t min_entropylen = drbg->min_entropylen;
/*
* Extra entropy for the random nonce in the absence of a
* get_nonce callback, see comment in RAND_DRBG_instantiate().
*/
if (drbg->min_noncelen > 0 && drbg->get_nonce == NULL) {
min_entropy += drbg->strength / 2;
min_entropylen += drbg->min_noncelen;
}
/*
* Convert entropy requirement from bits to bytes
* (dividing by 8 without rounding upwards, because
* all entropy requirements are divisible by 8).
*/
min_entropy >>= 3;
/* Return a value that satisfies both requirements */
return min_entropy > min_entropylen ? min_entropy : min_entropylen;
}
/* Implements the default OpenSSL RAND_add() method */ /* Implements the default OpenSSL RAND_add() method */
static int drbg_add(const void *buf, int num, double randomness) static int drbg_add(const void *buf, int num, double randomness)
{ {
int ret = 0; int ret = 0;
RAND_DRBG *drbg = RAND_DRBG_get0_master(); RAND_DRBG *drbg = RAND_DRBG_get0_master();
size_t buflen;
size_t seedlen = rand_drbg_seedlen(drbg);
if (drbg == NULL) if (drbg == NULL)
return 0; return 0;
@ -1043,7 +1085,31 @@ static int drbg_add(const void *buf, int num, double randomness)
if (num < 0 || randomness < 0.0) if (num < 0 || randomness < 0.0)
return 0; return 0;
if (randomness > (double)RAND_DRBG_STRENGTH) { buflen = (size_t)num;
if (buflen < seedlen || randomness < (double) seedlen) {
#if defined(OPENSSL_RAND_SEED_NONE)
/*
* If no os entropy source is available, a reseeding will fail
* inevitably. So we use a trick to mix the buffer contents into
* the DRBG state without forcing a reseeding: we generate a
* dummy random byte, using the buffer content as additional data.
*/
unsigned char dummy[1];
return RAND_DRBG_generate(drbg, dummy, sizeof(dummy), 0, buf, buflen);
#else
/*
* If an os entropy source is avaible then we declare the buffer content
* as additional data by setting randomness to zero and trigger a regular
* reseeding.
*/
randomness = 0.0;
#endif
}
if (randomness > (double)seedlen) {
/* /*
* The purpose of this check is to bound |randomness| by a * The purpose of this check is to bound |randomness| by a
* relatively small value in order to prevent an integer * relatively small value in order to prevent an integer
@ -1052,13 +1118,11 @@ static int drbg_add(const void *buf, int num, double randomness)
* not bits, so this value corresponds to eight times the * not bits, so this value corresponds to eight times the
* security strength. * security strength.
*/ */
randomness = (double)RAND_DRBG_STRENGTH; randomness = (double)seedlen;
} }
rand_drbg_lock(drbg); rand_drbg_lock(drbg);
ret = rand_drbg_restart(drbg, buf, ret = rand_drbg_restart(drbg, buf, buflen, (size_t)(8 * randomness));
(size_t)(unsigned int)num,
(size_t)(8*randomness));
rand_drbg_unlock(drbg); rand_drbg_unlock(drbg);
return ret; return ret;

View file

@ -677,7 +677,7 @@ static int test_drbg_reseed(int expect_success,
* setup correctly, in particular whether reseeding works * setup correctly, in particular whether reseeding works
* as designed. * as designed.
*/ */
static int test_rand_reseed(void) static int test_rand_drbg_reseed(void)
{ {
RAND_DRBG *master, *public, *private; RAND_DRBG *master, *public, *private;
unsigned char rand_add_buf[256]; unsigned char rand_add_buf[256];
@ -884,64 +884,107 @@ static int test_multi_thread(void)
} }
#endif #endif
#ifdef OPENSSL_RAND_SEED_NONE
/* /*
* This function only returns the entropy already added with RAND_add(), * Calculates the minimum buffer length which needs to be
* and does not get entropy from the OS. * provided to RAND_seed() in order to successfully
* instantiate the DRBG.
* *
* Returns 0 on failure and the size of the buffer on success. * Copied from rand_drbg_seedlen() in rand_drbg.c
*/ */
static size_t get_pool_entropy(RAND_DRBG *drbg, static size_t rand_drbg_seedlen(RAND_DRBG *drbg)
unsigned char **pout,
int entropy, size_t min_len, size_t max_len,
int prediction_resistance)
{ {
if (drbg->pool == NULL) /*
return 0; * If no os entropy source is available then RAND_seed(buffer, bufsize)
* is expected to succeed if and only if the buffer length satisfies
* the following requirements, which follow from the calculations
* in RAND_DRBG_instantiate().
*/
size_t min_entropy = drbg->strength;
size_t min_entropylen = drbg->min_entropylen;
if (drbg->pool->entropy < (size_t)entropy || drbg->pool->len < min_len /*
|| drbg->pool->len > max_len) * Extra entropy for the random nonce in the absence of a
return 0; * get_nonce callback, see comment in RAND_DRBG_instantiate().
*/
*pout = drbg->pool->buffer; if (drbg->min_noncelen > 0 && drbg->get_nonce == NULL) {
return drbg->pool->len; min_entropy += drbg->strength / 2;
min_entropylen += drbg->min_noncelen;
} }
/* /*
* Clean up the entropy that get_pool_entropy() returned. * Convert entropy requirement from bits to bytes
* (dividing by 8 without rounding upwards, because
* all entropy requirements are divisible by 8).
*/ */
static void cleanup_pool_entropy(RAND_DRBG *drbg, unsigned char *out, size_t outlen) min_entropy >>= 3;
/* Return a value that satisfies both requirements */
return min_entropy > min_entropylen ? min_entropy : min_entropylen;
}
#endif /*OPENSSL_RAND_SEED_NONE*/
/*
* Test that instantiation with RAND_seed() works as expected
*
* If no os entropy source is available then RAND_seed(buffer, bufsize)
* is expected to succeed if and only if the buffer length is at least
* rand_drbg_seedlen(master) bytes.
*
* If an os entropy source is available then RAND_seed(buffer, bufsize)
* is expected to succeed always.
*/
static int test_rand_seed(void)
{ {
OPENSSL_free(drbg->pool); RAND_DRBG *master = RAND_DRBG_get0_master();
drbg->pool = NULL; unsigned char rand_buf[256];
size_t rand_buflen;
#ifdef OPENSSL_RAND_SEED_NONE
size_t required_seed_buflen = rand_drbg_seedlen(master);
#else
size_t required_seed_buflen = 0;
#endif
memset(rand_buf, 0xCD, sizeof(rand_buf));
for ( rand_buflen = 256 ; rand_buflen > 0 ; --rand_buflen ) {
RAND_DRBG_uninstantiate(master);
RAND_seed(rand_buf, rand_buflen);
if (!TEST_int_eq(RAND_status(),
(rand_buflen >= required_seed_buflen)))
return 0;
}
return 1;
} }
/* /*
* Test that instantiating works when OS entropy is not available and that * Test that adding additional data with RAND_add() works as expected
* RAND_add() is enough to reseed it. * when the master DRBG is instantiated (and below its reseed limit).
*
* This should succeed regardless of whether an os entropy source is
* available or not.
*/ */
static int test_rand_add(void) static int test_rand_add(void)
{ {
RAND_DRBG *master = RAND_DRBG_get0_master(); unsigned char rand_buf[256];
RAND_DRBG_get_entropy_fn old_get_entropy = master->get_entropy; size_t rand_buflen;
RAND_DRBG_cleanup_entropy_fn old_cleanup_entropy = master->cleanup_entropy;
int rv = 0;
unsigned char rand_add_buf[256];
master->get_entropy = get_pool_entropy; memset(rand_buf, 0xCD, sizeof(rand_buf));
master->cleanup_entropy = cleanup_pool_entropy;
master->reseed_prop_counter++;
RAND_DRBG_uninstantiate(master);
memset(rand_add_buf, 0xCD, sizeof(rand_add_buf));
RAND_add(rand_add_buf, sizeof(rand_add_buf), sizeof(rand_add_buf));
if (!TEST_true(RAND_DRBG_instantiate(master, NULL, 0)))
goto error;
rv = 1; /* make sure it's instantiated */
RAND_seed(rand_buf, sizeof(rand_buf));
if (!TEST_true(RAND_status()))
return 0;
error: for ( rand_buflen = 256 ; rand_buflen > 0 ; --rand_buflen ) {
master->get_entropy = old_get_entropy; RAND_add(rand_buf, rand_buflen, 0.0);
master->cleanup_entropy = old_cleanup_entropy; if (!TEST_true(RAND_status()))
return rv; return 0;
}
return 1;
} }
static int test_multi_set(void) static int test_multi_set(void)
@ -1067,7 +1110,8 @@ int setup_tests(void)
ADD_ALL_TESTS(test_kats, OSSL_NELEM(drbg_test)); ADD_ALL_TESTS(test_kats, OSSL_NELEM(drbg_test));
ADD_ALL_TESTS(test_error_checks, OSSL_NELEM(drbg_test)); ADD_ALL_TESTS(test_error_checks, OSSL_NELEM(drbg_test));
ADD_TEST(test_rand_reseed); ADD_TEST(test_rand_drbg_reseed);
ADD_TEST(test_rand_seed);
ADD_TEST(test_rand_add); ADD_TEST(test_rand_add);
ADD_TEST(test_multi_set); ADD_TEST(test_multi_set);
ADD_TEST(test_set_defaults); ADD_TEST(test_set_defaults);