2019-08-13 13:30:48 +02:00
|
|
|
/**@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>
|
2019-08-13 13:30:56 +02:00
|
|
|
#include <IndustryStandard/Xen/arch-x86/hvm/start_info.h>
|
2019-08-13 13:31:01 +02:00
|
|
|
#include <Library/XenHypercallLib.h>
|
2019-08-13 13:30:48 +02:00
|
|
|
|
|
|
|
#include "Platform.h"
|
|
|
|
#include "Xen.h"
|
|
|
|
|
|
|
|
STATIC UINT32 mXenLeaf = 0;
|
|
|
|
|
|
|
|
EFI_XEN_INFO mXenInfo;
|
|
|
|
|
2019-08-13 13:30:54 +02:00
|
|
|
//
|
|
|
|
// Location of the firmware info struct setup by hvmloader.
|
|
|
|
// Only the E820 table is used by OVMF.
|
|
|
|
//
|
|
|
|
EFI_XEN_OVMF_INFO *mXenHvmloaderInfo;
|
|
|
|
|
2019-08-13 13:30:48 +02:00
|
|
|
/**
|
|
|
|
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
|
|
|
|
)
|
|
|
|
{
|
2019-08-13 13:30:55 +02:00
|
|
|
//
|
|
|
|
// 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;
|
2019-08-13 13:30:48 +02:00
|
|
|
}
|
|
|
|
|
2019-08-13 13:30:55 +02:00
|
|
|
return EFI_NOT_FOUND;
|
2019-08-13 13:30:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Connects to the Hypervisor.
|
|
|
|
|
|
|
|
@return EFI_STATUS
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
XenConnect (
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 Index;
|
|
|
|
UINT32 TransferReg;
|
|
|
|
UINT32 TransferPages;
|
|
|
|
UINT32 XenVersion;
|
2019-08-13 13:30:54 +02:00
|
|
|
EFI_XEN_OVMF_INFO *Info;
|
|
|
|
CHAR8 Sig[sizeof (Info->Signature) + 1];
|
2019-08-13 13:30:56 +02:00
|
|
|
UINT32 *PVHResetVectorData;
|
2019-08-13 13:31:01 +02:00
|
|
|
RETURN_STATUS Status;
|
2019-08-13 13:30:48 +02:00
|
|
|
|
2019-08-13 13:31:03 +02:00
|
|
|
ASSERT (mXenLeaf != 0);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Prepare HyperPages to be able to make hypercalls
|
|
|
|
//
|
|
|
|
|
|
|
|
AsmCpuid (mXenLeaf + 2, &TransferPages, &TransferReg, NULL, NULL);
|
2019-08-13 13:30:48 +02:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2019-08-13 13:31:03 +02:00
|
|
|
//
|
|
|
|
// Find out the Xen version
|
|
|
|
//
|
|
|
|
|
|
|
|
AsmCpuid (mXenLeaf + 1, &XenVersion, NULL, NULL, NULL);
|
2019-08-13 13:30:48 +02:00
|
|
|
DEBUG ((DEBUG_ERROR, "Detected Xen version %d.%d\n",
|
|
|
|
XenVersion >> 16, XenVersion & 0xFFFF));
|
|
|
|
mXenInfo.VersionMajor = (UINT16)(XenVersion >> 16);
|
|
|
|
mXenInfo.VersionMinor = (UINT16)(XenVersion & 0xFFFF);
|
|
|
|
|
2019-08-13 13:30:54 +02:00
|
|
|
//
|
|
|
|
// 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;
|
|
|
|
}
|
2019-08-13 13:30:48 +02:00
|
|
|
|
2019-08-13 13:30:56 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-13 13:30:48 +02:00
|
|
|
BuildGuidDataHob (
|
|
|
|
&gEfiXenInfoGuid,
|
|
|
|
&mXenInfo,
|
|
|
|
sizeof(mXenInfo)
|
|
|
|
);
|
|
|
|
|
2019-08-13 13:31:01 +02:00
|
|
|
//
|
|
|
|
// Initialize the XenHypercall library, now that the XenInfo HOB is
|
|
|
|
// available
|
|
|
|
//
|
|
|
|
Status = XenHypercallLibInit ();
|
|
|
|
ASSERT_RETURN_ERROR (Status);
|
|
|
|
|
2019-08-13 13:30:48 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-08-13 13:31:02 +02:00
|
|
|
BOOLEAN
|
|
|
|
XenHvmloaderDetected (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return (mXenHvmloaderInfo != NULL);
|
|
|
|
}
|
2019-08-13 13:30:48 +02:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|