mirror of
https://github.com/openssl/openssl.git
synced 2024-12-21 06:09:35 +08:00
82a7b2fb00
Reviewed-by: Matt Caswell <matt@openssl.org> Reviewed-by: Paul Dale <paul.dale@oracle.com> Reviewed-by: Ben Kaduk <kaduk@mit.edu> (Merged from https://github.com/openssl/openssl/pull/12608)
269 lines
10 KiB
C
269 lines
10 KiB
C
/*
|
|
* Copyright 1995-2020 The OpenSSL Project Authors. All Rights Reserved.
|
|
*
|
|
* Licensed under the Apache License 2.0 (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
|
|
*/
|
|
|
|
#ifndef OSSL_CRYPTO_PROV_LOCAL_H
|
|
# define OSSL_CRYPTO_PROV_LOCAL_H
|
|
|
|
# include <openssl/evp.h>
|
|
# include <openssl/core_dispatch.h>
|
|
# include <openssl/core_names.h>
|
|
# include <openssl/params.h>
|
|
# include "internal/tsan_assist.h"
|
|
# include "internal/nelem.h"
|
|
# include "internal/numbers.h"
|
|
|
|
/* How many times to read the TSC as a randomness source. */
|
|
# define TSC_READ_COUNT 4
|
|
|
|
/* Maximum reseed intervals */
|
|
# define MAX_RESEED_INTERVAL (1 << 24)
|
|
# define MAX_RESEED_TIME_INTERVAL (1 << 20) /* approx. 12 days */
|
|
|
|
/* Default reseed intervals */
|
|
# define RESEED_INTERVAL (1 << 8)
|
|
# define TIME_INTERVAL (60*60) /* 1 hour */
|
|
|
|
/*
|
|
* The number of bytes that constitutes an atomic lump of entropy with respect
|
|
* to the FIPS 140-2 section 4.9.2 Conditional Tests. The size is somewhat
|
|
* arbitrary, the smaller the value, the less entropy is consumed on first
|
|
* read but the higher the probability of the test failing by accident.
|
|
*
|
|
* The value is in bytes.
|
|
*/
|
|
#define CRNGT_BUFSIZ 16
|
|
|
|
/*
|
|
* Maximum input size for the DRBG (entropy, nonce, personalization string)
|
|
*
|
|
* NIST SP800 90Ar1 allows a maximum of (1 << 35) bits i.e., (1 << 32) bytes.
|
|
*
|
|
* We lower it to 'only' INT32_MAX bytes, which is equivalent to 2 gigabytes.
|
|
*/
|
|
# define DRBG_MAX_LENGTH INT32_MAX
|
|
|
|
/* The default nonce */
|
|
#ifdef CHARSET_EBCDIC
|
|
# define DRBG_DEFAULT_PERS_STRING { 0x4f, 0x70, 0x65, 0x6e, 0x53, 0x53, \
|
|
0x4c, 0x20, 0x4e, 0x49, 0x53, 0x54, 0x20, 0x53, 0x50, 0x20, 0x38, 0x30, \
|
|
0x30, 0x2d, 0x39, 0x30, 0x41, 0x20, 0x44, 0x52, 0x42, 0x47, 0x00};
|
|
#else
|
|
# define DRBG_DEFAULT_PERS_STRING "OpenSSL NIST SP 800-90A DRBG"
|
|
#endif
|
|
|
|
typedef struct prov_drbg_st PROV_DRBG;
|
|
|
|
/* DRBG status values */
|
|
typedef enum drbg_status_e {
|
|
DRBG_UNINITIALISED,
|
|
DRBG_READY,
|
|
DRBG_ERROR
|
|
} DRBG_STATUS;
|
|
|
|
/*
|
|
* The state of all types of DRBGs.
|
|
*/
|
|
struct prov_drbg_st {
|
|
CRYPTO_RWLOCK *lock;
|
|
void *provctx;
|
|
|
|
/* Virtual functions are cache here */
|
|
int (*instantiate)(PROV_DRBG *drbg,
|
|
const unsigned char *entropy, size_t entropylen,
|
|
const unsigned char *nonce, size_t noncelen,
|
|
const unsigned char *pers, size_t perslen);
|
|
int (*uninstantiate)(PROV_DRBG *ctx);
|
|
int (*reseed)(PROV_DRBG *drbg, const unsigned char *ent, size_t ent_len,
|
|
const unsigned char *adin, size_t adin_len);
|
|
int (*generate)(PROV_DRBG *, unsigned char *out, size_t outlen,
|
|
const unsigned char *adin, size_t adin_len);
|
|
|
|
/* Parent PROV_RAND and its dispatch table functions */
|
|
void *parent;
|
|
OSSL_FUNC_rand_enable_locking_fn *parent_enable_locking;
|
|
OSSL_FUNC_rand_lock_fn *parent_lock;
|
|
OSSL_FUNC_rand_unlock_fn *parent_unlock;
|
|
OSSL_FUNC_rand_get_ctx_params_fn *parent_get_ctx_params;
|
|
OSSL_FUNC_rand_generate_fn *parent_generate;
|
|
OSSL_FUNC_rand_nonce_fn *parent_nonce;
|
|
|
|
const OSSL_DISPATCH *parent_dispatch;
|
|
|
|
/*
|
|
* Stores the return value of openssl_get_fork_id() as of when we last
|
|
* reseeded. The DRBG reseeds automatically whenever drbg->fork_id !=
|
|
* openssl_get_fork_id(). Used to provide fork-safety and reseed this
|
|
* DRBG in the child process.
|
|
*/
|
|
int fork_id;
|
|
unsigned short flags; /* various external flags */
|
|
|
|
/*
|
|
* The random_data is used by PROV_add()/drbg_add() to attach random
|
|
* data to the global drbg, such that the rand_drbg_get_entropy() callback
|
|
* can pull it during instantiation and reseeding. This is necessary to
|
|
* reconcile the different philosophies of the PROV and the PROV_DRBG
|
|
* with respect to how randomness is added to the RNG during reseeding
|
|
* (see PR #4328).
|
|
*/
|
|
struct rand_pool_st *seed_pool;
|
|
|
|
/*
|
|
* Auxiliary pool for additional data.
|
|
*/
|
|
struct rand_pool_st *adin_pool;
|
|
|
|
/*
|
|
* The following parameters are setup by the per-type "init" function.
|
|
*
|
|
* The supported types and their init functions are:
|
|
* (1) CTR_DRBG: drbg_ctr_init().
|
|
* (2) HMAC_DRBG: drbg_hmac_init().
|
|
* (3) HASH_DRBG: drbg_hash_init().
|
|
*
|
|
* The parameters are closely related to the ones described in
|
|
* section '10.2.1 CTR_DRBG' of [NIST SP 800-90Ar1], with one
|
|
* crucial difference: In the NIST standard, all counts are given
|
|
* in bits, whereas in OpenSSL entropy counts are given in bits
|
|
* and buffer lengths are given in bytes.
|
|
*
|
|
* Since this difference has lead to some confusion in the past,
|
|
* (see [GitHub Issue #2443], formerly [rt.openssl.org #4055])
|
|
* the 'len' suffix has been added to all buffer sizes for
|
|
* clarification.
|
|
*/
|
|
|
|
unsigned int strength;
|
|
size_t max_request;
|
|
size_t min_entropylen, max_entropylen;
|
|
size_t min_noncelen, max_noncelen;
|
|
size_t max_perslen, max_adinlen;
|
|
|
|
/*
|
|
* Counts the number of generate requests since the last reseed
|
|
* (Starts at 1). This value is the reseed_counter as defined in
|
|
* NIST SP 800-90Ar1
|
|
*/
|
|
unsigned int reseed_gen_counter;
|
|
/*
|
|
* Maximum number of generate requests until a reseed is required.
|
|
* This value is ignored if it is zero.
|
|
*/
|
|
unsigned int reseed_interval;
|
|
/* Stores the time when the last reseeding occurred */
|
|
time_t reseed_time;
|
|
/*
|
|
* Specifies the maximum time interval (in seconds) between reseeds.
|
|
* This value is ignored if it is zero.
|
|
*/
|
|
time_t reseed_time_interval;
|
|
/*
|
|
* Counts the number of reseeds since instantiation.
|
|
* This value is ignored if it is zero.
|
|
*
|
|
* This counter is used only for seed propagation from the <master> DRBG
|
|
* to its two children, the <public> and <private> DRBG. This feature is
|
|
* very special and its sole purpose is to ensure that any randomness which
|
|
* is added by PROV_add() or PROV_seed() will have an immediate effect on
|
|
* the output of PROV_bytes() resp. PROV_priv_bytes().
|
|
*/
|
|
TSAN_QUALIFIER unsigned int reseed_counter;
|
|
unsigned int reseed_next_counter;
|
|
unsigned int parent_reseed_counter;
|
|
|
|
size_t seedlen;
|
|
DRBG_STATUS state;
|
|
|
|
/* DRBG specific data */
|
|
void *data;
|
|
|
|
/* Entropy and nonce gathering callbacks */
|
|
void *callback_arg;
|
|
OSSL_INOUT_CALLBACK *get_entropy_fn;
|
|
OSSL_CALLBACK *cleanup_entropy_fn;
|
|
OSSL_INOUT_CALLBACK *get_nonce_fn;
|
|
OSSL_CALLBACK *cleanup_nonce_fn;
|
|
};
|
|
|
|
PROV_DRBG *prov_rand_drbg_new
|
|
(void *provctx, void *parent, const OSSL_DISPATCH *parent_dispatch,
|
|
int (*dnew)(PROV_DRBG *ctx),
|
|
int (*instantiate)(PROV_DRBG *drbg,
|
|
const unsigned char *entropy, size_t entropylen,
|
|
const unsigned char *nonce, size_t noncelen,
|
|
const unsigned char *pers, size_t perslen),
|
|
int (*uninstantiate)(PROV_DRBG *ctx),
|
|
int (*reseed)(PROV_DRBG *drbg, const unsigned char *ent, size_t ent_len,
|
|
const unsigned char *adin, size_t adin_len),
|
|
int (*generate)(PROV_DRBG *, unsigned char *out, size_t outlen,
|
|
const unsigned char *adin, size_t adin_len));
|
|
void prov_rand_drbg_free(PROV_DRBG *drbg);
|
|
|
|
int PROV_DRBG_instantiate(PROV_DRBG *drbg, unsigned int strength,
|
|
int prediction_resistance,
|
|
const unsigned char *pers, size_t perslen);
|
|
|
|
int PROV_DRBG_uninstantiate(PROV_DRBG *drbg);
|
|
|
|
int PROV_DRBG_reseed(PROV_DRBG *drbg, int prediction_resistance,
|
|
const unsigned char *ent, size_t ent_len,
|
|
const unsigned char *adin, size_t adinlen);
|
|
|
|
int PROV_DRBG_generate(PROV_DRBG *drbg, unsigned char *out, size_t outlen,
|
|
unsigned int strength, int prediction_resistance,
|
|
const unsigned char *adin, size_t adinlen);
|
|
|
|
/* Verify that an array of numeric values is all zero */
|
|
#define PROV_DRBG_VERYIFY_ZEROIZATION(v) \
|
|
{ \
|
|
size_t i; \
|
|
\
|
|
for (i = 0; i < OSSL_NELEM(v); i++) \
|
|
if ((v)[i] != 0) \
|
|
return 0; \
|
|
}
|
|
|
|
/* locking api */
|
|
OSSL_FUNC_rand_enable_locking_fn drbg_enable_locking;
|
|
OSSL_FUNC_rand_lock_fn drbg_lock;
|
|
OSSL_FUNC_rand_unlock_fn drbg_unlock;
|
|
|
|
/* Common parameters for all of our DRBGs */
|
|
int drbg_get_ctx_params(PROV_DRBG *drbg, OSSL_PARAM params[]);
|
|
int drbg_set_ctx_params(PROV_DRBG *drbg, const OSSL_PARAM params[]);
|
|
|
|
#define OSSL_PARAM_DRBG_SETTABLE_CTX_COMMON \
|
|
OSSL_PARAM_uint(OSSL_DRBG_PARAM_RESEED_REQUESTS, NULL), \
|
|
OSSL_PARAM_uint64(OSSL_DRBG_PARAM_RESEED_TIME_INTERVAL, NULL)
|
|
|
|
#define OSSL_PARAM_DRBG_GETTABLE_CTX_COMMON \
|
|
OSSL_PARAM_int(OSSL_RAND_PARAM_STATE, NULL), \
|
|
OSSL_PARAM_uint(OSSL_RAND_PARAM_STRENGTH, NULL), \
|
|
OSSL_PARAM_size_t(OSSL_DRBG_PARAM_MAX_REQUEST, NULL), \
|
|
OSSL_PARAM_size_t(OSSL_DRBG_PARAM_MIN_ENTROPYLEN, NULL), \
|
|
OSSL_PARAM_size_t(OSSL_DRBG_PARAM_MAX_ENTROPYLEN, NULL), \
|
|
OSSL_PARAM_size_t(OSSL_DRBG_PARAM_MIN_NONCELEN, NULL), \
|
|
OSSL_PARAM_size_t(OSSL_DRBG_PARAM_MAX_NONCELEN, NULL), \
|
|
OSSL_PARAM_size_t(OSSL_DRBG_PARAM_MAX_PERSLEN, NULL), \
|
|
OSSL_PARAM_size_t(OSSL_DRBG_PARAM_MAX_ADINLEN, NULL), \
|
|
OSSL_PARAM_uint(OSSL_DRBG_PARAM_RESEED_CTR, NULL), \
|
|
OSSL_PARAM_time_t(OSSL_DRBG_PARAM_RESEED_TIME, NULL), \
|
|
OSSL_PARAM_uint(OSSL_DRBG_PARAM_RESEED_REQUESTS, NULL), \
|
|
OSSL_PARAM_uint64(OSSL_DRBG_PARAM_RESEED_TIME_INTERVAL, NULL)
|
|
|
|
/* Continuous test "entropy" calls */
|
|
size_t prov_crngt_get_entropy(PROV_DRBG *drbg,
|
|
unsigned char **pout,
|
|
int entropy, size_t min_len, size_t max_len,
|
|
int prediction_resistance);
|
|
void prov_crngt_cleanup_entropy(PROV_DRBG *drbg,
|
|
unsigned char *out, size_t outlen);
|
|
|
|
#endif
|