2021-08-02 12:46:23 +02:00
|
|
|
/** @file
|
|
|
|
This library provides helper functions to set/clear Secure Boot
|
|
|
|
keys and databases.
|
|
|
|
|
|
|
|
Copyright (c) 2011 - 2018, Intel Corporation. All rights reserved.<BR>
|
|
|
|
(C) Copyright 2018 Hewlett Packard Enterprise Development LP<BR>
|
|
|
|
Copyright (c) 2021, ARM Ltd. All rights reserved.<BR>
|
|
|
|
Copyright (c) 2021, Semihalf All rights reserved.<BR>
|
2022-04-11 00:35:59 +02:00
|
|
|
Copyright (c) Microsoft Corporation.
|
2021-08-02 12:46:23 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
|
|
|
**/
|
2022-04-11 00:35:59 +02:00
|
|
|
#include <Uefi.h>
|
2022-04-15 22:38:11 +02:00
|
|
|
#include <UefiSecureBoot.h>
|
2021-08-02 12:46:23 +02:00
|
|
|
#include <Guid/GlobalVariable.h>
|
|
|
|
#include <Guid/AuthenticatedVariableFormat.h>
|
|
|
|
#include <Guid/ImageAuthentication.h>
|
|
|
|
#include <Library/BaseLib.h>
|
|
|
|
#include <Library/BaseMemoryLib.h>
|
|
|
|
#include <Library/DebugLib.h>
|
|
|
|
#include <Library/UefiLib.h>
|
|
|
|
#include <Library/MemoryAllocationLib.h>
|
|
|
|
#include <Library/UefiRuntimeServicesTableLib.h>
|
|
|
|
#include <Library/SecureBootVariableLib.h>
|
2022-04-15 22:03:22 +02:00
|
|
|
#include <Library/PlatformPKProtectionLib.h>
|
2021-08-02 12:46:23 +02:00
|
|
|
|
2022-04-11 00:35:59 +02:00
|
|
|
// This time can be used when deleting variables, as it should be greater than any variable time.
|
|
|
|
EFI_TIME mMaxTimestamp = {
|
|
|
|
0xFFFF, // Year
|
|
|
|
0xFF, // Month
|
|
|
|
0xFF, // Day
|
|
|
|
0xFF, // Hour
|
|
|
|
0xFF, // Minute
|
|
|
|
0xFF, // Second
|
|
|
|
0x00,
|
|
|
|
0x00000000, // Nanosecond
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0x00
|
|
|
|
};
|
|
|
|
|
2022-04-15 22:03:22 +02:00
|
|
|
//
|
|
|
|
// This epoch time is the date that is used when creating SecureBoot default variables.
|
|
|
|
// NOTE: This is a placeholder date that doesn't correspond to anything else.
|
|
|
|
//
|
|
|
|
EFI_TIME mDefaultPayloadTimestamp = {
|
|
|
|
1970, // Year (1970)
|
|
|
|
1, // Month (Jan)
|
|
|
|
1, // Day (1)
|
|
|
|
0, // Hour
|
|
|
|
0, // Minute
|
|
|
|
0, // Second
|
|
|
|
0, // Pad1
|
|
|
|
0, // Nanosecond
|
|
|
|
0, // Timezone (Dummy value)
|
|
|
|
0, // Daylight (Dummy value)
|
|
|
|
0 // Pad2
|
|
|
|
};
|
|
|
|
|
2021-08-02 12:46:23 +02:00
|
|
|
/** Creates EFI Signature List structure.
|
|
|
|
|
|
|
|
@param[in] Data A pointer to signature data.
|
|
|
|
@param[in] Size Size of signature data.
|
|
|
|
@param[out] SigList Created Signature List.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Signature List was created successfully.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES Failed to allocate memory.
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
CreateSigList (
|
|
|
|
IN VOID *Data,
|
|
|
|
IN UINTN Size,
|
|
|
|
OUT EFI_SIGNATURE_LIST **SigList
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN SigListSize;
|
|
|
|
EFI_SIGNATURE_LIST *TmpSigList;
|
|
|
|
EFI_SIGNATURE_DATA *SigData;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Allocate data for Signature Database
|
|
|
|
//
|
|
|
|
SigListSize = sizeof (EFI_SIGNATURE_LIST) + sizeof (EFI_SIGNATURE_DATA) - 1 + Size;
|
|
|
|
TmpSigList = (EFI_SIGNATURE_LIST *)AllocateZeroPool (SigListSize);
|
|
|
|
if (TmpSigList == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Only gEfiCertX509Guid type is supported
|
|
|
|
//
|
|
|
|
TmpSigList->SignatureListSize = (UINT32)SigListSize;
|
|
|
|
TmpSigList->SignatureSize = (UINT32)(sizeof (EFI_SIGNATURE_DATA) - 1 + Size);
|
|
|
|
TmpSigList->SignatureHeaderSize = 0;
|
|
|
|
CopyGuid (&TmpSigList->SignatureType, &gEfiCertX509Guid);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Copy key data
|
|
|
|
//
|
|
|
|
SigData = (EFI_SIGNATURE_DATA *)(TmpSigList + 1);
|
|
|
|
CopyGuid (&SigData->SignatureOwner, &gEfiGlobalVariableGuid);
|
|
|
|
CopyMem (&SigData->SignatureData[0], Data, Size);
|
|
|
|
|
|
|
|
*SigList = TmpSigList;
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Adds new signature list to signature database.
|
|
|
|
|
|
|
|
@param[in] SigLists A pointer to signature database.
|
|
|
|
@param[in] SigListAppend A signature list to be added.
|
|
|
|
@param[out] *SigListOut Created signature database.
|
|
|
|
@param[in, out] SigListsSize A size of created signature database.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Signature List was added successfully.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES Failed to allocate memory.
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
ConcatenateSigList (
|
|
|
|
IN EFI_SIGNATURE_LIST *SigLists,
|
|
|
|
IN EFI_SIGNATURE_LIST *SigListAppend,
|
|
|
|
OUT EFI_SIGNATURE_LIST **SigListOut,
|
|
|
|
IN OUT UINTN *SigListsSize
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_SIGNATURE_LIST *TmpSigList;
|
|
|
|
UINT8 *Offset;
|
|
|
|
UINTN NewSigListsSize;
|
|
|
|
|
|
|
|
NewSigListsSize = *SigListsSize + SigListAppend->SignatureListSize;
|
|
|
|
|
|
|
|
TmpSigList = (EFI_SIGNATURE_LIST *)AllocateZeroPool (NewSigListsSize);
|
|
|
|
if (TmpSigList == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
CopyMem (TmpSigList, SigLists, *SigListsSize);
|
|
|
|
|
|
|
|
Offset = (UINT8 *)TmpSigList;
|
|
|
|
Offset += *SigListsSize;
|
|
|
|
CopyMem ((VOID *)Offset, SigListAppend, SigListAppend->SignatureListSize);
|
|
|
|
|
|
|
|
*SigListsSize = NewSigListsSize;
|
|
|
|
*SigListOut = TmpSigList;
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-04-15 22:38:11 +02:00
|
|
|
Create a EFI Signature List with data supplied from input argument.
|
|
|
|
The input certificates from KeyInfo parameter should be DER-encoded
|
|
|
|
format.
|
2021-08-02 12:46:23 +02:00
|
|
|
|
|
|
|
@param[out] SigListsSize A pointer to size of signature list
|
2022-04-15 22:38:11 +02:00
|
|
|
@param[out] SigListOut A pointer to a callee-allocated buffer with signature lists
|
|
|
|
@param[in] KeyInfoCount The number of certificate pointer and size pairs inside KeyInfo.
|
|
|
|
@param[in] KeyInfo A pointer to all certificates, in the format of DER-encoded,
|
|
|
|
to be concatenated into signature lists.
|
2021-08-02 12:46:23 +02:00
|
|
|
|
2022-04-15 22:38:11 +02:00
|
|
|
@retval EFI_SUCCESS Created signature list from payload successfully.
|
2021-08-02 12:46:23 +02:00
|
|
|
@retval EFI_NOT_FOUND Section with key has not been found.
|
2022-04-15 22:38:11 +02:00
|
|
|
@retval EFI_INVALID_PARAMETER Embedded key has a wrong format or input pointers are NULL.
|
2021-08-02 12:46:23 +02:00
|
|
|
@retval Others Unexpected error happens.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
2022-04-15 22:38:11 +02:00
|
|
|
EFIAPI
|
|
|
|
SecureBootCreateDataFromInput (
|
|
|
|
OUT UINTN *SigListsSize,
|
|
|
|
OUT EFI_SIGNATURE_LIST **SigListOut,
|
|
|
|
IN UINTN KeyInfoCount,
|
|
|
|
IN CONST SECURE_BOOT_CERTIFICATE_INFO *KeyInfo
|
2021-08-02 12:46:23 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_SIGNATURE_LIST *EfiSig;
|
|
|
|
EFI_SIGNATURE_LIST *TmpEfiSig;
|
|
|
|
EFI_SIGNATURE_LIST *TmpEfiSig2;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
VOID *Buffer;
|
|
|
|
UINTN Size;
|
2022-04-15 22:38:11 +02:00
|
|
|
UINTN InputIndex;
|
2021-08-02 12:46:23 +02:00
|
|
|
UINTN KeyIndex;
|
|
|
|
|
2022-04-15 22:38:11 +02:00
|
|
|
if ((SigListOut == NULL) || (SigListsSize == NULL)) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((KeyInfoCount == 0) || (KeyInfo == NULL)) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
InputIndex = 0;
|
2021-08-02 12:46:23 +02:00
|
|
|
KeyIndex = 0;
|
|
|
|
EfiSig = NULL;
|
|
|
|
*SigListsSize = 0;
|
2022-04-15 22:38:11 +02:00
|
|
|
while (InputIndex < KeyInfoCount) {
|
|
|
|
if (KeyInfo[InputIndex].Data != NULL) {
|
|
|
|
Size = KeyInfo[InputIndex].DataSize;
|
|
|
|
Buffer = AllocateCopyPool (Size, KeyInfo[InputIndex].Data);
|
|
|
|
if (Buffer == NULL) {
|
2021-08-02 12:46:23 +02:00
|
|
|
if (EfiSig != NULL) {
|
|
|
|
FreePool (EfiSig);
|
|
|
|
}
|
2021-12-05 23:54:12 +01:00
|
|
|
|
2022-04-15 22:38:11 +02:00
|
|
|
return EFI_OUT_OF_RESOURCES;
|
2021-08-02 12:46:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Status = CreateSigList (Buffer, Size, &TmpEfiSig);
|
|
|
|
|
2022-04-15 22:38:11 +02:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
FreePool (Buffer);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-08-02 12:46:23 +02:00
|
|
|
//
|
|
|
|
// Concatenate lists if more than one section found
|
|
|
|
//
|
|
|
|
if (KeyIndex == 0) {
|
|
|
|
EfiSig = TmpEfiSig;
|
|
|
|
*SigListsSize = TmpEfiSig->SignatureListSize;
|
|
|
|
} else {
|
|
|
|
ConcatenateSigList (EfiSig, TmpEfiSig, &TmpEfiSig2, SigListsSize);
|
|
|
|
FreePool (EfiSig);
|
|
|
|
FreePool (TmpEfiSig);
|
|
|
|
EfiSig = TmpEfiSig2;
|
|
|
|
}
|
|
|
|
|
|
|
|
KeyIndex++;
|
|
|
|
FreePool (Buffer);
|
|
|
|
}
|
2021-12-05 23:54:12 +01:00
|
|
|
|
2022-04-15 22:38:11 +02:00
|
|
|
InputIndex++;
|
2021-08-02 12:46:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (KeyIndex == 0) {
|
|
|
|
return EFI_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
*SigListOut = EfiSig;
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Create a time based data payload by concatenating the EFI_VARIABLE_AUTHENTICATION_2
|
|
|
|
descriptor with the input data. NO authentication is required in this function.
|
|
|
|
|
|
|
|
@param[in, out] DataSize On input, the size of Data buffer in bytes.
|
|
|
|
On output, the size of data returned in Data
|
|
|
|
buffer in bytes.
|
|
|
|
@param[in, out] Data On input, Pointer to data buffer to be wrapped or
|
|
|
|
pointer to NULL to wrap an empty payload.
|
|
|
|
On output, Pointer to the new payload date buffer allocated from pool,
|
|
|
|
it's caller's responsibility to free the memory when finish using it.
|
2022-04-11 00:35:59 +02:00
|
|
|
@param[in] Time Pointer to time information to created time based payload.
|
2021-08-02 12:46:23 +02:00
|
|
|
|
|
|
|
@retval EFI_SUCCESS Create time based payload successfully.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES There are not enough memory resources to create time based payload.
|
|
|
|
@retval EFI_INVALID_PARAMETER The parameter is invalid.
|
|
|
|
@retval Others Unexpected error happens.
|
|
|
|
|
2022-04-11 00:35:59 +02:00
|
|
|
--*/
|
2021-08-02 12:46:23 +02:00
|
|
|
EFI_STATUS
|
2022-04-11 00:35:59 +02:00
|
|
|
EFIAPI
|
2021-08-02 12:46:23 +02:00
|
|
|
CreateTimeBasedPayload (
|
2022-04-11 00:35:59 +02:00
|
|
|
IN OUT UINTN *DataSize,
|
|
|
|
IN OUT UINT8 **Data,
|
|
|
|
IN EFI_TIME *Time
|
2021-08-02 12:46:23 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 *NewData;
|
|
|
|
UINT8 *Payload;
|
|
|
|
UINTN PayloadSize;
|
|
|
|
EFI_VARIABLE_AUTHENTICATION_2 *DescriptorData;
|
|
|
|
UINTN DescriptorSize;
|
2021-12-05 23:54:12 +01:00
|
|
|
|
2022-04-11 00:35:59 +02:00
|
|
|
if ((Data == NULL) || (DataSize == NULL) || (Time == NULL)) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a(), invalid arg\n", __func__));
|
2021-08-02 12:46:23 +02:00
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// In Setup mode or Custom mode, the variable does not need to be signed but the
|
|
|
|
// parameters to the SetVariable() call still need to be prepared as authenticated
|
|
|
|
// variable. So we create EFI_VARIABLE_AUTHENTICATED_2 descriptor without certificate
|
|
|
|
// data in it.
|
|
|
|
//
|
|
|
|
Payload = *Data;
|
|
|
|
PayloadSize = *DataSize;
|
|
|
|
|
|
|
|
DescriptorSize = OFFSET_OF (EFI_VARIABLE_AUTHENTICATION_2, AuthInfo) + OFFSET_OF (WIN_CERTIFICATE_UEFI_GUID, CertData);
|
|
|
|
NewData = (UINT8 *)AllocateZeroPool (DescriptorSize + PayloadSize);
|
|
|
|
if (NewData == NULL) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a() Out of resources.\n", __func__));
|
2021-08-02 12:46:23 +02:00
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((Payload != NULL) && (PayloadSize != 0)) {
|
|
|
|
CopyMem (NewData + DescriptorSize, Payload, PayloadSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
DescriptorData = (EFI_VARIABLE_AUTHENTICATION_2 *)(NewData);
|
|
|
|
|
2022-04-11 00:35:59 +02:00
|
|
|
CopyMem (&DescriptorData->TimeStamp, Time, sizeof (EFI_TIME));
|
2021-08-02 12:46:23 +02:00
|
|
|
|
|
|
|
DescriptorData->AuthInfo.Hdr.dwLength = OFFSET_OF (WIN_CERTIFICATE_UEFI_GUID, CertData);
|
|
|
|
DescriptorData->AuthInfo.Hdr.wRevision = 0x0200;
|
|
|
|
DescriptorData->AuthInfo.Hdr.wCertificateType = WIN_CERT_TYPE_EFI_GUID;
|
|
|
|
CopyGuid (&DescriptorData->AuthInfo.CertType, &gEfiCertPkcs7Guid);
|
|
|
|
|
|
|
|
if (Payload != NULL) {
|
|
|
|
FreePool (Payload);
|
2022-04-11 00:35:59 +02:00
|
|
|
Payload = NULL;
|
2021-08-02 12:46:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
*DataSize = DescriptorSize + PayloadSize;
|
|
|
|
*Data = NewData;
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Internal helper function to delete a Variable given its name and GUID, NO authentication
|
|
|
|
required.
|
|
|
|
|
|
|
|
@param[in] VariableName Name of the Variable.
|
|
|
|
@param[in] VendorGuid GUID of the Variable.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Variable deleted successfully.
|
|
|
|
@retval Others The driver failed to start the device.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
2022-04-11 00:35:59 +02:00
|
|
|
EFIAPI
|
2021-08-02 12:46:23 +02:00
|
|
|
DeleteVariable (
|
|
|
|
IN CHAR16 *VariableName,
|
|
|
|
IN EFI_GUID *VendorGuid
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
VOID *Variable;
|
|
|
|
UINT8 *Data;
|
|
|
|
UINTN DataSize;
|
|
|
|
UINT32 Attr;
|
|
|
|
|
|
|
|
GetVariable2 (VariableName, VendorGuid, &Variable, NULL);
|
|
|
|
if (Variable == NULL) {
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
2021-12-05 23:54:12 +01:00
|
|
|
|
2021-08-02 12:46:23 +02:00
|
|
|
FreePool (Variable);
|
|
|
|
|
|
|
|
Data = NULL;
|
|
|
|
DataSize = 0;
|
|
|
|
Attr = EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_RUNTIME_ACCESS | EFI_VARIABLE_BOOTSERVICE_ACCESS
|
|
|
|
| EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS;
|
|
|
|
|
2022-04-11 00:35:59 +02:00
|
|
|
Status = CreateTimeBasedPayload (&DataSize, &Data, &mMaxTimestamp);
|
2021-08-02 12:46:23 +02:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "Fail to create time-based data payload: %r", Status));
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
VariableName,
|
|
|
|
VendorGuid,
|
|
|
|
Attr,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
if (Data != NULL) {
|
|
|
|
FreePool (Data);
|
|
|
|
}
|
2021-12-05 23:54:12 +01:00
|
|
|
|
2021-08-02 12:46:23 +02:00
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
Set the platform secure boot mode into "Custom" or "Standard" mode.
|
|
|
|
|
|
|
|
@param[in] SecureBootMode New secure boot mode: STANDARD_SECURE_BOOT_MODE or
|
|
|
|
CUSTOM_SECURE_BOOT_MODE.
|
|
|
|
|
|
|
|
@return EFI_SUCCESS The platform has switched to the special mode successfully.
|
|
|
|
@return other Fail to operate the secure boot mode.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
2022-04-11 00:35:59 +02:00
|
|
|
EFIAPI
|
2021-08-02 12:46:23 +02:00
|
|
|
SetSecureBootMode (
|
|
|
|
IN UINT8 SecureBootMode
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return gRT->SetVariable (
|
|
|
|
EFI_CUSTOM_MODE_NAME,
|
|
|
|
&gEfiCustomModeEnableGuid,
|
|
|
|
EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS,
|
|
|
|
sizeof (UINT8),
|
|
|
|
&SecureBootMode
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Fetches the value of SetupMode variable.
|
|
|
|
|
|
|
|
@param[out] SetupMode Pointer to UINT8 for SetupMode output
|
|
|
|
|
|
|
|
@retval other Retval from GetVariable.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
GetSetupMode (
|
|
|
|
OUT UINT8 *SetupMode
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN Size;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Size = sizeof (*SetupMode);
|
|
|
|
Status = gRT->GetVariable (
|
|
|
|
EFI_SETUP_MODE_NAME,
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
NULL,
|
|
|
|
&Size,
|
|
|
|
SetupMode
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2022-04-15 22:03:22 +02:00
|
|
|
/**
|
|
|
|
Helper function to quickly determine whether SecureBoot is enabled.
|
|
|
|
|
|
|
|
@retval TRUE SecureBoot is verifiably enabled.
|
|
|
|
@retval FALSE SecureBoot is either disabled or an error prevented checking.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
IsSecureBootEnabled (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT8 *SecureBoot;
|
|
|
|
|
|
|
|
SecureBoot = NULL;
|
|
|
|
|
|
|
|
Status = GetEfiGlobalVariable2 (EFI_SECURE_BOOT_MODE_NAME, (VOID **)&SecureBoot, NULL);
|
|
|
|
//
|
|
|
|
// Skip verification if SecureBoot variable doesn't exist.
|
|
|
|
//
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "Cannot check SecureBoot variable %r \n ", Status));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Skip verification if SecureBoot is disabled but not AuditMode
|
|
|
|
//
|
|
|
|
if (*SecureBoot == SECURE_BOOT_MODE_DISABLE) {
|
|
|
|
FreePool (SecureBoot);
|
|
|
|
return FALSE;
|
|
|
|
} else {
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-02 12:46:23 +02:00
|
|
|
/**
|
|
|
|
Clears the content of the 'db' variable.
|
|
|
|
|
|
|
|
@retval EFI_OUT_OF_RESOURCES If memory allocation for EFI_VARIABLE_AUTHENTICATION_2 fails
|
|
|
|
while VendorGuid is NULL.
|
|
|
|
@retval other Errors from GetVariable2 (), GetTime () and SetVariable ()
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
DeleteDb (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Status = DeleteVariable (
|
|
|
|
EFI_IMAGE_SECURITY_DATABASE,
|
|
|
|
&gEfiImageSecurityDatabaseGuid
|
|
|
|
);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clears the content of the 'dbx' variable.
|
|
|
|
|
|
|
|
@retval EFI_OUT_OF_RESOURCES If memory allocation for EFI_VARIABLE_AUTHENTICATION_2 fails
|
|
|
|
while VendorGuid is NULL.
|
|
|
|
@retval other Errors from GetVariable2 (), GetTime () and SetVariable ()
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
DeleteDbx (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Status = DeleteVariable (
|
|
|
|
EFI_IMAGE_SECURITY_DATABASE1,
|
|
|
|
&gEfiImageSecurityDatabaseGuid
|
|
|
|
);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clears the content of the 'dbt' variable.
|
|
|
|
|
|
|
|
@retval EFI_OUT_OF_RESOURCES If memory allocation for EFI_VARIABLE_AUTHENTICATION_2 fails
|
|
|
|
while VendorGuid is NULL.
|
|
|
|
@retval other Errors from GetVariable2 (), GetTime () and SetVariable ()
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
DeleteDbt (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Status = DeleteVariable (
|
|
|
|
EFI_IMAGE_SECURITY_DATABASE2,
|
|
|
|
&gEfiImageSecurityDatabaseGuid
|
|
|
|
);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clears the content of the 'KEK' variable.
|
|
|
|
|
|
|
|
@retval EFI_OUT_OF_RESOURCES If memory allocation for EFI_VARIABLE_AUTHENTICATION_2 fails
|
|
|
|
while VendorGuid is NULL.
|
|
|
|
@retval other Errors from GetVariable2 (), GetTime () and SetVariable ()
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
DeleteKEK (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Status = DeleteVariable (
|
|
|
|
EFI_KEY_EXCHANGE_KEY_NAME,
|
|
|
|
&gEfiGlobalVariableGuid
|
|
|
|
);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Remove the PK variable.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Delete PK successfully.
|
|
|
|
@retval Others Could not allow to delete PK.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
DeletePlatformKey (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Status = SetSecureBootMode (CUSTOM_SECURE_BOOT_MODE);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = DeleteVariable (
|
|
|
|
EFI_PLATFORM_KEY_NAME,
|
|
|
|
&gEfiGlobalVariableGuid
|
|
|
|
);
|
|
|
|
return Status;
|
|
|
|
}
|
2022-04-15 22:03:22 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
This function will delete the secure boot keys, thus
|
|
|
|
disabling secure boot.
|
|
|
|
|
|
|
|
@return EFI_SUCCESS or underlying failure code.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
DeleteSecureBootVariables (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status, TempStatus;
|
|
|
|
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "%a - Attempting to delete the Secure Boot variables.\n", __func__));
|
2022-04-15 22:03:22 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Step 1: Notify that a PK update is coming shortly...
|
|
|
|
Status = DisablePKProtection ();
|
|
|
|
if (EFI_ERROR (Status)) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a - Failed to signal PK update start! %r\n", __func__, Status));
|
2022-04-15 22:03:22 +02:00
|
|
|
// Classify this as a PK deletion error.
|
|
|
|
Status = EFI_ABORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Step 2: Attempt to delete the PK.
|
|
|
|
// Let's try to nuke the PK, why not...
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Status = DeletePlatformKey ();
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "%a - PK Delete = %r\n", __func__, Status));
|
2022-04-15 22:03:22 +02:00
|
|
|
// If the PK is not found, then our work here is done.
|
|
|
|
if (Status == EFI_NOT_FOUND) {
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
// If any other error occurred, let's inform the caller that the PK delete in particular failed.
|
|
|
|
else if (EFI_ERROR (Status)) {
|
|
|
|
Status = EFI_ABORTED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Step 3: Attempt to delete remaining keys/databases...
|
|
|
|
// Now that the PK is deleted (assuming Status == EFI_SUCCESS) the system is in SETUP_MODE.
|
|
|
|
// Arguably we could leave these variables in place and let them be deleted by whoever wants to
|
|
|
|
// update all the SecureBoot variables. However, for cleanliness sake, let's try to
|
|
|
|
// get rid of them here.
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// If any of THESE steps have an error, report the error but attempt to delete all keys.
|
|
|
|
// Using TempStatus will prevent an error from being trampled by an EFI_SUCCESS.
|
|
|
|
// Overwrite Status ONLY if TempStatus is an error.
|
|
|
|
//
|
|
|
|
// If the error is EFI_NOT_FOUND, we can safely ignore it since we were trying to delete
|
|
|
|
// the variables anyway.
|
|
|
|
//
|
|
|
|
TempStatus = DeleteKEK ();
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "%a - KEK Delete = %r\n", __func__, TempStatus));
|
2022-04-15 22:03:22 +02:00
|
|
|
if (EFI_ERROR (TempStatus) && (TempStatus != EFI_NOT_FOUND)) {
|
|
|
|
Status = EFI_ACCESS_DENIED;
|
|
|
|
}
|
|
|
|
|
|
|
|
TempStatus = DeleteDb ();
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "%a - db Delete = %r\n", __func__, TempStatus));
|
2022-04-15 22:03:22 +02:00
|
|
|
if (EFI_ERROR (TempStatus) && (TempStatus != EFI_NOT_FOUND)) {
|
|
|
|
Status = EFI_ACCESS_DENIED;
|
|
|
|
}
|
|
|
|
|
|
|
|
TempStatus = DeleteDbx ();
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "%a - dbx Delete = %r\n", __func__, TempStatus));
|
2022-04-15 22:03:22 +02:00
|
|
|
if (EFI_ERROR (TempStatus) && (TempStatus != EFI_NOT_FOUND)) {
|
|
|
|
Status = EFI_ACCESS_DENIED;
|
|
|
|
}
|
|
|
|
|
|
|
|
TempStatus = DeleteDbt ();
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "%a - dbt Delete = %r\n", __func__, TempStatus));
|
2022-04-15 22:03:22 +02:00
|
|
|
if (EFI_ERROR (TempStatus) && (TempStatus != EFI_NOT_FOUND)) {
|
|
|
|
Status = EFI_ACCESS_DENIED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}// DeleteSecureBootVariables()
|
|
|
|
|
|
|
|
/**
|
|
|
|
A helper function to take in a variable payload, wrap it in the
|
|
|
|
proper authenticated variable structure, and install it in the
|
|
|
|
EFI variable space.
|
|
|
|
|
|
|
|
@param[in] VariableName The name of the key/database.
|
|
|
|
@param[in] VendorGuid The namespace (ie. vendor GUID) of the variable
|
|
|
|
@param[in] DataSize Size parameter for target secure boot variable.
|
|
|
|
@param[in] Data Pointer to signature list formatted secure boot variable content.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The enrollment for authenticated variable was successful.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES There are not enough memory resources to create time based payload.
|
|
|
|
@retval EFI_INVALID_PARAMETER The parameter is invalid.
|
|
|
|
@retval Others Unexpected error happens.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
EnrollFromInput (
|
|
|
|
IN CHAR16 *VariableName,
|
|
|
|
IN EFI_GUID *VendorGuid,
|
|
|
|
IN UINTN DataSize,
|
|
|
|
IN VOID *Data
|
|
|
|
)
|
|
|
|
{
|
|
|
|
VOID *Payload;
|
|
|
|
UINTN PayloadSize;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Payload = NULL;
|
|
|
|
|
|
|
|
if ((VariableName == NULL) || (VendorGuid == 0)) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "Input vendor variable invalid: %p and %p\n", VariableName, VendorGuid));
|
|
|
|
Status = EFI_INVALID_PARAMETER;
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((Data == NULL) || (DataSize == 0)) {
|
|
|
|
// You might as well just use DeleteVariable...
|
|
|
|
DEBUG ((DEBUG_ERROR, "Input argument invalid: %p: %x\n", Data, DataSize));
|
|
|
|
Status = EFI_INVALID_PARAMETER;
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bring in the noise...
|
|
|
|
PayloadSize = DataSize;
|
|
|
|
Payload = AllocateZeroPool (DataSize);
|
|
|
|
// Bring in the funk...
|
|
|
|
if (Payload == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
} else {
|
|
|
|
CopyMem (Payload, Data, DataSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = CreateTimeBasedPayload (&PayloadSize, (UINT8 **)&Payload, &mDefaultPayloadTimestamp);
|
|
|
|
if (EFI_ERROR (Status) || (Payload == NULL)) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "Fail to create time-based data payload: %r\n", Status));
|
|
|
|
Payload = NULL;
|
|
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Allocate memory for auth variable
|
|
|
|
//
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
VariableName,
|
|
|
|
VendorGuid,
|
|
|
|
(EFI_VARIABLE_NON_VOLATILE |
|
|
|
|
EFI_VARIABLE_BOOTSERVICE_ACCESS |
|
|
|
|
EFI_VARIABLE_RUNTIME_ACCESS |
|
|
|
|
EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS),
|
|
|
|
PayloadSize,
|
|
|
|
Payload
|
|
|
|
);
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
DEBUG ((
|
|
|
|
DEBUG_ERROR,
|
|
|
|
"error: %a (\"%s\", %g): %r\n",
|
2023-04-06 21:50:26 +02:00
|
|
|
__func__,
|
2022-04-15 22:03:22 +02:00
|
|
|
VariableName,
|
|
|
|
VendorGuid,
|
|
|
|
Status
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
Exit:
|
|
|
|
//
|
|
|
|
// Always Put Away Your Toys
|
|
|
|
// Payload will be reassigned by CreateTimeBasedPayload()...
|
|
|
|
if (Payload != NULL) {
|
|
|
|
FreePool (Payload);
|
|
|
|
Payload = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Similar to DeleteSecureBootVariables, this function is used to unilaterally
|
|
|
|
force the state of related SB variables (db, dbx, dbt, KEK, PK, etc.) to be
|
|
|
|
the built-in, hardcoded default vars.
|
|
|
|
|
|
|
|
@param[in] SecureBootPayload Payload information for secure boot related keys.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS SecureBoot keys are now set to defaults.
|
|
|
|
@retval EFI_ABORTED SecureBoot keys are not empty. Please delete keys first
|
|
|
|
or follow standard methods of altering keys (ie. use the signing system).
|
|
|
|
@retval EFI_SECURITY_VIOLATION Failed to create the PK.
|
|
|
|
@retval Others Something failed in one of the subfunctions.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SetSecureBootVariablesToDefault (
|
|
|
|
IN CONST SECURE_BOOT_PAYLOAD_INFO *SecureBootPayload
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT8 *Data;
|
|
|
|
UINTN DataSize;
|
|
|
|
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "%a() Entry\n", __func__));
|
2022-04-15 22:03:22 +02:00
|
|
|
|
|
|
|
if (SecureBootPayload == NULL) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a - Invalid SecureBoot payload is supplied!\n", __func__));
|
2022-04-15 22:03:22 +02:00
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Right off the bat, if SecureBoot is currently enabled, bail.
|
|
|
|
if (IsSecureBootEnabled ()) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a - Cannot set default keys while SecureBoot is enabled!\n", __func__));
|
2022-04-15 22:03:22 +02:00
|
|
|
return EFI_ABORTED;
|
|
|
|
}
|
|
|
|
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "%a - Setting up key %s!\n", __func__, SecureBootPayload->SecureBootKeyName));
|
2022-04-15 22:03:22 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Start running down the list, creating variables in our wake.
|
|
|
|
// dbx is a good place to start.
|
|
|
|
Data = (UINT8 *)SecureBootPayload->DbxPtr;
|
|
|
|
DataSize = SecureBootPayload->DbxSize;
|
|
|
|
Status = EnrollFromInput (
|
|
|
|
EFI_IMAGE_SECURITY_DATABASE1,
|
|
|
|
&gEfiImageSecurityDatabaseGuid,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
|
|
|
|
// If that went well, try the db (make sure to pick the right one!).
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Data = (UINT8 *)SecureBootPayload->DbPtr;
|
|
|
|
DataSize = SecureBootPayload->DbSize;
|
|
|
|
Status = EnrollFromInput (
|
|
|
|
EFI_IMAGE_SECURITY_DATABASE,
|
|
|
|
&gEfiImageSecurityDatabaseGuid,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a - Failed to enroll DB %r!\n", __func__, Status));
|
2022-04-15 22:03:22 +02:00
|
|
|
}
|
|
|
|
} else {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a - Failed to enroll DBX %r!\n", __func__, Status));
|
2022-04-15 22:03:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Keep it going. Keep it going. dbt if supplied...
|
|
|
|
if (!EFI_ERROR (Status) && (SecureBootPayload->DbtPtr != NULL)) {
|
|
|
|
Data = (UINT8 *)SecureBootPayload->DbtPtr;
|
|
|
|
DataSize = SecureBootPayload->DbtSize;
|
|
|
|
Status = EnrollFromInput (
|
|
|
|
EFI_IMAGE_SECURITY_DATABASE2,
|
|
|
|
&gEfiImageSecurityDatabaseGuid,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a - Failed to enroll DBT %r!\n", __func__, Status));
|
2022-04-15 22:03:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keep it going. Keep it going. KEK...
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Data = (UINT8 *)SecureBootPayload->KekPtr;
|
|
|
|
DataSize = SecureBootPayload->KekSize;
|
|
|
|
Status = EnrollFromInput (
|
|
|
|
EFI_KEY_EXCHANGE_KEY_NAME,
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a - Failed to enroll KEK %r!\n", __func__, Status));
|
2022-04-15 22:03:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Finally! The Big Daddy of them all.
|
|
|
|
// The PK!
|
|
|
|
//
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// Finally, install the key.
|
|
|
|
Data = (UINT8 *)SecureBootPayload->PkPtr;
|
|
|
|
DataSize = SecureBootPayload->PkSize;
|
|
|
|
Status = EnrollFromInput (
|
|
|
|
EFI_PLATFORM_KEY_NAME,
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Report PK creation errors.
|
|
|
|
if (EFI_ERROR (Status)) {
|
2023-04-06 21:50:26 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a - Failed to update the PK! - %r\n", __func__, Status));
|
2022-04-15 22:03:22 +02:00
|
|
|
Status = EFI_SECURITY_VIOLATION;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|