2021-09-13 16:20:57 +02:00
|
|
|
/** @file
|
|
|
|
TPM Platform Hierarchy configuration library.
|
|
|
|
|
|
|
|
This library provides functions for customizing the TPM's Platform Hierarchy
|
|
|
|
Authorization Value (platformAuth) and Platform Hierarchy Authorization
|
|
|
|
Policy (platformPolicy) can be defined through this function.
|
|
|
|
|
|
|
|
Copyright (c) 2019, Intel Corporation. All rights reserved.<BR>
|
|
|
|
Copyright (c) Microsoft Corporation.<BR>
|
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
|
|
|
|
|
|
|
@par Specification Reference:
|
|
|
|
https://trustedcomputinggroup.org/resource/tcg-tpm-v2-0-provisioning-guidance/
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <Uefi.h>
|
|
|
|
|
|
|
|
#include <Library/BaseMemoryLib.h>
|
|
|
|
#include <Library/DebugLib.h>
|
|
|
|
#include <Library/MemoryAllocationLib.h>
|
|
|
|
#include <Library/RngLib.h>
|
|
|
|
#include <Library/Tpm2CommandLib.h>
|
|
|
|
#include <Library/Tpm2DeviceLib.h>
|
|
|
|
|
|
|
|
//
|
|
|
|
// The authorization value may be no larger than the digest produced by the hash
|
|
|
|
// algorithm used for context integrity.
|
|
|
|
//
|
|
|
|
|
2021-12-05 23:54:12 +01:00
|
|
|
UINT16 mAuthSize;
|
2021-09-13 16:20:57 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
Generate high-quality entropy source through RDRAND.
|
|
|
|
|
|
|
|
@param[in] Length Size of the buffer, in bytes, to fill with.
|
|
|
|
@param[out] Entropy Pointer to the buffer to store the entropy data.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Entropy generation succeeded.
|
|
|
|
@retval EFI_NOT_READY Failed to request random data.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
RdRandGenerateEntropy (
|
2021-12-05 23:54:12 +01:00
|
|
|
IN UINTN Length,
|
|
|
|
OUT UINT8 *Entropy
|
2021-09-13 16:20:57 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN BlockCount;
|
|
|
|
UINT64 Seed[2];
|
|
|
|
UINT8 *Ptr;
|
|
|
|
|
2021-12-05 23:54:12 +01:00
|
|
|
Status = EFI_NOT_READY;
|
|
|
|
BlockCount = Length / sizeof (Seed);
|
|
|
|
Ptr = (UINT8 *)Entropy;
|
2021-09-13 16:20:57 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Generate high-quality seed for DRBG Entropy
|
|
|
|
//
|
|
|
|
while (BlockCount > 0) {
|
|
|
|
Status = GetRandomNumber128 (Seed);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2021-12-05 23:54:12 +01:00
|
|
|
|
|
|
|
CopyMem (Ptr, Seed, sizeof (Seed));
|
2021-09-13 16:20:57 +02:00
|
|
|
|
|
|
|
BlockCount--;
|
2021-12-05 23:54:12 +01:00
|
|
|
Ptr = Ptr + sizeof (Seed);
|
2021-09-13 16:20:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Populate the remained data as request.
|
|
|
|
//
|
|
|
|
Status = GetRandomNumber128 (Seed);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2021-12-05 23:54:12 +01:00
|
|
|
|
|
|
|
CopyMem (Ptr, Seed, (Length % sizeof (Seed)));
|
2021-09-13 16:20:57 +02:00
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function returns the maximum size of TPM2B_AUTH; this structure is used for an authorization value
|
|
|
|
and limits an authValue to being no larger than the largest digest produced by a TPM.
|
|
|
|
|
|
|
|
@param[out] AuthSize Tpm2 Auth size
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Auth size returned.
|
|
|
|
@retval EFI_DEVICE_ERROR Can not return platform auth due to device error.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
GetAuthSize (
|
2021-12-05 23:54:12 +01:00
|
|
|
OUT UINT16 *AuthSize
|
2021-09-13 16:20:57 +02:00
|
|
|
)
|
|
|
|
{
|
2021-12-05 23:54:12 +01:00
|
|
|
EFI_STATUS Status;
|
|
|
|
TPML_PCR_SELECTION Pcrs;
|
|
|
|
UINTN Index;
|
|
|
|
UINT16 DigestSize;
|
2021-09-13 16:20:57 +02:00
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
while (mAuthSize == 0) {
|
|
|
|
mAuthSize = SHA1_DIGEST_SIZE;
|
|
|
|
ZeroMem (&Pcrs, sizeof (TPML_PCR_SELECTION));
|
|
|
|
Status = Tpm2GetCapabilityPcrs (&Pcrs);
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "Tpm2GetCapabilityPcrs fail!\n"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG ((DEBUG_ERROR, "Tpm2GetCapabilityPcrs - %08x\n", Pcrs.count));
|
|
|
|
|
|
|
|
for (Index = 0; Index < Pcrs.count; Index++) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "alg - %x\n", Pcrs.pcrSelections[Index].hash));
|
|
|
|
|
|
|
|
switch (Pcrs.pcrSelections[Index].hash) {
|
2021-12-05 23:54:12 +01:00
|
|
|
case TPM_ALG_SHA1:
|
|
|
|
DigestSize = SHA1_DIGEST_SIZE;
|
|
|
|
break;
|
|
|
|
case TPM_ALG_SHA256:
|
|
|
|
DigestSize = SHA256_DIGEST_SIZE;
|
|
|
|
break;
|
|
|
|
case TPM_ALG_SHA384:
|
|
|
|
DigestSize = SHA384_DIGEST_SIZE;
|
|
|
|
break;
|
|
|
|
case TPM_ALG_SHA512:
|
|
|
|
DigestSize = SHA512_DIGEST_SIZE;
|
|
|
|
break;
|
|
|
|
case TPM_ALG_SM3_256:
|
|
|
|
DigestSize = SM3_256_DIGEST_SIZE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DigestSize = SHA1_DIGEST_SIZE;
|
|
|
|
break;
|
2021-09-13 16:20:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (DigestSize > mAuthSize) {
|
|
|
|
mAuthSize = DigestSize;
|
|
|
|
}
|
|
|
|
}
|
2021-12-05 23:54:12 +01:00
|
|
|
|
2021-09-13 16:20:57 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
*AuthSize = mAuthSize;
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Set PlatformAuth to random value.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
RandomizePlatformAuth (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
2021-12-05 23:54:12 +01:00
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT16 AuthSize;
|
|
|
|
TPM2B_AUTH NewPlatformAuth;
|
2021-09-13 16:20:57 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Send Tpm2HierarchyChange Auth with random value to avoid PlatformAuth being null
|
|
|
|
//
|
|
|
|
|
|
|
|
GetAuthSize (&AuthSize);
|
|
|
|
|
|
|
|
NewPlatformAuth.size = AuthSize;
|
|
|
|
|
|
|
|
//
|
2021-09-13 16:20:58 +02:00
|
|
|
// Create the random bytes in the destination buffer
|
2021-09-13 16:20:57 +02:00
|
|
|
//
|
|
|
|
|
2021-09-13 16:20:58 +02:00
|
|
|
RdRandGenerateEntropy (NewPlatformAuth.size, NewPlatformAuth.buffer);
|
2021-09-13 16:20:57 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Send Tpm2HierarchyChangeAuth command with the new Auth value
|
|
|
|
//
|
|
|
|
Status = Tpm2HierarchyChangeAuth (TPM_RH_PLATFORM, NULL, &NewPlatformAuth);
|
|
|
|
DEBUG ((DEBUG_INFO, "Tpm2HierarchyChangeAuth Result: - %r\n", Status));
|
|
|
|
ZeroMem (NewPlatformAuth.buffer, AuthSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Disable the TPM platform hierarchy.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The TPM was disabled successfully.
|
|
|
|
@retval Others An error occurred attempting to disable the TPM platform hierarchy.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
DisableTpmPlatformHierarchy (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
// Make sure that we have use of the TPM.
|
|
|
|
Status = Tpm2RequestUseTpm ();
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "%a:%a() - Tpm2RequestUseTpm Failed! %r\n", gEfiCallerBaseName, __FUNCTION__, Status));
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Let's do what we can to shut down the hierarchies.
|
|
|
|
|
|
|
|
// Disable the PH NV.
|
|
|
|
// IMPORTANT NOTE: We *should* be able to disable the PH NV here, but TPM parts have
|
|
|
|
// been known to store the EK cert in the PH NV. If we disable it, the
|
|
|
|
// EK cert will be unreadable.
|
|
|
|
|
|
|
|
// Disable the PH.
|
|
|
|
Status = Tpm2HierarchyControl (
|
|
|
|
TPM_RH_PLATFORM, // AuthHandle
|
|
|
|
NULL, // AuthSession
|
|
|
|
TPM_RH_PLATFORM, // Hierarchy
|
|
|
|
NO // State
|
|
|
|
);
|
|
|
|
DEBUG ((DEBUG_VERBOSE, "%a:%a() - Disable PH = %r\n", gEfiCallerBaseName, __FUNCTION__, Status));
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "%a:%a() - Disable PH Failed! %r\n", gEfiCallerBaseName, __FUNCTION__, Status));
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This service defines the configuration of the Platform Hierarchy Authorization Value (platformAuth)
|
2021-09-13 16:21:00 +02:00
|
|
|
and Platform Hierarchy Authorization Policy (platformPolicy).
|
2021-09-13 16:20:57 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
ConfigureTpmPlatformHierarchy (
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (PcdGetBool (PcdRandomizePlatformHierarchy)) {
|
|
|
|
//
|
|
|
|
// Send Tpm2HierarchyChange Auth with random value to avoid PlatformAuth being null
|
|
|
|
//
|
|
|
|
RandomizePlatformAuth ();
|
|
|
|
} else {
|
|
|
|
//
|
|
|
|
// Disable the hierarchy entirely (do not randomize it)
|
|
|
|
//
|
|
|
|
DisableTpmPlatformHierarchy ();
|
|
|
|
}
|
|
|
|
}
|