Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Star Zeng <star.zeng@intel.com> Reviewed-by: Liming Gao <liming.gao@intel.com> git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17461 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			1934 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1934 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   Support routines for SMRAM profile.
 | |
| 
 | |
|   Copyright (c) 2014 - 2015, 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 "PiSmmCore.h"
 | |
| 
 | |
| #define IS_SMRAM_PROFILE_ENABLED ((PcdGet8 (PcdMemoryProfilePropertyMask) & BIT1) != 0)
 | |
| 
 | |
| typedef struct {
 | |
|   UINT32                        Signature;
 | |
|   MEMORY_PROFILE_CONTEXT        Context;
 | |
|   LIST_ENTRY                    *DriverInfoList;
 | |
| } MEMORY_PROFILE_CONTEXT_DATA;
 | |
| 
 | |
| typedef struct {
 | |
|   UINT32                        Signature;
 | |
|   MEMORY_PROFILE_DRIVER_INFO    DriverInfo;
 | |
|   LIST_ENTRY                    *AllocInfoList;
 | |
|   LIST_ENTRY                    Link;
 | |
| } MEMORY_PROFILE_DRIVER_INFO_DATA;
 | |
| 
 | |
| typedef struct {
 | |
|   UINT32                        Signature;
 | |
|   MEMORY_PROFILE_ALLOC_INFO     AllocInfo;
 | |
|   LIST_ENTRY                    Link;
 | |
| } MEMORY_PROFILE_ALLOC_INFO_DATA;
 | |
| 
 | |
| //
 | |
| // When free memory less than 4 pages, dump it.
 | |
| //
 | |
| #define SMRAM_INFO_DUMP_PAGE_THRESHOLD  4
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED MEMORY_PROFILE_FREE_MEMORY mSmramFreeMemory = {
 | |
|   {
 | |
|     MEMORY_PROFILE_FREE_MEMORY_SIGNATURE,
 | |
|     sizeof (MEMORY_PROFILE_FREE_MEMORY),
 | |
|     MEMORY_PROFILE_FREE_MEMORY_REVISION
 | |
|   },
 | |
|   0,
 | |
|   0
 | |
| };
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED LIST_ENTRY  mImageQueue = INITIALIZE_LIST_HEAD_VARIABLE (mImageQueue);
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED MEMORY_PROFILE_CONTEXT_DATA mSmramProfileContext = {
 | |
|   MEMORY_PROFILE_CONTEXT_SIGNATURE,
 | |
|   {
 | |
|     {
 | |
|       MEMORY_PROFILE_CONTEXT_SIGNATURE,
 | |
|       sizeof (MEMORY_PROFILE_CONTEXT),
 | |
|       MEMORY_PROFILE_CONTEXT_REVISION
 | |
|     },
 | |
|     0,
 | |
|     0,
 | |
|     {0},
 | |
|     {0},
 | |
|     0,
 | |
|     0,
 | |
|     0
 | |
|   },
 | |
|   &mImageQueue,
 | |
| };
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED MEMORY_PROFILE_CONTEXT_DATA *mSmramProfileContextPtr;
 | |
| 
 | |
| BOOLEAN mSmramReadyToLock;
 | |
| BOOLEAN mSmramProfileRecordingStatus = FALSE;
 | |
| 
 | |
| /**
 | |
|   Return SMRAM profile context.
 | |
| 
 | |
|   @return SMRAM profile context.
 | |
| 
 | |
| **/
 | |
| MEMORY_PROFILE_CONTEXT_DATA *
 | |
| GetSmramProfileContext (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   return mSmramProfileContextPtr;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Retrieves the magic value from the PE/COFF header.
 | |
| 
 | |
|   @param Hdr    The buffer in which to return the PE32, PE32+, or TE header.
 | |
| 
 | |
|   @return EFI_IMAGE_NT_OPTIONAL_HDR32_MAGIC - Image is PE32
 | |
|   @return EFI_IMAGE_NT_OPTIONAL_HDR64_MAGIC - Image is PE32+
 | |
| 
 | |
| **/
 | |
| UINT16
 | |
| InternalPeCoffGetPeHeaderMagicValue (
 | |
|   IN  EFI_IMAGE_OPTIONAL_HEADER_PTR_UNION  Hdr
 | |
|   )
 | |
| {
 | |
|   //
 | |
|   // NOTE: Some versions of Linux ELILO for Itanium have an incorrect magic value
 | |
|   //       in the PE/COFF Header.  If the MachineType is Itanium(IA64) and the
 | |
|   //       Magic value in the OptionalHeader is  EFI_IMAGE_NT_OPTIONAL_HDR32_MAGIC
 | |
|   //       then override the returned value to EFI_IMAGE_NT_OPTIONAL_HDR64_MAGIC
 | |
|   //
 | |
|   if (Hdr.Pe32->FileHeader.Machine == IMAGE_FILE_MACHINE_IA64 && Hdr.Pe32->OptionalHeader.Magic == EFI_IMAGE_NT_OPTIONAL_HDR32_MAGIC) {
 | |
|     return EFI_IMAGE_NT_OPTIONAL_HDR64_MAGIC;
 | |
|   }
 | |
|   //
 | |
|   // Return the magic value from the PC/COFF Optional Header
 | |
|   //
 | |
|   return Hdr.Pe32->OptionalHeader.Magic;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Retrieves and returns the Subsystem of a PE/COFF image that has been loaded into system memory.
 | |
|   If Pe32Data is NULL, then ASSERT().
 | |
| 
 | |
|   @param Pe32Data   The pointer to the PE/COFF image that is loaded in system memory.
 | |
| 
 | |
|   @return The Subsystem of the PE/COFF image.
 | |
| 
 | |
| **/
 | |
| UINT16
 | |
| InternalPeCoffGetSubsystem (
 | |
|   IN VOID  *Pe32Data
 | |
|   )
 | |
| {
 | |
|   EFI_IMAGE_OPTIONAL_HEADER_PTR_UNION  Hdr;
 | |
|   EFI_IMAGE_DOS_HEADER                 *DosHdr;
 | |
|   UINT16                               Magic;
 | |
| 
 | |
|   ASSERT (Pe32Data != NULL);
 | |
| 
 | |
|   DosHdr = (EFI_IMAGE_DOS_HEADER *) Pe32Data;
 | |
|   if (DosHdr->e_magic == EFI_IMAGE_DOS_SIGNATURE) {
 | |
|     //
 | |
|     // DOS image header is present, so read the PE header after the DOS image header.
 | |
|     //
 | |
|     Hdr.Pe32 = (EFI_IMAGE_NT_HEADERS32 *) ((UINTN) Pe32Data + (UINTN) ((DosHdr->e_lfanew) & 0x0ffff));
 | |
|   } else {
 | |
|     //
 | |
|     // DOS image header is not present, so PE header is at the image base.
 | |
|     //
 | |
|     Hdr.Pe32 = (EFI_IMAGE_NT_HEADERS32 *) Pe32Data;
 | |
|   }
 | |
| 
 | |
|   if (Hdr.Te->Signature == EFI_TE_IMAGE_HEADER_SIGNATURE) {
 | |
|     return Hdr.Te->Subsystem;
 | |
|   } else if (Hdr.Pe32->Signature == EFI_IMAGE_NT_SIGNATURE)  {
 | |
|     Magic = InternalPeCoffGetPeHeaderMagicValue (Hdr);
 | |
|     if (Magic == EFI_IMAGE_NT_OPTIONAL_HDR32_MAGIC) {
 | |
|       return Hdr.Pe32->OptionalHeader.Subsystem;
 | |
|     } else if (Magic == EFI_IMAGE_NT_OPTIONAL_HDR64_MAGIC) {
 | |
|       return Hdr.Pe32Plus->OptionalHeader.Subsystem;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return 0x0000;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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
 | |
|   )
 | |
| {
 | |
|   EFI_IMAGE_DOS_HEADER                  *DosHdr;
 | |
|   EFI_IMAGE_OPTIONAL_HEADER_PTR_UNION   Hdr;
 | |
| 
 | |
|   ASSERT (Pe32Data   != NULL);
 | |
|   ASSERT (EntryPoint != NULL);
 | |
| 
 | |
|   DosHdr = (EFI_IMAGE_DOS_HEADER *) Pe32Data;
 | |
|   if (DosHdr->e_magic == EFI_IMAGE_DOS_SIGNATURE) {
 | |
|     //
 | |
|     // DOS image header is present, so read the PE header after the DOS image header.
 | |
|     //
 | |
|     Hdr.Pe32 = (EFI_IMAGE_NT_HEADERS32 *) ((UINTN) Pe32Data + (UINTN) ((DosHdr->e_lfanew) & 0x0ffff));
 | |
|   } else {
 | |
|     //
 | |
|     // DOS image header is not present, so PE header is at the image base.
 | |
|     //
 | |
|     Hdr.Pe32 = (EFI_IMAGE_NT_HEADERS32 *) Pe32Data;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Calculate the entry point relative to the start of the image.
 | |
|   // AddressOfEntryPoint is common for PE32 & PE32+
 | |
|   //
 | |
|   if (Hdr.Te->Signature == EFI_TE_IMAGE_HEADER_SIGNATURE) {
 | |
|     *EntryPoint = (VOID *) ((UINTN) Pe32Data + (UINTN) (Hdr.Te->AddressOfEntryPoint & 0x0ffffffff) + sizeof (EFI_TE_IMAGE_HEADER) - Hdr.Te->StrippedSize);
 | |
|     return RETURN_SUCCESS;
 | |
|   } else if (Hdr.Pe32->Signature == EFI_IMAGE_NT_SIGNATURE) {
 | |
|     *EntryPoint = (VOID *) ((UINTN) Pe32Data + (UINTN) (Hdr.Pe32->OptionalHeader.AddressOfEntryPoint & 0x0ffffffff));
 | |
|     return RETURN_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   return RETURN_UNSUPPORTED;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Build driver info.
 | |
| 
 | |
|   @param ContextData    Memory profile context.
 | |
|   @param FileName       File name of the image.
 | |
|   @param ImageBase      Image base address.
 | |
|   @param ImageSize      Image size.
 | |
|   @param EntryPoint     Entry point of the image.
 | |
|   @param ImageSubsystem Image subsystem of the image.
 | |
| 
 | |
|   @param FileType       File type of the image.
 | |
| 
 | |
|   @return Pointer to memory profile driver info.
 | |
| 
 | |
| **/
 | |
| MEMORY_PROFILE_DRIVER_INFO_DATA *
 | |
| BuildDriverInfo (
 | |
|   IN MEMORY_PROFILE_CONTEXT_DATA    *ContextData,
 | |
|   IN EFI_GUID                       *FileName,
 | |
|   IN PHYSICAL_ADDRESS               ImageBase,
 | |
|   IN UINT64                         ImageSize,
 | |
|   IN PHYSICAL_ADDRESS               EntryPoint,
 | |
|   IN UINT16                         ImageSubsystem,
 | |
|   IN EFI_FV_FILETYPE                FileType
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   MEMORY_PROFILE_DRIVER_INFO        *DriverInfo;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   VOID                              *EntryPointInImage;
 | |
| 
 | |
|   //
 | |
|   // Use SmmInternalAllocatePool() that will not update profile for this AllocatePool action.
 | |
|   //
 | |
|   Status = SmmInternalAllocatePool (
 | |
|              EfiRuntimeServicesData,
 | |
|              sizeof (*DriverInfoData) + sizeof (LIST_ENTRY),
 | |
|              (VOID **) &DriverInfoData
 | |
|              );
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   ZeroMem (DriverInfoData, sizeof (*DriverInfoData));
 | |
| 
 | |
|   DriverInfo = &DriverInfoData->DriverInfo;
 | |
|   DriverInfoData->Signature = MEMORY_PROFILE_DRIVER_INFO_SIGNATURE;
 | |
|   DriverInfo->Header.Signature = MEMORY_PROFILE_DRIVER_INFO_SIGNATURE;
 | |
|   DriverInfo->Header.Length = sizeof (MEMORY_PROFILE_DRIVER_INFO);
 | |
|   DriverInfo->Header.Revision = MEMORY_PROFILE_DRIVER_INFO_REVISION;
 | |
|   if (FileName != NULL) {
 | |
|     CopyMem (&DriverInfo->FileName, FileName, sizeof (EFI_GUID));
 | |
|   }
 | |
|   DriverInfo->ImageBase = ImageBase;
 | |
|   DriverInfo->ImageSize = ImageSize;
 | |
|   DriverInfo->EntryPoint = EntryPoint;
 | |
|   DriverInfo->ImageSubsystem = ImageSubsystem;
 | |
|   if ((EntryPoint != 0) && ((EntryPoint < ImageBase) || (EntryPoint >= (ImageBase + 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 ((VOID *) (UINTN) ImageBase, &EntryPointInImage);
 | |
|     ASSERT_EFI_ERROR (Status);
 | |
|     DriverInfo->ImageBase = ImageBase + EntryPoint - (PHYSICAL_ADDRESS) (UINTN) EntryPointInImage;
 | |
|   }
 | |
|   DriverInfo->FileType = FileType;
 | |
|   DriverInfoData->AllocInfoList = (LIST_ENTRY *) (DriverInfoData + 1);
 | |
|   InitializeListHead (DriverInfoData->AllocInfoList);
 | |
|   DriverInfo->CurrentUsage = 0;
 | |
|   DriverInfo->PeakUsage = 0;
 | |
|   DriverInfo->AllocRecordCount = 0;
 | |
| 
 | |
|   InsertTailList (ContextData->DriverInfoList, &DriverInfoData->Link);
 | |
|   ContextData->Context.ImageCount ++;
 | |
|   ContextData->Context.TotalImageSize += DriverInfo->ImageSize;
 | |
| 
 | |
|   return DriverInfoData;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Register image to DXE.
 | |
| 
 | |
|   @param FileName       File name of the image.
 | |
|   @param ImageBase      Image base address.
 | |
|   @param ImageSize      Image size.
 | |
|   @param FileType       File type of the image.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| RegisterImageToDxe (
 | |
|   IN EFI_GUID                       *FileName,
 | |
|   IN PHYSICAL_ADDRESS               ImageBase,
 | |
|   IN UINT64                         ImageSize,
 | |
|   IN EFI_FV_FILETYPE                FileType
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   EDKII_MEMORY_PROFILE_PROTOCOL     *ProfileProtocol;
 | |
|   MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *FilePath;
 | |
|   UINT8                             TempBuffer[sizeof (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH) + sizeof (EFI_DEVICE_PATH_PROTOCOL)];
 | |
| 
 | |
|   if (IS_SMRAM_PROFILE_ENABLED) {
 | |
| 
 | |
|     FilePath = (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *)TempBuffer;
 | |
|     Status = gBS->LocateProtocol (&gEdkiiMemoryProfileGuid, NULL, (VOID **) &ProfileProtocol);
 | |
|     if (!EFI_ERROR (Status)) {
 | |
|       EfiInitializeFwVolDevicepathNode (FilePath, FileName);
 | |
|       SetDevicePathEndNode (FilePath + 1);
 | |
| 
 | |
|       Status = ProfileProtocol->RegisterImage (
 | |
|                                   ProfileProtocol,
 | |
|                                   (EFI_DEVICE_PATH_PROTOCOL *) FilePath,
 | |
|                                   ImageBase,
 | |
|                                   ImageSize,
 | |
|                                   FileType
 | |
|                                   );
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Unregister image from DXE.
 | |
| 
 | |
|   @param FileName       File name of the image.
 | |
|   @param ImageBase      Image base address.
 | |
|   @param ImageSize      Image size.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| UnregisterImageFromDxe (
 | |
|   IN EFI_GUID                       *FileName,
 | |
|   IN PHYSICAL_ADDRESS               ImageBase,
 | |
|   IN UINT64                         ImageSize
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   EDKII_MEMORY_PROFILE_PROTOCOL     *ProfileProtocol;
 | |
|   MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *FilePath;
 | |
|   UINT8                             TempBuffer[sizeof (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH) + sizeof (EFI_DEVICE_PATH_PROTOCOL)];
 | |
| 
 | |
|   if (IS_SMRAM_PROFILE_ENABLED) {
 | |
| 
 | |
|     FilePath = (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *)TempBuffer;
 | |
|     Status = gBS->LocateProtocol (&gEdkiiMemoryProfileGuid, NULL, (VOID *) &ProfileProtocol);
 | |
|     if (!EFI_ERROR (Status)) {
 | |
|       EfiInitializeFwVolDevicepathNode (FilePath, FileName);
 | |
|       SetDevicePathEndNode (FilePath + 1);
 | |
| 
 | |
|       Status = ProfileProtocol->UnregisterImage (
 | |
|                                   ProfileProtocol,
 | |
|                                   (EFI_DEVICE_PATH_PROTOCOL *) FilePath,
 | |
|                                   ImageBase,
 | |
|                                   ImageSize
 | |
|                                   );
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Register SMM Core to SMRAM profile.
 | |
| 
 | |
|   @param ContextData    SMRAM profile context.
 | |
| 
 | |
|   @retval TRUE          Register success.
 | |
|   @retval FALSE         Register fail.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| RegisterSmmCore (
 | |
|   IN MEMORY_PROFILE_CONTEXT_DATA    *ContextData
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   PHYSICAL_ADDRESS                  ImageBase;
 | |
| 
 | |
|   ASSERT (ContextData != NULL);
 | |
| 
 | |
|   RegisterImageToDxe (
 | |
|     &gEfiCallerIdGuid,
 | |
|     gSmmCorePrivate->PiSmmCoreImageBase,
 | |
|     gSmmCorePrivate->PiSmmCoreImageSize,
 | |
|     EFI_FV_FILETYPE_SMM_CORE
 | |
|     );
 | |
| 
 | |
|   ImageBase = gSmmCorePrivate->PiSmmCoreImageBase;
 | |
|   DriverInfoData = BuildDriverInfo (
 | |
|                      ContextData,
 | |
|                      &gEfiCallerIdGuid,
 | |
|                      ImageBase,
 | |
|                      gSmmCorePrivate->PiSmmCoreImageSize,
 | |
|                      gSmmCorePrivate->PiSmmCoreEntryPoint,
 | |
|                      InternalPeCoffGetSubsystem ((VOID *) (UINTN) ImageBase),
 | |
|                      EFI_FV_FILETYPE_SMM_CORE
 | |
|                      );
 | |
|   if (DriverInfoData == NULL) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Initialize SMRAM profile.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileInit (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT_DATA *SmramProfileContext;
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   SmramProfileContext = GetSmramProfileContext ();
 | |
|   if (SmramProfileContext != NULL) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   mSmramProfileRecordingStatus = TRUE;
 | |
|   mSmramProfileContextPtr = &mSmramProfileContext;
 | |
| 
 | |
|   RegisterSmmCore (&mSmramProfileContext);
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "SmramProfileInit SmramProfileContext - 0x%x\n", &mSmramProfileContext));
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Register SMM image to SMRAM profile.
 | |
| 
 | |
|   @param DriverEntry    SMM image info.
 | |
|   @param RegisterToDxe  Register image to DXE.
 | |
| 
 | |
|   @retval TRUE          Register success.
 | |
|   @retval FALSE         Register fail.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| RegisterSmramProfileImage (
 | |
|   IN EFI_SMM_DRIVER_ENTRY   *DriverEntry,
 | |
|   IN BOOLEAN                RegisterToDxe
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT_DATA       *ContextData;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   if (RegisterToDxe) {
 | |
|     RegisterImageToDxe (
 | |
|       &DriverEntry->FileName,
 | |
|       DriverEntry->ImageBuffer,
 | |
|       EFI_PAGES_TO_SIZE (DriverEntry->NumberOfPage),
 | |
|       EFI_FV_FILETYPE_SMM
 | |
|       );
 | |
|   }
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   DriverInfoData = BuildDriverInfo (
 | |
|                      ContextData,
 | |
|                      &DriverEntry->FileName,
 | |
|                      DriverEntry->ImageBuffer,
 | |
|                      EFI_PAGES_TO_SIZE (DriverEntry->NumberOfPage),
 | |
|                      DriverEntry->ImageEntryPoint,
 | |
|                      InternalPeCoffGetSubsystem ((VOID *) (UINTN) DriverEntry->ImageBuffer),
 | |
|                      EFI_FV_FILETYPE_SMM
 | |
|                      );
 | |
|   if (DriverInfoData == NULL) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Search image from memory profile.
 | |
| 
 | |
|   @param ContextData    Memory profile context.
 | |
|   @param FileName       Image file name.
 | |
|   @param Address        Image Address.
 | |
| 
 | |
|   @return Pointer to memory profile driver info.
 | |
| 
 | |
| **/
 | |
| MEMORY_PROFILE_DRIVER_INFO_DATA *
 | |
| GetMemoryProfileDriverInfoByFileNameAndAddress (
 | |
|   IN MEMORY_PROFILE_CONTEXT_DATA    *ContextData,
 | |
|   IN EFI_GUID                       *FileName,
 | |
|   IN PHYSICAL_ADDRESS               Address
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_DRIVER_INFO        *DriverInfo;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   LIST_ENTRY                        *DriverLink;
 | |
|   LIST_ENTRY                        *DriverInfoList;
 | |
| 
 | |
|   DriverInfoList = ContextData->DriverInfoList;
 | |
| 
 | |
|   for (DriverLink = DriverInfoList->ForwardLink;
 | |
|        DriverLink != DriverInfoList;
 | |
|        DriverLink = DriverLink->ForwardLink) {
 | |
|     DriverInfoData = CR (
 | |
|                        DriverLink,
 | |
|                        MEMORY_PROFILE_DRIVER_INFO_DATA,
 | |
|                        Link,
 | |
|                        MEMORY_PROFILE_DRIVER_INFO_SIGNATURE
 | |
|                        );
 | |
|     DriverInfo = &DriverInfoData->DriverInfo;
 | |
|     if ((CompareGuid (&DriverInfo->FileName, FileName)) &&
 | |
|         (Address >= DriverInfo->ImageBase) &&
 | |
|         (Address < (DriverInfo->ImageBase + DriverInfo->ImageSize))) {
 | |
|       return DriverInfoData;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Search dummy image from SMRAM profile.
 | |
| 
 | |
|   @param ContextData    Memory profile context.
 | |
| 
 | |
|   @return Pointer to memory profile driver info.
 | |
| 
 | |
| **/
 | |
| MEMORY_PROFILE_DRIVER_INFO_DATA *
 | |
| FindDummyImage (
 | |
|   IN MEMORY_PROFILE_CONTEXT_DATA    *ContextData
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   LIST_ENTRY                        *DriverLink;
 | |
|   LIST_ENTRY                        *DriverInfoList;
 | |
| 
 | |
|   DriverInfoList = ContextData->DriverInfoList;
 | |
| 
 | |
|   for (DriverLink = DriverInfoList->ForwardLink;
 | |
|        DriverLink != DriverInfoList;
 | |
|        DriverLink = DriverLink->ForwardLink) {
 | |
|     DriverInfoData = CR (
 | |
|                    DriverLink,
 | |
|                    MEMORY_PROFILE_DRIVER_INFO_DATA,
 | |
|                    Link,
 | |
|                    MEMORY_PROFILE_DRIVER_INFO_SIGNATURE
 | |
|                    );
 | |
|     if (CompareGuid (&gZeroGuid, &DriverInfoData->DriverInfo.FileName)) {
 | |
|       return DriverInfoData;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return BuildDriverInfo (ContextData, &gZeroGuid, 0, 0, 0, 0, 0);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Search image from memory profile.
 | |
|   It will return image, if (Address >= ImageBuffer) AND (Address < ImageBuffer + ImageSize)
 | |
| 
 | |
|   @param ContextData    Memory profile context.
 | |
|   @param Address        Image or Function address.
 | |
| 
 | |
|   @return Pointer to memory profile driver info.
 | |
| 
 | |
| **/
 | |
| MEMORY_PROFILE_DRIVER_INFO_DATA *
 | |
| GetMemoryProfileDriverInfoFromAddress (
 | |
|   IN MEMORY_PROFILE_CONTEXT_DATA    *ContextData,
 | |
|   IN PHYSICAL_ADDRESS               Address
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_DRIVER_INFO        *DriverInfo;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   LIST_ENTRY                        *DriverLink;
 | |
|   LIST_ENTRY                        *DriverInfoList;
 | |
| 
 | |
|   DriverInfoList = ContextData->DriverInfoList;
 | |
| 
 | |
|   for (DriverLink = DriverInfoList->ForwardLink;
 | |
|        DriverLink != DriverInfoList;
 | |
|        DriverLink = DriverLink->ForwardLink) {
 | |
|     DriverInfoData = CR (
 | |
|                        DriverLink,
 | |
|                        MEMORY_PROFILE_DRIVER_INFO_DATA,
 | |
|                        Link,
 | |
|                        MEMORY_PROFILE_DRIVER_INFO_SIGNATURE
 | |
|                        );
 | |
|     DriverInfo = &DriverInfoData->DriverInfo;
 | |
|     if ((Address >= DriverInfo->ImageBase) &&
 | |
|         (Address < (DriverInfo->ImageBase + DriverInfo->ImageSize))) {
 | |
|       return DriverInfoData;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Should never come here.
 | |
|   //
 | |
|   return FindDummyImage (ContextData);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Unregister image from SMRAM profile.
 | |
| 
 | |
|   @param DriverEntry        SMM image info.
 | |
|   @param UnregisterFromDxe  Unregister image from DXE.
 | |
| 
 | |
|   @retval TRUE              Unregister success.
 | |
|   @retval FALSE             Unregister fail.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| UnregisterSmramProfileImage (
 | |
|   IN EFI_SMM_DRIVER_ENTRY  *DriverEntry,
 | |
|   IN BOOLEAN               UnregisterFromDxe
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA       *ContextData;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   EFI_GUID                          *FileName;
 | |
|   PHYSICAL_ADDRESS                  ImageAddress;
 | |
|   VOID                              *EntryPointInImage;
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   if (UnregisterFromDxe) {
 | |
|     UnregisterImageFromDxe (
 | |
|       &DriverEntry->FileName,
 | |
|       DriverEntry->ImageBuffer,
 | |
|       EFI_PAGES_TO_SIZE (DriverEntry->NumberOfPage)
 | |
|       );
 | |
|   }
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   DriverInfoData = NULL;
 | |
|   FileName = &DriverEntry->FileName;
 | |
|   ImageAddress = DriverEntry->ImageBuffer;
 | |
|   if ((DriverEntry->ImageEntryPoint < ImageAddress) || (DriverEntry->ImageEntryPoint >= (ImageAddress + EFI_PAGES_TO_SIZE (DriverEntry->NumberOfPage)))) {
 | |
|     //
 | |
|     // If the EntryPoint is not in the range of image buffer, it should come from emulation environment.
 | |
|     // So patch ImageAddress here to align the EntryPoint.
 | |
|     //
 | |
|     Status = InternalPeCoffGetEntryPoint ((VOID *) (UINTN) ImageAddress, &EntryPointInImage);
 | |
|     ASSERT_EFI_ERROR (Status);
 | |
|     ImageAddress = ImageAddress + (UINTN) DriverEntry->ImageEntryPoint - (UINTN) EntryPointInImage;
 | |
|   }
 | |
|   if (FileName != NULL) {
 | |
|     DriverInfoData = GetMemoryProfileDriverInfoByFileNameAndAddress (ContextData, FileName, ImageAddress);
 | |
|   }
 | |
|   if (DriverInfoData == NULL) {
 | |
|     DriverInfoData = GetMemoryProfileDriverInfoFromAddress (ContextData, ImageAddress);
 | |
|   }
 | |
|   if (DriverInfoData == NULL) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   ContextData->Context.TotalImageSize -= DriverInfoData->DriverInfo.ImageSize;
 | |
| 
 | |
|   DriverInfoData->DriverInfo.ImageBase = 0;
 | |
|   DriverInfoData->DriverInfo.ImageSize = 0;
 | |
| 
 | |
|   if (DriverInfoData->DriverInfo.PeakUsage == 0) {
 | |
|     ContextData->Context.ImageCount --;
 | |
|     RemoveEntryList (&DriverInfoData->Link);
 | |
|     //
 | |
|     // Use SmmInternalFreePool() that will not update profile for this FreePool action.
 | |
|     //
 | |
|     SmmInternalFreePool (DriverInfoData);
 | |
|   }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return if this memory type needs to be recorded into memory profile.
 | |
|   Only need to record EfiRuntimeServicesCode and EfiRuntimeServicesData for SMRAM profile.
 | |
| 
 | |
|   @param MemoryType     Memory type.
 | |
| 
 | |
|   @retval TRUE          This memory type need to be recorded.
 | |
|   @retval FALSE         This memory type need not to be recorded.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| SmmCoreNeedRecordProfile (
 | |
|   IN EFI_MEMORY_TYPE    MemoryType
 | |
|   )
 | |
| {
 | |
|   UINT64 TestBit;
 | |
| 
 | |
|   if (MemoryType != EfiRuntimeServicesCode &&
 | |
|       MemoryType != EfiRuntimeServicesData) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   TestBit = LShiftU64 (1, MemoryType);
 | |
| 
 | |
|   if ((PcdGet64 (PcdMemoryProfileMemoryType) & TestBit) != 0) {
 | |
|     return TRUE;
 | |
|   } else {
 | |
|     return FALSE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Convert EFI memory type to profile memory index. The rule is:
 | |
|   If BIOS memory type (0 ~ EfiMaxMemoryType - 1), ProfileMemoryIndex = MemoryType.
 | |
|   As SMRAM profile is only to record EfiRuntimeServicesCode and EfiRuntimeServicesData,
 | |
|   so return input memory type directly.
 | |
| 
 | |
|   @param MemoryType     Memory type.
 | |
| 
 | |
|   @return EFI memory type as profile memory index.
 | |
| 
 | |
| **/
 | |
| EFI_MEMORY_TYPE
 | |
| GetProfileMemoryIndex (
 | |
|   IN EFI_MEMORY_TYPE    MemoryType
 | |
|   )
 | |
| {
 | |
|   return MemoryType;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Update SMRAM profile FreeMemoryPages information
 | |
| 
 | |
|   @param ContextData    Memory profile context.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileUpdateFreePages (
 | |
|   IN MEMORY_PROFILE_CONTEXT_DATA  *ContextData
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY                      *Node;
 | |
|   FREE_PAGE_LIST                  *Pages;
 | |
|   LIST_ENTRY                      *FreePageList;
 | |
|   UINTN                           NumberOfPages;
 | |
| 
 | |
|   NumberOfPages = 0;
 | |
|   FreePageList = &mSmmMemoryMap;
 | |
|   for (Node = FreePageList->BackLink;
 | |
|        Node != FreePageList;
 | |
|        Node = Node->BackLink) {
 | |
|     Pages = BASE_CR (Node, FREE_PAGE_LIST, Link);
 | |
|     NumberOfPages += Pages->NumberOfPages;
 | |
|   }
 | |
| 
 | |
|   mSmramFreeMemory.TotalFreeMemoryPages = NumberOfPages;
 | |
| 
 | |
|   if (NumberOfPages <= SMRAM_INFO_DUMP_PAGE_THRESHOLD) {
 | |
|     DumpSmramInfo ();
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Update SMRAM profile Allocate information.
 | |
| 
 | |
|   @param CallerAddress  Address of caller who call Allocate.
 | |
|   @param Action         This Allocate action.
 | |
|   @param MemoryType     Memory type.
 | |
|   @param Size           Buffer size.
 | |
|   @param Buffer         Buffer address.
 | |
| 
 | |
|   @retval TRUE          Profile udpate success.
 | |
|   @retval FALSE         Profile update fail.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| SmmCoreUpdateProfileAllocate (
 | |
|   IN PHYSICAL_ADDRESS       CallerAddress,
 | |
|   IN MEMORY_PROFILE_ACTION  Action,
 | |
|   IN EFI_MEMORY_TYPE        MemoryType,
 | |
|   IN UINTN                  Size,
 | |
|   IN VOID                   *Buffer
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   MEMORY_PROFILE_CONTEXT           *Context;
 | |
|   MEMORY_PROFILE_DRIVER_INFO       *DriverInfo;
 | |
|   MEMORY_PROFILE_ALLOC_INFO        *AllocInfo;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA       *ContextData;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   MEMORY_PROFILE_ALLOC_INFO_DATA    *AllocInfoData;
 | |
|   EFI_MEMORY_TYPE                   ProfileMemoryIndex;
 | |
| 
 | |
|   AllocInfoData = NULL;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   DriverInfoData = GetMemoryProfileDriverInfoFromAddress (ContextData, CallerAddress);
 | |
|   ASSERT (DriverInfoData != NULL);
 | |
| 
 | |
|   //
 | |
|   // Use SmmInternalAllocatePool() that will not update profile for this AllocatePool action.
 | |
|   //
 | |
|   Status = SmmInternalAllocatePool (
 | |
|              EfiRuntimeServicesData,
 | |
|              sizeof (*AllocInfoData),
 | |
|              (VOID **) &AllocInfoData
 | |
|              );
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return FALSE;
 | |
|   }
 | |
|   AllocInfo = &AllocInfoData->AllocInfo;
 | |
|   AllocInfoData->Signature      = MEMORY_PROFILE_ALLOC_INFO_SIGNATURE;
 | |
|   AllocInfo->Header.Signature   = MEMORY_PROFILE_ALLOC_INFO_SIGNATURE;
 | |
|   AllocInfo->Header.Length      = sizeof (MEMORY_PROFILE_ALLOC_INFO);
 | |
|   AllocInfo->Header.Revision    = MEMORY_PROFILE_ALLOC_INFO_REVISION;
 | |
|   AllocInfo->CallerAddress      = CallerAddress;
 | |
|   AllocInfo->SequenceId         = ContextData->Context.SequenceCount;
 | |
|   AllocInfo->Action             = Action;
 | |
|   AllocInfo->MemoryType         = MemoryType;
 | |
|   AllocInfo->Buffer             = (PHYSICAL_ADDRESS) (UINTN) Buffer;
 | |
|   AllocInfo->Size               = Size;
 | |
| 
 | |
|   InsertTailList (DriverInfoData->AllocInfoList, &AllocInfoData->Link);
 | |
| 
 | |
|   ProfileMemoryIndex = GetProfileMemoryIndex (MemoryType);
 | |
| 
 | |
|   DriverInfo = &DriverInfoData->DriverInfo;
 | |
|   DriverInfo->CurrentUsage += Size;
 | |
|   if (DriverInfo->PeakUsage < DriverInfo->CurrentUsage) {
 | |
|     DriverInfo->PeakUsage = DriverInfo->CurrentUsage;
 | |
|   }
 | |
|   DriverInfo->CurrentUsageByType[ProfileMemoryIndex] += Size;
 | |
|   if (DriverInfo->PeakUsageByType[ProfileMemoryIndex] < DriverInfo->CurrentUsageByType[ProfileMemoryIndex]) {
 | |
|     DriverInfo->PeakUsageByType[ProfileMemoryIndex] = DriverInfo->CurrentUsageByType[ProfileMemoryIndex];
 | |
|   }
 | |
|   DriverInfo->AllocRecordCount ++;
 | |
| 
 | |
|   Context = &ContextData->Context;
 | |
|   Context->CurrentTotalUsage += Size;
 | |
|   if (Context->PeakTotalUsage < Context->CurrentTotalUsage) {
 | |
|     Context->PeakTotalUsage = Context->CurrentTotalUsage;
 | |
|   }
 | |
|   Context->CurrentTotalUsageByType[ProfileMemoryIndex] += Size;
 | |
|   if (Context->PeakTotalUsageByType[ProfileMemoryIndex] < Context->CurrentTotalUsageByType[ProfileMemoryIndex]) {
 | |
|     Context->PeakTotalUsageByType[ProfileMemoryIndex] = Context->CurrentTotalUsageByType[ProfileMemoryIndex];
 | |
|   }
 | |
|   Context->SequenceCount ++;
 | |
| 
 | |
|   SmramProfileUpdateFreePages (ContextData);
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get memory profile alloc info from memory profile
 | |
| 
 | |
|   @param DriverInfoData     Driver info
 | |
|   @param Action             This Free action
 | |
|   @param Size               Buffer size
 | |
|   @param Buffer             Buffer address
 | |
| 
 | |
|   @return Pointer to memory profile alloc info.
 | |
| **/
 | |
| MEMORY_PROFILE_ALLOC_INFO_DATA *
 | |
| GetMemoryProfileAllocInfoFromAddress (
 | |
|   IN MEMORY_PROFILE_DRIVER_INFO_DATA    *DriverInfoData,
 | |
|   IN MEMORY_PROFILE_ACTION              Action,
 | |
|   IN UINTN                              Size,
 | |
|   IN VOID                               *Buffer
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY                        *AllocInfoList;
 | |
|   LIST_ENTRY                        *AllocLink;
 | |
|   MEMORY_PROFILE_ALLOC_INFO         *AllocInfo;
 | |
|   MEMORY_PROFILE_ALLOC_INFO_DATA    *AllocInfoData;
 | |
| 
 | |
|   AllocInfoList = DriverInfoData->AllocInfoList;
 | |
| 
 | |
|   for (AllocLink = AllocInfoList->ForwardLink;
 | |
|        AllocLink != AllocInfoList;
 | |
|        AllocLink = AllocLink->ForwardLink) {
 | |
|     AllocInfoData = CR (
 | |
|                       AllocLink,
 | |
|                       MEMORY_PROFILE_ALLOC_INFO_DATA,
 | |
|                       Link,
 | |
|                       MEMORY_PROFILE_ALLOC_INFO_SIGNATURE
 | |
|                       );
 | |
|     AllocInfo = &AllocInfoData->AllocInfo;
 | |
|     if (AllocInfo->Action != Action) {
 | |
|       continue;
 | |
|     }
 | |
|     switch (Action) {
 | |
|       case MemoryProfileActionAllocatePages:
 | |
|         if ((AllocInfo->Buffer <= (PHYSICAL_ADDRESS) (UINTN) Buffer) &&
 | |
|             ((AllocInfo->Buffer + AllocInfo->Size) >= ((PHYSICAL_ADDRESS) (UINTN) Buffer + Size))) {
 | |
|           return AllocInfoData;
 | |
|         }
 | |
|         break;
 | |
|       case MemoryProfileActionAllocatePool:
 | |
|         if (AllocInfo->Buffer == (PHYSICAL_ADDRESS) (UINTN) Buffer) {
 | |
|           return AllocInfoData;
 | |
|         }
 | |
|         break;
 | |
|       default:
 | |
|         ASSERT (FALSE);
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Update SMRAM profile Free information.
 | |
| 
 | |
|   @param CallerAddress  Address of caller who call Free.
 | |
|   @param Action         This Free action.
 | |
|   @param Size           Buffer size.
 | |
|   @param Buffer         Buffer address.
 | |
| 
 | |
|   @retval TRUE          Profile udpate success.
 | |
|   @retval FALSE         Profile update fail.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| SmmCoreUpdateProfileFree (
 | |
|   IN PHYSICAL_ADDRESS       CallerAddress,
 | |
|   IN MEMORY_PROFILE_ACTION  Action,
 | |
|   IN UINTN                  Size,
 | |
|   IN VOID                   *Buffer
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT           *Context;
 | |
|   MEMORY_PROFILE_DRIVER_INFO       *DriverInfo;
 | |
|   MEMORY_PROFILE_ALLOC_INFO        *AllocInfo;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA      *ContextData;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA  *DriverInfoData;
 | |
|   LIST_ENTRY                       *DriverLink;
 | |
|   LIST_ENTRY                       *DriverInfoList;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA  *ThisDriverInfoData;
 | |
|   MEMORY_PROFILE_ALLOC_INFO_DATA   *AllocInfoData;
 | |
|   EFI_MEMORY_TYPE                  ProfileMemoryIndex;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   DriverInfoData = GetMemoryProfileDriverInfoFromAddress (ContextData, CallerAddress);
 | |
|   ASSERT (DriverInfoData != NULL);
 | |
| 
 | |
|   switch (Action) {
 | |
|     case MemoryProfileActionFreePages:
 | |
|       AllocInfoData = GetMemoryProfileAllocInfoFromAddress (DriverInfoData, MemoryProfileActionAllocatePages, Size, Buffer);
 | |
|       break;
 | |
|     case MemoryProfileActionFreePool:
 | |
|       AllocInfoData = GetMemoryProfileAllocInfoFromAddress (DriverInfoData, MemoryProfileActionAllocatePool, 0, Buffer);
 | |
|       break;
 | |
|     default:
 | |
|       ASSERT (FALSE);
 | |
|       AllocInfoData = NULL;
 | |
|       break;
 | |
|   }
 | |
|   if (AllocInfoData == NULL) {
 | |
|     //
 | |
|     // Legal case, because driver A might free memory allocated by driver B, by some protocol.
 | |
|     //
 | |
|     DriverInfoList = ContextData->DriverInfoList;
 | |
| 
 | |
|     for (DriverLink = DriverInfoList->ForwardLink;
 | |
|          DriverLink != DriverInfoList;
 | |
|          DriverLink = DriverLink->ForwardLink) {
 | |
|       ThisDriverInfoData = CR (
 | |
|                              DriverLink,
 | |
|                              MEMORY_PROFILE_DRIVER_INFO_DATA,
 | |
|                              Link,
 | |
|                              MEMORY_PROFILE_DRIVER_INFO_SIGNATURE
 | |
|                              );
 | |
|       switch (Action) {
 | |
|         case MemoryProfileActionFreePages:
 | |
|           AllocInfoData = GetMemoryProfileAllocInfoFromAddress (ThisDriverInfoData, MemoryProfileActionAllocatePages, Size, Buffer);
 | |
|           break;
 | |
|         case MemoryProfileActionFreePool:
 | |
|           AllocInfoData = GetMemoryProfileAllocInfoFromAddress (ThisDriverInfoData, MemoryProfileActionAllocatePool, 0, Buffer);
 | |
|           break;
 | |
|         default:
 | |
|           ASSERT (FALSE);
 | |
|           AllocInfoData = NULL;
 | |
|           break;
 | |
|       }
 | |
|       if (AllocInfoData != NULL) {
 | |
|         DriverInfoData = ThisDriverInfoData;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (AllocInfoData == NULL) {
 | |
|       //
 | |
|       // No matched allocate operation is found for this free operation.
 | |
|       // It is because the specified memory type allocate operation has been
 | |
|       // filtered by CoreNeedRecordProfile(), but free operations have no
 | |
|       // memory type information, they can not be filtered by CoreNeedRecordProfile().
 | |
|       // Then, they will be filtered here.
 | |
|       //
 | |
|       return FALSE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Context = &ContextData->Context;
 | |
|   DriverInfo = &DriverInfoData->DriverInfo;
 | |
|   AllocInfo = &AllocInfoData->AllocInfo;
 | |
| 
 | |
|   ProfileMemoryIndex = GetProfileMemoryIndex (AllocInfo->MemoryType);
 | |
| 
 | |
|   Context->CurrentTotalUsage -= AllocInfo->Size;
 | |
|   Context->CurrentTotalUsageByType[ProfileMemoryIndex] -= AllocInfo->Size;
 | |
| 
 | |
|   DriverInfo->CurrentUsage -= AllocInfo->Size;
 | |
|   DriverInfo->CurrentUsageByType[ProfileMemoryIndex] -= AllocInfo->Size;
 | |
|   DriverInfo->AllocRecordCount --;
 | |
| 
 | |
|   RemoveEntryList (&AllocInfoData->Link);
 | |
| 
 | |
|   if (Action == MemoryProfileActionFreePages) {
 | |
|     if (AllocInfo->Buffer != (PHYSICAL_ADDRESS) (UINTN) Buffer) {
 | |
|       SmmCoreUpdateProfileAllocate (
 | |
|         AllocInfo->CallerAddress,
 | |
|         MemoryProfileActionAllocatePages,
 | |
|         AllocInfo->MemoryType,
 | |
|         (UINTN) ((PHYSICAL_ADDRESS) (UINTN) Buffer - AllocInfo->Buffer),
 | |
|         (VOID *) (UINTN) AllocInfo->Buffer
 | |
|         );
 | |
|     }
 | |
|     if (AllocInfo->Buffer + AllocInfo->Size != ((PHYSICAL_ADDRESS) (UINTN) Buffer + Size)) {
 | |
|       SmmCoreUpdateProfileAllocate (
 | |
|         AllocInfo->CallerAddress,
 | |
|         MemoryProfileActionAllocatePages,
 | |
|         AllocInfo->MemoryType,
 | |
|         (UINTN) ((AllocInfo->Buffer + AllocInfo->Size) - ((PHYSICAL_ADDRESS) (UINTN) Buffer + Size)),
 | |
|         (VOID *) ((UINTN) Buffer + Size)
 | |
|         );
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Use SmmInternalFreePool() that will not update profile for this FreePool action.
 | |
|   //
 | |
|   SmmInternalFreePool (AllocInfoData);
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Update SMRAM profile information.
 | |
| 
 | |
|   @param CallerAddress  Address of caller who call Allocate or Free.
 | |
|   @param Action         This Allocate or Free action.
 | |
|   @param MemoryType     Memory type.
 | |
|   @param Size           Buffer size.
 | |
|   @param Buffer         Buffer address.
 | |
| 
 | |
|   @retval TRUE          Profile udpate success.
 | |
|   @retval FALSE         Profile update fail.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| SmmCoreUpdateProfile (
 | |
|   IN PHYSICAL_ADDRESS       CallerAddress,
 | |
|   IN MEMORY_PROFILE_ACTION  Action,
 | |
|   IN EFI_MEMORY_TYPE        MemoryType, // Valid for AllocatePages/AllocatePool
 | |
|   IN UINTN                  Size,       // Valid for AllocatePages/FreePages/AllocatePool
 | |
|   IN VOID                   *Buffer
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT_DATA   *ContextData;
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   if (!mSmramProfileRecordingStatus) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Free operations have no memory type information, so skip the check.
 | |
|   //
 | |
|   if ((Action == MemoryProfileActionAllocatePages) || (Action == MemoryProfileActionAllocatePool)) {
 | |
|     //
 | |
|     // Only record limited MemoryType.
 | |
|     //
 | |
|     if (!SmmCoreNeedRecordProfile (MemoryType)) {
 | |
|       return FALSE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   switch (Action) {
 | |
|     case MemoryProfileActionAllocatePages:
 | |
|       SmmCoreUpdateProfileAllocate (CallerAddress, Action, MemoryType, Size, Buffer);
 | |
|       break;
 | |
|     case MemoryProfileActionFreePages:
 | |
|       SmmCoreUpdateProfileFree (CallerAddress, Action, Size, Buffer);
 | |
|       break;
 | |
|     case MemoryProfileActionAllocatePool:
 | |
|       SmmCoreUpdateProfileAllocate (CallerAddress, Action, MemoryType, Size, Buffer);
 | |
|       break;
 | |
|     case MemoryProfileActionFreePool:
 | |
|       SmmCoreUpdateProfileFree (CallerAddress, Action, 0, Buffer);
 | |
|       break;
 | |
|     default:
 | |
|       ASSERT (FALSE);
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMRAM profile ready to lock callback function.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileReadyToLock (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "SmramProfileReadyToLock\n"));
 | |
|   mSmramReadyToLock = TRUE;
 | |
| }
 | |
| 
 | |
| ////////////////////
 | |
| 
 | |
| /**
 | |
|   Get SMRAM profile data size.
 | |
| 
 | |
|   @return SMRAM profile data size.
 | |
| 
 | |
| **/
 | |
| UINTN
 | |
| SmramProfileGetDataSize (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT_DATA      *ContextData;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA  *DriverInfoData;
 | |
|   LIST_ENTRY                      *DriverInfoList;
 | |
|   LIST_ENTRY                      *DriverLink;
 | |
|   UINTN                           TotalSize;
 | |
|   LIST_ENTRY                      *Node;
 | |
|   LIST_ENTRY                      *FreePageList;
 | |
|   LIST_ENTRY                      *FreePoolList;
 | |
|   FREE_POOL_HEADER                *Pool;
 | |
|   UINTN                           PoolListIndex;
 | |
|   UINTN                           Index;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   TotalSize = sizeof (MEMORY_PROFILE_CONTEXT);
 | |
|   TotalSize += sizeof (MEMORY_PROFILE_DRIVER_INFO) * (UINTN) ContextData->Context.ImageCount;
 | |
| 
 | |
|   DriverInfoList = ContextData->DriverInfoList;
 | |
|   for (DriverLink = DriverInfoList->ForwardLink;
 | |
|        DriverLink != DriverInfoList;
 | |
|        DriverLink = DriverLink->ForwardLink) {
 | |
|     DriverInfoData = CR (
 | |
|                    DriverLink,
 | |
|                    MEMORY_PROFILE_DRIVER_INFO_DATA,
 | |
|                    Link,
 | |
|                    MEMORY_PROFILE_DRIVER_INFO_SIGNATURE
 | |
|                    );
 | |
|     TotalSize += sizeof (MEMORY_PROFILE_ALLOC_INFO) * (UINTN) DriverInfoData->DriverInfo.AllocRecordCount;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   Index = 0;
 | |
|   FreePageList = &mSmmMemoryMap;
 | |
|   for (Node = FreePageList->BackLink;
 | |
|        Node != FreePageList;
 | |
|        Node = Node->BackLink) {
 | |
|     Index++;
 | |
|   }
 | |
|   for (PoolListIndex = 0; PoolListIndex < MAX_POOL_INDEX; PoolListIndex++) {
 | |
|     FreePoolList = &mSmmPoolLists[PoolListIndex];
 | |
|     for (Node = FreePoolList->BackLink;
 | |
|          Node != FreePoolList;
 | |
|          Node = Node->BackLink) {
 | |
|       Pool = BASE_CR (Node, FREE_POOL_HEADER, Link);
 | |
|       if (Pool->Header.Available) {
 | |
|         Index++;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
| 
 | |
|   TotalSize += (sizeof (MEMORY_PROFILE_FREE_MEMORY) + Index * sizeof (MEMORY_PROFILE_DESCRIPTOR));
 | |
|   TotalSize += (sizeof (MEMORY_PROFILE_MEMORY_RANGE) + mFullSmramRangeCount * sizeof (MEMORY_PROFILE_DESCRIPTOR));
 | |
| 
 | |
|   return TotalSize;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Copy SMRAM profile data.
 | |
| 
 | |
|   @param ProfileBuffer  The buffer to hold SMRAM profile data.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileCopyData (
 | |
|   IN VOID   *ProfileBuffer
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT           *Context;
 | |
|   MEMORY_PROFILE_DRIVER_INFO       *DriverInfo;
 | |
|   MEMORY_PROFILE_ALLOC_INFO        *AllocInfo;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA      *ContextData;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA  *DriverInfoData;
 | |
|   MEMORY_PROFILE_ALLOC_INFO_DATA   *AllocInfoData;
 | |
|   LIST_ENTRY                      *DriverInfoList;
 | |
|   LIST_ENTRY                      *DriverLink;
 | |
|   LIST_ENTRY                      *AllocInfoList;
 | |
|   LIST_ENTRY                      *AllocLink;
 | |
|   LIST_ENTRY                      *Node;
 | |
|   FREE_PAGE_LIST                  *Pages;
 | |
|   LIST_ENTRY                      *FreePageList;
 | |
|   LIST_ENTRY                      *FreePoolList;
 | |
|   FREE_POOL_HEADER                *Pool;
 | |
|   UINTN                           PoolListIndex;
 | |
|   UINT32                          Index;
 | |
|   MEMORY_PROFILE_FREE_MEMORY      *FreeMemory;
 | |
|   MEMORY_PROFILE_MEMORY_RANGE     *MemoryRange;
 | |
|   MEMORY_PROFILE_DESCRIPTOR       *MemoryProfileDescriptor;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   Context = ProfileBuffer;
 | |
|   CopyMem (Context, &ContextData->Context, sizeof (MEMORY_PROFILE_CONTEXT));
 | |
|   DriverInfo = (MEMORY_PROFILE_DRIVER_INFO *) (Context + 1);
 | |
| 
 | |
|   DriverInfoList = ContextData->DriverInfoList;
 | |
|   for (DriverLink = DriverInfoList->ForwardLink;
 | |
|        DriverLink != DriverInfoList;
 | |
|        DriverLink = DriverLink->ForwardLink) {
 | |
|     DriverInfoData = CR (
 | |
|                        DriverLink,
 | |
|                        MEMORY_PROFILE_DRIVER_INFO_DATA,
 | |
|                        Link,
 | |
|                        MEMORY_PROFILE_DRIVER_INFO_SIGNATURE
 | |
|                        );
 | |
|     CopyMem (DriverInfo, &DriverInfoData->DriverInfo, sizeof (MEMORY_PROFILE_DRIVER_INFO));
 | |
|     AllocInfo = (MEMORY_PROFILE_ALLOC_INFO *) (DriverInfo + 1);
 | |
| 
 | |
|     AllocInfoList = DriverInfoData->AllocInfoList;
 | |
|     for (AllocLink = AllocInfoList->ForwardLink;
 | |
|          AllocLink != AllocInfoList;
 | |
|          AllocLink = AllocLink->ForwardLink) {
 | |
|       AllocInfoData = CR (
 | |
|                         AllocLink,
 | |
|                         MEMORY_PROFILE_ALLOC_INFO_DATA,
 | |
|                         Link,
 | |
|                         MEMORY_PROFILE_ALLOC_INFO_SIGNATURE
 | |
|                         );
 | |
|       CopyMem (AllocInfo, &AllocInfoData->AllocInfo, sizeof (MEMORY_PROFILE_ALLOC_INFO));
 | |
|       AllocInfo += 1;
 | |
|     }
 | |
| 
 | |
|     DriverInfo = (MEMORY_PROFILE_DRIVER_INFO *) ((UINTN) (DriverInfo + 1) + sizeof (MEMORY_PROFILE_ALLOC_INFO) * (UINTN) DriverInfo->AllocRecordCount);
 | |
|   }
 | |
| 
 | |
| 
 | |
|   FreeMemory = (MEMORY_PROFILE_FREE_MEMORY *) DriverInfo;
 | |
|   CopyMem (FreeMemory, &mSmramFreeMemory, sizeof (MEMORY_PROFILE_FREE_MEMORY));
 | |
|   MemoryProfileDescriptor = (MEMORY_PROFILE_DESCRIPTOR *) (FreeMemory + 1);
 | |
|   Index = 0;
 | |
|   FreePageList = &mSmmMemoryMap;
 | |
|   for (Node = FreePageList->BackLink;
 | |
|        Node != FreePageList;
 | |
|        Node = Node->BackLink) {
 | |
|     Pages = BASE_CR (Node, FREE_PAGE_LIST, Link);
 | |
|     MemoryProfileDescriptor->Header.Signature = MEMORY_PROFILE_DESCRIPTOR_SIGNATURE;
 | |
|     MemoryProfileDescriptor->Header.Length = sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|     MemoryProfileDescriptor->Header.Revision = MEMORY_PROFILE_DESCRIPTOR_REVISION;
 | |
|     MemoryProfileDescriptor->Address = (PHYSICAL_ADDRESS) (UINTN) Pages;
 | |
|     MemoryProfileDescriptor->Size = EFI_PAGES_TO_SIZE (Pages->NumberOfPages);
 | |
|     MemoryProfileDescriptor++;
 | |
|     Index++;
 | |
|   }
 | |
|   for (PoolListIndex = 0; PoolListIndex < MAX_POOL_INDEX; PoolListIndex++) {
 | |
|     FreePoolList = &mSmmPoolLists[MAX_POOL_INDEX - PoolListIndex - 1];
 | |
|     for (Node = FreePoolList->BackLink;
 | |
|          Node != FreePoolList;
 | |
|          Node = Node->BackLink) {
 | |
|       Pool = BASE_CR (Node, FREE_POOL_HEADER, Link);
 | |
|       if (Pool->Header.Available) {
 | |
|         MemoryProfileDescriptor->Header.Signature = MEMORY_PROFILE_DESCRIPTOR_SIGNATURE;
 | |
|         MemoryProfileDescriptor->Header.Length = sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|         MemoryProfileDescriptor->Header.Revision = MEMORY_PROFILE_DESCRIPTOR_REVISION;
 | |
|         MemoryProfileDescriptor->Address = (PHYSICAL_ADDRESS) (UINTN) Pool;
 | |
|         MemoryProfileDescriptor->Size = Pool->Header.Size;
 | |
|         MemoryProfileDescriptor++;
 | |
|         Index++;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   FreeMemory->FreeMemoryEntryCount = Index;
 | |
| 
 | |
|   MemoryRange = (MEMORY_PROFILE_MEMORY_RANGE *) MemoryProfileDescriptor;
 | |
|   MemoryRange->Header.Signature = MEMORY_PROFILE_MEMORY_RANGE_SIGNATURE;
 | |
|   MemoryRange->Header.Length = sizeof (MEMORY_PROFILE_MEMORY_RANGE);
 | |
|   MemoryRange->Header.Revision = MEMORY_PROFILE_MEMORY_RANGE_REVISION;
 | |
|   MemoryRange->MemoryRangeCount = (UINT32) mFullSmramRangeCount;
 | |
|   MemoryProfileDescriptor = (MEMORY_PROFILE_DESCRIPTOR *) (MemoryRange + 1);
 | |
|   for (Index = 0; Index < mFullSmramRangeCount; Index++) {
 | |
|     MemoryProfileDescriptor->Header.Signature = MEMORY_PROFILE_DESCRIPTOR_SIGNATURE;
 | |
|     MemoryProfileDescriptor->Header.Length = sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|     MemoryProfileDescriptor->Header.Revision = MEMORY_PROFILE_DESCRIPTOR_REVISION;
 | |
|     MemoryProfileDescriptor->Address = mFullSmramRanges[Index].PhysicalStart;
 | |
|     MemoryProfileDescriptor->Size = mFullSmramRanges[Index].PhysicalSize;
 | |
|     MemoryProfileDescriptor++; 
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMRAM profile handler to get profile info.
 | |
| 
 | |
|   @param SmramProfileParameterGetInfo The parameter of SMM profile get size.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileHandlerGetInfo (
 | |
|   IN SMRAM_PROFILE_PARAMETER_GET_PROFILE_INFO   *SmramProfileParameterGetInfo
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT_DATA   *ContextData;
 | |
|   BOOLEAN                       SmramProfileRecordingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileRecordingStatus = mSmramProfileRecordingStatus;
 | |
|   mSmramProfileRecordingStatus = FALSE;
 | |
| 
 | |
|   SmramProfileParameterGetInfo->ProfileSize = SmramProfileGetDataSize();
 | |
|   SmramProfileParameterGetInfo->Header.ReturnStatus = 0;
 | |
| 
 | |
|   mSmramProfileRecordingStatus = SmramProfileRecordingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMRAM profile handler to get profile data.
 | |
| 
 | |
|   @param SmramProfileParameterGetData The parameter of SMM profile get data.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileHandlerGetData (
 | |
|   IN SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA   *SmramProfileParameterGetData
 | |
|   )
 | |
| {
 | |
|   UINT64                                    ProfileSize;
 | |
|   SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA  SmramProfileGetData;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA               *ContextData;
 | |
|   BOOLEAN                                   SmramProfileRecordingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileRecordingStatus = mSmramProfileRecordingStatus;
 | |
|   mSmramProfileRecordingStatus = FALSE;
 | |
| 
 | |
| 
 | |
|   CopyMem (&SmramProfileGetData, SmramProfileParameterGetData, sizeof (SmramProfileGetData));
 | |
| 
 | |
|   ProfileSize = SmramProfileGetDataSize();
 | |
| 
 | |
|   //
 | |
|   // Sanity check
 | |
|   //
 | |
|   if (!SmmIsBufferOutsideSmmValid ((UINTN) SmramProfileGetData.ProfileBuffer, (UINTN) ProfileSize)) {
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerGetData: SMM ProfileBuffer in SMRAM or overflow!\n"));
 | |
|     SmramProfileParameterGetData->ProfileSize = ProfileSize;
 | |
|     SmramProfileParameterGetData->Header.ReturnStatus = (UINT64) (INT64) (INTN) EFI_ACCESS_DENIED;
 | |
|     goto Done;
 | |
|   }
 | |
| 
 | |
|   if (SmramProfileGetData.ProfileSize < ProfileSize) {
 | |
|     SmramProfileParameterGetData->ProfileSize = ProfileSize;
 | |
|     SmramProfileParameterGetData->Header.ReturnStatus = (UINT64) (INT64) (INTN) EFI_BUFFER_TOO_SMALL;
 | |
|     goto Done;
 | |
|   }
 | |
| 
 | |
|   SmramProfileParameterGetData->ProfileSize = ProfileSize;
 | |
|   SmramProfileCopyData ((VOID *) (UINTN) SmramProfileGetData.ProfileBuffer);
 | |
|   SmramProfileParameterGetData->Header.ReturnStatus = 0;
 | |
| 
 | |
| Done:
 | |
|   mSmramProfileRecordingStatus = SmramProfileRecordingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMRAM profile handler to register SMM image.
 | |
| 
 | |
|   @param SmramProfileParameterRegisterImage The parameter of SMM profile register image.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileHandlerRegisterImage (
 | |
|   IN SMRAM_PROFILE_PARAMETER_REGISTER_IMAGE *SmramProfileParameterRegisterImage
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   EFI_SMM_DRIVER_ENTRY              DriverEntry;
 | |
|   VOID                              *EntryPointInImage;
 | |
|   BOOLEAN                           Ret;
 | |
| 
 | |
|   ZeroMem (&DriverEntry, sizeof (DriverEntry));
 | |
|   CopyMem (&DriverEntry.FileName, &SmramProfileParameterRegisterImage->FileName, sizeof(EFI_GUID));
 | |
|   DriverEntry.ImageBuffer = SmramProfileParameterRegisterImage->ImageBuffer;
 | |
|   DriverEntry.NumberOfPage = (UINTN) SmramProfileParameterRegisterImage->NumberOfPage;
 | |
|   Status = InternalPeCoffGetEntryPoint ((VOID *) (UINTN) DriverEntry.ImageBuffer, &EntryPointInImage);
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
|   DriverEntry.ImageEntryPoint = (PHYSICAL_ADDRESS) (UINTN) EntryPointInImage;
 | |
| 
 | |
|   Ret = RegisterSmramProfileImage (&DriverEntry, FALSE);
 | |
|   if (Ret) {
 | |
|     SmramProfileParameterRegisterImage->Header.ReturnStatus = 0;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMRAM profile handler to unregister SMM image.
 | |
| 
 | |
|   @param SmramProfileParameterUnregisterImage The parameter of SMM profile unregister image.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileHandlerUnregisterImage (
 | |
|   IN SMRAM_PROFILE_PARAMETER_UNREGISTER_IMAGE *SmramProfileParameterUnregisterImage
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   EFI_SMM_DRIVER_ENTRY              DriverEntry;
 | |
|   VOID                              *EntryPointInImage;
 | |
|   BOOLEAN                           Ret;
 | |
| 
 | |
|   ZeroMem (&DriverEntry, sizeof (DriverEntry));
 | |
|   CopyMem (&DriverEntry.FileName, &SmramProfileParameterUnregisterImage->FileName, sizeof (EFI_GUID));
 | |
|   DriverEntry.ImageBuffer = SmramProfileParameterUnregisterImage->ImageBuffer;
 | |
|   DriverEntry.NumberOfPage = (UINTN) SmramProfileParameterUnregisterImage->NumberOfPage;
 | |
|   Status = InternalPeCoffGetEntryPoint ((VOID *) (UINTN) DriverEntry.ImageBuffer, &EntryPointInImage);
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
|   DriverEntry.ImageEntryPoint = (PHYSICAL_ADDRESS) (UINTN) EntryPointInImage;
 | |
| 
 | |
|   Ret = UnregisterSmramProfileImage (&DriverEntry, FALSE);
 | |
|   if (Ret) {
 | |
|     SmramProfileParameterUnregisterImage->Header.ReturnStatus = 0;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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
 | |
| SmramProfileHandler (
 | |
|   IN EFI_HANDLE  DispatchHandle,
 | |
|   IN CONST VOID  *Context         OPTIONAL,
 | |
|   IN OUT VOID    *CommBuffer      OPTIONAL,
 | |
|   IN OUT UINTN   *CommBufferSize  OPTIONAL
 | |
|   )
 | |
| {
 | |
|   SMRAM_PROFILE_PARAMETER_HEADER           *SmramProfileParameterHeader;
 | |
|   UINTN                                    TempCommBufferSize;
 | |
| 
 | |
|   DEBUG ((EFI_D_ERROR, "SmramProfileHandler Enter\n"));
 | |
| 
 | |
|   //
 | |
|   // If input is invalid, stop processing this SMI
 | |
|   //
 | |
|   if (CommBuffer == NULL || CommBufferSize == NULL) {
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   TempCommBufferSize = *CommBufferSize;
 | |
| 
 | |
|   if (TempCommBufferSize < sizeof (SMRAM_PROFILE_PARAMETER_HEADER)) {
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   if (mSmramReadyToLock && !SmmIsBufferOutsideSmmValid ((UINTN)CommBuffer, TempCommBufferSize)) {
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer in SMRAM or overflow!\n"));
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   SmramProfileParameterHeader = (SMRAM_PROFILE_PARAMETER_HEADER *) ((UINTN) CommBuffer);
 | |
| 
 | |
|   SmramProfileParameterHeader->ReturnStatus = (UINT64)-1;
 | |
| 
 | |
|   if (GetSmramProfileContext () == NULL) {
 | |
|     SmramProfileParameterHeader->ReturnStatus = (UINT64) (INT64) (INTN) EFI_UNSUPPORTED;
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   switch (SmramProfileParameterHeader->Command) {
 | |
|   case SMRAM_PROFILE_COMMAND_GET_PROFILE_INFO:
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerGetInfo\n"));
 | |
|     if (TempCommBufferSize != sizeof (SMRAM_PROFILE_PARAMETER_GET_PROFILE_INFO)) {
 | |
|       DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     SmramProfileHandlerGetInfo ((SMRAM_PROFILE_PARAMETER_GET_PROFILE_INFO *) (UINTN) CommBuffer);
 | |
|     break;
 | |
|   case SMRAM_PROFILE_COMMAND_GET_PROFILE_DATA:
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerGetData\n"));
 | |
|     if (TempCommBufferSize != sizeof (SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA)) {
 | |
|       DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     SmramProfileHandlerGetData ((SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA *) (UINTN) CommBuffer);
 | |
|     break;
 | |
|   case SMRAM_PROFILE_COMMAND_REGISTER_IMAGE:
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerRegisterImage\n"));
 | |
|     if (TempCommBufferSize != sizeof (SMRAM_PROFILE_PARAMETER_REGISTER_IMAGE)) {
 | |
|       DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     if (mSmramReadyToLock) {
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     SmramProfileHandlerRegisterImage ((SMRAM_PROFILE_PARAMETER_REGISTER_IMAGE *) (UINTN) CommBuffer);
 | |
|     break;
 | |
|   case SMRAM_PROFILE_COMMAND_UNREGISTER_IMAGE:
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerUnregisterImage\n"));
 | |
|     if (TempCommBufferSize != sizeof (SMRAM_PROFILE_PARAMETER_UNREGISTER_IMAGE)) {
 | |
|       DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     if (mSmramReadyToLock) {
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     SmramProfileHandlerUnregisterImage ((SMRAM_PROFILE_PARAMETER_UNREGISTER_IMAGE *) (UINTN) CommBuffer);
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   DEBUG ((EFI_D_ERROR, "SmramProfileHandler Exit\n"));
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Register SMRAM profile handler.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| RegisterSmramProfileHandler (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS    Status;
 | |
|   EFI_HANDLE    DispatchHandle;
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Status = SmiHandlerRegister (
 | |
|              SmramProfileHandler,
 | |
|              &gEdkiiMemoryProfileGuid,
 | |
|              &DispatchHandle
 | |
|              );
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
| }
 | |
| 
 | |
| ////////////////////
 | |
| 
 | |
| /**
 | |
|   Dump SMRAM range.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| DumpSmramRange (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   UINTN                         Index;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA   *ContextData;
 | |
|   BOOLEAN                       SmramProfileRecordingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileRecordingStatus = mSmramProfileRecordingStatus;
 | |
|   mSmramProfileRecordingStatus = FALSE;
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "FullSmramRange address - 0x%08x\n", mFullSmramRanges));
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "======= SmramProfile begin =======\n"));
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "FullSmramRange:\n"));
 | |
|   for (Index = 0; Index < mFullSmramRangeCount; Index++) {
 | |
|     DEBUG ((EFI_D_INFO, "  FullSmramRange (0x%x)\n", Index));
 | |
|     DEBUG ((EFI_D_INFO, "    PhysicalStart      - 0x%016lx\n", mFullSmramRanges[Index].PhysicalStart));
 | |
|     DEBUG ((EFI_D_INFO, "    CpuStart           - 0x%016lx\n", mFullSmramRanges[Index].CpuStart));
 | |
|     DEBUG ((EFI_D_INFO, "    PhysicalSize       - 0x%016lx\n", mFullSmramRanges[Index].PhysicalSize));
 | |
|     DEBUG ((EFI_D_INFO, "    RegionState        - 0x%016lx\n", mFullSmramRanges[Index].RegionState));
 | |
|   }
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "======= SmramProfile end =======\n"));
 | |
| 
 | |
|   mSmramProfileRecordingStatus = SmramProfileRecordingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dump SMRAM free page list.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| DumpFreePagesList (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY                    *FreePageList;
 | |
|   LIST_ENTRY                    *Node;
 | |
|   FREE_PAGE_LIST                *Pages;
 | |
|   UINTN                         Index;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA   *ContextData;
 | |
|   BOOLEAN                       SmramProfileRecordingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileRecordingStatus = mSmramProfileRecordingStatus;
 | |
|   mSmramProfileRecordingStatus = FALSE;
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "======= SmramProfile begin =======\n"));
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "FreePagesList:\n"));
 | |
|   FreePageList = &mSmmMemoryMap;
 | |
|   for (Node = FreePageList->BackLink, Index = 0;
 | |
|        Node != FreePageList;
 | |
|        Node = Node->BackLink, Index++) {
 | |
|     Pages = BASE_CR (Node, FREE_PAGE_LIST, Link);
 | |
|     DEBUG ((EFI_D_INFO, "  Index - 0x%x\n", Index));
 | |
|     DEBUG ((EFI_D_INFO, "    PhysicalStart - 0x%016lx\n", (PHYSICAL_ADDRESS) (UINTN) Pages));
 | |
|     DEBUG ((EFI_D_INFO, "    NumberOfPages - 0x%08x\n", Pages->NumberOfPages));
 | |
|   }
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "======= SmramProfile end =======\n"));
 | |
| 
 | |
|   mSmramProfileRecordingStatus = SmramProfileRecordingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dump SMRAM free pool list.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| DumpFreePoolList (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   LIST_ENTRY                     *FreePoolList;
 | |
|   LIST_ENTRY                     *Node;
 | |
|   FREE_POOL_HEADER               *Pool;
 | |
|   UINTN                          Index;
 | |
|   UINTN                          PoolListIndex;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA   *ContextData;
 | |
|   BOOLEAN                       SmramProfileRecordingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileRecordingStatus = mSmramProfileRecordingStatus;
 | |
|   mSmramProfileRecordingStatus = FALSE;
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "======= SmramProfile begin =======\n"));
 | |
| 
 | |
|   for (PoolListIndex = 0; PoolListIndex < MAX_POOL_INDEX; PoolListIndex++) {
 | |
|     DEBUG ((EFI_D_INFO, "FreePoolList (%d):\n", PoolListIndex));
 | |
|     FreePoolList = &mSmmPoolLists[PoolListIndex];
 | |
|     for (Node = FreePoolList->BackLink, Index = 0;
 | |
|          Node != FreePoolList;
 | |
|          Node = Node->BackLink, Index++) {
 | |
|       Pool = BASE_CR (Node, FREE_POOL_HEADER, Link);
 | |
|       DEBUG ((EFI_D_INFO, "  Index - 0x%x\n", Index));
 | |
|       DEBUG ((EFI_D_INFO, "    PhysicalStart - 0x%016lx\n", (PHYSICAL_ADDRESS) (UINTN) Pool));
 | |
|       DEBUG ((EFI_D_INFO, "    Size          - 0x%08x\n", Pool->Header.Size));
 | |
|       DEBUG ((EFI_D_INFO, "    Available     - 0x%02x\n", Pool->Header.Available));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "======= SmramProfile end =======\n"));
 | |
| 
 | |
|   mSmramProfileRecordingStatus = SmramProfileRecordingStatus;
 | |
| }
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED CHAR16 *mActionString[] = {
 | |
|   L"Unknown",
 | |
|   L"AllocatePages",
 | |
|   L"FreePages",
 | |
|   L"AllocatePool",
 | |
|   L"FreePool",
 | |
| };
 | |
| 
 | |
| typedef struct {
 | |
|   EFI_MEMORY_TYPE   MemoryType;
 | |
|   CHAR16            *MemoryTypeStr;
 | |
| } PROFILE_MEMORY_TYPE_STRING;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED PROFILE_MEMORY_TYPE_STRING mMemoryTypeString[] = {
 | |
|   {EfiRuntimeServicesCode, L"EfiRuntimeServicesCode"},
 | |
|   {EfiRuntimeServicesData, L"EfiRuntimeServicesData"}
 | |
| };
 | |
| 
 | |
| /**
 | |
|   Memory type to string.
 | |
| 
 | |
|   @param[in] MemoryType Memory type.
 | |
| 
 | |
|   @return Pointer to string.
 | |
| 
 | |
| **/
 | |
| CHAR16 *
 | |
| ProfileMemoryTypeToStr (
 | |
|   IN EFI_MEMORY_TYPE    MemoryType
 | |
|   )
 | |
| {
 | |
|   UINTN     Index;
 | |
|   for (Index = 0; Index < sizeof (mMemoryTypeString) / sizeof (mMemoryTypeString[0]); Index++) {
 | |
|     if (mMemoryTypeString[Index].MemoryType == MemoryType) {
 | |
|       return mMemoryTypeString[Index].MemoryTypeStr;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return L"UnexpectedMemoryType";
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dump SMRAM profile.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| DumpSmramProfile (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT            *Context;
 | |
|   MEMORY_PROFILE_DRIVER_INFO        *DriverInfo;
 | |
|   MEMORY_PROFILE_ALLOC_INFO         *AllocInfo;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA       *ContextData;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   MEMORY_PROFILE_ALLOC_INFO_DATA    *AllocInfoData;
 | |
|   LIST_ENTRY                        *SmramDriverInfoList;
 | |
|   UINTN                             DriverIndex;
 | |
|   LIST_ENTRY                        *DriverLink;
 | |
|   LIST_ENTRY                        *AllocInfoList;
 | |
|   UINTN                             AllocIndex;
 | |
|   LIST_ENTRY                        *AllocLink;
 | |
|   BOOLEAN                           SmramProfileRecordingStatus;
 | |
|   UINTN                             TypeIndex;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileRecordingStatus = mSmramProfileRecordingStatus;
 | |
|   mSmramProfileRecordingStatus = FALSE;
 | |
| 
 | |
|   Context = &ContextData->Context;
 | |
|   DEBUG ((EFI_D_INFO, "======= SmramProfile begin =======\n"));
 | |
|   DEBUG ((EFI_D_INFO, "MEMORY_PROFILE_CONTEXT\n"));
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "  CurrentTotalUsage     - 0x%016lx\n", Context->CurrentTotalUsage));
 | |
|   DEBUG ((EFI_D_INFO, "  PeakTotalUsage        - 0x%016lx\n", Context->PeakTotalUsage));
 | |
|   for (TypeIndex = 0; TypeIndex < sizeof (Context->CurrentTotalUsageByType) / sizeof (Context->CurrentTotalUsageByType[0]); TypeIndex++) {
 | |
|     if ((Context->CurrentTotalUsageByType[TypeIndex] != 0) ||
 | |
|         (Context->PeakTotalUsageByType[TypeIndex] != 0)) {
 | |
|       DEBUG ((EFI_D_INFO, "  CurrentTotalUsage[0x%02x]  - 0x%016lx (%s)\n", TypeIndex, Context->CurrentTotalUsageByType[TypeIndex], ProfileMemoryTypeToStr (TypeIndex)));
 | |
|       DEBUG ((EFI_D_INFO, "  PeakTotalUsage[0x%02x]     - 0x%016lx (%s)\n", TypeIndex, Context->PeakTotalUsageByType[TypeIndex], ProfileMemoryTypeToStr (TypeIndex)));
 | |
|     }
 | |
|   }
 | |
|   DEBUG ((EFI_D_INFO, "  TotalImageSize        - 0x%016lx\n", Context->TotalImageSize));
 | |
|   DEBUG ((EFI_D_INFO, "  ImageCount            - 0x%08x\n", Context->ImageCount));
 | |
|   DEBUG ((EFI_D_INFO, "  SequenceCount         - 0x%08x\n", Context->SequenceCount));
 | |
| 
 | |
|   SmramDriverInfoList = ContextData->DriverInfoList;
 | |
|   for (DriverLink = SmramDriverInfoList->ForwardLink, DriverIndex = 0;
 | |
|        DriverLink != SmramDriverInfoList;
 | |
|        DriverLink = DriverLink->ForwardLink, DriverIndex++) {
 | |
|     DriverInfoData = CR (
 | |
|                    DriverLink,
 | |
|                    MEMORY_PROFILE_DRIVER_INFO_DATA,
 | |
|                    Link,
 | |
|                    MEMORY_PROFILE_DRIVER_INFO_SIGNATURE
 | |
|                    );
 | |
|     DriverInfo = &DriverInfoData->DriverInfo;
 | |
|     DEBUG ((EFI_D_INFO, "  MEMORY_PROFILE_DRIVER_INFO (0x%x)\n", DriverIndex));
 | |
|     DEBUG ((EFI_D_INFO, "    FileName            - %g\n", &DriverInfo->FileName));
 | |
|     DEBUG ((EFI_D_INFO, "    ImageBase           - 0x%016lx\n", DriverInfo->ImageBase));
 | |
|     DEBUG ((EFI_D_INFO, "    ImageSize           - 0x%016lx\n", DriverInfo->ImageSize));
 | |
|     DEBUG ((EFI_D_INFO, "    EntryPoint          - 0x%016lx\n", DriverInfo->EntryPoint));
 | |
|     DEBUG ((EFI_D_INFO, "    ImageSubsystem      - 0x%04x\n", DriverInfo->ImageSubsystem));
 | |
|     DEBUG ((EFI_D_INFO, "    FileType            - 0x%02x\n", DriverInfo->FileType));
 | |
|     DEBUG ((EFI_D_INFO, "    CurrentUsage        - 0x%016lx\n", DriverInfo->CurrentUsage));
 | |
|     DEBUG ((EFI_D_INFO, "    PeakUsage           - 0x%016lx\n", DriverInfo->PeakUsage));
 | |
|     for (TypeIndex = 0; TypeIndex < sizeof (DriverInfo->CurrentUsageByType) / sizeof (DriverInfo->CurrentUsageByType[0]); TypeIndex++) {
 | |
|       if ((DriverInfo->CurrentUsageByType[TypeIndex] != 0) ||
 | |
|           (DriverInfo->PeakUsageByType[TypeIndex] != 0)) {
 | |
|         DEBUG ((EFI_D_INFO, "    CurrentUsage[0x%02x]     - 0x%016lx (%s)\n", TypeIndex, DriverInfo->CurrentUsageByType[TypeIndex], ProfileMemoryTypeToStr (TypeIndex)));
 | |
|         DEBUG ((EFI_D_INFO, "    PeakUsage[0x%02x]        - 0x%016lx (%s)\n", TypeIndex, DriverInfo->PeakUsageByType[TypeIndex], ProfileMemoryTypeToStr (TypeIndex)));
 | |
|       }
 | |
|     }
 | |
|     DEBUG ((EFI_D_INFO, "    AllocRecordCount    - 0x%08x\n", DriverInfo->AllocRecordCount));
 | |
| 
 | |
|     AllocInfoList = DriverInfoData->AllocInfoList;
 | |
|     for (AllocLink = AllocInfoList->ForwardLink, AllocIndex = 0;
 | |
|          AllocLink != AllocInfoList;
 | |
|          AllocLink = AllocLink->ForwardLink, AllocIndex++) {
 | |
|       AllocInfoData = CR (
 | |
|                      AllocLink,
 | |
|                      MEMORY_PROFILE_ALLOC_INFO_DATA,
 | |
|                      Link,
 | |
|                      MEMORY_PROFILE_ALLOC_INFO_SIGNATURE
 | |
|                      );
 | |
|       AllocInfo = &AllocInfoData->AllocInfo;
 | |
|       DEBUG ((EFI_D_INFO, "    MEMORY_PROFILE_ALLOC_INFO (0x%x)\n", AllocIndex));
 | |
|       DEBUG ((EFI_D_INFO, "      CallerAddress  - 0x%016lx (Offset: 0x%08x)\n", AllocInfo->CallerAddress, AllocInfo->CallerAddress - DriverInfo->ImageBase));
 | |
|       DEBUG ((EFI_D_INFO, "      SequenceId     - 0x%08x\n", AllocInfo->SequenceId));
 | |
|       DEBUG ((EFI_D_INFO, "      Action         - 0x%08x (%s)\n", AllocInfo->Action, mActionString[(AllocInfo->Action < sizeof(mActionString)/sizeof(mActionString[0])) ? AllocInfo->Action : 0]));
 | |
|       DEBUG ((EFI_D_INFO, "      MemoryType     - 0x%08x\n", AllocInfo->MemoryType));
 | |
|       DEBUG ((EFI_D_INFO, "      Buffer         - 0x%016lx\n", AllocInfo->Buffer));
 | |
|       DEBUG ((EFI_D_INFO, "      Size           - 0x%016lx\n", AllocInfo->Size));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "======= SmramProfile end =======\n"));
 | |
| 
 | |
|   mSmramProfileRecordingStatus = SmramProfileRecordingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dump SMRAM infromation.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| DumpSmramInfo (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   DEBUG_CODE (
 | |
|     if (IS_SMRAM_PROFILE_ENABLED) {
 | |
|       DumpSmramProfile ();
 | |
|       DumpFreePagesList ();
 | |
|       DumpFreePoolList ();
 | |
|       DumpSmramRange ();
 | |
|     }
 | |
|   );
 | |
| }
 | |
| 
 |