mirror of
				https://github.com/acidanthera/audk.git
				synced 2025-10-25 17:23:53 +02:00 
			
		
		
		
	Includes the profiler SMI in the profile itself for completeness. Co-authored-by: Michael Kubacki <michael.kubacki@microsoft.com> Signed-off-by: Michael Kubacki <michael.kubacki@microsoft.com>
		
			
				
	
	
		
			1429 lines
		
	
	
		
			46 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1429 lines
		
	
	
		
			46 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   SMI handler profile support.
 | |
| 
 | |
| Copyright (c) 2017, Intel Corporation. All rights reserved.<BR>
 | |
| Copyright (c) Microsoft Corporation.
 | |
| SPDX-License-Identifier: BSD-2-Clause-Patent
 | |
| 
 | |
| **/
 | |
| 
 | |
| #include <PiSmm.h>
 | |
| #include <Library/BaseLib.h>
 | |
| #include <Library/BaseMemoryLib.h>
 | |
| #include <Library/MemoryAllocationLib.h>
 | |
| #include <Library/UefiBootServicesTableLib.h>
 | |
| #include <Library/SmmServicesTableLib.h>
 | |
| #include <Library/DebugLib.h>
 | |
| #include <Library/PrintLib.h>
 | |
| #include <Library/UefiLib.h>
 | |
| #include <Library/DevicePathLib.h>
 | |
| #include <Library/PeCoffGetEntryPointLib.h>
 | |
| #include <Protocol/LoadedImage.h>
 | |
| #include <Protocol/SmmAccess2.h>
 | |
| #include <Protocol/SmmReadyToLock.h>
 | |
| #include <Protocol/SmmEndOfDxe.h>
 | |
| 
 | |
| #include <Guid/SmiHandlerProfile.h>
 | |
| 
 | |
| #include "PiSmmCore.h"
 | |
| 
 | |
| #define GET_OCCUPIED_SIZE(ActualSize, Alignment) \
 | |
|   ((ActualSize) + (((Alignment) - ((ActualSize) & ((Alignment) - 1))) & ((Alignment) - 1)))
 | |
| 
 | |
| typedef struct {
 | |
|   EFI_GUID            FileGuid;
 | |
|   PHYSICAL_ADDRESS    EntryPoint;
 | |
|   PHYSICAL_ADDRESS    ImageBase;
 | |
|   UINT64              ImageSize;
 | |
|   UINT32              ImageRef;
 | |
|   UINT16              PdbStringSize;
 | |
|   CHAR8               *PdbString;
 | |
| } IMAGE_STRUCT;
 | |
| 
 | |
| /**
 | |
|   Register SMI handler profile handler.
 | |
| **/
 | |
| VOID
 | |
| RegisterSmiHandlerProfileHandler (
 | |
|   VOID
 | |
|   );
 | |
| 
 | |
| /**
 | |
|   Build SMI handler profile database.
 | |
| **/
 | |
| VOID
 | |
| BuildSmiHandlerProfileDatabase (
 | |
|   VOID
 | |
|   );
 | |
| 
 | |
| /**
 | |
|   Retrieves and returns a pointer to the entry point to a PE/COFF image that has been loaded
 | |
|   into system memory with the PE/COFF Loader Library functions.
 | |
| 
 | |
|   Retrieves the entry point to the PE/COFF image specified by Pe32Data and returns this entry
 | |
|   point in EntryPoint.  If the entry point could not be retrieved from the PE/COFF image, then
 | |
|   return RETURN_INVALID_PARAMETER.  Otherwise return RETURN_SUCCESS.
 | |
|   If Pe32Data is NULL, then ASSERT().
 | |
|   If EntryPoint is NULL, then ASSERT().
 | |
| 
 | |
|   @param  Pe32Data                  The pointer to the PE/COFF image that is loaded in system memory.
 | |
|   @param  EntryPoint                The pointer to entry point to the PE/COFF image to return.
 | |
| 
 | |
|   @retval RETURN_SUCCESS            EntryPoint was returned.
 | |
|   @retval RETURN_INVALID_PARAMETER  The entry point could not be found in the PE/COFF image.
 | |
| 
 | |
| **/
 | |
| RETURN_STATUS
 | |
| InternalPeCoffGetEntryPoint (
 | |
|   IN  VOID  *Pe32Data,
 | |
|   OUT VOID  **EntryPoint
 | |
|   );
 | |
| 
 | |
| extern LIST_ENTRY  mSmiEntryList;
 | |
| extern LIST_ENTRY  mHardwareSmiEntryList;
 | |
| extern SMI_ENTRY   mRootSmiEntry;
 | |
| 
 | |
| extern SMI_HANDLER_PROFILE_PROTOCOL  mSmiHandlerProfile;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED LIST_ENTRY  mHardwareSmiEntryList = INITIALIZE_LIST_HEAD_VARIABLE (mHardwareSmiEntryList);
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED LIST_ENTRY  mRootSmiEntryList = INITIALIZE_LIST_HEAD_VARIABLE (mRootSmiEntryList);
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED LIST_ENTRY  *mSmmCoreRootSmiEntryList     = &mRootSmiEntryList;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED LIST_ENTRY  *mSmmCoreSmiEntryList         = &mSmiEntryList;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED LIST_ENTRY  *mSmmCoreHardwareSmiEntryList = &mHardwareSmiEntryList;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED IMAGE_STRUCT  *mImageStruct;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINT32        mImageStructCountMax;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINT32        mImageStructCount;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED VOID   *mSmiHandlerProfileDatabase;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINTN  mSmiHandlerProfileDatabaseSize;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINTN  mSmmImageDatabaseSize;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINTN  mSmmRootSmiDatabaseSize;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINTN  mSmmSmiDatabaseSize;
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED UINTN  mSmmHardwareSmiDatabaseSize;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED BOOLEAN  mSmiHandlerProfileRecordingStatus;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED SMI_HANDLER_PROFILE_PROTOCOL  mSmiHandlerProfile = {
 | |
|   SmiHandlerProfileRegisterHandler,
 | |
|   SmiHandlerProfileUnregisterHandler,
 | |
| };
 | |
| 
 | |
| /**
 | |
|   This function dump raw data.
 | |
| 
 | |
|   @param  Data  raw data
 | |
|   @param  Size  raw data size
 | |
| **/
 | |
| VOID
 | |
| InternalDumpData (
 | |
|   IN UINT8  *Data,
 | |
|   IN UINTN  Size
 | |
|   )
 | |
| {
 | |
|   UINTN  Index;
 | |
| 
 | |
|   for (Index = 0; Index < Size; Index++) {
 | |
|     DEBUG ((DEBUG_INFO, "%02x ", (UINTN)Data[Index]));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get GUID name for an image.
 | |
| 
 | |
|   @param[in]  LoadedImage LoadedImage protocol.
 | |
|   @param[out] Guid        Guid of the FFS
 | |
| **/
 | |
| VOID
 | |
| GetDriverGuid (
 | |
|   IN  EFI_LOADED_IMAGE_PROTOCOL  *LoadedImage,
 | |
|   OUT EFI_GUID                   *Guid
 | |
|   )
 | |
| {
 | |
|   EFI_GUID  *FileName;
 | |
| 
 | |
|   FileName = NULL;
 | |
|   if ((DevicePathType (LoadedImage->FilePath) == MEDIA_DEVICE_PATH) &&
 | |
|       (DevicePathSubType (LoadedImage->FilePath) == MEDIA_PIWG_FW_FILE_DP))
 | |
|   {
 | |
|     FileName = &((MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *)LoadedImage->FilePath)->FvFileName;
 | |
|   }
 | |
| 
 | |
|   if (FileName != NULL) {
 | |
|     CopyGuid (Guid, FileName);
 | |
|   } else {
 | |
|     ZeroMem (Guid, sizeof (EFI_GUID));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Add image structure.
 | |
| 
 | |
|   @param  ImageBase         image base
 | |
|   @param  ImageSize         image size
 | |
|   @param  EntryPoint        image entry point
 | |
|   @param  Guid              FFS GUID of the image
 | |
|   @param  PdbString         image PDB string
 | |
| **/
 | |
| VOID
 | |
| AddImageStruct (
 | |
|   IN PHYSICAL_ADDRESS  ImageBase,
 | |
|   IN UINT64            ImageSize,
 | |
|   IN PHYSICAL_ADDRESS  EntryPoint,
 | |
|   IN EFI_GUID          *Guid,
 | |
|   IN CHAR8             *PdbString
 | |
|   )
 | |
| {
 | |
|   UINTN  PdbStringSize;
 | |
| 
 | |
|   if (mImageStructCount >= mImageStructCountMax) {
 | |
|     ASSERT (FALSE);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   CopyGuid (&mImageStruct[mImageStructCount].FileGuid, Guid);
 | |
|   mImageStruct[mImageStructCount].ImageRef   = mImageStructCount;
 | |
|   mImageStruct[mImageStructCount].ImageBase  = ImageBase;
 | |
|   mImageStruct[mImageStructCount].ImageSize  = ImageSize;
 | |
|   mImageStruct[mImageStructCount].EntryPoint = EntryPoint;
 | |
|   if (PdbString != NULL) {
 | |
|     PdbStringSize                             = AsciiStrSize (PdbString);
 | |
|     mImageStruct[mImageStructCount].PdbString = AllocateCopyPool (PdbStringSize, PdbString);
 | |
|     if (mImageStruct[mImageStructCount].PdbString != NULL) {
 | |
|       mImageStruct[mImageStructCount].PdbStringSize = (UINT16)PdbStringSize;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   mImageStructCount++;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   return an image structure based upon image address.
 | |
| 
 | |
|   @param  Address  image address
 | |
| 
 | |
|   @return image structure
 | |
| **/
 | |
| IMAGE_STRUCT *
 | |
| AddressToImageStruct (
 | |
|   IN UINTN  Address
 | |
|   )
 | |
| {
 | |
|   UINTN  Index;
 | |
| 
 | |
|   for (Index = 0; Index < mImageStructCount; Index++) {
 | |
|     if ((Address >= mImageStruct[Index].ImageBase) &&
 | |
|         (Address < mImageStruct[Index].ImageBase + mImageStruct[Index].ImageSize))
 | |
|     {
 | |
|       return &mImageStruct[Index];
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   return an image reference index based upon image address.
 | |
| 
 | |
|   @param  Address  image address
 | |
| 
 | |
|   @return image reference index
 | |
| **/
 | |
| UINT32
 | |
| AddressToImageRef (
 | |
|   IN UINTN  Address
 | |
|   )
 | |
| {
 | |
|   IMAGE_STRUCT  *ImageStruct;
 | |
| 
 | |
|   ImageStruct = AddressToImageStruct (Address);
 | |
|   if (ImageStruct != NULL) {
 | |
|     return ImageStruct->ImageRef;
 | |
|   }
 | |
| 
 | |
|   return (UINT32)-1;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Collect SMM image information based upon loaded image protocol.
 | |
| **/
 | |
| VOID
 | |
| GetSmmLoadedImage (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                 Status;
 | |
|   UINTN                      NoHandles;
 | |
|   UINTN                      HandleBufferSize;
 | |
|   EFI_HANDLE                 *HandleBuffer;
 | |
|   UINTN                      Index;
 | |
|   EFI_LOADED_IMAGE_PROTOCOL  *LoadedImage;
 | |
|   CHAR16                     *PathStr;
 | |
|   EFI_SMM_DRIVER_ENTRY       *LoadedImagePrivate;
 | |
|   PHYSICAL_ADDRESS           EntryPoint;
 | |
|   VOID                       *EntryPointInImage;
 | |
|   EFI_GUID                   Guid;
 | |
|   CHAR8                      *PdbString;
 | |
|   PHYSICAL_ADDRESS           RealImageBase;
 | |
| 
 | |
|   HandleBufferSize = 0;
 | |
|   HandleBuffer     = NULL;
 | |
|   Status           = gSmst->SmmLocateHandle (
 | |
|                               ByProtocol,
 | |
|                               &gEfiLoadedImageProtocolGuid,
 | |
|                               NULL,
 | |
|                               &HandleBufferSize,
 | |
|                               HandleBuffer
 | |
|                               );
 | |
|   if (Status != EFI_BUFFER_TOO_SMALL) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   HandleBuffer = AllocateZeroPool (HandleBufferSize);
 | |
|   if (HandleBuffer == NULL) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Status = gSmst->SmmLocateHandle (
 | |
|                     ByProtocol,
 | |
|                     &gEfiLoadedImageProtocolGuid,
 | |
|                     NULL,
 | |
|                     &HandleBufferSize,
 | |
|                     HandleBuffer
 | |
|                     );
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   NoHandles            = HandleBufferSize/sizeof (EFI_HANDLE);
 | |
|   mImageStructCountMax = (UINT32)NoHandles;
 | |
|   mImageStruct         = AllocateZeroPool (mImageStructCountMax * sizeof (IMAGE_STRUCT));
 | |
|   if (mImageStruct == NULL) {
 | |
|     goto Done;
 | |
|   }
 | |
| 
 | |
|   for (Index = 0; Index < NoHandles; Index++) {
 | |
|     Status = gSmst->SmmHandleProtocol (
 | |
|                       HandleBuffer[Index],
 | |
|                       &gEfiLoadedImageProtocolGuid,
 | |
|                       (VOID **)&LoadedImage
 | |
|                       );
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     PathStr = ConvertDevicePathToText (LoadedImage->FilePath, TRUE, TRUE);
 | |
|     GetDriverGuid (LoadedImage, &Guid);
 | |
|     DEBUG ((DEBUG_INFO, "Image: %g ", &Guid));
 | |
| 
 | |
|     EntryPoint         = 0;
 | |
|     LoadedImagePrivate = BASE_CR (LoadedImage, EFI_SMM_DRIVER_ENTRY, SmmLoadedImage);
 | |
|     RealImageBase      = (UINTN)LoadedImage->ImageBase;
 | |
|     if (LoadedImagePrivate->Signature == EFI_SMM_DRIVER_ENTRY_SIGNATURE) {
 | |
|       EntryPoint = LoadedImagePrivate->ImageEntryPoint;
 | |
|       if ((EntryPoint != 0) && ((EntryPoint < (UINTN)LoadedImage->ImageBase) || (EntryPoint >= ((UINTN)LoadedImage->ImageBase + LoadedImage->ImageSize)))) {
 | |
|         //
 | |
|         // If the EntryPoint is not in the range of image buffer, it should come from emulation environment.
 | |
|         // So patch ImageBuffer here to align the EntryPoint.
 | |
|         //
 | |
|         Status = InternalPeCoffGetEntryPoint (LoadedImage->ImageBase, &EntryPointInImage);
 | |
|         ASSERT_EFI_ERROR (Status);
 | |
|         RealImageBase = (UINTN)LoadedImage->ImageBase + EntryPoint - (UINTN)EntryPointInImage;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     DEBUG ((DEBUG_INFO, "(0x%lx - 0x%lx", RealImageBase, LoadedImage->ImageSize));
 | |
|     if (EntryPoint != 0) {
 | |
|       DEBUG ((DEBUG_INFO, ", EntryPoint:0x%lx", EntryPoint));
 | |
|     }
 | |
| 
 | |
|     DEBUG ((DEBUG_INFO, ")\n"));
 | |
| 
 | |
|     if (RealImageBase != 0) {
 | |
|       PdbString = PeCoffLoaderGetPdbPointer ((VOID *)(UINTN)RealImageBase);
 | |
|       DEBUG ((DEBUG_INFO, "       pdb - %a\n", PdbString));
 | |
|     } else {
 | |
|       PdbString = NULL;
 | |
|     }
 | |
| 
 | |
|     DEBUG ((DEBUG_INFO, "       (%s)\n", PathStr));
 | |
| 
 | |
|     AddImageStruct (RealImageBase, LoadedImage->ImageSize, EntryPoint, &Guid, PdbString);
 | |
|   }
 | |
| 
 | |
| Done:
 | |
|   FreePool (HandleBuffer);
 | |
|   return;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dump SMI child context.
 | |
| 
 | |
|   @param HandlerType  the handler type
 | |
|   @param Context      the handler context
 | |
|   @param ContextSize  the handler context size
 | |
| **/
 | |
| VOID
 | |
| DumpSmiChildContext (
 | |
|   IN EFI_GUID  *HandlerType,
 | |
|   IN VOID      *Context,
 | |
|   IN UINTN     ContextSize
 | |
|   )
 | |
| {
 | |
|   CHAR16  *Str;
 | |
| 
 | |
|   if (CompareGuid (HandlerType, &gEfiSmmSwDispatch2ProtocolGuid)) {
 | |
|     DEBUG ((DEBUG_INFO, "  SwSmi - 0x%lx\n", ((SMI_HANDLER_PROFILE_SW_REGISTER_CONTEXT *)Context)->SwSmiInputValue));
 | |
|   } else if (CompareGuid (HandlerType, &gEfiSmmSxDispatch2ProtocolGuid)) {
 | |
|     DEBUG ((DEBUG_INFO, "  SxType - 0x%x\n", ((EFI_SMM_SX_REGISTER_CONTEXT *)Context)->Type));
 | |
|     DEBUG ((DEBUG_INFO, "  SxPhase - 0x%x\n", ((EFI_SMM_SX_REGISTER_CONTEXT *)Context)->Phase));
 | |
|   } else if (CompareGuid (HandlerType, &gEfiSmmPowerButtonDispatch2ProtocolGuid)) {
 | |
|     DEBUG ((DEBUG_INFO, "  PowerButtonPhase - 0x%x\n", ((EFI_SMM_POWER_BUTTON_REGISTER_CONTEXT *)Context)->Phase));
 | |
|   } else if (CompareGuid (HandlerType, &gEfiSmmStandbyButtonDispatch2ProtocolGuid)) {
 | |
|     DEBUG ((DEBUG_INFO, "  StandbyButtonPhase - 0x%x\n", ((EFI_SMM_STANDBY_BUTTON_REGISTER_CONTEXT *)Context)->Phase));
 | |
|   } else if (CompareGuid (HandlerType, &gEfiSmmPeriodicTimerDispatch2ProtocolGuid)) {
 | |
|     DEBUG ((DEBUG_INFO, "  PeriodicTimerPeriod - %ld\n", ((EFI_SMM_PERIODIC_TIMER_REGISTER_CONTEXT *)Context)->Period));
 | |
|     DEBUG ((DEBUG_INFO, "  PeriodicTimerSmiTickInterval - %ld\n", ((EFI_SMM_PERIODIC_TIMER_REGISTER_CONTEXT *)Context)->SmiTickInterval));
 | |
|   } else if (CompareGuid (HandlerType, &gEfiSmmGpiDispatch2ProtocolGuid)) {
 | |
|     DEBUG ((DEBUG_INFO, "  GpiNum - 0x%lx\n", ((EFI_SMM_GPI_REGISTER_CONTEXT *)Context)->GpiNum));
 | |
|   } else if (CompareGuid (HandlerType, &gEfiSmmIoTrapDispatch2ProtocolGuid)) {
 | |
|     DEBUG ((DEBUG_INFO, "  IoTrapAddress - 0x%x\n", ((EFI_SMM_IO_TRAP_REGISTER_CONTEXT *)Context)->Address));
 | |
|     DEBUG ((DEBUG_INFO, "  IoTrapLength - 0x%x\n", ((EFI_SMM_IO_TRAP_REGISTER_CONTEXT *)Context)->Length));
 | |
|     DEBUG ((DEBUG_INFO, "  IoTrapType - 0x%x\n", ((EFI_SMM_IO_TRAP_REGISTER_CONTEXT *)Context)->Type));
 | |
|   } else if (CompareGuid (HandlerType, &gEfiSmmUsbDispatch2ProtocolGuid)) {
 | |
|     DEBUG ((DEBUG_INFO, "  UsbType - 0x%x\n", ((SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT *)Context)->Type));
 | |
|     Str = ConvertDevicePathToText ((EFI_DEVICE_PATH_PROTOCOL *)(((SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT *)Context) + 1), TRUE, TRUE);
 | |
|     DEBUG ((DEBUG_INFO, "  UsbDevicePath - %s\n", Str));
 | |
|     if (Str != NULL) {
 | |
|       FreePool (Str);
 | |
|     }
 | |
|   } else {
 | |
|     DEBUG ((DEBUG_INFO, "  Context - "));
 | |
|     InternalDumpData (Context, ContextSize);
 | |
|     DEBUG ((DEBUG_INFO, "\n"));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dump all SMI handlers associated with SmiEntry.
 | |
| 
 | |
|   @param SmiEntry  SMI entry.
 | |
| **/
 | |
| VOID
 | |
| DumpSmiHandlerOnSmiEntry (
 | |
|   IN SMI_ENTRY  *SmiEntry
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY    *ListEntry;
 | |
|   SMI_HANDLER   *SmiHandler;
 | |
|   IMAGE_STRUCT  *ImageStruct;
 | |
| 
 | |
|   ListEntry = &SmiEntry->SmiHandlers;
 | |
|   for (ListEntry = ListEntry->ForwardLink;
 | |
|        ListEntry != &SmiEntry->SmiHandlers;
 | |
|        ListEntry = ListEntry->ForwardLink)
 | |
|   {
 | |
|     SmiHandler  = CR (ListEntry, SMI_HANDLER, Link, SMI_HANDLER_SIGNATURE);
 | |
|     ImageStruct = AddressToImageStruct ((UINTN)SmiHandler->Handler);
 | |
|     if (ImageStruct != NULL) {
 | |
|       DEBUG ((DEBUG_INFO, " Module - %g", &ImageStruct->FileGuid));
 | |
|     }
 | |
| 
 | |
|     if ((ImageStruct != NULL) && (ImageStruct->PdbString[0] != 0)) {
 | |
|       DEBUG ((DEBUG_INFO, " (Pdb - %a)", ImageStruct->PdbString));
 | |
|     }
 | |
| 
 | |
|     DEBUG ((DEBUG_INFO, "\n"));
 | |
|     if (SmiHandler->ContextSize != 0) {
 | |
|       DumpSmiChildContext (&SmiEntry->HandlerType, SmiHandler->Context, SmiHandler->ContextSize);
 | |
|     }
 | |
| 
 | |
|     DEBUG ((DEBUG_INFO, "  Handler - 0x%x", SmiHandler->Handler));
 | |
|     if (ImageStruct != NULL) {
 | |
|       DEBUG ((DEBUG_INFO, " <== RVA - 0x%x", (UINTN)SmiHandler->Handler - (UINTN)ImageStruct->ImageBase));
 | |
|     }
 | |
| 
 | |
|     DEBUG ((DEBUG_INFO, "\n"));
 | |
|     DEBUG ((DEBUG_INFO, "  CallerAddr - 0x%x", SmiHandler->CallerAddr));
 | |
|     if (ImageStruct != NULL) {
 | |
|       DEBUG ((DEBUG_INFO, " <== RVA - 0x%x", SmiHandler->CallerAddr - (UINTN)ImageStruct->ImageBase));
 | |
|     }
 | |
| 
 | |
|     DEBUG ((DEBUG_INFO, "\n"));
 | |
|   }
 | |
| 
 | |
|   return;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dump all SMI entry on the list.
 | |
| 
 | |
|   @param SmiEntryList a list of SMI entry.
 | |
| **/
 | |
| VOID
 | |
| DumpSmiEntryList (
 | |
|   IN LIST_ENTRY  *SmiEntryList
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY  *ListEntry;
 | |
|   SMI_ENTRY   *SmiEntry;
 | |
| 
 | |
|   ListEntry = SmiEntryList;
 | |
|   for (ListEntry = ListEntry->ForwardLink;
 | |
|        ListEntry != SmiEntryList;
 | |
|        ListEntry = ListEntry->ForwardLink)
 | |
|   {
 | |
|     SmiEntry = CR (ListEntry, SMI_ENTRY, AllEntries, SMI_ENTRY_SIGNATURE);
 | |
|     DEBUG ((DEBUG_INFO, "SmiEntry - %g\n", &SmiEntry->HandlerType));
 | |
|     DumpSmiHandlerOnSmiEntry (SmiEntry);
 | |
|   }
 | |
| 
 | |
|   return;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMM Ready To Lock event notification handler.
 | |
| 
 | |
|   This function collects all SMM image information and build SmiHandleProfile database,
 | |
|   and register SmiHandlerProfile SMI handler.
 | |
| 
 | |
|   @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
 | |
| SmmReadyToLockInSmiHandlerProfile (
 | |
|   IN CONST EFI_GUID  *Protocol,
 | |
|   IN VOID            *Interface,
 | |
|   IN EFI_HANDLE      Handle
 | |
|   )
 | |
| {
 | |
|   RegisterSmiHandlerProfileHandler ();
 | |
| 
 | |
|   //
 | |
|   // Dump all image
 | |
|   //
 | |
|   DEBUG ((DEBUG_INFO, "##################\n"));
 | |
|   DEBUG ((DEBUG_INFO, "# IMAGE DATABASE #\n"));
 | |
|   DEBUG ((DEBUG_INFO, "##################\n"));
 | |
|   GetSmmLoadedImage ();
 | |
|   DEBUG ((DEBUG_INFO, "\n"));
 | |
| 
 | |
|   //
 | |
|   // Dump SMI Handler
 | |
|   //
 | |
|   DEBUG ((DEBUG_INFO, "########################\n"));
 | |
|   DEBUG ((DEBUG_INFO, "# SMI Handler DATABASE #\n"));
 | |
|   DEBUG ((DEBUG_INFO, "########################\n"));
 | |
| 
 | |
|   DEBUG ((DEBUG_INFO, "# 1. ROOT SMI Handler #\n"));
 | |
|   DEBUG_CODE (
 | |
|     DumpSmiEntryList (mSmmCoreRootSmiEntryList);
 | |
|     );
 | |
| 
 | |
|   DEBUG ((DEBUG_INFO, "# 2. GUID SMI Handler #\n"));
 | |
|   DEBUG_CODE (
 | |
|     DumpSmiEntryList (mSmmCoreSmiEntryList);
 | |
|     );
 | |
| 
 | |
|   DEBUG ((DEBUG_INFO, "# 3. Hardware SMI Handler #\n"));
 | |
|   DEBUG_CODE (
 | |
|     DumpSmiEntryList (mSmmCoreHardwareSmiEntryList);
 | |
|     );
 | |
| 
 | |
|   DEBUG ((DEBUG_INFO, "\n"));
 | |
| 
 | |
|   BuildSmiHandlerProfileDatabase ();
 | |
| 
 | |
|   if (mImageStruct != NULL) {
 | |
|     FreePool (mImageStruct);
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   returns SMM image data base size.
 | |
| 
 | |
|   @return SMM image data base size.
 | |
| **/
 | |
| UINTN
 | |
| GetSmmImageDatabaseSize (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   UINTN   Size;
 | |
|   UINT32  Index;
 | |
| 
 | |
|   Size = 0;
 | |
|   for (Index = 0; Index < mImageStructCount; Index++) {
 | |
|     Size += sizeof (SMM_CORE_IMAGE_DATABASE_STRUCTURE) + GET_OCCUPIED_SIZE (mImageStruct[Index].PdbStringSize, sizeof (UINT64));
 | |
|   }
 | |
| 
 | |
|   return Size;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   returns all SMI handlers' size associated with SmiEntry.
 | |
| 
 | |
|   @param SmiEntry  SMI entry.
 | |
| 
 | |
|   @return all SMI handlers' size associated with SmiEntry.
 | |
| **/
 | |
| UINTN
 | |
| GetSmmSmiHandlerSizeOnSmiEntry (
 | |
|   IN SMI_ENTRY  *SmiEntry
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY   *ListEntry;
 | |
|   SMI_HANDLER  *SmiHandler;
 | |
|   UINTN        Size;
 | |
| 
 | |
|   Size      = 0;
 | |
|   ListEntry = &SmiEntry->SmiHandlers;
 | |
|   for (ListEntry = ListEntry->ForwardLink;
 | |
|        ListEntry != &SmiEntry->SmiHandlers;
 | |
|        ListEntry = ListEntry->ForwardLink)
 | |
|   {
 | |
|     SmiHandler = CR (ListEntry, SMI_HANDLER, Link, SMI_HANDLER_SIGNATURE);
 | |
|     Size      += sizeof (SMM_CORE_SMI_HANDLER_STRUCTURE) + GET_OCCUPIED_SIZE (SmiHandler->ContextSize, sizeof (UINT64));
 | |
|   }
 | |
| 
 | |
|   return Size;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   return all SMI handler database size on the SMI entry list.
 | |
| 
 | |
|   @param SmiEntryList a list of SMI entry.
 | |
| 
 | |
|   @return all SMI handler database size on the SMI entry list.
 | |
| **/
 | |
| UINTN
 | |
| GetSmmSmiDatabaseSize (
 | |
|   IN LIST_ENTRY  *SmiEntryList
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY  *ListEntry;
 | |
|   SMI_ENTRY   *SmiEntry;
 | |
|   UINTN       Size;
 | |
| 
 | |
|   Size      = 0;
 | |
|   ListEntry = SmiEntryList;
 | |
|   for (ListEntry = ListEntry->ForwardLink;
 | |
|        ListEntry != SmiEntryList;
 | |
|        ListEntry = ListEntry->ForwardLink)
 | |
|   {
 | |
|     SmiEntry = CR (ListEntry, SMI_ENTRY, AllEntries, SMI_ENTRY_SIGNATURE);
 | |
|     Size    += sizeof (SMM_CORE_SMI_DATABASE_STRUCTURE);
 | |
|     Size    += GetSmmSmiHandlerSizeOnSmiEntry (SmiEntry);
 | |
|   }
 | |
| 
 | |
|   return Size;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   return SMI handler profile database size.
 | |
| 
 | |
|   @return SMI handler profile database size.
 | |
| **/
 | |
| UINTN
 | |
| GetSmiHandlerProfileDatabaseSize (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   mSmmImageDatabaseSize       = GetSmmImageDatabaseSize ();
 | |
|   mSmmRootSmiDatabaseSize     = GetSmmSmiDatabaseSize (mSmmCoreRootSmiEntryList);
 | |
|   mSmmSmiDatabaseSize         = GetSmmSmiDatabaseSize (mSmmCoreSmiEntryList);
 | |
|   mSmmHardwareSmiDatabaseSize = GetSmmSmiDatabaseSize (mSmmCoreHardwareSmiEntryList);
 | |
| 
 | |
|   return mSmmImageDatabaseSize + mSmmSmiDatabaseSize + mSmmRootSmiDatabaseSize + mSmmHardwareSmiDatabaseSize;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   get SMM image database.
 | |
| 
 | |
|   @param Data           The buffer to hold SMM image database
 | |
|   @param ExpectedSize   The expected size of the SMM image database
 | |
| 
 | |
|   @return SMM image data base size.
 | |
| **/
 | |
| UINTN
 | |
| GetSmmImageDatabaseData (
 | |
|   IN OUT VOID   *Data,
 | |
|   IN     UINTN  ExpectedSize
 | |
|   )
 | |
| {
 | |
|   SMM_CORE_IMAGE_DATABASE_STRUCTURE  *ImageStruct;
 | |
|   UINTN                              Size;
 | |
|   UINTN                              Index;
 | |
| 
 | |
|   ImageStruct = Data;
 | |
|   Size        = 0;
 | |
|   for (Index = 0; Index < mImageStructCount; Index++) {
 | |
|     if (Size >= ExpectedSize) {
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     if (sizeof (SMM_CORE_IMAGE_DATABASE_STRUCTURE) + GET_OCCUPIED_SIZE (mImageStruct[Index].PdbStringSize, sizeof (UINT64)) > ExpectedSize - Size) {
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     ImageStruct->Header.Signature = SMM_CORE_IMAGE_DATABASE_SIGNATURE;
 | |
|     ImageStruct->Header.Length    = (UINT32)(sizeof (SMM_CORE_IMAGE_DATABASE_STRUCTURE) + GET_OCCUPIED_SIZE (mImageStruct[Index].PdbStringSize, sizeof (UINT64)));
 | |
|     ImageStruct->Header.Revision  = SMM_CORE_IMAGE_DATABASE_REVISION;
 | |
|     CopyGuid (&ImageStruct->FileGuid, &mImageStruct[Index].FileGuid);
 | |
|     ImageStruct->ImageRef   = mImageStruct[Index].ImageRef;
 | |
|     ImageStruct->EntryPoint = mImageStruct[Index].EntryPoint;
 | |
|     ImageStruct->ImageBase  = mImageStruct[Index].ImageBase;
 | |
|     ImageStruct->ImageSize  = mImageStruct[Index].ImageSize;
 | |
|     if (mImageStruct[Index].PdbStringSize != 0) {
 | |
|       ImageStruct->PdbStringOffset = sizeof (SMM_CORE_IMAGE_DATABASE_STRUCTURE);
 | |
|       CopyMem ((VOID *)((UINTN)ImageStruct + ImageStruct->PdbStringOffset), mImageStruct[Index].PdbString, mImageStruct[Index].PdbStringSize);
 | |
|     } else {
 | |
|       ImageStruct->PdbStringOffset = 0;
 | |
|     }
 | |
| 
 | |
|     ImageStruct = (SMM_CORE_IMAGE_DATABASE_STRUCTURE *)((UINTN)ImageStruct + ImageStruct->Header.Length);
 | |
|     Size       += sizeof (SMM_CORE_IMAGE_DATABASE_STRUCTURE) + GET_OCCUPIED_SIZE (mImageStruct[Index].PdbStringSize, sizeof (UINT64));
 | |
|   }
 | |
| 
 | |
|   if (ExpectedSize != Size) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   return Size;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   get all SMI handler data associated with SmiEntry.
 | |
| 
 | |
|   @param SmiEntry       SMI entry.
 | |
|   @param Data           The buffer to hold all SMI handler data
 | |
|   @param MaxSize        The max size of the SMM image database
 | |
|   @param Count          The count of the SMI handler.
 | |
| 
 | |
|   @return SMM image data base size.
 | |
| **/
 | |
| UINTN
 | |
| GetSmmSmiHandlerDataOnSmiEntry (
 | |
|   IN     SMI_ENTRY  *SmiEntry,
 | |
|   IN OUT VOID       *Data,
 | |
|   IN     UINTN      MaxSize,
 | |
|   OUT UINT32        *Count
 | |
|   )
 | |
| {
 | |
|   SMM_CORE_SMI_HANDLER_STRUCTURE  *SmiHandlerStruct;
 | |
|   LIST_ENTRY                      *ListEntry;
 | |
|   SMI_HANDLER                     *SmiHandler;
 | |
|   UINTN                           Size;
 | |
| 
 | |
|   SmiHandlerStruct = Data;
 | |
|   Size             = 0;
 | |
|   *Count           = 0;
 | |
|   ListEntry        = &SmiEntry->SmiHandlers;
 | |
|   for (ListEntry = ListEntry->ForwardLink;
 | |
|        ListEntry != &SmiEntry->SmiHandlers;
 | |
|        ListEntry = ListEntry->ForwardLink)
 | |
|   {
 | |
|     SmiHandler = CR (ListEntry, SMI_HANDLER, Link, SMI_HANDLER_SIGNATURE);
 | |
|     if (Size >= MaxSize) {
 | |
|       *Count = 0;
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     if (sizeof (SMM_CORE_SMI_HANDLER_STRUCTURE) + GET_OCCUPIED_SIZE (SmiHandler->ContextSize, sizeof (UINT64)) > MaxSize - Size) {
 | |
|       *Count = 0;
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     SmiHandlerStruct->Length            = (UINT32)(sizeof (SMM_CORE_SMI_HANDLER_STRUCTURE) + GET_OCCUPIED_SIZE (SmiHandler->ContextSize, sizeof (UINT64)));
 | |
|     SmiHandlerStruct->CallerAddr        = (UINTN)SmiHandler->CallerAddr;
 | |
|     SmiHandlerStruct->Handler           = (UINTN)SmiHandler->Handler;
 | |
|     SmiHandlerStruct->ImageRef          = AddressToImageRef ((UINTN)SmiHandler->Handler);
 | |
|     SmiHandlerStruct->ContextBufferSize = (UINT32)SmiHandler->ContextSize;
 | |
|     if (SmiHandler->ContextSize != 0) {
 | |
|       SmiHandlerStruct->ContextBufferOffset = sizeof (SMM_CORE_SMI_HANDLER_STRUCTURE);
 | |
|       CopyMem ((UINT8 *)SmiHandlerStruct + SmiHandlerStruct->ContextBufferOffset, SmiHandler->Context, SmiHandler->ContextSize);
 | |
|     } else {
 | |
|       SmiHandlerStruct->ContextBufferOffset = 0;
 | |
|     }
 | |
| 
 | |
|     Size            += sizeof (SMM_CORE_SMI_HANDLER_STRUCTURE) + GET_OCCUPIED_SIZE (SmiHandler->ContextSize, sizeof (UINT64));
 | |
|     SmiHandlerStruct = (SMM_CORE_SMI_HANDLER_STRUCTURE *)((UINTN)SmiHandlerStruct + SmiHandlerStruct->Length);
 | |
|     *Count           = *Count + 1;
 | |
|   }
 | |
| 
 | |
|   return Size;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   get all SMI handler database on the SMI entry list.
 | |
| 
 | |
|   @param SmiEntryList     a list of SMI entry.
 | |
|   @param HandlerCategory  The handler category
 | |
|   @param Data             The buffer to hold all SMI handler database
 | |
|   @param ExpectedSize     The expected size of the SMM image database
 | |
| 
 | |
|   @return all SMI database size on the SMI entry list.
 | |
| **/
 | |
| UINTN
 | |
| GetSmmSmiDatabaseData (
 | |
|   IN     LIST_ENTRY  *SmiEntryList,
 | |
|   IN     UINT32      HandlerCategory,
 | |
|   IN OUT VOID        *Data,
 | |
|   IN     UINTN       ExpectedSize
 | |
|   )
 | |
| {
 | |
|   SMM_CORE_SMI_DATABASE_STRUCTURE  *SmiStruct;
 | |
|   LIST_ENTRY                       *ListEntry;
 | |
|   SMI_ENTRY                        *SmiEntry;
 | |
|   UINTN                            Size;
 | |
|   UINTN                            SmiHandlerSize;
 | |
|   UINT32                           SmiHandlerCount;
 | |
| 
 | |
|   SmiStruct = Data;
 | |
|   Size      = 0;
 | |
|   ListEntry = SmiEntryList;
 | |
|   for (ListEntry = ListEntry->ForwardLink;
 | |
|        ListEntry != SmiEntryList;
 | |
|        ListEntry = ListEntry->ForwardLink)
 | |
|   {
 | |
|     SmiEntry = CR (ListEntry, SMI_ENTRY, AllEntries, SMI_ENTRY_SIGNATURE);
 | |
|     if (Size >= ExpectedSize) {
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     if (sizeof (SMM_CORE_SMI_DATABASE_STRUCTURE) > ExpectedSize - Size) {
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     SmiStruct->Header.Signature = SMM_CORE_SMI_DATABASE_SIGNATURE;
 | |
|     SmiStruct->Header.Length    = sizeof (SMM_CORE_SMI_DATABASE_STRUCTURE);
 | |
|     SmiStruct->Header.Revision  = SMM_CORE_SMI_DATABASE_REVISION;
 | |
|     SmiStruct->HandlerCategory  = HandlerCategory;
 | |
|     CopyGuid (&SmiStruct->HandlerType, &SmiEntry->HandlerType);
 | |
|     Size                     += sizeof (SMM_CORE_SMI_DATABASE_STRUCTURE);
 | |
|     SmiHandlerSize            = GetSmmSmiHandlerDataOnSmiEntry (SmiEntry, (UINT8 *)SmiStruct + SmiStruct->Header.Length, ExpectedSize - Size, &SmiHandlerCount);
 | |
|     SmiStruct->HandlerCount   = SmiHandlerCount;
 | |
|     Size                     += SmiHandlerSize;
 | |
|     SmiStruct->Header.Length += (UINT32)SmiHandlerSize;
 | |
|     SmiStruct                 = (VOID *)((UINTN)SmiStruct + SmiStruct->Header.Length);
 | |
|   }
 | |
| 
 | |
|   if (ExpectedSize != Size) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   return Size;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get SMI handler profile database.
 | |
| 
 | |
|   @param Data the buffer to hold SMI handler profile database
 | |
| 
 | |
|   @retval EFI_SUCCESS            the database is got.
 | |
|   @retval EFI_INVALID_PARAMETER  the database size mismatch.
 | |
| **/
 | |
| EFI_STATUS
 | |
| GetSmiHandlerProfileDatabaseData (
 | |
|   IN OUT VOID  *Data
 | |
|   )
 | |
| {
 | |
|   UINTN  SmmImageDatabaseSize;
 | |
|   UINTN  SmmSmiDatabaseSize;
 | |
|   UINTN  SmmRootSmiDatabaseSize;
 | |
|   UINTN  SmmHardwareSmiDatabaseSize;
 | |
| 
 | |
|   DEBUG ((DEBUG_VERBOSE, "GetSmiHandlerProfileDatabaseData\n"));
 | |
|   SmmImageDatabaseSize = GetSmmImageDatabaseData (Data, mSmmImageDatabaseSize);
 | |
|   if (SmmImageDatabaseSize != mSmmImageDatabaseSize) {
 | |
|     DEBUG ((DEBUG_ERROR, "GetSmiHandlerProfileDatabaseData - SmmImageDatabaseSize mismatch!\n"));
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   SmmRootSmiDatabaseSize = GetSmmSmiDatabaseData (mSmmCoreRootSmiEntryList, SmmCoreSmiHandlerCategoryRootHandler, (UINT8 *)Data + SmmImageDatabaseSize, mSmmRootSmiDatabaseSize);
 | |
|   if (SmmRootSmiDatabaseSize != mSmmRootSmiDatabaseSize) {
 | |
|     DEBUG ((DEBUG_ERROR, "GetSmiHandlerProfileDatabaseData - SmmRootSmiDatabaseSize mismatch!\n"));
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   SmmSmiDatabaseSize = GetSmmSmiDatabaseData (mSmmCoreSmiEntryList, SmmCoreSmiHandlerCategoryGuidHandler, (UINT8 *)Data + SmmImageDatabaseSize + mSmmRootSmiDatabaseSize, mSmmSmiDatabaseSize);
 | |
|   if (SmmSmiDatabaseSize != mSmmSmiDatabaseSize) {
 | |
|     DEBUG ((DEBUG_ERROR, "GetSmiHandlerProfileDatabaseData - SmmSmiDatabaseSize mismatch!\n"));
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   SmmHardwareSmiDatabaseSize = GetSmmSmiDatabaseData (mSmmCoreHardwareSmiEntryList, SmmCoreSmiHandlerCategoryHardwareHandler, (UINT8 *)Data + SmmImageDatabaseSize + SmmRootSmiDatabaseSize + SmmSmiDatabaseSize, mSmmHardwareSmiDatabaseSize);
 | |
|   if (SmmHardwareSmiDatabaseSize != mSmmHardwareSmiDatabaseSize) {
 | |
|     DEBUG ((DEBUG_ERROR, "GetSmiHandlerProfileDatabaseData - SmmHardwareSmiDatabaseSize mismatch!\n"));
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Build SMI handler profile database.
 | |
| **/
 | |
| VOID
 | |
| BuildSmiHandlerProfileDatabase (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
| 
 | |
|   mSmiHandlerProfileDatabaseSize = GetSmiHandlerProfileDatabaseSize ();
 | |
|   mSmiHandlerProfileDatabase     = AllocatePool (mSmiHandlerProfileDatabaseSize);
 | |
|   if (mSmiHandlerProfileDatabase == NULL) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Status = GetSmiHandlerProfileDatabaseData (mSmiHandlerProfileDatabase);
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     FreePool (mSmiHandlerProfileDatabase);
 | |
|     mSmiHandlerProfileDatabase = NULL;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Copy SMI handler profile data.
 | |
| 
 | |
|   @param DataBuffer  The buffer to hold SMI handler profile data.
 | |
|   @param DataSize    On input, data buffer size.
 | |
|                      On output, actual data buffer size copied.
 | |
|   @param DataOffset  On input, data buffer offset to copy.
 | |
|                      On output, next time data buffer offset to copy.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmiHandlerProfileCopyData (
 | |
|   OUT VOID       *DataBuffer,
 | |
|   IN OUT UINT64  *DataSize,
 | |
|   IN OUT UINT64  *DataOffset
 | |
|   )
 | |
| {
 | |
|   if (*DataOffset >= mSmiHandlerProfileDatabaseSize) {
 | |
|     *DataOffset = mSmiHandlerProfileDatabaseSize;
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (mSmiHandlerProfileDatabaseSize - *DataOffset < *DataSize) {
 | |
|     *DataSize = mSmiHandlerProfileDatabaseSize - *DataOffset;
 | |
|   }
 | |
| 
 | |
|   CopyMem (
 | |
|     DataBuffer,
 | |
|     (UINT8 *)mSmiHandlerProfileDatabase + *DataOffset,
 | |
|     (UINTN)*DataSize
 | |
|     );
 | |
|   *DataOffset = *DataOffset + *DataSize;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMI handler profile handler to get info.
 | |
| 
 | |
|   @param SmiHandlerProfileParameterGetInfo The parameter of SMI handler profile get info.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmiHandlerProfileHandlerGetInfo (
 | |
|   IN SMI_HANDLER_PROFILE_PARAMETER_GET_INFO  *SmiHandlerProfileParameterGetInfo
 | |
|   )
 | |
| {
 | |
|   BOOLEAN  SmiHandlerProfileRecordingStatus;
 | |
| 
 | |
|   SmiHandlerProfileRecordingStatus  = mSmiHandlerProfileRecordingStatus;
 | |
|   mSmiHandlerProfileRecordingStatus = FALSE;
 | |
| 
 | |
|   SmiHandlerProfileParameterGetInfo->DataSize            = mSmiHandlerProfileDatabaseSize;
 | |
|   SmiHandlerProfileParameterGetInfo->Header.ReturnStatus = 0;
 | |
| 
 | |
|   mSmiHandlerProfileRecordingStatus = SmiHandlerProfileRecordingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMI handler profile handler to get data by offset.
 | |
| 
 | |
|   @param SmiHandlerProfileParameterGetDataByOffset   The parameter of SMI handler profile get data by offset.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmiHandlerProfileHandlerGetDataByOffset (
 | |
|   IN SMI_HANDLER_PROFILE_PARAMETER_GET_DATA_BY_OFFSET  *SmiHandlerProfileParameterGetDataByOffset
 | |
|   )
 | |
| {
 | |
|   SMI_HANDLER_PROFILE_PARAMETER_GET_DATA_BY_OFFSET  SmiHandlerProfileGetDataByOffset;
 | |
|   BOOLEAN                                           SmiHandlerProfileRecordingStatus;
 | |
| 
 | |
|   SmiHandlerProfileRecordingStatus  = mSmiHandlerProfileRecordingStatus;
 | |
|   mSmiHandlerProfileRecordingStatus = FALSE;
 | |
| 
 | |
|   CopyMem (&SmiHandlerProfileGetDataByOffset, SmiHandlerProfileParameterGetDataByOffset, sizeof (SmiHandlerProfileGetDataByOffset));
 | |
| 
 | |
|   //
 | |
|   // Sanity check
 | |
|   //
 | |
|   if (!SmmIsBufferOutsideSmmValid ((UINTN)SmiHandlerProfileGetDataByOffset.DataBuffer, (UINTN)SmiHandlerProfileGetDataByOffset.DataSize)) {
 | |
|     DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandlerGetDataByOffset: SMI handler profile get data in SMRAM or overflow!\n"));
 | |
|     SmiHandlerProfileParameterGetDataByOffset->Header.ReturnStatus = (UINT64)(INT64)(INTN)EFI_ACCESS_DENIED;
 | |
|     goto Done;
 | |
|   }
 | |
| 
 | |
|   SmiHandlerProfileCopyData ((VOID *)(UINTN)SmiHandlerProfileGetDataByOffset.DataBuffer, &SmiHandlerProfileGetDataByOffset.DataSize, &SmiHandlerProfileGetDataByOffset.DataOffset);
 | |
|   CopyMem (SmiHandlerProfileParameterGetDataByOffset, &SmiHandlerProfileGetDataByOffset, sizeof (SmiHandlerProfileGetDataByOffset));
 | |
|   SmiHandlerProfileParameterGetDataByOffset->Header.ReturnStatus = 0;
 | |
| 
 | |
| Done:
 | |
|   mSmiHandlerProfileRecordingStatus = SmiHandlerProfileRecordingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dispatch function for a Software SMI handler.
 | |
| 
 | |
|   Caution: This function may receive untrusted input.
 | |
|   Communicate buffer and buffer size are external input, so this function will do basic validation.
 | |
| 
 | |
|   @param DispatchHandle  The unique handle assigned to this handler by SmiHandlerRegister().
 | |
|   @param Context         Points to an optional handler context which was specified when the
 | |
|                          handler was registered.
 | |
|   @param CommBuffer      A pointer to a collection of data in memory that will
 | |
|                          be conveyed from a non-SMM environment into an SMM environment.
 | |
|   @param CommBufferSize  The size of the CommBuffer.
 | |
| 
 | |
|   @retval EFI_SUCCESS Command is handled successfully.
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmiHandlerProfileHandler (
 | |
|   IN EFI_HANDLE  DispatchHandle,
 | |
|   IN CONST VOID  *Context         OPTIONAL,
 | |
|   IN OUT VOID    *CommBuffer      OPTIONAL,
 | |
|   IN OUT UINTN   *CommBufferSize  OPTIONAL
 | |
|   )
 | |
| {
 | |
|   SMI_HANDLER_PROFILE_PARAMETER_HEADER  *SmiHandlerProfileParameterHeader;
 | |
|   UINTN                                 TempCommBufferSize;
 | |
| 
 | |
|   DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandler Enter\n"));
 | |
| 
 | |
|   if (mSmiHandlerProfileDatabase == NULL) {
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // If input is invalid, stop processing this SMI
 | |
|   //
 | |
|   if ((CommBuffer == NULL) || (CommBufferSize == NULL)) {
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   TempCommBufferSize = *CommBufferSize;
 | |
| 
 | |
|   if (TempCommBufferSize < sizeof (SMI_HANDLER_PROFILE_PARAMETER_HEADER)) {
 | |
|     DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   if (!SmmIsBufferOutsideSmmValid ((UINTN)CommBuffer, TempCommBufferSize)) {
 | |
|     DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandler: SMM communication buffer in SMRAM or overflow!\n"));
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   SmiHandlerProfileParameterHeader               = (SMI_HANDLER_PROFILE_PARAMETER_HEADER *)((UINTN)CommBuffer);
 | |
|   SmiHandlerProfileParameterHeader->ReturnStatus = (UINT64)-1;
 | |
| 
 | |
|   switch (SmiHandlerProfileParameterHeader->Command) {
 | |
|     case SMI_HANDLER_PROFILE_COMMAND_GET_INFO:
 | |
|       DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandlerGetInfo\n"));
 | |
|       if (TempCommBufferSize != sizeof (SMI_HANDLER_PROFILE_PARAMETER_GET_INFO)) {
 | |
|         DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|         return EFI_SUCCESS;
 | |
|       }
 | |
| 
 | |
|       SmiHandlerProfileHandlerGetInfo ((SMI_HANDLER_PROFILE_PARAMETER_GET_INFO *)(UINTN)CommBuffer);
 | |
|       break;
 | |
|     case SMI_HANDLER_PROFILE_COMMAND_GET_DATA_BY_OFFSET:
 | |
|       DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandlerGetDataByOffset\n"));
 | |
|       if (TempCommBufferSize != sizeof (SMI_HANDLER_PROFILE_PARAMETER_GET_DATA_BY_OFFSET)) {
 | |
|         DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|         return EFI_SUCCESS;
 | |
|       }
 | |
| 
 | |
|       SmiHandlerProfileHandlerGetDataByOffset ((SMI_HANDLER_PROFILE_PARAMETER_GET_DATA_BY_OFFSET *)(UINTN)CommBuffer);
 | |
|       break;
 | |
|     default:
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   DEBUG ((DEBUG_ERROR, "SmiHandlerProfileHandler Exit\n"));
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Register SMI handler profile handler.
 | |
| **/
 | |
| VOID
 | |
| RegisterSmiHandlerProfileHandler (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   EFI_HANDLE  DispatchHandle;
 | |
| 
 | |
|   Status = gSmst->SmiHandlerRegister (
 | |
|                     SmiHandlerProfileHandler,
 | |
|                     &gSmiHandlerProfileGuid,
 | |
|                     &DispatchHandle
 | |
|                     );
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Finds the SMI entry for the requested handler type.
 | |
| 
 | |
|   @param  HandlerType            The type of the interrupt
 | |
|   @param  Create                 Create a new entry if not found
 | |
| 
 | |
|   @return SMI entry
 | |
| **/
 | |
| SMI_ENTRY  *
 | |
| SmmCoreFindHardwareSmiEntry (
 | |
|   IN EFI_GUID  *HandlerType,
 | |
|   IN BOOLEAN   Create
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY  *Link;
 | |
|   SMI_ENTRY   *Item;
 | |
|   SMI_ENTRY   *SmiEntry;
 | |
| 
 | |
|   //
 | |
|   // Search the SMI entry list for the matching GUID
 | |
|   //
 | |
|   SmiEntry = NULL;
 | |
|   for (Link = mHardwareSmiEntryList.ForwardLink;
 | |
|        Link != &mHardwareSmiEntryList;
 | |
|        Link = Link->ForwardLink)
 | |
|   {
 | |
|     Item = CR (Link, SMI_ENTRY, AllEntries, SMI_ENTRY_SIGNATURE);
 | |
|     if (CompareGuid (&Item->HandlerType, HandlerType)) {
 | |
|       //
 | |
|       // This is the SMI entry
 | |
|       //
 | |
|       SmiEntry = Item;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // If the protocol entry was not found and Create is TRUE, then
 | |
|   // allocate a new entry
 | |
|   //
 | |
|   if ((SmiEntry == NULL) && Create) {
 | |
|     SmiEntry = AllocatePool (sizeof (SMI_ENTRY));
 | |
|     if (SmiEntry != NULL) {
 | |
|       //
 | |
|       // Initialize new SMI entry structure
 | |
|       //
 | |
|       SmiEntry->Signature = SMI_ENTRY_SIGNATURE;
 | |
|       CopyGuid ((VOID *)&SmiEntry->HandlerType, HandlerType);
 | |
|       InitializeListHead (&SmiEntry->SmiHandlers);
 | |
| 
 | |
|       //
 | |
|       // Add it to SMI entry list
 | |
|       //
 | |
|       InsertTailList (&mHardwareSmiEntryList, &SmiEntry->AllEntries);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return SmiEntry;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Convert EFI_SMM_USB_REGISTER_CONTEXT to SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT.
 | |
| 
 | |
|   @param UsbContext                   A pointer to EFI_SMM_USB_REGISTER_CONTEXT
 | |
|   @param UsbContextSize               The size of EFI_SMM_USB_REGISTER_CONTEXT in bytes
 | |
|   @param SmiHandlerUsbContextSize     The size of SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT in bytes
 | |
| 
 | |
|   @return SmiHandlerUsbContext  A pointer to SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT
 | |
| **/
 | |
| SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT *
 | |
| ConvertSmiHandlerUsbContext (
 | |
|   IN EFI_SMM_USB_REGISTER_CONTEXT  *UsbContext,
 | |
|   IN UINTN                         UsbContextSize,
 | |
|   OUT UINTN                        *SmiHandlerUsbContextSize
 | |
|   )
 | |
| {
 | |
|   UINTN                                     DevicePathSize;
 | |
|   SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT  *SmiHandlerUsbContext;
 | |
| 
 | |
|   ASSERT (UsbContextSize == sizeof (EFI_SMM_USB_REGISTER_CONTEXT));
 | |
| 
 | |
|   DevicePathSize       = GetDevicePathSize (UsbContext->Device);
 | |
|   SmiHandlerUsbContext = AllocatePool (sizeof (SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT) + DevicePathSize);
 | |
|   if (SmiHandlerUsbContext == NULL) {
 | |
|     *SmiHandlerUsbContextSize = 0;
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   SmiHandlerUsbContext->Type           = UsbContext->Type;
 | |
|   SmiHandlerUsbContext->DevicePathSize = (UINT32)DevicePathSize;
 | |
|   CopyMem (SmiHandlerUsbContext + 1, UsbContext->Device, DevicePathSize);
 | |
|   *SmiHandlerUsbContextSize = sizeof (SMI_HANDLER_PROFILE_USB_REGISTER_CONTEXT) + DevicePathSize;
 | |
|   return SmiHandlerUsbContext;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Convert EFI_SMM_SW_REGISTER_CONTEXT to SMI_HANDLER_PROFILE_SW_REGISTER_CONTEXT.
 | |
| 
 | |
|   @param SwContext                    A pointer to EFI_SMM_SW_REGISTER_CONTEXT
 | |
|   @param SwContextSize                The size of EFI_SMM_SW_REGISTER_CONTEXT in bytes
 | |
|   @param SmiHandlerSwContextSize      The size of SMI_HANDLER_PROFILE_SW_REGISTER_CONTEXT in bytes
 | |
| 
 | |
|   @return SmiHandlerSwContext   A pointer to SMI_HANDLER_PROFILE_SW_REGISTER_CONTEXT
 | |
| **/
 | |
| SMI_HANDLER_PROFILE_SW_REGISTER_CONTEXT *
 | |
| ConvertSmiHandlerSwContext (
 | |
|   IN EFI_SMM_SW_REGISTER_CONTEXT  *SwContext,
 | |
|   IN UINTN                        SwContextSize,
 | |
|   OUT UINTN                       *SmiHandlerSwContextSize
 | |
|   )
 | |
| {
 | |
|   SMI_HANDLER_PROFILE_SW_REGISTER_CONTEXT  *SmiHandlerSwContext;
 | |
| 
 | |
|   ASSERT (SwContextSize == sizeof (EFI_SMM_SW_REGISTER_CONTEXT));
 | |
| 
 | |
|   SmiHandlerSwContext = AllocatePool (sizeof (SMI_HANDLER_PROFILE_SW_REGISTER_CONTEXT));
 | |
|   if (SmiHandlerSwContext == NULL) {
 | |
|     *SmiHandlerSwContextSize = 0;
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   SmiHandlerSwContext->SwSmiInputValue = SwContext->SwSmiInputValue;
 | |
|   *SmiHandlerSwContextSize             = sizeof (SMI_HANDLER_PROFILE_SW_REGISTER_CONTEXT);
 | |
|   return SmiHandlerSwContext;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   This function is called by SmmChildDispatcher module to report
 | |
|   a new SMI handler is registered, to SmmCore.
 | |
| 
 | |
|   @param This            The protocol instance
 | |
|   @param HandlerGuid     The GUID to identify the type of the handler.
 | |
|                          For the SmmChildDispatch protocol, the HandlerGuid
 | |
|                          must be the GUID of SmmChildDispatch protocol.
 | |
|   @param Handler         The SMI handler.
 | |
|   @param CallerAddress   The address of the module who registers the SMI handler.
 | |
|   @param Context         The context of the SMI handler.
 | |
|                          For the SmmChildDispatch protocol, the Context
 | |
|                          must match the one defined for SmmChildDispatch protocol.
 | |
|   @param ContextSize     The size of the context in bytes.
 | |
|                          For the SmmChildDispatch protocol, the Context
 | |
|                          must match the one defined for SmmChildDispatch protocol.
 | |
| 
 | |
|   @retval EFI_SUCCESS           The information is recorded.
 | |
|   @retval EFI_OUT_OF_RESOURCES  There is no enough resource to record the information.
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmiHandlerProfileRegisterHandler (
 | |
|   IN SMI_HANDLER_PROFILE_PROTOCOL  *This,
 | |
|   IN EFI_GUID                      *HandlerGuid,
 | |
|   IN EFI_SMM_HANDLER_ENTRY_POINT2  Handler,
 | |
|   IN PHYSICAL_ADDRESS              CallerAddress,
 | |
|   IN VOID                          *Context  OPTIONAL,
 | |
|   IN UINTN                         ContextSize OPTIONAL
 | |
|   )
 | |
| {
 | |
|   SMI_HANDLER  *SmiHandler;
 | |
|   SMI_ENTRY    *SmiEntry;
 | |
|   LIST_ENTRY   *List;
 | |
| 
 | |
|   if (((ContextSize == 0) && (Context != NULL)) ||
 | |
|       ((ContextSize != 0) && (Context == NULL)))
 | |
|   {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   SmiHandler = AllocateZeroPool (sizeof (SMI_HANDLER));
 | |
|   if (SmiHandler == NULL) {
 | |
|     return EFI_OUT_OF_RESOURCES;
 | |
|   }
 | |
| 
 | |
|   SmiHandler->Signature   = SMI_HANDLER_SIGNATURE;
 | |
|   SmiHandler->Handler     = Handler;
 | |
|   SmiHandler->CallerAddr  = (UINTN)CallerAddress;
 | |
|   SmiHandler->Context     = Context;
 | |
|   SmiHandler->ContextSize = ContextSize;
 | |
| 
 | |
|   if (Context != NULL) {
 | |
|     if (CompareGuid (HandlerGuid, &gEfiSmmUsbDispatch2ProtocolGuid)) {
 | |
|       SmiHandler->Context = ConvertSmiHandlerUsbContext (Context, ContextSize, &SmiHandler->ContextSize);
 | |
|     } else if (CompareGuid (HandlerGuid, &gEfiSmmSwDispatch2ProtocolGuid)) {
 | |
|       SmiHandler->Context = ConvertSmiHandlerSwContext (Context, ContextSize, &SmiHandler->ContextSize);
 | |
|     } else {
 | |
|       SmiHandler->Context = AllocateCopyPool (ContextSize, Context);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (SmiHandler->Context == NULL) {
 | |
|     SmiHandler->ContextSize = 0;
 | |
|   }
 | |
| 
 | |
|   SmiEntry = SmmCoreFindHardwareSmiEntry (HandlerGuid, TRUE);
 | |
|   if (SmiEntry == NULL) {
 | |
|     if (SmiHandler->Context != NULL) {
 | |
|       FreePool (SmiHandler->Context);
 | |
|     }
 | |
| 
 | |
|     FreePool (SmiHandler);
 | |
|     return EFI_OUT_OF_RESOURCES;
 | |
|   }
 | |
| 
 | |
|   List = &SmiEntry->SmiHandlers;
 | |
| 
 | |
|   SmiHandler->SmiEntry = SmiEntry;
 | |
|   InsertTailList (List, &SmiHandler->Link);
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   This function is called by SmmChildDispatcher module to report
 | |
|   an existing SMI handler is unregistered, to SmmCore.
 | |
| 
 | |
|   @param This            The protocol instance
 | |
|   @param HandlerGuid     The GUID to identify the type of the handler.
 | |
|                          For the SmmChildDispatch protocol, the HandlerGuid
 | |
|                          must be the GUID of SmmChildDispatch protocol.
 | |
|   @param Handler         The SMI handler.
 | |
|   @param Context         The context of the SMI handler.
 | |
|                          If it is NOT NULL, it will be used to check what is registered.
 | |
|   @param ContextSize     The size of the context in bytes.
 | |
|                          If Context is NOT NULL, it will be used to check what is registered.
 | |
| 
 | |
|   @retval EFI_SUCCESS           The original record is removed.
 | |
|   @retval EFI_NOT_FOUND         There is no record for the HandlerGuid and handler.
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmiHandlerProfileUnregisterHandler (
 | |
|   IN SMI_HANDLER_PROFILE_PROTOCOL  *This,
 | |
|   IN EFI_GUID                      *HandlerGuid,
 | |
|   IN EFI_SMM_HANDLER_ENTRY_POINT2  Handler,
 | |
|   IN VOID                          *Context  OPTIONAL,
 | |
|   IN UINTN                         ContextSize OPTIONAL
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY   *Link;
 | |
|   LIST_ENTRY   *Head;
 | |
|   SMI_HANDLER  *SmiHandler;
 | |
|   SMI_ENTRY    *SmiEntry;
 | |
|   SMI_HANDLER  *TargetSmiHandler;
 | |
|   VOID         *SearchContext;
 | |
|   UINTN        SearchContextSize;
 | |
| 
 | |
|   if (((ContextSize == 0) && (Context != NULL)) ||
 | |
|       ((ContextSize != 0) && (Context == NULL)))
 | |
|   {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   SmiEntry = SmmCoreFindHardwareSmiEntry (HandlerGuid, FALSE);
 | |
|   if (SmiEntry == NULL) {
 | |
|     return EFI_NOT_FOUND;
 | |
|   }
 | |
| 
 | |
|   SearchContext     = Context;
 | |
|   SearchContextSize = ContextSize;
 | |
|   if (Context != NULL) {
 | |
|     if (CompareGuid (HandlerGuid, &gEfiSmmUsbDispatch2ProtocolGuid)) {
 | |
|       SearchContext = ConvertSmiHandlerUsbContext (Context, ContextSize, &SearchContextSize);
 | |
|     } else if (CompareGuid (HandlerGuid, &gEfiSmmSwDispatch2ProtocolGuid)) {
 | |
|       SearchContext = ConvertSmiHandlerSwContext (Context, ContextSize, &SearchContextSize);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   TargetSmiHandler = NULL;
 | |
|   Head             = &SmiEntry->SmiHandlers;
 | |
|   for (Link = Head->ForwardLink; Link != Head; Link = Link->ForwardLink) {
 | |
|     SmiHandler = CR (Link, SMI_HANDLER, Link, SMI_HANDLER_SIGNATURE);
 | |
|     if (SmiHandler->Handler == Handler) {
 | |
|       if ((SearchContext == NULL) ||
 | |
|           ((SearchContextSize == SmiHandler->ContextSize) && (CompareMem (SearchContext, SmiHandler->Context, SearchContextSize) == 0)))
 | |
|       {
 | |
|         TargetSmiHandler = SmiHandler;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (SearchContext != NULL) {
 | |
|     if (CompareGuid (HandlerGuid, &gEfiSmmUsbDispatch2ProtocolGuid)) {
 | |
|       FreePool (SearchContext);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (TargetSmiHandler == NULL) {
 | |
|     return EFI_NOT_FOUND;
 | |
|   }
 | |
| 
 | |
|   SmiHandler = TargetSmiHandler;
 | |
| 
 | |
|   RemoveEntryList (&SmiHandler->Link);
 | |
|   if (SmiHandler->Context != NULL) {
 | |
|     FreePool (SmiHandler->Context);
 | |
|   }
 | |
| 
 | |
|   FreePool (SmiHandler);
 | |
| 
 | |
|   if (IsListEmpty (&SmiEntry->SmiHandlers)) {
 | |
|     RemoveEntryList (&SmiEntry->AllEntries);
 | |
|     FreePool (SmiEntry);
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Initialize SmiHandler profile feature.
 | |
| **/
 | |
| VOID
 | |
| SmmCoreInitializeSmiHandlerProfile (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   VOID        *Registration;
 | |
|   EFI_HANDLE  Handle;
 | |
| 
 | |
|   if ((PcdGet8 (PcdSmiHandlerProfilePropertyMask) & 0x1) != 0) {
 | |
|     InsertTailList (&mRootSmiEntryList, &mRootSmiEntry.AllEntries);
 | |
| 
 | |
|     Status = gSmst->SmmRegisterProtocolNotify (
 | |
|                       &gEfiSmmReadyToLockProtocolGuid,
 | |
|                       SmmReadyToLockInSmiHandlerProfile,
 | |
|                       &Registration
 | |
|                       );
 | |
|     ASSERT_EFI_ERROR (Status);
 | |
| 
 | |
|     Handle = NULL;
 | |
|     Status = gSmst->SmmInstallProtocolInterface (
 | |
|                       &Handle,
 | |
|                       &gSmiHandlerProfileGuid,
 | |
|                       EFI_NATIVE_INTERFACE,
 | |
|                       &mSmiHandlerProfile
 | |
|                       );
 | |
|     ASSERT_EFI_ERROR (Status);
 | |
|   }
 | |
| }
 |