We are going to need to make an hypercall in order to retreive the E820 table from the hypervisor before been able to setup the memory. Calling XenConnect earlier will allow to setup the XenHypercallLib earlier to allow to make hypercalls. While here, add some comments in XenConnect(). Ref: https://bugzilla.tianocore.org/show_bug.cgi?id=1689 Signed-off-by: Anthony PERARD <anthony.perard@citrix.com> Acked-by: Laszlo Ersek <lersek@redhat.com> Message-Id: <20190813113119.14804-20-anthony.perard@citrix.com>
283 lines
6.2 KiB
C
283 lines
6.2 KiB
C
/**@file
|
|
Xen Platform PEI support
|
|
|
|
Copyright (c) 2006 - 2016, Intel Corporation. All rights reserved.<BR>
|
|
Copyright (c) 2011, Andrei Warkentin <andreiw@motorola.com>
|
|
Copyright (c) 2019, Citrix Systems, Inc.
|
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
|
|
|
**/
|
|
|
|
//
|
|
// The package level header files this module uses
|
|
//
|
|
#include <PiPei.h>
|
|
|
|
//
|
|
// The Library classes this module consumes
|
|
//
|
|
#include <Library/DebugLib.h>
|
|
#include <Library/HobLib.h>
|
|
#include <Library/MemoryAllocationLib.h>
|
|
#include <Library/PcdLib.h>
|
|
#include <Guid/XenInfo.h>
|
|
#include <IndustryStandard/E820.h>
|
|
#include <Library/ResourcePublicationLib.h>
|
|
#include <Library/MtrrLib.h>
|
|
#include <IndustryStandard/Xen/arch-x86/hvm/start_info.h>
|
|
#include <Library/XenHypercallLib.h>
|
|
|
|
#include "Platform.h"
|
|
#include "Xen.h"
|
|
|
|
STATIC UINT32 mXenLeaf = 0;
|
|
|
|
EFI_XEN_INFO mXenInfo;
|
|
|
|
//
|
|
// Location of the firmware info struct setup by hvmloader.
|
|
// Only the E820 table is used by OVMF.
|
|
//
|
|
EFI_XEN_OVMF_INFO *mXenHvmloaderInfo;
|
|
|
|
/**
|
|
Returns E820 map provided by Xen
|
|
|
|
@param Entries Pointer to E820 map
|
|
@param Count Number of entries
|
|
|
|
@return EFI_STATUS
|
|
**/
|
|
EFI_STATUS
|
|
XenGetE820Map (
|
|
EFI_E820_ENTRY64 **Entries,
|
|
UINT32 *Count
|
|
)
|
|
{
|
|
//
|
|
// Get E820 produced by hvmloader
|
|
//
|
|
if (mXenHvmloaderInfo != NULL) {
|
|
ASSERT (mXenHvmloaderInfo->E820 < MAX_ADDRESS);
|
|
*Entries = (EFI_E820_ENTRY64 *)(UINTN) mXenHvmloaderInfo->E820;
|
|
*Count = mXenHvmloaderInfo->E820EntriesCount;
|
|
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
return EFI_NOT_FOUND;
|
|
}
|
|
|
|
/**
|
|
Connects to the Hypervisor.
|
|
|
|
@return EFI_STATUS
|
|
|
|
**/
|
|
EFI_STATUS
|
|
XenConnect (
|
|
)
|
|
{
|
|
UINT32 Index;
|
|
UINT32 TransferReg;
|
|
UINT32 TransferPages;
|
|
UINT32 XenVersion;
|
|
EFI_XEN_OVMF_INFO *Info;
|
|
CHAR8 Sig[sizeof (Info->Signature) + 1];
|
|
UINT32 *PVHResetVectorData;
|
|
RETURN_STATUS Status;
|
|
|
|
ASSERT (mXenLeaf != 0);
|
|
|
|
//
|
|
// Prepare HyperPages to be able to make hypercalls
|
|
//
|
|
|
|
AsmCpuid (mXenLeaf + 2, &TransferPages, &TransferReg, NULL, NULL);
|
|
mXenInfo.HyperPages = AllocatePages (TransferPages);
|
|
if (!mXenInfo.HyperPages) {
|
|
return EFI_OUT_OF_RESOURCES;
|
|
}
|
|
|
|
for (Index = 0; Index < TransferPages; Index++) {
|
|
AsmWriteMsr64 (TransferReg,
|
|
(UINTN) mXenInfo.HyperPages +
|
|
(Index << EFI_PAGE_SHIFT) + Index);
|
|
}
|
|
|
|
//
|
|
// Find out the Xen version
|
|
//
|
|
|
|
AsmCpuid (mXenLeaf + 1, &XenVersion, NULL, NULL, NULL);
|
|
DEBUG ((DEBUG_ERROR, "Detected Xen version %d.%d\n",
|
|
XenVersion >> 16, XenVersion & 0xFFFF));
|
|
mXenInfo.VersionMajor = (UINT16)(XenVersion >> 16);
|
|
mXenInfo.VersionMinor = (UINT16)(XenVersion & 0xFFFF);
|
|
|
|
//
|
|
// Check if there are information left by hvmloader
|
|
//
|
|
|
|
Info = (EFI_XEN_OVMF_INFO *)(UINTN) OVMF_INFO_PHYSICAL_ADDRESS;
|
|
//
|
|
// Copy the signature, and make it null-terminated.
|
|
//
|
|
AsciiStrnCpyS (Sig, sizeof (Sig), (CHAR8 *) &Info->Signature,
|
|
sizeof (Info->Signature));
|
|
if (AsciiStrCmp (Sig, "XenHVMOVMF") == 0) {
|
|
mXenHvmloaderInfo = Info;
|
|
} else {
|
|
mXenHvmloaderInfo = NULL;
|
|
}
|
|
|
|
mXenInfo.RsdpPvh = NULL;
|
|
|
|
//
|
|
// Locate and use information from the start of day structure if we have
|
|
// booted via the PVH entry point.
|
|
//
|
|
|
|
PVHResetVectorData = (VOID *)(UINTN) PcdGet32 (PcdXenPvhStartOfDayStructPtr);
|
|
//
|
|
// That magic value is written in XenResetVector/Ia32/XenPVHMain.asm
|
|
//
|
|
if (PVHResetVectorData[1] == SIGNATURE_32 ('X', 'P', 'V', 'H')) {
|
|
struct hvm_start_info *HVMStartInfo;
|
|
|
|
HVMStartInfo = (VOID *)(UINTN) PVHResetVectorData[0];
|
|
if (HVMStartInfo->magic == XEN_HVM_START_MAGIC_VALUE) {
|
|
ASSERT (HVMStartInfo->rsdp_paddr != 0);
|
|
if (HVMStartInfo->rsdp_paddr != 0) {
|
|
mXenInfo.RsdpPvh = (VOID *)(UINTN)HVMStartInfo->rsdp_paddr;
|
|
}
|
|
}
|
|
}
|
|
|
|
BuildGuidDataHob (
|
|
&gEfiXenInfoGuid,
|
|
&mXenInfo,
|
|
sizeof(mXenInfo)
|
|
);
|
|
|
|
//
|
|
// Initialize the XenHypercall library, now that the XenInfo HOB is
|
|
// available
|
|
//
|
|
Status = XenHypercallLibInit ();
|
|
ASSERT_RETURN_ERROR (Status);
|
|
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
Figures out if we are running inside Xen HVM.
|
|
|
|
@retval TRUE Xen was detected
|
|
@retval FALSE Xen was not detected
|
|
|
|
**/
|
|
BOOLEAN
|
|
XenDetect (
|
|
VOID
|
|
)
|
|
{
|
|
UINT8 Signature[13];
|
|
|
|
if (mXenLeaf != 0) {
|
|
return TRUE;
|
|
}
|
|
|
|
Signature[12] = '\0';
|
|
for (mXenLeaf = 0x40000000; mXenLeaf < 0x40010000; mXenLeaf += 0x100) {
|
|
AsmCpuid (mXenLeaf,
|
|
NULL,
|
|
(UINT32 *) &Signature[0],
|
|
(UINT32 *) &Signature[4],
|
|
(UINT32 *) &Signature[8]);
|
|
|
|
if (!AsciiStrCmp ((CHAR8 *) Signature, "XenVMMXenVMM")) {
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
mXenLeaf = 0;
|
|
return FALSE;
|
|
}
|
|
|
|
BOOLEAN
|
|
XenHvmloaderDetected (
|
|
VOID
|
|
)
|
|
{
|
|
return (mXenHvmloaderInfo != NULL);
|
|
}
|
|
|
|
VOID
|
|
XenPublishRamRegions (
|
|
VOID
|
|
)
|
|
{
|
|
EFI_E820_ENTRY64 *E820Map;
|
|
UINT32 E820EntriesCount;
|
|
EFI_STATUS Status;
|
|
|
|
DEBUG ((DEBUG_INFO, "Using memory map provided by Xen\n"));
|
|
|
|
//
|
|
// Parse RAM in E820 map
|
|
//
|
|
E820EntriesCount = 0;
|
|
Status = XenGetE820Map (&E820Map, &E820EntriesCount);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
if (E820EntriesCount > 0) {
|
|
EFI_E820_ENTRY64 *Entry;
|
|
UINT32 Loop;
|
|
|
|
for (Loop = 0; Loop < E820EntriesCount; Loop++) {
|
|
Entry = E820Map + Loop;
|
|
|
|
//
|
|
// Only care about RAM
|
|
//
|
|
if (Entry->Type != EfiAcpiAddressRangeMemory) {
|
|
continue;
|
|
}
|
|
|
|
AddMemoryBaseSizeHob (Entry->BaseAddr, Entry->Length);
|
|
|
|
MtrrSetMemoryAttribute (Entry->BaseAddr, Entry->Length, CacheWriteBack);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
Perform Xen PEI initialization.
|
|
|
|
@return EFI_SUCCESS Xen initialized successfully
|
|
@return EFI_NOT_FOUND Not running under Xen
|
|
|
|
**/
|
|
EFI_STATUS
|
|
InitializeXen (
|
|
VOID
|
|
)
|
|
{
|
|
RETURN_STATUS PcdStatus;
|
|
|
|
//
|
|
// Reserve away HVMLOADER reserved memory [0xFC000000,0xFD000000).
|
|
// This needs to match HVMLOADER RESERVED_MEMBASE/RESERVED_MEMSIZE.
|
|
//
|
|
AddReservedMemoryBaseSizeHob (0xFC000000, 0x1000000, FALSE);
|
|
|
|
PcdStatus = PcdSetBoolS (PcdPciDisableBusEnumeration, TRUE);
|
|
ASSERT_RETURN_ERROR (PcdStatus);
|
|
|
|
return EFI_SUCCESS;
|
|
}
|