2016-05-26 13:12:14 +02:00
|
|
|
/** @file
|
|
|
|
Serialize Variables Library implementation
|
|
|
|
|
|
|
|
Copyright (c) 2004 - 2011, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:06:33 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2016-05-26 13:12:14 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "SerializeVariablesLib.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
Serialization format:
|
|
|
|
|
|
|
|
The SerializeVariablesLib interface does not specify a format
|
|
|
|
for the serialization of the variable data. This library uses
|
|
|
|
a packed array of a non-uniformly sized data structure elements.
|
|
|
|
|
|
|
|
Each variable is stored (packed) as:
|
|
|
|
UINT32 VendorNameSize; // Name size in bytes
|
|
|
|
CHAR16 VendorName[?]; // The variable unicode name including the
|
|
|
|
// null terminating character.
|
|
|
|
EFI_GUID VendorGuid; // The variable GUID
|
|
|
|
UINT32 DataSize; // The size of variable data in bytes
|
|
|
|
UINT8 Data[?]; // The variable data
|
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Unpacks the next variable from the buffer
|
|
|
|
|
|
|
|
@param[in] Buffer - Buffer pointing to the next variable instance
|
|
|
|
On subsequent calls, the pointer should be incremented
|
|
|
|
by the returned SizeUsed value.
|
|
|
|
@param[in] MaxSize - Max allowable size for the variable data
|
|
|
|
On subsequent calls, this should be decremented
|
|
|
|
by the returned SizeUsed value.
|
|
|
|
@param[out] Name - Variable name string (address in Buffer)
|
|
|
|
@param[out] NameSize - Size of Name in bytes
|
|
|
|
@param[out] Guid - GUID of variable (address in Buffer)
|
|
|
|
@param[out] Attributes - Attributes of variable
|
|
|
|
@param[out] Data - Buffer containing Data for variable (address in Buffer)
|
|
|
|
@param[out] DataSize - Size of Data in bytes
|
|
|
|
@param[out] SizeUsed - Total size used for this variable instance in Buffer
|
|
|
|
|
|
|
|
@return EFI_STATUS based on the success or failure of the operation
|
|
|
|
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
UnpackVariableFromBuffer (
|
|
|
|
IN VOID *Buffer,
|
|
|
|
IN UINTN MaxSize,
|
|
|
|
OUT CHAR16 **Name,
|
|
|
|
OUT UINT32 *NameSize,
|
|
|
|
OUT EFI_GUID **Guid,
|
|
|
|
OUT UINT32 *Attributes,
|
|
|
|
OUT UINT32 *DataSize,
|
|
|
|
OUT VOID **Data,
|
|
|
|
OUT UINTN *SizeUsed
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 *BytePtr;
|
|
|
|
UINTN Offset;
|
|
|
|
|
|
|
|
BytePtr = (UINT8*)Buffer;
|
|
|
|
Offset = 0;
|
|
|
|
|
|
|
|
*NameSize = *(UINT32*) (BytePtr + Offset);
|
|
|
|
Offset = Offset + sizeof (UINT32);
|
|
|
|
|
|
|
|
if (Offset > MaxSize) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*Name = (CHAR16*) (BytePtr + Offset);
|
|
|
|
Offset = Offset + *(UINT32*)BytePtr;
|
|
|
|
if (Offset > MaxSize) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*Guid = (EFI_GUID*) (BytePtr + Offset);
|
|
|
|
Offset = Offset + sizeof (EFI_GUID);
|
|
|
|
if (Offset > MaxSize) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*Attributes = *(UINT32*) (BytePtr + Offset);
|
|
|
|
Offset = Offset + sizeof (UINT32);
|
|
|
|
if (Offset > MaxSize) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*DataSize = *(UINT32*) (BytePtr + Offset);
|
|
|
|
Offset = Offset + sizeof (UINT32);
|
|
|
|
if (Offset > MaxSize) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*Data = (VOID*) (BytePtr + Offset);
|
|
|
|
Offset = Offset + *DataSize;
|
|
|
|
if (Offset > MaxSize) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*SizeUsed = Offset;
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Iterates through the variables in the buffer, and calls a callback
|
|
|
|
function for each variable found.
|
|
|
|
|
|
|
|
@param[in] CallbackFunction - Function called for each variable instance
|
|
|
|
@param[in] Context - Passed to each call of CallbackFunction
|
|
|
|
@param[in] Buffer - Buffer containing serialized variables
|
|
|
|
@param[in] MaxSize - Size of Buffer in bytes
|
|
|
|
|
|
|
|
@return EFI_STATUS based on the success or failure of the operation
|
|
|
|
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
IterateVariablesInBuffer (
|
|
|
|
IN VARIABLE_SERIALIZATION_ITERATION_CALLBACK CallbackFunction,
|
|
|
|
IN VOID *CallbackContext,
|
|
|
|
IN VOID *Buffer,
|
|
|
|
IN UINTN MaxSize
|
|
|
|
)
|
|
|
|
{
|
|
|
|
RETURN_STATUS Status;
|
|
|
|
UINTN TotalSizeUsed;
|
|
|
|
UINTN SizeUsed;
|
|
|
|
|
|
|
|
CHAR16 *Name;
|
|
|
|
UINT32 NameSize;
|
|
|
|
CHAR16 *AlignedName;
|
|
|
|
UINT32 AlignedNameMaxSize;
|
|
|
|
EFI_GUID *Guid;
|
|
|
|
UINT32 Attributes;
|
|
|
|
UINT32 DataSize;
|
|
|
|
VOID *Data;
|
|
|
|
|
|
|
|
SizeUsed = 0;
|
|
|
|
AlignedName = NULL;
|
|
|
|
AlignedNameMaxSize = 0;
|
|
|
|
Name = NULL;
|
|
|
|
Guid = NULL;
|
|
|
|
Attributes = 0;
|
|
|
|
DataSize = 0;
|
|
|
|
Data = NULL;
|
|
|
|
|
|
|
|
for (
|
|
|
|
Status = EFI_SUCCESS, TotalSizeUsed = 0;
|
|
|
|
!EFI_ERROR (Status) && (TotalSizeUsed < MaxSize);
|
|
|
|
) {
|
|
|
|
Status = UnpackVariableFromBuffer (
|
|
|
|
(VOID*) ((UINT8*) Buffer + TotalSizeUsed),
|
|
|
|
(MaxSize - TotalSizeUsed),
|
|
|
|
&Name,
|
|
|
|
&NameSize,
|
|
|
|
&Guid,
|
|
|
|
&Attributes,
|
|
|
|
&DataSize,
|
|
|
|
&Data,
|
|
|
|
&SizeUsed
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// We copy the name to a separately allocated buffer,
|
|
|
|
// to be sure it is 16-bit aligned.
|
|
|
|
//
|
|
|
|
if (NameSize > AlignedNameMaxSize) {
|
|
|
|
if (AlignedName != NULL) {
|
|
|
|
FreePool (AlignedName);
|
|
|
|
}
|
|
|
|
AlignedName = AllocatePool (NameSize);
|
|
|
|
}
|
|
|
|
if (AlignedName == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
CopyMem (AlignedName, Name, NameSize);
|
|
|
|
|
|
|
|
TotalSizeUsed = TotalSizeUsed + SizeUsed;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Run the callback function
|
|
|
|
//
|
|
|
|
Status = (*CallbackFunction) (
|
|
|
|
CallbackContext,
|
|
|
|
AlignedName,
|
|
|
|
Guid,
|
|
|
|
Attributes,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (AlignedName != NULL) {
|
|
|
|
FreePool (AlignedName);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Make sure the entire buffer was used, or else return an error
|
|
|
|
//
|
|
|
|
if (TotalSizeUsed != MaxSize) {
|
|
|
|
DEBUG ((
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG_ERROR,
|
2016-05-26 13:12:14 +02:00
|
|
|
"Deserialize variables error: TotalSizeUsed(%Lu) != MaxSize(%Lu)\n",
|
|
|
|
(UINT64)TotalSizeUsed,
|
|
|
|
(UINT64)MaxSize
|
|
|
|
));
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
STATIC
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
IterateVariablesCallbackNop (
|
|
|
|
IN VOID *Context,
|
|
|
|
IN CHAR16 *VariableName,
|
|
|
|
IN EFI_GUID *VendorGuid,
|
|
|
|
IN UINT32 Attributes,
|
|
|
|
IN UINTN DataSize,
|
|
|
|
IN VOID *Data
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
STATIC
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
IterateVariablesCallbackSetInInstance (
|
|
|
|
IN VOID *Context,
|
|
|
|
IN CHAR16 *VariableName,
|
|
|
|
IN EFI_GUID *VendorGuid,
|
|
|
|
IN UINT32 Attributes,
|
|
|
|
IN UINTN DataSize,
|
|
|
|
IN VOID *Data
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_HANDLE Instance;
|
|
|
|
|
|
|
|
Instance = (EFI_HANDLE) Context;
|
|
|
|
|
|
|
|
return SerializeVariablesAddVariable (
|
|
|
|
Instance,
|
|
|
|
VariableName,
|
|
|
|
VendorGuid,
|
|
|
|
Attributes,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
STATIC
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
IterateVariablesCallbackSetSystemVariable (
|
|
|
|
IN VOID *Context,
|
|
|
|
IN CHAR16 *VariableName,
|
|
|
|
IN EFI_GUID *VendorGuid,
|
|
|
|
IN UINT32 Attributes,
|
|
|
|
IN UINTN DataSize,
|
|
|
|
IN VOID *Data
|
|
|
|
)
|
|
|
|
{
|
2013-05-28 19:21:37 +02:00
|
|
|
EFI_STATUS Status;
|
|
|
|
STATIC CONST UINT32 AuthMask =
|
|
|
|
EFI_VARIABLE_AUTHENTICATED_WRITE_ACCESS |
|
|
|
|
EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS;
|
|
|
|
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
VariableName,
|
|
|
|
VendorGuid,
|
|
|
|
Attributes,
|
|
|
|
DataSize,
|
|
|
|
Data
|
|
|
|
);
|
|
|
|
|
|
|
|
if (Status == EFI_SECURITY_VIOLATION && (Attributes & AuthMask) != 0) {
|
|
|
|
DEBUG ((DEBUG_WARN, "%a: setting authenticated variable \"%s\" "
|
|
|
|
"failed with EFI_SECURITY_VIOLATION, ignoring\n", __FUNCTION__,
|
|
|
|
VariableName));
|
|
|
|
Status = EFI_SUCCESS;
|
2016-05-26 12:24:14 +02:00
|
|
|
} else if (Status == EFI_WRITE_PROTECTED) {
|
|
|
|
DEBUG ((DEBUG_WARN, "%a: setting ReadOnly variable \"%s\" "
|
|
|
|
"failed with EFI_WRITE_PROTECTED, ignoring\n", __FUNCTION__,
|
|
|
|
VariableName));
|
|
|
|
Status = EFI_SUCCESS;
|
2013-05-28 19:21:37 +02:00
|
|
|
}
|
|
|
|
return Status;
|
2016-05-26 13:12:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
STATIC
|
|
|
|
RETURN_STATUS
|
|
|
|
EnsureExtraBufferSpace (
|
|
|
|
IN SV_INSTANCE *Instance,
|
|
|
|
IN UINTN Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
VOID *NewBuffer;
|
|
|
|
UINTN NewSize;
|
|
|
|
|
|
|
|
NewSize = Instance->DataSize + Size;
|
|
|
|
if (NewSize <= Instance->BufferSize) {
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Double the required size to lessen the need to re-allocate in the future
|
|
|
|
//
|
|
|
|
NewSize = 2 * NewSize;
|
|
|
|
|
|
|
|
NewBuffer = AllocatePool (NewSize);
|
|
|
|
if (NewBuffer == NULL) {
|
|
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Instance->BufferPtr != NULL) {
|
|
|
|
CopyMem (NewBuffer, Instance->BufferPtr, Instance->DataSize);
|
|
|
|
FreePool (Instance->BufferPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
Instance->BufferPtr = NewBuffer;
|
|
|
|
Instance->BufferSize = NewSize;
|
|
|
|
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
AppendToBuffer (
|
|
|
|
IN SV_INSTANCE *Instance,
|
|
|
|
IN VOID *Data,
|
|
|
|
IN UINTN Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN NewSize;
|
|
|
|
|
|
|
|
ASSERT (Instance != NULL);
|
|
|
|
ASSERT (Data != NULL);
|
|
|
|
|
|
|
|
NewSize = Instance->DataSize + Size;
|
|
|
|
ASSERT ((Instance->DataSize + Size) <= Instance->BufferSize);
|
|
|
|
|
|
|
|
CopyMem (
|
|
|
|
(VOID*) (((UINT8*) (Instance->BufferPtr)) + Instance->DataSize),
|
|
|
|
Data,
|
|
|
|
Size
|
|
|
|
);
|
|
|
|
|
|
|
|
Instance->DataSize = NewSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Creates a new variable serialization instance
|
|
|
|
|
|
|
|
@param[out] Handle - Handle for a variable serialization instance
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS - The variable serialization instance was
|
|
|
|
successfully created.
|
|
|
|
@retval RETURN_OUT_OF_RESOURCES - There we not enough resources to
|
|
|
|
create the variable serialization instance.
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SerializeVariablesNewInstance (
|
|
|
|
OUT EFI_HANDLE *Handle
|
|
|
|
)
|
|
|
|
{
|
|
|
|
SV_INSTANCE *New;
|
|
|
|
|
|
|
|
New = AllocateZeroPool (sizeof (*New));
|
|
|
|
if (New == NULL) {
|
|
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
New->Signature = SV_SIGNATURE;
|
|
|
|
|
|
|
|
*Handle = (EFI_HANDLE) New;
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Free memory associated with a variable serialization instance
|
|
|
|
|
|
|
|
@param[in] Handle - Handle for a variable serialization instance
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS - The variable serialization instance was
|
|
|
|
successfully freed.
|
|
|
|
@retval RETURN_INVALID_PARAMETER - Handle was not a valid
|
|
|
|
variable serialization instance.
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SerializeVariablesFreeInstance (
|
|
|
|
IN EFI_HANDLE Handle
|
|
|
|
)
|
|
|
|
{
|
|
|
|
SV_INSTANCE *Instance;
|
|
|
|
|
|
|
|
Instance = SV_FROM_HANDLE (Handle);
|
|
|
|
|
|
|
|
if (Instance->Signature != SV_SIGNATURE) {
|
|
|
|
return RETURN_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
Instance->Signature = 0;
|
|
|
|
|
|
|
|
if (Instance->BufferPtr != NULL) {
|
|
|
|
FreePool (Instance->BufferPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
FreePool (Instance);
|
|
|
|
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Creates a new variable serialization instance using the given
|
|
|
|
binary representation of the variables to fill the new instance
|
|
|
|
|
|
|
|
@param[out] Handle - Handle for a variable serialization instance
|
|
|
|
@param[in] Buffer - A buffer with the serialized representation
|
|
|
|
of the variables. Must be the same format as produced
|
|
|
|
by SerializeVariablesToBuffer.
|
|
|
|
@param[in] Size - This is the size of the binary representation
|
|
|
|
of the variables.
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS - The binary representation was successfully
|
|
|
|
imported into a new variable serialization instance
|
|
|
|
@retval RETURN_OUT_OF_RESOURCES - There we not enough resources to
|
|
|
|
create the new variable serialization instance
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SerializeVariablesNewInstanceFromBuffer (
|
|
|
|
OUT EFI_HANDLE *Handle,
|
|
|
|
IN VOID *Buffer,
|
|
|
|
IN UINTN Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
RETURN_STATUS Status;
|
|
|
|
|
|
|
|
Status = SerializeVariablesNewInstance (Handle);
|
|
|
|
if (RETURN_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = IterateVariablesInBuffer (
|
|
|
|
IterateVariablesCallbackNop,
|
|
|
|
NULL,
|
|
|
|
Buffer,
|
|
|
|
Size
|
|
|
|
);
|
|
|
|
if (RETURN_ERROR (Status)) {
|
|
|
|
SerializeVariablesFreeInstance (*Handle);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = IterateVariablesInBuffer (
|
|
|
|
IterateVariablesCallbackSetInInstance,
|
|
|
|
(VOID*) *Handle,
|
|
|
|
Buffer,
|
|
|
|
Size
|
|
|
|
);
|
|
|
|
if (RETURN_ERROR (Status)) {
|
|
|
|
SerializeVariablesFreeInstance (*Handle);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Iterates all variables found with RuntimeServices GetNextVariableName
|
|
|
|
|
|
|
|
@param[in] CallbackFunction - Function called for each variable instance
|
|
|
|
@param[in] Context - Passed to each call of CallbackFunction
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS - All variables were iterated without the
|
|
|
|
CallbackFunction returning an error
|
|
|
|
@retval RETURN_OUT_OF_RESOURCES - There we not enough resources to
|
|
|
|
iterate through the variables
|
|
|
|
@return Any of RETURN_ERROR indicates an error reading the variable
|
|
|
|
or an error was returned from CallbackFunction
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SerializeVariablesIterateSystemVariables (
|
|
|
|
IN VARIABLE_SERIALIZATION_ITERATION_CALLBACK CallbackFunction,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
RETURN_STATUS Status;
|
|
|
|
UINTN VariableNameBufferSize;
|
|
|
|
UINTN VariableNameSize;
|
|
|
|
CHAR16 *VariableName;
|
|
|
|
EFI_GUID VendorGuid;
|
|
|
|
UINTN VariableDataBufferSize;
|
|
|
|
UINTN VariableDataSize;
|
|
|
|
VOID *VariableData;
|
|
|
|
UINT32 VariableAttributes;
|
|
|
|
VOID *NewBuffer;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Initialize the variable name and data buffer variables.
|
|
|
|
//
|
|
|
|
VariableNameBufferSize = sizeof (CHAR16);
|
|
|
|
VariableName = AllocateZeroPool (VariableNameBufferSize);
|
|
|
|
|
|
|
|
VariableDataBufferSize = 0;
|
|
|
|
VariableData = NULL;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
//
|
|
|
|
// Get the next variable name and guid
|
|
|
|
//
|
|
|
|
VariableNameSize = VariableNameBufferSize;
|
|
|
|
Status = gRT->GetNextVariableName (
|
|
|
|
&VariableNameSize,
|
|
|
|
VariableName,
|
|
|
|
&VendorGuid
|
|
|
|
);
|
|
|
|
if (Status == EFI_BUFFER_TOO_SMALL) {
|
|
|
|
//
|
|
|
|
// The currently allocated VariableName buffer is too small,
|
|
|
|
// so we allocate a larger buffer, and copy the old buffer
|
|
|
|
// to it.
|
|
|
|
//
|
|
|
|
NewBuffer = AllocatePool (VariableNameSize);
|
|
|
|
if (NewBuffer == NULL) {
|
|
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
CopyMem (NewBuffer, VariableName, VariableNameBufferSize);
|
|
|
|
if (VariableName != NULL) {
|
|
|
|
FreePool (VariableName);
|
|
|
|
}
|
|
|
|
VariableName = NewBuffer;
|
|
|
|
VariableNameBufferSize = VariableNameSize;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Try to get the next variable name again with the larger buffer.
|
|
|
|
//
|
|
|
|
Status = gRT->GetNextVariableName (
|
|
|
|
&VariableNameSize,
|
|
|
|
VariableName,
|
|
|
|
&VendorGuid
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
if (Status == EFI_NOT_FOUND) {
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Get the variable data and attributes
|
|
|
|
//
|
|
|
|
VariableDataSize = VariableDataBufferSize;
|
|
|
|
Status = gRT->GetVariable (
|
|
|
|
VariableName,
|
|
|
|
&VendorGuid,
|
|
|
|
&VariableAttributes,
|
|
|
|
&VariableDataSize,
|
|
|
|
VariableData
|
|
|
|
);
|
|
|
|
if (Status == EFI_BUFFER_TOO_SMALL) {
|
|
|
|
//
|
|
|
|
// The currently allocated VariableData buffer is too small,
|
|
|
|
// so we allocate a larger buffer.
|
|
|
|
//
|
|
|
|
if (VariableDataBufferSize != 0) {
|
|
|
|
FreePool (VariableData);
|
|
|
|
VariableData = NULL;
|
|
|
|
VariableDataBufferSize = 0;
|
|
|
|
}
|
|
|
|
VariableData = AllocatePool (VariableDataSize);
|
|
|
|
if (VariableData == NULL) {
|
|
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
VariableDataBufferSize = VariableDataSize;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Try to read the variable again with the larger buffer.
|
|
|
|
//
|
|
|
|
Status = gRT->GetVariable (
|
|
|
|
VariableName,
|
|
|
|
&VendorGuid,
|
|
|
|
&VariableAttributes,
|
|
|
|
&VariableDataSize,
|
|
|
|
VariableData
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Run the callback function
|
|
|
|
//
|
|
|
|
Status = (*CallbackFunction) (
|
|
|
|
Context,
|
|
|
|
VariableName,
|
|
|
|
&VendorGuid,
|
|
|
|
VariableAttributes,
|
|
|
|
VariableDataSize,
|
|
|
|
VariableData
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VariableName != NULL) {
|
|
|
|
FreePool (VariableName);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VariableData != NULL) {
|
|
|
|
FreePool (VariableData);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Iterates all variables found in the variable serialization instance
|
|
|
|
|
|
|
|
@param[in] Handle - Handle for a variable serialization instance
|
|
|
|
@param[in] CallbackFunction - Function called for each variable instance
|
|
|
|
@param[in] Context - Passed to each call of CallbackFunction
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS - All variables were iterated without the
|
|
|
|
CallbackFunction returning an error
|
|
|
|
@retval RETURN_OUT_OF_RESOURCES - There we not enough resources to
|
|
|
|
iterate through the variables
|
|
|
|
@return Any of RETURN_ERROR indicates an error reading the variable
|
|
|
|
or an error was returned from CallbackFunction
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SerializeVariablesIterateInstanceVariables (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN VARIABLE_SERIALIZATION_ITERATION_CALLBACK CallbackFunction,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
SV_INSTANCE *Instance;
|
|
|
|
|
|
|
|
Instance = SV_FROM_HANDLE (Handle);
|
|
|
|
|
|
|
|
if ((Instance->BufferPtr != NULL) && (Instance->DataSize != 0)) {
|
|
|
|
return IterateVariablesInBuffer (
|
|
|
|
CallbackFunction,
|
|
|
|
Context,
|
|
|
|
Instance->BufferPtr,
|
|
|
|
Instance->DataSize
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Sets all variables found in the variable serialization instance
|
|
|
|
|
|
|
|
@param[in] Handle - Handle for a variable serialization instance
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS - All variables were set successfully
|
|
|
|
@retval RETURN_OUT_OF_RESOURCES - There we not enough resources to
|
|
|
|
set all the variables
|
|
|
|
@return Any of RETURN_ERROR indicates an error reading the variables
|
|
|
|
or in attempting to set a variable
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SerializeVariablesSetSerializedVariables (
|
|
|
|
IN EFI_HANDLE Handle
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return SerializeVariablesIterateInstanceVariables (
|
|
|
|
Handle,
|
|
|
|
IterateVariablesCallbackSetSystemVariable,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Adds a variable to the variable serialization instance
|
|
|
|
|
|
|
|
@param[in] Handle - Handle for a variable serialization instance
|
|
|
|
@param[in] VariableName - Refer to RuntimeServices GetVariable
|
|
|
|
@param[in] VendorGuid - Refer to RuntimeServices GetVariable
|
|
|
|
@param[in] Attributes - Refer to RuntimeServices GetVariable
|
|
|
|
@param[in] DataSize - Refer to RuntimeServices GetVariable
|
|
|
|
@param[in] Data - Refer to RuntimeServices GetVariable
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS - All variables were set successfully
|
|
|
|
@retval RETURN_OUT_OF_RESOURCES - There we not enough resources to
|
|
|
|
add the variable
|
|
|
|
@retval RETURN_INVALID_PARAMETER - Handle was not a valid
|
|
|
|
variable serialization instance or
|
|
|
|
VariableName, VariableGuid or Data are NULL.
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SerializeVariablesAddVariable (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN CHAR16 *VariableName,
|
|
|
|
IN EFI_GUID *VendorGuid,
|
|
|
|
IN UINT32 Attributes,
|
|
|
|
IN UINTN DataSize,
|
|
|
|
IN VOID *Data
|
|
|
|
)
|
|
|
|
{
|
|
|
|
RETURN_STATUS Status;
|
|
|
|
SV_INSTANCE *Instance;
|
|
|
|
UINT32 SerializedNameSize;
|
|
|
|
UINT32 SerializedDataSize;
|
|
|
|
UINTN SerializedSize;
|
|
|
|
|
|
|
|
Instance = SV_FROM_HANDLE (Handle);
|
|
|
|
|
|
|
|
if ((Instance->Signature != SV_SIGNATURE) ||
|
|
|
|
(VariableName == NULL) || (VendorGuid == NULL) || (Data == NULL)) {
|
|
|
|
}
|
|
|
|
|
|
|
|
SerializedNameSize = (UINT32) StrSize (VariableName);
|
|
|
|
|
|
|
|
SerializedSize =
|
|
|
|
sizeof (SerializedNameSize) +
|
|
|
|
SerializedNameSize +
|
|
|
|
sizeof (*VendorGuid) +
|
|
|
|
sizeof (Attributes) +
|
|
|
|
sizeof (SerializedDataSize) +
|
|
|
|
DataSize;
|
|
|
|
|
|
|
|
Status = EnsureExtraBufferSpace (
|
|
|
|
Instance,
|
|
|
|
SerializedSize
|
|
|
|
);
|
|
|
|
if (RETURN_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Add name size (UINT32)
|
|
|
|
//
|
|
|
|
AppendToBuffer (Instance, (VOID*) &SerializedNameSize, sizeof (SerializedNameSize));
|
|
|
|
|
|
|
|
//
|
|
|
|
// Add variable unicode name string
|
|
|
|
//
|
|
|
|
AppendToBuffer (Instance, (VOID*) VariableName, SerializedNameSize);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Add variable GUID
|
|
|
|
//
|
|
|
|
AppendToBuffer (Instance, (VOID*) VendorGuid, sizeof (*VendorGuid));
|
|
|
|
|
|
|
|
//
|
|
|
|
// Add variable attributes
|
|
|
|
//
|
|
|
|
AppendToBuffer (Instance, (VOID*) &Attributes, sizeof (Attributes));
|
|
|
|
|
|
|
|
//
|
|
|
|
// Add variable data size (UINT32)
|
|
|
|
//
|
|
|
|
SerializedDataSize = (UINT32) DataSize;
|
|
|
|
AppendToBuffer (Instance, (VOID*) &SerializedDataSize, sizeof (SerializedDataSize));
|
|
|
|
|
|
|
|
//
|
|
|
|
// Add variable data
|
|
|
|
//
|
|
|
|
AppendToBuffer (Instance, Data, DataSize);
|
|
|
|
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Serializes the variables known to this instance into the
|
|
|
|
provided buffer.
|
|
|
|
|
|
|
|
@param[in] Handle - Handle for a variable serialization instance
|
|
|
|
@param[out] Buffer - A buffer to store the binary representation
|
|
|
|
of the variables.
|
|
|
|
@param[in,out] Size - On input this is the size of the buffer.
|
|
|
|
On output this is the size of the binary representation
|
|
|
|
of the variables.
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS - The binary representation was successfully
|
|
|
|
completed and returned in the buffer.
|
|
|
|
@retval RETURN_OUT_OF_RESOURCES - There we not enough resources to
|
|
|
|
save the variables to the buffer.
|
|
|
|
@retval RETURN_INVALID_PARAMETER - Handle was not a valid
|
|
|
|
variable serialization instance or
|
|
|
|
Size or Buffer were NULL.
|
|
|
|
@retval RETURN_BUFFER_TOO_SMALL - The Buffer size as indicated by
|
|
|
|
the Size parameter was too small for the serialized
|
|
|
|
variable data. Size is returned with the required size.
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SerializeVariablesToBuffer (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
OUT VOID *Buffer,
|
|
|
|
IN OUT UINTN *Size
|
|
|
|
)
|
|
|
|
{
|
|
|
|
SV_INSTANCE *Instance;
|
|
|
|
|
|
|
|
Instance = SV_FROM_HANDLE (Handle);
|
|
|
|
|
|
|
|
if (Size == NULL) {
|
|
|
|
return RETURN_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*Size < Instance->DataSize) {
|
|
|
|
*Size = Instance->DataSize;
|
|
|
|
return RETURN_BUFFER_TOO_SMALL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Buffer == NULL) {
|
|
|
|
return RETURN_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*Size = Instance->DataSize;
|
|
|
|
CopyMem (Buffer, Instance->BufferPtr, Instance->DataSize);
|
|
|
|
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|