2015-11-23 10:34:55 +01:00
|
|
|
/** @file
|
|
|
|
Variable operation that will be used by bootmaint
|
|
|
|
|
2018-06-27 15:08:52 +02:00
|
|
|
Copyright (c) 2004 - 2018, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:05:13 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "BootMaintenanceManager.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
Delete Boot Option that represent a Deleted state in BootOptionMenu.
|
|
|
|
|
2018-06-27 15:08:52 +02:00
|
|
|
@retval EFI_SUCCESS If all boot load option EFI Variables corresponding to
|
2015-11-23 10:34:55 +01:00
|
|
|
BM_LOAD_CONTEXT marked for deletion is deleted.
|
|
|
|
@retval EFI_NOT_FOUND If can not find the boot option want to be deleted.
|
2018-06-27 15:08:52 +02:00
|
|
|
@return Others If failed to update the "BootOrder" variable after deletion.
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_DelBootOption (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BM_MENU_ENTRY *NewMenuEntry;
|
|
|
|
BM_LOAD_CONTEXT *NewLoadContext;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN Index;
|
|
|
|
UINTN Index2;
|
|
|
|
|
|
|
|
Index2 = 0;
|
|
|
|
for (Index = 0; Index < BootOptionMenu.MenuNumber; Index++) {
|
|
|
|
NewMenuEntry = BOpt_GetMenuEntry (&BootOptionMenu, (Index - Index2));
|
|
|
|
if (NULL == NewMenuEntry) {
|
|
|
|
return EFI_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
NewLoadContext = (BM_LOAD_CONTEXT *)NewMenuEntry->VariableContext;
|
|
|
|
if (!NewLoadContext->Deleted) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-02-27 06:33:06 +01:00
|
|
|
Status = EfiBootManagerDeleteLoadOptionVariable (NewMenuEntry->OptionNumber, LoadOptionTypeBoot);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2021-12-05 23:54:02 +01:00
|
|
|
|
2015-11-23 10:34:55 +01:00
|
|
|
Index2++;
|
|
|
|
//
|
|
|
|
// If current Load Option is the same as BootNext,
|
|
|
|
// must delete BootNext in order to make sure
|
|
|
|
// there will be no panic on next boot
|
|
|
|
//
|
|
|
|
if (NewLoadContext->IsBootNext) {
|
|
|
|
EfiLibDeleteVariable (L"BootNext", &gEfiGlobalVariableGuid);
|
|
|
|
}
|
|
|
|
|
|
|
|
RemoveEntryList (&NewMenuEntry->Link);
|
|
|
|
BOpt_DestroyMenuEntry (NewMenuEntry);
|
|
|
|
NewMenuEntry = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
BootOptionMenu.MenuNumber -= Index2;
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-02-27 06:33:06 +01:00
|
|
|
Delete Load Option that represent a Deleted state in DriverOptionMenu.
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
@retval EFI_SUCCESS Load Option is successfully updated.
|
|
|
|
@retval EFI_NOT_FOUND Fail to find the driver option want to be deleted.
|
|
|
|
@return Other value than EFI_SUCCESS if failed to update "Driver Order" EFI
|
|
|
|
Variable.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_DelDriverOption (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BM_MENU_ENTRY *NewMenuEntry;
|
|
|
|
BM_LOAD_CONTEXT *NewLoadContext;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN Index;
|
|
|
|
UINTN Index2;
|
|
|
|
|
|
|
|
Index2 = 0;
|
|
|
|
for (Index = 0; Index < DriverOptionMenu.MenuNumber; Index++) {
|
|
|
|
NewMenuEntry = BOpt_GetMenuEntry (&DriverOptionMenu, (Index - Index2));
|
|
|
|
if (NULL == NewMenuEntry) {
|
|
|
|
return EFI_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
NewLoadContext = (BM_LOAD_CONTEXT *)NewMenuEntry->VariableContext;
|
|
|
|
if (!NewLoadContext->Deleted) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-12-05 23:54:02 +01:00
|
|
|
|
2017-02-27 06:33:06 +01:00
|
|
|
Status = EfiBootManagerDeleteLoadOptionVariable (NewMenuEntry->OptionNumber, LoadOptionTypeDriver);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
Index2++;
|
|
|
|
|
|
|
|
RemoveEntryList (&NewMenuEntry->Link);
|
|
|
|
BOpt_DestroyMenuEntry (NewMenuEntry);
|
|
|
|
NewMenuEntry = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
DriverOptionMenu.MenuNumber -= Index2;
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function delete and build multi-instance device path for
|
|
|
|
specified type of console device.
|
|
|
|
|
|
|
|
This function clear the EFI variable defined by ConsoleName and
|
|
|
|
gEfiGlobalVariableGuid. It then build the multi-instance device
|
2018-06-27 15:08:52 +02:00
|
|
|
path by appending the device path of the Console (In/Out/Err) instance
|
2015-11-23 10:34:55 +01:00
|
|
|
in ConsoleMenu. Then it scan all corresponding console device by
|
|
|
|
scanning Terminal (built from device supporting Serial I/O instances)
|
|
|
|
devices in TerminalMenu. At last, it save a EFI variable specifed
|
|
|
|
by ConsoleName and gEfiGlobalVariableGuid.
|
|
|
|
|
|
|
|
@param ConsoleName The name for the console device type. They are
|
|
|
|
usually "ConIn", "ConOut" and "ErrOut".
|
|
|
|
@param ConsoleMenu The console memu which is a list of console devices.
|
|
|
|
@param UpdatePageId The flag specifying which type of console device
|
|
|
|
to be processed.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The function complete successfully.
|
|
|
|
@return The EFI variable can not be saved. See gRT->SetVariable for detail return information.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateConsoleOption (
|
|
|
|
IN UINT16 *ConsoleName,
|
|
|
|
IN BM_MENU_OPTION *ConsoleMenu,
|
|
|
|
IN UINT16 UpdatePageId
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *ConDevicePath;
|
|
|
|
BM_MENU_ENTRY *NewMenuEntry;
|
|
|
|
BM_CONSOLE_CONTEXT *NewConsoleContext;
|
|
|
|
BM_TERMINAL_CONTEXT *NewTerminalContext;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
VENDOR_DEVICE_PATH Vendor;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *TerminalDevicePath;
|
|
|
|
UINTN Index;
|
|
|
|
|
|
|
|
GetEfiGlobalVariable2 (ConsoleName, (VOID **)&ConDevicePath, NULL);
|
|
|
|
if (ConDevicePath != NULL) {
|
|
|
|
EfiLibDeleteVariable (ConsoleName, &gEfiGlobalVariableGuid);
|
|
|
|
FreePool (ConDevicePath);
|
|
|
|
ConDevicePath = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// First add all console input device from console input menu
|
|
|
|
//
|
|
|
|
for (Index = 0; Index < ConsoleMenu->MenuNumber; Index++) {
|
|
|
|
NewMenuEntry = BOpt_GetMenuEntry (ConsoleMenu, Index);
|
|
|
|
|
|
|
|
NewConsoleContext = (BM_CONSOLE_CONTEXT *)NewMenuEntry->VariableContext;
|
|
|
|
if (NewConsoleContext->IsActive) {
|
|
|
|
ConDevicePath = AppendDevicePathInstance (
|
|
|
|
ConDevicePath,
|
|
|
|
NewConsoleContext->DevicePath
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Index = 0; Index < TerminalMenu.MenuNumber; Index++) {
|
|
|
|
NewMenuEntry = BOpt_GetMenuEntry (&TerminalMenu, Index);
|
|
|
|
|
|
|
|
NewTerminalContext = (BM_TERMINAL_CONTEXT *)NewMenuEntry->VariableContext;
|
|
|
|
if (((NewTerminalContext->IsConIn != 0) && (UpdatePageId == FORM_CON_IN_ID)) ||
|
|
|
|
((NewTerminalContext->IsConOut != 0) && (UpdatePageId == FORM_CON_OUT_ID)) ||
|
|
|
|
((NewTerminalContext->IsStdErr != 0) && (UpdatePageId == FORM_CON_ERR_ID))
|
|
|
|
)
|
|
|
|
{
|
|
|
|
Vendor.Header.Type = MESSAGING_DEVICE_PATH;
|
|
|
|
Vendor.Header.SubType = MSG_VENDOR_DP;
|
2018-06-27 15:08:52 +02:00
|
|
|
|
2016-10-26 19:00:44 +02:00
|
|
|
ASSERT (NewTerminalContext->TerminalType < (ARRAY_SIZE (TerminalTypeGuid)));
|
2015-11-23 10:34:55 +01:00
|
|
|
CopyMem (
|
|
|
|
&Vendor.Guid,
|
|
|
|
&TerminalTypeGuid[NewTerminalContext->TerminalType],
|
|
|
|
sizeof (EFI_GUID)
|
|
|
|
);
|
|
|
|
SetDevicePathNodeLength (&Vendor.Header, sizeof (VENDOR_DEVICE_PATH));
|
|
|
|
TerminalDevicePath = AppendDevicePathNode (
|
|
|
|
NewTerminalContext->DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&Vendor
|
|
|
|
);
|
|
|
|
ASSERT (TerminalDevicePath != NULL);
|
|
|
|
ChangeTerminalDevicePath (TerminalDevicePath, TRUE);
|
|
|
|
ConDevicePath = AppendDevicePathInstance (
|
|
|
|
ConDevicePath,
|
|
|
|
TerminalDevicePath
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ConDevicePath != NULL) {
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
ConsoleName,
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
VAR_FLAG,
|
|
|
|
GetDevicePathSize (ConDevicePath),
|
|
|
|
ConDevicePath
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function delete and build multi-instance device path ConIn
|
|
|
|
console device.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The function complete successfully.
|
|
|
|
@return The EFI variable can not be saved. See gRT->SetVariable for detail return information.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateConsoleInpOption (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return Var_UpdateConsoleOption (L"ConIn", &ConsoleInpMenu, FORM_CON_IN_ID);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function delete and build multi-instance device path ConOut
|
|
|
|
console device.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The function complete successfully.
|
|
|
|
@return The EFI variable can not be saved. See gRT->SetVariable for detail return information.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateConsoleOutOption (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return Var_UpdateConsoleOption (L"ConOut", &ConsoleOutMenu, FORM_CON_OUT_ID);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function delete and build multi-instance device path ErrOut
|
|
|
|
console device.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The function complete successfully.
|
2018-06-27 15:08:52 +02:00
|
|
|
@return The EFI variable can not be saved. See gRT->SetVariable for detail return information.
|
2015-11-23 10:34:55 +01:00
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateErrorOutOption (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return Var_UpdateConsoleOption (L"ErrOut", &ConsoleErrMenu, FORM_CON_ERR_ID);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-06-27 15:08:52 +02:00
|
|
|
This function create a currently loaded Drive Option from
|
|
|
|
the BMM. It then appends this Driver Option to the end of
|
2015-11-23 10:34:55 +01:00
|
|
|
the "DriverOrder" list. It append this Driver Opotion to the end
|
|
|
|
of DriverOptionMenu.
|
|
|
|
|
|
|
|
@param CallbackData The BMM context data.
|
|
|
|
@param HiiHandle The HII handle associated with the BMM formset.
|
|
|
|
@param DescriptionData The description of this driver option.
|
|
|
|
@param OptionalData The optional load option.
|
|
|
|
@param ForceReconnect If to force reconnect.
|
|
|
|
|
2016-10-17 08:51:37 +02:00
|
|
|
@retval other Contain some errors when excuting this function.See function
|
|
|
|
EfiBootManagerInitializeLoadOption/EfiBootManagerAddLoadOptionVariabl
|
|
|
|
for detail return information.
|
2015-11-23 10:34:55 +01:00
|
|
|
@retval EFI_SUCCESS If function completes successfully.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateDriverOption (
|
|
|
|
IN BMM_CALLBACK_DATA *CallbackData,
|
|
|
|
IN EFI_HII_HANDLE HiiHandle,
|
|
|
|
IN UINT16 *DescriptionData,
|
|
|
|
IN UINT16 *OptionalData,
|
|
|
|
IN UINT8 ForceReconnect
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT16 Index;
|
|
|
|
UINT16 DriverString[12];
|
|
|
|
BM_MENU_ENTRY *NewMenuEntry;
|
|
|
|
BM_LOAD_CONTEXT *NewLoadContext;
|
|
|
|
BOOLEAN OptionalDataExist;
|
|
|
|
EFI_STATUS Status;
|
2016-01-26 09:07:18 +01:00
|
|
|
EFI_BOOT_MANAGER_LOAD_OPTION LoadOption;
|
|
|
|
UINT8 *OptionalDesData;
|
|
|
|
UINT32 OptionalDataSize;
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
OptionalDataExist = FALSE;
|
2016-01-26 09:07:18 +01:00
|
|
|
OptionalDesData = NULL;
|
|
|
|
OptionalDataSize = 0;
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
Index = BOpt_GetDriverOptionNumber ();
|
|
|
|
UnicodeSPrint (
|
|
|
|
DriverString,
|
|
|
|
sizeof (DriverString),
|
|
|
|
L"Driver%04x",
|
|
|
|
Index
|
|
|
|
);
|
|
|
|
|
|
|
|
if (*DescriptionData == 0x0000) {
|
|
|
|
StrCpyS (DescriptionData, MAX_MENU_NUMBER, DriverString);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*OptionalData != 0x0000) {
|
|
|
|
OptionalDataExist = TRUE;
|
2016-01-26 09:07:18 +01:00
|
|
|
OptionalDesData = (UINT8 *)OptionalData;
|
|
|
|
OptionalDataSize = (UINT32)StrSize (OptionalData);
|
2015-11-23 10:34:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
NewMenuEntry = BOpt_CreateMenuEntry (BM_LOAD_CONTEXT_SELECT);
|
|
|
|
if (NULL == NewMenuEntry) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
2016-01-26 09:07:18 +01:00
|
|
|
Status = EfiBootManagerInitializeLoadOption (
|
|
|
|
&LoadOption,
|
|
|
|
Index,
|
|
|
|
LoadOptionTypeDriver,
|
|
|
|
LOAD_OPTION_ACTIVE | (ForceReconnect << 1),
|
|
|
|
DescriptionData,
|
|
|
|
CallbackData->LoadContext->FilePathList,
|
|
|
|
OptionalDesData,
|
|
|
|
OptionalDataSize
|
|
|
|
);
|
2016-10-17 08:51:37 +02:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = EfiBootManagerAddLoadOptionVariable (&LoadOption, (UINTN)-1);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
EfiBootManagerFreeLoadOption (&LoadOption);
|
|
|
|
return Status;
|
2016-01-26 09:07:18 +01:00
|
|
|
}
|
|
|
|
|
2015-11-23 10:34:55 +01:00
|
|
|
NewLoadContext = (BM_LOAD_CONTEXT *)NewMenuEntry->VariableContext;
|
|
|
|
NewLoadContext->Deleted = FALSE;
|
2016-01-26 09:07:18 +01:00
|
|
|
NewLoadContext->Attributes = LoadOption.Attributes;
|
|
|
|
NewLoadContext->FilePathListLength = (UINT16)GetDevicePathSize (LoadOption.FilePath);
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
NewLoadContext->Description = AllocateZeroPool (StrSize (DescriptionData));
|
|
|
|
ASSERT (NewLoadContext->Description != NULL);
|
|
|
|
NewMenuEntry->DisplayString = NewLoadContext->Description;
|
|
|
|
CopyMem (
|
|
|
|
NewLoadContext->Description,
|
2016-01-26 09:07:18 +01:00
|
|
|
LoadOption.Description,
|
2015-11-23 10:34:55 +01:00
|
|
|
StrSize (DescriptionData)
|
|
|
|
);
|
|
|
|
|
|
|
|
NewLoadContext->FilePathList = AllocateZeroPool (GetDevicePathSize (CallbackData->LoadContext->FilePathList));
|
|
|
|
ASSERT (NewLoadContext->FilePathList != NULL);
|
|
|
|
CopyMem (
|
|
|
|
NewLoadContext->FilePathList,
|
2016-01-26 09:07:18 +01:00
|
|
|
LoadOption.FilePath,
|
2015-11-23 10:34:55 +01:00
|
|
|
GetDevicePathSize (CallbackData->LoadContext->FilePathList)
|
|
|
|
);
|
|
|
|
|
|
|
|
NewMenuEntry->HelpString = UiDevicePathToStr (NewLoadContext->FilePathList);
|
|
|
|
NewMenuEntry->OptionNumber = Index;
|
|
|
|
NewMenuEntry->DisplayStringToken = HiiSetString (HiiHandle, 0, NewMenuEntry->DisplayString, NULL);
|
|
|
|
NewMenuEntry->HelpStringToken = HiiSetString (HiiHandle, 0, NewMenuEntry->HelpString, NULL);
|
|
|
|
|
|
|
|
if (OptionalDataExist) {
|
2016-01-26 09:07:18 +01:00
|
|
|
NewLoadContext->OptionalData = AllocateZeroPool (LoadOption.OptionalDataSize);
|
2016-02-25 08:09:39 +01:00
|
|
|
ASSERT (NewLoadContext->OptionalData != NULL);
|
2015-11-23 10:34:55 +01:00
|
|
|
CopyMem (
|
2016-01-26 09:07:18 +01:00
|
|
|
NewLoadContext->OptionalData,
|
|
|
|
LoadOption.OptionalData,
|
|
|
|
LoadOption.OptionalDataSize
|
2015-11-23 10:34:55 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
InsertTailList (&DriverOptionMenu.Head, &NewMenuEntry->Link);
|
|
|
|
DriverOptionMenu.MenuNumber++;
|
|
|
|
|
2016-01-26 09:07:18 +01:00
|
|
|
EfiBootManagerFreeLoadOption (&LoadOption);
|
|
|
|
|
2015-11-23 10:34:55 +01:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-06-27 15:08:52 +02:00
|
|
|
This function create a currently loaded Boot Option from
|
|
|
|
the BMM. It then appends this Boot Option to the end of
|
2015-11-23 10:34:55 +01:00
|
|
|
the "BootOrder" list. It also append this Boot Opotion to the end
|
|
|
|
of BootOptionMenu.
|
|
|
|
|
|
|
|
@param CallbackData The BMM context data.
|
|
|
|
|
2016-10-17 08:51:37 +02:00
|
|
|
@retval other Contain some errors when excuting this function. See function
|
|
|
|
EfiBootManagerInitializeLoadOption/EfiBootManagerAddLoadOptionVariabl
|
|
|
|
for detail return information.
|
2015-11-23 10:34:55 +01:00
|
|
|
@retval EFI_SUCCESS If function completes successfully.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateBootOption (
|
|
|
|
IN BMM_CALLBACK_DATA *CallbackData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT16 BootString[10];
|
|
|
|
UINT16 Index;
|
|
|
|
BM_MENU_ENTRY *NewMenuEntry;
|
|
|
|
BM_LOAD_CONTEXT *NewLoadContext;
|
|
|
|
BOOLEAN OptionalDataExist;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
BMM_FAKE_NV_DATA *NvRamMap;
|
2016-01-26 09:07:18 +01:00
|
|
|
EFI_BOOT_MANAGER_LOAD_OPTION LoadOption;
|
|
|
|
UINT8 *OptionalData;
|
|
|
|
UINT32 OptionalDataSize;
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
OptionalDataExist = FALSE;
|
|
|
|
NvRamMap = &CallbackData->BmmFakeNvData;
|
2016-01-26 09:07:18 +01:00
|
|
|
OptionalData = NULL;
|
|
|
|
OptionalDataSize = 0;
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
Index = BOpt_GetBootOptionNumber ();
|
|
|
|
UnicodeSPrint (BootString, sizeof (BootString), L"Boot%04x", Index);
|
|
|
|
|
|
|
|
if (NvRamMap->BootDescriptionData[0] == 0x0000) {
|
|
|
|
StrCpyS (NvRamMap->BootDescriptionData, sizeof (NvRamMap->BootDescriptionData) / sizeof (NvRamMap->BootDescriptionData[0]), BootString);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (NvRamMap->BootOptionalData[0] != 0x0000) {
|
|
|
|
OptionalDataExist = TRUE;
|
2016-01-26 09:07:18 +01:00
|
|
|
OptionalData = (UINT8 *)NvRamMap->BootOptionalData;
|
|
|
|
OptionalDataSize = (UINT32)StrSize (NvRamMap->BootOptionalData);
|
2015-11-23 10:34:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
NewMenuEntry = BOpt_CreateMenuEntry (BM_LOAD_CONTEXT_SELECT);
|
|
|
|
if (NULL == NewMenuEntry) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
2016-01-26 09:07:18 +01:00
|
|
|
Status = EfiBootManagerInitializeLoadOption (
|
|
|
|
&LoadOption,
|
|
|
|
Index,
|
|
|
|
LoadOptionTypeBoot,
|
|
|
|
LOAD_OPTION_ACTIVE,
|
|
|
|
NvRamMap->BootDescriptionData,
|
|
|
|
CallbackData->LoadContext->FilePathList,
|
|
|
|
OptionalData,
|
|
|
|
OptionalDataSize
|
|
|
|
);
|
2016-10-17 08:51:37 +02:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = EfiBootManagerAddLoadOptionVariable (&LoadOption, (UINTN)-1);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
EfiBootManagerFreeLoadOption (&LoadOption);
|
|
|
|
return Status;
|
2016-01-26 09:07:18 +01:00
|
|
|
}
|
|
|
|
|
2015-11-23 10:34:55 +01:00
|
|
|
NewLoadContext = (BM_LOAD_CONTEXT *)NewMenuEntry->VariableContext;
|
|
|
|
NewLoadContext->Deleted = FALSE;
|
2016-01-26 09:07:18 +01:00
|
|
|
NewLoadContext->Attributes = LoadOption.Attributes;
|
|
|
|
NewLoadContext->FilePathListLength = (UINT16)GetDevicePathSize (LoadOption.FilePath);
|
2015-11-23 10:34:55 +01:00
|
|
|
|
|
|
|
NewLoadContext->Description = AllocateZeroPool (StrSize (NvRamMap->BootDescriptionData));
|
|
|
|
ASSERT (NewLoadContext->Description != NULL);
|
|
|
|
|
|
|
|
NewMenuEntry->DisplayString = NewLoadContext->Description;
|
2016-01-26 09:07:18 +01:00
|
|
|
|
2015-11-23 10:34:55 +01:00
|
|
|
CopyMem (
|
|
|
|
NewLoadContext->Description,
|
2016-01-26 09:07:18 +01:00
|
|
|
LoadOption.Description,
|
2015-11-23 10:34:55 +01:00
|
|
|
StrSize (NvRamMap->BootDescriptionData)
|
|
|
|
);
|
|
|
|
|
|
|
|
NewLoadContext->FilePathList = AllocateZeroPool (GetDevicePathSize (CallbackData->LoadContext->FilePathList));
|
|
|
|
ASSERT (NewLoadContext->FilePathList != NULL);
|
|
|
|
CopyMem (
|
|
|
|
NewLoadContext->FilePathList,
|
2016-01-26 09:07:18 +01:00
|
|
|
LoadOption.FilePath,
|
2015-11-23 10:34:55 +01:00
|
|
|
GetDevicePathSize (CallbackData->LoadContext->FilePathList)
|
|
|
|
);
|
|
|
|
|
|
|
|
NewMenuEntry->HelpString = UiDevicePathToStr (NewLoadContext->FilePathList);
|
|
|
|
NewMenuEntry->OptionNumber = Index;
|
|
|
|
NewMenuEntry->DisplayStringToken = HiiSetString (CallbackData->BmmHiiHandle, 0, NewMenuEntry->DisplayString, NULL);
|
|
|
|
NewMenuEntry->HelpStringToken = HiiSetString (CallbackData->BmmHiiHandle, 0, NewMenuEntry->HelpString, NULL);
|
|
|
|
|
|
|
|
if (OptionalDataExist) {
|
2016-01-26 09:07:18 +01:00
|
|
|
NewLoadContext->OptionalData = AllocateZeroPool (LoadOption.OptionalDataSize);
|
2016-02-25 08:09:39 +01:00
|
|
|
ASSERT (NewLoadContext->OptionalData != NULL);
|
2016-01-26 09:07:18 +01:00
|
|
|
CopyMem (
|
|
|
|
NewLoadContext->OptionalData,
|
|
|
|
LoadOption.OptionalData,
|
|
|
|
LoadOption.OptionalDataSize
|
|
|
|
);
|
2015-11-23 10:34:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
InsertTailList (&BootOptionMenu.Head, &NewMenuEntry->Link);
|
|
|
|
BootOptionMenu.MenuNumber++;
|
|
|
|
|
2016-01-26 09:07:18 +01:00
|
|
|
EfiBootManagerFreeLoadOption (&LoadOption);
|
|
|
|
|
2015-11-23 10:34:55 +01:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-06-27 15:08:52 +02:00
|
|
|
This function update the "BootNext" EFI Variable. If there is
|
2015-11-23 10:34:55 +01:00
|
|
|
no "BootNext" specified in BMM, this EFI Variable is deleted.
|
|
|
|
It also update the BMM context data specified the "BootNext"
|
|
|
|
vaule.
|
|
|
|
|
|
|
|
@param CallbackData The BMM context data.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The function complete successfully.
|
2018-06-27 15:08:52 +02:00
|
|
|
@return The EFI variable can be saved. See gRT->SetVariable
|
2015-11-23 10:34:55 +01:00
|
|
|
for detail return information.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateBootNext (
|
|
|
|
IN BMM_CALLBACK_DATA *CallbackData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
BM_MENU_ENTRY *NewMenuEntry;
|
|
|
|
BM_LOAD_CONTEXT *NewLoadContext;
|
|
|
|
BMM_FAKE_NV_DATA *CurrentFakeNVMap;
|
|
|
|
UINT16 Index;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
CurrentFakeNVMap = &CallbackData->BmmFakeNvData;
|
|
|
|
for (Index = 0; Index < BootOptionMenu.MenuNumber; Index++) {
|
|
|
|
NewMenuEntry = BOpt_GetMenuEntry (&BootOptionMenu, Index);
|
|
|
|
ASSERT (NULL != NewMenuEntry);
|
|
|
|
|
|
|
|
NewLoadContext = (BM_LOAD_CONTEXT *)NewMenuEntry->VariableContext;
|
|
|
|
NewLoadContext->IsBootNext = FALSE;
|
|
|
|
}
|
|
|
|
|
2016-01-06 01:55:12 +01:00
|
|
|
if (CurrentFakeNVMap->BootNext == NONE_BOOTNEXT_VALUE) {
|
2015-11-23 10:34:55 +01:00
|
|
|
EfiLibDeleteVariable (L"BootNext", &gEfiGlobalVariableGuid);
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
NewMenuEntry = BOpt_GetMenuEntry (
|
|
|
|
&BootOptionMenu,
|
|
|
|
CurrentFakeNVMap->BootNext
|
|
|
|
);
|
|
|
|
ASSERT (NewMenuEntry != NULL);
|
|
|
|
|
|
|
|
NewLoadContext = (BM_LOAD_CONTEXT *)NewMenuEntry->VariableContext;
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
L"BootNext",
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
VAR_FLAG,
|
|
|
|
sizeof (UINT16),
|
|
|
|
&NewMenuEntry->OptionNumber
|
|
|
|
);
|
|
|
|
NewLoadContext->IsBootNext = TRUE;
|
|
|
|
CallbackData->BmmOldFakeNVData.BootNext = CurrentFakeNVMap->BootNext;
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function update the "BootOrder" EFI Variable based on
|
|
|
|
BMM Formset's NV map. It then refresh BootOptionMenu
|
|
|
|
with the new "BootOrder" list.
|
|
|
|
|
|
|
|
@param CallbackData The BMM context data.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The function complete successfully.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES Not enough memory to complete the function.
|
|
|
|
@return The EFI variable can not be saved. See gRT->SetVariable for detail return information.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateBootOrder (
|
|
|
|
IN BMM_CALLBACK_DATA *CallbackData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT16 Index;
|
|
|
|
UINT16 OrderIndex;
|
|
|
|
UINT16 *BootOrder;
|
|
|
|
UINTN BootOrderSize;
|
|
|
|
UINT16 OptionNumber;
|
|
|
|
|
|
|
|
//
|
|
|
|
// First check whether BootOrder is present in current configuration
|
|
|
|
//
|
|
|
|
GetEfiGlobalVariable2 (L"BootOrder", (VOID **)&BootOrder, &BootOrderSize);
|
|
|
|
if (BootOrder == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT (BootOptionMenu.MenuNumber <= (sizeof (CallbackData->BmmFakeNvData.BootOptionOrder) / sizeof (CallbackData->BmmFakeNvData.BootOptionOrder[0])));
|
|
|
|
|
|
|
|
//
|
|
|
|
// OptionOrder is subset of BootOrder
|
|
|
|
//
|
|
|
|
for (OrderIndex = 0; (OrderIndex < BootOptionMenu.MenuNumber) && (CallbackData->BmmFakeNvData.BootOptionOrder[OrderIndex] != 0); OrderIndex++) {
|
|
|
|
for (Index = OrderIndex; Index < BootOrderSize / sizeof (UINT16); Index++) {
|
|
|
|
if ((BootOrder[Index] == (UINT16)(CallbackData->BmmFakeNvData.BootOptionOrder[OrderIndex] - 1)) && (OrderIndex != Index)) {
|
|
|
|
OptionNumber = BootOrder[Index];
|
|
|
|
CopyMem (&BootOrder[OrderIndex + 1], &BootOrder[OrderIndex], (Index - OrderIndex) * sizeof (UINT16));
|
|
|
|
BootOrder[OrderIndex] = OptionNumber;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
L"BootOrder",
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
VAR_FLAG,
|
|
|
|
BootOrderSize,
|
|
|
|
BootOrder
|
|
|
|
);
|
|
|
|
FreePool (BootOrder);
|
2018-06-27 15:08:52 +02:00
|
|
|
|
2015-11-23 10:34:55 +01:00
|
|
|
BOpt_FreeMenu (&BootOptionMenu);
|
|
|
|
BOpt_GetBootOptions (CallbackData);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function update the "DriverOrder" EFI Variable based on
|
|
|
|
BMM Formset's NV map. It then refresh DriverOptionMenu
|
|
|
|
with the new "DriverOrder" list.
|
|
|
|
|
|
|
|
@param CallbackData The BMM context data.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The function complete successfully.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES Not enough memory to complete the function.
|
|
|
|
@return The EFI variable can not be saved. See gRT->SetVariable for detail return information.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateDriverOrder (
|
|
|
|
IN BMM_CALLBACK_DATA *CallbackData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT16 Index;
|
|
|
|
UINT16 *DriverOrderList;
|
|
|
|
UINT16 *NewDriverOrderList;
|
|
|
|
UINTN DriverOrderListSize;
|
|
|
|
|
|
|
|
DriverOrderList = NULL;
|
|
|
|
DriverOrderListSize = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
// First check whether DriverOrder is present in current configuration
|
|
|
|
//
|
|
|
|
GetEfiGlobalVariable2 (L"DriverOrder", (VOID **)&DriverOrderList, &DriverOrderListSize);
|
|
|
|
NewDriverOrderList = AllocateZeroPool (DriverOrderListSize);
|
|
|
|
|
|
|
|
if (NewDriverOrderList == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
2021-12-05 23:54:02 +01:00
|
|
|
|
2015-11-23 10:34:55 +01:00
|
|
|
//
|
|
|
|
// If exists, delete it to hold new DriverOrder
|
|
|
|
//
|
|
|
|
if (DriverOrderList != NULL) {
|
|
|
|
EfiLibDeleteVariable (L"DriverOrder", &gEfiGlobalVariableGuid);
|
|
|
|
FreePool (DriverOrderList);
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT (DriverOptionMenu.MenuNumber <= (sizeof (CallbackData->BmmFakeNvData.DriverOptionOrder) / sizeof (CallbackData->BmmFakeNvData.DriverOptionOrder[0])));
|
|
|
|
for (Index = 0; Index < DriverOptionMenu.MenuNumber; Index++) {
|
|
|
|
NewDriverOrderList[Index] = (UINT16)(CallbackData->BmmFakeNvData.DriverOptionOrder[Index] - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
L"DriverOrder",
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
VAR_FLAG,
|
|
|
|
DriverOrderListSize,
|
|
|
|
NewDriverOrderList
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOpt_FreeMenu (&DriverOptionMenu);
|
|
|
|
BOpt_GetDriverOptions (CallbackData);
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Update the Text Mode of Console.
|
|
|
|
|
|
|
|
@param CallbackData The context data for BMM.
|
|
|
|
|
|
|
|
@retval EFI_SUCCSS If the Text Mode of Console is updated.
|
|
|
|
@return Other value if the Text Mode of Console is not updated.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
Var_UpdateConMode (
|
|
|
|
IN BMM_CALLBACK_DATA *CallbackData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN Mode;
|
|
|
|
CONSOLE_OUT_MODE ModeInfo;
|
|
|
|
|
|
|
|
Mode = CallbackData->BmmFakeNvData.ConsoleOutMode;
|
|
|
|
|
|
|
|
Status = gST->ConOut->QueryMode (gST->ConOut, Mode, &(ModeInfo.Column), &(ModeInfo.Row));
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Status = PcdSet32S (PcdSetupConOutColumn, (UINT32)ModeInfo.Column);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Status = PcdSet32S (PcdSetupConOutRow, (UINT32)ModeInfo.Row);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|