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>
		
	
		
			
				
	
	
		
			550 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			550 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @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;
 | |
| }
 |