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:
		| @@ -110,6 +110,7 @@ | ||||
| !include MdePkg/MdeLibs.dsc.inc | ||||
|  | ||||
| [LibraryClasses] | ||||
|   SmmRelocationLib|OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.inf | ||||
|   PcdLib|MdePkg/Library/BasePcdLibNull/BasePcdLibNull.inf | ||||
|   TimerLib|OvmfPkg/Library/AcpiTimerLib/BaseAcpiTimerLib.inf | ||||
|   ResetSystemLib|OvmfPkg/Library/ResetSystemLib/BaseResetSystemLib.inf | ||||
|   | ||||
| @@ -126,6 +126,7 @@ | ||||
| !include MdePkg/MdeLibs.dsc.inc | ||||
|  | ||||
| [LibraryClasses] | ||||
|   SmmRelocationLib|OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.inf | ||||
|   PcdLib|MdePkg/Library/BasePcdLibNull/BasePcdLibNull.inf | ||||
|   TimerLib|OvmfPkg/Library/AcpiTimerLib/BaseAcpiTimerLib.inf | ||||
|   ResetSystemLib|OvmfPkg/Library/ResetSystemLib/BaseResetSystemLib.inf | ||||
|   | ||||
							
								
								
									
										39
									
								
								OvmfPkg/Library/SmmRelocationLib/Ia32/Semaphore.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								OvmfPkg/Library/SmmRelocationLib/Ia32/Semaphore.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| /** @file | ||||
|   Semaphore mechanism to indicate to the BSP that an AP has exited SMM | ||||
|   after SMBASE relocation. | ||||
|  | ||||
|   Copyright (c) 2024, Intel Corporation. All rights reserved.<BR> | ||||
|   SPDX-License-Identifier: BSD-2-Clause-Patent | ||||
|  | ||||
| **/ | ||||
|  | ||||
| #include "InternalSmmRelocationLib.h" | ||||
|  | ||||
| UINTN             mSmmRelocationOriginalAddress; | ||||
| volatile BOOLEAN  *mRebasedFlag; | ||||
|  | ||||
| /** | ||||
|   Hook return address of SMM Save State so that semaphore code | ||||
|   can be executed immediately after AP exits SMM to indicate to | ||||
|   the BSP that an AP has exited SMM after SMBASE relocation. | ||||
|  | ||||
|   @param[in] RebasedFlag  A pointer to a flag that is set to TRUE | ||||
|                           immediately after AP exits SMM. | ||||
|  | ||||
| **/ | ||||
| VOID | ||||
| SemaphoreHook ( | ||||
|   IN volatile BOOLEAN  *RebasedFlag | ||||
|   ) | ||||
| { | ||||
|   SMRAM_SAVE_STATE_MAP  *CpuState; | ||||
|  | ||||
|   mRebasedFlag = RebasedFlag; | ||||
|  | ||||
|   CpuState                      = (SMRAM_SAVE_STATE_MAP *)(UINTN)(SMM_DEFAULT_SMBASE + SMRAM_SAVE_STATE_MAP_OFFSET); | ||||
|   mSmmRelocationOriginalAddress = (UINTN)HookReturnFromSmm ( | ||||
|                                            CpuState, | ||||
|                                            (UINT64)(UINTN)&SmmRelocationSemaphoreComplete, | ||||
|                                            (UINT64)(UINTN)&SmmRelocationSemaphoreComplete | ||||
|                                            ); | ||||
| } | ||||
							
								
								
									
										151
									
								
								OvmfPkg/Library/SmmRelocationLib/Ia32/SmmInit.nasm
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								OvmfPkg/Library/SmmRelocationLib/Ia32/SmmInit.nasm
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,151 @@ | ||||
| ;------------------------------------------------------------------------------ ; | ||||
| ; Copyright (c) 2024, Intel Corporation. All rights reserved.<BR> | ||||
| ; SPDX-License-Identifier: BSD-2-Clause-Patent | ||||
| ; | ||||
| ; Module Name: | ||||
| ; | ||||
| ;   SmmInit.nasm | ||||
| ; | ||||
| ; Abstract: | ||||
| ; | ||||
| ;   Functions for relocating SMBASE's for all processors | ||||
| ; | ||||
| ;------------------------------------------------------------------------------- | ||||
|  | ||||
| %include "StuffRsbNasm.inc" | ||||
|  | ||||
| extern ASM_PFX(SmmInitHandler) | ||||
| extern ASM_PFX(mRebasedFlag) | ||||
| extern ASM_PFX(mSmmRelocationOriginalAddress) | ||||
|  | ||||
| global ASM_PFX(gPatchSmmInitCr3) | ||||
| global ASM_PFX(gPatchSmmInitCr4) | ||||
| global ASM_PFX(gPatchSmmInitCr0) | ||||
| global ASM_PFX(gPatchSmmInitStack) | ||||
| global ASM_PFX(gcSmmInitGdtr) | ||||
| global ASM_PFX(gcSmmInitSize) | ||||
| global ASM_PFX(gcSmmInitTemplate) | ||||
|  | ||||
| %define PROTECT_MODE_CS 0x8 | ||||
| %define PROTECT_MODE_DS 0x20 | ||||
|  | ||||
|     SECTION .data | ||||
|  | ||||
| NullSeg: DQ 0                   ; reserved by architecture | ||||
| CodeSeg32: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x9b | ||||
|             DB      0xcf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| ProtModeCodeSeg32: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x9b | ||||
|             DB      0xcf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| ProtModeSsSeg32: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x93 | ||||
|             DB      0xcf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| DataSeg32: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x93 | ||||
|             DB      0xcf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| CodeSeg16: | ||||
|             DW      -1 | ||||
|             DW      0 | ||||
|             DB      0 | ||||
|             DB      0x9b | ||||
|             DB      0x8f | ||||
|             DB      0 | ||||
| DataSeg16: | ||||
|             DW      -1 | ||||
|             DW      0 | ||||
|             DB      0 | ||||
|             DB      0x93 | ||||
|             DB      0x8f | ||||
|             DB      0 | ||||
| CodeSeg64: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x9b | ||||
|             DB      0xaf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| GDT_SIZE equ $ - NullSeg | ||||
|  | ||||
| ASM_PFX(gcSmmInitGdtr): | ||||
|     DW      GDT_SIZE - 1 | ||||
|     DD      NullSeg | ||||
|  | ||||
|  | ||||
|     SECTION .text | ||||
|  | ||||
| global ASM_PFX(SmmStartup) | ||||
|  | ||||
| BITS 16 | ||||
| ASM_PFX(SmmStartup): | ||||
|     mov     eax, 0x80000001             ; read capability | ||||
|     cpuid | ||||
|     mov     ebx, edx                    ; rdmsr will change edx. keep it in ebx. | ||||
|     and     ebx, BIT20                  ; extract NX capability bit | ||||
|     shr     ebx, 9                      ; shift bit to IA32_EFER.NXE[BIT11] position | ||||
|     mov     eax, strict dword 0         ; source operand will be patched | ||||
| ASM_PFX(gPatchSmmInitCr3): | ||||
|     mov     cr3, eax | ||||
| o32 lgdt    [cs:ebp + (ASM_PFX(gcSmmInitGdtr) - ASM_PFX(SmmStartup))] | ||||
|     mov     eax, strict dword 0         ; source operand will be patched | ||||
| ASM_PFX(gPatchSmmInitCr4): | ||||
|     mov     cr4, eax | ||||
|     mov     ecx, 0xc0000080             ; IA32_EFER MSR | ||||
|     rdmsr | ||||
|     or      eax, ebx                    ; set NXE bit if NX is available | ||||
|     wrmsr | ||||
|     mov     eax, strict dword 0         ; source operand will be patched | ||||
| ASM_PFX(gPatchSmmInitCr0): | ||||
|     mov     di, PROTECT_MODE_DS | ||||
|     mov     cr0, eax | ||||
|     jmp     PROTECT_MODE_CS : dword @32bit | ||||
|  | ||||
| BITS 32 | ||||
| @32bit: | ||||
|     mov     ds, edi | ||||
|     mov     es, edi | ||||
|     mov     fs, edi | ||||
|     mov     gs, edi | ||||
|     mov     ss, edi | ||||
|     mov     esp, strict dword 0         ; source operand will be patched | ||||
| ASM_PFX(gPatchSmmInitStack): | ||||
|     call    ASM_PFX(SmmInitHandler) | ||||
|     StuffRsb32 | ||||
|     rsm | ||||
|  | ||||
| BITS 16 | ||||
| ASM_PFX(gcSmmInitTemplate): | ||||
|     mov ebp, ASM_PFX(SmmStartup) | ||||
|     sub ebp, 0x30000 | ||||
|     jmp ebp | ||||
|  | ||||
| ASM_PFX(gcSmmInitSize): DW $ - ASM_PFX(gcSmmInitTemplate) | ||||
|  | ||||
| BITS 32 | ||||
| global ASM_PFX(SmmRelocationSemaphoreComplete) | ||||
| ASM_PFX(SmmRelocationSemaphoreComplete): | ||||
|     push    eax | ||||
|     mov     eax, [ASM_PFX(mRebasedFlag)] | ||||
|     mov     byte [eax], 1 | ||||
|     pop     eax | ||||
|     jmp     [ASM_PFX(mSmmRelocationOriginalAddress)] | ||||
|  | ||||
| global ASM_PFX(SmmInitFixupAddress) | ||||
| ASM_PFX(SmmInitFixupAddress): | ||||
|     ret | ||||
							
								
								
									
										127
									
								
								OvmfPkg/Library/SmmRelocationLib/InternalSmmRelocationLib.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								OvmfPkg/Library/SmmRelocationLib/InternalSmmRelocationLib.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,127 @@ | ||||
| /** @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 | ||||
|  | ||||
| **/ | ||||
|  | ||||
| #ifndef INTERNAL_SMM_RELOCATION_LIB_H_ | ||||
| #define INTERNAL_SMM_RELOCATION_LIB_H_ | ||||
|  | ||||
| #include <Library/BaseLib.h> | ||||
| #include <Library/BaseMemoryLib.h> | ||||
| #include <Library/CpuExceptionHandlerLib.h> | ||||
| #include <Library/DebugLib.h> | ||||
| #include <Library/HobLib.h> | ||||
| #include <Library/LocalApicLib.h> | ||||
| #include <Library/MemoryAllocationLib.h> | ||||
| #include <Library/PcdLib.h> | ||||
| #include <Library/PeimEntryPoint.h> | ||||
| #include <Library/PeiServicesLib.h> | ||||
| #include <Library/SmmRelocationLib.h> | ||||
| #include <Guid/SmramMemoryReserve.h> | ||||
| #include <Guid/SmmBaseHob.h> | ||||
| #include <Register/Intel/Cpuid.h> | ||||
| #include <Register/Intel/SmramSaveStateMap.h> | ||||
| #include <Protocol/MmCpu.h> | ||||
|  | ||||
| extern IA32_DESCRIPTOR  gcSmmInitGdtr; | ||||
| extern CONST UINT16     gcSmmInitSize; | ||||
| extern CONST UINT8      gcSmmInitTemplate[]; | ||||
|  | ||||
| X86_ASSEMBLY_PATCH_LABEL  gPatchSmmInitCr0; | ||||
| X86_ASSEMBLY_PATCH_LABEL  gPatchSmmInitCr3; | ||||
| X86_ASSEMBLY_PATCH_LABEL  gPatchSmmInitCr4; | ||||
| X86_ASSEMBLY_PATCH_LABEL  gPatchSmmInitStack; | ||||
|  | ||||
| // | ||||
| // The size 0x20 must be bigger than | ||||
| // the size of template code of SmmInit. Currently, | ||||
| // the size of SmmInit requires the 0x16 Bytes buffer | ||||
| // at least. | ||||
| // | ||||
| #define BACK_BUF_SIZE  0x20 | ||||
|  | ||||
| #define CR4_CET_ENABLE  BIT23 | ||||
|  | ||||
| // | ||||
| // EFER register LMA bit | ||||
| // | ||||
| #define LMA  BIT10 | ||||
|  | ||||
| /** | ||||
|   This function configures the SmBase on the currently executing CPU. | ||||
|  | ||||
|   @param[in]     SmBase             The SmBase on the currently executing CPU. | ||||
|  | ||||
| **/ | ||||
| VOID | ||||
| EFIAPI | ||||
| ConfigureSmBase ( | ||||
|   IN UINT64  SmBase | ||||
|   ); | ||||
|  | ||||
| /** | ||||
|   Semaphore operation for all processor relocate SMMBase. | ||||
| **/ | ||||
| VOID | ||||
| EFIAPI | ||||
| SmmRelocationSemaphoreComplete ( | ||||
|   VOID | ||||
|   ); | ||||
|  | ||||
| /** | ||||
|   Hook the code executed immediately after an RSM instruction on the currently | ||||
|   executing CPU.  The mode of code executed immediately after RSM must be | ||||
|   detected, and the appropriate hook must be selected.  Always clear the auto | ||||
|   HALT restart flag if it is set. | ||||
|  | ||||
|   @param[in,out] CpuState                 Pointer to SMRAM Save State Map for the | ||||
|                                           currently executing CPU. | ||||
|   @param[in]     NewInstructionPointer32  Instruction pointer to use if resuming to | ||||
|                                           32-bit mode from 64-bit SMM. | ||||
|   @param[in]     NewInstructionPointer    Instruction pointer to use if resuming to | ||||
|                                           same mode as SMM. | ||||
|  | ||||
|   @retval The value of the original instruction pointer before it was hooked. | ||||
|  | ||||
| **/ | ||||
| UINT64 | ||||
| EFIAPI | ||||
| HookReturnFromSmm ( | ||||
|   IN OUT SMRAM_SAVE_STATE_MAP  *CpuState, | ||||
|   IN     UINT64                NewInstructionPointer32, | ||||
|   IN     UINT64                NewInstructionPointer | ||||
|   ); | ||||
|  | ||||
| /** | ||||
|   Hook return address of SMM Save State so that semaphore code | ||||
|   can be executed immediately after AP exits SMM to indicate to | ||||
|   the BSP that an AP has exited SMM after SMBASE relocation. | ||||
|  | ||||
|   @param[in] RebasedFlag  A pointer to a flag that is set to TRUE | ||||
|                           immediately after AP exits SMM. | ||||
|  | ||||
| **/ | ||||
| VOID | ||||
| SemaphoreHook ( | ||||
|   IN volatile BOOLEAN  *RebasedFlag | ||||
|   ); | ||||
|  | ||||
| /** | ||||
|   This function fixes up the address of the global variable or function | ||||
|   referred in SmmInit assembly files to be the absolute address. | ||||
| **/ | ||||
| VOID | ||||
| EFIAPI | ||||
| SmmInitFixupAddress ( | ||||
|   ); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										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; | ||||
| } | ||||
							
								
								
									
										60
									
								
								OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.inf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.inf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| ## @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 | ||||
| # | ||||
| ## | ||||
|  | ||||
| [Defines] | ||||
|   INF_VERSION                    = 0x00010005 | ||||
|   BASE_NAME                      = SmmRelocationLib | ||||
|   FILE_GUID                      = 51834F51-CCE0-4743-B553-935D0C8A53FF | ||||
|   MODULE_TYPE                    = PEIM | ||||
|   VERSION_STRING                 = 1.0 | ||||
|   LIBRARY_CLASS                  = SmmRelocationLib | ||||
|  | ||||
| [Sources] | ||||
|   InternalSmmRelocationLib.h | ||||
|   SmramSaveStateConfig.c | ||||
|   SmmRelocationLib.c | ||||
|  | ||||
| [Sources.Ia32] | ||||
|   Ia32/Semaphore.c | ||||
|   Ia32/SmmInit.nasm | ||||
|  | ||||
| [Sources.X64] | ||||
|   X64/Semaphore.c | ||||
|   X64/SmmInit.nasm | ||||
|  | ||||
| [Packages] | ||||
|   MdePkg/MdePkg.dec | ||||
|   MdeModulePkg/MdeModulePkg.dec | ||||
|   UefiCpuPkg/UefiCpuPkg.dec | ||||
|  | ||||
| [LibraryClasses] | ||||
|   BaseLib | ||||
|   BaseMemoryLib | ||||
|   CpuExceptionHandlerLib | ||||
|   DebugLib | ||||
|   HobLib | ||||
|   LocalApicLib | ||||
|   MemoryAllocationLib | ||||
|   PcdLib | ||||
|   PeiServicesLib | ||||
|  | ||||
| [Guids] | ||||
|   gSmmBaseHobGuid                               ## HOB ALWAYS_PRODUCED | ||||
|   gEfiSmmSmramMemoryGuid                        ## CONSUMES | ||||
|  | ||||
| [Pcd] | ||||
|   gUefiCpuPkgTokenSpaceGuid.PcdCpuMaxLogicalProcessorNumber | ||||
|  | ||||
| [FeaturePcd] | ||||
|   gUefiCpuPkgTokenSpaceGuid.PcdCpuHotPlugSupport                        ## CONSUMES | ||||
							
								
								
									
										100
									
								
								OvmfPkg/Library/SmmRelocationLib/SmramSaveStateConfig.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								OvmfPkg/Library/SmmRelocationLib/SmramSaveStateConfig.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | ||||
| /** @file | ||||
|   Config SMRAM Save State for SmmBases Relocation. | ||||
|  | ||||
|   Copyright (c) 2024, Intel Corporation. All rights reserved.<BR> | ||||
|   SPDX-License-Identifier: BSD-2-Clause-Patent | ||||
|  | ||||
| **/ | ||||
| #include "InternalSmmRelocationLib.h" | ||||
| #include <Register/Amd/SmramSaveStateMap.h> | ||||
|  | ||||
| /** | ||||
|   This function configures the SmBase on the currently executing CPU. | ||||
|  | ||||
|   @param[in]     SmBase             The SmBase on the currently executing CPU. | ||||
|  | ||||
| **/ | ||||
| VOID | ||||
| EFIAPI | ||||
| ConfigureSmBase ( | ||||
|   IN UINT64  SmBase | ||||
|   ) | ||||
| { | ||||
|   AMD_SMRAM_SAVE_STATE_MAP  *CpuSaveState; | ||||
|  | ||||
|   CpuSaveState = (AMD_SMRAM_SAVE_STATE_MAP *)(UINTN)(SMM_DEFAULT_SMBASE + SMRAM_SAVE_STATE_MAP_OFFSET); | ||||
|  | ||||
|   if ((CpuSaveState->x86.SMMRevId & 0xFFFF) == 0) { | ||||
|     CpuSaveState->x86.SMBASE = (UINT32)SmBase; | ||||
|   } else { | ||||
|     CpuSaveState->x64.SMBASE = (UINT32)SmBase; | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** | ||||
|   This function updates the SMRAM save state on the currently executing CPU | ||||
|   to resume execution at a specific address after an RSM instruction.  This | ||||
|   function must evaluate the SMRAM save state to determine the execution mode | ||||
|   the RSM instruction resumes and update the resume execution address with | ||||
|   either NewInstructionPointer32 or NewInstructionPoint.  The auto HALT restart | ||||
|   flag in the SMRAM save state must always be cleared.  This function returns | ||||
|   the value of the instruction pointer from the SMRAM save state that was | ||||
|   replaced.  If this function returns 0, then the SMRAM save state was not | ||||
|   modified. | ||||
|  | ||||
|   This function is called during the very first SMI on each CPU after | ||||
|   SmmCpuFeaturesInitializeProcessor() to set a flag in normal execution mode | ||||
|   to signal that the SMBASE of each CPU has been updated before the default | ||||
|   SMBASE address is used for the first SMI to the next CPU. | ||||
|  | ||||
|   @param[in,out] CpuState                 Pointer to SMRAM Save State Map for the | ||||
|                                           currently executing CPU. | ||||
|   @param[in]     NewInstructionPointer32  Instruction pointer to use if resuming to | ||||
|                                           32-bit mode from 64-bit SMM. | ||||
|   @param[in]     NewInstructionPointer    Instruction pointer to use if resuming to | ||||
|                                           same mode as SMM. | ||||
|  | ||||
|   @retval The value of the original instruction pointer before it was hooked. | ||||
|  | ||||
| **/ | ||||
| UINT64 | ||||
| EFIAPI | ||||
| HookReturnFromSmm ( | ||||
|   IN OUT SMRAM_SAVE_STATE_MAP  *CpuState, | ||||
|   IN     UINT64                NewInstructionPointer32, | ||||
|   IN     UINT64                NewInstructionPointer | ||||
|   ) | ||||
| { | ||||
|   UINT64                    OriginalInstructionPointer; | ||||
|   AMD_SMRAM_SAVE_STATE_MAP  *CpuSaveState; | ||||
|  | ||||
|   CpuSaveState = (AMD_SMRAM_SAVE_STATE_MAP *)CpuState; | ||||
|   if ((CpuSaveState->x86.SMMRevId & 0xFFFF) == 0) { | ||||
|     OriginalInstructionPointer = (UINT64)CpuSaveState->x86._EIP; | ||||
|     CpuSaveState->x86._EIP     = (UINT32)NewInstructionPointer; | ||||
|     // | ||||
|     // Clear the auto HALT restart flag so the RSM instruction returns | ||||
|     // program control to the instruction following the HLT instruction. | ||||
|     // | ||||
|     if ((CpuSaveState->x86.AutoHALTRestart & BIT0) != 0) { | ||||
|       CpuSaveState->x86.AutoHALTRestart &= ~BIT0; | ||||
|     } | ||||
|   } else { | ||||
|     OriginalInstructionPointer = CpuSaveState->x64._RIP; | ||||
|     if ((CpuSaveState->x64.EFER & LMA) == 0) { | ||||
|       CpuSaveState->x64._RIP = (UINT32)NewInstructionPointer32; | ||||
|     } else { | ||||
|       CpuSaveState->x64._RIP = (UINT32)NewInstructionPointer; | ||||
|     } | ||||
|  | ||||
|     // | ||||
|     // Clear the auto HALT restart flag so the RSM instruction returns | ||||
|     // program control to the instruction following the HLT instruction. | ||||
|     // | ||||
|     if ((CpuSaveState->x64.AutoHALTRestart & BIT0) != 0) { | ||||
|       CpuSaveState->x64.AutoHALTRestart &= ~BIT0; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return OriginalInstructionPointer; | ||||
| } | ||||
							
								
								
									
										66
									
								
								OvmfPkg/Library/SmmRelocationLib/X64/Semaphore.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								OvmfPkg/Library/SmmRelocationLib/X64/Semaphore.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| /** @file | ||||
|   Semaphore mechanism to indicate to the BSP that an AP has exited SMM | ||||
|   after SMBASE relocation. | ||||
|  | ||||
|   Copyright (c) 2024, Intel Corporation. All rights reserved.<BR> | ||||
|   SPDX-License-Identifier: BSD-2-Clause-Patent | ||||
|  | ||||
| **/ | ||||
|  | ||||
| #include "InternalSmmRelocationLib.h" | ||||
|  | ||||
| X86_ASSEMBLY_PATCH_LABEL  gPatchSmmRelocationOriginalAddressPtr32; | ||||
| X86_ASSEMBLY_PATCH_LABEL  gPatchRebasedFlagAddr32; | ||||
|  | ||||
| UINTN             mSmmRelocationOriginalAddress; | ||||
| volatile BOOLEAN  *mRebasedFlag; | ||||
|  | ||||
| /** | ||||
| AP Semaphore operation in 32-bit mode while BSP runs in 64-bit mode. | ||||
| **/ | ||||
| VOID | ||||
| SmmRelocationSemaphoreComplete32 ( | ||||
|   VOID | ||||
|   ); | ||||
|  | ||||
| /** | ||||
|   Hook return address of SMM Save State so that semaphore code | ||||
|   can be executed immediately after AP exits SMM to indicate to | ||||
|   the BSP that an AP has exited SMM after SMBASE relocation. | ||||
|  | ||||
|   @param[in] RebasedFlag  A pointer to a flag that is set to TRUE | ||||
|                           immediately after AP exits SMM. | ||||
|  | ||||
| **/ | ||||
| VOID | ||||
| SemaphoreHook ( | ||||
|   IN volatile BOOLEAN  *RebasedFlag | ||||
|   ) | ||||
| { | ||||
|   SMRAM_SAVE_STATE_MAP  *CpuState; | ||||
|   UINTN                 TempValue; | ||||
|  | ||||
|   mRebasedFlag = RebasedFlag; | ||||
|   PatchInstructionX86 ( | ||||
|     gPatchRebasedFlagAddr32, | ||||
|     (UINT32)(UINTN)mRebasedFlag, | ||||
|     4 | ||||
|     ); | ||||
|  | ||||
|   CpuState                      = (SMRAM_SAVE_STATE_MAP *)(UINTN)(SMM_DEFAULT_SMBASE + SMRAM_SAVE_STATE_MAP_OFFSET); | ||||
|   mSmmRelocationOriginalAddress = HookReturnFromSmm ( | ||||
|                                     CpuState, | ||||
|                                     (UINT64)(UINTN)&SmmRelocationSemaphoreComplete32, | ||||
|                                     (UINT64)(UINTN)&SmmRelocationSemaphoreComplete | ||||
|                                     ); | ||||
|  | ||||
|   // | ||||
|   // Use temp value to fix ICC compiler warning | ||||
|   // | ||||
|   TempValue = (UINTN)&mSmmRelocationOriginalAddress; | ||||
|   PatchInstructionX86 ( | ||||
|     gPatchSmmRelocationOriginalAddressPtr32, | ||||
|     (UINT32)TempValue, | ||||
|     4 | ||||
|     ); | ||||
| } | ||||
							
								
								
									
										201
									
								
								OvmfPkg/Library/SmmRelocationLib/X64/SmmInit.nasm
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										201
									
								
								OvmfPkg/Library/SmmRelocationLib/X64/SmmInit.nasm
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,201 @@ | ||||
| ;------------------------------------------------------------------------------ ; | ||||
| ; Copyright (c) 2024, Intel Corporation. All rights reserved.<BR> | ||||
| ; SPDX-License-Identifier: BSD-2-Clause-Patent | ||||
| ; | ||||
| ; Module Name: | ||||
| ; | ||||
| ;   SmmInit.nasm | ||||
| ; | ||||
| ; Abstract: | ||||
| ; | ||||
| ;   Functions for relocating SMBASE's for all processors | ||||
| ; | ||||
| ;------------------------------------------------------------------------------- | ||||
|  | ||||
| %include "StuffRsbNasm.inc" | ||||
|  | ||||
| extern ASM_PFX(SmmInitHandler) | ||||
| extern ASM_PFX(mRebasedFlag) | ||||
| extern ASM_PFX(mSmmRelocationOriginalAddress) | ||||
|  | ||||
| global ASM_PFX(gPatchSmmInitCr3) | ||||
| global ASM_PFX(gPatchSmmInitCr4) | ||||
| global ASM_PFX(gPatchSmmInitCr0) | ||||
| global ASM_PFX(gPatchSmmInitStack) | ||||
| global ASM_PFX(gcSmmInitGdtr) | ||||
| global ASM_PFX(gcSmmInitSize) | ||||
| global ASM_PFX(gcSmmInitTemplate) | ||||
| global ASM_PFX(gPatchRebasedFlagAddr32) | ||||
| global ASM_PFX(gPatchSmmRelocationOriginalAddressPtr32) | ||||
|  | ||||
| %define LONG_MODE_CS 0x38 | ||||
|  | ||||
|     SECTION .data | ||||
|  | ||||
| NullSeg: DQ 0                   ; reserved by architecture | ||||
| CodeSeg32: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x9b | ||||
|             DB      0xcf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| ProtModeCodeSeg32: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x9b | ||||
|             DB      0xcf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| ProtModeSsSeg32: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x93 | ||||
|             DB      0xcf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| DataSeg32: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x93 | ||||
|             DB      0xcf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| CodeSeg16: | ||||
|             DW      -1 | ||||
|             DW      0 | ||||
|             DB      0 | ||||
|             DB      0x9b | ||||
|             DB      0x8f | ||||
|             DB      0 | ||||
| DataSeg16: | ||||
|             DW      -1 | ||||
|             DW      0 | ||||
|             DB      0 | ||||
|             DB      0x93 | ||||
|             DB      0x8f | ||||
|             DB      0 | ||||
| CodeSeg64: | ||||
|             DW      -1                  ; LimitLow | ||||
|             DW      0                   ; BaseLow | ||||
|             DB      0                   ; BaseMid | ||||
|             DB      0x9b | ||||
|             DB      0xaf                ; LimitHigh | ||||
|             DB      0                   ; BaseHigh | ||||
| GDT_SIZE equ $ -   NullSeg | ||||
|  | ||||
| ASM_PFX(gcSmmInitGdtr): | ||||
|     DW      GDT_SIZE - 1 | ||||
|     DQ      NullSeg | ||||
|  | ||||
|  | ||||
|     DEFAULT REL | ||||
|     SECTION .text | ||||
|  | ||||
| global ASM_PFX(SmmStartup) | ||||
|  | ||||
| BITS 16 | ||||
| ASM_PFX(SmmStartup): | ||||
|     mov     eax, 0x80000001             ; read capability | ||||
|     cpuid | ||||
|     mov     ebx, edx                    ; rdmsr will change edx. keep it in ebx. | ||||
|     mov     eax, strict dword 0         ; source operand will be patched | ||||
| ASM_PFX(gPatchSmmInitCr3): | ||||
|     mov     cr3, eax | ||||
| o32 lgdt    [cs:ebp + (ASM_PFX(gcSmmInitGdtr) - ASM_PFX(SmmStartup))] | ||||
|     mov     eax, strict dword 0         ; source operand will be patched | ||||
| ASM_PFX(gPatchSmmInitCr4): | ||||
|     or      ah,  2                      ; enable XMM registers access | ||||
|     mov     cr4, eax | ||||
|     mov     ecx, 0xc0000080             ; IA32_EFER MSR | ||||
|     rdmsr | ||||
|     or      ah, BIT0                    ; set LME bit | ||||
|     test    ebx, BIT20                  ; check NXE capability | ||||
|     jz      .1 | ||||
|     or      ah, BIT3                    ; set NXE bit | ||||
| .1: | ||||
|     wrmsr | ||||
|     mov     eax, strict dword 0         ; source operand will be patched | ||||
| ASM_PFX(gPatchSmmInitCr0): | ||||
|     mov     cr0, eax                    ; enable protected mode & paging | ||||
|     jmp     LONG_MODE_CS : dword 0      ; offset will be patched to @LongMode | ||||
| @PatchLongModeOffset: | ||||
|  | ||||
| BITS 64 | ||||
| @LongMode:                              ; long-mode starts here | ||||
|     mov     rsp, strict qword 0         ; source operand will be patched | ||||
| ASM_PFX(gPatchSmmInitStack): | ||||
|     and     sp, 0xfff0                  ; make sure RSP is 16-byte aligned | ||||
|     ; | ||||
|     ; According to X64 calling convention, XMM0~5 are volatile, we need to save | ||||
|     ; them before calling C-function. | ||||
|     ; | ||||
|     sub     rsp, 0x60 | ||||
|     movdqa  [rsp], xmm0 | ||||
|     movdqa  [rsp + 0x10], xmm1 | ||||
|     movdqa  [rsp + 0x20], xmm2 | ||||
|     movdqa  [rsp + 0x30], xmm3 | ||||
|     movdqa  [rsp + 0x40], xmm4 | ||||
|     movdqa  [rsp + 0x50], xmm5 | ||||
|  | ||||
|     add     rsp, -0x20 | ||||
|     call    ASM_PFX(SmmInitHandler) | ||||
|     add     rsp, 0x20 | ||||
|  | ||||
|     ; | ||||
|     ; Restore XMM0~5 after calling C-function. | ||||
|     ; | ||||
|     movdqa  xmm0, [rsp] | ||||
|     movdqa  xmm1, [rsp + 0x10] | ||||
|     movdqa  xmm2, [rsp + 0x20] | ||||
|     movdqa  xmm3, [rsp + 0x30] | ||||
|     movdqa  xmm4, [rsp + 0x40] | ||||
|     movdqa  xmm5, [rsp + 0x50] | ||||
|  | ||||
|     StuffRsb64 | ||||
|     rsm | ||||
|  | ||||
| BITS 16 | ||||
| ASM_PFX(gcSmmInitTemplate): | ||||
|     mov ebp, [cs:@L1 - ASM_PFX(gcSmmInitTemplate) + 0x8000] | ||||
|     sub ebp, 0x30000 | ||||
|     jmp ebp | ||||
| @L1: | ||||
|     DQ     0; ASM_PFX(SmmStartup) | ||||
|  | ||||
| ASM_PFX(gcSmmInitSize): DW $ - ASM_PFX(gcSmmInitTemplate) | ||||
|  | ||||
| BITS 64 | ||||
| global ASM_PFX(SmmRelocationSemaphoreComplete) | ||||
| ASM_PFX(SmmRelocationSemaphoreComplete): | ||||
|     push    rax | ||||
|     mov     rax, [ASM_PFX(mRebasedFlag)] | ||||
|     mov     byte [rax], 1 | ||||
|     pop     rax | ||||
|     jmp     [ASM_PFX(mSmmRelocationOriginalAddress)] | ||||
|  | ||||
| ; | ||||
| ; Semaphore code running in 32-bit mode | ||||
| ; | ||||
| BITS 32 | ||||
| global ASM_PFX(SmmRelocationSemaphoreComplete32) | ||||
| ASM_PFX(SmmRelocationSemaphoreComplete32): | ||||
|     push    eax | ||||
|     mov     eax, strict dword 0                ; source operand will be patched | ||||
| ASM_PFX(gPatchRebasedFlagAddr32): | ||||
|     mov     byte [eax], 1 | ||||
|     pop     eax | ||||
|     jmp     dword [dword 0]                    ; destination will be patched | ||||
| ASM_PFX(gPatchSmmRelocationOriginalAddressPtr32): | ||||
|  | ||||
| BITS 64 | ||||
| global ASM_PFX(SmmInitFixupAddress) | ||||
| ASM_PFX(SmmInitFixupAddress): | ||||
|     lea    rax, [@LongMode] | ||||
|     lea    rcx, [@PatchLongModeOffset - 6] | ||||
|     mov    dword [rcx], eax | ||||
|  | ||||
|     lea    rax, [ASM_PFX(SmmStartup)] | ||||
|     lea    rcx, [@L1] | ||||
|     mov    qword [rcx], rax | ||||
|     ret | ||||
| @@ -129,6 +129,7 @@ | ||||
| !include MdePkg/MdeLibs.dsc.inc | ||||
|  | ||||
| [LibraryClasses] | ||||
|   SmmRelocationLib|OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.inf | ||||
|   PcdLib|MdePkg/Library/BasePcdLibNull/BasePcdLibNull.inf | ||||
|   TimerLib|MdePkg/Library/SecPeiDxeTimerLibCpu/SecPeiDxeTimerLibCpu.inf | ||||
|   ResetSystemLib|OvmfPkg/Library/ResetSystemLib/BaseResetSystemLibMicrovm.inf | ||||
|   | ||||
| @@ -129,6 +129,7 @@ | ||||
| !include MdePkg/MdeLibs.dsc.inc | ||||
|  | ||||
| [LibraryClasses] | ||||
|   SmmRelocationLib|OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.inf | ||||
|   PcdLib|MdePkg/Library/BasePcdLibNull/BasePcdLibNull.inf | ||||
|   TimerLib|OvmfPkg/Library/AcpiTimerLib/BaseAcpiTimerLib.inf | ||||
|   ResetSystemLib|OvmfPkg/Library/ResetSystemLib/BaseResetSystemLib.inf | ||||
|   | ||||
| @@ -134,6 +134,7 @@ | ||||
| !include MdePkg/MdeLibs.dsc.inc | ||||
|  | ||||
| [LibraryClasses] | ||||
|   SmmRelocationLib|OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.inf | ||||
|   PcdLib|MdePkg/Library/BasePcdLibNull/BasePcdLibNull.inf | ||||
|   TimerLib|OvmfPkg/Library/AcpiTimerLib/BaseAcpiTimerLib.inf | ||||
|   ResetSystemLib|OvmfPkg/Library/ResetSystemLib/BaseResetSystemLib.inf | ||||
|   | ||||
| @@ -147,6 +147,7 @@ | ||||
| !include MdePkg/MdeLibs.dsc.inc | ||||
|  | ||||
| [LibraryClasses] | ||||
|   SmmRelocationLib|OvmfPkg/Library/SmmRelocationLib/SmmRelocationLib.inf | ||||
|   PcdLib|MdePkg/Library/BasePcdLibNull/BasePcdLibNull.inf | ||||
|   TimerLib|OvmfPkg/Library/AcpiTimerLib/BaseAcpiTimerLib.inf | ||||
|   ResetSystemLib|OvmfPkg/Library/ResetSystemLib/BaseResetSystemLib.inf | ||||
|   | ||||
		Reference in New Issue
	
	Block a user