mirror of
				https://git.proxmox.com/git/mirror_edk2
				synced 2025-10-25 14:51:07 +00:00 
			
		
		
		
	 7d467158e0
			
		
	
	
		7d467158e0
		
	
	
	
	
		
			
			For a question, its question id can not be zero. This patch is to fix the issue that using zero as question id. Cc: Liming Gao <liming.gao@intel.com> Cc: Eric Dong <eric.dong@intel.com> Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Dandan Bi <dandan.bi@intel.com> Reviewed-by: Eric Dong <eric.dong@intel.com> Reviewed-by: Liming Gao <liming.gao@intel.com>
		
			
				
	
	
		
			675 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			675 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
| 
 | |
|   This library class defines a set of interfaces to customize Ui module
 | |
| 
 | |
| Copyright (c) 2016, Intel Corporation. All rights reserved.<BR>
 | |
| This program and the accompanying materials are licensed and made available under
 | |
| the terms and conditions of the BSD License that 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 <Uefi.h>
 | |
| 
 | |
| #include <Guid/MdeModuleHii.h>
 | |
| #include <Guid/GlobalVariable.h>
 | |
| 
 | |
| #include <Protocol/HiiConfigAccess.h>
 | |
| #include <Protocol/HiiString.h>
 | |
| 
 | |
| #include <Library/HiiLib.h>
 | |
| #include <Library/DebugLib.h>
 | |
| #include <Library/UefiLib.h>
 | |
| #include <Library/BaseMemoryLib.h>
 | |
| #include <Library/PcdLib.h>
 | |
| #include <Library/MemoryAllocationLib.h>
 | |
| #include <Library/UefiRuntimeServicesTableLib.h>
 | |
| #include <Library/UefiHiiServicesLib.h>
 | |
| #include <Library/DevicePathLib.h>
 | |
| #include <Library/UefiBootServicesTableLib.h>
 | |
| #include "FrontPageCustomizedUiSupport.h"
 | |
| 
 | |
| //
 | |
| // This is the VFR compiler generated header file which defines the
 | |
| // string identifiers.
 | |
| //
 | |
| #define PRINTABLE_LANGUAGE_NAME_STRING_ID     0x0001
 | |
| 
 | |
| #define UI_HII_DRIVER_LIST_SIZE               0x8
 | |
| 
 | |
| #define FRONT_PAGE_KEY_CONTINUE               0x1000
 | |
| #define FRONT_PAGE_KEY_RESET                  0x1001
 | |
| #define FRONT_PAGE_KEY_LANGUAGE               0x1002
 | |
| #define FRONT_PAGE_KEY_DRIVER                 0x2000
 | |
| 
 | |
| typedef struct {
 | |
|   EFI_STRING_ID   PromptId;
 | |
|   EFI_STRING_ID   HelpId;
 | |
|   EFI_STRING_ID   DevicePathId;
 | |
|   EFI_GUID        FormSetGuid;
 | |
|   BOOLEAN         EmptyLineAfter;
 | |
| } UI_HII_DRIVER_INSTANCE;
 | |
| 
 | |
| CHAR8                        *gLanguageString;
 | |
| EFI_STRING_ID                *gLanguageToken;
 | |
| UI_HII_DRIVER_INSTANCE       *gHiiDriverList;
 | |
| extern EFI_HII_HANDLE        gStringPackHandle;
 | |
| UINT8                        gCurrentLanguageIndex;
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Get next language from language code list (with separator ';').
 | |
| 
 | |
|   If LangCode is NULL, then ASSERT.
 | |
|   If Lang is NULL, then ASSERT.
 | |
| 
 | |
|   @param  LangCode    On input: point to first language in the list. On
 | |
|                                  output: point to next language in the list, or
 | |
|                                  NULL if no more language in the list.
 | |
|   @param  Lang           The first language in the list.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| GetNextLanguage (
 | |
|   IN OUT CHAR8      **LangCode,
 | |
|   OUT CHAR8         *Lang
 | |
|   )
 | |
| {
 | |
|   UINTN  Index;
 | |
|   CHAR8  *StringPtr;
 | |
| 
 | |
|   ASSERT (LangCode != NULL);
 | |
|   ASSERT (*LangCode != NULL);
 | |
|   ASSERT (Lang != NULL);
 | |
| 
 | |
|   Index = 0;
 | |
|   StringPtr = *LangCode;
 | |
|   while (StringPtr[Index] != 0 && StringPtr[Index] != ';') {
 | |
|     Index++;
 | |
|   }
 | |
| 
 | |
|   CopyMem (Lang, StringPtr, Index);
 | |
|   Lang[Index] = 0;
 | |
| 
 | |
|   if (StringPtr[Index] == ';') {
 | |
|     Index++;
 | |
|   }
 | |
|   *LangCode = StringPtr + Index;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   This function processes the language changes in configuration.
 | |
| 
 | |
|   @param Value           A pointer to the data being sent to the original exporting driver.
 | |
| 
 | |
| 
 | |
|   @retval  TRUE          The callback successfully handled the action.
 | |
|   @retval  FALSE         The callback not supported in this handler.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| LanguageChangeHandler (
 | |
|   IN  EFI_IFR_TYPE_VALUE                     *Value
 | |
|   )
 | |
| {
 | |
|   CHAR8                         *LangCode;
 | |
|   CHAR8                         *Lang;
 | |
|   UINTN                         Index;
 | |
|   EFI_STATUS                    Status;
 | |
| 
 | |
|   //
 | |
|   // Allocate working buffer for RFC 4646 language in supported LanguageString.
 | |
|   //
 | |
|   Lang = AllocatePool (AsciiStrSize (gLanguageString));
 | |
|   ASSERT (Lang != NULL);
 | |
| 
 | |
|   Index = 0;
 | |
|   LangCode = gLanguageString;
 | |
|   while (*LangCode != 0) {
 | |
|     GetNextLanguage (&LangCode, Lang);
 | |
| 
 | |
|     if (Index == Value->u8) {
 | |
|       gCurrentLanguageIndex = Value->u8;
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     Index++;
 | |
|   }
 | |
| 
 | |
|   if (Index == Value->u8) {
 | |
|     Status = gRT->SetVariable (
 | |
|                     L"PlatformLang",
 | |
|                     &gEfiGlobalVariableGuid,
 | |
|                     EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,
 | |
|                     AsciiStrSize (Lang),
 | |
|                     Lang
 | |
|                     );
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       FreePool (Lang);
 | |
|       return EFI_DEVICE_ERROR;
 | |
|     }
 | |
|   } else {
 | |
|     ASSERT (FALSE);
 | |
|   }
 | |
|   FreePool (Lang);
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   This function processes the results of changes in configuration.
 | |
| 
 | |
| 
 | |
|   @param HiiHandle       Points to the hii handle for this formset.
 | |
|   @param Action          Specifies the type of action taken by the browser.
 | |
|   @param QuestionId      A unique value which is sent to the original exporting driver
 | |
|                          so that it can identify the type of data to expect.
 | |
|   @param Type            The type of value for the question.
 | |
|   @param Value           A pointer to the data being sent to the original exporting driver.
 | |
|   @param ActionRequest   On return, points to the action requested by the callback function.
 | |
|   @param Status          Return the handle status.
 | |
| 
 | |
|   @retval  TRUE          The callback successfully handled the action.
 | |
|   @retval  FALSE         The callback not supported in this handler.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| UiSupportLibCallbackHandler (
 | |
|   IN  EFI_HII_HANDLE                         HiiHandle,
 | |
|   IN  EFI_BROWSER_ACTION                     Action,
 | |
|   IN  EFI_QUESTION_ID                        QuestionId,
 | |
|   IN  UINT8                                  Type,
 | |
|   IN  EFI_IFR_TYPE_VALUE                     *Value,
 | |
|   OUT EFI_BROWSER_ACTION_REQUEST             *ActionRequest,
 | |
|   OUT EFI_STATUS                             *Status
 | |
|   )
 | |
| {
 | |
|   if (QuestionId != FRONT_PAGE_KEY_CONTINUE &&
 | |
|       QuestionId != FRONT_PAGE_KEY_RESET &&
 | |
|       QuestionId != FRONT_PAGE_KEY_LANGUAGE) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   if (Action == EFI_BROWSER_ACTION_RETRIEVE) {
 | |
|     if (QuestionId == FRONT_PAGE_KEY_LANGUAGE) {
 | |
|       Value->u8 = gCurrentLanguageIndex;
 | |
|       *Status = EFI_SUCCESS;
 | |
|     } else {
 | |
|       *Status = EFI_UNSUPPORTED;
 | |
|     }
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   if (Action != EFI_BROWSER_ACTION_CHANGED) {
 | |
|     //
 | |
|     // Do nothing for other UEFI Action. Only do call back when data is changed.
 | |
|     //
 | |
|     *Status = EFI_UNSUPPORTED;
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   if (Action == EFI_BROWSER_ACTION_CHANGED) {
 | |
|     if ((Value == NULL) || (ActionRequest == NULL)) {
 | |
|       *Status = EFI_INVALID_PARAMETER;
 | |
|       return TRUE;
 | |
|     }
 | |
| 
 | |
|     *Status = EFI_SUCCESS;
 | |
|     switch (QuestionId) {
 | |
|     case FRONT_PAGE_KEY_CONTINUE:
 | |
|       //
 | |
|       // This is the continue - clear the screen and return an error to get out of FrontPage loop
 | |
|       //
 | |
|       *ActionRequest = EFI_BROWSER_ACTION_REQUEST_EXIT;
 | |
|       break;
 | |
| 
 | |
|     case FRONT_PAGE_KEY_LANGUAGE:
 | |
|       *Status = LanguageChangeHandler(Value);
 | |
|       break;
 | |
| 
 | |
|     case FRONT_PAGE_KEY_RESET:
 | |
|       //
 | |
|       // Reset
 | |
|       //
 | |
|       gRT->ResetSystem (EfiResetCold, EFI_SUCCESS, 0, NULL);
 | |
|       *Status = EFI_UNSUPPORTED;
 | |
| 
 | |
|     default:
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Create Select language menu in the front page with oneof opcode.
 | |
| 
 | |
|   @param[in]    HiiHandle           The hii handle for the Uiapp driver.
 | |
|   @param[in]    StartOpCodeHandle   The opcode handle to save the new opcode.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| UiCreateLanguageMenu (
 | |
|   IN EFI_HII_HANDLE              HiiHandle,
 | |
|   IN VOID                        *StartOpCodeHandle
 | |
|   )
 | |
| {
 | |
|   CHAR8                       *LangCode;
 | |
|   CHAR8                       *Lang;
 | |
|   UINTN                       LangSize;
 | |
|   CHAR8                       *CurrentLang;
 | |
|   UINTN                       OptionCount;
 | |
|   CHAR16                      *StringBuffer;
 | |
|   VOID                        *OptionsOpCodeHandle;
 | |
|   UINTN                       StringSize;
 | |
|   EFI_STATUS                  Status;
 | |
|   EFI_HII_STRING_PROTOCOL     *HiiString;
 | |
| 
 | |
|   Lang         = NULL;
 | |
|   StringBuffer = NULL;
 | |
| 
 | |
|   //
 | |
|   // Init OpCode Handle and Allocate space for creation of UpdateData Buffer
 | |
|   //
 | |
|   OptionsOpCodeHandle = HiiAllocateOpCodeHandle ();
 | |
|   ASSERT (OptionsOpCodeHandle != NULL);
 | |
| 
 | |
|   GetEfiGlobalVariable2 (L"PlatformLang", (VOID**)&CurrentLang, NULL);
 | |
| 
 | |
|   //
 | |
|   // Get Support language list from variable.
 | |
|   //
 | |
|   GetEfiGlobalVariable2 (L"PlatformLangCodes", (VOID**)&gLanguageString, NULL);
 | |
|   if (gLanguageString == NULL) {
 | |
|     gLanguageString = AllocateCopyPool (
 | |
|                                AsciiStrSize ((CHAR8 *) PcdGetPtr (PcdUefiVariableDefaultPlatformLangCodes)),
 | |
|                                (CHAR8 *) PcdGetPtr (PcdUefiVariableDefaultPlatformLangCodes)
 | |
|                                );
 | |
|     ASSERT (gLanguageString != NULL);
 | |
|   }
 | |
| 
 | |
|   if (gLanguageToken == NULL) {
 | |
|     //
 | |
|     // Count the language list number.
 | |
|     //
 | |
|     LangCode = gLanguageString;
 | |
|     Lang = AllocatePool (AsciiStrSize (gLanguageString));
 | |
|     ASSERT (Lang != NULL);
 | |
| 
 | |
|     OptionCount = 0;
 | |
|     while (*LangCode != 0) {
 | |
|       GetNextLanguage (&LangCode, Lang);
 | |
|       OptionCount ++;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     // Allocate extra 1 as the end tag.
 | |
|     //
 | |
|     gLanguageToken = AllocateZeroPool ((OptionCount + 1) * sizeof (EFI_STRING_ID));
 | |
|     ASSERT (gLanguageToken != NULL);
 | |
| 
 | |
|     Status = gBS->LocateProtocol (&gEfiHiiStringProtocolGuid, NULL, (VOID **) &HiiString);
 | |
|     ASSERT_EFI_ERROR (Status);
 | |
| 
 | |
|     LangCode     = gLanguageString;
 | |
|     OptionCount  = 0;
 | |
|     while (*LangCode != 0) {
 | |
|       GetNextLanguage (&LangCode, Lang);
 | |
| 
 | |
|       StringSize = 0;
 | |
|       Status = HiiString->GetString (HiiString, Lang, HiiHandle, PRINTABLE_LANGUAGE_NAME_STRING_ID, StringBuffer, &StringSize, NULL);
 | |
|       if (Status == EFI_BUFFER_TOO_SMALL) {
 | |
|         StringBuffer = AllocateZeroPool (StringSize);
 | |
|         ASSERT (StringBuffer != NULL);
 | |
|         Status = HiiString->GetString (HiiString, Lang, HiiHandle, PRINTABLE_LANGUAGE_NAME_STRING_ID, StringBuffer, &StringSize, NULL);
 | |
|         ASSERT_EFI_ERROR (Status);
 | |
|       }
 | |
| 
 | |
|       if (EFI_ERROR (Status)) {
 | |
|         LangSize = AsciiStrSize (Lang);
 | |
|         StringBuffer = AllocatePool (LangSize * sizeof (CHAR16));
 | |
|         ASSERT (StringBuffer != NULL);
 | |
|         AsciiStrToUnicodeStrS (Lang, StringBuffer, LangSize);
 | |
|       }
 | |
| 
 | |
|       ASSERT (StringBuffer != NULL);
 | |
|       gLanguageToken[OptionCount] = HiiSetString (HiiHandle, 0, StringBuffer, NULL);
 | |
|       FreePool (StringBuffer);
 | |
| 
 | |
|       OptionCount++;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   ASSERT (gLanguageToken != NULL);
 | |
|   LangCode = gLanguageString;
 | |
|   OptionCount = 0;
 | |
|   if (Lang == NULL) {
 | |
|     Lang = AllocatePool (AsciiStrSize (gLanguageString));
 | |
|     ASSERT (Lang != NULL);
 | |
|   }
 | |
|   while (*LangCode != 0) {
 | |
|     GetNextLanguage (&LangCode, Lang);
 | |
| 
 | |
|     if (CurrentLang != NULL && AsciiStrCmp (Lang, CurrentLang) == 0) {
 | |
|       HiiCreateOneOfOptionOpCode (
 | |
|         OptionsOpCodeHandle,
 | |
|         gLanguageToken[OptionCount],
 | |
|         EFI_IFR_OPTION_DEFAULT,
 | |
|         EFI_IFR_NUMERIC_SIZE_1,
 | |
|         (UINT8) OptionCount
 | |
|         );
 | |
|       gCurrentLanguageIndex = (UINT8) OptionCount;
 | |
|     } else {
 | |
|       HiiCreateOneOfOptionOpCode (
 | |
|         OptionsOpCodeHandle,
 | |
|         gLanguageToken[OptionCount],
 | |
|         0,
 | |
|         EFI_IFR_NUMERIC_SIZE_1,
 | |
|         (UINT8) OptionCount
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     OptionCount++;
 | |
|   }
 | |
| 
 | |
|   if (CurrentLang != NULL) {
 | |
|     FreePool (CurrentLang);
 | |
|   }
 | |
|   FreePool (Lang);
 | |
| 
 | |
|   HiiCreateOneOfOpCode (
 | |
|     StartOpCodeHandle,
 | |
|     FRONT_PAGE_KEY_LANGUAGE,
 | |
|     0,
 | |
|     0,
 | |
|     STRING_TOKEN (STR_LANGUAGE_SELECT),
 | |
|     STRING_TOKEN (STR_LANGUAGE_SELECT_HELP),
 | |
|     EFI_IFR_FLAG_CALLBACK,
 | |
|     EFI_IFR_NUMERIC_SIZE_1,
 | |
|     OptionsOpCodeHandle,
 | |
|     NULL
 | |
|     );
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Create continue menu in the front page.
 | |
| 
 | |
|   @param[in]    HiiHandle           The hii handle for the Uiapp driver.
 | |
|   @param[in]    StartOpCodeHandle   The opcode handle to save the new opcode.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| UiCreateContinueMenu (
 | |
|   IN EFI_HII_HANDLE              HiiHandle,
 | |
|   IN VOID                        *StartOpCodeHandle
 | |
|   )
 | |
| {
 | |
|   HiiCreateActionOpCode (
 | |
|     StartOpCodeHandle,
 | |
|     FRONT_PAGE_KEY_CONTINUE,
 | |
|     STRING_TOKEN (STR_CONTINUE_PROMPT),
 | |
|     STRING_TOKEN (STR_CONTINUE_PROMPT),
 | |
|     EFI_IFR_FLAG_CALLBACK,
 | |
|     0
 | |
|     );
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Create empty line menu in the front page.
 | |
| 
 | |
|   @param    HiiHandle           The hii handle for the Uiapp driver.
 | |
|   @param    StartOpCodeHandle   The opcode handle to save the new opcode.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| UiCreateEmptyLine (
 | |
|   IN EFI_HII_HANDLE              HiiHandle,
 | |
|   IN VOID                        *StartOpCodeHandle
 | |
|   )
 | |
| {
 | |
|   HiiCreateSubTitleOpCode (StartOpCodeHandle, STRING_TOKEN (STR_NULL_STRING), 0, 0, 0);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Create Reset menu in the front page.
 | |
| 
 | |
|   @param[in]    HiiHandle           The hii handle for the Uiapp driver.
 | |
|   @param[in]    StartOpCodeHandle   The opcode handle to save the new opcode.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| UiCreateResetMenu (
 | |
|   IN EFI_HII_HANDLE              HiiHandle,
 | |
|   IN VOID                        *StartOpCodeHandle
 | |
|   )
 | |
| {
 | |
|   HiiCreateActionOpCode (
 | |
|     StartOpCodeHandle,
 | |
|     FRONT_PAGE_KEY_RESET,
 | |
|     STRING_TOKEN (STR_RESET_STRING),
 | |
|     STRING_TOKEN (STR_RESET_STRING),
 | |
|     EFI_IFR_FLAG_CALLBACK,
 | |
|     0
 | |
|     );
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Extract device path for given HII handle and class guid.
 | |
| 
 | |
|   @param Handle          The HII handle.
 | |
| 
 | |
|   @retval  NULL          Fail to get the device path string.
 | |
|   @return  PathString    Get the device path string.
 | |
| 
 | |
| **/
 | |
| CHAR16 *
 | |
| ExtractDevicePathFromHiiHandle (
 | |
|   IN      EFI_HII_HANDLE      Handle
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                       Status;
 | |
|   EFI_HANDLE                       DriverHandle;
 | |
| 
 | |
|   ASSERT (Handle != NULL);
 | |
| 
 | |
|   if (Handle == NULL) {
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   Status = gHiiDatabase->GetPackageListHandle (gHiiDatabase, Handle, &DriverHandle);
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   return ConvertDevicePathToText(DevicePathFromHandle (DriverHandle), FALSE, FALSE);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Check whether this driver need to be shown in the front page.
 | |
| 
 | |
|   @param    HiiHandle           The hii handle for the driver.
 | |
|   @param    Guid                The special guid for the driver which is the target.
 | |
|   @param    PromptId            Return the prompt string id.
 | |
|   @param    HelpId              Return the help string id.
 | |
|   @param    FormsetGuid         Return the formset guid info.
 | |
| 
 | |
|   @retval   EFI_SUCCESS         Search the driver success
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| RequiredDriver (
 | |
|   IN  EFI_HII_HANDLE              HiiHandle,
 | |
|   IN  EFI_GUID                    *Guid,
 | |
|   OUT EFI_STRING_ID               *PromptId,
 | |
|   OUT EFI_STRING_ID               *HelpId,
 | |
|   OUT VOID                        *FormsetGuid
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                  Status;
 | |
|   UINT8                       ClassGuidNum;
 | |
|   EFI_GUID                    *ClassGuid;
 | |
|   EFI_IFR_FORM_SET            *Buffer;
 | |
|   UINTN                       BufferSize;
 | |
|   UINT8                       *Ptr;
 | |
|   UINTN                       TempSize;
 | |
|   BOOLEAN                     RetVal;
 | |
| 
 | |
|   Status = HiiGetFormSetFromHiiHandle(HiiHandle, &Buffer,&BufferSize);
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   RetVal = FALSE;
 | |
|   TempSize = 0;
 | |
|   Ptr = (UINT8 *) Buffer;
 | |
|   while(TempSize < BufferSize)  {
 | |
|     TempSize += ((EFI_IFR_OP_HEADER *) Ptr)->Length;
 | |
| 
 | |
|     if (((EFI_IFR_OP_HEADER *) Ptr)->Length <= OFFSET_OF (EFI_IFR_FORM_SET, Flags)){
 | |
|       Ptr += ((EFI_IFR_OP_HEADER *) Ptr)->Length;
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     ClassGuidNum = (UINT8) (((EFI_IFR_FORM_SET *)Ptr)->Flags & 0x3);
 | |
|     ClassGuid = (EFI_GUID *) (VOID *)(Ptr + sizeof (EFI_IFR_FORM_SET));
 | |
|     while (ClassGuidNum-- > 0) {
 | |
|       if (!CompareGuid (Guid, ClassGuid)){
 | |
|         ClassGuid ++;
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       *PromptId = ((EFI_IFR_FORM_SET *)Ptr)->FormSetTitle;
 | |
|       *HelpId = ((EFI_IFR_FORM_SET *)Ptr)->Help;
 | |
|       CopyMem (FormsetGuid, &((EFI_IFR_FORM_SET *) Ptr)->Guid, sizeof (EFI_GUID));
 | |
|       RetVal = TRUE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   FreePool (Buffer);
 | |
| 
 | |
|   return RetVal;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Search the drivers in the system which need to show in the front page
 | |
|   and insert the menu to the front page.
 | |
| 
 | |
|   @param    HiiHandle           The hii handle for the Uiapp driver.
 | |
|   @param    ClassGuid           The class guid for the driver which is the target.
 | |
|   @param    SpecialHandlerFn    The pointer to the specail handler function, if any.
 | |
|   @param    StartOpCodeHandle   The opcode handle to save the new opcode.
 | |
| 
 | |
|   @retval   EFI_SUCCESS         Search the driver success
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| UiListThirdPartyDrivers (
 | |
|   IN EFI_HII_HANDLE              HiiHandle,
 | |
|   IN EFI_GUID                    *ClassGuid,
 | |
|   IN DRIVER_SPECIAL_HANDLER      SpecialHandlerFn,
 | |
|   IN VOID                        *StartOpCodeHandle
 | |
|   )
 | |
| {
 | |
|   UINTN                       Index;
 | |
|   EFI_STRING                  String;
 | |
|   EFI_STRING_ID               Token;
 | |
|   EFI_STRING_ID               TokenHelp;
 | |
|   EFI_HII_HANDLE              *HiiHandles;
 | |
|   CHAR16                      *DevicePathStr;
 | |
|   UINTN                       Count;
 | |
|   UINTN                       CurrentSize;
 | |
|   UI_HII_DRIVER_INSTANCE      *DriverListPtr;
 | |
|   EFI_STRING                  NewName;
 | |
|   BOOLEAN                     EmptyLineAfter;
 | |
| 
 | |
|   if (gHiiDriverList != NULL) {
 | |
|     FreePool (gHiiDriverList);
 | |
|   }
 | |
| 
 | |
|   HiiHandles = HiiGetHiiHandles (NULL);
 | |
|   ASSERT (HiiHandles != NULL);
 | |
| 
 | |
|   gHiiDriverList = AllocateZeroPool (UI_HII_DRIVER_LIST_SIZE * sizeof (UI_HII_DRIVER_INSTANCE));
 | |
|   ASSERT (gHiiDriverList != NULL);
 | |
|   DriverListPtr = gHiiDriverList;
 | |
|   CurrentSize = UI_HII_DRIVER_LIST_SIZE;
 | |
| 
 | |
|   for (Index = 0, Count = 0; HiiHandles[Index] != NULL; Index++) {
 | |
|     if (!RequiredDriver (HiiHandles[Index], ClassGuid, &Token, &TokenHelp, &gHiiDriverList[Count].FormSetGuid)) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     String = HiiGetString (HiiHandles[Index], Token, NULL);
 | |
|     if (String == NULL) {
 | |
|       String = HiiGetString (gStringPackHandle, STRING_TOKEN (STR_MISSING_STRING), NULL);
 | |
|       ASSERT (String != NULL);
 | |
|     } else if (SpecialHandlerFn != NULL) {
 | |
|       //
 | |
|       // Check whether need to rename the driver name.
 | |
|       //
 | |
|       EmptyLineAfter = FALSE;
 | |
|       if (SpecialHandlerFn (String, &NewName, &EmptyLineAfter)) {
 | |
|         FreePool (String);
 | |
|         String = NewName;
 | |
|         DriverListPtr[Count].EmptyLineAfter = EmptyLineAfter;
 | |
|       }
 | |
|     }
 | |
|     DriverListPtr[Count].PromptId = HiiSetString (HiiHandle, 0, String, NULL);
 | |
|     FreePool (String);
 | |
| 
 | |
|     String = HiiGetString (HiiHandles[Index], TokenHelp, NULL);
 | |
|     if (String == NULL) {
 | |
|       String = HiiGetString (gStringPackHandle, STRING_TOKEN (STR_MISSING_STRING), NULL);
 | |
|       ASSERT (String != NULL);
 | |
|     }
 | |
|     DriverListPtr[Count].HelpId = HiiSetString (HiiHandle, 0, String, NULL);
 | |
|     FreePool (String);
 | |
| 
 | |
|     DevicePathStr = ExtractDevicePathFromHiiHandle(HiiHandles[Index]);
 | |
|     if (DevicePathStr != NULL){
 | |
|       DriverListPtr[Count].DevicePathId = HiiSetString (HiiHandle, 0, DevicePathStr, NULL);
 | |
|       FreePool (DevicePathStr);
 | |
|     } else {
 | |
|       DriverListPtr[Count].DevicePathId = 0;
 | |
|     }
 | |
| 
 | |
|     Count++;
 | |
|     if (Count >= CurrentSize) {
 | |
|       DriverListPtr = AllocateCopyPool ((Count + UI_HII_DRIVER_LIST_SIZE) * sizeof (UI_HII_DRIVER_INSTANCE), gHiiDriverList);
 | |
|       ASSERT (DriverListPtr != NULL);
 | |
|       FreePool (gHiiDriverList);
 | |
|       gHiiDriverList = DriverListPtr;
 | |
|       CurrentSize += UI_HII_DRIVER_LIST_SIZE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   FreePool (HiiHandles);
 | |
| 
 | |
|   Index = 0;
 | |
|   while (gHiiDriverList[Index].PromptId != 0) {
 | |
|     HiiCreateGotoExOpCode (
 | |
|       StartOpCodeHandle,
 | |
|       0,
 | |
|       gHiiDriverList[Index].PromptId,
 | |
|       gHiiDriverList[Index].HelpId,
 | |
|       0,
 | |
|       (EFI_QUESTION_ID) (Index + FRONT_PAGE_KEY_DRIVER),
 | |
|       0,
 | |
|       &gHiiDriverList[Index].FormSetGuid,
 | |
|       gHiiDriverList[Index].DevicePathId
 | |
|     );
 | |
| 
 | |
|     if (gHiiDriverList[Index].EmptyLineAfter) {
 | |
|       UiCreateEmptyLine (HiiHandle, StartOpCodeHandle);
 | |
|     }
 | |
| 
 | |
|     Index ++;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 |