OvmfPkg/SmmRelocationLib: Add library instance for OVMF
There are below 2 differences between AMD & OVMF according existing implementation: 1.The mode of the CPU check is different between the AMD & OVMF. OVMF: CpuSaveState->x86.SMMRevId & 0Xffff AMD: LMAValue = (UINT32)AsmReadMsr64 (EFER_ADDRESS) & LMA 2.Existing SmBase configuration is different between the AMD & OVMF. OVMF: if ((CpuSaveState->x86.SMMRevId & 0xFFFF) == 0) { CpuSaveState->x86.SMBASE = mSmBaseForAllCpus[CpuIndex]; } else { CpuSaveState->x64.SMBASE = mSmBaseForAllCpus[CpuIndex]; } AMD: AmdCpuState->x64.SMBASE = mSmBaseForAllCpus[CpuIndex]; This patch provides the SmmRelocationLib library instance for OVMF to handle the logic difference, and it won't change the existing implementation code logic. Cc: Ray Ni <ray.ni@intel.com> Cc: Zeng Star <star.zeng@intel.com> Cc: Ard Biesheuvel <ardb+tianocore@kernel.org> Cc: Jiewen Yao <jiewen.yao@intel.com> Cc: Gerd Hoffmann <kraxel@redhat.com> Cc: Rahul Kumar <rahul1.kumar@intel.com> Signed-off-by: Jiaxin Wu <jiaxin.wu@intel.com> Tested-by: Gerd Hoffmann <kraxel@redhat.com> Acked-by: Gerd Hoffmann <kraxel@redhat.com> Acked-by: Jiewen Yao <Jiewen.yao@intel.com>
This commit is contained in:
549
OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.c
Normal file
549
OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.c
Normal file
@@ -0,0 +1,549 @@
|
||||
/** @file
|
||||
SMM Relocation Lib for each processor.
|
||||
|
||||
This Lib produces the SMM_BASE_HOB in HOB database which tells
|
||||
the PiSmmCpuDxeSmm driver (runs at a later phase) about the new
|
||||
SMBASE for each processor. PiSmmCpuDxeSmm driver installs the
|
||||
SMI handler at the SMM_BASE_HOB.SmBase[Index]+0x8000 for processor
|
||||
Index.
|
||||
|
||||
Copyright (c) 2024, Intel Corporation. All rights reserved.<BR>
|
||||
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||
|
||||
**/
|
||||
#include "InternalSmmRelocationLib.h"
|
||||
|
||||
UINTN mMaxNumberOfCpus = 1;
|
||||
UINTN mNumberOfCpus = 1;
|
||||
|
||||
//
|
||||
// IDT used during SMM Init
|
||||
//
|
||||
IA32_DESCRIPTOR gcSmmInitIdtr;
|
||||
|
||||
//
|
||||
// Smbase for current CPU
|
||||
//
|
||||
UINT64 mSmBase;
|
||||
|
||||
//
|
||||
// SmBase Rebased flag for current CPU
|
||||
//
|
||||
volatile BOOLEAN mRebased;
|
||||
|
||||
/**
|
||||
This function will get the SmBase for CpuIndex.
|
||||
|
||||
@param[in] CpuIndex The processor index.
|
||||
@param[in] SmmRelocationStart The start address of Smm relocated memory in SMRAM.
|
||||
@param[in] TileSize The total size required for a CPU save state, any
|
||||
additional CPU-specific context and the size of code
|
||||
for the SMI entry point.
|
||||
|
||||
@retval The value of SmBase for CpuIndex.
|
||||
|
||||
**/
|
||||
UINTN
|
||||
GetSmBase (
|
||||
IN UINTN CpuIndex,
|
||||
IN EFI_PHYSICAL_ADDRESS SmmRelocationStart,
|
||||
IN UINTN TileSize
|
||||
)
|
||||
{
|
||||
return (UINTN)(SmmRelocationStart) + CpuIndex * TileSize - SMM_HANDLER_OFFSET;
|
||||
}
|
||||
|
||||
/**
|
||||
This function will create SmBase for all CPUs.
|
||||
|
||||
@param[in] SmmRelocationStart The start address of Smm relocated memory in SMRAM.
|
||||
@param[in] TileSize The total size required for a CPU save state, any
|
||||
additional CPU-specific context and the size of code
|
||||
for the SMI entry point.
|
||||
|
||||
@retval EFI_SUCCESS Create SmBase for all CPUs successfully.
|
||||
@retval Others Failed to create SmBase for all CPUs.
|
||||
|
||||
**/
|
||||
EFI_STATUS
|
||||
CreateSmmBaseHob (
|
||||
IN EFI_PHYSICAL_ADDRESS SmmRelocationStart,
|
||||
IN UINTN TileSize
|
||||
)
|
||||
{
|
||||
UINTN Index;
|
||||
SMM_BASE_HOB_DATA *SmmBaseHobData;
|
||||
UINT32 CpuCount;
|
||||
UINT32 NumberOfProcessorsInHob;
|
||||
UINT32 MaxCapOfProcessorsInHob;
|
||||
UINT32 HobCount;
|
||||
|
||||
SmmBaseHobData = NULL;
|
||||
CpuCount = 0;
|
||||
NumberOfProcessorsInHob = 0;
|
||||
MaxCapOfProcessorsInHob = 0;
|
||||
HobCount = 0;
|
||||
|
||||
//
|
||||
// Count the HOB instance maximum capacity of CPU (MaxCapOfProcessorsInHob) since the max HobLength is 0xFFF8.
|
||||
//
|
||||
MaxCapOfProcessorsInHob = (0xFFF8 - sizeof (EFI_HOB_GUID_TYPE) - sizeof (SMM_BASE_HOB_DATA)) / sizeof (UINT64) + 1;
|
||||
DEBUG ((DEBUG_INFO, "CreateSmmBaseHob - MaxCapOfProcessorsInHob: %d\n", MaxCapOfProcessorsInHob));
|
||||
|
||||
//
|
||||
// Create Guided SMM Base HOB Instances.
|
||||
//
|
||||
while (CpuCount != mMaxNumberOfCpus) {
|
||||
NumberOfProcessorsInHob = MIN ((UINT32)mMaxNumberOfCpus - CpuCount, MaxCapOfProcessorsInHob);
|
||||
|
||||
SmmBaseHobData = BuildGuidHob (
|
||||
&gSmmBaseHobGuid,
|
||||
sizeof (SMM_BASE_HOB_DATA) + sizeof (UINT64) * NumberOfProcessorsInHob
|
||||
);
|
||||
if (SmmBaseHobData == NULL) {
|
||||
return EFI_OUT_OF_RESOURCES;
|
||||
}
|
||||
|
||||
SmmBaseHobData->ProcessorIndex = CpuCount;
|
||||
SmmBaseHobData->NumberOfProcessors = NumberOfProcessorsInHob;
|
||||
|
||||
DEBUG ((DEBUG_INFO, "CreateSmmBaseHob - SmmBaseHobData[%d]->ProcessorIndex: %d\n", HobCount, SmmBaseHobData->ProcessorIndex));
|
||||
DEBUG ((DEBUG_INFO, "CreateSmmBaseHob - SmmBaseHobData[%d]->NumberOfProcessors: %d\n", HobCount, SmmBaseHobData->NumberOfProcessors));
|
||||
for (Index = 0; Index < SmmBaseHobData->NumberOfProcessors; Index++) {
|
||||
//
|
||||
// Calculate the new SMBASE address
|
||||
//
|
||||
SmmBaseHobData->SmBase[Index] = GetSmBase (Index + CpuCount, SmmRelocationStart, TileSize);
|
||||
DEBUG ((DEBUG_INFO, "CreateSmmBaseHob - SmmBaseHobData[%d]->SmBase[%d]: 0x%08x\n", HobCount, Index, SmmBaseHobData->SmBase[Index]));
|
||||
}
|
||||
|
||||
CpuCount += NumberOfProcessorsInHob;
|
||||
HobCount++;
|
||||
SmmBaseHobData = NULL;
|
||||
}
|
||||
|
||||
return EFI_SUCCESS;
|
||||
}
|
||||
|
||||
/**
|
||||
C function for SMI handler. To change all processor's SMMBase Register.
|
||||
|
||||
**/
|
||||
VOID
|
||||
EFIAPI
|
||||
SmmInitHandler (
|
||||
VOID
|
||||
)
|
||||
{
|
||||
//
|
||||
// Update SMM IDT entries' code segment and load IDT
|
||||
//
|
||||
AsmWriteIdtr (&gcSmmInitIdtr);
|
||||
|
||||
//
|
||||
// Configure SmBase.
|
||||
//
|
||||
ConfigureSmBase (mSmBase);
|
||||
|
||||
//
|
||||
// Hook return after RSM to set SMM re-based flag
|
||||
// SMM re-based flag can't be set before RSM, because SMM save state context might be override
|
||||
// by next AP flow before it take effect.
|
||||
//
|
||||
SemaphoreHook (&mRebased);
|
||||
}
|
||||
|
||||
/**
|
||||
Relocate SmmBases for each processor.
|
||||
Execute on first boot and all S3 resumes
|
||||
|
||||
@param[in] MpServices2 Pointer to this instance of the MpServices.
|
||||
@param[in] SmmRelocationStart The start address of Smm relocated memory in SMRAM.
|
||||
@param[in] TileSize The total size required for a CPU save state, any
|
||||
additional CPU-specific context and the size of code
|
||||
for the SMI entry point.
|
||||
|
||||
**/
|
||||
VOID
|
||||
SmmRelocateBases (
|
||||
IN EDKII_PEI_MP_SERVICES2_PPI *MpServices2,
|
||||
IN EFI_PHYSICAL_ADDRESS SmmRelocationStart,
|
||||
IN UINTN TileSize
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
UINT8 BakBuf[BACK_BUF_SIZE];
|
||||
SMRAM_SAVE_STATE_MAP BakBuf2;
|
||||
SMRAM_SAVE_STATE_MAP *CpuStatePtr;
|
||||
UINT8 *U8Ptr;
|
||||
UINTN Index;
|
||||
UINTN BspIndex;
|
||||
UINT32 BspApicId;
|
||||
EFI_PROCESSOR_INFORMATION ProcessorInfo;
|
||||
|
||||
//
|
||||
// Make sure the reserved size is large enough for procedure SmmInitTemplate.
|
||||
//
|
||||
ASSERT (sizeof (BakBuf) >= gcSmmInitSize);
|
||||
|
||||
//
|
||||
// Patch ASM code template with current CR0, CR3, and CR4 values
|
||||
//
|
||||
PatchInstructionX86 (gPatchSmmInitCr0, AsmReadCr0 (), 4);
|
||||
PatchInstructionX86 (gPatchSmmInitCr3, AsmReadCr3 (), 4);
|
||||
PatchInstructionX86 (gPatchSmmInitCr4, AsmReadCr4 () & (~CR4_CET_ENABLE), 4);
|
||||
|
||||
U8Ptr = (UINT8 *)(UINTN)(SMM_DEFAULT_SMBASE + SMM_HANDLER_OFFSET);
|
||||
CpuStatePtr = (SMRAM_SAVE_STATE_MAP *)(UINTN)(SMM_DEFAULT_SMBASE + SMRAM_SAVE_STATE_MAP_OFFSET);
|
||||
|
||||
//
|
||||
// Backup original contents at address 0x38000
|
||||
//
|
||||
CopyMem (BakBuf, U8Ptr, sizeof (BakBuf));
|
||||
CopyMem (&BakBuf2, CpuStatePtr, sizeof (BakBuf2));
|
||||
|
||||
//
|
||||
// Load image for relocation
|
||||
//
|
||||
CopyMem (U8Ptr, gcSmmInitTemplate, gcSmmInitSize);
|
||||
|
||||
//
|
||||
// Retrieve the local APIC ID of current processor
|
||||
//
|
||||
BspApicId = GetApicId ();
|
||||
|
||||
//
|
||||
// Relocate SM bases for all APs
|
||||
// This is APs' 1st SMI - rebase will be done here, and APs' default SMI handler will be overridden by gcSmmInitTemplate
|
||||
//
|
||||
BspIndex = (UINTN)-1;
|
||||
for (Index = 0; Index < mNumberOfCpus; Index++) {
|
||||
Status = MpServices2->GetProcessorInfo (MpServices2, Index | CPU_V2_EXTENDED_TOPOLOGY, &ProcessorInfo);
|
||||
ASSERT_EFI_ERROR (Status);
|
||||
|
||||
if (BspApicId != (UINT32)ProcessorInfo.ProcessorId) {
|
||||
mRebased = FALSE;
|
||||
mSmBase = GetSmBase (Index, SmmRelocationStart, TileSize);
|
||||
SendSmiIpi ((UINT32)ProcessorInfo.ProcessorId);
|
||||
//
|
||||
// Wait for this AP to finish its 1st SMI
|
||||
//
|
||||
while (!mRebased) {
|
||||
}
|
||||
} else {
|
||||
//
|
||||
// BSP will be Relocated later
|
||||
//
|
||||
BspIndex = Index;
|
||||
}
|
||||
}
|
||||
|
||||
//
|
||||
// Relocate BSP's SMM base
|
||||
//
|
||||
ASSERT (BspIndex != (UINTN)-1);
|
||||
mRebased = FALSE;
|
||||
mSmBase = GetSmBase (BspIndex, SmmRelocationStart, TileSize);
|
||||
SendSmiIpi (BspApicId);
|
||||
|
||||
//
|
||||
// Wait for the BSP to finish its 1st SMI
|
||||
//
|
||||
while (!mRebased) {
|
||||
}
|
||||
|
||||
//
|
||||
// Restore contents at address 0x38000
|
||||
//
|
||||
CopyMem (CpuStatePtr, &BakBuf2, sizeof (BakBuf2));
|
||||
CopyMem (U8Ptr, BakBuf, sizeof (BakBuf));
|
||||
}
|
||||
|
||||
/**
|
||||
Initialize IDT to setup exception handlers in SMM.
|
||||
|
||||
**/
|
||||
VOID
|
||||
InitSmmIdt (
|
||||
VOID
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
BOOLEAN InterruptState;
|
||||
IA32_DESCRIPTOR PeiIdtr;
|
||||
CONST EFI_PEI_SERVICES **PeiServices;
|
||||
|
||||
//
|
||||
// There are 32 (not 255) entries in it since only processor
|
||||
// generated exceptions will be handled.
|
||||
//
|
||||
gcSmmInitIdtr.Limit = (sizeof (IA32_IDT_GATE_DESCRIPTOR) * 32) - 1;
|
||||
|
||||
//
|
||||
// Allocate for IDT.
|
||||
// sizeof (UINTN) is for the PEI Services Table pointer.
|
||||
//
|
||||
gcSmmInitIdtr.Base = (UINTN)AllocateZeroPool (gcSmmInitIdtr.Limit + 1 + sizeof (UINTN));
|
||||
ASSERT (gcSmmInitIdtr.Base != 0);
|
||||
gcSmmInitIdtr.Base += sizeof (UINTN);
|
||||
|
||||
//
|
||||
// Disable Interrupt, save InterruptState and save PEI IDT table
|
||||
//
|
||||
InterruptState = SaveAndDisableInterrupts ();
|
||||
AsmReadIdtr (&PeiIdtr);
|
||||
|
||||
//
|
||||
// Save the PEI Services Table pointer
|
||||
// The PEI Services Table pointer will be stored in the sizeof (UINTN) bytes
|
||||
// immediately preceding the IDT in memory.
|
||||
//
|
||||
PeiServices = (CONST EFI_PEI_SERVICES **)(*(UINTN *)(PeiIdtr.Base - sizeof (UINTN)));
|
||||
(*(UINTN *)(gcSmmInitIdtr.Base - sizeof (UINTN))) = (UINTN)PeiServices;
|
||||
|
||||
//
|
||||
// Load SMM temporary IDT table
|
||||
//
|
||||
AsmWriteIdtr (&gcSmmInitIdtr);
|
||||
|
||||
//
|
||||
// Setup SMM default exception handlers, SMM IDT table
|
||||
// will be updated and saved in gcSmmInitIdtr
|
||||
//
|
||||
Status = InitializeCpuExceptionHandlers (NULL);
|
||||
ASSERT_EFI_ERROR (Status);
|
||||
|
||||
//
|
||||
// Restore PEI IDT table and CPU InterruptState
|
||||
//
|
||||
AsmWriteIdtr ((IA32_DESCRIPTOR *)&PeiIdtr);
|
||||
SetInterruptState (InterruptState);
|
||||
}
|
||||
|
||||
/**
|
||||
This routine will split SmramReserve HOB to reserve SmmRelocationSize for Smm relocated memory.
|
||||
|
||||
@param[in] SmmRelocationSize SmmRelocationSize for all processors.
|
||||
@param[in,out] SmmRelocationStart Return the start address of Smm relocated memory in SMRAM.
|
||||
|
||||
@retval EFI_SUCCESS The gEfiSmmSmramMemoryGuid is split successfully.
|
||||
@retval EFI_DEVICE_ERROR Failed to build new HOB for gEfiSmmSmramMemoryGuid.
|
||||
@retval EFI_NOT_FOUND The gEfiSmmSmramMemoryGuid is not found.
|
||||
|
||||
**/
|
||||
EFI_STATUS
|
||||
SplitSmramHobForSmmRelocation (
|
||||
IN UINT64 SmmRelocationSize,
|
||||
IN OUT EFI_PHYSICAL_ADDRESS *SmmRelocationStart
|
||||
)
|
||||
{
|
||||
EFI_HOB_GUID_TYPE *GuidHob;
|
||||
EFI_SMRAM_HOB_DESCRIPTOR_BLOCK *Block;
|
||||
EFI_SMRAM_HOB_DESCRIPTOR_BLOCK *NewBlock;
|
||||
UINTN NewBlockSize;
|
||||
|
||||
ASSERT (SmmRelocationStart != NULL);
|
||||
|
||||
//
|
||||
// Retrieve the GUID HOB data that contains the set of SMRAM descriptors
|
||||
//
|
||||
GuidHob = GetFirstGuidHob (&gEfiSmmSmramMemoryGuid);
|
||||
if (GuidHob == NULL) {
|
||||
return EFI_NOT_FOUND;
|
||||
}
|
||||
|
||||
Block = (EFI_SMRAM_HOB_DESCRIPTOR_BLOCK *)GET_GUID_HOB_DATA (GuidHob);
|
||||
|
||||
//
|
||||
// Allocate one extra EFI_SMRAM_DESCRIPTOR to describe smram carved out for all SMBASE
|
||||
//
|
||||
NewBlockSize = sizeof (EFI_SMRAM_HOB_DESCRIPTOR_BLOCK) + (Block->NumberOfSmmReservedRegions * sizeof (EFI_SMRAM_DESCRIPTOR));
|
||||
|
||||
NewBlock = (EFI_SMRAM_HOB_DESCRIPTOR_BLOCK *)BuildGuidHob (
|
||||
&gEfiSmmSmramMemoryGuid,
|
||||
NewBlockSize
|
||||
);
|
||||
ASSERT (NewBlock != NULL);
|
||||
if (NewBlock == NULL) {
|
||||
return EFI_DEVICE_ERROR;
|
||||
}
|
||||
|
||||
//
|
||||
// Copy old EFI_SMRAM_HOB_DESCRIPTOR_BLOCK to new allocated region
|
||||
//
|
||||
CopyMem ((VOID *)NewBlock, Block, NewBlockSize - sizeof (EFI_SMRAM_DESCRIPTOR));
|
||||
|
||||
//
|
||||
// Increase the number of SMRAM descriptors by 1 to make room for the ALLOCATED descriptor of size EFI_PAGE_SIZE
|
||||
//
|
||||
NewBlock->NumberOfSmmReservedRegions = (UINT32)(Block->NumberOfSmmReservedRegions + 1);
|
||||
|
||||
ASSERT (Block->NumberOfSmmReservedRegions >= 1);
|
||||
//
|
||||
// Copy last entry to the end - we assume TSEG is last entry.
|
||||
//
|
||||
CopyMem (&NewBlock->Descriptor[Block->NumberOfSmmReservedRegions], &NewBlock->Descriptor[Block->NumberOfSmmReservedRegions - 1], sizeof (EFI_SMRAM_DESCRIPTOR));
|
||||
|
||||
//
|
||||
// Update the entry in the array with a size of SmmRelocationSize and put into the ALLOCATED state
|
||||
//
|
||||
NewBlock->Descriptor[Block->NumberOfSmmReservedRegions - 1].PhysicalSize = SmmRelocationSize;
|
||||
NewBlock->Descriptor[Block->NumberOfSmmReservedRegions - 1].RegionState |= EFI_ALLOCATED;
|
||||
|
||||
//
|
||||
// Return the start address of Smm relocated memory in SMRAM.
|
||||
//
|
||||
*SmmRelocationStart = NewBlock->Descriptor[Block->NumberOfSmmReservedRegions - 1].CpuStart;
|
||||
|
||||
//
|
||||
// Reduce the size of the last SMRAM descriptor by SmmRelocationSize
|
||||
//
|
||||
NewBlock->Descriptor[Block->NumberOfSmmReservedRegions].PhysicalStart += SmmRelocationSize;
|
||||
NewBlock->Descriptor[Block->NumberOfSmmReservedRegions].CpuStart += SmmRelocationSize;
|
||||
NewBlock->Descriptor[Block->NumberOfSmmReservedRegions].PhysicalSize -= SmmRelocationSize;
|
||||
|
||||
//
|
||||
// Last step, we can scrub old one
|
||||
//
|
||||
ZeroMem (&GuidHob->Name, sizeof (GuidHob->Name));
|
||||
|
||||
return EFI_SUCCESS;
|
||||
}
|
||||
|
||||
/**
|
||||
CPU SmmBase Relocation Init.
|
||||
|
||||
This function is to relocate CPU SmmBase.
|
||||
|
||||
@param[in] MpServices2 Pointer to this instance of the MpServices.
|
||||
|
||||
@retval EFI_SUCCESS CPU SmmBase Relocated successfully.
|
||||
@retval Others CPU SmmBase Relocation failed.
|
||||
|
||||
**/
|
||||
EFI_STATUS
|
||||
EFIAPI
|
||||
SmmRelocationInit (
|
||||
IN EDKII_PEI_MP_SERVICES2_PPI *MpServices2
|
||||
)
|
||||
{
|
||||
EFI_STATUS Status;
|
||||
UINTN NumberOfEnabledCpus;
|
||||
UINTN TileSize;
|
||||
UINT64 SmmRelocationSize;
|
||||
EFI_PHYSICAL_ADDRESS SmmRelocationStart;
|
||||
UINTN SmmStackSize;
|
||||
UINT8 *SmmStacks;
|
||||
|
||||
SmmRelocationStart = 0;
|
||||
SmmStacks = NULL;
|
||||
|
||||
DEBUG ((DEBUG_INFO, "SmmRelocationInit Start \n"));
|
||||
if (MpServices2 == NULL) {
|
||||
return EFI_INVALID_PARAMETER;
|
||||
}
|
||||
|
||||
//
|
||||
// Get the number of processors
|
||||
//
|
||||
Status = MpServices2->GetNumberOfProcessors (
|
||||
MpServices2,
|
||||
&mNumberOfCpus,
|
||||
&NumberOfEnabledCpus
|
||||
);
|
||||
if (EFI_ERROR (Status)) {
|
||||
goto ON_EXIT;
|
||||
}
|
||||
|
||||
if (FeaturePcdGet (PcdCpuHotPlugSupport)) {
|
||||
mMaxNumberOfCpus = PcdGet32 (PcdCpuMaxLogicalProcessorNumber);
|
||||
} else {
|
||||
mMaxNumberOfCpus = mNumberOfCpus;
|
||||
}
|
||||
|
||||
ASSERT (mNumberOfCpus <= mMaxNumberOfCpus);
|
||||
|
||||
//
|
||||
// Calculate SmmRelocationSize for all of the tiles.
|
||||
//
|
||||
// The CPU save state and code for the SMI entry point are tiled within an SMRAM
|
||||
// allocated buffer. The minimum size of this buffer for a uniprocessor system
|
||||
// is 32 KB, because the entry point is SMBASE + 32KB, and CPU save state area
|
||||
// just below SMBASE + 64KB. If more than one CPU is present in the platform,
|
||||
// then the SMI entry point and the CPU save state areas can be tiles to minimize
|
||||
// the total amount SMRAM required for all the CPUs. The tile size can be computed
|
||||
// by adding the CPU save state size, any extra CPU specific context, and
|
||||
// the size of code that must be placed at the SMI entry point to transfer
|
||||
// control to a C function in the native SMM execution mode. This size is
|
||||
// rounded up to the nearest power of 2 to give the tile size for a each CPU.
|
||||
// The total amount of memory required is the maximum number of CPUs that
|
||||
// platform supports times the tile size.
|
||||
//
|
||||
TileSize = SIZE_8KB;
|
||||
SmmRelocationSize = EFI_PAGES_TO_SIZE (EFI_SIZE_TO_PAGES (SIZE_32KB + TileSize * (mMaxNumberOfCpus - 1)));
|
||||
|
||||
//
|
||||
// Split SmramReserve HOB to reserve SmmRelocationSize for Smm relocated memory
|
||||
//
|
||||
Status = SplitSmramHobForSmmRelocation (
|
||||
SmmRelocationSize,
|
||||
&SmmRelocationStart
|
||||
);
|
||||
if (EFI_ERROR (Status)) {
|
||||
goto ON_EXIT;
|
||||
}
|
||||
|
||||
ASSERT (SmmRelocationStart != 0);
|
||||
DEBUG ((DEBUG_INFO, "SmmRelocationInit - SmmRelocationSize: 0x%08x\n", SmmRelocationSize));
|
||||
DEBUG ((DEBUG_INFO, "SmmRelocationInit - SmmRelocationStart: 0x%08x\n", SmmRelocationStart));
|
||||
|
||||
//
|
||||
// Fix up the address of the global variable or function referred in
|
||||
// SmmInit assembly files to be the absolute address
|
||||
//
|
||||
SmmInitFixupAddress ();
|
||||
|
||||
//
|
||||
// Patch SMI stack for SMM base relocation
|
||||
// Note: No need allocate stack for all CPUs since the relocation
|
||||
// occurs serially for each CPU
|
||||
//
|
||||
SmmStackSize = EFI_PAGE_SIZE;
|
||||
SmmStacks = (UINT8 *)AllocatePages (EFI_SIZE_TO_PAGES (SmmStackSize));
|
||||
if (SmmStacks == NULL) {
|
||||
Status = EFI_OUT_OF_RESOURCES;
|
||||
goto ON_EXIT;
|
||||
}
|
||||
|
||||
DEBUG ((DEBUG_INFO, "SmmRelocationInit - SmmStackSize: 0x%08x\n", SmmStackSize));
|
||||
DEBUG ((DEBUG_INFO, "SmmRelocationInit - SmmStacks: 0x%08x\n", SmmStacks));
|
||||
|
||||
PatchInstructionX86 (
|
||||
gPatchSmmInitStack,
|
||||
(UINTN)(SmmStacks + SmmStackSize - sizeof (UINTN)),
|
||||
sizeof (UINTN)
|
||||
);
|
||||
|
||||
//
|
||||
// Initialize the SMM IDT for SMM base relocation
|
||||
//
|
||||
InitSmmIdt ();
|
||||
|
||||
//
|
||||
// Relocate SmmBases for each processor.
|
||||
//
|
||||
SmmRelocateBases (MpServices2, SmmRelocationStart, TileSize);
|
||||
|
||||
//
|
||||
// Create the SmBase HOB for all CPUs
|
||||
//
|
||||
Status = CreateSmmBaseHob (SmmRelocationStart, TileSize);
|
||||
|
||||
ON_EXIT:
|
||||
if (SmmStacks != NULL) {
|
||||
FreePages (SmmStacks, EFI_SIZE_TO_PAGES (SmmStackSize));
|
||||
}
|
||||
|
||||
DEBUG ((DEBUG_INFO, "SmmRelocationInit Done\n"));
|
||||
return Status;
|
||||
}
|
Reference in New Issue
Block a user