mirror of https://github.com/acidanthera/audk.git
700 lines
20 KiB
C
700 lines
20 KiB
C
/**@file
|
|
|
|
This file contains the keyboard processing code to the HII database.
|
|
|
|
Copyright (c) 2006 - 2008, Intel Corporation
|
|
All rights reserved. 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 "HiiDatabase.h"
|
|
#include "HiiHandle.h"
|
|
|
|
EFI_GUID gFrameworkHiiCompatbilityGuid = EFI_IFR_FRAMEWORK_GUID;
|
|
EFI_GUID gTianoHiiIfrGuid = EFI_IFR_TIANO_GUID;
|
|
|
|
|
|
EFI_HII_HANDLE
|
|
FwHiiHandleToUefiHiiHandle (
|
|
IN CONST HII_THUNK_PRIVATE_DATA *Private,
|
|
IN FRAMEWORK_EFI_HII_HANDLE FwHiiHandle
|
|
)
|
|
{
|
|
HII_THUNK_CONTEXT *ThunkContext;
|
|
|
|
ASSERT (FwHiiHandle != (FRAMEWORK_EFI_HII_HANDLE) 0);
|
|
ASSERT (Private != NULL);
|
|
|
|
ThunkContext = FwHiiHandleToThunkContext (Private, FwHiiHandle);
|
|
|
|
if (ThunkContext != NULL) {
|
|
return ThunkContext->UefiHiiHandle;
|
|
}
|
|
|
|
return (EFI_HII_HANDLE) NULL;
|
|
}
|
|
|
|
|
|
HII_THUNK_CONTEXT *
|
|
FwHiiHandleToThunkContext (
|
|
IN CONST HII_THUNK_PRIVATE_DATA *Private,
|
|
IN FRAMEWORK_EFI_HII_HANDLE FwHiiHandle
|
|
)
|
|
{
|
|
LIST_ENTRY *Link;
|
|
HII_THUNK_CONTEXT *ThunkContext;
|
|
|
|
|
|
Link = GetFirstNode (&Private->ThunkContextListHead);
|
|
|
|
while (!IsNull (&Private->ThunkContextListHead, Link)) {
|
|
ThunkContext = HII_THUNK_CONTEXT_FROM_LINK (Link);
|
|
|
|
if (FwHiiHandle == ThunkContext->FwHiiHandle) {
|
|
return ThunkContext;
|
|
}
|
|
|
|
Link = GetNextNode (&Private->ThunkContextListHead, Link);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
HII_THUNK_CONTEXT *
|
|
UefiHiiHandleToThunkContext (
|
|
IN CONST HII_THUNK_PRIVATE_DATA *Private,
|
|
IN EFI_HII_HANDLE UefiHiiHandle
|
|
)
|
|
{
|
|
LIST_ENTRY *Link;
|
|
HII_THUNK_CONTEXT *ThunkContext;
|
|
|
|
Link = GetFirstNode (&Private->ThunkContextListHead);
|
|
|
|
while (!IsNull (&Private->ThunkContextListHead, Link)) {
|
|
ThunkContext = HII_THUNK_CONTEXT_FROM_LINK (Link);
|
|
|
|
if (UefiHiiHandle == ThunkContext->UefiHiiHandle) {
|
|
return ThunkContext;
|
|
}
|
|
Link = GetNextNode (&Private->ThunkContextListHead, Link);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
HII_THUNK_CONTEXT *
|
|
TagGuidToIfrPackThunkContext (
|
|
IN CONST HII_THUNK_PRIVATE_DATA *Private,
|
|
IN CONST EFI_GUID *Guid
|
|
)
|
|
{
|
|
LIST_ENTRY *Link;
|
|
HII_THUNK_CONTEXT *ThunkContext;
|
|
|
|
Link = GetFirstNode (&Private->ThunkContextListHead);
|
|
|
|
while (!IsNull (&Private->ThunkContextListHead, Link)) {
|
|
ThunkContext = HII_THUNK_CONTEXT_FROM_LINK (Link);
|
|
|
|
if (CompareGuid (Guid, &ThunkContext->TagGuid) && (ThunkContext->IfrPackageCount != 0)) {
|
|
return ThunkContext;
|
|
}
|
|
|
|
Link = GetNextNode (&Private->ThunkContextListHead, Link);
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
VOID
|
|
DestroyThunkContextForUefiHiiHandle (
|
|
IN HII_THUNK_PRIVATE_DATA *Private,
|
|
IN EFI_HII_HANDLE UefiHiiHandle
|
|
)
|
|
{
|
|
HII_THUNK_CONTEXT *ThunkContext;
|
|
|
|
ThunkContext = UefiHiiHandleToThunkContext (Private, UefiHiiHandle);
|
|
ASSERT (ThunkContext != NULL);
|
|
|
|
DestroyThunkContext (ThunkContext);
|
|
}
|
|
|
|
|
|
/**
|
|
This function create a HII_THUNK_CONTEXT for a package list registered
|
|
by a module calling EFI_HII_DATABASE_PROTOCOL.NewPackageList. It records
|
|
the PackageListGuid in EFI_HII_PACKAGE_LIST_HEADER in the TagGuid in
|
|
HII_THUNK_CONTEXT created. This TagGuid will be used as a key to s
|
|
|
|
**/
|
|
HII_THUNK_CONTEXT *
|
|
CreateThunkContextForUefiHiiHandle (
|
|
IN EFI_HII_HANDLE UefiHiiHandle
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
EFI_GUID PackageGuid;
|
|
HII_THUNK_CONTEXT *ThunkContext;
|
|
|
|
ThunkContext = AllocateZeroPool (sizeof (*ThunkContext));
|
|
ASSERT (ThunkContext != NULL);
|
|
|
|
ThunkContext->Signature = HII_THUNK_CONTEXT_SIGNATURE;
|
|
|
|
Status = AllocateHiiHandle (&ThunkContext->FwHiiHandle);
|
|
if (EFI_ERROR (Status)) {
|
|
return NULL;
|
|
}
|
|
|
|
ThunkContext->UefiHiiHandle = UefiHiiHandle;
|
|
|
|
Status = HiiLibExtractGuidFromHiiHandle (UefiHiiHandle, &PackageGuid);
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
CopyGuid(&ThunkContext->TagGuid, &PackageGuid);
|
|
|
|
InitializeListHead (&ThunkContext->QuestionIdMapListHead);
|
|
InitializeListHead (&ThunkContext->OneOfOptionMapListHead);
|
|
|
|
return ThunkContext;
|
|
}
|
|
|
|
|
|
UINTN
|
|
GetPackageCountByType (
|
|
IN CONST EFI_HII_PACKAGE_LIST_HEADER *PackageListHeader,
|
|
IN UINT8 PackageType
|
|
)
|
|
{
|
|
UINTN Count;
|
|
EFI_HII_PACKAGE_HEADER *PackageHeader;
|
|
|
|
PackageHeader = (EFI_HII_PACKAGE_HEADER *) ((UINT8 *) PackageListHeader + sizeof (EFI_HII_PACKAGE_LIST_HEADER));
|
|
Count = 0;
|
|
|
|
while (PackageHeader->Type != EFI_HII_PACKAGE_END) {
|
|
if (PackageHeader->Type == PackageType ) {
|
|
Count++;
|
|
}
|
|
PackageHeader = (EFI_HII_PACKAGE_HEADER *) ((UINT8 *) PackageHeader + PackageHeader->Length);
|
|
}
|
|
|
|
|
|
return Count;
|
|
}
|
|
|
|
LIST_ENTRY *
|
|
GetOneOfOptionMapEntryListHead (
|
|
IN CONST HII_THUNK_CONTEXT *ThunkContext,
|
|
IN UINT16 QuestionId
|
|
)
|
|
{
|
|
LIST_ENTRY *Link;
|
|
ONE_OF_OPTION_MAP *Map;
|
|
|
|
Link = GetFirstNode (&ThunkContext->OneOfOptionMapListHead);
|
|
|
|
while (!IsNull (&ThunkContext->OneOfOptionMapListHead, Link)) {
|
|
Map = ONE_OF_OPTION_MAP_FROM_LINK (Link);
|
|
if (QuestionId == Map->QuestionId) {
|
|
return &Map->OneOfOptionMapEntryListHead;
|
|
}
|
|
Link = GetNextNode (&ThunkContext->OneOfOptionMapListHead, Link);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
EFI_HII_PACKAGE_HEADER *
|
|
GetIfrPackage (
|
|
IN CONST EFI_HII_PACKAGES *Packages
|
|
)
|
|
{
|
|
UINTN Index;
|
|
TIANO_AUTOGEN_PACKAGES_HEADER **TianoAutogenPackageHdrArray;
|
|
|
|
ASSERT (Packages != NULL);
|
|
|
|
TianoAutogenPackageHdrArray = (TIANO_AUTOGEN_PACKAGES_HEADER **) (((UINT8 *) &Packages->GuidId) + sizeof (Packages->GuidId));
|
|
|
|
for (Index = 0; Index < Packages->NumberOfPackages; Index++) {
|
|
//
|
|
// The current UEFI HII build tool generate a binary in the format defined by
|
|
// TIANO_AUTOGEN_PACKAGES_HEADER. We assume that all packages generated in
|
|
// this binary is with same package type. So the returned IfrPackageCount and StringPackageCount
|
|
// may not be the exact number of valid package number in the binary generated
|
|
// by HII Build tool.
|
|
//
|
|
switch (TianoAutogenPackageHdrArray[Index]->PackageHeader.Type) {
|
|
case EFI_HII_PACKAGE_FORM:
|
|
return &TianoAutogenPackageHdrArray[Index]->PackageHeader;
|
|
break;
|
|
case EFI_HII_PACKAGE_STRINGS:
|
|
case EFI_HII_PACKAGE_SIMPLE_FONTS:
|
|
break;
|
|
|
|
//
|
|
// The following fonts are invalid for a module that using Framework to UEFI thunk layer.
|
|
//
|
|
case EFI_HII_PACKAGE_KEYBOARD_LAYOUT:
|
|
case EFI_HII_PACKAGE_FONTS:
|
|
case EFI_HII_PACKAGE_IMAGES:
|
|
default:
|
|
ASSERT (FALSE);
|
|
return NULL;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
VOID
|
|
GetFormSetGuid (
|
|
IN EFI_HII_PACKAGE_HEADER *Package,
|
|
OUT EFI_GUID *FormSetGuid
|
|
)
|
|
{
|
|
UINTN Offset;
|
|
EFI_IFR_OP_HEADER *OpCode;
|
|
EFI_IFR_FORM_SET *FormSet;
|
|
|
|
Offset = sizeof (EFI_HII_PACKAGE_HEADER);
|
|
while (Offset < Package->Length) {
|
|
OpCode = (EFI_IFR_OP_HEADER *)((UINT8 *) Package + Offset);
|
|
|
|
switch (OpCode->OpCode) {
|
|
case EFI_IFR_FORM_SET_OP:
|
|
FormSet = (EFI_IFR_FORM_SET *) OpCode;
|
|
CopyMem (FormSetGuid, &FormSet->Guid, sizeof (EFI_GUID));
|
|
return;
|
|
|
|
default:
|
|
break;
|
|
|
|
}
|
|
Offset += OpCode->Length;
|
|
}
|
|
|
|
//
|
|
// A proper IFR must have a formset opcode.
|
|
//
|
|
ASSERT (FALSE);
|
|
|
|
}
|
|
|
|
|
|
VOID
|
|
GetAttributesOfFirstFormSet (
|
|
IN OUT HII_THUNK_CONTEXT *ThunkContext
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
EFI_HII_PACKAGE_LIST_HEADER *List;
|
|
EFI_HII_PACKAGE_HEADER *Package;
|
|
UINTN Size;
|
|
EFI_IFR_OP_HEADER *OpCode;
|
|
UINTN Offset;
|
|
EFI_IFR_GUID_CLASS *Class;
|
|
EFI_IFR_FORM_SET *FormSet;
|
|
EFI_IFR_GUID_SUBCLASS *SubClass;
|
|
|
|
Status = HiiLibExportPackageLists (ThunkContext->UefiHiiHandle, &List, &Size);
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
//
|
|
// There must be at least one EFI_HII_PACKAGE_FORM in the package list.
|
|
//
|
|
ASSERT (GetPackageCountByType (List, EFI_HII_PACKAGE_FORM) >= 1);
|
|
|
|
//
|
|
// Skip the package list header.
|
|
//
|
|
Package = (EFI_HII_PACKAGE_HEADER *) (List + 1);
|
|
|
|
while (Package->Type != EFI_HII_PACKAGE_END) {
|
|
|
|
if (Package->Type == EFI_HII_PACKAGE_FORM) {
|
|
|
|
//
|
|
// Skip the package header
|
|
//
|
|
Offset = sizeof (EFI_HII_PACKAGE_HEADER);
|
|
while (Offset < Package->Length) {
|
|
OpCode = (EFI_IFR_OP_HEADER *)((UINT8 *) Package + Offset);
|
|
|
|
switch (OpCode->OpCode) {
|
|
case EFI_IFR_FORM_SET_OP:
|
|
FormSet = (EFI_IFR_FORM_SET *) OpCode;
|
|
ThunkContext->FormSetTitle = FormSet->FormSetTitle;
|
|
ThunkContext->FormSetHelp = FormSet->Help;
|
|
break;
|
|
|
|
|
|
case EFI_IFR_GUID_OP:
|
|
Class = (EFI_IFR_GUID_CLASS*) OpCode;
|
|
if (CompareMem (&Class->Guid, &gTianoHiiIfrGuid, sizeof (EFI_GUID)) == 0) {
|
|
Class = (EFI_IFR_GUID_CLASS *) OpCode;
|
|
|
|
switch (Class->ExtendOpCode) {
|
|
case EFI_IFR_EXTEND_OP_CLASS:
|
|
ThunkContext->FormSetClass = Class->Class;
|
|
break;
|
|
case EFI_IFR_EXTEND_OP_SUBCLASS:
|
|
SubClass = (EFI_IFR_GUID_SUBCLASS *) OpCode;
|
|
ThunkContext->FormSetSubClass = SubClass->SubClass;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
|
|
}
|
|
|
|
Offset += OpCode->Length;
|
|
}
|
|
//
|
|
// The attributes of first FormSet is ready now.
|
|
//
|
|
FreePool (List);
|
|
return;
|
|
|
|
break;
|
|
}
|
|
|
|
Package = (EFI_HII_PACKAGE_HEADER *) ((UINT8 *) Package + Package->Length);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
EFI_STATUS
|
|
CreateQuestionIdMap (
|
|
IN OUT HII_THUNK_CONTEXT *ThunkContext
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
EFI_HII_PACKAGE_LIST_HEADER *List;
|
|
EFI_HII_PACKAGE_HEADER *Package;
|
|
UINTN Size;
|
|
EFI_IFR_OP_HEADER *OpCode;
|
|
UINTN Offset;
|
|
QUESTION_ID_MAP *IdMap;
|
|
EFI_IFR_VARSTORE *VarStore;
|
|
EFI_IFR_FORM_SET *FormSet;
|
|
EFI_IFR_QUESTION_HEADER *Question;
|
|
LIST_ENTRY *QuestionIdMapEntryListHead;
|
|
LIST_ENTRY *OneOfOptinMapEntryListHead;
|
|
QUESTION_ID_MAP_ENTRY *IdMapEntry;
|
|
EFI_IFR_GUID_OPTIONKEY *OptionMap;
|
|
ONE_OF_OPTION_MAP *OneOfOptionMap;
|
|
ONE_OF_OPTION_MAP_ENTRY *OneOfOptionMapEntry;
|
|
EFI_IFR_GUID_CLASS *Class;
|
|
EFI_IFR_GUID_SUBCLASS *SubClass;
|
|
UINT8 OneOfType;
|
|
EFI_IFR_ONE_OF *OneOfOpcode;
|
|
|
|
//
|
|
// Set to a invalid value.
|
|
//
|
|
OneOfType = (UINT8) -1;
|
|
|
|
|
|
Status = HiiLibExportPackageLists (ThunkContext->UefiHiiHandle, &List, &Size);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
//
|
|
// Get all VarStoreId and build the the QuestionId map.
|
|
// EFI_IFR_QUESTION_HEADER.VarStoreInfo.VarOffset -> Framework Question ID
|
|
// EFI_IFR_QUESTION_HEADER.QuestionId -> UEFI Question ID
|
|
//
|
|
|
|
//
|
|
// Skip the package list header.
|
|
//
|
|
Package = (EFI_HII_PACKAGE_HEADER *) (List + 1);
|
|
|
|
while (Package->Type != EFI_HII_PACKAGE_END) {
|
|
|
|
if (Package->Type == EFI_HII_PACKAGE_FORM) {
|
|
|
|
//
|
|
// Skip the package header
|
|
//
|
|
Offset = sizeof (EFI_HII_PACKAGE_HEADER);
|
|
while (Offset < Package->Length) {
|
|
OpCode = (EFI_IFR_OP_HEADER *)((UINT8 *) Package + Offset);
|
|
|
|
switch (OpCode->OpCode) {
|
|
case EFI_IFR_FORM_SET_OP:
|
|
FormSet = (EFI_IFR_FORM_SET *) OpCode;
|
|
ThunkContext->FormSetTitle = FormSet->FormSetTitle;
|
|
ThunkContext->FormSetHelp = FormSet->Help;
|
|
break;
|
|
|
|
case EFI_IFR_VARSTORE_OP:
|
|
//
|
|
// IFR built from Framework VFR only has UEFI Buffer Type Storage
|
|
//
|
|
VarStore = (EFI_IFR_VARSTORE *) OpCode;
|
|
IdMap = AllocateZeroPool (sizeof (QUESTION_ID_MAP));
|
|
ASSERT (IdMap != NULL);
|
|
|
|
IdMap->Signature = QUESTION_ID_MAP_SIGNATURE;
|
|
IdMap->VarStoreId = VarStore->VarStoreId;
|
|
IdMap->VarSize = VarStore->Size;
|
|
InitializeListHead (&IdMap->MapEntryListHead);
|
|
InsertTailList (&ThunkContext->QuestionIdMapListHead, &IdMap->Link);
|
|
break;
|
|
|
|
case EFI_IFR_NUMERIC_OP:
|
|
case EFI_IFR_CHECKBOX_OP:
|
|
case EFI_IFR_ONE_OF_OP:
|
|
case EFI_IFR_ORDERED_LIST_OP:
|
|
case EFI_IFR_STRING_OP:
|
|
//case EFI_IFR_PASSWORD_OP:
|
|
Question = (EFI_IFR_QUESTION_HEADER *)(OpCode + 1);
|
|
QuestionIdMapEntryListHead = GetMapEntryListHead (ThunkContext, Question->VarStoreId);
|
|
|
|
if (QuestionIdMapEntryListHead != NULL) {
|
|
//
|
|
// If the Question is using Buffer (EFI_IFR_VARSTORE_OP) type VarStore.
|
|
//
|
|
IdMapEntry = AllocateZeroPool (sizeof (QUESTION_ID_MAP_ENTRY));
|
|
ASSERT (IdMapEntry != NULL);
|
|
|
|
IdMapEntry->FwQId = Question->VarStoreInfo.VarOffset;
|
|
IdMapEntry->UefiQid = Question->QuestionId;
|
|
IdMapEntry->Signature = QUESTION_ID_MAP_ENTRY_SIGNATURE;
|
|
|
|
InsertTailList (QuestionIdMapEntryListHead, &IdMapEntry->Link);
|
|
}
|
|
|
|
if (OpCode->OpCode == EFI_IFR_ONE_OF_OP) {
|
|
OneOfOpcode = (EFI_IFR_ONE_OF *) OpCode;
|
|
OneOfType = OneOfOpcode->Flags & EFI_IFR_NUMERIC_SIZE;
|
|
}
|
|
|
|
break;
|
|
|
|
case EFI_IFR_GUID_OP:
|
|
OptionMap = (EFI_IFR_GUID_OPTIONKEY *) OpCode;
|
|
if (CompareMem (&OptionMap->Guid, &gFrameworkHiiCompatbilityGuid, sizeof (EFI_GUID)) == 0) {
|
|
if (OptionMap->ExtendOpCode == EFI_IFR_EXTEND_OP_OPTIONKEY) {
|
|
OneOfOptinMapEntryListHead = GetOneOfOptionMapEntryListHead (ThunkContext, OptionMap->QuestionId);
|
|
if (OneOfOptinMapEntryListHead == NULL) {
|
|
OneOfOptionMap = AllocateZeroPool (sizeof (ONE_OF_OPTION_MAP));
|
|
ASSERT (OneOfOptionMap != NULL);
|
|
|
|
OneOfOptionMap->Signature = ONE_OF_OPTION_MAP_SIGNATURE;
|
|
OneOfOptionMap->QuestionId = OptionMap->QuestionId;
|
|
|
|
//
|
|
// Make sure OneOfType is initialized.
|
|
//
|
|
ASSERT (OneOfType != (UINT8) -1);
|
|
OneOfOptionMap->ValueType = OneOfType;
|
|
InitializeListHead (&OneOfOptionMap->OneOfOptionMapEntryListHead);
|
|
OneOfOptinMapEntryListHead = &OneOfOptionMap->OneOfOptionMapEntryListHead;
|
|
InsertTailList (&ThunkContext->OneOfOptionMapListHead, &OneOfOptionMap->Link);
|
|
}
|
|
OneOfOptionMapEntry = AllocateZeroPool (sizeof (ONE_OF_OPTION_MAP_ENTRY));
|
|
ASSERT (OneOfOptionMapEntry != NULL);
|
|
|
|
OneOfOptionMapEntry->Signature = ONE_OF_OPTION_MAP_ENTRY_SIGNATURE;
|
|
OneOfOptionMapEntry->FwKey = OptionMap->KeyValue;
|
|
CopyMem (&OneOfOptionMapEntry->Value, &OptionMap->OptionValue, sizeof (EFI_IFR_TYPE_VALUE));
|
|
|
|
InsertTailList (OneOfOptinMapEntryListHead, &OneOfOptionMapEntry->Link);
|
|
}
|
|
} else if (CompareMem (&OptionMap->Guid, &gTianoHiiIfrGuid, sizeof (EFI_GUID)) == 0) {
|
|
Class = (EFI_IFR_GUID_CLASS *) OpCode;
|
|
|
|
switch (Class->ExtendOpCode) {
|
|
case EFI_IFR_EXTEND_OP_CLASS:
|
|
ThunkContext->FormSetClass = Class->Class;
|
|
break;
|
|
case EFI_IFR_EXTEND_OP_SUBCLASS:
|
|
SubClass = (EFI_IFR_GUID_SUBCLASS *) OpCode;
|
|
ThunkContext->FormSetSubClass = SubClass->SubClass;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
|
|
}
|
|
|
|
Offset += OpCode->Length;
|
|
}
|
|
//
|
|
// Only Form Package is in a Package List.
|
|
//
|
|
break;
|
|
}
|
|
|
|
Package = (EFI_HII_PACKAGE_HEADER *) (UINT8 *) Package + Package->Length;
|
|
}
|
|
|
|
FreePool (List);
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
|
|
LIST_ENTRY *
|
|
GetMapEntryListHead (
|
|
IN CONST HII_THUNK_CONTEXT *ThunkContext,
|
|
IN UINT16 VarStoreId
|
|
)
|
|
{
|
|
LIST_ENTRY *Link;
|
|
QUESTION_ID_MAP *Map;
|
|
|
|
Link = GetFirstNode (&ThunkContext->QuestionIdMapListHead);
|
|
|
|
while (!IsNull (&ThunkContext->QuestionIdMapListHead, Link)) {
|
|
Map = QUESTION_ID_MAP_FROM_LINK (Link);
|
|
if (VarStoreId == Map->VarStoreId) {
|
|
return &Map->MapEntryListHead;
|
|
}
|
|
Link = GetNextNode (&ThunkContext->QuestionIdMapListHead, Link);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
HII_THUNK_CONTEXT *
|
|
CreateThunkContext (
|
|
IN HII_THUNK_PRIVATE_DATA *Private,
|
|
IN UINTN StringPackageCount,
|
|
IN UINTN IfrPackageCount
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
HII_THUNK_CONTEXT *ThunkContext;
|
|
|
|
ThunkContext = AllocateZeroPool (sizeof (HII_THUNK_CONTEXT));
|
|
ASSERT (ThunkContext != NULL);
|
|
|
|
ThunkContext->Signature = HII_THUNK_CONTEXT_SIGNATURE;
|
|
ThunkContext->IfrPackageCount = IfrPackageCount;
|
|
ThunkContext->StringPackageCount = StringPackageCount;
|
|
Status = AllocateHiiHandle (&ThunkContext->FwHiiHandle);
|
|
if (EFI_ERROR (Status)) {
|
|
return NULL;
|
|
}
|
|
|
|
InitializeListHead (&ThunkContext->QuestionIdMapListHead);
|
|
InitializeListHead (&ThunkContext->OneOfOptionMapListHead);
|
|
|
|
|
|
return ThunkContext;
|
|
|
|
}
|
|
|
|
VOID
|
|
DestroyThunkContext (
|
|
IN HII_THUNK_CONTEXT *ThunkContext
|
|
)
|
|
{
|
|
ASSERT (ThunkContext != NULL);
|
|
|
|
FreeHiiHandle (ThunkContext->FwHiiHandle);
|
|
|
|
DestroyQuestionIdMap (&ThunkContext->QuestionIdMapListHead);
|
|
|
|
DestoryOneOfOptionMap (&ThunkContext->OneOfOptionMapListHead);
|
|
|
|
RemoveEntryList (&ThunkContext->Link);
|
|
|
|
FreePool (ThunkContext);
|
|
}
|
|
|
|
|
|
VOID
|
|
DestroyQuestionIdMap (
|
|
IN LIST_ENTRY *QuestionIdMapListHead
|
|
)
|
|
{
|
|
QUESTION_ID_MAP *IdMap;
|
|
QUESTION_ID_MAP_ENTRY *IdMapEntry;
|
|
LIST_ENTRY *Link;
|
|
LIST_ENTRY *Link2;
|
|
|
|
while (!IsListEmpty (QuestionIdMapListHead)) {
|
|
Link = GetFirstNode (QuestionIdMapListHead);
|
|
|
|
IdMap = QUESTION_ID_MAP_FROM_LINK (Link);
|
|
|
|
while (!IsListEmpty (&IdMap->MapEntryListHead)) {
|
|
Link2 = GetFirstNode (&IdMap->MapEntryListHead);
|
|
|
|
IdMapEntry = QUESTION_ID_MAP_ENTRY_FROM_LINK (Link2);
|
|
|
|
RemoveEntryList (Link2);
|
|
|
|
FreePool (IdMapEntry);
|
|
}
|
|
|
|
RemoveEntryList (Link);
|
|
FreePool (IdMap);
|
|
}
|
|
}
|
|
|
|
VOID
|
|
DestoryOneOfOptionMap (
|
|
IN LIST_ENTRY *OneOfOptionMapListHead
|
|
)
|
|
{
|
|
ONE_OF_OPTION_MAP *Map;
|
|
ONE_OF_OPTION_MAP_ENTRY *MapEntry;
|
|
LIST_ENTRY *Link;
|
|
LIST_ENTRY *Link2;
|
|
|
|
while (!IsListEmpty (OneOfOptionMapListHead)) {
|
|
Link = GetFirstNode (OneOfOptionMapListHead);
|
|
|
|
Map = ONE_OF_OPTION_MAP_FROM_LINK (Link);
|
|
|
|
while (!IsListEmpty (&Map->OneOfOptionMapEntryListHead)) {
|
|
Link2 = GetFirstNode (&Map->OneOfOptionMapEntryListHead);
|
|
|
|
MapEntry = ONE_OF_OPTION_MAP_ENTRY_FROM_LINK (Link2);
|
|
|
|
RemoveEntryList (Link2);
|
|
|
|
FreePool (MapEntry);
|
|
}
|
|
|
|
RemoveEntryList (Link);
|
|
FreePool (Map);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|