Cc: Jiewen Yao <jiewen.yao@intel.com> Cc: Eric Dong <eric.dong@intel.com> Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Jeff Fan <jeff.fan@intel.com> Reviewed-by: Eric Dong <eric.dong@intel.com>
		
			
				
	
	
		
			1300 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1300 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   SMM STM support functions
 | |
| 
 | |
|   Copyright (c) 2015 - 2017, Intel Corporation. All rights reserved.<BR>
 | |
|   This program and the accompanying materials
 | |
|   are licensed and made available under the terms and conditions of the BSD License
 | |
|   which accompanies this distribution.  The full text of the license may be found at
 | |
|   http://opensource.org/licenses/bsd-license.php.
 | |
| 
 | |
|   THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
 | |
|   WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
 | |
| 
 | |
| **/
 | |
| 
 | |
| #include <PiSmm.h>
 | |
| #include <Library/BaseLib.h>
 | |
| #include <Library/BaseMemoryLib.h>
 | |
| #include <Library/MemoryAllocationLib.h>
 | |
| #include <Library/HobLib.h>
 | |
| #include <Library/DebugLib.h>
 | |
| #include <Library/UefiBootServicesTableLib.h>
 | |
| #include <Library/SmmServicesTableLib.h>
 | |
| #include <Library/TpmMeasurementLib.h>
 | |
| #include <Register/Cpuid.h>
 | |
| #include <Register/ArchitecturalMsr.h>
 | |
| #include <Register/SmramSaveStateMap.h>
 | |
| 
 | |
| #include <Protocol/MpService.h>
 | |
| 
 | |
| #include "SmmStm.h"
 | |
| 
 | |
| #define TXT_EVTYPE_BASE                  0x400
 | |
| #define TXT_EVTYPE_STM_HASH              (TXT_EVTYPE_BASE + 14)
 | |
| 
 | |
| #define RDWR_ACCS             3
 | |
| #define FULL_ACCS             7
 | |
| 
 | |
| /**
 | |
|   The constructor function
 | |
| 
 | |
|   @param[in]  ImageHandle  The firmware allocated handle for the EFI image.
 | |
|   @param[in]  SystemTable  A pointer to the EFI System Table.
 | |
| 
 | |
|   @retval EFI_SUCCESS      The constructor always returns EFI_SUCCESS.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmmCpuFeaturesLibConstructor (
 | |
|   IN EFI_HANDLE        ImageHandle,
 | |
|   IN EFI_SYSTEM_TABLE  *SystemTable
 | |
|   );
 | |
| 
 | |
| EFI_HANDLE  mStmSmmCpuHandle = NULL;
 | |
| 
 | |
| BOOLEAN mLockLoadMonitor = FALSE;
 | |
| 
 | |
| //
 | |
| // Template of STM_RSC_END structure for copying.
 | |
| //
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED STM_RSC_END mRscEndNode = {
 | |
|   {END_OF_RESOURCES, sizeof (STM_RSC_END)},
 | |
| };
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINT8  *mStmResourcesPtr         = NULL;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINTN  mStmResourceTotalSize     = 0x0;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINTN  mStmResourceSizeUsed      = 0x0;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINTN  mStmResourceSizeAvailable = 0x0;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINT32  mStmState = 0;
 | |
| 
 | |
| //
 | |
| // System Configuration Table pointing to STM Configuration Table
 | |
| //
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED
 | |
| EFI_SM_MONITOR_INIT_PROTOCOL mSmMonitorInitProtocol = {
 | |
|   LoadMonitor,
 | |
|   AddPiResource,
 | |
|   DeletePiResource,
 | |
|   GetPiResource,
 | |
|   GetMonitorState,
 | |
| };
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| #define   CPUID1_EDX_XD_SUPPORT      0x100000
 | |
| 
 | |
| //
 | |
| // External global variables associated with SMI Handler Template
 | |
| //
 | |
| extern CONST TXT_PROCESSOR_SMM_DESCRIPTOR  gcStmPsd;
 | |
| extern UINT32                              gStmSmbase;
 | |
| extern volatile UINT32                     gStmSmiStack;
 | |
| extern UINT32                              gStmSmiCr3;
 | |
| extern volatile UINT8                      gcStmSmiHandlerTemplate[];
 | |
| extern CONST UINT16                        gcStmSmiHandlerSize;
 | |
| extern UINT16                              gcStmSmiHandlerOffset;
 | |
| extern BOOLEAN                             gStmXdSupported;
 | |
| 
 | |
| //
 | |
| // Variables used by SMI Handler
 | |
| //
 | |
| IA32_DESCRIPTOR  gStmSmiHandlerIdtr;
 | |
| 
 | |
| //
 | |
| // MP Services Protocol
 | |
| //
 | |
| EFI_MP_SERVICES_PROTOCOL  *mSmmCpuFeaturesLibMpService = NULL;
 | |
| 
 | |
| //
 | |
| // MSEG Base and Length in SMRAM
 | |
| //
 | |
| UINTN  mMsegBase = 0;
 | |
| UINTN  mMsegSize = 0;
 | |
| 
 | |
| BOOLEAN  mStmConfigurationTableInitialized = FALSE;
 | |
| 
 | |
| 
 | |
| /**
 | |
|   The constructor function
 | |
| 
 | |
|   @param[in]  ImageHandle  The firmware allocated handle for the EFI image.
 | |
|   @param[in]  SystemTable  A pointer to the EFI System Table.
 | |
| 
 | |
|   @retval EFI_SUCCESS      The constructor always returns EFI_SUCCESS.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmmCpuFeaturesLibStmConstructor (
 | |
|   IN EFI_HANDLE        ImageHandle,
 | |
|   IN EFI_SYSTEM_TABLE  *SystemTable
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
|   CPUID_VERSION_INFO_ECX  RegEcx;
 | |
|   EFI_HOB_GUID_TYPE       *GuidHob;
 | |
|   EFI_SMRAM_DESCRIPTOR    *SmramDescriptor;
 | |
| 
 | |
|   //
 | |
|   // Call the common constructor function
 | |
|   //
 | |
|   Status = SmmCpuFeaturesLibConstructor (ImageHandle, SystemTable);
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
| 
 | |
|   //
 | |
|   // Lookup the MP Services Protocol
 | |
|   //
 | |
|   Status = gBS->LocateProtocol (
 | |
|                   &gEfiMpServiceProtocolGuid,
 | |
|                   NULL,
 | |
|                   (VOID **)&mSmmCpuFeaturesLibMpService
 | |
|                   );
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
| 
 | |
|   //
 | |
|   // If CPU supports VMX, then determine SMRAM range for MSEG.
 | |
|   //
 | |
|   AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, &RegEcx.Uint32, NULL);
 | |
|   if (RegEcx.Bits.VMX == 1) {
 | |
|     GuidHob = GetFirstGuidHob (&gMsegSmramGuid);
 | |
|     if (GuidHob != NULL) {
 | |
|       //
 | |
|       // Retrieve MSEG location from MSEG SRAM HOB
 | |
|       //
 | |
|       SmramDescriptor = (EFI_SMRAM_DESCRIPTOR *) GET_GUID_HOB_DATA (GuidHob);
 | |
|       if (SmramDescriptor->PhysicalSize > 0) {
 | |
|         mMsegBase       = (UINTN)SmramDescriptor->CpuStart;
 | |
|         mMsegSize       = (UINTN)SmramDescriptor->PhysicalSize;
 | |
|       }
 | |
|     } else if (PcdGet32 (PcdCpuMsegSize) > 0) {
 | |
|       //
 | |
|       // Allocate MSEG from SMRAM memory
 | |
|       //
 | |
|       mMsegBase = (UINTN)AllocatePages (EFI_SIZE_TO_PAGES (PcdGet32 (PcdCpuMsegSize)));
 | |
|       if (mMsegBase > 0) {
 | |
|         mMsegSize = ALIGN_VALUE (PcdGet32 (PcdCpuMsegSize), EFI_PAGE_SIZE);
 | |
|       } else {
 | |
|         DEBUG ((DEBUG_ERROR, "Not enough SMRAM resource to allocate MSEG size %08x\n", PcdGet32 (PcdCpuMsegSize)));
 | |
|       }
 | |
|     }
 | |
|     if (mMsegBase > 0) {
 | |
|       DEBUG ((DEBUG_INFO, "MsegBase: 0x%08x, MsegSize: 0x%08x\n", mMsegBase, mMsegSize));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Internal worker function that is called to complete CPU initialization at the
 | |
|   end of SmmCpuFeaturesInitializeProcessor().
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| FinishSmmCpuFeaturesInitializeProcessor (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   MSR_IA32_SMM_MONITOR_CTL_REGISTER  SmmMonitorCtl;
 | |
| 
 | |
|   //
 | |
|   // Set MSEG Base Address in SMM Monitor Control MSR.
 | |
|   //
 | |
|   if (mMsegBase > 0) {
 | |
|     SmmMonitorCtl.Uint64        = 0;
 | |
|     SmmMonitorCtl.Bits.MsegBase = (UINT32)mMsegBase >> 12;
 | |
|     SmmMonitorCtl.Bits.Valid    = 1;
 | |
|     AsmWriteMsr64 (MSR_IA32_SMM_MONITOR_CTL, SmmMonitorCtl.Uint64);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return the size, in bytes, of a custom SMI Handler in bytes.  If 0 is
 | |
|   returned, then a custom SMI handler is not provided by this library,
 | |
|   and the default SMI handler must be used.
 | |
| 
 | |
|   @retval 0    Use the default SMI handler.
 | |
|   @retval > 0  Use the SMI handler installed by SmmCpuFeaturesInstallSmiHandler()
 | |
|                The caller is required to allocate enough SMRAM for each CPU to
 | |
|                support the size of the custom SMI handler.
 | |
| **/
 | |
| UINTN
 | |
| EFIAPI
 | |
| SmmCpuFeaturesGetSmiHandlerSize (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   return gcStmSmiHandlerSize;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Install a custom SMI handler for the CPU specified by CpuIndex.  This function
 | |
|   is only called if SmmCpuFeaturesGetSmiHandlerSize() returns a size is greater
 | |
|   than zero and is called by the CPU that was elected as monarch during System
 | |
|   Management Mode initialization.
 | |
| 
 | |
|   @param[in] CpuIndex   The index of the CPU to install the custom SMI handler.
 | |
|                         The value must be between 0 and the NumberOfCpus field
 | |
|                         in the System Management System Table (SMST).
 | |
|   @param[in] SmBase     The SMBASE address for the CPU specified by CpuIndex.
 | |
|   @param[in] SmiStack   The stack to use when an SMI is processed by the
 | |
|                         the CPU specified by CpuIndex.
 | |
|   @param[in] StackSize  The size, in bytes, if the stack used when an SMI is
 | |
|                         processed by the CPU specified by CpuIndex.
 | |
|   @param[in] GdtBase    The base address of the GDT to use when an SMI is
 | |
|                         processed by the CPU specified by CpuIndex.
 | |
|   @param[in] GdtSize    The size, in bytes, of the GDT used when an SMI is
 | |
|                         processed by the CPU specified by CpuIndex.
 | |
|   @param[in] IdtBase    The base address of the IDT to use when an SMI is
 | |
|                         processed by the CPU specified by CpuIndex.
 | |
|   @param[in] IdtSize    The size, in bytes, of the IDT used when an SMI is
 | |
|                         processed by the CPU specified by CpuIndex.
 | |
|   @param[in] Cr3        The base address of the page tables to use when an SMI
 | |
|                         is processed by the CPU specified by CpuIndex.
 | |
| **/
 | |
| VOID
 | |
| EFIAPI
 | |
| SmmCpuFeaturesInstallSmiHandler (
 | |
|   IN UINTN   CpuIndex,
 | |
|   IN UINT32  SmBase,
 | |
|   IN VOID    *SmiStack,
 | |
|   IN UINTN   StackSize,
 | |
|   IN UINTN   GdtBase,
 | |
|   IN UINTN   GdtSize,
 | |
|   IN UINTN   IdtBase,
 | |
|   IN UINTN   IdtSize,
 | |
|   IN UINT32  Cr3
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                     Status;
 | |
|   TXT_PROCESSOR_SMM_DESCRIPTOR   *Psd;
 | |
|   VOID                           *Hob;
 | |
|   UINT32                         RegEax;
 | |
|   UINT32                         RegEdx;
 | |
|   EFI_PROCESSOR_INFORMATION      ProcessorInfo;
 | |
| 
 | |
|   CopyMem ((VOID *)((UINTN)SmBase + TXT_SMM_PSD_OFFSET), &gcStmPsd, sizeof (gcStmPsd));
 | |
|   Psd = (TXT_PROCESSOR_SMM_DESCRIPTOR *)(VOID *)((UINTN)SmBase + TXT_SMM_PSD_OFFSET);
 | |
|   Psd->SmmGdtPtr = GdtBase;
 | |
|   Psd->SmmGdtSize = (UINT32)GdtSize;
 | |
| 
 | |
|   //
 | |
|   // Initialize values in template before copy
 | |
|   //
 | |
|   gStmSmiStack             = (UINT32)((UINTN)SmiStack + StackSize - sizeof (UINTN));
 | |
|   gStmSmiCr3               = Cr3;
 | |
|   gStmSmbase               = SmBase;
 | |
|   gStmSmiHandlerIdtr.Base  = IdtBase;
 | |
|   gStmSmiHandlerIdtr.Limit = (UINT16)(IdtSize - 1);
 | |
| 
 | |
|   if (gStmXdSupported) {
 | |
|     AsmCpuid (CPUID_EXTENDED_FUNCTION, &RegEax, NULL, NULL, NULL);
 | |
|     if (RegEax <= CPUID_EXTENDED_FUNCTION) {
 | |
|       //
 | |
|       // Extended CPUID functions are not supported on this processor.
 | |
|       //
 | |
|       gStmXdSupported = FALSE;
 | |
|     }
 | |
| 
 | |
|     AsmCpuid (CPUID_EXTENDED_CPU_SIG, NULL, NULL, NULL, &RegEdx);
 | |
|     if ((RegEdx & CPUID1_EDX_XD_SUPPORT) == 0) {
 | |
|       //
 | |
|       // Execute Disable Bit feature is not supported on this processor.
 | |
|       //
 | |
|       gStmXdSupported = FALSE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Set the value at the top of the CPU stack to the CPU Index
 | |
|   //
 | |
|   *(UINTN*)(UINTN)gStmSmiStack = CpuIndex;
 | |
| 
 | |
|   //
 | |
|   // Copy template to CPU specific SMI handler location
 | |
|   //
 | |
|   CopyMem (
 | |
|     (VOID*)((UINTN)SmBase + SMM_HANDLER_OFFSET),
 | |
|     (VOID*)gcStmSmiHandlerTemplate,
 | |
|     gcStmSmiHandlerSize
 | |
|     );
 | |
| 
 | |
|   Psd->SmmSmiHandlerRip = SmBase + SMM_HANDLER_OFFSET + gcStmSmiHandlerOffset;
 | |
|   Psd->SmmSmiHandlerRsp = (UINTN)SmiStack + StackSize - sizeof(UINTN);
 | |
|   Psd->SmmCr3           = Cr3;
 | |
| 
 | |
|   DEBUG((DEBUG_INFO, "CpuSmmStmExceptionStackSize - %x\n", PcdGet32(PcdCpuSmmStmExceptionStackSize)));
 | |
|   DEBUG((DEBUG_INFO, "Pages - %x\n", EFI_SIZE_TO_PAGES(PcdGet32(PcdCpuSmmStmExceptionStackSize))));
 | |
|   Psd->StmProtectionExceptionHandler.SpeRsp = (UINT64)(UINTN)AllocatePages (EFI_SIZE_TO_PAGES (PcdGet32 (PcdCpuSmmStmExceptionStackSize)));
 | |
|   Psd->StmProtectionExceptionHandler.SpeRsp += EFI_PAGES_TO_SIZE (EFI_SIZE_TO_PAGES (PcdGet32 (PcdCpuSmmStmExceptionStackSize)));
 | |
| 
 | |
|   Psd->BiosHwResourceRequirementsPtr        = (UINT64)(UINTN)GetStmResource ();
 | |
| 
 | |
|   //
 | |
|   // Get the APIC ID for the CPU specified by CpuIndex
 | |
|   //
 | |
|   Status = mSmmCpuFeaturesLibMpService->GetProcessorInfo (
 | |
|              mSmmCpuFeaturesLibMpService,
 | |
|              CpuIndex,
 | |
|              &ProcessorInfo
 | |
|              );
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
| 
 | |
|   Psd->LocalApicId = (UINT32)ProcessorInfo.ProcessorId;
 | |
|   Psd->AcpiRsdp = 0;
 | |
| 
 | |
|   Hob = GetFirstHob (EFI_HOB_TYPE_CPU);
 | |
|   if (Hob != NULL) {
 | |
|     Psd->PhysicalAddressBits = ((EFI_HOB_CPU *) Hob)->SizeOfMemorySpace;
 | |
|   } else {
 | |
|     AsmCpuid (0x80000000, &RegEax, NULL, NULL, NULL);
 | |
|     if (RegEax >= 0x80000008) {
 | |
|       AsmCpuid (0x80000008, &RegEax, NULL, NULL, NULL);
 | |
|       Psd->PhysicalAddressBits = (UINT8) RegEax;
 | |
|     } else {
 | |
|       Psd->PhysicalAddressBits = 36;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!mStmConfigurationTableInitialized) {
 | |
|     StmSmmConfigurationTableInit ();
 | |
|     mStmConfigurationTableInitialized = TRUE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMM End Of Dxe event notification handler.
 | |
| 
 | |
|   STM support need patch AcpiRsdp in TXT_PROCESSOR_SMM_DESCRIPTOR.
 | |
| 
 | |
|   @param[in] Protocol   Points to the protocol's unique identifier.
 | |
|   @param[in] Interface  Points to the interface instance.
 | |
|   @param[in] Handle     The handle on which the interface was installed.
 | |
| 
 | |
|   @retval EFI_SUCCESS   Notification handler runs successfully.
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmmEndOfDxeEventNotify (
 | |
|   IN CONST EFI_GUID  *Protocol,
 | |
|   IN VOID            *Interface,
 | |
|   IN EFI_HANDLE      Handle
 | |
|   )
 | |
| {
 | |
|   VOID                          *Rsdp;
 | |
|   UINTN                         Index;
 | |
|   TXT_PROCESSOR_SMM_DESCRIPTOR  *Psd;
 | |
| 
 | |
|   DEBUG ((DEBUG_INFO, "SmmEndOfDxeEventNotify\n"));
 | |
| 
 | |
|   //
 | |
|   // found ACPI table RSD_PTR from system table
 | |
|   //
 | |
|   Rsdp = NULL;
 | |
|   for (Index = 0; Index < gST->NumberOfTableEntries; Index++) {
 | |
|     if (CompareGuid (&(gST->ConfigurationTable[Index].VendorGuid), &gEfiAcpi20TableGuid)) {
 | |
|       //
 | |
|       // A match was found.
 | |
|       //
 | |
|       Rsdp = gST->ConfigurationTable[Index].VendorTable;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   if (Rsdp == NULL) {
 | |
|     for (Index = 0; Index < gST->NumberOfTableEntries; Index++) {
 | |
|       if (CompareGuid (&(gST->ConfigurationTable[Index].VendorGuid), &gEfiAcpi10TableGuid)) {
 | |
|         //
 | |
|         // A match was found.
 | |
|         //
 | |
|         Rsdp = gST->ConfigurationTable[Index].VendorTable;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (Index = 0; Index < gSmst->NumberOfCpus; Index++) {
 | |
|     Psd = (TXT_PROCESSOR_SMM_DESCRIPTOR *)((UINTN)gSmst->CpuSaveState[Index] - SMRAM_SAVE_STATE_MAP_OFFSET + TXT_SMM_PSD_OFFSET);
 | |
|     DEBUG ((DEBUG_INFO, "Index=%d  Psd=%p  Rsdp=%p\n", Index, Psd, Rsdp));
 | |
|     Psd->AcpiRsdp = (UINT64)(UINTN)Rsdp;
 | |
|   }
 | |
| 
 | |
|   mLockLoadMonitor = TRUE;
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   This function initializes the STM configuration table.
 | |
| **/
 | |
| VOID
 | |
| StmSmmConfigurationTableInit (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS    Status;
 | |
|     VOID        *Registration;
 | |
| 
 | |
|   Status = gSmst->SmmInstallProtocolInterface (
 | |
|                     &mStmSmmCpuHandle,
 | |
|                     &gEfiSmMonitorInitProtocolGuid,
 | |
|                     EFI_NATIVE_INTERFACE,
 | |
|                     &mSmMonitorInitProtocol
 | |
|                     );
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
| 
 | |
|   //
 | |
|   //
 | |
|   // Register SMM End of DXE Event
 | |
|   //
 | |
|   Status = gSmst->SmmRegisterProtocolNotify (
 | |
|                     &gEfiSmmEndOfDxeProtocolGuid,
 | |
|                     SmmEndOfDxeEventNotify,
 | |
|                     &Registration
 | |
|                     );
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Get STM state.
 | |
| 
 | |
|   @return STM state
 | |
| 
 | |
| **/
 | |
| EFI_SM_MONITOR_STATE
 | |
| EFIAPI
 | |
| GetMonitorState (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   return mStmState;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Handle single Resource to see if it can be merged into Record.
 | |
| 
 | |
|   @param Resource  A pointer to resource node to be added
 | |
|   @param Record    A pointer to record node to be merged
 | |
| 
 | |
|   @retval TRUE  resource handled
 | |
|   @retval FALSE resource is not handled
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| HandleSingleResource (
 | |
|   IN  STM_RSC      *Resource,
 | |
|   IN  STM_RSC      *Record
 | |
|   )
 | |
| {
 | |
|   UINT64      ResourceLo;
 | |
|   UINT64      ResourceHi;
 | |
|   UINT64      RecordLo;
 | |
|   UINT64      RecordHi;
 | |
| 
 | |
|   ResourceLo = 0;
 | |
|   ResourceHi = 0;
 | |
|   RecordLo = 0;
 | |
|   RecordHi = 0;
 | |
| 
 | |
|   //
 | |
|   // Calling code is responsible for making sure that
 | |
|   // Resource->Header.RscType == (*Record)->Header.RscType
 | |
|   // thus we use just one of them as switch variable.
 | |
|   //
 | |
|   switch (Resource->Header.RscType) {
 | |
|   case MEM_RANGE:
 | |
|   case MMIO_RANGE:
 | |
|     ResourceLo = Resource->Mem.Base;
 | |
|     ResourceHi = Resource->Mem.Base + Resource->Mem.Length;
 | |
|     RecordLo = Record->Mem.Base;
 | |
|     RecordHi = Record->Mem.Base + Record->Mem.Length;
 | |
|     if (Resource->Mem.RWXAttributes != Record->Mem.RWXAttributes) {
 | |
|       if ((ResourceLo == RecordLo) && (ResourceHi == RecordHi)) {
 | |
|         Record->Mem.RWXAttributes = Resource->Mem.RWXAttributes | Record->Mem.RWXAttributes;
 | |
|         return TRUE;
 | |
|       } else {
 | |
|         return FALSE;
 | |
|       }
 | |
|     }
 | |
|     break;
 | |
|   case IO_RANGE:
 | |
|   case TRAPPED_IO_RANGE:
 | |
|     ResourceLo = (UINT64) Resource->Io.Base;
 | |
|     ResourceHi = (UINT64) Resource->Io.Base + (UINT64) Resource->Io.Length;
 | |
|     RecordLo = (UINT64) Record->Io.Base;
 | |
|     RecordHi = (UINT64) Record->Io.Base + (UINT64) Record->Io.Length;
 | |
|     break;
 | |
|   case PCI_CFG_RANGE:
 | |
|     if ((Resource->PciCfg.OriginatingBusNumber != Record->PciCfg.OriginatingBusNumber) ||
 | |
|         (Resource->PciCfg.LastNodeIndex != Record->PciCfg.LastNodeIndex)) {
 | |
|       return FALSE;
 | |
|     }
 | |
|     if (CompareMem (Resource->PciCfg.PciDevicePath, Record->PciCfg.PciDevicePath, sizeof(STM_PCI_DEVICE_PATH_NODE) * (Resource->PciCfg.LastNodeIndex + 1)) != 0) {
 | |
|       return FALSE;
 | |
|     }
 | |
|     ResourceLo = (UINT64) Resource->PciCfg.Base;
 | |
|     ResourceHi = (UINT64) Resource->PciCfg.Base + (UINT64) Resource->PciCfg.Length;
 | |
|     RecordLo = (UINT64) Record->PciCfg.Base;
 | |
|     RecordHi = (UINT64) Record->PciCfg.Base + (UINT64) Record->PciCfg.Length;
 | |
|     if (Resource->PciCfg.RWAttributes != Record->PciCfg.RWAttributes) {
 | |
|       if ((ResourceLo == RecordLo) && (ResourceHi == RecordHi)) {
 | |
|         Record->PciCfg.RWAttributes = Resource->PciCfg.RWAttributes | Record->PciCfg.RWAttributes;
 | |
|         return TRUE;
 | |
|       } else {
 | |
|         return FALSE;
 | |
|       }
 | |
|     }
 | |
|     break;
 | |
|   case MACHINE_SPECIFIC_REG:
 | |
|     //
 | |
|     // Special case - merge MSR masks in place.
 | |
|     //
 | |
|     if (Resource->Msr.MsrIndex != Record->Msr.MsrIndex) {
 | |
|       return FALSE;
 | |
|     }
 | |
|     Record->Msr.ReadMask |= Resource->Msr.ReadMask;
 | |
|     Record->Msr.WriteMask |= Resource->Msr.WriteMask;
 | |
|     return TRUE;
 | |
|   default:
 | |
|     return FALSE;
 | |
|   }
 | |
|   //
 | |
|   // If resources are disjoint
 | |
|   //
 | |
|   if ((ResourceHi < RecordLo) || (ResourceLo > RecordHi)) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // If resource is consumed by record.
 | |
|   //
 | |
|   if ((ResourceLo >= RecordLo) && (ResourceHi <= RecordHi)) {
 | |
|     return TRUE;
 | |
|   }
 | |
|   //
 | |
|   // Resources are overlapping.
 | |
|   // Resource and record are merged.
 | |
|   //
 | |
|   ResourceLo = (ResourceLo < RecordLo) ? ResourceLo : RecordLo;
 | |
|   ResourceHi = (ResourceHi > RecordHi) ? ResourceHi : RecordHi;
 | |
| 
 | |
|   switch (Resource->Header.RscType) {
 | |
|   case MEM_RANGE:
 | |
|   case MMIO_RANGE:
 | |
|     Record->Mem.Base = ResourceLo;
 | |
|     Record->Mem.Length = ResourceHi - ResourceLo;
 | |
|     break;
 | |
|   case IO_RANGE:
 | |
|   case TRAPPED_IO_RANGE:
 | |
|     Record->Io.Base = (UINT16) ResourceLo;
 | |
|     Record->Io.Length = (UINT16) (ResourceHi - ResourceLo);
 | |
|     break;
 | |
|   case PCI_CFG_RANGE:
 | |
|     Record->PciCfg.Base = (UINT16) ResourceLo;
 | |
|     Record->PciCfg.Length = (UINT16) (ResourceHi - ResourceLo);
 | |
|     break;
 | |
|   default:
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Add resource node.
 | |
| 
 | |
|   @param Resource  A pointer to resource node to be added
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| AddSingleResource (
 | |
|   IN  STM_RSC    *Resource
 | |
|   )
 | |
| {
 | |
|   STM_RSC    *Record;
 | |
| 
 | |
|   Record = (STM_RSC *)mStmResourcesPtr;
 | |
| 
 | |
|   while (TRUE) {
 | |
|     if (Record->Header.RscType == END_OF_RESOURCES) {
 | |
|       break;
 | |
|     }
 | |
|     //
 | |
|     // Go to next record if resource and record types don't match.
 | |
|     //
 | |
|     if (Resource->Header.RscType != Record->Header.RscType) {
 | |
|       Record = (STM_RSC *)((UINTN)Record + Record->Header.Length);
 | |
|       continue;
 | |
|     }
 | |
|     //
 | |
|     // Record is handled inside of procedure - don't adjust.
 | |
|     //
 | |
|     if (HandleSingleResource (Resource, Record)) {
 | |
|       return ;
 | |
|     }
 | |
|     Record = (STM_RSC *)((UINTN)Record + Record->Header.Length);
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Add resource to the end of area.
 | |
|   //
 | |
|   CopyMem (
 | |
|     mStmResourcesPtr + mStmResourceSizeUsed - sizeof(mRscEndNode),
 | |
|     Resource,
 | |
|     Resource->Header.Length
 | |
|     );
 | |
|   CopyMem (
 | |
|     mStmResourcesPtr + mStmResourceSizeUsed - sizeof(mRscEndNode) + Resource->Header.Length,
 | |
|     &mRscEndNode,
 | |
|     sizeof(mRscEndNode)
 | |
|     );
 | |
|   mStmResourceSizeUsed += Resource->Header.Length;
 | |
|   mStmResourceSizeAvailable = mStmResourceTotalSize - mStmResourceSizeUsed;
 | |
| 
 | |
|   return ;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Add resource list.
 | |
| 
 | |
|   @param ResourceList  A pointer to resource list to be added
 | |
|   @param NumEntries    Optional number of entries.
 | |
|                        If 0, list must be terminated by END_OF_RESOURCES.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| AddResource (
 | |
|   IN  STM_RSC    *ResourceList,
 | |
|   IN  UINT32      NumEntries OPTIONAL
 | |
|   )
 | |
| {
 | |
|   UINT32      Count;
 | |
|   UINTN       Index;
 | |
|   STM_RSC    *Resource;
 | |
| 
 | |
|   if (NumEntries == 0) {
 | |
|     Count = 0xFFFFFFFF;
 | |
|   } else {
 | |
|     Count = NumEntries;
 | |
|   }
 | |
| 
 | |
|   Resource = ResourceList;
 | |
| 
 | |
|   for (Index = 0; Index < Count; Index++) {
 | |
|     if (Resource->Header.RscType == END_OF_RESOURCES) {
 | |
|       return ;
 | |
|     }
 | |
|     AddSingleResource (Resource);
 | |
|     Resource = (STM_RSC *)((UINTN)Resource + Resource->Header.Length);
 | |
|   }
 | |
|   return ;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Validate resource list.
 | |
| 
 | |
|   @param ResourceList  A pointer to resource list to be added
 | |
|   @param NumEntries    Optional number of entries.
 | |
|                        If 0, list must be terminated by END_OF_RESOURCES.
 | |
| 
 | |
|   @retval TRUE  resource valid
 | |
|   @retval FALSE resource invalid
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| ValidateResource (
 | |
|   IN  STM_RSC    *ResourceList,
 | |
|   IN  UINT32      NumEntries OPTIONAL
 | |
|   )
 | |
| {
 | |
|   UINT32      Count;
 | |
|   UINTN       Index;
 | |
|   STM_RSC    *Resource;
 | |
|   UINTN       SubIndex;
 | |
| 
 | |
|   //
 | |
|   // If NumEntries == 0 make it very big. Scan will be terminated by
 | |
|   // END_OF_RESOURCES.
 | |
|   //
 | |
|   if (NumEntries == 0) {
 | |
|     Count = 0xFFFFFFFF;
 | |
|   } else {
 | |
|     Count = NumEntries;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Start from beginning of resource list.
 | |
|   //
 | |
|   Resource = ResourceList;
 | |
| 
 | |
|   for (Index = 0; Index < Count; Index++) {
 | |
|     DEBUG ((DEBUG_INFO, "ValidateResource (%d) - RscType(%x)\n", Index, Resource->Header.RscType));
 | |
|     //
 | |
|     // Validate resource.
 | |
|     //
 | |
|     switch (Resource->Header.RscType) {
 | |
|       case END_OF_RESOURCES:
 | |
|         if (Resource->Header.Length != sizeof (STM_RSC_END)) {
 | |
|           return  FALSE;
 | |
|         }
 | |
|         //
 | |
|         // If we are passed actual number of resources to add,
 | |
|         // END_OF_RESOURCES structure between them is considered an
 | |
|         // error. If NumEntries == 0 END_OF_RESOURCES is a termination.
 | |
|         //
 | |
|         if (NumEntries != 0) {
 | |
|           return  FALSE;
 | |
|         } else {
 | |
|           //
 | |
|           // If NumEntries == 0 and list reached end - return success.
 | |
|           //
 | |
|           return TRUE;
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       case MEM_RANGE:
 | |
|       case MMIO_RANGE:
 | |
|         if (Resource->Header.Length != sizeof (STM_RSC_MEM_DESC)) {
 | |
|           return FALSE;
 | |
|         }
 | |
| 
 | |
|         if (Resource->Mem.RWXAttributes > FULL_ACCS) {
 | |
|           return FALSE;
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       case IO_RANGE:
 | |
|       case TRAPPED_IO_RANGE:
 | |
|         if (Resource->Header.Length != sizeof (STM_RSC_IO_DESC)) {
 | |
|           return FALSE;
 | |
|         }
 | |
| 
 | |
|         if ((Resource->Io.Base + Resource->Io.Length) > 0xFFFF) {
 | |
|           return FALSE;
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       case PCI_CFG_RANGE:
 | |
|         DEBUG ((DEBUG_INFO, "ValidateResource - PCI (0x%02x, 0x%08x, 0x%02x, 0x%02x)\n", Resource->PciCfg.OriginatingBusNumber, Resource->PciCfg.LastNodeIndex, Resource->PciCfg.PciDevicePath[0].PciDevice, Resource->PciCfg.PciDevicePath[0].PciFunction));
 | |
|         if (Resource->Header.Length != sizeof (STM_RSC_PCI_CFG_DESC) + (sizeof(STM_PCI_DEVICE_PATH_NODE) * Resource->PciCfg.LastNodeIndex)) {
 | |
|           return FALSE;
 | |
|         }
 | |
|         for (SubIndex = 0; SubIndex <= Resource->PciCfg.LastNodeIndex; SubIndex++) {
 | |
|           if ((Resource->PciCfg.PciDevicePath[SubIndex].PciDevice > 0x1F) || (Resource->PciCfg.PciDevicePath[SubIndex].PciFunction > 7)) {
 | |
|             return FALSE;
 | |
|           }
 | |
|         }
 | |
|         if ((Resource->PciCfg.Base + Resource->PciCfg.Length) > 0x1000) {
 | |
|           return FALSE;
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       case MACHINE_SPECIFIC_REG:
 | |
|         if (Resource->Header.Length != sizeof (STM_RSC_MSR_DESC)) {
 | |
|           return FALSE;
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       default :
 | |
|         DEBUG ((DEBUG_ERROR, "ValidateResource - Unknown RscType(%x)\n", Resource->Header.RscType));
 | |
|         return FALSE;
 | |
|     }
 | |
|     Resource = (STM_RSC *)((UINTN)Resource + Resource->Header.Length);
 | |
|   }
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Get resource list.
 | |
|   EndResource is excluded.
 | |
| 
 | |
|   @param ResourceList  A pointer to resource list to be added
 | |
|   @param NumEntries    Optional number of entries.
 | |
|                        If 0, list must be terminated by END_OF_RESOURCES.
 | |
| 
 | |
|   @retval TRUE  resource valid
 | |
|   @retval FALSE resource invalid
 | |
| 
 | |
| **/
 | |
| UINTN
 | |
| GetResourceSize (
 | |
|   IN  STM_RSC    *ResourceList,
 | |
|   IN  UINT32      NumEntries OPTIONAL
 | |
|   )
 | |
| {
 | |
|   UINT32      Count;
 | |
|   UINTN       Index;
 | |
|   STM_RSC    *Resource;
 | |
| 
 | |
|   Resource = ResourceList;
 | |
| 
 | |
|   //
 | |
|   // If NumEntries == 0 make it very big. Scan will be terminated by
 | |
|   // END_OF_RESOURCES.
 | |
|   //
 | |
|   if (NumEntries == 0) {
 | |
|     Count = 0xFFFFFFFF;
 | |
|   } else {
 | |
|     Count = NumEntries;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Start from beginning of resource list.
 | |
|   //
 | |
|   Resource = ResourceList;
 | |
| 
 | |
|   for (Index = 0; Index < Count; Index++) {
 | |
|     if (Resource->Header.RscType == END_OF_RESOURCES) {
 | |
|       break;
 | |
|     }
 | |
|     Resource = (STM_RSC *)((UINTN)Resource + Resource->Header.Length);
 | |
|   }
 | |
| 
 | |
|   return (UINTN)Resource - (UINTN)ResourceList;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Add resources in list to database. Allocate new memory areas as needed.
 | |
| 
 | |
|   @param ResourceList  A pointer to resource list to be added
 | |
|   @param NumEntries    Optional number of entries.
 | |
|                        If 0, list must be terminated by END_OF_RESOURCES.
 | |
| 
 | |
|   @retval EFI_SUCCESS            If resources are added
 | |
|   @retval EFI_INVALID_PARAMETER  If nested procedure detected resource failer
 | |
|   @retval EFI_OUT_OF_RESOURCES   If nested procedure returned it and we cannot allocate more areas.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| AddPiResource (
 | |
|   IN  STM_RSC    *ResourceList,
 | |
|   IN  UINT32      NumEntries OPTIONAL
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS            Status;
 | |
|   UINTN                 ResourceSize;
 | |
|   EFI_PHYSICAL_ADDRESS  NewResource;
 | |
|   UINTN                 NewResourceSize;
 | |
| 
 | |
|   DEBUG ((DEBUG_INFO, "AddPiResource - Enter\n"));
 | |
| 
 | |
|   if (!ValidateResource (ResourceList, NumEntries)) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   ResourceSize = GetResourceSize (ResourceList, NumEntries);
 | |
|   DEBUG ((DEBUG_INFO, "ResourceSize - 0x%08x\n", ResourceSize));
 | |
|   if (ResourceSize == 0) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   if (mStmResourcesPtr == NULL) {
 | |
|     //
 | |
|     // First time allocation
 | |
|     //
 | |
|     NewResourceSize = EFI_PAGES_TO_SIZE (EFI_SIZE_TO_PAGES (ResourceSize + sizeof(mRscEndNode)));
 | |
|     DEBUG ((DEBUG_INFO, "Allocate - 0x%08x\n", NewResourceSize));
 | |
|     Status = gSmst->SmmAllocatePages (
 | |
|                       AllocateAnyPages,
 | |
|                       EfiRuntimeServicesData,
 | |
|                       EFI_SIZE_TO_PAGES (NewResourceSize),
 | |
|                       &NewResource
 | |
|                       );
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     // Copy EndResource for intialization
 | |
|     //
 | |
|     mStmResourcesPtr = (UINT8 *)(UINTN)NewResource;
 | |
|     mStmResourceTotalSize = NewResourceSize;
 | |
|     CopyMem (mStmResourcesPtr, &mRscEndNode, sizeof(mRscEndNode));
 | |
|     mStmResourceSizeUsed      = sizeof(mRscEndNode);
 | |
|     mStmResourceSizeAvailable = mStmResourceTotalSize - sizeof(mRscEndNode);
 | |
| 
 | |
|     //
 | |
|     // Let SmmCore change resource ptr
 | |
|     //
 | |
|     NotifyStmResourceChange (mStmResourcesPtr);
 | |
|   } else if (mStmResourceSizeAvailable < ResourceSize) {
 | |
|     //
 | |
|     // Need enlarge
 | |
|     //
 | |
|     NewResourceSize = mStmResourceTotalSize + (ResourceSize - mStmResourceSizeAvailable);
 | |
|     NewResourceSize = EFI_PAGES_TO_SIZE (EFI_SIZE_TO_PAGES (NewResourceSize));
 | |
|     DEBUG ((DEBUG_INFO, "ReAllocate - 0x%08x\n", NewResourceSize));
 | |
|     Status = gSmst->SmmAllocatePages (
 | |
|                       AllocateAnyPages,
 | |
|                       EfiRuntimeServicesData,
 | |
|                       EFI_SIZE_TO_PAGES (NewResourceSize),
 | |
|                       &NewResource
 | |
|                       );
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
|     CopyMem ((VOID *)(UINTN)NewResource, mStmResourcesPtr, mStmResourceSizeUsed);
 | |
|     mStmResourceSizeAvailable = NewResourceSize - mStmResourceSizeUsed;
 | |
| 
 | |
|     gSmst->SmmFreePages (
 | |
|              (EFI_PHYSICAL_ADDRESS)(UINTN)mStmResourcesPtr,
 | |
|              EFI_SIZE_TO_PAGES (mStmResourceTotalSize)
 | |
|              );
 | |
| 
 | |
|     mStmResourceTotalSize = NewResourceSize;
 | |
|     mStmResourcesPtr = (UINT8 *)(UINTN)NewResource;
 | |
| 
 | |
|     //
 | |
|     // Let SmmCore change resource ptr
 | |
|     //
 | |
|     NotifyStmResourceChange (mStmResourcesPtr);
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Check duplication
 | |
|   //
 | |
|   AddResource (ResourceList, NumEntries);
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Delete resources in list to database.
 | |
| 
 | |
|   @param ResourceList  A pointer to resource list to be deleted
 | |
|                        NULL means delete all resources.
 | |
|   @param NumEntries    Optional number of entries.
 | |
|                        If 0, list must be terminated by END_OF_RESOURCES.
 | |
| 
 | |
|   @retval EFI_SUCCESS            If resources are deleted
 | |
|   @retval EFI_INVALID_PARAMETER  If nested procedure detected resource failer
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| DeletePiResource (
 | |
|   IN  STM_RSC    *ResourceList,
 | |
|   IN  UINT32      NumEntries OPTIONAL
 | |
|   )
 | |
| {
 | |
|   if (ResourceList != NULL) {
 | |
|     // TBD
 | |
|     ASSERT (FALSE);
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
|   //
 | |
|   // Delete all
 | |
|   //
 | |
|   CopyMem (mStmResourcesPtr, &mRscEndNode, sizeof(mRscEndNode));
 | |
|   mStmResourceSizeUsed      = sizeof(mRscEndNode);
 | |
|   mStmResourceSizeAvailable = mStmResourceTotalSize - sizeof(mRscEndNode);
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Get BIOS resources.
 | |
| 
 | |
|   @param ResourceList  A pointer to resource list to be filled
 | |
|   @param ResourceSize  On input it means size of resource list input.
 | |
|                        On output it means size of resource list filled,
 | |
|                        or the size of resource list to be filled if size of too small.
 | |
| 
 | |
|   @retval EFI_SUCCESS            If resources are returned.
 | |
|   @retval EFI_BUFFER_TOO_SMALL   If resource list buffer is too small to hold the whole resources.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| GetPiResource (
 | |
|   OUT    STM_RSC *ResourceList,
 | |
|   IN OUT UINT32  *ResourceSize
 | |
|   )
 | |
| {
 | |
|   if (*ResourceSize < mStmResourceSizeUsed) {
 | |
|     *ResourceSize = (UINT32)mStmResourceSizeUsed;
 | |
|     return EFI_BUFFER_TOO_SMALL;
 | |
|   }
 | |
| 
 | |
|   CopyMem (ResourceList, mStmResourcesPtr, mStmResourceSizeUsed);
 | |
|   *ResourceSize = (UINT32)mStmResourceSizeUsed;
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Set valid bit for MSEG MSR.
 | |
| 
 | |
|   @param Buffer Ap function buffer. (not used)
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| EFIAPI
 | |
| EnableMsegMsr (
 | |
|   IN VOID  *Buffer
 | |
|   )
 | |
| {
 | |
|   MSR_IA32_SMM_MONITOR_CTL_REGISTER  SmmMonitorCtl;
 | |
| 
 | |
|   SmmMonitorCtl.Uint64 = AsmReadMsr64 (MSR_IA32_SMM_MONITOR_CTL);
 | |
|   SmmMonitorCtl.Bits.Valid = 1;
 | |
|   AsmWriteMsr64 (MSR_IA32_SMM_MONITOR_CTL, SmmMonitorCtl.Uint64);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Get 4K page aligned VMCS size.
 | |
| 
 | |
|   @return 4K page aligned VMCS size
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| GetVmcsSize (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   MSR_IA32_VMX_BASIC_REGISTER  VmxBasic;
 | |
| 
 | |
|   //
 | |
|   // Read VMCS size and and align to 4KB
 | |
|   //
 | |
|   VmxBasic.Uint64 = AsmReadMsr64 (MSR_IA32_VMX_BASIC);
 | |
|   return ALIGN_VALUE (VmxBasic.Bits.VmcsSize, SIZE_4KB);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Check STM image size.
 | |
| 
 | |
|   @param StmImage      STM image
 | |
|   @param StmImageSize  STM image size
 | |
| 
 | |
|   @retval TRUE  check pass
 | |
|   @retval FALSE check fail
 | |
| **/
 | |
| BOOLEAN
 | |
| StmCheckStmImage (
 | |
|   IN EFI_PHYSICAL_ADDRESS StmImage,
 | |
|   IN UINTN                StmImageSize
 | |
|   )
 | |
| {
 | |
|   UINTN                     MinMsegSize;
 | |
|   STM_HEADER                *StmHeader;
 | |
|   IA32_VMX_MISC_REGISTER    VmxMiscMsr;
 | |
| 
 | |
|   //
 | |
|   // Check to see if STM image is compatible with CPU
 | |
|   //
 | |
|   StmHeader = (STM_HEADER *)(UINTN)StmImage;
 | |
|   VmxMiscMsr.Uint64 = AsmReadMsr64 (MSR_IA32_VMX_MISC);
 | |
|   if (StmHeader->HwStmHdr.MsegHeaderRevision != VmxMiscMsr.Bits.MsegRevisionIdentifier) {
 | |
|     DEBUG ((DEBUG_ERROR, "STM Image not compatible with CPU\n"));
 | |
|     DEBUG ((DEBUG_ERROR, "  StmHeader->HwStmHdr.MsegHeaderRevision = %08x\n", StmHeader->HwStmHdr.MsegHeaderRevision));
 | |
|     DEBUG ((DEBUG_ERROR, "  VmxMiscMsr.Bits.MsegRevisionIdentifier = %08x\n", VmxMiscMsr.Bits.MsegRevisionIdentifier));
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Get Minimal required Mseg size
 | |
|   //
 | |
|   MinMsegSize = (EFI_PAGES_TO_SIZE (EFI_SIZE_TO_PAGES (StmHeader->SwStmHdr.StaticImageSize)) +
 | |
|                  StmHeader->SwStmHdr.AdditionalDynamicMemorySize +
 | |
|                  (StmHeader->SwStmHdr.PerProcDynamicMemorySize + GetVmcsSize () * 2) * gSmst->NumberOfCpus);
 | |
|   if (MinMsegSize < StmImageSize) {
 | |
|     MinMsegSize = StmImageSize;
 | |
|   }
 | |
| 
 | |
|   if (StmHeader->HwStmHdr.Cr3Offset >= StmHeader->SwStmHdr.StaticImageSize) {
 | |
|     //
 | |
|     // We will create page table, just in case that SINIT does not create it.
 | |
|     //
 | |
|     if (MinMsegSize < StmHeader->HwStmHdr.Cr3Offset + EFI_PAGES_TO_SIZE(6)) {
 | |
|       MinMsegSize = StmHeader->HwStmHdr.Cr3Offset + EFI_PAGES_TO_SIZE(6);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Check if it exceeds MSEG size
 | |
|   //
 | |
|   if (MinMsegSize > mMsegSize) {
 | |
|     DEBUG ((DEBUG_ERROR, "MSEG too small.  Min MSEG Size = %08x  Current MSEG Size = %08x\n", MinMsegSize, mMsegSize));
 | |
|     DEBUG ((DEBUG_ERROR, "  StmHeader->SwStmHdr.StaticImageSize             = %08x\n", StmHeader->SwStmHdr.StaticImageSize));
 | |
|     DEBUG ((DEBUG_ERROR, "  StmHeader->SwStmHdr.AdditionalDynamicMemorySize = %08x\n", StmHeader->SwStmHdr.AdditionalDynamicMemorySize));
 | |
|     DEBUG ((DEBUG_ERROR, "  StmHeader->SwStmHdr.PerProcDynamicMemorySize    = %08x\n", StmHeader->SwStmHdr.PerProcDynamicMemorySize));
 | |
|     DEBUG ((DEBUG_ERROR, "  VMCS Size                                       = %08x\n", GetVmcsSize ()));
 | |
|     DEBUG ((DEBUG_ERROR, "  Max CPUs                                        = %08x\n", gSmst->NumberOfCpus));
 | |
|     DEBUG ((DEBUG_ERROR, "  StmHeader->HwStmHdr.Cr3Offset                   = %08x\n", StmHeader->HwStmHdr.Cr3Offset));
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Load STM image to MSEG.
 | |
| 
 | |
|   @param StmImage      STM image
 | |
|   @param StmImageSize  STM image size
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| StmLoadStmImage (
 | |
|   IN EFI_PHYSICAL_ADDRESS StmImage,
 | |
|   IN UINTN                StmImageSize
 | |
|   )
 | |
| {
 | |
|   MSR_IA32_SMM_MONITOR_CTL_REGISTER  SmmMonitorCtl;
 | |
|   UINT32                             MsegBase;
 | |
|   STM_HEADER                         *StmHeader;
 | |
| 
 | |
|   //
 | |
|   // Get MSEG base address from MSR_IA32_SMM_MONITOR_CTL
 | |
|   //
 | |
|   SmmMonitorCtl.Uint64 = AsmReadMsr64 (MSR_IA32_SMM_MONITOR_CTL);
 | |
|   MsegBase = SmmMonitorCtl.Bits.MsegBase << 12;
 | |
| 
 | |
|   //
 | |
|   // Zero all of MSEG base address
 | |
|   //
 | |
|   ZeroMem ((VOID *)(UINTN)MsegBase, mMsegSize);
 | |
| 
 | |
|   //
 | |
|   // Copy STM Image into MSEG
 | |
|   //
 | |
|   CopyMem ((VOID *)(UINTN)MsegBase, (VOID *)(UINTN)StmImage, StmImageSize);
 | |
| 
 | |
|   //
 | |
|   // STM Header is at the beginning of the STM Image
 | |
|   //
 | |
|   StmHeader = (STM_HEADER *)(UINTN)StmImage;
 | |
| 
 | |
|   StmGen4GPageTable ((UINTN)MsegBase + StmHeader->HwStmHdr.Cr3Offset);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Load STM image to MSEG.
 | |
| 
 | |
|   @param StmImage      STM image
 | |
|   @param StmImageSize  STM image size
 | |
| 
 | |
|   @retval EFI_SUCCESS            Load STM to MSEG successfully
 | |
|   @retval EFI_ALREADY_STARTED    STM image is already loaded to MSEG
 | |
|   @retval EFI_BUFFER_TOO_SMALL   MSEG is smaller than minimal requirement of STM image
 | |
|   @retval EFI_UNSUPPORTED        MSEG is not enabled
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| LoadMonitor (
 | |
|   IN EFI_PHYSICAL_ADDRESS StmImage,
 | |
|   IN UINTN                StmImageSize
 | |
|   )
 | |
| {
 | |
|   MSR_IA32_SMM_MONITOR_CTL_REGISTER  SmmMonitorCtl;
 | |
| 
 | |
|   if (mLockLoadMonitor) {
 | |
|     return EFI_ACCESS_DENIED;
 | |
|   }
 | |
| 
 | |
|   SmmMonitorCtl.Uint64 = AsmReadMsr64 (MSR_IA32_SMM_MONITOR_CTL);
 | |
|   if (SmmMonitorCtl.Bits.MsegBase == 0) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   if (!StmCheckStmImage (StmImage, StmImageSize)) {
 | |
|     return EFI_BUFFER_TOO_SMALL;
 | |
|   }
 | |
| 
 | |
|   // Record STM_HASH to PCR 0, just in case it is NOT TXT launch, we still need provide the evidence.
 | |
|   TpmMeasureAndLogData(
 | |
|     0,                        // PcrIndex
 | |
|     TXT_EVTYPE_STM_HASH,      // EventType
 | |
|     NULL,                     // EventLog
 | |
|     0,                        // LogLen
 | |
|     (VOID *)(UINTN)StmImage,  // HashData
 | |
|     StmImageSize              // HashDataLen
 | |
|     );
 | |
| 
 | |
|   StmLoadStmImage (StmImage, StmImageSize);
 | |
| 
 | |
|   mStmState |= EFI_SM_MONITOR_STATE_ENABLED;
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   This function return BIOS STM resource.
 | |
|   Produced by SmmStm.
 | |
|   Comsumed by SmmMpService when Init.
 | |
| 
 | |
|   @return BIOS STM resource
 | |
| 
 | |
| **/
 | |
| VOID *
 | |
| GetStmResource(
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   return mStmResourcesPtr;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   This function notify STM resource change.
 | |
| 
 | |
|   @param StmResource BIOS STM resource
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| NotifyStmResourceChange (
 | |
|   VOID *StmResource
 | |
|   )
 | |
| {
 | |
|   UINTN                         Index;
 | |
|   TXT_PROCESSOR_SMM_DESCRIPTOR  *Psd;
 | |
| 
 | |
|   for (Index = 0; Index < gSmst->NumberOfCpus; Index++) {
 | |
|     Psd = (TXT_PROCESSOR_SMM_DESCRIPTOR *)((UINTN)gSmst->CpuSaveState[Index] - SMRAM_SAVE_STATE_MAP_OFFSET + TXT_SMM_PSD_OFFSET);
 | |
|     Psd->BiosHwResourceRequirementsPtr = (UINT64)(UINTN)StmResource;
 | |
|   }
 | |
|   return ;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   This is STM setup BIOS callback.
 | |
| **/
 | |
| VOID
 | |
| EFIAPI
 | |
| SmmStmSetup (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   mStmState |= EFI_SM_MONITOR_STATE_ACTIVATED;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   This is STM teardown BIOS callback.
 | |
| **/
 | |
| VOID
 | |
| EFIAPI
 | |
| SmmStmTeardown (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   mStmState &= ~EFI_SM_MONITOR_STATE_ACTIVATED;
 | |
| }
 | |
| 
 |