mirror of
https://github.com/acidanthera/audk.git
synced 2025-07-27 07:34:06 +02:00
Fix a bug that the invalid public key will never be removed from public key database.
Signed-off-by: Fu Siyuan <siyuan.fu@intel.com> Reviewed-by: Ye Ting <ting.ye@intel.com> Reviewed-by: Dong Guo <guo.dong@intel.com> git-svn-id: https://edk2.svn.sourceforge.net/svnroot/edk2/trunk/edk2@14041 6f19259b-4bc3-4df7-8a09-765794883524
This commit is contained in:
parent
8b5b3d421f
commit
83758cdc84
@ -15,7 +15,7 @@
|
|||||||
They will do basic validation for authentication data structure, then call crypto library
|
They will do basic validation for authentication data structure, then call crypto library
|
||||||
to verify the signature.
|
to verify the signature.
|
||||||
|
|
||||||
Copyright (c) 2009 - 2012, Intel Corporation. All rights reserved.<BR>
|
Copyright (c) 2009 - 2013, Intel Corporation. All rights reserved.<BR>
|
||||||
This program and the accompanying materials
|
This program and the accompanying materials
|
||||||
are licensed and made available under the terms and conditions of the BSD License
|
are licensed and made available under the terms and conditions of the BSD License
|
||||||
which accompanies this distribution. The full text of the license may be found at
|
which accompanies this distribution. The full text of the license may be found at
|
||||||
@ -437,6 +437,8 @@ AddPubKeyInStore (
|
|||||||
UINT32 Index;
|
UINT32 Index;
|
||||||
VARIABLE_POINTER_TRACK Variable;
|
VARIABLE_POINTER_TRACK Variable;
|
||||||
UINT8 *Ptr;
|
UINT8 *Ptr;
|
||||||
|
UINT8 *Data;
|
||||||
|
UINTN DataSize;
|
||||||
|
|
||||||
if (PubKey == NULL) {
|
if (PubKey == NULL) {
|
||||||
return 0;
|
return 0;
|
||||||
@ -468,9 +470,45 @@ AddPubKeyInStore (
|
|||||||
//
|
//
|
||||||
if (mPubKeyNumber == MAX_KEY_NUM) {
|
if (mPubKeyNumber == MAX_KEY_NUM) {
|
||||||
//
|
//
|
||||||
// Notes: Database is full, need enhancement here, currently just return 0.
|
// Public key dadatase is full, try to reclaim invalid key.
|
||||||
//
|
//
|
||||||
return 0;
|
if (AtRuntime ()) {
|
||||||
|
//
|
||||||
|
// NV storage can't reclaim at runtime.
|
||||||
|
//
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = Reclaim (
|
||||||
|
mVariableModuleGlobal->VariableGlobal.NonVolatileVariableBase,
|
||||||
|
&mVariableModuleGlobal->NonVolatileLastVariableOffset,
|
||||||
|
FALSE,
|
||||||
|
NULL,
|
||||||
|
TRUE,
|
||||||
|
TRUE
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = FindVariable (
|
||||||
|
AUTHVAR_KEYDB_NAME,
|
||||||
|
&gEfiAuthenticatedVariableGuid,
|
||||||
|
&Variable,
|
||||||
|
&mVariableModuleGlobal->VariableGlobal,
|
||||||
|
FALSE
|
||||||
|
);
|
||||||
|
ASSERT_EFI_ERROR (Status);
|
||||||
|
|
||||||
|
DataSize = DataSizeOfVariable (Variable.CurrPtr);
|
||||||
|
Data = GetVariableDataPtr (Variable.CurrPtr);
|
||||||
|
ASSERT ((DataSize != 0) && (Data != NULL));
|
||||||
|
CopyMem (mPubKeyStore, (UINT8 *) Data, DataSize);
|
||||||
|
mPubKeyNumber = (UINT32) (DataSize / EFI_CERT_TYPE_RSA2048_SIZE);
|
||||||
|
|
||||||
|
if (mPubKeyNumber == MAX_KEY_NUM) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
CopyMem (mPubKeyStore + mPubKeyNumber * EFI_CERT_TYPE_RSA2048_SIZE, PubKey, EFI_CERT_TYPE_RSA2048_SIZE);
|
CopyMem (mPubKeyStore + mPubKeyNumber * EFI_CERT_TYPE_RSA2048_SIZE, PubKey, EFI_CERT_TYPE_RSA2048_SIZE);
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
VariableServiceSetVariable() should also check authenticate data to avoid buffer overflow,
|
VariableServiceSetVariable() should also check authenticate data to avoid buffer overflow,
|
||||||
integer overflow. It should also check attribute to avoid authentication bypass.
|
integer overflow. It should also check attribute to avoid authentication bypass.
|
||||||
|
|
||||||
Copyright (c) 2009 - 2012, Intel Corporation. All rights reserved.<BR>
|
Copyright (c) 2009 - 2013, Intel Corporation. All rights reserved.<BR>
|
||||||
This program and the accompanying materials
|
This program and the accompanying materials
|
||||||
are licensed and made available under the terms and conditions of the BSD License
|
are licensed and made available under the terms and conditions of the BSD License
|
||||||
which accompanies this distribution. The full text of the license may be found at
|
which accompanies this distribution. The full text of the license may be found at
|
||||||
@ -511,21 +511,157 @@ GetEndPointer (
|
|||||||
return (VARIABLE_HEADER *) HEADER_ALIGN ((UINTN) VarStoreHeader + VarStoreHeader->Size);
|
return (VARIABLE_HEADER *) HEADER_ALIGN ((UINTN) VarStoreHeader + VarStoreHeader->Size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
|
||||||
|
Check the PubKeyIndex is a valid key or not.
|
||||||
|
|
||||||
|
This function will iterate the NV storage to see if this PubKeyIndex is still referenced
|
||||||
|
by any valid count-based auth variabe.
|
||||||
|
|
||||||
|
@param[in] PubKeyIndex Index of the public key in public key store.
|
||||||
|
|
||||||
|
@retval TRUE The PubKeyIndex is still in use.
|
||||||
|
@retval FALSE The PubKeyIndex is not referenced by any count-based auth variabe.
|
||||||
|
|
||||||
|
**/
|
||||||
|
BOOLEAN
|
||||||
|
IsValidPubKeyIndex (
|
||||||
|
IN UINT32 PubKeyIndex
|
||||||
|
)
|
||||||
|
{
|
||||||
|
VARIABLE_HEADER *Variable;
|
||||||
|
|
||||||
|
if (PubKeyIndex > mPubKeyNumber) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
Variable = GetStartPointer (mNvVariableCache);
|
||||||
|
|
||||||
|
while (IsValidVariableHeader (Variable)) {
|
||||||
|
if ((Variable->State == VAR_ADDED || Variable->State == (VAR_IN_DELETED_TRANSITION & VAR_ADDED)) &&
|
||||||
|
Variable->PubKeyIndex == PubKeyIndex) {
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
Variable = GetNextVariablePtr (Variable);
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
|
||||||
|
Get the number of valid public key in PubKeyStore.
|
||||||
|
|
||||||
|
@param[in] PubKeyNumber Number of the public key in public key store.
|
||||||
|
|
||||||
|
@return Number of valid public key in PubKeyStore.
|
||||||
|
|
||||||
|
**/
|
||||||
|
UINT32
|
||||||
|
GetValidPubKeyNumber (
|
||||||
|
IN UINT32 PubKeyNumber
|
||||||
|
)
|
||||||
|
{
|
||||||
|
UINT32 PubKeyIndex;
|
||||||
|
UINT32 Counter;
|
||||||
|
|
||||||
|
Counter = 0;
|
||||||
|
|
||||||
|
for (PubKeyIndex = 1; PubKeyIndex <= PubKeyNumber; PubKeyIndex++) {
|
||||||
|
if (IsValidPubKeyIndex (PubKeyIndex)) {
|
||||||
|
Counter++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Counter;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
|
||||||
|
Filter the useless key in public key store.
|
||||||
|
|
||||||
|
This function will find out all valid public keys in public key database, save them in new allocated
|
||||||
|
buffer NewPubKeyStore, and give the new PubKeyIndex. The caller is responsible for freeing buffer
|
||||||
|
NewPubKeyIndex and NewPubKeyStore with FreePool().
|
||||||
|
|
||||||
|
@param[in] PubKeyStore Point to the public key database.
|
||||||
|
@param[in] PubKeyNumber Number of the public key in PubKeyStore.
|
||||||
|
@param[out] NewPubKeyIndex Point to an array of new PubKeyIndex corresponds to NewPubKeyStore.
|
||||||
|
@param[out] NewPubKeyStore Saved all valid public keys in PubKeyStore.
|
||||||
|
@param[out] NewPubKeySize Buffer size of the NewPubKeyStore.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Trim operation is complete successfully.
|
||||||
|
@retval EFI_OUT_OF_RESOURCES No enough memory resources, or no useless key in PubKeyStore.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
PubKeyStoreFilter (
|
||||||
|
IN UINT8 *PubKeyStore,
|
||||||
|
IN UINT32 PubKeyNumber,
|
||||||
|
OUT UINT32 **NewPubKeyIndex,
|
||||||
|
OUT UINT8 **NewPubKeyStore,
|
||||||
|
OUT UINT32 *NewPubKeySize
|
||||||
|
)
|
||||||
|
{
|
||||||
|
UINT32 PubKeyIndex;
|
||||||
|
UINT32 CopiedKey;
|
||||||
|
UINT32 NewPubKeyNumber;
|
||||||
|
|
||||||
|
NewPubKeyNumber = GetValidPubKeyNumber (PubKeyNumber);
|
||||||
|
if (NewPubKeyNumber == PubKeyNumber) {
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (NewPubKeyNumber != 0) {
|
||||||
|
*NewPubKeySize = NewPubKeyNumber * EFI_CERT_TYPE_RSA2048_SIZE;
|
||||||
|
} else {
|
||||||
|
*NewPubKeySize = sizeof (UINT8);
|
||||||
|
}
|
||||||
|
|
||||||
|
*NewPubKeyStore = AllocatePool (*NewPubKeySize);
|
||||||
|
if (*NewPubKeyStore == NULL) {
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
|
||||||
|
*NewPubKeyIndex = AllocateZeroPool ((PubKeyNumber + 1) * sizeof (UINT32));
|
||||||
|
if (*NewPubKeyIndex == NULL) {
|
||||||
|
FreePool (*NewPubKeyStore);
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
|
||||||
|
CopiedKey = 0;
|
||||||
|
for (PubKeyIndex = 1; PubKeyIndex <= PubKeyNumber; PubKeyIndex++) {
|
||||||
|
if (IsValidPubKeyIndex (PubKeyIndex)) {
|
||||||
|
CopyMem (
|
||||||
|
*NewPubKeyStore + CopiedKey * EFI_CERT_TYPE_RSA2048_SIZE,
|
||||||
|
PubKeyStore + (PubKeyIndex - 1) * EFI_CERT_TYPE_RSA2048_SIZE,
|
||||||
|
EFI_CERT_TYPE_RSA2048_SIZE
|
||||||
|
);
|
||||||
|
(*NewPubKeyIndex)[PubKeyIndex] = ++CopiedKey;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
||||||
Variable store garbage collection and reclaim operation.
|
Variable store garbage collection and reclaim operation.
|
||||||
|
|
||||||
@param VariableBase Base address of variable store.
|
If ReclaimPubKeyStore is FALSE, reclaim variable space by deleting the obsoleted varaibles.
|
||||||
@param LastVariableOffset Offset of last variable.
|
If ReclaimPubKeyStore is TRUE, reclaim invalid key in public key database and update the PubKeyIndex
|
||||||
@param IsVolatile The variable store is volatile or not;
|
for all the count-based authenticate variable in NV storage.
|
||||||
if it is non-volatile, need FTW.
|
|
||||||
@param UpdatingVariable Pointer to updating variable.
|
|
||||||
@param ReclaimAnyway If TRUE, do reclaim anyway.
|
|
||||||
|
|
||||||
@return EFI_OUT_OF_RESOURCES
|
@param[in] VariableBase Base address of variable store.
|
||||||
@return EFI_SUCCESS
|
@param[out] LastVariableOffset Offset of last variable.
|
||||||
@return Others
|
@param[in] IsVolatile The variable store is volatile or not;
|
||||||
|
if it is non-volatile, need FTW.
|
||||||
|
@param[in] UpdatingVariable Pointer to updating variable.
|
||||||
|
@param[in] ReclaimPubKeyStore Reclaim for public key database or not.
|
||||||
|
@param[in] ReclaimAnyway If TRUE, do reclaim anyway.
|
||||||
|
|
||||||
|
@return EFI_OUT_OF_RESOURCES No enough memory resources.
|
||||||
|
@return EFI_SUCCESS Reclaim operation has finished successfully.
|
||||||
|
@return Others Unexpect error happened during reclaim operation.
|
||||||
|
|
||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
@ -534,6 +670,7 @@ Reclaim (
|
|||||||
OUT UINTN *LastVariableOffset,
|
OUT UINTN *LastVariableOffset,
|
||||||
IN BOOLEAN IsVolatile,
|
IN BOOLEAN IsVolatile,
|
||||||
IN VARIABLE_HEADER *UpdatingVariable,
|
IN VARIABLE_HEADER *UpdatingVariable,
|
||||||
|
IN BOOLEAN ReclaimPubKeyStore,
|
||||||
IN BOOLEAN ReclaimAnyway
|
IN BOOLEAN ReclaimAnyway
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
@ -557,6 +694,10 @@ Reclaim (
|
|||||||
CHAR16 *UpdatingVariableNamePtr;
|
CHAR16 *UpdatingVariableNamePtr;
|
||||||
UINTN CommonVariableTotalSize;
|
UINTN CommonVariableTotalSize;
|
||||||
UINTN HwErrVariableTotalSize;
|
UINTN HwErrVariableTotalSize;
|
||||||
|
UINT32 *NewPubKeyIndex;
|
||||||
|
UINT8 *NewPubKeyStore;
|
||||||
|
UINT32 NewPubKeySize;
|
||||||
|
VARIABLE_HEADER *PubKeyHeader;
|
||||||
BOOLEAN NeedDoReclaim;
|
BOOLEAN NeedDoReclaim;
|
||||||
|
|
||||||
NeedDoReclaim = FALSE;
|
NeedDoReclaim = FALSE;
|
||||||
@ -564,7 +705,11 @@ Reclaim (
|
|||||||
|
|
||||||
CommonVariableTotalSize = 0;
|
CommonVariableTotalSize = 0;
|
||||||
HwErrVariableTotalSize = 0;
|
HwErrVariableTotalSize = 0;
|
||||||
|
NewPubKeyIndex = NULL;
|
||||||
|
NewPubKeyStore = NULL;
|
||||||
|
NewPubKeySize = 0;
|
||||||
|
PubKeyHeader = NULL;
|
||||||
|
|
||||||
//
|
//
|
||||||
// Start Pointers for the variable.
|
// Start Pointers for the variable.
|
||||||
//
|
//
|
||||||
@ -608,95 +753,42 @@ Reclaim (
|
|||||||
CopyMem (ValidBuffer, VariableStoreHeader, sizeof (VARIABLE_STORE_HEADER));
|
CopyMem (ValidBuffer, VariableStoreHeader, sizeof (VARIABLE_STORE_HEADER));
|
||||||
CurrPtr = (UINT8 *) GetStartPointer ((VARIABLE_STORE_HEADER *) ValidBuffer);
|
CurrPtr = (UINT8 *) GetStartPointer ((VARIABLE_STORE_HEADER *) ValidBuffer);
|
||||||
|
|
||||||
//
|
if (ReclaimPubKeyStore) {
|
||||||
// Reinstall all ADDED variables as long as they are not identical to Updating Variable.
|
//
|
||||||
//
|
// Trim the PubKeyStore and get new PubKeyIndex.
|
||||||
Variable = GetStartPointer (VariableStoreHeader);
|
//
|
||||||
while (IsValidVariableHeader (Variable)) {
|
Status = PubKeyStoreFilter (
|
||||||
NextVariable = GetNextVariablePtr (Variable);
|
mPubKeyStore,
|
||||||
if (Variable->State == VAR_ADDED) {
|
mPubKeyNumber,
|
||||||
if (UpdatingVariable != NULL) {
|
&NewPubKeyIndex,
|
||||||
if (UpdatingVariable == Variable) {
|
&NewPubKeyStore,
|
||||||
|
&NewPubKeySize
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
FreePool (ValidBuffer);
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Refresh the PubKeyIndex for all valid variables (ADDED and IN_DELETED_TRANSITION).
|
||||||
|
//
|
||||||
|
Variable = GetStartPointer (mNvVariableCache);
|
||||||
|
while (IsValidVariableHeader (Variable)) {
|
||||||
|
NextVariable = GetNextVariablePtr (Variable);
|
||||||
|
if (Variable->State == VAR_ADDED || Variable->State == (VAR_IN_DELETED_TRANSITION & VAR_ADDED)) {
|
||||||
|
if ((StrCmp (GetVariableNamePtr (Variable), AUTHVAR_KEYDB_NAME) == 0) &&
|
||||||
|
(CompareGuid (&Variable->VendorGuid, &gEfiAuthenticatedVariableGuid))) {
|
||||||
|
//
|
||||||
|
// Skip the public key database, it will be reinstalled later.
|
||||||
|
//
|
||||||
|
PubKeyHeader = Variable;
|
||||||
Variable = NextVariable;
|
Variable = NextVariable;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
VariableNameSize = NameSizeOfVariable(Variable);
|
|
||||||
UpdatingVariableNameSize = NameSizeOfVariable(UpdatingVariable);
|
|
||||||
|
|
||||||
VariableNamePtr = GetVariableNamePtr (Variable);
|
|
||||||
UpdatingVariableNamePtr = GetVariableNamePtr (UpdatingVariable);
|
|
||||||
if (CompareGuid (&Variable->VendorGuid, &UpdatingVariable->VendorGuid) &&
|
|
||||||
VariableNameSize == UpdatingVariableNameSize &&
|
|
||||||
CompareMem (VariableNamePtr, UpdatingVariableNamePtr, VariableNameSize) == 0 ) {
|
|
||||||
Variable = NextVariable;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
VariableSize = (UINTN) NextVariable - (UINTN) Variable;
|
|
||||||
CopyMem (CurrPtr, (UINT8 *) Variable, VariableSize);
|
|
||||||
CurrPtr += VariableSize;
|
|
||||||
if ((!IsVolatile) && ((Variable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) == EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
|
||||||
HwErrVariableTotalSize += VariableSize;
|
|
||||||
} else if ((!IsVolatile) && ((Variable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) != EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
|
||||||
CommonVariableTotalSize += VariableSize;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Variable = NextVariable;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// Reinstall the variable being updated if it is not NULL.
|
|
||||||
//
|
|
||||||
if (UpdatingVariable != NULL) {
|
|
||||||
VariableSize = (UINTN)(GetNextVariablePtr (UpdatingVariable)) - (UINTN)UpdatingVariable;
|
|
||||||
CopyMem (CurrPtr, (UINT8 *) UpdatingVariable, VariableSize);
|
|
||||||
CurrPtr += VariableSize;
|
|
||||||
if ((!IsVolatile) && ((UpdatingVariable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) == EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
|
||||||
HwErrVariableTotalSize += VariableSize;
|
|
||||||
} else if ((!IsVolatile) && ((UpdatingVariable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) != EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
|
||||||
CommonVariableTotalSize += VariableSize;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// Reinstall all in delete transition variables.
|
|
||||||
//
|
|
||||||
Variable = GetStartPointer (VariableStoreHeader);
|
|
||||||
while (IsValidVariableHeader (Variable)) {
|
|
||||||
NextVariable = GetNextVariablePtr (Variable);
|
|
||||||
if (Variable != UpdatingVariable && Variable->State == (VAR_IN_DELETED_TRANSITION & VAR_ADDED)) {
|
|
||||||
|
|
||||||
//
|
|
||||||
// Buffer has cached all ADDED variable.
|
|
||||||
// Per IN_DELETED variable, we have to guarantee that
|
|
||||||
// no ADDED one in previous buffer.
|
|
||||||
//
|
|
||||||
|
|
||||||
FoundAdded = FALSE;
|
|
||||||
AddedVariable = GetStartPointer ((VARIABLE_STORE_HEADER *) ValidBuffer);
|
|
||||||
while (IsValidVariableHeader (AddedVariable)) {
|
|
||||||
NextAddedVariable = GetNextVariablePtr (AddedVariable);
|
|
||||||
NameSize = NameSizeOfVariable (AddedVariable);
|
|
||||||
if (CompareGuid (&AddedVariable->VendorGuid, &Variable->VendorGuid) &&
|
|
||||||
NameSize == NameSizeOfVariable (Variable)
|
|
||||||
) {
|
|
||||||
Point0 = (VOID *) GetVariableNamePtr (AddedVariable);
|
|
||||||
Point1 = (VOID *) GetVariableNamePtr (Variable);
|
|
||||||
if (CompareMem (Point0, Point1, NameSizeOfVariable (AddedVariable)) == 0) {
|
|
||||||
FoundAdded = TRUE;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
AddedVariable = NextAddedVariable;
|
|
||||||
}
|
|
||||||
if (!FoundAdded) {
|
|
||||||
//
|
|
||||||
// Promote VAR_IN_DELETED_TRANSITION to VAR_ADDED.
|
|
||||||
//
|
|
||||||
VariableSize = (UINTN) NextVariable - (UINTN) Variable;
|
VariableSize = (UINTN) NextVariable - (UINTN) Variable;
|
||||||
CopyMem (CurrPtr, (UINT8 *) Variable, VariableSize);
|
CopyMem (CurrPtr, (UINT8 *) Variable, VariableSize);
|
||||||
((VARIABLE_HEADER *) CurrPtr)->State = VAR_ADDED;
|
((VARIABLE_HEADER*) CurrPtr)->PubKeyIndex = NewPubKeyIndex[Variable->PubKeyIndex];
|
||||||
CurrPtr += VariableSize;
|
CurrPtr += VariableSize;
|
||||||
if ((!IsVolatile) && ((Variable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) == EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
if ((!IsVolatile) && ((Variable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) == EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
||||||
HwErrVariableTotalSize += VariableSize;
|
HwErrVariableTotalSize += VariableSize;
|
||||||
@ -704,9 +796,120 @@ Reclaim (
|
|||||||
CommonVariableTotalSize += VariableSize;
|
CommonVariableTotalSize += VariableSize;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Variable = NextVariable;
|
||||||
}
|
}
|
||||||
|
|
||||||
Variable = NextVariable;
|
//
|
||||||
|
// Reinstall the new public key database.
|
||||||
|
//
|
||||||
|
CopyMem (CurrPtr, (UINT8*) PubKeyHeader, sizeof (VARIABLE_HEADER));
|
||||||
|
Variable = (VARIABLE_HEADER*) CurrPtr;
|
||||||
|
Variable->DataSize = NewPubKeySize;
|
||||||
|
StrCpy (GetVariableNamePtr (Variable), GetVariableNamePtr (PubKeyHeader));
|
||||||
|
CopyMem (GetVariableDataPtr (Variable), NewPubKeyStore, NewPubKeySize);
|
||||||
|
CurrPtr = (UINT8*) GetNextVariablePtr (Variable);
|
||||||
|
CommonVariableTotalSize += (UINTN) CurrPtr - (UINTN) Variable;
|
||||||
|
} else {
|
||||||
|
//
|
||||||
|
// Reinstall all ADDED variables as long as they are not identical to Updating Variable.
|
||||||
|
//
|
||||||
|
Variable = GetStartPointer (VariableStoreHeader);
|
||||||
|
while (IsValidVariableHeader (Variable)) {
|
||||||
|
NextVariable = GetNextVariablePtr (Variable);
|
||||||
|
if (Variable->State == VAR_ADDED) {
|
||||||
|
if (UpdatingVariable != NULL) {
|
||||||
|
if (UpdatingVariable == Variable) {
|
||||||
|
Variable = NextVariable;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
VariableNameSize = NameSizeOfVariable(Variable);
|
||||||
|
UpdatingVariableNameSize = NameSizeOfVariable(UpdatingVariable);
|
||||||
|
|
||||||
|
VariableNamePtr = GetVariableNamePtr (Variable);
|
||||||
|
UpdatingVariableNamePtr = GetVariableNamePtr (UpdatingVariable);
|
||||||
|
if (CompareGuid (&Variable->VendorGuid, &UpdatingVariable->VendorGuid) &&
|
||||||
|
VariableNameSize == UpdatingVariableNameSize &&
|
||||||
|
CompareMem (VariableNamePtr, UpdatingVariableNamePtr, VariableNameSize) == 0 ) {
|
||||||
|
Variable = NextVariable;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
VariableSize = (UINTN) NextVariable - (UINTN) Variable;
|
||||||
|
CopyMem (CurrPtr, (UINT8 *) Variable, VariableSize);
|
||||||
|
CurrPtr += VariableSize;
|
||||||
|
if ((!IsVolatile) && ((Variable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) == EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
||||||
|
HwErrVariableTotalSize += VariableSize;
|
||||||
|
} else if ((!IsVolatile) && ((Variable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) != EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
||||||
|
CommonVariableTotalSize += VariableSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Variable = NextVariable;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Reinstall the variable being updated if it is not NULL.
|
||||||
|
//
|
||||||
|
if (UpdatingVariable != NULL) {
|
||||||
|
VariableSize = (UINTN)(GetNextVariablePtr (UpdatingVariable)) - (UINTN)UpdatingVariable;
|
||||||
|
CopyMem (CurrPtr, (UINT8 *) UpdatingVariable, VariableSize);
|
||||||
|
CurrPtr += VariableSize;
|
||||||
|
if ((!IsVolatile) && ((UpdatingVariable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) == EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
||||||
|
HwErrVariableTotalSize += VariableSize;
|
||||||
|
} else if ((!IsVolatile) && ((UpdatingVariable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) != EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
||||||
|
CommonVariableTotalSize += VariableSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Reinstall all in delete transition variables.
|
||||||
|
//
|
||||||
|
Variable = GetStartPointer (VariableStoreHeader);
|
||||||
|
while (IsValidVariableHeader (Variable)) {
|
||||||
|
NextVariable = GetNextVariablePtr (Variable);
|
||||||
|
if (Variable != UpdatingVariable && Variable->State == (VAR_IN_DELETED_TRANSITION & VAR_ADDED)) {
|
||||||
|
|
||||||
|
//
|
||||||
|
// Buffer has cached all ADDED variable.
|
||||||
|
// Per IN_DELETED variable, we have to guarantee that
|
||||||
|
// no ADDED one in previous buffer.
|
||||||
|
//
|
||||||
|
|
||||||
|
FoundAdded = FALSE;
|
||||||
|
AddedVariable = GetStartPointer ((VARIABLE_STORE_HEADER *) ValidBuffer);
|
||||||
|
while (IsValidVariableHeader (AddedVariable)) {
|
||||||
|
NextAddedVariable = GetNextVariablePtr (AddedVariable);
|
||||||
|
NameSize = NameSizeOfVariable (AddedVariable);
|
||||||
|
if (CompareGuid (&AddedVariable->VendorGuid, &Variable->VendorGuid) &&
|
||||||
|
NameSize == NameSizeOfVariable (Variable)
|
||||||
|
) {
|
||||||
|
Point0 = (VOID *) GetVariableNamePtr (AddedVariable);
|
||||||
|
Point1 = (VOID *) GetVariableNamePtr (Variable);
|
||||||
|
if (CompareMem (Point0, Point1, NameSizeOfVariable (AddedVariable)) == 0) {
|
||||||
|
FoundAdded = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
AddedVariable = NextAddedVariable;
|
||||||
|
}
|
||||||
|
if (!FoundAdded) {
|
||||||
|
//
|
||||||
|
// Promote VAR_IN_DELETED_TRANSITION to VAR_ADDED.
|
||||||
|
//
|
||||||
|
VariableSize = (UINTN) NextVariable - (UINTN) Variable;
|
||||||
|
CopyMem (CurrPtr, (UINT8 *) Variable, VariableSize);
|
||||||
|
((VARIABLE_HEADER *) CurrPtr)->State = VAR_ADDED;
|
||||||
|
CurrPtr += VariableSize;
|
||||||
|
if ((!IsVolatile) && ((Variable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) == EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
||||||
|
HwErrVariableTotalSize += VariableSize;
|
||||||
|
} else if ((!IsVolatile) && ((Variable->Attributes & EFI_VARIABLE_HARDWARE_ERROR_RECORD) != EFI_VARIABLE_HARDWARE_ERROR_RECORD)) {
|
||||||
|
CommonVariableTotalSize += VariableSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Variable = NextVariable;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (IsVolatile) {
|
if (IsVolatile) {
|
||||||
@ -748,6 +951,14 @@ Reclaim (
|
|||||||
*LastVariableOffset = (UINTN) NextVariable - (UINTN) VariableBase;
|
*LastVariableOffset = (UINTN) NextVariable - (UINTN) VariableBase;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (NewPubKeyStore != NULL) {
|
||||||
|
FreePool (NewPubKeyStore);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (NewPubKeyIndex != NULL) {
|
||||||
|
FreePool (NewPubKeyIndex);
|
||||||
|
}
|
||||||
|
|
||||||
FreePool (ValidBuffer);
|
FreePool (ValidBuffer);
|
||||||
|
|
||||||
return Status;
|
return Status;
|
||||||
@ -1744,8 +1955,14 @@ UpdateVariable (
|
|||||||
//
|
//
|
||||||
// Perform garbage collection & reclaim operation.
|
// Perform garbage collection & reclaim operation.
|
||||||
//
|
//
|
||||||
Status = Reclaim (mVariableModuleGlobal->VariableGlobal.NonVolatileVariableBase,
|
Status = Reclaim (
|
||||||
&mVariableModuleGlobal->NonVolatileLastVariableOffset, FALSE, Variable->CurrPtr, FALSE);
|
mVariableModuleGlobal->VariableGlobal.NonVolatileVariableBase,
|
||||||
|
&mVariableModuleGlobal->NonVolatileLastVariableOffset,
|
||||||
|
FALSE,
|
||||||
|
Variable->CurrPtr,
|
||||||
|
FALSE,
|
||||||
|
FALSE
|
||||||
|
);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
goto Done;
|
goto Done;
|
||||||
}
|
}
|
||||||
@ -1859,8 +2076,14 @@ UpdateVariable (
|
|||||||
//
|
//
|
||||||
// Perform garbage collection & reclaim operation.
|
// Perform garbage collection & reclaim operation.
|
||||||
//
|
//
|
||||||
Status = Reclaim (mVariableModuleGlobal->VariableGlobal.VolatileVariableBase,
|
Status = Reclaim (
|
||||||
&mVariableModuleGlobal->VolatileLastVariableOffset, TRUE, Variable->CurrPtr, FALSE);
|
mVariableModuleGlobal->VariableGlobal.VolatileVariableBase,
|
||||||
|
&mVariableModuleGlobal->VolatileLastVariableOffset,
|
||||||
|
TRUE,
|
||||||
|
Variable->CurrPtr,
|
||||||
|
FALSE,
|
||||||
|
FALSE
|
||||||
|
);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
goto Done;
|
goto Done;
|
||||||
}
|
}
|
||||||
@ -2629,6 +2852,7 @@ ReclaimForOS(
|
|||||||
&mVariableModuleGlobal->NonVolatileLastVariableOffset,
|
&mVariableModuleGlobal->NonVolatileLastVariableOffset,
|
||||||
FALSE,
|
FALSE,
|
||||||
NULL,
|
NULL,
|
||||||
|
FALSE,
|
||||||
FALSE
|
FALSE
|
||||||
);
|
);
|
||||||
ASSERT_EFI_ERROR (Status);
|
ASSERT_EFI_ERROR (Status);
|
||||||
@ -2760,6 +2984,7 @@ VariableWriteServiceInitialize (
|
|||||||
&mVariableModuleGlobal->NonVolatileLastVariableOffset,
|
&mVariableModuleGlobal->NonVolatileLastVariableOffset,
|
||||||
FALSE,
|
FALSE,
|
||||||
NULL,
|
NULL,
|
||||||
|
FALSE,
|
||||||
TRUE
|
TRUE
|
||||||
);
|
);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
The internal header file includes the common header files, defines
|
The internal header file includes the common header files, defines
|
||||||
internal structure and functions used by Variable modules.
|
internal structure and functions used by Variable modules.
|
||||||
|
|
||||||
Copyright (c) 2009 - 2012, Intel Corporation. All rights reserved.<BR>
|
Copyright (c) 2009 - 2013, Intel Corporation. All rights reserved.<BR>
|
||||||
This program and the accompanying materials
|
This program and the accompanying materials
|
||||||
are licensed and made available under the terms and conditions of the BSD License
|
are licensed and made available under the terms and conditions of the BSD License
|
||||||
which accompanies this distribution. The full text of the license may be found at
|
which accompanies this distribution. The full text of the license may be found at
|
||||||
@ -366,6 +366,37 @@ VariableCommonInitialize (
|
|||||||
VOID
|
VOID
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
|
||||||
|
Variable store garbage collection and reclaim operation.
|
||||||
|
|
||||||
|
If ReclaimPubKeyStore is FALSE, reclaim variable space by deleting the obsoleted varaibles.
|
||||||
|
If ReclaimPubKeyStore is TRUE, reclaim invalid key in public key database and update the PubKeyIndex
|
||||||
|
for all the count-based authenticate variable in NV storage.
|
||||||
|
|
||||||
|
@param[in] VariableBase Base address of variable store.
|
||||||
|
@param[out] LastVariableOffset Offset of last variable.
|
||||||
|
@param[in] IsVolatile The variable store is volatile or not;
|
||||||
|
if it is non-volatile, need FTW.
|
||||||
|
@param[in] UpdatingVariable Pointer to updating variable.
|
||||||
|
@param[in] ReclaimPubKeyStore Reclaim for public key database or not.
|
||||||
|
@param[in] ReclaimAnyway If TRUE, do reclaim anyway.
|
||||||
|
|
||||||
|
@return EFI_OUT_OF_RESOURCES No enough memory resources.
|
||||||
|
@return EFI_SUCCESS Reclaim operation has finished successfully.
|
||||||
|
@return Others Unexpect error happened during reclaim operation.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
Reclaim (
|
||||||
|
IN EFI_PHYSICAL_ADDRESS VariableBase,
|
||||||
|
OUT UINTN *LastVariableOffset,
|
||||||
|
IN BOOLEAN IsVolatile,
|
||||||
|
IN VARIABLE_HEADER *UpdatingVariable,
|
||||||
|
IN BOOLEAN ReclaimPubKeyStore,
|
||||||
|
IN BOOLEAN ReclaimAnyway
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
This function reclaims variable storage if free size is below the threshold.
|
This function reclaims variable storage if free size is below the threshold.
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user