/**@file Xen Platform PEI support Copyright (c) 2006 - 2016, Intel Corporation. All rights reserved.
Copyright (c) 2011, Andrei Warkentin Copyright (c) 2019, Citrix Systems, Inc. SPDX-License-Identifier: BSD-2-Clause-Patent **/ // // The package level header files this module uses // #include // // The Library classes this module consumes // #include #include #include #include #include #include #include #include #include #include #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); } BOOLEAN XenPvhDetected ( VOID ) { // // This function should only be used after XenConnect // ASSERT (mXenInfo.HyperPages != NULL); 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; }