2013-01-25 12:28:06 +01:00
|
|
|
/** @file
|
|
|
|
*
|
2015-07-14 16:30:08 +02:00
|
|
|
* Copyright (c) 2011-2015, ARM Limited. All rights reserved.
|
2013-01-25 12:28:06 +01:00
|
|
|
*
|
|
|
|
* This program and the accompanying materials
|
|
|
|
* are licensed and made available under the terms and conditions of the BSD License
|
|
|
|
* which accompanies this distribution. The full text of the license may be found at
|
|
|
|
* http://opensource.org/licenses/bsd-license.php
|
|
|
|
*
|
|
|
|
* THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|
|
|
*
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "BdsInternal.h"
|
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
/**
|
|
|
|
Locate an EFI application in a the Firmware Volumes by its Name
|
|
|
|
|
|
|
|
@param EfiAppGuid Guid of the EFI Application into the Firmware Volume
|
|
|
|
@param DevicePath EFI Device Path of the EFI application
|
|
|
|
|
|
|
|
@return EFI_SUCCESS The function completed successfully.
|
|
|
|
@return EFI_NOT_FOUND The protocol could not be located.
|
|
|
|
@return EFI_OUT_OF_RESOURCES There are not enough resources to find the protocol.
|
2013-01-25 12:28:06 +01:00
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
**/
|
2013-01-25 12:28:06 +01:00
|
|
|
EFI_STATUS
|
2015-07-14 16:30:08 +02:00
|
|
|
LocateEfiApplicationInFvByName (
|
|
|
|
IN CONST CHAR16* EfiAppName,
|
|
|
|
OUT EFI_DEVICE_PATH **DevicePath
|
2013-01-25 12:28:06 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
VOID *Key;
|
|
|
|
EFI_STATUS Status, FileStatus;
|
|
|
|
EFI_GUID NameGuid;
|
|
|
|
EFI_FV_FILETYPE FileType;
|
|
|
|
EFI_FV_FILE_ATTRIBUTES Attributes;
|
|
|
|
UINTN Size;
|
|
|
|
UINTN UiStringLen;
|
|
|
|
CHAR16 *UiSection;
|
|
|
|
UINT32 Authentication;
|
|
|
|
EFI_DEVICE_PATH *FvDevicePath;
|
|
|
|
MEDIA_FW_VOL_FILEPATH_DEVICE_PATH FileDevicePath;
|
2015-07-14 16:30:08 +02:00
|
|
|
EFI_HANDLE *HandleBuffer;
|
|
|
|
UINTN NumberOfHandles;
|
|
|
|
UINTN Index;
|
|
|
|
EFI_FIRMWARE_VOLUME2_PROTOCOL *FvInstance;
|
2013-01-25 12:28:06 +01:00
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
ASSERT (DevicePath != NULL);
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
// Length of FilePath
|
2015-07-14 16:30:08 +02:00
|
|
|
UiStringLen = StrLen (EfiAppName);
|
|
|
|
|
|
|
|
// Locate all the Firmware Volume protocols.
|
|
|
|
Status = gBS->LocateHandleBuffer (
|
|
|
|
ByProtocol,
|
|
|
|
&gEfiFirmwareVolume2ProtocolGuid,
|
|
|
|
NULL,
|
|
|
|
&NumberOfHandles,
|
|
|
|
&HandleBuffer
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
*DevicePath = NULL;
|
|
|
|
|
|
|
|
// Looking for FV with ACPI storage file
|
|
|
|
for (Index = 0; Index < NumberOfHandles; Index++) {
|
|
|
|
//
|
|
|
|
// Get the protocol on this handle
|
|
|
|
// This should not fail because of LocateHandleBuffer
|
|
|
|
//
|
|
|
|
Status = gBS->HandleProtocol (
|
|
|
|
HandleBuffer[Index],
|
|
|
|
&gEfiFirmwareVolume2ProtocolGuid,
|
|
|
|
(VOID**) &FvInstance
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto FREE_HANDLE_BUFFER;
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
// Allocate Key
|
|
|
|
Key = AllocatePool (FvInstance->KeySize);
|
|
|
|
ASSERT (Key != NULL);
|
|
|
|
ZeroMem (Key, FvInstance->KeySize);
|
|
|
|
|
|
|
|
do {
|
|
|
|
// Search in all files
|
|
|
|
FileType = EFI_FV_FILETYPE_ALL;
|
|
|
|
|
|
|
|
Status = FvInstance->GetNextFile (FvInstance, Key, &FileType, &NameGuid, &Attributes, &Size);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
UiSection = NULL;
|
|
|
|
FileStatus = FvInstance->ReadSection (
|
|
|
|
FvInstance,
|
|
|
|
&NameGuid,
|
|
|
|
EFI_SECTION_USER_INTERFACE,
|
|
|
|
0,
|
|
|
|
(VOID **)&UiSection,
|
|
|
|
&Size,
|
|
|
|
&Authentication
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (FileStatus)) {
|
|
|
|
if (StrnCmp (EfiAppName, UiSection, UiStringLen) == 0) {
|
|
|
|
//
|
|
|
|
// We found a UiString match.
|
|
|
|
//
|
|
|
|
Status = gBS->HandleProtocol (HandleBuffer[Index], &gEfiDevicePathProtocolGuid, (VOID **)&FvDevicePath);
|
|
|
|
|
|
|
|
// Generate the Device Path for the file
|
|
|
|
EfiInitializeFwVolDevicepathNode (&FileDevicePath, &NameGuid);
|
|
|
|
*DevicePath = AppendDevicePathNode (FvDevicePath, (EFI_DEVICE_PATH_PROTOCOL *)&FileDevicePath);
|
|
|
|
ASSERT (*DevicePath != NULL);
|
|
|
|
|
|
|
|
FreePool (Key);
|
|
|
|
FreePool (UiSection);
|
|
|
|
FreePool (HandleBuffer);
|
|
|
|
return FileStatus;
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
FreePool (UiSection);
|
|
|
|
}
|
|
|
|
}
|
2015-07-14 16:30:08 +02:00
|
|
|
} while (!EFI_ERROR (Status));
|
2013-01-25 12:28:06 +01:00
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
FreePool (Key);
|
|
|
|
}
|
|
|
|
|
|
|
|
FREE_HANDLE_BUFFER:
|
|
|
|
FreePool (HandleBuffer);
|
|
|
|
return EFI_NOT_FOUND;
|
2013-01-25 12:28:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-07-14 16:30:08 +02:00
|
|
|
Locate an EFI application in a the Firmware Volumes by its GUID
|
2013-01-25 12:28:06 +01:00
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
@param EfiAppGuid Guid of the EFI Application into the Firmware Volume
|
|
|
|
@param DevicePath EFI Device Path of the EFI application
|
2013-01-25 12:28:06 +01:00
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
@return EFI_SUCCESS The function completed successfully.
|
|
|
|
@return EFI_NOT_FOUND The protocol could not be located.
|
|
|
|
@return EFI_OUT_OF_RESOURCES There are not enough resources to find the protocol.
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
2015-07-14 16:30:08 +02:00
|
|
|
LocateEfiApplicationInFvByGuid (
|
|
|
|
IN CONST EFI_GUID *EfiAppGuid,
|
|
|
|
OUT EFI_DEVICE_PATH **DevicePath
|
2013-01-25 12:28:06 +01:00
|
|
|
)
|
|
|
|
{
|
2015-07-14 16:30:08 +02:00
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_DEVICE_PATH *FvDevicePath;
|
|
|
|
EFI_HANDLE *HandleBuffer;
|
|
|
|
UINTN NumberOfHandles;
|
|
|
|
UINTN Index;
|
|
|
|
EFI_FIRMWARE_VOLUME2_PROTOCOL *FvInstance;
|
|
|
|
EFI_FV_FILE_ATTRIBUTES Attributes;
|
|
|
|
UINT32 AuthenticationStatus;
|
|
|
|
EFI_FV_FILETYPE Type;
|
|
|
|
UINTN Size;
|
|
|
|
CHAR16 *UiSection;
|
|
|
|
MEDIA_FW_VOL_FILEPATH_DEVICE_PATH FvFileDevicePath;
|
|
|
|
|
|
|
|
ASSERT (DevicePath != NULL);
|
|
|
|
|
|
|
|
// Locate all the Firmware Volume protocols.
|
|
|
|
Status = gBS->LocateHandleBuffer (
|
|
|
|
ByProtocol,
|
|
|
|
&gEfiFirmwareVolume2ProtocolGuid,
|
|
|
|
NULL,
|
|
|
|
&NumberOfHandles,
|
|
|
|
&HandleBuffer
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
2013-01-25 12:28:06 +01:00
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
*DevicePath = NULL;
|
|
|
|
|
|
|
|
// Looking for FV with ACPI storage file
|
|
|
|
for (Index = 0; Index < NumberOfHandles; Index++) {
|
|
|
|
//
|
|
|
|
// Get the protocol on this handle
|
|
|
|
// This should not fail because of LocateHandleBuffer
|
|
|
|
//
|
|
|
|
Status = gBS->HandleProtocol (
|
|
|
|
HandleBuffer[Index],
|
|
|
|
&gEfiFirmwareVolume2ProtocolGuid,
|
|
|
|
(VOID**) &FvInstance
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto FREE_HANDLE_BUFFER;
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
Status = FvInstance->ReadFile (
|
|
|
|
FvInstance,
|
|
|
|
EfiAppGuid,
|
|
|
|
NULL,
|
|
|
|
&Size,
|
|
|
|
&Type,
|
|
|
|
&Attributes,
|
|
|
|
&AuthenticationStatus
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// Skip if no EFI application file in the FV
|
|
|
|
//
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
UiSection = NULL;
|
|
|
|
Status = FvInstance->ReadSection (
|
|
|
|
FvInstance,
|
|
|
|
EfiAppGuid,
|
|
|
|
EFI_SECTION_USER_INTERFACE,
|
|
|
|
0,
|
|
|
|
(VOID **)&UiSection,
|
|
|
|
&Size,
|
|
|
|
&AuthenticationStatus
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// Create the EFI Device Path for the application using the Filename of the application
|
|
|
|
//
|
|
|
|
*DevicePath = FileDevicePath (HandleBuffer[Index], UiSection);
|
|
|
|
} else {
|
|
|
|
Status = gBS->HandleProtocol (HandleBuffer[Index], &gEfiDevicePathProtocolGuid, (VOID**)&FvDevicePath);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Create the EFI Device Path for the application using the EFI GUID of the application
|
|
|
|
//
|
|
|
|
EfiInitializeFwVolDevicepathNode (&FvFileDevicePath, EfiAppGuid);
|
|
|
|
|
|
|
|
*DevicePath = AppendDevicePathNode (FvDevicePath, (EFI_DEVICE_PATH_PROTOCOL *)&FvFileDevicePath);
|
|
|
|
ASSERT (*DevicePath != NULL);
|
|
|
|
}
|
|
|
|
break;
|
2013-01-25 12:28:06 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-14 16:30:08 +02:00
|
|
|
FREE_HANDLE_BUFFER:
|
|
|
|
//
|
|
|
|
// Free any allocated buffers
|
|
|
|
//
|
|
|
|
FreePool (HandleBuffer);
|
|
|
|
|
|
|
|
if (*DevicePath == NULL) {
|
|
|
|
return EFI_NOT_FOUND;
|
|
|
|
} else {
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
}
|