| /** @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 EFI_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 EFI_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; | |
| } |