| /** @file | |
| Config SMRAM Save State for SmmBases Relocation. | |
| Copyright (C) 2023 - 2024 Advanced Micro Devices, Inc. All rights reserved.<BR> | |
| Copyright (c) 2024, Intel Corporation. All rights reserved.<BR> | |
| SPDX-License-Identifier: BSD-2-Clause-Patent | |
| **/ | |
| #include "InternalSmmRelocationLib.h" | |
| #include <Register/Amd/SmramSaveStateMap.h> | |
| /** | |
| Get the mode of the CPU at the time an SMI occurs | |
| @retval EFI_MM_SAVE_STATE_REGISTER_LMA_32BIT 32 bit. | |
| @retval EFI_MM_SAVE_STATE_REGISTER_LMA_64BIT 64 bit. | |
| **/ | |
| UINT8 | |
| GetMmSaveStateRegisterLma ( | |
| VOID | |
| ) | |
| { | |
| UINT8 SmmSaveStateRegisterLma; | |
| MSR_IA32_EFER_REGISTER Msr; | |
| Msr.Uint64 = AsmReadMsr64 (MSR_IA32_EFER); | |
| SmmSaveStateRegisterLma = (UINT8)EFI_MM_SAVE_STATE_REGISTER_LMA_32BIT; | |
| if (Msr.Bits.LMA) { | |
| SmmSaveStateRegisterLma = (UINT8)EFI_MM_SAVE_STATE_REGISTER_LMA_64BIT; | |
| } | |
| return SmmSaveStateRegisterLma; | |
| } | |
| /** | |
| 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 *AmdCpuState; | |
| AmdCpuState = (AMD_SMRAM_SAVE_STATE_MAP *)(UINTN)(SMM_DEFAULT_SMBASE + SMRAM_SAVE_STATE_MAP_OFFSET); | |
| AmdCpuState->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 *AmdCpuState; | |
| MSR_IA32_EFER_REGISTER Msr; | |
| AmdCpuState = (AMD_SMRAM_SAVE_STATE_MAP *)CpuState; | |
| OriginalInstructionPointer = AmdCpuState->x64._RIP; | |
| Msr.Uint64 = AmdCpuState->x64.EFER; | |
| if (!Msr.Bits.LMA) { | |
| AmdCpuState->x64._RIP = NewInstructionPointer32; | |
| } else { | |
| AmdCpuState->x64._RIP = NewInstructionPointer; | |
| } | |
| // | |
| // Clear the auto HALT restart flag so the RSM instruction returns | |
| // program control to the instruction following the HLT instruction. | |
| // | |
| if ((AmdCpuState->x64.AutoHALTRestart & BIT0) != 0) { | |
| AmdCpuState->x64.AutoHALTRestart &= ~BIT0; | |
| } | |
| return OriginalInstructionPointer; | |
| } |