mirror of https://github.com/ipxe/ipxe.git
[rng] Add RDRAND as an entropy source
Signed-off-by: Michael Brown <mcb30@ipxe.org>pull/898/head
parent
6a004be0cc
commit
cff857461b
|
@ -0,0 +1,99 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2023 Michael Brown <mbrown@fensystems.co.uk>.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2 of the
|
||||||
|
* License, or any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* You can also choose to distribute this program under the terms of
|
||||||
|
* the Unmodified Binary Distribution Licence (as given in the file
|
||||||
|
* COPYING.UBDL), provided that you have satisfied its requirements.
|
||||||
|
*/
|
||||||
|
|
||||||
|
FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
|
||||||
|
|
||||||
|
/** @file
|
||||||
|
*
|
||||||
|
* Hardware random number generator
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <ipxe/cpuid.h>
|
||||||
|
#include <ipxe/entropy.h>
|
||||||
|
|
||||||
|
/** Number of times to retry RDRAND instruction */
|
||||||
|
#define RDRAND_RETRY_COUNT 16
|
||||||
|
|
||||||
|
/** Colour for debug messages */
|
||||||
|
#define colour CPUID_FEATURES_INTEL_ECX_RDRAND
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enable entropy gathering
|
||||||
|
*
|
||||||
|
* @ret rc Return status code
|
||||||
|
*/
|
||||||
|
static int rdrand_entropy_enable ( void ) {
|
||||||
|
struct x86_features features;
|
||||||
|
|
||||||
|
/* Check that RDRAND is supported */
|
||||||
|
x86_features ( &features );
|
||||||
|
if ( ! ( features.intel.ecx & CPUID_FEATURES_INTEL_ECX_RDRAND ) ) {
|
||||||
|
DBGC ( colour, "RDRAND not supported\n" );
|
||||||
|
return -ENOTSUP;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Disable entropy gathering
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
static void rdrand_entropy_disable ( void ) {
|
||||||
|
|
||||||
|
/* Nothing to do */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get noise sample
|
||||||
|
*
|
||||||
|
* @ret noise Noise sample
|
||||||
|
* @ret rc Return status code
|
||||||
|
*/
|
||||||
|
static int rdrand_get_noise ( noise_sample_t *noise ) {
|
||||||
|
unsigned int result;
|
||||||
|
unsigned int discard_c;
|
||||||
|
unsigned int ok;
|
||||||
|
|
||||||
|
/* Issue RDRAND, retrying until CF is set */
|
||||||
|
__asm__ ( "\n1:\n\t"
|
||||||
|
"rdrand %0\n\t"
|
||||||
|
"sbb %1, %1\n\t"
|
||||||
|
"loopz 1b\n\t"
|
||||||
|
: "=r" ( result ), "=r" ( ok ), "=c" ( discard_c )
|
||||||
|
: "2" ( RDRAND_RETRY_COUNT ) );
|
||||||
|
if ( ! ok ) {
|
||||||
|
DBGC ( colour, "RDRAND failed to become ready\n" );
|
||||||
|
return -EBUSY;
|
||||||
|
}
|
||||||
|
|
||||||
|
*noise = result;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
PROVIDE_ENTROPY_INLINE ( rdrand, min_entropy_per_sample );
|
||||||
|
PROVIDE_ENTROPY ( rdrand, entropy_enable, rdrand_entropy_enable );
|
||||||
|
PROVIDE_ENTROPY ( rdrand, entropy_disable, rdrand_entropy_disable );
|
||||||
|
PROVIDE_ENTROPY ( rdrand, get_noise, rdrand_get_noise );
|
|
@ -10,5 +10,6 @@
|
||||||
FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
|
FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
|
||||||
|
|
||||||
#include <ipxe/rtc_entropy.h>
|
#include <ipxe/rtc_entropy.h>
|
||||||
|
#include <ipxe/rdrand.h>
|
||||||
|
|
||||||
#endif /* _BITS_ENTROPY_H */
|
#endif /* _BITS_ENTROPY_H */
|
||||||
|
|
|
@ -28,6 +28,7 @@ FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
|
||||||
#define ERRFILE_cpuid ( ERRFILE_ARCH | ERRFILE_CORE | 0x00110000 )
|
#define ERRFILE_cpuid ( ERRFILE_ARCH | ERRFILE_CORE | 0x00110000 )
|
||||||
#define ERRFILE_rdtsc_timer ( ERRFILE_ARCH | ERRFILE_CORE | 0x00120000 )
|
#define ERRFILE_rdtsc_timer ( ERRFILE_ARCH | ERRFILE_CORE | 0x00120000 )
|
||||||
#define ERRFILE_acpi_timer ( ERRFILE_ARCH | ERRFILE_CORE | 0x00130000 )
|
#define ERRFILE_acpi_timer ( ERRFILE_ARCH | ERRFILE_CORE | 0x00130000 )
|
||||||
|
#define ERRFILE_rdrand ( ERRFILE_ARCH | ERRFILE_CORE | 0x00140000 )
|
||||||
|
|
||||||
#define ERRFILE_bootsector ( ERRFILE_ARCH | ERRFILE_IMAGE | 0x00000000 )
|
#define ERRFILE_bootsector ( ERRFILE_ARCH | ERRFILE_IMAGE | 0x00000000 )
|
||||||
#define ERRFILE_bzimage ( ERRFILE_ARCH | ERRFILE_IMAGE | 0x00010000 )
|
#define ERRFILE_bzimage ( ERRFILE_ARCH | ERRFILE_IMAGE | 0x00010000 )
|
||||||
|
|
|
@ -39,6 +39,9 @@ struct x86_features {
|
||||||
/** Get standard features */
|
/** Get standard features */
|
||||||
#define CPUID_FEATURES 0x00000001UL
|
#define CPUID_FEATURES 0x00000001UL
|
||||||
|
|
||||||
|
/** RDRAND instruction is supported */
|
||||||
|
#define CPUID_FEATURES_INTEL_ECX_RDRAND 0x40000000UL
|
||||||
|
|
||||||
/** Hypervisor is present */
|
/** Hypervisor is present */
|
||||||
#define CPUID_FEATURES_INTEL_ECX_HYPERVISOR 0x80000000UL
|
#define CPUID_FEATURES_INTEL_ECX_HYPERVISOR 0x80000000UL
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
#ifndef _IPXE_RDRAND_H
|
||||||
|
#define _IPXE_RDRAND_H
|
||||||
|
|
||||||
|
/** @file
|
||||||
|
*
|
||||||
|
* Hardware random number generator
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <ipxe/drbg.h>
|
||||||
|
|
||||||
|
#ifdef ENTROPY_RDRAND
|
||||||
|
#define ENTROPY_PREFIX_rdrand
|
||||||
|
#else
|
||||||
|
#define ENTROPY_PREFIX_rdrand __rdrand_
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* min-entropy per sample
|
||||||
|
*
|
||||||
|
* @ret min_entropy min-entropy of each sample
|
||||||
|
*/
|
||||||
|
static inline __always_inline min_entropy_t
|
||||||
|
ENTROPY_INLINE ( rdrand, min_entropy_per_sample ) ( void ) {
|
||||||
|
|
||||||
|
/* Data returned by RDRAND is theoretically full entropy, up
|
||||||
|
* to a security strength of 128 bits.
|
||||||
|
*/
|
||||||
|
if ( DRBG_SECURITY_STRENGTH > 128 )
|
||||||
|
return 0;
|
||||||
|
return MIN_ENTROPY ( 8 * sizeof ( noise_sample_t ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* _IPXE_RDRAND_H */
|
Loading…
Reference in New Issue