RngGetBytes() relies on the RngLib. The RngLib might use the RNDR instruction if the FEAT_RNG feature is present. RngGetInfo and RngGetRNG both must check that RngGetBytes() is working before advertising/using it. To do so, allocate an array storing the available algorithms. The Rng algorithm at the lowest index will be the default Rng algorithm. The array is shared between RngGetInfo and RngGetRNG. This array is allocated when the driver is loaded, and freed when unloaded. This patch also prevents from having PcdCpuRngSupportedAlgorithm let to a zero GUID, but let the possibility to have no valid Rng algorithm in such case. Signed-off-by: Pierre Gondois <Pierre.Gondois@arm.com> Acked-by: Jiewen Yao <jiewen.yao@intel.com>
151 lines
3.8 KiB
C
151 lines
3.8 KiB
C
/** @file
|
|
RNG Driver to produce the UEFI Random Number Generator protocol.
|
|
|
|
The driver uses CPU RNG instructions to produce high-quality,
|
|
high-performance entropy and random number.
|
|
|
|
RNG Algorithms defined in UEFI 2.4:
|
|
- EFI_RNG_ALGORITHM_SP800_90_CTR_256_GUID
|
|
- EFI_RNG_ALGORITHM_RAW
|
|
- EFI_RNG_ALGORITHM_SP800_90_HMAC_256_GUID
|
|
- EFI_RNG_ALGORITHM_SP800_90_HASH_256_GUID
|
|
- EFI_RNG_ALGORITHM_X9_31_3DES_GUID
|
|
- EFI_RNG_ALGORITHM_X9_31_AES_GUID
|
|
|
|
Copyright (c) 2013 - 2018, Intel Corporation. All rights reserved.<BR>
|
|
(C) Copyright 2015 Hewlett Packard Enterprise Development LP<BR>
|
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
|
|
|
**/
|
|
|
|
#include <Library/BaseLib.h>
|
|
#include <Library/BaseMemoryLib.h>
|
|
#include <Library/UefiBootServicesTableLib.h>
|
|
#include <Library/RngLib.h>
|
|
#include <Protocol/Rng.h>
|
|
|
|
#include "RngDxeInternals.h"
|
|
|
|
//
|
|
// Array containing the validated Rng algorithm.
|
|
// The entry with the lowest index will be the default algorithm.
|
|
//
|
|
UINTN mAvailableAlgoArrayCount;
|
|
EFI_RNG_ALGORITHM *mAvailableAlgoArray;
|
|
|
|
//
|
|
// The Random Number Generator (RNG) protocol
|
|
//
|
|
EFI_RNG_PROTOCOL mRngRdRand = {
|
|
RngGetInfo,
|
|
RngGetRNG
|
|
};
|
|
|
|
/**
|
|
The user Entry Point for the Random Number Generator (RNG) driver.
|
|
|
|
@param[in] ImageHandle The firmware allocated handle for the EFI image.
|
|
@param[in] SystemTable A pointer to the EFI System Table.
|
|
|
|
@retval EFI_SUCCESS The entry point is executed successfully.
|
|
@retval EFI_NOT_SUPPORTED Platform does not support RNG.
|
|
@retval Other Some error occurs when executing this entry point.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
EFIAPI
|
|
RngDriverEntry (
|
|
IN EFI_HANDLE ImageHandle,
|
|
IN EFI_SYSTEM_TABLE *SystemTable
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
EFI_HANDLE Handle;
|
|
|
|
//
|
|
// Install UEFI RNG (Random Number Generator) Protocol
|
|
//
|
|
Handle = NULL;
|
|
Status = gBS->InstallMultipleProtocolInterfaces (
|
|
&Handle,
|
|
&gEfiRngProtocolGuid,
|
|
&mRngRdRand,
|
|
NULL
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
//
|
|
// Get the list of available algorithm.
|
|
//
|
|
return GetAvailableAlgorithms ();
|
|
}
|
|
|
|
/**
|
|
This is the unload handle for RndgDxe module.
|
|
|
|
Disconnect the driver specified by ImageHandle from all the devices in the handle database.
|
|
Uninstall all the protocols installed in the driver entry point.
|
|
|
|
@param[in] ImageHandle The drivers' driver image.
|
|
|
|
@retval EFI_SUCCESS The image is unloaded.
|
|
@retval Others Failed to unload the image.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
EFIAPI
|
|
RngDriverUnLoad (
|
|
IN EFI_HANDLE ImageHandle
|
|
)
|
|
{
|
|
//
|
|
// Free the list of available algorithm.
|
|
//
|
|
FreeAvailableAlgorithms ();
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
Runs CPU RNG instruction to fill a buffer of arbitrary size with random bytes.
|
|
|
|
@param[in] Length Size of the buffer, in bytes, to fill with.
|
|
@param[out] RandBuffer Pointer to the buffer to store the random result.
|
|
|
|
@retval EFI_SUCCESS Random bytes generation succeeded.
|
|
@retval EFI_NOT_READY Failed to request random bytes.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
EFIAPI
|
|
RngGetBytes (
|
|
IN UINTN Length,
|
|
OUT UINT8 *RandBuffer
|
|
)
|
|
{
|
|
BOOLEAN IsRandom;
|
|
UINT64 TempRand[2];
|
|
|
|
while (Length > 0) {
|
|
IsRandom = GetRandomNumber128 (TempRand);
|
|
if (!IsRandom) {
|
|
return EFI_NOT_READY;
|
|
}
|
|
|
|
if (Length >= sizeof (TempRand)) {
|
|
WriteUnaligned64 ((UINT64 *)RandBuffer, TempRand[0]);
|
|
RandBuffer += sizeof (UINT64);
|
|
WriteUnaligned64 ((UINT64 *)RandBuffer, TempRand[1]);
|
|
RandBuffer += sizeof (UINT64);
|
|
Length -= sizeof (TempRand);
|
|
} else {
|
|
CopyMem (RandBuffer, TempRand, Length);
|
|
Length = 0;
|
|
}
|
|
}
|
|
|
|
return EFI_SUCCESS;
|
|
}
|