mirror of
https://github.com/espressif/mbedtls.git
synced 2026-06-05 21:14:47 +00:00
216 lines
7.8 KiB
C
216 lines
7.8 KiB
C
/** \file psa_crypto_random_impl.h
|
|
*
|
|
* \brief PSA crypto random generator implementation abstraction.
|
|
*/
|
|
/*
|
|
* Copyright The Mbed TLS Contributors
|
|
* SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later
|
|
*/
|
|
|
|
#ifndef TF_PSA_CRYPTO_PSA_CRYPTO_RANDOM_IMPL_H
|
|
#define TF_PSA_CRYPTO_PSA_CRYPTO_RANDOM_IMPL_H
|
|
|
|
#include "mbedtls/psa_util.h"
|
|
|
|
#if defined(MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG)
|
|
|
|
typedef mbedtls_psa_external_random_context_t mbedtls_psa_random_context_t;
|
|
|
|
#else /* MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG */
|
|
|
|
#include "mbedtls/private/entropy.h"
|
|
#include "mbedtls/private/error_common.h"
|
|
|
|
#if !defined(PSA_WANT_ALG_SHA_256)
|
|
MBEDTLS_STATIC_ASSERT(MBEDTLS_PSA_CRYPTO_RNG_HASH != PSA_ALG_SHA_256,
|
|
"SHA_256 used as the hash for the random generator, but not enabled");
|
|
#endif
|
|
|
|
#if !defined(PSA_WANT_ALG_SHA_512)
|
|
MBEDTLS_STATIC_ASSERT(MBEDTLS_PSA_CRYPTO_RNG_HASH != PSA_ALG_SHA_512,
|
|
"SHA_512 used as the hash for the random generator, but not enabled");
|
|
#endif
|
|
|
|
MBEDTLS_STATIC_ASSERT((MBEDTLS_PSA_CRYPTO_RNG_HASH == PSA_ALG_SHA_256) || \
|
|
(MBEDTLS_PSA_CRYPTO_RNG_HASH == PSA_ALG_SHA_512),
|
|
"Invalid hashing algorithm for MBEDTLS_PSA_CRYPTO_RNG_HASH");
|
|
|
|
MBEDTLS_STATIC_ASSERT(PSA_BYTES_TO_BITS(PSA_HASH_LENGTH(
|
|
MBEDTLS_PSA_CRYPTO_RNG_HASH))
|
|
>= MBEDTLS_PSA_CRYPTO_RNG_STRENGTH,
|
|
"The hash size (in bits) of MBEDTLS_PSA_CRYPTO_RNG_HASH must be at least MBEDTLS_PSA_CRYPTO_RNG_STRENGTH");
|
|
|
|
/* Choose a DRBG based on configuration and availability */
|
|
#if defined(MBEDTLS_CTR_DRBG_C)
|
|
|
|
#include "mbedtls/private/ctr_drbg.h"
|
|
|
|
#if (MBEDTLS_PSA_CRYPTO_RNG_STRENGTH > 128) && \
|
|
defined(MBEDTLS_AES_ONLY_128_BIT_KEY_LENGTH)
|
|
#error "CTR_DRBG cannot meet the configured RNG strength using only 128-bit AES keys."
|
|
#endif
|
|
|
|
#if MBEDTLS_PSA_CRYPTO_RNG_STRENGTH > PSA_BYTES_TO_BITS(MBEDTLS_CTR_DRBG_KEYSIZE)
|
|
#error "The CTR_DRBG key size (in bits) must be at least MBEDTLS_PSA_CRYPTO_RNG_STRENGTH"
|
|
#endif
|
|
|
|
#undef MBEDTLS_PSA_HMAC_DRBG_MD_TYPE
|
|
|
|
#elif defined(MBEDTLS_HMAC_DRBG_C)
|
|
|
|
#include "mbedtls/private/hmac_drbg.h"
|
|
#define MBEDTLS_PSA_HMAC_DRBG_MD_TYPE MBEDTLS_ENTROPY_MD
|
|
|
|
#else /* !MBEDTLS_CTR_DRBG_C && !MBEDTLS_HMAC_DRBG_C*/
|
|
|
|
#error "No DRBG module available for the psa_crypto module."
|
|
|
|
#endif /* !MBEDTLS_CTR_DRBG_C && !MBEDTLS_HMAC_DRBG_C*/
|
|
|
|
/* The maximum number of bytes that mbedtls_psa_get_random() is expected to return. */
|
|
#if defined(MBEDTLS_CTR_DRBG_C)
|
|
#define MBEDTLS_PSA_RANDOM_MAX_REQUEST MBEDTLS_CTR_DRBG_MAX_REQUEST
|
|
#elif defined(MBEDTLS_HMAC_DRBG_C)
|
|
#define MBEDTLS_PSA_RANDOM_MAX_REQUEST MBEDTLS_HMAC_DRBG_MAX_REQUEST
|
|
#endif
|
|
|
|
#if defined(MBEDTLS_CTR_DRBG_C)
|
|
typedef mbedtls_ctr_drbg_context mbedtls_psa_drbg_context_t;
|
|
#elif defined(MBEDTLS_HMAC_DRBG_C)
|
|
typedef mbedtls_hmac_drbg_context mbedtls_psa_drbg_context_t;
|
|
#endif /* !MBEDTLS_CTR_DRBG_C && !MBEDTLS_HMAC_DRBG_C */
|
|
|
|
typedef struct {
|
|
void (* entropy_init)(mbedtls_entropy_context *ctx);
|
|
void (* entropy_free)(mbedtls_entropy_context *ctx);
|
|
mbedtls_entropy_context entropy;
|
|
mbedtls_psa_drbg_context_t drbg;
|
|
#if defined(MBEDTLS_PLATFORM_IS_UNIXLIKE)
|
|
/* Fork protection: normally pid = getpid(). If the value changes,
|
|
* we are in a (grand)*child of the original process, so reseed
|
|
* the RNG to ensure that the child and the original process have
|
|
* distinct RNG states. See psa_random_internal_generate().
|
|
*
|
|
* The type is intmax_t, not pid_t, for portability reasons:
|
|
* pid_t is defined in `unistd.h`, but on some platforms, it may
|
|
* only be defined if a certain compatibility level is requested
|
|
* by defining a macro such as _POSIX_C_SOURCE or _XOPEN_SOURCE.
|
|
* The macro needs to be defined before any system header, which
|
|
* may be hard to do in some C files that include this header
|
|
* (e.g. test suites). So we sidestep this complication, at the
|
|
* cost of possibly a few more instructions to compare pid values.
|
|
*/
|
|
intmax_t pid;
|
|
#endif
|
|
} mbedtls_psa_random_context_t;
|
|
|
|
/** Initialize the PSA DRBG.
|
|
*
|
|
* \param p_rng Pointer to the Mbed TLS DRBG state.
|
|
*/
|
|
static inline void mbedtls_psa_drbg_init(mbedtls_psa_drbg_context_t *p_rng)
|
|
{
|
|
#if defined(MBEDTLS_CTR_DRBG_C)
|
|
mbedtls_ctr_drbg_init(p_rng);
|
|
#elif defined(MBEDTLS_HMAC_DRBG_C)
|
|
mbedtls_hmac_drbg_init(p_rng);
|
|
#endif
|
|
}
|
|
|
|
/** Deinitialize the PSA DRBG.
|
|
*
|
|
* \param p_rng Pointer to the Mbed TLS DRBG state.
|
|
*/
|
|
static inline void mbedtls_psa_drbg_free(mbedtls_psa_drbg_context_t *p_rng)
|
|
{
|
|
#if defined(MBEDTLS_CTR_DRBG_C)
|
|
mbedtls_ctr_drbg_free(p_rng);
|
|
#elif defined(MBEDTLS_HMAC_DRBG_C)
|
|
mbedtls_hmac_drbg_free(p_rng);
|
|
#endif
|
|
}
|
|
|
|
/** Seed the PSA DRBG.
|
|
*
|
|
* \param drbg_ctx The DRBG context to seed.
|
|
* It must be initialized but not active.
|
|
* \param entropy An entropy context to read the seed from.
|
|
* \param custom The personalization string.
|
|
* This can be \c NULL, in which case the personalization
|
|
* string is empty regardless of the value of \p len.
|
|
* \param len The length of the personalization string.
|
|
*
|
|
* \return \c 0 on success.
|
|
* \return An Mbed TLS error code (\c MBEDTLS_ERR_xxx) on failure.
|
|
*/
|
|
static inline int mbedtls_psa_drbg_seed(mbedtls_psa_drbg_context_t *drbg_ctx,
|
|
mbedtls_entropy_context *entropy,
|
|
const unsigned char *custom, size_t len)
|
|
{
|
|
#if defined(MBEDTLS_CTR_DRBG_C)
|
|
return mbedtls_ctr_drbg_seed(drbg_ctx, mbedtls_entropy_func, entropy, custom, len);
|
|
#elif defined(MBEDTLS_HMAC_DRBG_C)
|
|
const mbedtls_md_info_t *md_info = mbedtls_md_info_from_type(MBEDTLS_PSA_HMAC_DRBG_MD_TYPE);
|
|
return mbedtls_hmac_drbg_seed(drbg_ctx, md_info, mbedtls_entropy_func, entropy, custom, len);
|
|
#endif /* MBEDTLS_HMAC_DRBG_C */
|
|
}
|
|
|
|
/** Reseed the PSA DRBG.
|
|
*
|
|
* \param drbg_ctx The DRBG context to reseed.
|
|
* It must be active.
|
|
* \param additional Additional data to inject.
|
|
* \param len The length of \p additional in bytes.
|
|
* This can be 0 to simply reseed from the entropy source.
|
|
*
|
|
* \return \c 0 on success.
|
|
* \return An Mbed TLS error code (\c MBEDTLS_ERR_xxx) on failure.
|
|
*/
|
|
static inline int mbedtls_psa_drbg_reseed(mbedtls_psa_drbg_context_t *drbg_ctx,
|
|
const unsigned char *additional,
|
|
size_t len)
|
|
{
|
|
#if defined(MBEDTLS_CTR_DRBG_C)
|
|
return mbedtls_ctr_drbg_reseed(drbg_ctx, additional, len);
|
|
#elif defined(MBEDTLS_HMAC_DRBG_C)
|
|
return mbedtls_hmac_drbg_reseed(drbg_ctx, additional, len);
|
|
#endif
|
|
}
|
|
|
|
/** Deplete the PSA DRBG, i.e. cause it to reseed the next time it is used.
|
|
*
|
|
* \note This function is not thread-safe.
|
|
*
|
|
* \param drbg_ctx The DRBG context to deplete.
|
|
* It must be active.
|
|
*/
|
|
static inline void mbedtls_psa_drbg_deplete(mbedtls_psa_drbg_context_t *drbg_ctx)
|
|
{
|
|
drbg_ctx->reseed_counter = drbg_ctx->reseed_interval;
|
|
}
|
|
|
|
#if MBEDTLS_ENTROPY_TRUE_SOURCES > 0
|
|
/** Set prediction resistance in the PSA DRBG.
|
|
*
|
|
* \note This function is not thread-safe.
|
|
*
|
|
* \param drbg_ctx The DRBG context to reconfigure.
|
|
* It must be active.
|
|
* \param enabled \c 1 to enable, or \c 0 to disable.
|
|
*/
|
|
static inline void mbedtls_psa_drbg_set_prediction_resistance(
|
|
mbedtls_psa_drbg_context_t *drbg_ctx,
|
|
unsigned enabled)
|
|
{
|
|
#if defined(MBEDTLS_CTR_DRBG_C)
|
|
mbedtls_ctr_drbg_set_prediction_resistance(drbg_ctx, enabled);
|
|
#elif defined(MBEDTLS_HMAC_DRBG_C)
|
|
mbedtls_hmac_drbg_set_prediction_resistance(drbg_ctx, enabled);
|
|
#endif
|
|
}
|
|
#endif /* MBEDTLS_ENTROPY_TRUE_SOURCES > 0 */
|
|
|
|
#endif /* MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG */
|
|
|
|
#endif /* TF_PSA_CRYPTO_PSA_CRYPTO_RANDOM_IMPL_H */
|