2015-08-03 07:26:01 +02:00
|
|
|
/** @file
|
|
|
|
Platform BDS customizations.
|
|
|
|
|
2019-06-14 03:14:29 +02:00
|
|
|
Copyright (c) 2004 - 2019, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:06:33 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "BdsPlatform.h"
|
|
|
|
#include <Guid/RootBridgesConnectedEventGroup.h>
|
2016-07-08 02:49:45 +02:00
|
|
|
#include <Protocol/FirmwareVolume2.h>
|
2019-02-20 01:08:58 +01:00
|
|
|
#include <Library/PlatformBmPrintScLib.h>
|
OvmfPkg/PlatformBootManagerLib: process TPM PPI request
Call Tcg2PhysicalPresenceLibProcessRequest() to process pending PPI
requests from PlatformBootManagerAfterConsole().
Laszlo understanding of edk2 is that the PPI operation processing was
meant to occur *entirely* before End-Of-Dxe, so that 3rd party UEFI
drivers couldn't interfere with PPI opcode processing *at all*.
He suggested that we should *not* call
Tcg2PhysicalPresenceLibProcessRequest() from BeforeConsole(). Because,
an "auth" console, i.e. one that does not depend on a 3rd party
driver, is *in general* impossible to guarantee. Instead we could opt
to trust 3rd party drivers, and use the "normal" console(s) in
AfterConsole(), in order to let the user confirm the PPI requests. It
will depend on the user to enable Secure Boot, so that the
trustworthiness of those 3rd party drivers is ensured. If an attacker
roots the guest OS from within, queues some TPM2 PPI requests, and
also modifies drivers on the EFI system partition and/or in GPU option
ROMs (?), then those drivers will not load after guest reboot, and
thus the dependent console(s) won't be used for confirming the PPI
requests.
Contributed-under: TianoCore Contribution Agreement 1.1
Signed-off-by: Marc-André Lureau <marcandre.lureau@redhat.com>
Reviewed-by: Laszlo Ersek <lersek@redhat.com>
2018-05-18 14:23:04 +02:00
|
|
|
#include <Library/Tcg2PhysicalPresenceLib.h>
|
2019-08-13 13:31:11 +02:00
|
|
|
#include <Library/XenPlatformLib.h>
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Global data
|
|
|
|
//
|
|
|
|
|
|
|
|
VOID *mEfiDevPathNotifyReg;
|
|
|
|
EFI_EVENT mEfiDevPathEvent;
|
|
|
|
VOID *mEmuVariableEventReg;
|
|
|
|
EFI_EVENT mEmuVariableEvent;
|
|
|
|
UINT16 mHostBridgeDevId;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Table of host IRQs matching PCI IRQs A-D
|
|
|
|
// (for configuring PCI Interrupt Line register)
|
|
|
|
//
|
|
|
|
CONST UINT8 PciHostIrqs[] = {
|
OvmfPkg/PlatformBootManagerLib: fix PCI interrupt link (LNKx)
This patch fixes an issue with the current programming of the i440fx
PCI Interrupt routing assignment.
Explanation by Laszlo Ersek:
(1) The rotating pattern is a map:
(slot, function) --> (interrupt link) [LNKA..LNKD]
(more precisely, it is a pattern from (slot, pin) to (interrupt link),
but function<->pin is an identity mapping in the QEMU hardware, so we
can just use (slot, function) rather than (slot, pin) on the left hand
side. But I digress.)
The ACPI _PRT object is generated by QEMU; it describes this map.
(2) Another map is
(interrupt link) --> { set of possible interrupt numbers,
for this link }
This map is given by the LNK[A..D] ACPI objects, also given by QEMU.
(3) What the firmware is expected to do is:
(3a) for each interrupt link, select an *actual* interrupt from the set
that's possible for that link, yielding a deterministic map
(interrupt link) --> (actual interrupt number)
and
(3b) for each PCI device/function with an interrupt pin, resolve the
(slot, function) --> (interrupt link) --> (actual interrupt number)
functional composition, and program the result into the Interrupt Line
register of the device.
In OVMF, we do not parse the rotating map described under (1) from
QEMU's _PRT object. Instead, we duplicate the code. This is not a
problem.
In OVMF, we also do not parse the map described under (2) from QEMU's
ACPI content. Instead, we pick a specific selection (3a) that we
"apriori" know satisfies (2). This is also not a problem. OVMF's
particular selection is the PciHostIrqs table.
(
Table (2) from QEMU is
LNKA -> { 5, 10, 11 }
LNKB -> { 5, 10, 11 }
LNKC -> { 5, 10, 11 }
LNKD -> { 5, 10, 11 }
and our specific pick in OVMF, in the PciHostIrqs table, is
LNKA -> 10
LNKB -> 10
LNKC -> 11
LNKD -> 11
)
In OVMF, we also cover step (3b), in the SetPciIntLine() function.
What's missing in OVMF -- and what this patch corrects -- is that we
currently fail to program our selection for table (3) into the hardware.
We pick a specific LNKx->IRQ# mapping for each interrupt link, and we
correctly program the PCI Interrupt Line registers through those
link-to-IRQ mappings -- but we don't tell the hardware about the
link-to-IRQ mappings. More precisely, we program such a link-to-IRQ
mapping table into the hardware that is then not matched by the mapping
we use for programming the PCI device/function interrupt lines. As a
result, some PCI Interrupt Line registers will have impossible values --
a given (slot, function) may use a particular link, but also report an
interrupt number that was never picked for that link.
Output of Linux PCI Interrupt Links for i440fx before the patch:
[ 0.327305] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 10 *11)
[ 0.327944] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 10 *11)
[ 0.328582] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 *10 11)
[ 0.329208] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 *10 11)
[ 0.329807] ACPI: PCI Interrupt Link [LNKS] (IRQs *9)
after the patch:
[ 0.327292] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.327934] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.328564] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.329195] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.329785] ACPI: PCI Interrupt Link [LNKS] (IRQs *9)
Output of Linux PCI Interrupt Links for q35 before the patch:
[ 0.307474] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.308027] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.308764] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.309310] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.309853] ACPI: PCI Interrupt Link [LNKE] (IRQs 5 *10 11)
[ 0.310508] ACPI: PCI Interrupt Link [LNKF] (IRQs 5 *10 11)
[ 0.311051] ACPI: PCI Interrupt Link [LNKG] (IRQs 5 10 *11)
[ 0.311589] ACPI: PCI Interrupt Link [LNKH] (IRQs 5 10 *11)
after the patch:
[ 0.301991] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.302833] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.303354] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.303873] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.304399] ACPI: PCI Interrupt Link [LNKE] (IRQs 5 *10 11)
[ 0.304918] ACPI: PCI Interrupt Link [LNKF] (IRQs 5 *10 11)
[ 0.305436] ACPI: PCI Interrupt Link [LNKG] (IRQs 5 10 *11)
[ 0.305954] ACPI: PCI Interrupt Link [LNKH] (IRQs 5 10 *11)
Signed-off-by: Hendrik Borghorst <hborghor@amazon.de>
Reviewed-by: David Woodhouse <dwmw@amazon.co.uk>
Message-Id: <8dbedc4c7a1c3fd390aca915270814e3b35e13a5.camel@amazon.com>
Reviewed-by: Laszlo Ersek <lersek@redhat.com>
2020-12-17 15:12:22 +01:00
|
|
|
0x0a, // LNKA, LNKE
|
|
|
|
0x0a, // LNKB, LNKF
|
|
|
|
0x0b, // LNKC, LNKG
|
|
|
|
0x0b // LNKD, LNKH
|
2015-08-03 07:26:01 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
//
|
|
|
|
// Type definitions
|
|
|
|
//
|
|
|
|
|
|
|
|
typedef
|
|
|
|
EFI_STATUS
|
|
|
|
(EFIAPI *PROTOCOL_INSTANCE_CALLBACK)(
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN VOID *Instance,
|
|
|
|
IN VOID *Context
|
|
|
|
);
|
|
|
|
|
|
|
|
/**
|
|
|
|
@param[in] Handle - Handle of PCI device instance
|
|
|
|
@param[in] PciIo - PCI IO protocol instance
|
|
|
|
@param[in] Pci - PCI Header register block
|
|
|
|
**/
|
|
|
|
typedef
|
|
|
|
EFI_STATUS
|
|
|
|
(EFIAPI *VISIT_PCI_INSTANCE_CALLBACK)(
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN EFI_PCI_IO_PROTOCOL *PciIo,
|
|
|
|
IN PCI_TYPE00 *Pci
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Function prototypes
|
|
|
|
//
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
VisitAllInstancesOfProtocol (
|
|
|
|
IN EFI_GUID *Id,
|
|
|
|
IN PROTOCOL_INSTANCE_CALLBACK CallBackFunction,
|
|
|
|
IN VOID *Context
|
|
|
|
);
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
VisitAllPciInstancesOfProtocol (
|
|
|
|
IN VISIT_PCI_INSTANCE_CALLBACK CallBackFunction
|
|
|
|
);
|
|
|
|
|
|
|
|
VOID
|
|
|
|
InstallDevicePathCallback (
|
|
|
|
VOID
|
|
|
|
);
|
|
|
|
|
2016-04-20 11:49:15 +02:00
|
|
|
VOID
|
|
|
|
PlatformRegisterFvBootOption (
|
|
|
|
EFI_GUID *FileGuid,
|
|
|
|
CHAR16 *Description,
|
|
|
|
UINT32 Attributes
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
INTN OptionIndex;
|
|
|
|
EFI_BOOT_MANAGER_LOAD_OPTION NewOption;
|
|
|
|
EFI_BOOT_MANAGER_LOAD_OPTION *BootOptions;
|
|
|
|
UINTN BootOptionCount;
|
|
|
|
MEDIA_FW_VOL_FILEPATH_DEVICE_PATH FileNode;
|
|
|
|
EFI_LOADED_IMAGE_PROTOCOL *LoadedImage;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
|
|
|
|
|
|
|
Status = gBS->HandleProtocol (
|
|
|
|
gImageHandle,
|
|
|
|
&gEfiLoadedImageProtocolGuid,
|
|
|
|
(VOID **) &LoadedImage
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
EfiInitializeFwVolDevicepathNode (&FileNode, FileGuid);
|
|
|
|
DevicePath = DevicePathFromHandle (LoadedImage->DeviceHandle);
|
|
|
|
ASSERT (DevicePath != NULL);
|
|
|
|
DevicePath = AppendDevicePathNode (
|
|
|
|
DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *) &FileNode
|
|
|
|
);
|
|
|
|
ASSERT (DevicePath != NULL);
|
|
|
|
|
|
|
|
Status = EfiBootManagerInitializeLoadOption (
|
|
|
|
&NewOption,
|
|
|
|
LoadOptionNumberUnassigned,
|
|
|
|
LoadOptionTypeBoot,
|
|
|
|
Attributes,
|
|
|
|
Description,
|
|
|
|
DevicePath,
|
|
|
|
NULL,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
FreePool (DevicePath);
|
|
|
|
|
|
|
|
BootOptions = EfiBootManagerGetLoadOptions (
|
|
|
|
&BootOptionCount, LoadOptionTypeBoot
|
|
|
|
);
|
|
|
|
|
|
|
|
OptionIndex = EfiBootManagerFindLoadOption (
|
|
|
|
&NewOption, BootOptions, BootOptionCount
|
|
|
|
);
|
|
|
|
|
|
|
|
if (OptionIndex == -1) {
|
|
|
|
Status = EfiBootManagerAddLoadOptionVariable (&NewOption, MAX_UINTN);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
}
|
|
|
|
EfiBootManagerFreeLoadOption (&NewOption);
|
|
|
|
EfiBootManagerFreeLoadOptions (BootOptions, BootOptionCount);
|
|
|
|
}
|
|
|
|
|
2016-07-08 02:49:45 +02:00
|
|
|
/**
|
|
|
|
Remove all MemoryMapped(...)/FvFile(...) and Fv(...)/FvFile(...) boot options
|
|
|
|
whose device paths do not resolve exactly to an FvFile in the system.
|
|
|
|
|
|
|
|
This removes any boot options that point to binaries built into the firmware
|
|
|
|
and have become stale due to any of the following:
|
|
|
|
- DXEFV's base address or size changed (historical),
|
|
|
|
- DXEFV's FvNameGuid changed,
|
|
|
|
- the FILE_GUID of the pointed-to binary changed,
|
|
|
|
- the referenced binary is no longer built into the firmware.
|
|
|
|
|
|
|
|
EfiBootManagerFindLoadOption() used in PlatformRegisterFvBootOption() only
|
|
|
|
avoids exact duplicates.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
RemoveStaleFvFileOptions (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_BOOT_MANAGER_LOAD_OPTION *BootOptions;
|
|
|
|
UINTN BootOptionCount;
|
|
|
|
UINTN Index;
|
|
|
|
|
|
|
|
BootOptions = EfiBootManagerGetLoadOptions (&BootOptionCount,
|
|
|
|
LoadOptionTypeBoot);
|
|
|
|
|
|
|
|
for (Index = 0; Index < BootOptionCount; ++Index) {
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *Node1, *Node2, *SearchNode;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_HANDLE FvHandle;
|
|
|
|
|
|
|
|
//
|
|
|
|
// If the device path starts with neither MemoryMapped(...) nor Fv(...),
|
|
|
|
// then keep the boot option.
|
|
|
|
//
|
|
|
|
Node1 = BootOptions[Index].FilePath;
|
|
|
|
if (!(DevicePathType (Node1) == HARDWARE_DEVICE_PATH &&
|
|
|
|
DevicePathSubType (Node1) == HW_MEMMAP_DP) &&
|
|
|
|
!(DevicePathType (Node1) == MEDIA_DEVICE_PATH &&
|
|
|
|
DevicePathSubType (Node1) == MEDIA_PIWG_FW_VOL_DP)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// If the second device path node is not FvFile(...), then keep the boot
|
|
|
|
// option.
|
|
|
|
//
|
|
|
|
Node2 = NextDevicePathNode (Node1);
|
|
|
|
if (DevicePathType (Node2) != MEDIA_DEVICE_PATH ||
|
|
|
|
DevicePathSubType (Node2) != MEDIA_PIWG_FW_FILE_DP) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Locate the Firmware Volume2 protocol instance that is denoted by the
|
|
|
|
// boot option. If this lookup fails (i.e., the boot option references a
|
|
|
|
// firmware volume that doesn't exist), then we'll proceed to delete the
|
|
|
|
// boot option.
|
|
|
|
//
|
|
|
|
SearchNode = Node1;
|
|
|
|
Status = gBS->LocateDevicePath (&gEfiFirmwareVolume2ProtocolGuid,
|
|
|
|
&SearchNode, &FvHandle);
|
|
|
|
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// The firmware volume was found; now let's see if it contains the FvFile
|
|
|
|
// identified by GUID.
|
|
|
|
//
|
|
|
|
EFI_FIRMWARE_VOLUME2_PROTOCOL *FvProtocol;
|
|
|
|
MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *FvFileNode;
|
|
|
|
UINTN BufferSize;
|
|
|
|
EFI_FV_FILETYPE FoundType;
|
|
|
|
EFI_FV_FILE_ATTRIBUTES FileAttributes;
|
|
|
|
UINT32 AuthenticationStatus;
|
|
|
|
|
|
|
|
Status = gBS->HandleProtocol (FvHandle, &gEfiFirmwareVolume2ProtocolGuid,
|
|
|
|
(VOID **)&FvProtocol);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
FvFileNode = (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *)Node2;
|
|
|
|
//
|
|
|
|
// Buffer==NULL means we request metadata only: BufferSize, FoundType,
|
|
|
|
// FileAttributes.
|
|
|
|
//
|
|
|
|
Status = FvProtocol->ReadFile (
|
|
|
|
FvProtocol,
|
|
|
|
&FvFileNode->FvFileName, // NameGuid
|
|
|
|
NULL, // Buffer
|
|
|
|
&BufferSize,
|
|
|
|
&FoundType,
|
|
|
|
&FileAttributes,
|
|
|
|
&AuthenticationStatus
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// The FvFile was found. Keep the boot option.
|
|
|
|
//
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Delete the boot option.
|
|
|
|
//
|
|
|
|
Status = EfiBootManagerDeleteLoadOptionVariable (
|
|
|
|
BootOptions[Index].OptionNumber, LoadOptionTypeBoot);
|
|
|
|
DEBUG_CODE (
|
|
|
|
CHAR16 *DevicePathString;
|
|
|
|
|
|
|
|
DevicePathString = ConvertDevicePathToText(BootOptions[Index].FilePath,
|
|
|
|
FALSE, FALSE);
|
|
|
|
DEBUG ((
|
2020-04-29 23:53:27 +02:00
|
|
|
EFI_ERROR (Status) ? DEBUG_WARN : DEBUG_VERBOSE,
|
2016-07-08 02:49:45 +02:00
|
|
|
"%a: removing stale Boot#%04x %s: %r\n",
|
|
|
|
__FUNCTION__,
|
|
|
|
(UINT32)BootOptions[Index].OptionNumber,
|
|
|
|
DevicePathString == NULL ? L"<unavailable>" : DevicePathString,
|
|
|
|
Status
|
|
|
|
));
|
|
|
|
if (DevicePathString != NULL) {
|
|
|
|
FreePool (DevicePathString);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
EfiBootManagerFreeLoadOptions (BootOptions, BootOptionCount);
|
|
|
|
}
|
|
|
|
|
2016-04-20 11:49:15 +02:00
|
|
|
VOID
|
|
|
|
PlatformRegisterOptionsAndKeys (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_INPUT_KEY Enter;
|
|
|
|
EFI_INPUT_KEY F2;
|
|
|
|
EFI_INPUT_KEY Esc;
|
|
|
|
EFI_BOOT_MANAGER_LOAD_OPTION BootOption;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Register ENTER as CONTINUE key
|
|
|
|
//
|
|
|
|
Enter.ScanCode = SCAN_NULL;
|
|
|
|
Enter.UnicodeChar = CHAR_CARRIAGE_RETURN;
|
|
|
|
Status = EfiBootManagerRegisterContinueKeyOption (0, &Enter, NULL);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Map F2 to Boot Manager Menu
|
|
|
|
//
|
|
|
|
F2.ScanCode = SCAN_F2;
|
|
|
|
F2.UnicodeChar = CHAR_NULL;
|
|
|
|
Esc.ScanCode = SCAN_ESC;
|
|
|
|
Esc.UnicodeChar = CHAR_NULL;
|
|
|
|
Status = EfiBootManagerGetBootManagerMenu (&BootOption);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
Status = EfiBootManagerAddKeyOptionVariable (
|
|
|
|
NULL, (UINT16) BootOption.OptionNumber, 0, &F2, NULL
|
|
|
|
);
|
|
|
|
ASSERT (Status == EFI_SUCCESS || Status == EFI_ALREADY_STARTED);
|
|
|
|
Status = EfiBootManagerAddKeyOptionVariable (
|
|
|
|
NULL, (UINT16) BootOption.OptionNumber, 0, &Esc, NULL
|
|
|
|
);
|
|
|
|
ASSERT (Status == EFI_SUCCESS || Status == EFI_ALREADY_STARTED);
|
|
|
|
}
|
|
|
|
|
2015-08-03 07:26:01 +02:00
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
ConnectRootBridge (
|
|
|
|
IN EFI_HANDLE RootBridgeHandle,
|
|
|
|
IN VOID *Instance,
|
|
|
|
IN VOID *Context
|
|
|
|
);
|
|
|
|
|
2018-05-17 21:51:11 +02:00
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
ConnectVirtioPciRng (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN VOID *Instance,
|
|
|
|
IN VOID *Context
|
|
|
|
);
|
|
|
|
|
2015-08-03 07:26:01 +02:00
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
SaveS3BootScript (
|
|
|
|
VOID
|
|
|
|
);
|
|
|
|
|
|
|
|
//
|
|
|
|
// BDS Platform Functions
|
|
|
|
//
|
2018-03-15 14:57:39 +01:00
|
|
|
/**
|
|
|
|
Do the platform init, can be customized by OEM/IBV
|
|
|
|
|
|
|
|
Possible things that can be done in PlatformBootManagerBeforeConsole:
|
|
|
|
|
|
|
|
> Update console variable: 1. include hot-plug devices;
|
|
|
|
> 2. Clear ConIn and add SOL for AMT
|
|
|
|
> Register new Driver#### or Boot####
|
|
|
|
> Register new Key####: e.g.: F12
|
|
|
|
> Signal ReadyToLock event
|
|
|
|
> Authentication action: 1. connect Auth devices;
|
|
|
|
> 2. Identify auto logon user.
|
|
|
|
**/
|
2015-08-03 07:26:01 +02:00
|
|
|
VOID
|
|
|
|
EFIAPI
|
2016-04-20 09:46:46 +02:00
|
|
|
PlatformBootManagerBeforeConsole (
|
2015-08-03 07:26:01 +02:00
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
2016-10-21 11:59:36 +02:00
|
|
|
EFI_HANDLE Handle;
|
|
|
|
EFI_STATUS Status;
|
2020-03-04 10:44:12 +01:00
|
|
|
UINT16 FrontPageTimeout;
|
2016-10-21 11:59:36 +02:00
|
|
|
RETURN_STATUS PcdStatus;
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "PlatformBootManagerBeforeConsole\n"));
|
2015-08-03 07:26:01 +02:00
|
|
|
InstallDevicePathCallback ();
|
|
|
|
|
|
|
|
VisitAllInstancesOfProtocol (&gEfiPciRootBridgeIoProtocolGuid,
|
|
|
|
ConnectRootBridge, NULL);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Signal the ACPI platform driver that it can download QEMU ACPI tables.
|
|
|
|
//
|
|
|
|
EfiEventGroupSignal (&gRootBridgesConnectedEventGroupGuid);
|
|
|
|
|
|
|
|
//
|
|
|
|
// We can't signal End-of-Dxe earlier than this. Namely, End-of-Dxe triggers
|
|
|
|
// the preparation of S3 system information. That logic has a hard dependency
|
|
|
|
// on the presence of the FACS ACPI table. Since our ACPI tables are only
|
|
|
|
// installed after PCI enumeration completes, we must not trigger the S3 save
|
|
|
|
// earlier, hence we can't signal End-of-Dxe earlier.
|
|
|
|
//
|
|
|
|
EfiEventGroupSignal (&gEfiEndOfDxeEventGroupGuid);
|
|
|
|
|
2021-08-31 03:31:14 +02:00
|
|
|
if (PcdGetBool (PcdAcpiS3Enable)) {
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
|
|
|
// Save the boot script too. Note that this will require us to emit the
|
|
|
|
// DxeSmmReadyToLock event just below, which in turn locks down SMM.
|
|
|
|
//
|
|
|
|
SaveS3BootScript ();
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Prevent further changes to LockBoxes or SMRAM.
|
|
|
|
//
|
|
|
|
Handle = NULL;
|
|
|
|
Status = gBS->InstallProtocolInterface (&Handle,
|
|
|
|
&gEfiDxeSmmReadyToLockProtocolGuid, EFI_NATIVE_INTERFACE,
|
|
|
|
NULL);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
2016-04-20 11:36:14 +02:00
|
|
|
|
2016-11-08 13:16:39 +01:00
|
|
|
//
|
2018-03-15 14:19:22 +01:00
|
|
|
// Dispatch deferred images after EndOfDxe event and ReadyToLock
|
|
|
|
// installation.
|
2016-11-08 13:16:39 +01:00
|
|
|
//
|
|
|
|
EfiBootManagerDispatchDeferredImages ();
|
|
|
|
|
2019-08-13 13:31:15 +02:00
|
|
|
PlatformInitializeConsole (
|
|
|
|
XenDetected() ? gXenPlatformConsole : gPlatformConsole);
|
2020-03-04 10:44:12 +01:00
|
|
|
|
|
|
|
FrontPageTimeout = GetFrontPageTimeoutFromQemu ();
|
|
|
|
PcdStatus = PcdSet16S (PcdPlatformBootTimeOut, FrontPageTimeout);
|
2016-10-21 11:59:36 +02:00
|
|
|
ASSERT_RETURN_ERROR (PcdStatus);
|
2020-03-04 10:44:12 +01:00
|
|
|
//
|
|
|
|
// Reflect the PCD in the standard Timeout variable.
|
|
|
|
//
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
EFI_TIME_OUT_VARIABLE_NAME,
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
(EFI_VARIABLE_NON_VOLATILE |
|
|
|
|
EFI_VARIABLE_BOOTSERVICE_ACCESS |
|
|
|
|
EFI_VARIABLE_RUNTIME_ACCESS),
|
|
|
|
sizeof FrontPageTimeout,
|
|
|
|
&FrontPageTimeout
|
|
|
|
);
|
|
|
|
DEBUG ((
|
|
|
|
EFI_ERROR (Status) ? DEBUG_ERROR : DEBUG_VERBOSE,
|
|
|
|
"%a: SetVariable(%s, %u): %r\n",
|
|
|
|
__FUNCTION__,
|
|
|
|
EFI_TIME_OUT_VARIABLE_NAME,
|
|
|
|
FrontPageTimeout,
|
|
|
|
Status
|
|
|
|
));
|
2016-04-20 11:49:15 +02:00
|
|
|
|
|
|
|
PlatformRegisterOptionsAndKeys ();
|
2018-05-17 21:51:11 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Install both VIRTIO_DEVICE_PROTOCOL and (dependent) EFI_RNG_PROTOCOL
|
|
|
|
// instances on Virtio PCI RNG devices.
|
|
|
|
//
|
|
|
|
VisitAllInstancesOfProtocol (&gEfiPciIoProtocolGuid, ConnectVirtioPciRng,
|
|
|
|
NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
ConnectRootBridge (
|
|
|
|
IN EFI_HANDLE RootBridgeHandle,
|
|
|
|
IN VOID *Instance,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Make the PCI bus driver connect the root bridge, non-recursively. This
|
|
|
|
// will produce a number of child handles with PciIo on them.
|
|
|
|
//
|
|
|
|
Status = gBS->ConnectController (
|
|
|
|
RootBridgeHandle, // ControllerHandle
|
|
|
|
NULL, // DriverImageHandle
|
|
|
|
NULL, // RemainingDevicePath -- produce all
|
|
|
|
// children
|
|
|
|
FALSE // Recursive
|
|
|
|
);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-17 21:51:11 +02:00
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
ConnectVirtioPciRng (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN VOID *Instance,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_PCI_IO_PROTOCOL *PciIo;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT16 VendorId;
|
|
|
|
UINT16 DeviceId;
|
|
|
|
UINT8 RevisionId;
|
|
|
|
BOOLEAN Virtio10;
|
|
|
|
UINT16 SubsystemId;
|
|
|
|
|
|
|
|
PciIo = Instance;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Read and check VendorId.
|
|
|
|
//
|
|
|
|
Status = PciIo->Pci.Read (PciIo, EfiPciIoWidthUint16, PCI_VENDOR_ID_OFFSET,
|
|
|
|
1, &VendorId);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
if (VendorId != VIRTIO_VENDOR_ID) {
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Read DeviceId and RevisionId.
|
|
|
|
//
|
|
|
|
Status = PciIo->Pci.Read (PciIo, EfiPciIoWidthUint16, PCI_DEVICE_ID_OFFSET,
|
|
|
|
1, &DeviceId);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
Status = PciIo->Pci.Read (PciIo, EfiPciIoWidthUint8, PCI_REVISION_ID_OFFSET,
|
|
|
|
1, &RevisionId);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// From DeviceId and RevisionId, determine whether the device is a
|
|
|
|
// modern-only Virtio 1.0 device. In case of Virtio 1.0, DeviceId can
|
|
|
|
// immediately be restricted to VIRTIO_SUBSYSTEM_ENTROPY_SOURCE, and
|
|
|
|
// SubsystemId will only play a sanity-check role. Otherwise, DeviceId can
|
|
|
|
// only be sanity-checked, and SubsystemId will decide.
|
|
|
|
//
|
|
|
|
if (DeviceId == 0x1040 + VIRTIO_SUBSYSTEM_ENTROPY_SOURCE &&
|
|
|
|
RevisionId >= 0x01) {
|
|
|
|
Virtio10 = TRUE;
|
|
|
|
} else if (DeviceId >= 0x1000 && DeviceId <= 0x103F && RevisionId == 0x00) {
|
|
|
|
Virtio10 = FALSE;
|
|
|
|
} else {
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Read and check SubsystemId as dictated by Virtio10.
|
|
|
|
//
|
|
|
|
Status = PciIo->Pci.Read (PciIo, EfiPciIoWidthUint16,
|
|
|
|
PCI_SUBSYSTEM_ID_OFFSET, 1, &SubsystemId);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
if ((Virtio10 && SubsystemId >= 0x40) ||
|
|
|
|
(!Virtio10 && SubsystemId == VIRTIO_SUBSYSTEM_ENTROPY_SOURCE)) {
|
|
|
|
Status = gBS->ConnectController (
|
|
|
|
Handle, // ControllerHandle
|
|
|
|
NULL, // DriverImageHandle -- connect all drivers
|
|
|
|
NULL, // RemainingDevicePath -- produce all child handles
|
|
|
|
FALSE // Recursive -- don't follow child handles
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
|
|
|
|
Error:
|
|
|
|
DEBUG ((DEBUG_ERROR, "%a: %r\n", __FUNCTION__, Status));
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
/**
|
|
|
|
Add IsaKeyboard to ConIn; add IsaSerial to ConOut, ConIn, ErrOut.
|
|
|
|
|
|
|
|
@param[in] DeviceHandle Handle of the LPC Bridge device.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Console devices on the LPC bridge have been added to
|
|
|
|
ConOut, ConIn, and ErrOut.
|
|
|
|
|
|
|
|
@return Error codes, due to EFI_DEVICE_PATH_PROTOCOL missing
|
|
|
|
from DeviceHandle.
|
|
|
|
**/
|
2015-08-03 07:26:01 +02:00
|
|
|
EFI_STATUS
|
|
|
|
PrepareLpcBridgeDevicePath (
|
|
|
|
IN EFI_HANDLE DeviceHandle
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *TempDevicePath;
|
|
|
|
CHAR16 *DevPathStr;
|
|
|
|
|
|
|
|
DevicePath = NULL;
|
|
|
|
Status = gBS->HandleProtocol (
|
|
|
|
DeviceHandle,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
(VOID*)&DevicePath
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
TempDevicePath = DevicePath;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Register Keyboard
|
|
|
|
//
|
2018-03-15 14:19:22 +01:00
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gPnpPs2KeyboardDeviceNode);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2016-04-20 11:19:14 +02:00
|
|
|
EfiBootManagerUpdateConsoleVariable (ConIn, DevicePath, NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Register COM1
|
|
|
|
//
|
|
|
|
DevicePath = TempDevicePath;
|
|
|
|
gPnp16550ComPortDeviceNode.UID = 0;
|
|
|
|
|
2018-03-15 14:19:22 +01:00
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gPnp16550ComPortDeviceNode);
|
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gUartDeviceNode);
|
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gTerminalTypeDeviceNode);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Print Device Path
|
|
|
|
//
|
2016-04-20 11:24:38 +02:00
|
|
|
DevPathStr = ConvertDevicePathToText (DevicePath, FALSE, FALSE);
|
2015-08-03 07:26:01 +02:00
|
|
|
if (DevPathStr != NULL) {
|
|
|
|
DEBUG((
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG_INFO,
|
2015-08-03 07:26:01 +02:00
|
|
|
"BdsPlatform.c+%d: COM%d DevPath: %s\n",
|
|
|
|
__LINE__,
|
|
|
|
gPnp16550ComPortDeviceNode.UID + 1,
|
|
|
|
DevPathStr
|
|
|
|
));
|
|
|
|
FreePool(DevPathStr);
|
|
|
|
}
|
|
|
|
|
2016-04-20 11:19:14 +02:00
|
|
|
EfiBootManagerUpdateConsoleVariable (ConOut, DevicePath, NULL);
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ConIn, DevicePath, NULL);
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ErrOut, DevicePath, NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Register COM2
|
|
|
|
//
|
|
|
|
DevicePath = TempDevicePath;
|
|
|
|
gPnp16550ComPortDeviceNode.UID = 1;
|
|
|
|
|
2018-03-15 14:19:22 +01:00
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gPnp16550ComPortDeviceNode);
|
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gUartDeviceNode);
|
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gTerminalTypeDeviceNode);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Print Device Path
|
|
|
|
//
|
2016-04-20 11:24:38 +02:00
|
|
|
DevPathStr = ConvertDevicePathToText (DevicePath, FALSE, FALSE);
|
2015-08-03 07:26:01 +02:00
|
|
|
if (DevPathStr != NULL) {
|
|
|
|
DEBUG((
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG_INFO,
|
2015-08-03 07:26:01 +02:00
|
|
|
"BdsPlatform.c+%d: COM%d DevPath: %s\n",
|
|
|
|
__LINE__,
|
|
|
|
gPnp16550ComPortDeviceNode.UID + 1,
|
|
|
|
DevPathStr
|
|
|
|
));
|
|
|
|
FreePool(DevPathStr);
|
|
|
|
}
|
|
|
|
|
2016-04-20 11:19:14 +02:00
|
|
|
EfiBootManagerUpdateConsoleVariable (ConOut, DevicePath, NULL);
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ConIn, DevicePath, NULL);
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ErrOut, DevicePath, NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
GetGopDevicePath (
|
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *PciDevicePath,
|
|
|
|
OUT EFI_DEVICE_PATH_PROTOCOL **GopDevicePath
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN Index;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_HANDLE PciDeviceHandle;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *TempDevicePath;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *TempPciDevicePath;
|
|
|
|
UINTN GopHandleCount;
|
|
|
|
EFI_HANDLE *GopHandleBuffer;
|
|
|
|
|
|
|
|
if (PciDevicePath == NULL || GopDevicePath == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Initialize the GopDevicePath to be PciDevicePath
|
|
|
|
//
|
|
|
|
*GopDevicePath = PciDevicePath;
|
|
|
|
TempPciDevicePath = PciDevicePath;
|
|
|
|
|
|
|
|
Status = gBS->LocateDevicePath (
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
&TempPciDevicePath,
|
|
|
|
&PciDeviceHandle
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2016-10-19 09:01:31 +02:00
|
|
|
// Try to connect this handle, so that GOP driver could start on this
|
2015-08-03 07:26:01 +02:00
|
|
|
// device and create child handles with GraphicsOutput Protocol installed
|
|
|
|
// on them, then we get device paths of these child handles and select
|
|
|
|
// them as possible console device.
|
|
|
|
//
|
|
|
|
gBS->ConnectController (PciDeviceHandle, NULL, NULL, FALSE);
|
|
|
|
|
|
|
|
Status = gBS->LocateHandleBuffer (
|
|
|
|
ByProtocol,
|
|
|
|
&gEfiGraphicsOutputProtocolGuid,
|
|
|
|
NULL,
|
|
|
|
&GopHandleCount,
|
|
|
|
&GopHandleBuffer
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// Add all the child handles as possible Console Device
|
|
|
|
//
|
|
|
|
for (Index = 0; Index < GopHandleCount; Index++) {
|
2018-03-15 14:19:22 +01:00
|
|
|
Status = gBS->HandleProtocol (GopHandleBuffer[Index],
|
|
|
|
&gEfiDevicePathProtocolGuid, (VOID*)&TempDevicePath);
|
2015-08-03 07:26:01 +02:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (CompareMem (
|
|
|
|
PciDevicePath,
|
|
|
|
TempDevicePath,
|
|
|
|
GetDevicePathSize (PciDevicePath) - END_DEVICE_PATH_LENGTH
|
|
|
|
) == 0) {
|
|
|
|
//
|
|
|
|
// In current implementation, we only enable one of the child handles
|
|
|
|
// as console device, i.e. sotre one of the child handle's device
|
|
|
|
// path to variable "ConOut"
|
2016-10-19 09:01:31 +02:00
|
|
|
// In future, we could select all child handles to be console device
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
|
|
|
|
|
|
|
*GopDevicePath = TempDevicePath;
|
|
|
|
|
|
|
|
//
|
2018-03-15 14:19:22 +01:00
|
|
|
// Delete the PCI device's path that added by
|
|
|
|
// GetPlugInPciVgaDevicePath(). Add the integrity GOP device path.
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2016-04-20 11:19:14 +02:00
|
|
|
EfiBootManagerUpdateConsoleVariable (ConOutDev, NULL, PciDevicePath);
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ConOutDev, TempDevicePath, NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
gBS->FreePool (GopHandleBuffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
/**
|
|
|
|
Add PCI display to ConOut.
|
|
|
|
|
|
|
|
@param[in] DeviceHandle Handle of the PCI display device.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The PCI display device has been added to ConOut.
|
|
|
|
|
|
|
|
@return Error codes, due to EFI_DEVICE_PATH_PROTOCOL missing
|
|
|
|
from DeviceHandle.
|
|
|
|
**/
|
2015-08-03 07:26:01 +02:00
|
|
|
EFI_STATUS
|
2016-08-18 11:33:39 +02:00
|
|
|
PreparePciDisplayDevicePath (
|
2015-08-03 07:26:01 +02:00
|
|
|
IN EFI_HANDLE DeviceHandle
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *GopDevicePath;
|
|
|
|
|
|
|
|
DevicePath = NULL;
|
|
|
|
GopDevicePath = NULL;
|
|
|
|
Status = gBS->HandleProtocol (
|
|
|
|
DeviceHandle,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
(VOID*)&DevicePath
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
GetGopDevicePath (DevicePath, &GopDevicePath);
|
|
|
|
DevicePath = GopDevicePath;
|
|
|
|
|
2016-04-20 11:19:14 +02:00
|
|
|
EfiBootManagerUpdateConsoleVariable (ConOut, DevicePath, NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
/**
|
2015-08-03 07:26:01 +02:00
|
|
|
Add PCI Serial to ConOut, ConIn, ErrOut.
|
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
@param[in] DeviceHandle Handle of the PCI serial device.
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
@retval EFI_SUCCESS The PCI serial device has been added to ConOut, ConIn,
|
|
|
|
ErrOut.
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
@return Error codes, due to EFI_DEVICE_PATH_PROTOCOL missing
|
|
|
|
from DeviceHandle.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
PreparePciSerialDevicePath (
|
|
|
|
IN EFI_HANDLE DeviceHandle
|
|
|
|
)
|
2015-08-03 07:26:01 +02:00
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
|
|
|
|
|
|
|
DevicePath = NULL;
|
|
|
|
Status = gBS->HandleProtocol (
|
|
|
|
DeviceHandle,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
(VOID*)&DevicePath
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2018-03-15 14:19:22 +01:00
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gUartDeviceNode);
|
|
|
|
DevicePath = AppendDevicePathNode (DevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *)&gTerminalTypeDeviceNode);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2016-04-20 11:19:14 +02:00
|
|
|
EfiBootManagerUpdateConsoleVariable (ConOut, DevicePath, NULL);
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ConIn, DevicePath, NULL);
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ErrOut, DevicePath, NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
VisitAllInstancesOfProtocol (
|
|
|
|
IN EFI_GUID *Id,
|
|
|
|
IN PROTOCOL_INSTANCE_CALLBACK CallBackFunction,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN HandleCount;
|
|
|
|
EFI_HANDLE *HandleBuffer;
|
|
|
|
UINTN Index;
|
|
|
|
VOID *Instance;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Start to check all the PciIo to find all possible device
|
|
|
|
//
|
|
|
|
HandleCount = 0;
|
|
|
|
HandleBuffer = NULL;
|
|
|
|
Status = gBS->LocateHandleBuffer (
|
|
|
|
ByProtocol,
|
|
|
|
Id,
|
|
|
|
NULL,
|
|
|
|
&HandleCount,
|
|
|
|
&HandleBuffer
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Index = 0; Index < HandleCount; Index++) {
|
|
|
|
Status = gBS->HandleProtocol (HandleBuffer[Index], Id, &Instance);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = (*CallBackFunction) (
|
|
|
|
HandleBuffer[Index],
|
|
|
|
Instance,
|
|
|
|
Context
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
gBS->FreePool (HandleBuffer);
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
VisitingAPciInstance (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN VOID *Instance,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_PCI_IO_PROTOCOL *PciIo;
|
|
|
|
PCI_TYPE00 Pci;
|
|
|
|
|
|
|
|
PciIo = (EFI_PCI_IO_PROTOCOL*) Instance;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Check for all PCI device
|
|
|
|
//
|
|
|
|
Status = PciIo->Pci.Read (
|
|
|
|
PciIo,
|
|
|
|
EfiPciIoWidthUint32,
|
|
|
|
0,
|
|
|
|
sizeof (Pci) / sizeof (UINT32),
|
|
|
|
&Pci
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (*(VISIT_PCI_INSTANCE_CALLBACK)(UINTN) Context) (
|
|
|
|
Handle,
|
|
|
|
PciIo,
|
|
|
|
&Pci
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
VisitAllPciInstances (
|
|
|
|
IN VISIT_PCI_INSTANCE_CALLBACK CallBackFunction
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return VisitAllInstancesOfProtocol (
|
|
|
|
&gEfiPciIoProtocolGuid,
|
|
|
|
VisitingAPciInstance,
|
|
|
|
(VOID*)(UINTN) CallBackFunction
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Do platform specific PCI Device check and add them to
|
|
|
|
ConOut, ConIn, ErrOut.
|
|
|
|
|
|
|
|
@param[in] Handle - Handle of PCI device instance
|
|
|
|
@param[in] PciIo - PCI IO protocol instance
|
|
|
|
@param[in] Pci - PCI Header register block
|
|
|
|
|
2018-03-15 14:19:22 +01:00
|
|
|
@retval EFI_SUCCESS - PCI Device check and Console variable update
|
|
|
|
successfully.
|
2015-08-03 07:26:01 +02:00
|
|
|
@retval EFI_STATUS - PCI Device check or Console variable update fail.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
DetectAndPreparePlatformPciDevicePath (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN EFI_PCI_IO_PROTOCOL *PciIo,
|
|
|
|
IN PCI_TYPE00 *Pci
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
Status = PciIo->Attributes (
|
|
|
|
PciIo,
|
|
|
|
EfiPciIoAttributeOperationEnable,
|
|
|
|
EFI_PCI_DEVICE_ENABLE,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
2018-05-13 00:13:53 +02:00
|
|
|
//
|
|
|
|
// Here we decide whether it is LPC Bridge
|
|
|
|
//
|
|
|
|
if ((IS_PCI_LPC (Pci)) ||
|
|
|
|
((IS_PCI_ISA_PDECODE (Pci)) &&
|
|
|
|
(Pci->Hdr.VendorId == 0x8086) &&
|
|
|
|
(Pci->Hdr.DeviceId == 0x7000)
|
|
|
|
)
|
|
|
|
) {
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2018-05-13 00:13:53 +02:00
|
|
|
// Add IsaKeyboard to ConIn,
|
|
|
|
// add IsaSerial to ConOut, ConIn, ErrOut
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "Found LPC Bridge device\n"));
|
2018-05-13 00:13:53 +02:00
|
|
|
PrepareLpcBridgeDevicePath (Handle);
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// Here we decide which Serial device to enable in PCI bus
|
|
|
|
//
|
|
|
|
if (IS_PCI_16550SERIAL (Pci)) {
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2018-05-13 00:13:53 +02:00
|
|
|
// Add them to ConOut, ConIn, ErrOut.
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "Found PCI 16550 SERIAL device\n"));
|
2018-05-13 00:13:53 +02:00
|
|
|
PreparePciSerialDevicePath (Handle);
|
|
|
|
return EFI_SUCCESS;
|
2015-08-03 07:26:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2016-08-18 11:33:39 +02:00
|
|
|
// Here we decide which display device to enable in PCI bus
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2016-08-18 11:33:39 +02:00
|
|
|
if (IS_PCI_DISPLAY (Pci)) {
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
|
|
|
// Add them to ConOut.
|
|
|
|
//
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "Found PCI display device\n"));
|
2016-08-18 11:33:39 +02:00
|
|
|
PreparePciDisplayDevicePath (Handle);
|
2015-08-03 07:26:01 +02:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
/**
|
|
|
|
Connect the predefined platform default console device.
|
|
|
|
|
|
|
|
Always try to find and enable PCI display devices.
|
|
|
|
|
|
|
|
@param[in] PlatformConsole Predefined platform default console device array.
|
|
|
|
**/
|
2016-04-20 11:36:14 +02:00
|
|
|
VOID
|
|
|
|
PlatformInitializeConsole (
|
|
|
|
IN PLATFORM_CONSOLE_CONNECT_ENTRY *PlatformConsole
|
2015-08-03 07:26:01 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN Index;
|
|
|
|
|
|
|
|
//
|
2018-05-13 00:13:53 +02:00
|
|
|
// Do platform specific PCI Device check and add them to ConOut, ConIn,
|
|
|
|
// ErrOut
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2018-05-13 00:13:53 +02:00
|
|
|
VisitAllPciInstances (DetectAndPreparePlatformPciDevicePath);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2018-05-13 00:13:53 +02:00
|
|
|
//
|
|
|
|
// Have chance to connect the platform default console,
|
|
|
|
// the platform default console is the minimum device group
|
|
|
|
// the platform should support
|
|
|
|
//
|
|
|
|
for (Index = 0; PlatformConsole[Index].DevicePath != NULL; ++Index) {
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2018-05-13 00:13:53 +02:00
|
|
|
// Update the console variable with the connect type
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2018-05-13 00:13:53 +02:00
|
|
|
if ((PlatformConsole[Index].ConnectType & CONSOLE_IN) == CONSOLE_IN) {
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ConIn,
|
|
|
|
PlatformConsole[Index].DevicePath, NULL);
|
|
|
|
}
|
|
|
|
if ((PlatformConsole[Index].ConnectType & CONSOLE_OUT) == CONSOLE_OUT) {
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ConOut,
|
|
|
|
PlatformConsole[Index].DevicePath, NULL);
|
|
|
|
}
|
|
|
|
if ((PlatformConsole[Index].ConnectType & STD_ERROR) == STD_ERROR) {
|
|
|
|
EfiBootManagerUpdateConsoleVariable (ErrOut,
|
|
|
|
PlatformConsole[Index].DevicePath, NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Configure PCI Interrupt Line register for applicable devices
|
|
|
|
Ported from SeaBIOS, src/fw/pciinit.c, *_pci_slot_get_irq()
|
|
|
|
|
|
|
|
@param[in] Handle - Handle of PCI device instance
|
|
|
|
@param[in] PciIo - PCI IO protocol instance
|
|
|
|
@param[in] PciHdr - PCI Header register block
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS - PCI Interrupt Line register configured successfully.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
SetPciIntLine (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN EFI_PCI_IO_PROTOCOL *PciIo,
|
|
|
|
IN PCI_TYPE00 *PciHdr
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevPathNode;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevPath;
|
|
|
|
UINTN RootSlot;
|
|
|
|
UINTN Idx;
|
|
|
|
UINT8 IrqLine;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 RootBusNumber;
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
if (PciHdr->Device.InterruptPin != 0) {
|
|
|
|
|
|
|
|
DevPathNode = DevicePathFromHandle (Handle);
|
|
|
|
ASSERT (DevPathNode != NULL);
|
|
|
|
DevPath = DevPathNode;
|
|
|
|
|
|
|
|
RootBusNumber = 0;
|
|
|
|
if (DevicePathType (DevPathNode) == ACPI_DEVICE_PATH &&
|
|
|
|
DevicePathSubType (DevPathNode) == ACPI_DP &&
|
|
|
|
((ACPI_HID_DEVICE_PATH *)DevPathNode)->HID == EISA_PNP_ID(0x0A03)) {
|
|
|
|
RootBusNumber = ((ACPI_HID_DEVICE_PATH *)DevPathNode)->UID;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Compute index into PciHostIrqs[] table by walking
|
|
|
|
// the device path and adding up all device numbers
|
|
|
|
//
|
|
|
|
Status = EFI_NOT_FOUND;
|
|
|
|
RootSlot = 0;
|
|
|
|
Idx = PciHdr->Device.InterruptPin - 1;
|
|
|
|
while (!IsDevicePathEnd (DevPathNode)) {
|
|
|
|
if (DevicePathType (DevPathNode) == HARDWARE_DEVICE_PATH &&
|
|
|
|
DevicePathSubType (DevPathNode) == HW_PCI_DP) {
|
|
|
|
|
|
|
|
Idx += ((PCI_DEVICE_PATH *)DevPathNode)->Device;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Unlike SeaBIOS, which starts climbing from the leaf device
|
|
|
|
// up toward the root, we traverse the device path starting at
|
|
|
|
// the root moving toward the leaf node.
|
|
|
|
// The slot number of the top-level parent bridge is needed for
|
|
|
|
// Q35 cases with more than 24 slots on the root bus.
|
|
|
|
//
|
|
|
|
if (Status != EFI_SUCCESS) {
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
RootSlot = ((PCI_DEVICE_PATH *)DevPathNode)->Device;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DevPathNode = NextDevicePathNode (DevPathNode);
|
|
|
|
}
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
if (RootBusNumber == 0 && RootSlot == 0) {
|
|
|
|
DEBUG((
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG_ERROR,
|
2015-08-03 07:26:01 +02:00
|
|
|
"%a: PCI host bridge (00:00.0) should have no interrupts!\n",
|
|
|
|
__FUNCTION__
|
|
|
|
));
|
|
|
|
ASSERT (FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Final PciHostIrqs[] index calculation depends on the platform
|
|
|
|
// and should match SeaBIOS src/fw/pciinit.c *_pci_slot_get_irq()
|
|
|
|
//
|
|
|
|
switch (mHostBridgeDevId) {
|
|
|
|
case INTEL_82441_DEVICE_ID:
|
|
|
|
Idx -= 1;
|
|
|
|
break;
|
|
|
|
case INTEL_Q35_MCH_DEVICE_ID:
|
|
|
|
//
|
|
|
|
// SeaBIOS contains the following comment:
|
|
|
|
// "Slots 0-24 rotate slot:pin mapping similar to piix above, but
|
|
|
|
// with a different starting index - see q35-acpi-dsdt.dsl.
|
|
|
|
//
|
|
|
|
// Slots 25-31 all use LNKA mapping (or LNKE, but A:D = E:H)"
|
|
|
|
//
|
|
|
|
if (RootSlot > 24) {
|
|
|
|
//
|
|
|
|
// in this case, subtract back out RootSlot from Idx
|
|
|
|
// (SeaBIOS never adds it to begin with, but that would make our
|
|
|
|
// device path traversal loop above too awkward)
|
|
|
|
//
|
|
|
|
Idx -= RootSlot;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ASSERT (FALSE); // should never get here
|
|
|
|
}
|
|
|
|
Idx %= ARRAY_SIZE (PciHostIrqs);
|
|
|
|
IrqLine = PciHostIrqs[Idx];
|
|
|
|
|
|
|
|
DEBUG_CODE_BEGIN ();
|
|
|
|
{
|
|
|
|
CHAR16 *DevPathString;
|
|
|
|
STATIC CHAR16 Fallback[] = L"<failed to convert>";
|
|
|
|
UINTN Segment, Bus, Device, Function;
|
|
|
|
|
|
|
|
DevPathString = ConvertDevicePathToText (DevPath, FALSE, FALSE);
|
|
|
|
if (DevPathString == NULL) {
|
|
|
|
DevPathString = Fallback;
|
|
|
|
}
|
|
|
|
Status = PciIo->GetLocation (PciIo, &Segment, &Bus, &Device, &Function);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_VERBOSE, "%a: [%02x:%02x.%x] %s -> 0x%02x\n", __FUNCTION__,
|
2015-08-03 07:26:01 +02:00
|
|
|
(UINT32)Bus, (UINT32)Device, (UINT32)Function, DevPathString,
|
|
|
|
IrqLine));
|
|
|
|
|
|
|
|
if (DevPathString != Fallback) {
|
|
|
|
FreePool (DevPathString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DEBUG_CODE_END ();
|
|
|
|
|
|
|
|
//
|
|
|
|
// Set PCI Interrupt Line register for this device to PciHostIrqs[Idx]
|
|
|
|
//
|
|
|
|
Status = PciIo->Pci.Write (
|
|
|
|
PciIo,
|
|
|
|
EfiPciIoWidthUint8,
|
|
|
|
PCI_INT_LINE_OFFSET,
|
|
|
|
1,
|
|
|
|
&IrqLine
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VOID
|
|
|
|
PciAcpiInitialization (
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN Pmba;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Query Host Bridge DID to determine platform type
|
|
|
|
//
|
|
|
|
mHostBridgeDevId = PcdGet16 (PcdOvmfHostBridgePciDevId);
|
|
|
|
switch (mHostBridgeDevId) {
|
|
|
|
case INTEL_82441_DEVICE_ID:
|
|
|
|
Pmba = POWER_MGMT_REGISTER_PIIX4 (PIIX4_PMBA);
|
|
|
|
//
|
|
|
|
// 00:01.0 ISA Bridge (PIIX4) LNK routing targets
|
|
|
|
//
|
OvmfPkg/PlatformBootManagerLib: fix PCI interrupt link (LNKx)
This patch fixes an issue with the current programming of the i440fx
PCI Interrupt routing assignment.
Explanation by Laszlo Ersek:
(1) The rotating pattern is a map:
(slot, function) --> (interrupt link) [LNKA..LNKD]
(more precisely, it is a pattern from (slot, pin) to (interrupt link),
but function<->pin is an identity mapping in the QEMU hardware, so we
can just use (slot, function) rather than (slot, pin) on the left hand
side. But I digress.)
The ACPI _PRT object is generated by QEMU; it describes this map.
(2) Another map is
(interrupt link) --> { set of possible interrupt numbers,
for this link }
This map is given by the LNK[A..D] ACPI objects, also given by QEMU.
(3) What the firmware is expected to do is:
(3a) for each interrupt link, select an *actual* interrupt from the set
that's possible for that link, yielding a deterministic map
(interrupt link) --> (actual interrupt number)
and
(3b) for each PCI device/function with an interrupt pin, resolve the
(slot, function) --> (interrupt link) --> (actual interrupt number)
functional composition, and program the result into the Interrupt Line
register of the device.
In OVMF, we do not parse the rotating map described under (1) from
QEMU's _PRT object. Instead, we duplicate the code. This is not a
problem.
In OVMF, we also do not parse the map described under (2) from QEMU's
ACPI content. Instead, we pick a specific selection (3a) that we
"apriori" know satisfies (2). This is also not a problem. OVMF's
particular selection is the PciHostIrqs table.
(
Table (2) from QEMU is
LNKA -> { 5, 10, 11 }
LNKB -> { 5, 10, 11 }
LNKC -> { 5, 10, 11 }
LNKD -> { 5, 10, 11 }
and our specific pick in OVMF, in the PciHostIrqs table, is
LNKA -> 10
LNKB -> 10
LNKC -> 11
LNKD -> 11
)
In OVMF, we also cover step (3b), in the SetPciIntLine() function.
What's missing in OVMF -- and what this patch corrects -- is that we
currently fail to program our selection for table (3) into the hardware.
We pick a specific LNKx->IRQ# mapping for each interrupt link, and we
correctly program the PCI Interrupt Line registers through those
link-to-IRQ mappings -- but we don't tell the hardware about the
link-to-IRQ mappings. More precisely, we program such a link-to-IRQ
mapping table into the hardware that is then not matched by the mapping
we use for programming the PCI device/function interrupt lines. As a
result, some PCI Interrupt Line registers will have impossible values --
a given (slot, function) may use a particular link, but also report an
interrupt number that was never picked for that link.
Output of Linux PCI Interrupt Links for i440fx before the patch:
[ 0.327305] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 10 *11)
[ 0.327944] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 10 *11)
[ 0.328582] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 *10 11)
[ 0.329208] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 *10 11)
[ 0.329807] ACPI: PCI Interrupt Link [LNKS] (IRQs *9)
after the patch:
[ 0.327292] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.327934] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.328564] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.329195] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.329785] ACPI: PCI Interrupt Link [LNKS] (IRQs *9)
Output of Linux PCI Interrupt Links for q35 before the patch:
[ 0.307474] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.308027] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.308764] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.309310] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.309853] ACPI: PCI Interrupt Link [LNKE] (IRQs 5 *10 11)
[ 0.310508] ACPI: PCI Interrupt Link [LNKF] (IRQs 5 *10 11)
[ 0.311051] ACPI: PCI Interrupt Link [LNKG] (IRQs 5 10 *11)
[ 0.311589] ACPI: PCI Interrupt Link [LNKH] (IRQs 5 10 *11)
after the patch:
[ 0.301991] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.302833] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.303354] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.303873] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.304399] ACPI: PCI Interrupt Link [LNKE] (IRQs 5 *10 11)
[ 0.304918] ACPI: PCI Interrupt Link [LNKF] (IRQs 5 *10 11)
[ 0.305436] ACPI: PCI Interrupt Link [LNKG] (IRQs 5 10 *11)
[ 0.305954] ACPI: PCI Interrupt Link [LNKH] (IRQs 5 10 *11)
Signed-off-by: Hendrik Borghorst <hborghor@amazon.de>
Reviewed-by: David Woodhouse <dwmw@amazon.co.uk>
Message-Id: <8dbedc4c7a1c3fd390aca915270814e3b35e13a5.camel@amazon.com>
Reviewed-by: Laszlo Ersek <lersek@redhat.com>
2020-12-17 15:12:22 +01:00
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 1, 0, 0x60), PciHostIrqs[0]); // A
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 1, 0, 0x61), PciHostIrqs[1]); // B
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 1, 0, 0x62), PciHostIrqs[2]); // C
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 1, 0, 0x63), PciHostIrqs[3]); // D
|
2015-08-03 07:26:01 +02:00
|
|
|
break;
|
|
|
|
case INTEL_Q35_MCH_DEVICE_ID:
|
|
|
|
Pmba = POWER_MGMT_REGISTER_Q35 (ICH9_PMBASE);
|
|
|
|
//
|
|
|
|
// 00:1f.0 LPC Bridge (Q35) LNK routing targets
|
|
|
|
//
|
OvmfPkg/PlatformBootManagerLib: fix PCI interrupt link (LNKx)
This patch fixes an issue with the current programming of the i440fx
PCI Interrupt routing assignment.
Explanation by Laszlo Ersek:
(1) The rotating pattern is a map:
(slot, function) --> (interrupt link) [LNKA..LNKD]
(more precisely, it is a pattern from (slot, pin) to (interrupt link),
but function<->pin is an identity mapping in the QEMU hardware, so we
can just use (slot, function) rather than (slot, pin) on the left hand
side. But I digress.)
The ACPI _PRT object is generated by QEMU; it describes this map.
(2) Another map is
(interrupt link) --> { set of possible interrupt numbers,
for this link }
This map is given by the LNK[A..D] ACPI objects, also given by QEMU.
(3) What the firmware is expected to do is:
(3a) for each interrupt link, select an *actual* interrupt from the set
that's possible for that link, yielding a deterministic map
(interrupt link) --> (actual interrupt number)
and
(3b) for each PCI device/function with an interrupt pin, resolve the
(slot, function) --> (interrupt link) --> (actual interrupt number)
functional composition, and program the result into the Interrupt Line
register of the device.
In OVMF, we do not parse the rotating map described under (1) from
QEMU's _PRT object. Instead, we duplicate the code. This is not a
problem.
In OVMF, we also do not parse the map described under (2) from QEMU's
ACPI content. Instead, we pick a specific selection (3a) that we
"apriori" know satisfies (2). This is also not a problem. OVMF's
particular selection is the PciHostIrqs table.
(
Table (2) from QEMU is
LNKA -> { 5, 10, 11 }
LNKB -> { 5, 10, 11 }
LNKC -> { 5, 10, 11 }
LNKD -> { 5, 10, 11 }
and our specific pick in OVMF, in the PciHostIrqs table, is
LNKA -> 10
LNKB -> 10
LNKC -> 11
LNKD -> 11
)
In OVMF, we also cover step (3b), in the SetPciIntLine() function.
What's missing in OVMF -- and what this patch corrects -- is that we
currently fail to program our selection for table (3) into the hardware.
We pick a specific LNKx->IRQ# mapping for each interrupt link, and we
correctly program the PCI Interrupt Line registers through those
link-to-IRQ mappings -- but we don't tell the hardware about the
link-to-IRQ mappings. More precisely, we program such a link-to-IRQ
mapping table into the hardware that is then not matched by the mapping
we use for programming the PCI device/function interrupt lines. As a
result, some PCI Interrupt Line registers will have impossible values --
a given (slot, function) may use a particular link, but also report an
interrupt number that was never picked for that link.
Output of Linux PCI Interrupt Links for i440fx before the patch:
[ 0.327305] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 10 *11)
[ 0.327944] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 10 *11)
[ 0.328582] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 *10 11)
[ 0.329208] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 *10 11)
[ 0.329807] ACPI: PCI Interrupt Link [LNKS] (IRQs *9)
after the patch:
[ 0.327292] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.327934] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.328564] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.329195] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.329785] ACPI: PCI Interrupt Link [LNKS] (IRQs *9)
Output of Linux PCI Interrupt Links for q35 before the patch:
[ 0.307474] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.308027] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.308764] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.309310] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.309853] ACPI: PCI Interrupt Link [LNKE] (IRQs 5 *10 11)
[ 0.310508] ACPI: PCI Interrupt Link [LNKF] (IRQs 5 *10 11)
[ 0.311051] ACPI: PCI Interrupt Link [LNKG] (IRQs 5 10 *11)
[ 0.311589] ACPI: PCI Interrupt Link [LNKH] (IRQs 5 10 *11)
after the patch:
[ 0.301991] ACPI: PCI Interrupt Link [LNKA] (IRQs 5 *10 11)
[ 0.302833] ACPI: PCI Interrupt Link [LNKB] (IRQs 5 *10 11)
[ 0.303354] ACPI: PCI Interrupt Link [LNKC] (IRQs 5 10 *11)
[ 0.303873] ACPI: PCI Interrupt Link [LNKD] (IRQs 5 10 *11)
[ 0.304399] ACPI: PCI Interrupt Link [LNKE] (IRQs 5 *10 11)
[ 0.304918] ACPI: PCI Interrupt Link [LNKF] (IRQs 5 *10 11)
[ 0.305436] ACPI: PCI Interrupt Link [LNKG] (IRQs 5 10 *11)
[ 0.305954] ACPI: PCI Interrupt Link [LNKH] (IRQs 5 10 *11)
Signed-off-by: Hendrik Borghorst <hborghor@amazon.de>
Reviewed-by: David Woodhouse <dwmw@amazon.co.uk>
Message-Id: <8dbedc4c7a1c3fd390aca915270814e3b35e13a5.camel@amazon.com>
Reviewed-by: Laszlo Ersek <lersek@redhat.com>
2020-12-17 15:12:22 +01:00
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 0x1f, 0, 0x60), PciHostIrqs[0]); // A
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 0x1f, 0, 0x61), PciHostIrqs[1]); // B
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 0x1f, 0, 0x62), PciHostIrqs[2]); // C
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 0x1f, 0, 0x63), PciHostIrqs[3]); // D
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 0x1f, 0, 0x68), PciHostIrqs[0]); // E
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 0x1f, 0, 0x69), PciHostIrqs[1]); // F
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 0x1f, 0, 0x6a), PciHostIrqs[2]); // G
|
|
|
|
PciWrite8 (PCI_LIB_ADDRESS (0, 0x1f, 0, 0x6b), PciHostIrqs[3]); // H
|
2015-08-03 07:26:01 +02:00
|
|
|
break;
|
|
|
|
default:
|
2019-08-13 13:31:12 +02:00
|
|
|
if (XenDetected ()) {
|
|
|
|
//
|
|
|
|
// There is no PCI bus in this case.
|
|
|
|
//
|
|
|
|
return;
|
|
|
|
}
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "%a: Unknown Host Bridge Device ID: 0x%04x\n",
|
2015-08-03 07:26:01 +02:00
|
|
|
__FUNCTION__, mHostBridgeDevId));
|
|
|
|
ASSERT (FALSE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Initialize PCI_INTERRUPT_LINE for applicable present PCI devices
|
|
|
|
//
|
|
|
|
VisitAllPciInstances (SetPciIntLine);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Set ACPI SCI_EN bit in PMCNTRL
|
|
|
|
//
|
|
|
|
IoOr16 ((PciRead32 (Pmba) & ~BIT0) + 4, BIT0);
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
ConnectRecursivelyIfPciMassStorage (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN EFI_PCI_IO_PROTOCOL *Instance,
|
|
|
|
IN PCI_TYPE00 *PciHeader
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
|
|
|
CHAR16 *DevPathStr;
|
|
|
|
|
2016-06-01 12:26:20 +02:00
|
|
|
//
|
|
|
|
// Recognize PCI Mass Storage, and Xen PCI devices
|
|
|
|
//
|
|
|
|
if (IS_CLASS1 (PciHeader, PCI_CLASS_MASS_STORAGE) ||
|
|
|
|
(XenDetected() && IS_CLASS2 (PciHeader, 0xFF, 0x80))) {
|
2015-08-03 07:26:01 +02:00
|
|
|
DevicePath = NULL;
|
|
|
|
Status = gBS->HandleProtocol (
|
|
|
|
Handle,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
(VOID*)&DevicePath
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Print Device Path
|
|
|
|
//
|
2016-04-20 11:24:38 +02:00
|
|
|
DevPathStr = ConvertDevicePathToText (DevicePath, FALSE, FALSE);
|
2015-08-03 07:26:01 +02:00
|
|
|
if (DevPathStr != NULL) {
|
|
|
|
DEBUG((
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG_INFO,
|
2016-06-01 12:26:20 +02:00
|
|
|
"Found %s device: %s\n",
|
2018-03-15 14:19:22 +01:00
|
|
|
(IS_CLASS1 (PciHeader, PCI_CLASS_MASS_STORAGE) ?
|
|
|
|
L"Mass Storage" :
|
|
|
|
L"Xen"
|
|
|
|
),
|
2015-08-03 07:26:01 +02:00
|
|
|
DevPathStr
|
|
|
|
));
|
|
|
|
FreePool(DevPathStr);
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = gBS->ConnectController (Handle, NULL, NULL, TRUE);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
This notification function is invoked when the
|
|
|
|
EMU Variable FVB has been changed.
|
|
|
|
|
2016-09-09 22:32:15 +02:00
|
|
|
@param Event The event that occurred
|
|
|
|
@param Context For EFI compatibility. Not used.
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
EmuVariablesUpdatedCallback (
|
|
|
|
IN EFI_EVENT Event,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "EmuVariablesUpdatedCallback\n"));
|
2015-08-03 07:26:01 +02:00
|
|
|
UpdateNvVarsOnFileSystem ();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
VisitingFileSystemInstance (
|
|
|
|
IN EFI_HANDLE Handle,
|
|
|
|
IN VOID *Instance,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
STATIC BOOLEAN ConnectedToFileSystem = FALSE;
|
2016-10-21 11:59:36 +02:00
|
|
|
RETURN_STATUS PcdStatus;
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
if (ConnectedToFileSystem) {
|
|
|
|
return EFI_ALREADY_STARTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = ConnectNvVarsToFileSystem (Handle);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
ConnectedToFileSystem = TRUE;
|
|
|
|
mEmuVariableEvent =
|
|
|
|
EfiCreateProtocolNotifyEvent (
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
TPL_CALLBACK,
|
|
|
|
EmuVariablesUpdatedCallback,
|
|
|
|
NULL,
|
|
|
|
&mEmuVariableEventReg
|
|
|
|
);
|
2016-10-21 11:59:36 +02:00
|
|
|
PcdStatus = PcdSet64S (PcdEmuVariableEvent,
|
|
|
|
(UINT64)(UINTN) mEmuVariableEvent);
|
|
|
|
ASSERT_RETURN_ERROR (PcdStatus);
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VOID
|
|
|
|
PlatformBdsRestoreNvVarsFromHardDisk (
|
|
|
|
)
|
|
|
|
{
|
|
|
|
VisitAllPciInstances (ConnectRecursivelyIfPciMassStorage);
|
|
|
|
VisitAllInstancesOfProtocol (
|
|
|
|
&gEfiSimpleFileSystemProtocolGuid,
|
|
|
|
VisitingFileSystemInstance,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
/**
|
|
|
|
Connect with predefined platform connect sequence.
|
|
|
|
|
|
|
|
The OEM/IBV can customize with their own connect sequence.
|
|
|
|
**/
|
2015-08-03 07:26:01 +02:00
|
|
|
VOID
|
|
|
|
PlatformBdsConnectSequence (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
2018-03-13 19:32:55 +01:00
|
|
|
UINTN Index;
|
|
|
|
RETURN_STATUS Status;
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "PlatformBdsConnectSequence\n"));
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
Index = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Here we can get the customized platform connect sequence
|
|
|
|
// Notes: we can connect with new variable which record the
|
|
|
|
// last time boots connect device path sequence
|
|
|
|
//
|
|
|
|
while (gPlatformConnectSequence[Index] != NULL) {
|
|
|
|
//
|
|
|
|
// Build the platform boot option
|
|
|
|
//
|
2016-04-20 12:05:26 +02:00
|
|
|
EfiBootManagerConnectDevicePath (gPlatformConnectSequence[Index], NULL);
|
2015-08-03 07:26:01 +02:00
|
|
|
Index++;
|
|
|
|
}
|
|
|
|
|
2018-03-13 19:32:55 +01:00
|
|
|
Status = ConnectDevicesFromQemu ();
|
|
|
|
if (RETURN_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// Just use the simple policy to connect all devices
|
|
|
|
//
|
|
|
|
DEBUG ((DEBUG_INFO, "EfiBootManagerConnectAll\n"));
|
|
|
|
EfiBootManagerConnectAll ();
|
|
|
|
}
|
2015-08-03 07:26:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Save the S3 boot script.
|
|
|
|
|
|
|
|
Note that DxeSmmReadyToLock must be signaled after this function returns;
|
|
|
|
otherwise the script wouldn't be saved actually.
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
SaveS3BootScript (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_S3_SAVE_STATE_PROTOCOL *BootScript;
|
|
|
|
STATIC CONST UINT8 Info[] = { 0xDE, 0xAD, 0xBE, 0xEF };
|
|
|
|
|
|
|
|
Status = gBS->LocateProtocol (&gEfiS3SaveStateProtocolGuid, NULL,
|
|
|
|
(VOID **) &BootScript);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Despite the opcode documentation in the PI spec, the protocol
|
|
|
|
// implementation embeds a deep copy of the info in the boot script, rather
|
|
|
|
// than storing just a pointer to runtime or NVS storage.
|
|
|
|
//
|
|
|
|
Status = BootScript->Write(BootScript, EFI_BOOT_SCRIPT_INFORMATION_OPCODE,
|
|
|
|
(UINT32) sizeof Info,
|
|
|
|
(EFI_PHYSICAL_ADDRESS)(UINTN) &Info);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-15 14:57:39 +01:00
|
|
|
/**
|
|
|
|
Do the platform specific action after the console is ready
|
|
|
|
|
|
|
|
Possible things that can be done in PlatformBootManagerAfterConsole:
|
|
|
|
|
|
|
|
> Console post action:
|
|
|
|
> Dynamically switch output mode from 100x31 to 80x25 for certain senarino
|
|
|
|
> Signal console ready platform customized event
|
|
|
|
> Run diagnostics like memory testing
|
|
|
|
> Connect certain devices
|
|
|
|
> Dispatch aditional option roms
|
|
|
|
> Special boot: e.g.: USB boot, enter UI
|
|
|
|
**/
|
2015-08-03 07:26:01 +02:00
|
|
|
VOID
|
|
|
|
EFIAPI
|
2016-04-20 09:46:46 +02:00
|
|
|
PlatformBootManagerAfterConsole (
|
|
|
|
VOID
|
2015-08-03 07:26:01 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_BOOT_MODE BootMode;
|
|
|
|
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "PlatformBootManagerAfterConsole\n"));
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
if (PcdGetBool (PcdOvmfFlashVariablesEnable)) {
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "PlatformBdsPolicyBehavior: not restoring NvVars "
|
2015-08-03 07:26:01 +02:00
|
|
|
"from disk since flash variables appear to be supported.\n"));
|
|
|
|
} else {
|
|
|
|
//
|
|
|
|
// Try to restore variables from the hard disk early so
|
|
|
|
// they can be used for the other BDS connect operations.
|
|
|
|
//
|
|
|
|
PlatformBdsRestoreNvVarsFromHardDisk ();
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Get current Boot Mode
|
|
|
|
//
|
2016-04-20 12:15:17 +02:00
|
|
|
BootMode = GetBootModeHob ();
|
2017-09-04 20:34:18 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "Boot Mode:%x\n", BootMode));
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Go the different platform policy with different boot mode
|
|
|
|
// Notes: this part code can be change with the table policy
|
|
|
|
//
|
|
|
|
ASSERT (BootMode == BOOT_WITH_FULL_CONFIGURATION);
|
|
|
|
|
|
|
|
//
|
2016-04-21 08:25:53 +02:00
|
|
|
// Logo show
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2016-09-26 08:45:06 +02:00
|
|
|
BootLogoEnableLogo ();
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2018-03-15 15:20:08 +01:00
|
|
|
//
|
|
|
|
// Set PCI Interrupt Line registers and ACPI SCI_EN
|
|
|
|
//
|
|
|
|
PciAcpiInitialization ();
|
|
|
|
|
OvmfPkg/PlatformBootManagerLib: process TPM PPI request
Call Tcg2PhysicalPresenceLibProcessRequest() to process pending PPI
requests from PlatformBootManagerAfterConsole().
Laszlo understanding of edk2 is that the PPI operation processing was
meant to occur *entirely* before End-Of-Dxe, so that 3rd party UEFI
drivers couldn't interfere with PPI opcode processing *at all*.
He suggested that we should *not* call
Tcg2PhysicalPresenceLibProcessRequest() from BeforeConsole(). Because,
an "auth" console, i.e. one that does not depend on a 3rd party
driver, is *in general* impossible to guarantee. Instead we could opt
to trust 3rd party drivers, and use the "normal" console(s) in
AfterConsole(), in order to let the user confirm the PPI requests. It
will depend on the user to enable Secure Boot, so that the
trustworthiness of those 3rd party drivers is ensured. If an attacker
roots the guest OS from within, queues some TPM2 PPI requests, and
also modifies drivers on the EFI system partition and/or in GPU option
ROMs (?), then those drivers will not load after guest reboot, and
thus the dependent console(s) won't be used for confirming the PPI
requests.
Contributed-under: TianoCore Contribution Agreement 1.1
Signed-off-by: Marc-André Lureau <marcandre.lureau@redhat.com>
Reviewed-by: Laszlo Ersek <lersek@redhat.com>
2018-05-18 14:23:04 +02:00
|
|
|
//
|
|
|
|
// Process TPM PPI request
|
|
|
|
//
|
|
|
|
Tcg2PhysicalPresenceLibProcessRequest (NULL);
|
|
|
|
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2018-03-15 12:49:26 +01:00
|
|
|
// Process QEMU's -kernel command line option
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2018-03-15 12:49:26 +01:00
|
|
|
TryRunningQemuKernel ();
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
//
|
2018-03-15 12:49:26 +01:00
|
|
|
// Perform some platform specific connect sequence
|
2015-08-03 07:26:01 +02:00
|
|
|
//
|
2018-03-15 12:49:26 +01:00
|
|
|
PlatformBdsConnectSequence ();
|
2015-08-03 07:26:01 +02:00
|
|
|
|
2016-04-20 12:09:21 +02:00
|
|
|
EfiBootManagerRefreshAllBootOption ();
|
|
|
|
|
2016-05-11 10:40:49 +02:00
|
|
|
//
|
|
|
|
// Register UEFI Shell
|
|
|
|
//
|
|
|
|
PlatformRegisterFvBootOption (
|
2019-06-14 03:14:29 +02:00
|
|
|
&gUefiShellFileGuid, L"EFI Internal Shell", LOAD_OPTION_ACTIVE
|
2016-05-11 10:40:49 +02:00
|
|
|
);
|
|
|
|
|
2016-07-08 02:49:45 +02:00
|
|
|
RemoveStaleFvFileOptions ();
|
2016-05-17 19:22:17 +02:00
|
|
|
SetBootOrderFromQemu ();
|
2019-02-20 01:08:58 +01:00
|
|
|
|
|
|
|
PlatformBmPrintScRegisterHandler ();
|
2015-08-03 07:26:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This notification function is invoked when an instance of the
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL is produced.
|
|
|
|
|
2016-09-09 22:32:15 +02:00
|
|
|
@param Event The event that occurred
|
|
|
|
@param Context For EFI compatibility. Not used.
|
2015-08-03 07:26:01 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
NotifyDevPath (
|
|
|
|
IN EFI_EVENT Event,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_HANDLE Handle;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN BufferSize;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevPathNode;
|
|
|
|
ATAPI_DEVICE_PATH *Atapi;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Examine all new handles
|
|
|
|
//
|
|
|
|
for (;;) {
|
|
|
|
//
|
|
|
|
// Get the next handle
|
|
|
|
//
|
|
|
|
BufferSize = sizeof (Handle);
|
|
|
|
Status = gBS->LocateHandle (
|
|
|
|
ByRegisterNotify,
|
|
|
|
NULL,
|
|
|
|
mEfiDevPathNotifyReg,
|
|
|
|
&BufferSize,
|
|
|
|
&Handle
|
|
|
|
);
|
|
|
|
|
|
|
|
//
|
|
|
|
// If not found, we're done
|
|
|
|
//
|
|
|
|
if (EFI_NOT_FOUND == Status) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Get the DevicePath protocol on that handle
|
|
|
|
//
|
2018-03-15 14:19:22 +01:00
|
|
|
Status = gBS->HandleProtocol (Handle, &gEfiDevicePathProtocolGuid,
|
|
|
|
(VOID **)&DevPathNode);
|
2015-08-03 07:26:01 +02:00
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
while (!IsDevicePathEnd (DevPathNode)) {
|
|
|
|
//
|
|
|
|
// Find the handler to dump this device path node
|
|
|
|
//
|
|
|
|
if (
|
|
|
|
(DevicePathType(DevPathNode) == MESSAGING_DEVICE_PATH) &&
|
|
|
|
(DevicePathSubType(DevPathNode) == MSG_ATAPI_DP)
|
|
|
|
) {
|
|
|
|
Atapi = (ATAPI_DEVICE_PATH*) DevPathNode;
|
|
|
|
PciOr16 (
|
|
|
|
PCI_LIB_ADDRESS (
|
|
|
|
0,
|
|
|
|
1,
|
|
|
|
1,
|
|
|
|
(Atapi->PrimarySecondary == 1) ? 0x42: 0x40
|
|
|
|
),
|
|
|
|
BIT15
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Next device path node
|
|
|
|
//
|
|
|
|
DevPathNode = NextDevicePathNode (DevPathNode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VOID
|
|
|
|
InstallDevicePathCallback (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
2020-04-29 23:53:27 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "Registered NotifyDevPath Event\n"));
|
2015-08-03 07:26:01 +02:00
|
|
|
mEfiDevPathEvent = EfiCreateProtocolNotifyEvent (
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
TPL_CALLBACK,
|
|
|
|
NotifyDevPath,
|
|
|
|
NULL,
|
|
|
|
&mEfiDevPathNotifyReg
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-04-20 09:46:46 +02:00
|
|
|
/**
|
2018-03-15 14:19:22 +01:00
|
|
|
This function is called each second during the boot manager waits the
|
|
|
|
timeout.
|
2016-04-20 09:46:46 +02:00
|
|
|
|
|
|
|
@param TimeoutRemain The remaining timeout.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
PlatformBootManagerWaitCallback (
|
|
|
|
UINT16 TimeoutRemain
|
|
|
|
)
|
|
|
|
{
|
2016-05-25 14:00:06 +02:00
|
|
|
EFI_GRAPHICS_OUTPUT_BLT_PIXEL_UNION Black;
|
|
|
|
EFI_GRAPHICS_OUTPUT_BLT_PIXEL_UNION White;
|
2019-10-14 17:03:10 +02:00
|
|
|
UINT16 TimeoutInitial;
|
2016-05-25 14:00:06 +02:00
|
|
|
|
2019-10-14 17:03:10 +02:00
|
|
|
TimeoutInitial = PcdGet16 (PcdPlatformBootTimeOut);
|
|
|
|
|
|
|
|
//
|
|
|
|
// If PcdPlatformBootTimeOut is set to zero, then we consider
|
|
|
|
// that no progress update should be enacted (since we'd only
|
|
|
|
// ever display a one-shot progress of either 0% or 100%).
|
|
|
|
//
|
|
|
|
if (TimeoutInitial == 0) {
|
|
|
|
return;
|
|
|
|
}
|
2016-05-25 14:00:06 +02:00
|
|
|
|
|
|
|
Black.Raw = 0x00000000;
|
|
|
|
White.Raw = 0x00FFFFFF;
|
|
|
|
|
|
|
|
BootLogoUpdateProgress (
|
|
|
|
White.Pixel,
|
|
|
|
Black.Pixel,
|
|
|
|
L"Start boot option",
|
|
|
|
White.Pixel,
|
2019-10-14 17:03:10 +02:00
|
|
|
(TimeoutInitial - TimeoutRemain) * 100 / TimeoutInitial,
|
2016-05-25 14:00:06 +02:00
|
|
|
0
|
|
|
|
);
|
2016-04-20 09:46:46 +02:00
|
|
|
}
|
|
|
|
|
2018-07-03 08:04:55 +02:00
|
|
|
/**
|
|
|
|
The function is called when no boot option could be launched,
|
|
|
|
including platform recovery options and options pointing to applications
|
|
|
|
built into firmware volumes.
|
|
|
|
|
|
|
|
If this function returns, BDS attempts to enter an infinite loop.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
PlatformBootManagerUnableToBoot (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_INPUT_KEY Key;
|
|
|
|
EFI_BOOT_MANAGER_LOAD_OPTION BootManagerMenu;
|
|
|
|
UINTN Index;
|
|
|
|
|
|
|
|
//
|
|
|
|
// BootManagerMenu doesn't contain the correct information when return status
|
|
|
|
// is EFI_NOT_FOUND.
|
|
|
|
//
|
|
|
|
Status = EfiBootManagerGetBootManagerMenu (&BootManagerMenu);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// Normally BdsDxe does not print anything to the system console, but this is
|
|
|
|
// a last resort -- the end-user will likely not see any DEBUG messages
|
|
|
|
// logged in this situation.
|
|
|
|
//
|
|
|
|
// AsciiPrint() will NULL-check gST->ConOut internally. We check gST->ConIn
|
|
|
|
// here to see if it makes sense to request and wait for a keypress.
|
|
|
|
//
|
|
|
|
if (gST->ConIn != NULL) {
|
|
|
|
AsciiPrint (
|
|
|
|
"%a: No bootable option or device was found.\n"
|
|
|
|
"%a: Press any key to enter the Boot Manager Menu.\n",
|
|
|
|
gEfiCallerBaseName,
|
|
|
|
gEfiCallerBaseName
|
|
|
|
);
|
|
|
|
Status = gBS->WaitForEvent (1, &gST->ConIn->WaitForKey, &Index);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
ASSERT (Index == 0);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Drain any queued keys.
|
|
|
|
//
|
|
|
|
while (!EFI_ERROR (gST->ConIn->ReadKeyStroke (gST->ConIn, &Key))) {
|
|
|
|
//
|
|
|
|
// just throw away Key
|
|
|
|
//
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
EfiBootManagerBoot (&BootManagerMenu);
|
|
|
|
}
|
|
|
|
}
|