2011-09-02 09:49:32 +02:00
/** @file
Implement ReadOnly Variable Services required by PEIM and install PEI
ReadOnly Varaiable2 PPI . These services operates the non - volatile
storage space .
2013-01-25 07:17:43 +01:00
Copyright ( c ) 2009 - 2013 , Intel Corporation . All rights reserved . < BR >
2011-09-02 09:49:32 +02:00
This program and the accompanying materials
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
http : //opensource.org/licenses/bsd-license.php
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN " AS IS " BASIS ,
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND , EITHER EXPRESS OR IMPLIED .
* */
# include "Variable.h"
//
// Module globals
//
EFI_PEI_READ_ONLY_VARIABLE2_PPI mVariablePpi = {
PeiGetVariable ,
PeiGetNextVariableName
} ;
EFI_PEI_PPI_DESCRIPTOR mPpiListVariable = {
( EFI_PEI_PPI_DESCRIPTOR_PPI | EFI_PEI_PPI_DESCRIPTOR_TERMINATE_LIST ) ,
& gEfiPeiReadOnlyVariable2PpiGuid ,
& mVariablePpi
} ;
/**
Provide the functionality of the variable services .
@ param FileHandle Handle of the file being invoked .
Type EFI_PEI_FILE_HANDLE is defined in FfsFindNextFile ( ) .
@ param PeiServices General purpose services available to every PEIM .
@ retval EFI_SUCCESS If the interface could be successfully installed
@ retval Others Returned from PeiServicesInstallPpi ( )
* */
EFI_STATUS
EFIAPI
PeimInitializeVariableServices (
IN EFI_PEI_FILE_HANDLE FileHandle ,
IN CONST EFI_PEI_SERVICES * * PeiServices
)
{
return PeiServicesInstallPpi ( & mPpiListVariable ) ;
}
/**
Gets the pointer to the first variable header in given variable store area .
@ param VarStoreHeader Pointer to the Variable Store Header .
@ return Pointer to the first variable header
* */
VARIABLE_HEADER *
GetStartPointer (
IN VARIABLE_STORE_HEADER * VarStoreHeader
)
{
//
// The end of variable store
//
return ( VARIABLE_HEADER * ) HEADER_ALIGN ( VarStoreHeader + 1 ) ;
}
/**
This code gets the pointer to the last variable memory pointer byte .
@ param VarStoreHeader Pointer to the Variable Store Header .
@ return VARIABLE_HEADER * pointer to last unavailable Variable Header .
* */
VARIABLE_HEADER *
GetEndPointer (
IN VARIABLE_STORE_HEADER * VarStoreHeader
)
{
//
// The end of variable store
//
return ( VARIABLE_HEADER * ) HEADER_ALIGN ( ( UINTN ) VarStoreHeader + VarStoreHeader - > Size ) ;
}
/**
This code checks if variable header is valid or not .
@ param Variable Pointer to the Variable Header .
@ retval TRUE Variable header is valid .
@ retval FALSE Variable header is not valid .
* */
BOOLEAN
IsValidVariableHeader (
IN VARIABLE_HEADER * Variable
)
{
if ( Variable = = NULL | | Variable - > StartId ! = VARIABLE_DATA ) {
return FALSE ;
}
return TRUE ;
}
/**
This code gets the size of name of variable .
@ param Variable Pointer to the Variable Header .
@ return Size of variable in bytes in type UINTN .
* */
UINTN
NameSizeOfVariable (
IN VARIABLE_HEADER * Variable
)
{
if ( Variable - > State = = ( UINT8 ) ( - 1 ) | |
Variable - > DataSize = = ( UINT32 ) ( - 1 ) | |
Variable - > NameSize = = ( UINT32 ) ( - 1 ) | |
Variable - > Attributes = = ( UINT32 ) ( - 1 ) ) {
return 0 ;
}
return ( UINTN ) Variable - > NameSize ;
}
/**
This code gets the size of data of variable .
@ param Variable Pointer to the Variable Header .
@ return Size of variable in bytes in type UINTN .
* */
UINTN
DataSizeOfVariable (
IN VARIABLE_HEADER * Variable
)
{
if ( Variable - > State = = ( UINT8 ) ( - 1 ) | |
Variable - > DataSize = = ( UINT32 ) ( - 1 ) | |
Variable - > NameSize = = ( UINT32 ) ( - 1 ) | |
Variable - > Attributes = = ( UINT32 ) ( - 1 ) ) {
return 0 ;
}
return ( UINTN ) Variable - > DataSize ;
}
/**
This code gets the pointer to the variable name .
@ param Variable Pointer to the Variable Header .
@ return A CHAR16 * pointer to Variable Name .
* */
CHAR16 *
GetVariableNamePtr (
IN VARIABLE_HEADER * Variable
)
{
return ( CHAR16 * ) ( Variable + 1 ) ;
}
/**
This code gets the pointer to the variable data .
2013-07-03 11:09:42 +02:00
@ param Variable Pointer to the Variable Header .
@ param VariableHeader Pointer to the Variable Header that has consecutive content .
2011-09-02 09:49:32 +02:00
@ return A UINT8 * pointer to Variable Data .
* */
UINT8 *
GetVariableDataPtr (
2013-07-03 11:09:42 +02:00
IN VARIABLE_HEADER * Variable ,
IN VARIABLE_HEADER * VariableHeader
2011-09-02 09:49:32 +02:00
)
{
UINTN Value ;
//
// Be careful about pad size for alignment
//
Value = ( UINTN ) GetVariableNamePtr ( Variable ) ;
2013-07-03 11:09:42 +02:00
Value + = NameSizeOfVariable ( VariableHeader ) ;
Value + = GET_PAD_SIZE ( NameSizeOfVariable ( VariableHeader ) ) ;
2011-09-02 09:49:32 +02:00
return ( UINT8 * ) Value ;
}
/**
This code gets the pointer to the next variable header .
2013-07-03 11:09:42 +02:00
@ param StoreInfo Pointer to variable store info structure .
@ param Variable Pointer to the Variable Header .
@ param VariableHeader Pointer to the Variable Header that has consecutive content .
2011-09-02 09:49:32 +02:00
@ return A VARIABLE_HEADER * pointer to next variable header .
* */
VARIABLE_HEADER *
GetNextVariablePtr (
2013-07-03 11:09:42 +02:00
IN VARIABLE_STORE_INFO * StoreInfo ,
IN VARIABLE_HEADER * Variable ,
IN VARIABLE_HEADER * VariableHeader
2011-09-02 09:49:32 +02:00
)
{
2013-07-03 11:09:42 +02:00
EFI_PHYSICAL_ADDRESS TargetAddress ;
EFI_PHYSICAL_ADDRESS SpareAddress ;
UINTN Value ;
2011-09-02 09:49:32 +02:00
2013-07-03 11:09:42 +02:00
Value = ( UINTN ) GetVariableDataPtr ( Variable , VariableHeader ) ;
Value + = DataSizeOfVariable ( VariableHeader ) ;
Value + = GET_PAD_SIZE ( DataSizeOfVariable ( VariableHeader ) ) ;
2011-09-02 09:49:32 +02:00
//
// Be careful about pad size for alignment
//
2013-07-03 11:09:42 +02:00
Value = HEADER_ALIGN ( Value ) ;
if ( StoreInfo - > FtwLastWriteData ! = NULL ) {
TargetAddress = StoreInfo - > FtwLastWriteData - > TargetAddress ;
SpareAddress = StoreInfo - > FtwLastWriteData - > SpareAddress ;
if ( ( ( UINTN ) Variable < ( UINTN ) TargetAddress ) & & ( Value > = ( UINTN ) TargetAddress ) ) {
//
// Next variable is in spare block.
//
Value = ( UINTN ) SpareAddress + ( Value - ( UINTN ) TargetAddress ) ;
}
}
return ( VARIABLE_HEADER * ) Value ;
2011-09-02 09:49:32 +02:00
}
/**
2013-07-03 11:09:42 +02:00
Get variable store status .
2011-09-02 09:49:32 +02:00
@ param VarStoreHeader Pointer to the Variable Store Header .
@ retval EfiRaw Variable store is raw
@ retval EfiValid Variable store is valid
@ retval EfiInvalid Variable store is invalid
* */
VARIABLE_STORE_STATUS
GetVariableStoreStatus (
IN VARIABLE_STORE_HEADER * VarStoreHeader
)
{
if ( CompareGuid ( & VarStoreHeader - > Signature , & gEfiAuthenticatedVariableGuid ) & &
VarStoreHeader - > Format = = VARIABLE_STORE_FORMATTED & &
VarStoreHeader - > State = = VARIABLE_STORE_HEALTHY
) {
return EfiValid ;
}
if ( ( ( UINT32 * ) ( & VarStoreHeader - > Signature ) ) [ 0 ] = = 0xffffffff & &
( ( UINT32 * ) ( & VarStoreHeader - > Signature ) ) [ 1 ] = = 0xffffffff & &
( ( UINT32 * ) ( & VarStoreHeader - > Signature ) ) [ 2 ] = = 0xffffffff & &
( ( UINT32 * ) ( & VarStoreHeader - > Signature ) ) [ 3 ] = = 0xffffffff & &
VarStoreHeader - > Size = = 0xffffffff & &
VarStoreHeader - > Format = = 0xff & &
VarStoreHeader - > State = = 0xff
) {
return EfiRaw ;
} else {
return EfiInvalid ;
}
}
2013-07-03 11:09:42 +02:00
/**
Compare two variable names , one of them may be inconsecutive .
@ param StoreInfo Pointer to variable store info structure .
@ param Name1 Pointer to one variable name .
@ param Name2 Pointer to another variable name .
@ param NameSize Variable name size .
@ retval TRUE Name1 and Name2 are identical .
@ retval FALSE Name1 and Name2 are not identical .
* */
BOOLEAN
CompareVariableName (
IN VARIABLE_STORE_INFO * StoreInfo ,
IN CONST CHAR16 * Name1 ,
IN CONST CHAR16 * Name2 ,
IN UINTN NameSize
)
{
EFI_PHYSICAL_ADDRESS TargetAddress ;
EFI_PHYSICAL_ADDRESS SpareAddress ;
UINTN PartialNameSize ;
if ( StoreInfo - > FtwLastWriteData ! = NULL ) {
TargetAddress = StoreInfo - > FtwLastWriteData - > TargetAddress ;
SpareAddress = StoreInfo - > FtwLastWriteData - > SpareAddress ;
if ( ( ( UINTN ) Name1 < ( UINTN ) TargetAddress ) & & ( ( ( UINTN ) Name1 + NameSize ) > ( UINTN ) TargetAddress ) ) {
//
// Name1 is inconsecutive.
//
PartialNameSize = ( UINTN ) TargetAddress - ( UINTN ) Name1 ;
//
// Partial content is in NV storage.
//
if ( CompareMem ( ( UINT8 * ) Name1 , ( UINT8 * ) Name2 , PartialNameSize ) = = 0 ) {
//
// Another partial content is in spare block.
//
if ( CompareMem ( ( UINT8 * ) ( UINTN ) SpareAddress , ( UINT8 * ) Name2 + PartialNameSize , NameSize - PartialNameSize ) = = 0 ) {
return TRUE ;
}
}
return FALSE ;
} else if ( ( ( UINTN ) Name2 < ( UINTN ) TargetAddress ) & & ( ( ( UINTN ) Name2 + NameSize ) > ( UINTN ) TargetAddress ) ) {
//
// Name2 is inconsecutive.
//
PartialNameSize = ( UINTN ) TargetAddress - ( UINTN ) Name2 ;
//
// Partial content is in NV storage.
//
if ( CompareMem ( ( UINT8 * ) Name2 , ( UINT8 * ) Name1 , PartialNameSize ) = = 0 ) {
//
// Another partial content is in spare block.
//
if ( CompareMem ( ( UINT8 * ) ( UINTN ) SpareAddress , ( UINT8 * ) Name1 + PartialNameSize , NameSize - PartialNameSize ) = = 0 ) {
return TRUE ;
}
}
return FALSE ;
}
}
//
// Both Name1 and Name2 are consecutive.
//
if ( CompareMem ( ( UINT8 * ) Name1 , ( UINT8 * ) Name2 , NameSize ) = = 0 ) {
return TRUE ;
}
return FALSE ;
}
2011-09-02 09:49:32 +02:00
/**
This function compares a variable with variable entries in database .
2013-07-03 11:09:42 +02:00
@ param StoreInfo Pointer to variable store info structure .
2011-09-02 09:49:32 +02:00
@ param Variable Pointer to the variable in our database
2013-07-03 11:09:42 +02:00
@ param VariableHeader Pointer to the Variable Header that has consecutive content .
2011-09-02 09:49:32 +02:00
@ param VariableName Name of the variable to compare to ' Variable '
@ param VendorGuid GUID of the variable to compare to ' Variable '
@ param PtrTrack Variable Track Pointer structure that contains Variable Information .
@ retval EFI_SUCCESS Found match variable
@ retval EFI_NOT_FOUND Variable not found
* */
EFI_STATUS
CompareWithValidVariable (
2013-07-03 11:09:42 +02:00
IN VARIABLE_STORE_INFO * StoreInfo ,
2011-09-02 09:49:32 +02:00
IN VARIABLE_HEADER * Variable ,
2013-07-03 11:09:42 +02:00
IN VARIABLE_HEADER * VariableHeader ,
2011-09-02 09:49:32 +02:00
IN CONST CHAR16 * VariableName ,
IN CONST EFI_GUID * VendorGuid ,
OUT VARIABLE_POINTER_TRACK * PtrTrack
)
{
VOID * Point ;
if ( VariableName [ 0 ] = = 0 ) {
PtrTrack - > CurrPtr = Variable ;
return EFI_SUCCESS ;
} else {
//
// Don't use CompareGuid function here for performance reasons.
// Instead we compare the GUID a UINT32 at a time and branch
// on the first failed comparison.
//
2013-07-03 11:09:42 +02:00
if ( ( ( ( INT32 * ) VendorGuid ) [ 0 ] = = ( ( INT32 * ) & VariableHeader - > VendorGuid ) [ 0 ] ) & &
( ( ( INT32 * ) VendorGuid ) [ 1 ] = = ( ( INT32 * ) & VariableHeader - > VendorGuid ) [ 1 ] ) & &
( ( ( INT32 * ) VendorGuid ) [ 2 ] = = ( ( INT32 * ) & VariableHeader - > VendorGuid ) [ 2 ] ) & &
( ( ( INT32 * ) VendorGuid ) [ 3 ] = = ( ( INT32 * ) & VariableHeader - > VendorGuid ) [ 3 ] )
2011-09-02 09:49:32 +02:00
) {
2013-07-03 11:09:42 +02:00
ASSERT ( NameSizeOfVariable ( VariableHeader ) ! = 0 ) ;
2011-09-02 09:49:32 +02:00
Point = ( VOID * ) GetVariableNamePtr ( Variable ) ;
2013-07-03 11:09:42 +02:00
if ( CompareVariableName ( StoreInfo , VariableName , Point , NameSizeOfVariable ( VariableHeader ) ) ) {
2011-09-02 09:49:32 +02:00
PtrTrack - > CurrPtr = Variable ;
return EFI_SUCCESS ;
}
}
}
return EFI_NOT_FOUND ;
}
2011-10-19 14:40:52 +02:00
/**
2013-07-03 11:09:42 +02:00
Return the variable store header and the store info based on the Index .
2011-10-19 14:40:52 +02:00
2013-07-03 11:09:42 +02:00
@ param Type The type of the variable store .
@ param StoreInfo Return the store info .
2011-10-19 14:40:52 +02:00
@ return Pointer to the variable store header .
* */
VARIABLE_STORE_HEADER *
GetVariableStore (
IN VARIABLE_STORE_TYPE Type ,
2013-07-03 11:09:42 +02:00
OUT VARIABLE_STORE_INFO * StoreInfo
2011-10-19 14:40:52 +02:00
)
{
2013-07-03 11:09:42 +02:00
EFI_HOB_GUID_TYPE * GuidHob ;
EFI_FIRMWARE_VOLUME_HEADER * FvHeader ;
VARIABLE_STORE_HEADER * VariableStoreHeader ;
EFI_PHYSICAL_ADDRESS NvStorageBase ;
UINT32 NvStorageSize ;
FAULT_TOLERANT_WRITE_LAST_WRITE_DATA * FtwLastWriteData ;
UINT32 BackUpOffset ;
StoreInfo - > IndexTable = NULL ;
StoreInfo - > FtwLastWriteData = NULL ;
2011-10-19 14:40:52 +02:00
VariableStoreHeader = NULL ;
switch ( Type ) {
case VariableStoreTypeHob :
GuidHob = GetFirstGuidHob ( & gEfiAuthenticatedVariableGuid ) ;
if ( GuidHob ! = NULL ) {
VariableStoreHeader = ( VARIABLE_STORE_HEADER * ) GET_GUID_HOB_DATA ( GuidHob ) ;
}
break ;
case VariableStoreTypeNv :
if ( GetBootModeHob ( ) ! = BOOT_IN_RECOVERY_MODE ) {
//
// The content of NV storage for variable is not reliable in recovery boot mode.
//
2013-07-03 11:09:42 +02:00
NvStorageSize = PcdGet32 ( PcdFlashNvStorageVariableSize ) ;
NvStorageBase = ( EFI_PHYSICAL_ADDRESS ) ( PcdGet64 ( PcdFlashNvStorageVariableBase64 ) ! = 0 ?
PcdGet64 ( PcdFlashNvStorageVariableBase64 ) :
PcdGet32 ( PcdFlashNvStorageVariableBase )
) ;
//
// First let FvHeader point to NV storage base.
//
FvHeader = ( EFI_FIRMWARE_VOLUME_HEADER * ) ( UINTN ) NvStorageBase ;
//
// Check the FTW last write data hob.
//
BackUpOffset = 0 ;
GuidHob = GetFirstGuidHob ( & gEdkiiFaultTolerantWriteGuid ) ;
if ( GuidHob ! = NULL ) {
FtwLastWriteData = ( FAULT_TOLERANT_WRITE_LAST_WRITE_DATA * ) GET_GUID_HOB_DATA ( GuidHob ) ;
if ( FtwLastWriteData - > TargetAddress = = NvStorageBase ) {
//
// Let FvHeader point to spare block.
//
FvHeader = ( EFI_FIRMWARE_VOLUME_HEADER * ) ( UINTN ) FtwLastWriteData - > SpareAddress ;
DEBUG ( ( EFI_D_INFO , " PeiVariable: NV storage is backed up in spare block: 0x%x \n " , ( UINTN ) FtwLastWriteData - > SpareAddress ) ) ;
} else if ( ( FtwLastWriteData - > TargetAddress > NvStorageBase ) & & ( FtwLastWriteData - > TargetAddress < ( NvStorageBase + NvStorageSize ) ) ) {
StoreInfo - > FtwLastWriteData = FtwLastWriteData ;
//
// Flash NV storage from the offset is backed up in spare block.
//
BackUpOffset = ( UINT32 ) ( FtwLastWriteData - > TargetAddress - NvStorageBase ) ;
DEBUG ( ( EFI_D_INFO , " PeiVariable: High partial NV storage from offset: %x is backed up in spare block: 0x%x \n " , BackUpOffset , ( UINTN ) FtwLastWriteData - > SpareAddress ) ) ;
//
// At least one block data in flash NV storage is still valid, so still leave FvHeader point to NV storage base.
//
}
}
2011-12-13 10:42:36 +01:00
//
// Check if the Firmware Volume is not corrupted
//
if ( ( FvHeader - > Signature ! = EFI_FVH_SIGNATURE ) | | ( ! CompareGuid ( & gEfiSystemNvDataFvGuid , & FvHeader - > FileSystemGuid ) ) ) {
DEBUG ( ( EFI_D_ERROR , " Firmware Volume for Variable Store is corrupted \n " ) ) ;
break ;
}
2011-10-19 14:40:52 +02:00
VariableStoreHeader = ( VARIABLE_STORE_HEADER * ) ( ( UINT8 * ) FvHeader + FvHeader - > HeaderLength ) ;
2013-07-03 11:09:42 +02:00
GuidHob = GetFirstGuidHob ( & gEfiVariableIndexTableGuid ) ;
if ( GuidHob ! = NULL ) {
StoreInfo - > IndexTable = GET_GUID_HOB_DATA ( GuidHob ) ;
} else {
//
// If it's the first time to access variable region in flash, create a guid hob to record
// VAR_ADDED type variable info.
// Note that as the resource of PEI phase is limited, only store the limited number of
// VAR_ADDED type variables to reduce access time.
//
StoreInfo - > IndexTable = ( VARIABLE_INDEX_TABLE * ) BuildGuidHob ( & gEfiVariableIndexTableGuid , sizeof ( VARIABLE_INDEX_TABLE ) ) ;
StoreInfo - > IndexTable - > Length = 0 ;
StoreInfo - > IndexTable - > StartPtr = GetStartPointer ( VariableStoreHeader ) ;
StoreInfo - > IndexTable - > EndPtr = GetEndPointer ( VariableStoreHeader ) ;
StoreInfo - > IndexTable - > GoneThrough = 0 ;
2011-10-19 14:40:52 +02:00
}
}
break ;
default :
ASSERT ( FALSE ) ;
break ;
}
2013-07-03 11:09:42 +02:00
StoreInfo - > VariableStoreHeader = VariableStoreHeader ;
2011-10-19 14:40:52 +02:00
return VariableStoreHeader ;
}
2011-09-02 09:49:32 +02:00
2013-07-03 11:09:42 +02:00
/**
Get variable header that has consecutive content .
@ param StoreInfo Pointer to variable store info structure .
@ param Variable Pointer to the Variable Header .
@ param VariableHeader Pointer to Pointer to the Variable Header that has consecutive content .
@ retval TRUE Variable header is valid .
@ retval FALSE Variable header is not valid .
* */
BOOLEAN
GetVariableHeader (
IN VARIABLE_STORE_INFO * StoreInfo ,
IN VARIABLE_HEADER * Variable ,
OUT VARIABLE_HEADER * * VariableHeader
)
{
EFI_PHYSICAL_ADDRESS TargetAddress ;
EFI_PHYSICAL_ADDRESS SpareAddress ;
EFI_HOB_GUID_TYPE * GuidHob ;
UINTN PartialHeaderSize ;
//
// First assume variable header pointed by Variable is consecutive.
//
* VariableHeader = Variable ;
if ( ( Variable ! = NULL ) & & ( StoreInfo - > FtwLastWriteData ! = NULL ) ) {
TargetAddress = StoreInfo - > FtwLastWriteData - > TargetAddress ;
SpareAddress = StoreInfo - > FtwLastWriteData - > SpareAddress ;
if ( ( ( UINTN ) Variable < ( UINTN ) TargetAddress ) & & ( ( ( UINTN ) Variable + sizeof ( VARIABLE_HEADER ) ) > ( UINTN ) TargetAddress ) ) {
//
// Variable header pointed by Variable is inconsecutive,
// create a guid hob to combine the two partial variable header content together.
//
GuidHob = GetFirstGuidHob ( & gEfiCallerIdGuid ) ;
if ( GuidHob ! = NULL ) {
* VariableHeader = ( VARIABLE_HEADER * ) GET_GUID_HOB_DATA ( GuidHob ) ;
} else {
* VariableHeader = ( VARIABLE_HEADER * ) BuildGuidHob ( & gEfiCallerIdGuid , sizeof ( VARIABLE_HEADER ) ) ;
PartialHeaderSize = ( UINTN ) TargetAddress - ( UINTN ) Variable ;
//
// Partial content is in NV storage.
//
CopyMem ( ( UINT8 * ) * VariableHeader , ( UINT8 * ) Variable , PartialHeaderSize ) ;
//
// Another partial content is in spare block.
//
CopyMem ( ( UINT8 * ) * VariableHeader + PartialHeaderSize , ( UINT8 * ) ( UINTN ) SpareAddress , sizeof ( VARIABLE_HEADER ) - PartialHeaderSize ) ;
}
}
}
return IsValidVariableHeader ( * VariableHeader ) ;
}
/**
Get variable name or data to output buffer .
@ param StoreInfo Pointer to variable store info structure .
@ param NameOrData Pointer to the variable name / data that may be inconsecutive .
@ param Size Variable name / data size .
@ param Buffer Pointer to output buffer to hold the variable name / data .
* */
VOID
GetVariableNameOrData (
IN VARIABLE_STORE_INFO * StoreInfo ,
IN UINT8 * NameOrData ,
IN UINTN Size ,
OUT UINT8 * Buffer
)
{
EFI_PHYSICAL_ADDRESS TargetAddress ;
EFI_PHYSICAL_ADDRESS SpareAddress ;
UINTN PartialSize ;
if ( StoreInfo - > FtwLastWriteData ! = NULL ) {
TargetAddress = StoreInfo - > FtwLastWriteData - > TargetAddress ;
SpareAddress = StoreInfo - > FtwLastWriteData - > SpareAddress ;
if ( ( ( UINTN ) NameOrData < ( UINTN ) TargetAddress ) & & ( ( ( UINTN ) NameOrData + Size ) > ( UINTN ) TargetAddress ) ) {
//
// Variable name/data is inconsecutive.
//
PartialSize = ( UINTN ) TargetAddress - ( UINTN ) NameOrData ;
//
// Partial content is in NV storage.
//
CopyMem ( Buffer , NameOrData , PartialSize ) ;
//
// Another partial content is in spare block.
//
CopyMem ( Buffer + PartialSize , ( UINT8 * ) ( UINTN ) SpareAddress , Size - PartialSize ) ;
return ;
}
}
//
// Variable name/data is consecutive.
//
CopyMem ( Buffer , NameOrData , Size ) ;
}
2011-09-02 09:49:32 +02:00
/**
2011-10-19 14:40:52 +02:00
Find the variable in the specified variable store .
2011-09-02 09:49:32 +02:00
2013-07-03 11:09:42 +02:00
@ param StoreInfo Pointer to the store info structure .
2011-10-19 14:40:52 +02:00
@ param VariableName Name of the variable to be found
@ param VendorGuid Vendor GUID to be found .
@ param PtrTrack Variable Track Pointer structure that contains Variable Information .
2011-09-02 09:49:32 +02:00
@ retval EFI_SUCCESS Variable found successfully
@ retval EFI_NOT_FOUND Variable not found
@ retval EFI_INVALID_PARAMETER Invalid variable name
* */
EFI_STATUS
2011-10-19 14:40:52 +02:00
FindVariableEx (
2013-07-03 11:09:42 +02:00
IN VARIABLE_STORE_INFO * StoreInfo ,
2011-10-19 14:40:52 +02:00
IN CONST CHAR16 * VariableName ,
IN CONST EFI_GUID * VendorGuid ,
OUT VARIABLE_POINTER_TRACK * PtrTrack
2011-09-02 09:49:32 +02:00
)
{
VARIABLE_HEADER * Variable ;
VARIABLE_HEADER * LastVariable ;
VARIABLE_HEADER * MaxIndex ;
2011-10-19 14:40:52 +02:00
UINTN Index ;
UINTN Offset ;
2011-09-02 09:49:32 +02:00
BOOLEAN StopRecord ;
2013-01-25 07:17:43 +01:00
VARIABLE_HEADER * InDeletedVariable ;
2013-07-03 11:09:42 +02:00
VARIABLE_STORE_HEADER * VariableStoreHeader ;
VARIABLE_INDEX_TABLE * IndexTable ;
VARIABLE_HEADER * VariableHeader ;
VariableStoreHeader = StoreInfo - > VariableStoreHeader ;
2011-09-02 09:49:32 +02:00
2011-10-19 14:40:52 +02:00
if ( VariableStoreHeader = = NULL ) {
2011-09-02 09:49:32 +02:00
return EFI_INVALID_PARAMETER ;
}
2011-10-19 14:40:52 +02:00
if ( GetVariableStoreStatus ( VariableStoreHeader ) ! = EfiValid ) {
return EFI_UNSUPPORTED ;
}
if ( ~ VariableStoreHeader - > Size = = 0 ) {
return EFI_NOT_FOUND ;
}
2013-07-03 11:09:42 +02:00
IndexTable = StoreInfo - > IndexTable ;
2011-10-19 14:40:52 +02:00
PtrTrack - > StartPtr = GetStartPointer ( VariableStoreHeader ) ;
PtrTrack - > EndPtr = GetEndPointer ( VariableStoreHeader ) ;
2013-01-25 07:17:43 +01:00
InDeletedVariable = NULL ;
2011-09-02 09:49:32 +02:00
//
// No Variable Address equals zero, so 0 as initial value is safe.
//
2011-10-19 14:40:52 +02:00
MaxIndex = NULL ;
2013-07-03 11:09:42 +02:00
VariableHeader = NULL ;
2011-09-02 09:49:32 +02:00
2011-10-19 14:40:52 +02:00
if ( IndexTable ! = NULL ) {
2011-09-02 09:49:32 +02:00
//
2011-10-19 14:40:52 +02:00
// traverse the variable index table to look for varible.
// The IndexTable->Index[Index] records the distance of two neighbouring VAR_ADDED type variables.
2011-09-02 09:49:32 +02:00
//
2011-10-19 14:40:52 +02:00
for ( Offset = 0 , Index = 0 ; Index < IndexTable - > Length ; Index + + ) {
ASSERT ( Index < sizeof ( IndexTable - > Index ) / sizeof ( IndexTable - > Index [ 0 ] ) ) ;
Offset + = IndexTable - > Index [ Index ] ;
MaxIndex = ( VARIABLE_HEADER * ) ( ( UINT8 * ) IndexTable - > StartPtr + Offset ) ;
2013-07-03 11:09:42 +02:00
GetVariableHeader ( StoreInfo , MaxIndex , & VariableHeader ) ;
if ( CompareWithValidVariable ( StoreInfo , MaxIndex , VariableHeader , VariableName , VendorGuid , PtrTrack ) = = EFI_SUCCESS ) {
if ( VariableHeader - > State = = ( VAR_IN_DELETED_TRANSITION & VAR_ADDED ) ) {
2013-01-25 07:17:43 +01:00
InDeletedVariable = PtrTrack - > CurrPtr ;
} else {
return EFI_SUCCESS ;
}
2011-09-02 09:49:32 +02:00
}
}
if ( IndexTable - > GoneThrough ! = 0 ) {
2011-10-19 14:40:52 +02:00
//
2013-01-25 07:17:43 +01:00
// If the table has all the existing variables indexed, return.
2011-10-19 14:40:52 +02:00
//
2013-01-25 07:17:43 +01:00
PtrTrack - > CurrPtr = InDeletedVariable ;
return ( PtrTrack - > CurrPtr = = NULL ) ? EFI_NOT_FOUND : EFI_SUCCESS ;
2011-09-02 09:49:32 +02:00
}
}
2011-10-19 14:40:52 +02:00
2011-09-02 09:49:32 +02:00
if ( MaxIndex ! = NULL ) {
2011-10-19 14:40:52 +02:00
//
// HOB exists but the variable cannot be found in HOB
// If not found in HOB, then let's start from the MaxIndex we've found.
//
2013-07-03 11:09:42 +02:00
Variable = GetNextVariablePtr ( StoreInfo , MaxIndex , VariableHeader ) ;
2011-09-02 09:49:32 +02:00
LastVariable = MaxIndex ;
} else {
2011-10-19 14:40:52 +02:00
//
// Start Pointers for the variable.
// Actual Data Pointer where data can be written.
//
Variable = PtrTrack - > StartPtr ;
LastVariable = PtrTrack - > StartPtr ;
2011-09-02 09:49:32 +02:00
}
2011-10-19 14:40:52 +02:00
2011-09-02 09:49:32 +02:00
//
2013-01-25 07:17:43 +01:00
// Find the variable by walk through variable store
2011-09-02 09:49:32 +02:00
//
2011-10-19 14:40:52 +02:00
StopRecord = FALSE ;
2013-07-03 11:09:42 +02:00
while ( GetVariableHeader ( StoreInfo , Variable , & VariableHeader ) ) {
if ( VariableHeader - > State = = VAR_ADDED | | VariableHeader - > State = = ( VAR_IN_DELETED_TRANSITION & VAR_ADDED ) ) {
2011-09-02 09:49:32 +02:00
//
// Record Variable in VariableIndex HOB
//
2011-10-19 14:40:52 +02:00
if ( ( IndexTable ! = NULL ) & & ! StopRecord ) {
Offset = ( UINTN ) Variable - ( UINTN ) LastVariable ;
if ( ( Offset > 0x0FFFF ) | | ( IndexTable - > Length = = sizeof ( IndexTable - > Index ) / sizeof ( IndexTable - > Index [ 0 ] ) ) ) {
//
// Stop to record if the distance of two neighbouring VAR_ADDED variable is larger than the allowable scope(UINT16),
// or the record buffer is full.
//
2011-09-02 09:49:32 +02:00
StopRecord = TRUE ;
2011-10-19 14:40:52 +02:00
} else {
2011-09-02 09:49:32 +02:00
IndexTable - > Index [ IndexTable - > Length + + ] = ( UINT16 ) Offset ;
2011-10-19 14:40:52 +02:00
LastVariable = Variable ;
2011-09-02 09:49:32 +02:00
}
}
2013-07-03 11:09:42 +02:00
if ( CompareWithValidVariable ( StoreInfo , Variable , VariableHeader , VariableName , VendorGuid , PtrTrack ) = = EFI_SUCCESS ) {
if ( VariableHeader - > State = = ( VAR_IN_DELETED_TRANSITION & VAR_ADDED ) ) {
2013-01-25 07:17:43 +01:00
InDeletedVariable = PtrTrack - > CurrPtr ;
} else {
return EFI_SUCCESS ;
}
2011-09-02 09:49:32 +02:00
}
}
2013-07-03 11:09:42 +02:00
Variable = GetNextVariablePtr ( StoreInfo , Variable , VariableHeader ) ;
2011-09-02 09:49:32 +02:00
}
//
// If gone through the VariableStore, that means we never find in Firmware any more.
//
2011-10-19 14:40:52 +02:00
if ( ( IndexTable ! = NULL ) & & ! StopRecord ) {
2011-09-02 09:49:32 +02:00
IndexTable - > GoneThrough = 1 ;
}
2013-01-25 07:17:43 +01:00
PtrTrack - > CurrPtr = InDeletedVariable ;
2011-09-02 09:49:32 +02:00
2013-01-25 07:17:43 +01:00
return ( PtrTrack - > CurrPtr = = NULL ) ? EFI_NOT_FOUND : EFI_SUCCESS ;
2011-09-02 09:49:32 +02:00
}
2011-10-19 14:40:52 +02:00
/**
Find the variable in HOB and Non - Volatile variable storages .
@ param VariableName Name of the variable to be found
@ param VendorGuid Vendor GUID to be found .
@ param PtrTrack Variable Track Pointer structure that contains Variable Information .
2013-07-03 11:09:42 +02:00
@ param StoreInfo Return the store info .
2011-10-19 14:40:52 +02:00
@ retval EFI_SUCCESS Variable found successfully
@ retval EFI_NOT_FOUND Variable not found
@ retval EFI_INVALID_PARAMETER Invalid variable name
* */
EFI_STATUS
FindVariable (
IN CONST CHAR16 * VariableName ,
IN CONST EFI_GUID * VendorGuid ,
2013-07-03 11:09:42 +02:00
OUT VARIABLE_POINTER_TRACK * PtrTrack ,
OUT VARIABLE_STORE_INFO * StoreInfo
2011-10-19 14:40:52 +02:00
)
{
EFI_STATUS Status ;
VARIABLE_STORE_TYPE Type ;
if ( VariableName [ 0 ] ! = 0 & & VendorGuid = = NULL ) {
return EFI_INVALID_PARAMETER ;
}
for ( Type = ( VARIABLE_STORE_TYPE ) 0 ; Type < VariableStoreTypeMax ; Type + + ) {
2013-07-03 11:09:42 +02:00
GetVariableStore ( Type , StoreInfo ) ;
2011-10-19 14:40:52 +02:00
Status = FindVariableEx (
2013-07-03 11:09:42 +02:00
StoreInfo ,
2011-10-19 14:40:52 +02:00
VariableName ,
VendorGuid ,
PtrTrack
) ;
if ( ! EFI_ERROR ( Status ) ) {
return Status ;
}
}
return EFI_NOT_FOUND ;
}
2011-09-02 09:49:32 +02:00
/**
This service retrieves a variable ' s value using its name and GUID .
Read the specified variable from the UEFI variable store . If the Data
buffer is too small to hold the contents of the variable , the error
EFI_BUFFER_TOO_SMALL is returned and DataSize is set to the required buffer
size to obtain the data .
@ param This A pointer to this instance of the EFI_PEI_READ_ONLY_VARIABLE2_PPI .
@ param VariableName A pointer to a null - terminated string that is the variable ' s name .
@ param VariableGuid A pointer to an EFI_GUID that is the variable ' s GUID . The combination of
VariableGuid and VariableName must be unique .
@ param Attributes If non - NULL , on return , points to the variable ' s attributes .
@ param DataSize On entry , points to the size in bytes of the Data buffer .
On return , points to the size of the data returned in Data .
@ param Data Points to the buffer which will hold the returned variable value .
@ retval EFI_SUCCESS The variable was read successfully .
@ retval EFI_NOT_FOUND The variable could not be found .
@ retval EFI_BUFFER_TOO_SMALL The DataSize is too small for the resulting data .
DataSize is updated with the size required for
the specified variable .
@ retval EFI_INVALID_PARAMETER VariableName , VariableGuid , DataSize or Data is NULL .
@ retval EFI_DEVICE_ERROR The variable could not be retrieved because of a device error .
* */
EFI_STATUS
EFIAPI
PeiGetVariable (
IN CONST EFI_PEI_READ_ONLY_VARIABLE2_PPI * This ,
IN CONST CHAR16 * VariableName ,
IN CONST EFI_GUID * VariableGuid ,
OUT UINT32 * Attributes ,
IN OUT UINTN * DataSize ,
OUT VOID * Data
)
{
VARIABLE_POINTER_TRACK Variable ;
UINTN VarDataSize ;
EFI_STATUS Status ;
2013-07-03 11:09:42 +02:00
VARIABLE_STORE_INFO StoreInfo ;
VARIABLE_HEADER * VariableHeader ;
2011-09-02 09:49:32 +02:00
if ( VariableName = = NULL | | VariableGuid = = NULL | | DataSize = = NULL ) {
return EFI_INVALID_PARAMETER ;
}
2011-09-07 12:10:11 +02:00
2011-09-02 09:49:32 +02:00
//
// Find existing variable
//
2013-07-03 11:09:42 +02:00
Status = FindVariable ( VariableName , VariableGuid , & Variable , & StoreInfo ) ;
2011-10-19 14:40:52 +02:00
if ( EFI_ERROR ( Status ) ) {
2011-09-02 09:49:32 +02:00
return Status ;
}
2013-07-03 11:09:42 +02:00
GetVariableHeader ( & StoreInfo , Variable . CurrPtr , & VariableHeader ) ;
2011-09-02 09:49:32 +02:00
//
// Get data size
//
2013-07-03 11:09:42 +02:00
VarDataSize = DataSizeOfVariable ( VariableHeader ) ;
2011-09-02 09:49:32 +02:00
if ( * DataSize > = VarDataSize ) {
if ( Data = = NULL ) {
return EFI_INVALID_PARAMETER ;
}
2013-07-03 11:09:42 +02:00
GetVariableNameOrData ( & StoreInfo , GetVariableDataPtr ( Variable . CurrPtr , VariableHeader ) , VarDataSize , Data ) ;
2011-09-02 09:49:32 +02:00
if ( Attributes ! = NULL ) {
2013-07-03 11:09:42 +02:00
* Attributes = VariableHeader - > Attributes ;
2011-09-02 09:49:32 +02:00
}
* DataSize = VarDataSize ;
return EFI_SUCCESS ;
} else {
* DataSize = VarDataSize ;
return EFI_BUFFER_TOO_SMALL ;
}
}
/**
Return the next variable name and GUID .
This function is called multiple times to retrieve the VariableName
and VariableGuid of all variables currently available in the system .
On each call , the previous results are passed into the interface ,
and , on return , the interface returns the data for the next
interface . When the entire variable list has been returned ,
EFI_NOT_FOUND is returned .
@ param This A pointer to this instance of the EFI_PEI_READ_ONLY_VARIABLE2_PPI .
@ param VariableNameSize On entry , points to the size of the buffer pointed to by VariableName .
On return , the size of the variable name buffer .
@ param VariableName On entry , a pointer to a null - terminated string that is the variable ' s name .
On return , points to the next variable ' s null - terminated name string .
@ param VariableGuid On entry , a pointer to an EFI_GUID that is the variable ' s GUID .
On return , a pointer to the next variable ' s GUID .
@ retval EFI_SUCCESS The variable was read successfully .
@ retval EFI_NOT_FOUND The variable could not be found .
@ retval EFI_BUFFER_TOO_SMALL The VariableNameSize is too small for the resulting
data . VariableNameSize is updated with the size
required for the specified variable .
@ retval EFI_INVALID_PARAMETER VariableName , VariableGuid or
VariableNameSize is NULL .
@ retval EFI_DEVICE_ERROR The variable could not be retrieved because of a device error .
* */
EFI_STATUS
EFIAPI
PeiGetNextVariableName (
IN CONST EFI_PEI_READ_ONLY_VARIABLE2_PPI * This ,
IN OUT UINTN * VariableNameSize ,
IN OUT CHAR16 * VariableName ,
IN OUT EFI_GUID * VariableGuid
)
{
2011-10-19 14:40:52 +02:00
VARIABLE_STORE_TYPE Type ;
2011-09-02 09:49:32 +02:00
VARIABLE_POINTER_TRACK Variable ;
2011-10-19 14:40:52 +02:00
VARIABLE_POINTER_TRACK VariableInHob ;
2013-01-25 07:17:43 +01:00
VARIABLE_POINTER_TRACK VariablePtrTrack ;
2011-09-02 09:49:32 +02:00
UINTN VarNameSize ;
EFI_STATUS Status ;
2011-10-19 14:40:52 +02:00
VARIABLE_STORE_HEADER * VariableStoreHeader [ VariableStoreTypeMax ] ;
2013-07-03 11:09:42 +02:00
VARIABLE_HEADER * VariableHeader ;
VARIABLE_STORE_INFO StoreInfo ;
VARIABLE_STORE_INFO StoreInfoForNv ;
VARIABLE_STORE_INFO StoreInfoForHob ;
2011-09-02 09:49:32 +02:00
if ( VariableName = = NULL | | VariableGuid = = NULL | | VariableNameSize = = NULL ) {
return EFI_INVALID_PARAMETER ;
}
2013-07-03 11:09:42 +02:00
Status = FindVariable ( VariableName , VariableGuid , & Variable , & StoreInfo ) ;
2011-09-02 09:49:32 +02:00
if ( Variable . CurrPtr = = NULL | | Status ! = EFI_SUCCESS ) {
return Status ;
}
if ( VariableName [ 0 ] ! = 0 ) {
//
// If variable name is not NULL, get next variable
//
2013-07-03 11:09:42 +02:00
GetVariableHeader ( & StoreInfo , Variable . CurrPtr , & VariableHeader ) ;
Variable . CurrPtr = GetNextVariablePtr ( & StoreInfo , Variable . CurrPtr , VariableHeader ) ;
2011-09-02 09:49:32 +02:00
}
2013-07-03 11:09:42 +02:00
VariableStoreHeader [ VariableStoreTypeHob ] = GetVariableStore ( VariableStoreTypeHob , & StoreInfoForHob ) ;
VariableStoreHeader [ VariableStoreTypeNv ] = GetVariableStore ( VariableStoreTypeNv , & StoreInfoForNv ) ;
2011-09-02 09:49:32 +02:00
2011-10-19 14:40:52 +02:00
while ( TRUE ) {
//
// Switch from HOB to Non-Volatile.
//
2013-07-03 11:09:42 +02:00
while ( ! GetVariableHeader ( & StoreInfo , Variable . CurrPtr , & VariableHeader ) ) {
2011-10-19 14:40:52 +02:00
//
// Find current storage index
//
for ( Type = ( VARIABLE_STORE_TYPE ) 0 ; Type < VariableStoreTypeMax ; Type + + ) {
if ( ( VariableStoreHeader [ Type ] ! = NULL ) & & ( Variable . StartPtr = = GetStartPointer ( VariableStoreHeader [ Type ] ) ) ) {
break ;
}
}
ASSERT ( Type < VariableStoreTypeMax ) ;
//
// Switch to next storage
//
for ( Type + + ; Type < VariableStoreTypeMax ; Type + + ) {
if ( VariableStoreHeader [ Type ] ! = NULL ) {
break ;
}
}
//
// Capture the case that
// 1. current storage is the last one, or
// 2. no further storage
//
if ( Type = = VariableStoreTypeMax ) {
return EFI_NOT_FOUND ;
}
Variable . StartPtr = GetStartPointer ( VariableStoreHeader [ Type ] ) ;
Variable . EndPtr = GetEndPointer ( VariableStoreHeader [ Type ] ) ;
Variable . CurrPtr = Variable . StartPtr ;
2013-07-03 11:09:42 +02:00
GetVariableStore ( Type , & StoreInfo ) ;
2011-10-19 14:40:52 +02:00
}
2011-09-02 09:49:32 +02:00
2013-07-03 11:09:42 +02:00
if ( VariableHeader - > State = = VAR_ADDED | | VariableHeader - > State = = ( VAR_IN_DELETED_TRANSITION & VAR_ADDED ) ) {
if ( VariableHeader - > State = = ( VAR_IN_DELETED_TRANSITION & VAR_ADDED ) ) {
2013-01-25 07:17:43 +01:00
//
// If it is a IN_DELETED_TRANSITION variable,
// and there is also a same ADDED one at the same time,
// don't return it.
//
Status = FindVariableEx (
2013-07-03 11:09:42 +02:00
& StoreInfo ,
2013-01-25 07:17:43 +01:00
GetVariableNamePtr ( Variable . CurrPtr ) ,
2013-07-03 11:09:42 +02:00
& VariableHeader - > VendorGuid ,
2013-01-25 07:17:43 +01:00
& VariablePtrTrack
) ;
2013-07-03 11:09:42 +02:00
if ( ! EFI_ERROR ( Status ) & & VariablePtrTrack . CurrPtr ! = Variable . CurrPtr ) {
Variable . CurrPtr = GetNextVariablePtr ( & StoreInfo , Variable . CurrPtr , VariableHeader ) ;
2013-01-25 07:17:43 +01:00
continue ;
}
}
2011-09-02 09:49:32 +02:00
2011-10-19 14:40:52 +02:00
//
// Don't return NV variable when HOB overrides it
//
if ( ( VariableStoreHeader [ VariableStoreTypeHob ] ! = NULL ) & & ( VariableStoreHeader [ VariableStoreTypeNv ] ! = NULL ) & &
( Variable . StartPtr = = GetStartPointer ( VariableStoreHeader [ VariableStoreTypeNv ] ) )
) {
Status = FindVariableEx (
2013-07-03 11:09:42 +02:00
& StoreInfoForHob ,
2011-10-19 14:40:52 +02:00
GetVariableNamePtr ( Variable . CurrPtr ) ,
2013-07-03 11:09:42 +02:00
& VariableHeader - > VendorGuid ,
2011-10-19 14:40:52 +02:00
& VariableInHob
) ;
if ( ! EFI_ERROR ( Status ) ) {
2013-07-03 11:09:42 +02:00
Variable . CurrPtr = GetNextVariablePtr ( & StoreInfo , Variable . CurrPtr , VariableHeader ) ;
2011-10-19 14:40:52 +02:00
continue ;
2011-09-02 09:49:32 +02:00
}
2011-10-19 14:40:52 +02:00
}
2011-09-02 09:49:32 +02:00
2013-07-03 11:09:42 +02:00
VarNameSize = NameSizeOfVariable ( VariableHeader ) ;
2011-10-19 14:40:52 +02:00
ASSERT ( VarNameSize ! = 0 ) ;
if ( VarNameSize < = * VariableNameSize ) {
2013-07-03 11:09:42 +02:00
GetVariableNameOrData ( & StoreInfo , ( UINT8 * ) GetVariableNamePtr ( Variable . CurrPtr ) , VarNameSize , ( UINT8 * ) VariableName ) ;
2011-10-19 14:40:52 +02:00
2013-07-03 11:09:42 +02:00
CopyMem ( VariableGuid , & VariableHeader - > VendorGuid , sizeof ( EFI_GUID ) ) ;
2011-10-19 14:40:52 +02:00
Status = EFI_SUCCESS ;
2011-09-02 09:49:32 +02:00
} else {
2011-10-19 14:40:52 +02:00
Status = EFI_BUFFER_TOO_SMALL ;
2011-09-02 09:49:32 +02:00
}
2011-10-19 14:40:52 +02:00
* VariableNameSize = VarNameSize ;
//
// Variable is found
//
return Status ;
2011-09-02 09:49:32 +02:00
} else {
2013-07-03 11:09:42 +02:00
Variable . CurrPtr = GetNextVariablePtr ( & StoreInfo , Variable . CurrPtr , VariableHeader ) ;
2011-09-02 09:49:32 +02:00
}
}
}