PiSmmCore supports page level protection based upon the Memory Type (EfiRuntimeServicesCode/EfiRuntimeServicesData) and PE image. However, the Memory Type information is ignored in AllocatePool(). If a caller calls AllocatePool with EfiRuntimeServicesCode, the final memory is still allocated as EfiRuntimeServicesData. This patch supports AllocatePool with EfiRuntimeServicesCode. Cc: Jeff Fan <jeff.fan@intel.com> Cc: Michael D Kinney <michael.d.kinney@intel.com> Cc: Laszlo Ersek <lersek@redhat.com> Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Jiewen Yao <jiewen.yao@intel.com> Reviewed-by: Jeff Fan <jeff.fan@intel.com> Regression-tested-by: Laszlo Ersek <lersek@redhat.com>
		
			
				
	
	
		
			2855 lines
		
	
	
		
			99 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			2855 lines
		
	
	
		
			99 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   Support routines for SMRAM profile.
 | |
| 
 | |
|   Copyright (c) 2014 - 2016, 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)
 | |
| #define IS_UEFI_MEMORY_PROFILE_ENABLED ((PcdGet8 (PcdMemoryProfilePropertyMask) & BIT0) != 0)
 | |
| 
 | |
| #define GET_OCCUPIED_SIZE(ActualSize, Alignment) \
 | |
|   ((ActualSize) + (((Alignment) - ((ActualSize) & ((Alignment) - 1))) & ((Alignment) - 1)))
 | |
| 
 | |
| 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;
 | |
|   CHAR8                         *PdbString;
 | |
|   LIST_ENTRY                    Link;
 | |
| } MEMORY_PROFILE_DRIVER_INFO_DATA;
 | |
| 
 | |
| typedef struct {
 | |
|   UINT32                        Signature;
 | |
|   MEMORY_PROFILE_ALLOC_INFO     AllocInfo;
 | |
|   CHAR8                         *ActionString;
 | |
|   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 = NULL;
 | |
| 
 | |
| BOOLEAN mSmramReadyToLock;
 | |
| BOOLEAN mSmramProfileGettingStatus = FALSE;
 | |
| BOOLEAN mSmramProfileRecordingEnable = MEMORY_PROFILE_RECORDING_DISABLE;
 | |
| EFI_DEVICE_PATH_PROTOCOL *mSmramProfileDriverPath;
 | |
| UINTN                    mSmramProfileDriverPathSize;
 | |
| 
 | |
| /**
 | |
|   Dump SMRAM infromation.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| DumpSmramInfo (
 | |
|   VOID
 | |
|   );
 | |
| 
 | |
| /**
 | |
|   Get memory profile data.
 | |
| 
 | |
|   @param[in]      This              The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in, out] ProfileSize       On entry, points to the size in bytes of the ProfileBuffer.
 | |
|                                     On return, points to the size of the data returned in ProfileBuffer.
 | |
|   @param[out]     ProfileBuffer     Profile buffer.
 | |
|                       
 | |
|   @return EFI_SUCCESS               Get the memory profile data successfully.
 | |
|   @return EFI_UNSUPPORTED           Memory profile is unsupported.
 | |
|   @return EFI_BUFFER_TO_SMALL       The ProfileSize is too small for the resulting data. 
 | |
|                                     ProfileSize is updated with the size required.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolGetData (
 | |
|   IN     EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN OUT UINT64                             *ProfileSize,
 | |
|      OUT VOID                               *ProfileBuffer
 | |
|   );
 | |
| 
 | |
| /**
 | |
|   Register image to memory profile.
 | |
| 
 | |
|   @param[in] This               The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in] FilePath           File path of the image.
 | |
|   @param[in] ImageBase          Image base address.
 | |
|   @param[in] ImageSize          Image size.
 | |
|   @param[in] FileType           File type of the image.
 | |
| 
 | |
|   @return EFI_SUCCESS           Register successfully.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required.
 | |
|   @return EFI_OUT_OF_RESOURCE   No enough resource for this register.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolRegisterImage (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN EFI_DEVICE_PATH_PROTOCOL           *FilePath,
 | |
|   IN PHYSICAL_ADDRESS                   ImageBase,
 | |
|   IN UINT64                             ImageSize,
 | |
|   IN EFI_FV_FILETYPE                    FileType
 | |
|   );
 | |
| 
 | |
| /**
 | |
|   Unregister image from memory profile.
 | |
| 
 | |
|   @param[in] This               The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in] FilePath           File path of the image.
 | |
|   @param[in] ImageBase          Image base address.
 | |
|   @param[in] ImageSize          Image size.
 | |
| 
 | |
|   @return EFI_SUCCESS           Unregister successfully.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required.
 | |
|   @return EFI_NOT_FOUND         The image is not found.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolUnregisterImage (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN EFI_DEVICE_PATH_PROTOCOL           *FilePath,
 | |
|   IN PHYSICAL_ADDRESS                   ImageBase,
 | |
|   IN UINT64                             ImageSize
 | |
|   );
 | |
| 
 | |
| /**
 | |
|   Get memory profile recording state.
 | |
| 
 | |
|   @param[in]  This              The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[out] RecordingState    Recording state.
 | |
| 
 | |
|   @return EFI_SUCCESS           Memory profile recording state is returned.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported.
 | |
|   @return EFI_INVALID_PARAMETER RecordingState is NULL.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolGetRecordingState (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   OUT BOOLEAN                           *RecordingState
 | |
|   );
 | |
| 
 | |
| /**
 | |
|   Set memory profile recording state.
 | |
| 
 | |
|   @param[in] This               The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in] RecordingState     Recording state.
 | |
| 
 | |
|   @return EFI_SUCCESS           Set memory profile recording state successfully.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolSetRecordingState (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN BOOLEAN                            RecordingState
 | |
|   );
 | |
| 
 | |
| /**
 | |
|   Record memory profile of multilevel caller.
 | |
| 
 | |
|   @param[in] This               The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in] CallerAddress      Address of caller.
 | |
|   @param[in] Action             Memory profile action.
 | |
|   @param[in] MemoryType         Memory type.
 | |
|                                 EfiMaxMemoryType means the MemoryType is unknown.
 | |
|   @param[in] Buffer             Buffer address.
 | |
|   @param[in] Size               Buffer size.
 | |
|   @param[in] ActionString       String for memory profile action.
 | |
|                                 Only needed for user defined allocate action.
 | |
| 
 | |
|   @return EFI_SUCCESS           Memory profile is updated.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required,
 | |
|                                 or memory profile for the memory type is not required.
 | |
|   @return EFI_ACCESS_DENIED     It is during memory profile data getting.
 | |
|   @return EFI_ABORTED           Memory profile recording is not enabled.
 | |
|   @return EFI_OUT_OF_RESOURCES  No enough resource to update memory profile for allocate action.
 | |
|   @return EFI_NOT_FOUND         No matched allocate info found for free action.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolRecord (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN PHYSICAL_ADDRESS                   CallerAddress,
 | |
|   IN MEMORY_PROFILE_ACTION              Action,
 | |
|   IN EFI_MEMORY_TYPE                    MemoryType,
 | |
|   IN VOID                               *Buffer,
 | |
|   IN UINTN                              Size,
 | |
|   IN CHAR8                              *ActionString OPTIONAL
 | |
|   );
 | |
| 
 | |
| EDKII_SMM_MEMORY_PROFILE_PROTOCOL mSmmProfileProtocol = {
 | |
|   SmramProfileProtocolGetData,
 | |
|   SmramProfileProtocolRegisterImage,
 | |
|   SmramProfileProtocolUnregisterImage,
 | |
|   SmramProfileProtocolGetRecordingState,
 | |
|   SmramProfileProtocolSetRecordingState,
 | |
|   SmramProfileProtocolRecord,
 | |
| };
 | |
| 
 | |
| /**
 | |
|   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;
 | |
|   CHAR8                             *PdbString;
 | |
|   UINTN                             PdbSize;
 | |
|   UINTN                             PdbOccupiedSize;
 | |
| 
 | |
|   PdbSize = 0;
 | |
|   PdbOccupiedSize = 0;
 | |
|   PdbString = NULL;
 | |
|   if (ImageBase != 0) {
 | |
|     PdbString = PeCoffLoaderGetPdbPointer ((VOID*) (UINTN) ImageBase);
 | |
|     if (PdbString != NULL) {
 | |
|       PdbSize = AsciiStrSize (PdbString);
 | |
|       PdbOccupiedSize = GET_OCCUPIED_SIZE (PdbSize, sizeof (UINT64));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Use SmmInternalAllocatePool() that will not update profile for this AllocatePool action.
 | |
|   //
 | |
|   Status = SmmInternalAllocatePool (
 | |
|              EfiRuntimeServicesData,
 | |
|              sizeof (*DriverInfoData) + sizeof (LIST_ENTRY) + PdbSize,
 | |
|              (VOID **) &DriverInfoData
 | |
|              );
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return NULL;
 | |
|   }
 | |
|   ASSERT (DriverInfoData != 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 = (UINT16) (sizeof (MEMORY_PROFILE_DRIVER_INFO) + PdbOccupiedSize);
 | |
|   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;
 | |
|   if (PdbSize != 0) {
 | |
|     DriverInfo->PdbStringOffset = (UINT16) sizeof (MEMORY_PROFILE_DRIVER_INFO);
 | |
|     DriverInfoData->PdbString = (CHAR8 *) (DriverInfoData->AllocInfoList + 1);
 | |
|     CopyMem (DriverInfoData->PdbString, PdbString, PdbSize);
 | |
|   } else {
 | |
|     DriverInfo->PdbStringOffset = 0;
 | |
|     DriverInfoData->PdbString = NULL;
 | |
|   }
 | |
| 
 | |
|   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_UEFI_MEMORY_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_UEFI_MEMORY_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
 | |
|                                   );
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return if record for this driver is needed..
 | |
| 
 | |
|   @param DriverFilePath     Driver file path.
 | |
| 
 | |
|   @retval TRUE              Record for this driver is needed.
 | |
|   @retval FALSE             Record for this driver is not needed.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| NeedRecordThisDriver (
 | |
|   IN EFI_DEVICE_PATH_PROTOCOL       *DriverFilePath
 | |
|   )
 | |
| {
 | |
|   EFI_DEVICE_PATH_PROTOCOL    *TmpDevicePath;
 | |
|   EFI_DEVICE_PATH_PROTOCOL    *DevicePathInstance;
 | |
|   UINTN                       DevicePathSize;
 | |
|   UINTN                       FilePathSize;
 | |
| 
 | |
|   if (!IsDevicePathValid (mSmramProfileDriverPath, mSmramProfileDriverPathSize)) {
 | |
|     //
 | |
|     // Invalid Device Path means record all.
 | |
|     //
 | |
|     return TRUE;
 | |
|   }
 | |
|   
 | |
|   //
 | |
|   // Record FilePath without end node.
 | |
|   //
 | |
|   FilePathSize = GetDevicePathSize (DriverFilePath) - sizeof(EFI_DEVICE_PATH_PROTOCOL);
 | |
| 
 | |
|   DevicePathInstance = mSmramProfileDriverPath;
 | |
|   do {
 | |
|     //
 | |
|     // Find End node (it might be END_ENTIRE or END_INSTANCE)
 | |
|     //
 | |
|     TmpDevicePath = DevicePathInstance;
 | |
|     while (!IsDevicePathEndType (TmpDevicePath)) {
 | |
|       TmpDevicePath = NextDevicePathNode (TmpDevicePath);
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     // Do not compare END node
 | |
|     //
 | |
|     DevicePathSize = (UINTN)TmpDevicePath - (UINTN)DevicePathInstance;
 | |
|     if ((FilePathSize == DevicePathSize) &&
 | |
|         (CompareMem (DriverFilePath, DevicePathInstance, DevicePathSize) == 0)) {
 | |
|       return TRUE;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     // Get next instance
 | |
|     //
 | |
|     DevicePathInstance = (EFI_DEVICE_PATH_PROTOCOL *)((UINTN)DevicePathInstance + DevicePathSize + DevicePathNodeLength(TmpDevicePath));
 | |
|   } while (DevicePathSubType (TmpDevicePath) != END_ENTIRE_DEVICE_PATH_SUBTYPE);
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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;
 | |
|   UINT8                             TempBuffer[sizeof(MEDIA_FW_VOL_FILEPATH_DEVICE_PATH) + sizeof(EFI_DEVICE_PATH_PROTOCOL)];
 | |
|   MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *FilePath;
 | |
| 
 | |
|   FilePath = (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *) TempBuffer;
 | |
|   EfiInitializeFwVolDevicepathNode (FilePath, &gEfiCallerIdGuid);
 | |
|   SetDevicePathEndNode (FilePath + 1);
 | |
| 
 | |
|   if (!NeedRecordThisDriver ((EFI_DEVICE_PATH_PROTOCOL *) FilePath)) {
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   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;
 | |
| 
 | |
|   RegisterImageToDxe (
 | |
|     &gEfiCallerIdGuid,
 | |
|     gSmmCorePrivate->PiSmmCoreImageBase,
 | |
|     gSmmCorePrivate->PiSmmCoreImageSize,
 | |
|     EFI_FV_FILETYPE_SMM_CORE
 | |
|     );
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   SmramProfileContext = GetSmramProfileContext ();
 | |
|   if (SmramProfileContext != NULL) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   mSmramProfileGettingStatus = FALSE;
 | |
|   if ((PcdGet8 (PcdMemoryProfilePropertyMask) & BIT7) != 0) {
 | |
|     mSmramProfileRecordingEnable = MEMORY_PROFILE_RECORDING_DISABLE;
 | |
|   } else {
 | |
|     mSmramProfileRecordingEnable = MEMORY_PROFILE_RECORDING_ENABLE;
 | |
|   }
 | |
|   mSmramProfileDriverPathSize = PcdGetSize (PcdMemoryProfileDriverPath);
 | |
|   mSmramProfileDriverPath = AllocateCopyPool (mSmramProfileDriverPathSize, PcdGetPtr (PcdMemoryProfileDriverPath));
 | |
|   mSmramProfileContextPtr = &mSmramProfileContext;
 | |
| 
 | |
|   RegisterSmmCore (&mSmramProfileContext);
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "SmramProfileInit SmramProfileContext - 0x%x\n", &mSmramProfileContext));
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Install SMRAM profile protocol.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileInstallProtocol (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   EFI_HANDLE    Handle;
 | |
|   EFI_STATUS    Status;
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Handle = NULL;
 | |
|   Status = SmmInstallProtocolInterface (
 | |
|              &Handle,
 | |
|              &gEdkiiSmmMemoryProfileGuid,
 | |
|              EFI_NATIVE_INTERFACE,
 | |
|              &mSmmProfileProtocol
 | |
|              );
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get the GUID file name from the file path.
 | |
| 
 | |
|   @param FilePath  File path.
 | |
| 
 | |
|   @return The GUID file name from the file path.
 | |
| 
 | |
| **/
 | |
| EFI_GUID *
 | |
| GetFileNameFromFilePath (
 | |
|   IN EFI_DEVICE_PATH_PROTOCOL   *FilePath
 | |
|   )
 | |
| {
 | |
|   MEDIA_FW_VOL_FILEPATH_DEVICE_PATH     *ThisFilePath;
 | |
|   EFI_GUID                              *FileName;
 | |
| 
 | |
|   FileName = NULL;
 | |
|   if (FilePath != NULL) {
 | |
|     ThisFilePath = (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *) FilePath;
 | |
|     while (!IsDevicePathEnd (ThisFilePath)) {
 | |
|       FileName = EfiGetNameGuidFromFwVolDevicePathNode (ThisFilePath);
 | |
|       if (FileName != NULL) {
 | |
|         break;
 | |
|       }
 | |
|       ThisFilePath = (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *) NextDevicePathNode (ThisFilePath);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FileName;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Register SMM image to SMRAM profile.
 | |
| 
 | |
|   @param DriverEntry    SMM image info.
 | |
|   @param RegisterToDxe  Register image to DXE.
 | |
| 
 | |
|   @return EFI_SUCCESS           Register successfully.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required.
 | |
|   @return EFI_OUT_OF_RESOURCES  No enough resource for this register.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| RegisterSmramProfileImage (
 | |
|   IN EFI_SMM_DRIVER_ENTRY   *DriverEntry,
 | |
|   IN BOOLEAN                RegisterToDxe
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT_DATA       *ContextData;
 | |
|   MEMORY_PROFILE_DRIVER_INFO_DATA   *DriverInfoData;
 | |
|   UINT8                             TempBuffer[sizeof(MEDIA_FW_VOL_FILEPATH_DEVICE_PATH) + sizeof(EFI_DEVICE_PATH_PROTOCOL)];
 | |
|   MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *FilePath;
 | |
| 
 | |
|   if (RegisterToDxe) {
 | |
|     RegisterImageToDxe (
 | |
|       &DriverEntry->FileName,
 | |
|       DriverEntry->ImageBuffer,
 | |
|       EFI_PAGES_TO_SIZE (DriverEntry->NumberOfPage),
 | |
|       EFI_FV_FILETYPE_SMM
 | |
|       );
 | |
|   }
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   FilePath = (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *) TempBuffer;
 | |
|   EfiInitializeFwVolDevicepathNode (FilePath, &DriverEntry->FileName);
 | |
|   SetDevicePathEndNode (FilePath + 1);
 | |
| 
 | |
|   if (!NeedRecordThisDriver ((EFI_DEVICE_PATH_PROTOCOL *) FilePath)) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   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 EFI_OUT_OF_RESOURCES;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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 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;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Unregister image from SMRAM profile.
 | |
| 
 | |
|   @param DriverEntry        SMM image info.
 | |
|   @param UnregisterFromDxe  Unregister image from DXE.
 | |
| 
 | |
|   @return EFI_SUCCESS           Unregister successfully.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required.
 | |
|   @return EFI_NOT_FOUND         The image is not found.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| 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;
 | |
|   UINT8                             TempBuffer[sizeof(MEDIA_FW_VOL_FILEPATH_DEVICE_PATH) + sizeof(EFI_DEVICE_PATH_PROTOCOL)];
 | |
|   MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *FilePath;
 | |
| 
 | |
|   if (UnregisterFromDxe) {
 | |
|     UnregisterImageFromDxe (
 | |
|       &DriverEntry->FileName,
 | |
|       DriverEntry->ImageBuffer,
 | |
|       EFI_PAGES_TO_SIZE (DriverEntry->NumberOfPage)
 | |
|       );
 | |
|   }
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   FilePath = (MEDIA_FW_VOL_FILEPATH_DEVICE_PATH *) TempBuffer;
 | |
|   EfiInitializeFwVolDevicepathNode (FilePath, &DriverEntry->FileName);
 | |
|   SetDevicePathEndNode (FilePath + 1);
 | |
| 
 | |
|   if (!NeedRecordThisDriver ((EFI_DEVICE_PATH_PROTOCOL *) FilePath)) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   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 EFI_NOT_FOUND;
 | |
|   }
 | |
| 
 | |
|   ContextData->Context.TotalImageSize -= DriverInfoData->DriverInfo.ImageSize;
 | |
| 
 | |
|   // Keep the ImageBase for RVA calculation in Application.
 | |
|   //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 EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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.
 | |
|   @param ActionString   String for memory profile action.
 | |
| 
 | |
|   @return EFI_SUCCESS           Memory profile is updated.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required.
 | |
|   @return EFI_OUT_OF_RESOURCES  No enough resource to update memory profile for allocate action.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| SmmCoreUpdateProfileAllocate (
 | |
|   IN PHYSICAL_ADDRESS       CallerAddress,
 | |
|   IN MEMORY_PROFILE_ACTION  Action,
 | |
|   IN EFI_MEMORY_TYPE        MemoryType,
 | |
|   IN UINTN                  Size,
 | |
|   IN VOID                   *Buffer,
 | |
|   IN CHAR8                  *ActionString OPTIONAL
 | |
|   )
 | |
| {
 | |
|   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;
 | |
|   MEMORY_PROFILE_ACTION             BasicAction;
 | |
|   UINTN                             ActionStringSize;
 | |
|   UINTN                             ActionStringOccupiedSize;
 | |
| 
 | |
|   BasicAction = Action & MEMORY_PROFILE_ACTION_BASIC_MASK;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   DriverInfoData = GetMemoryProfileDriverInfoFromAddress (ContextData, CallerAddress);
 | |
|   if (DriverInfoData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   ActionStringSize = 0;
 | |
|   ActionStringOccupiedSize = 0;
 | |
|   if (ActionString != NULL) {
 | |
|     ActionStringSize = AsciiStrSize (ActionString);
 | |
|     ActionStringOccupiedSize = GET_OCCUPIED_SIZE (ActionStringSize, sizeof (UINT64));
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Use SmmInternalAllocatePool() that will not update profile for this AllocatePool action.
 | |
|   //
 | |
|   AllocInfoData = NULL;
 | |
|   Status = SmmInternalAllocatePool (
 | |
|              EfiRuntimeServicesData,
 | |
|              sizeof (*AllocInfoData) + ActionStringSize,
 | |
|              (VOID **) &AllocInfoData
 | |
|              );
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return EFI_OUT_OF_RESOURCES;
 | |
|   }
 | |
|   ASSERT (AllocInfoData != NULL);
 | |
| 
 | |
|   //
 | |
|   // Only update SequenceCount if and only if it is basic action.
 | |
|   //
 | |
|   if (Action == BasicAction) {
 | |
|     ContextData->Context.SequenceCount ++;
 | |
|   }
 | |
| 
 | |
|   AllocInfo = &AllocInfoData->AllocInfo;
 | |
|   AllocInfoData->Signature      = MEMORY_PROFILE_ALLOC_INFO_SIGNATURE;
 | |
|   AllocInfo->Header.Signature   = MEMORY_PROFILE_ALLOC_INFO_SIGNATURE;
 | |
|   AllocInfo->Header.Length      = (UINT16) (sizeof (MEMORY_PROFILE_ALLOC_INFO) + ActionStringOccupiedSize);
 | |
|   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;
 | |
|   if (ActionString != NULL) {
 | |
|     AllocInfo->ActionStringOffset = (UINT16) sizeof (MEMORY_PROFILE_ALLOC_INFO);
 | |
|     AllocInfoData->ActionString = (CHAR8 *) (AllocInfoData + 1);
 | |
|     CopyMem (AllocInfoData->ActionString, ActionString, ActionStringSize);
 | |
|   } else {
 | |
|     AllocInfo->ActionStringOffset = 0;
 | |
|     AllocInfoData->ActionString = NULL;
 | |
|   }
 | |
| 
 | |
|   InsertTailList (DriverInfoData->AllocInfoList, &AllocInfoData->Link);
 | |
| 
 | |
|   Context = &ContextData->Context;
 | |
|   DriverInfo = &DriverInfoData->DriverInfo;
 | |
|   DriverInfo->AllocRecordCount ++;
 | |
| 
 | |
|   //
 | |
|   // Update summary if and only if it is basic action.
 | |
|   //
 | |
|   if (Action == BasicAction) {
 | |
|     ProfileMemoryIndex = GetProfileMemoryIndex (MemoryType);
 | |
| 
 | |
|     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];
 | |
|     }
 | |
| 
 | |
|     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];
 | |
|     }
 | |
| 
 | |
|     SmramProfileUpdateFreePages (ContextData);
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get memory profile alloc info from memory profile
 | |
| 
 | |
|   @param DriverInfoData     Driver info
 | |
|   @param BasicAction        This Free basic 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              BasicAction,
 | |
|   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 & MEMORY_PROFILE_ACTION_BASIC_MASK) != BasicAction) {
 | |
|       continue;
 | |
|     }
 | |
|     switch (BasicAction) {
 | |
|       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.
 | |
| 
 | |
|   @return EFI_SUCCESS           Memory profile is updated.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported.
 | |
|   @return EFI_NOT_FOUND         No matched allocate info found for free action.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| 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;
 | |
|   MEMORY_PROFILE_ACTION            BasicAction;
 | |
|   BOOLEAN                          Found;
 | |
| 
 | |
|   BasicAction = Action & MEMORY_PROFILE_ACTION_BASIC_MASK;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   DriverInfoData = GetMemoryProfileDriverInfoFromAddress (ContextData, CallerAddress);
 | |
| 
 | |
|   //
 | |
|   // Do not return if DriverInfoData == NULL here,
 | |
|   // because driver A might free memory allocated by driver B.
 | |
|   //
 | |
| 
 | |
|   //
 | |
|   // Need use do-while loop to find all possible record,
 | |
|   // because one address might be recorded multiple times.
 | |
|   //
 | |
|   Found = FALSE;
 | |
|   AllocInfoData = NULL;
 | |
|   do {
 | |
|     if (DriverInfoData != NULL) {
 | |
|       switch (BasicAction) {
 | |
|         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 (BasicAction) {
 | |
|           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) {
 | |
|         //
 | |
|         // If (!Found), no matched allocate info is found for this free action.
 | |
|         // It is because the specified memory type allocate actions have been filtered by
 | |
|         // CoreNeedRecordProfile(), but free actions have no memory type information,
 | |
|         // they can not be filtered by CoreNeedRecordProfile(). Then, they will be
 | |
|         // filtered here.
 | |
|         //
 | |
|         // If (Found), it is normal exit path.
 | |
|         return (Found ? EFI_SUCCESS : EFI_NOT_FOUND);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     ASSERT (DriverInfoData != NULL);
 | |
|     ASSERT (AllocInfoData != NULL);
 | |
| 
 | |
|     Found = TRUE;
 | |
| 
 | |
|     Context = &ContextData->Context;
 | |
|     DriverInfo = &DriverInfoData->DriverInfo;
 | |
|     AllocInfo = &AllocInfoData->AllocInfo;
 | |
| 
 | |
|     DriverInfo->AllocRecordCount --;
 | |
|     //
 | |
|     // Update summary if and only if it is basic action.
 | |
|     //
 | |
|     if (AllocInfo->Action == (AllocInfo->Action & MEMORY_PROFILE_ACTION_BASIC_MASK)) {
 | |
|       ProfileMemoryIndex = GetProfileMemoryIndex (AllocInfo->MemoryType);
 | |
| 
 | |
|       Context->CurrentTotalUsage -= AllocInfo->Size;
 | |
|       Context->CurrentTotalUsageByType[ProfileMemoryIndex] -= AllocInfo->Size;
 | |
| 
 | |
|       DriverInfo->CurrentUsage -= AllocInfo->Size;
 | |
|       DriverInfo->CurrentUsageByType[ProfileMemoryIndex] -= AllocInfo->Size;
 | |
|     }
 | |
| 
 | |
|     RemoveEntryList (&AllocInfoData->Link);
 | |
| 
 | |
|     if (BasicAction == MemoryProfileActionFreePages) {
 | |
|       if (AllocInfo->Buffer != (PHYSICAL_ADDRESS) (UINTN) Buffer) {
 | |
|         SmmCoreUpdateProfileAllocate (
 | |
|           AllocInfo->CallerAddress,
 | |
|           AllocInfo->Action,
 | |
|           AllocInfo->MemoryType,
 | |
|           (UINTN) ((PHYSICAL_ADDRESS) (UINTN) Buffer - AllocInfo->Buffer),
 | |
|           (VOID *) (UINTN) AllocInfo->Buffer,
 | |
|           AllocInfoData->ActionString
 | |
|           );
 | |
|       }
 | |
|       if (AllocInfo->Buffer + AllocInfo->Size != ((PHYSICAL_ADDRESS) (UINTN) Buffer + Size)) {
 | |
|         SmmCoreUpdateProfileAllocate (
 | |
|           AllocInfo->CallerAddress,
 | |
|           AllocInfo->Action,
 | |
|           AllocInfo->MemoryType,
 | |
|           (UINTN) ((AllocInfo->Buffer + AllocInfo->Size) - ((PHYSICAL_ADDRESS) (UINTN) Buffer + Size)),
 | |
|           (VOID *) ((UINTN) Buffer + Size),
 | |
|           AllocInfoData->ActionString
 | |
|           );
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     // Use SmmInternalFreePool() that will not update profile for this FreePool action.
 | |
|     //
 | |
|     SmmInternalFreePool (AllocInfoData);
 | |
|   } while (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.
 | |
|                         EfiMaxMemoryType means the MemoryType is unknown.
 | |
|   @param Size           Buffer size.
 | |
|   @param Buffer         Buffer address.
 | |
|   @param ActionString   String for memory profile action.
 | |
|                         Only needed for user defined allocate action.
 | |
| 
 | |
|   @return EFI_SUCCESS           Memory profile is updated.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required,
 | |
|                                 or memory profile for the memory type is not required.
 | |
|   @return EFI_ACCESS_DENIED     It is during memory profile data getting.
 | |
|   @return EFI_ABORTED           Memory profile recording is not enabled.
 | |
|   @return EFI_OUT_OF_RESOURCES  No enough resource to update memory profile for allocate action.
 | |
|   @return EFI_NOT_FOUND         No matched allocate info found for free action.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| 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,
 | |
|   IN CHAR8                  *ActionString OPTIONAL
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                    Status;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA   *ContextData;
 | |
|   MEMORY_PROFILE_ACTION         BasicAction;
 | |
| 
 | |
|   if (!IS_SMRAM_PROFILE_ENABLED) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   if (mSmramProfileGettingStatus) {
 | |
|     return EFI_ACCESS_DENIED;
 | |
|   }
 | |
| 
 | |
|   if (!mSmramProfileRecordingEnable) {
 | |
|     return EFI_ABORTED;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Get the basic action to know how to process the record
 | |
|   //
 | |
|   BasicAction = Action & MEMORY_PROFILE_ACTION_BASIC_MASK;
 | |
| 
 | |
|   //
 | |
|   // Free operations have no memory type information, so skip the check.
 | |
|   //
 | |
|   if ((BasicAction == MemoryProfileActionAllocatePages) || (BasicAction == MemoryProfileActionAllocatePool)) {
 | |
|     //
 | |
|     // Only record limited MemoryType.
 | |
|     //
 | |
|     if (!SmmCoreNeedRecordProfile (MemoryType)) {
 | |
|       return EFI_UNSUPPORTED;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   switch (BasicAction) {
 | |
|     case MemoryProfileActionAllocatePages:
 | |
|       Status = SmmCoreUpdateProfileAllocate (CallerAddress, Action, MemoryType, Size, Buffer, ActionString);
 | |
|       break;
 | |
|     case MemoryProfileActionFreePages:
 | |
|       Status = SmmCoreUpdateProfileFree (CallerAddress, Action, Size, Buffer);
 | |
|       break;
 | |
|     case MemoryProfileActionAllocatePool:
 | |
|       Status = SmmCoreUpdateProfileAllocate (CallerAddress, Action, MemoryType, Size, Buffer, ActionString);
 | |
|       break;
 | |
|     case MemoryProfileActionFreePool:
 | |
|       Status = SmmCoreUpdateProfileFree (CallerAddress, Action, 0, Buffer);
 | |
|       break;
 | |
|     default:
 | |
|       ASSERT (FALSE);
 | |
|       Status = EFI_UNSUPPORTED;
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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;
 | |
|   MEMORY_PROFILE_ALLOC_INFO_DATA    *AllocInfoData;
 | |
|   LIST_ENTRY                        *DriverInfoList;
 | |
|   LIST_ENTRY                        *DriverLink;
 | |
|   LIST_ENTRY                        *AllocInfoList;
 | |
|   LIST_ENTRY                        *AllocLink;
 | |
|   UINTN                             TotalSize;
 | |
|   LIST_ENTRY                        *Node;
 | |
|   LIST_ENTRY                        *FreePageList;
 | |
|   LIST_ENTRY                        *FreePoolList;
 | |
|   FREE_POOL_HEADER                  *Pool;
 | |
|   UINTN                             PoolListIndex;
 | |
|   UINTN                             Index;
 | |
|   UINTN                             SmmPoolTypeIndex;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   TotalSize = sizeof (MEMORY_PROFILE_CONTEXT);
 | |
| 
 | |
|   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 += DriverInfoData->DriverInfo.Header.Length;
 | |
| 
 | |
|     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
 | |
|                         );
 | |
|       TotalSize += AllocInfoData->AllocInfo.Header.Length;
 | |
|     }
 | |
|   }
 | |
| 
 | |
| 
 | |
|   Index = 0;
 | |
|   FreePageList = &mSmmMemoryMap;
 | |
|   for (Node = FreePageList->BackLink;
 | |
|        Node != FreePageList;
 | |
|        Node = Node->BackLink) {
 | |
|     Index++;
 | |
|   }
 | |
|   for (SmmPoolTypeIndex = 0; SmmPoolTypeIndex < SmmPoolTypeMax; SmmPoolTypeIndex++) {
 | |
|     for (PoolListIndex = 0; PoolListIndex < MAX_POOL_INDEX; PoolListIndex++) {
 | |
|       FreePoolList = &mSmmPoolLists[SmmPoolTypeIndex][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.
 | |
|   @param ProfileSize    On input, profile buffer size.
 | |
|                         On output, actual profile data size copied.
 | |
|   @param ProfileOffset  On input, profile buffer offset to copy.
 | |
|                         On output, next time profile buffer offset to copy.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileCopyData (
 | |
|   OUT VOID      *ProfileBuffer,
 | |
|   IN OUT UINT64 *ProfileSize,
 | |
|   IN OUT UINT64 *ProfileOffset
 | |
|   )
 | |
| {
 | |
|   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;
 | |
|   UINT64                          Offset;
 | |
|   UINT64                          RemainingSize;
 | |
|   UINTN                           PdbSize;
 | |
|   UINTN                           ActionStringSize;
 | |
|   UINTN                           SmmPoolTypeIndex;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   RemainingSize = *ProfileSize;
 | |
|   Offset = 0;
 | |
| 
 | |
|   if (*ProfileOffset < sizeof (MEMORY_PROFILE_CONTEXT)) {
 | |
|     if (RemainingSize >= sizeof (MEMORY_PROFILE_CONTEXT)) {
 | |
|       Context = ProfileBuffer;
 | |
|       CopyMem (Context, &ContextData->Context, sizeof (MEMORY_PROFILE_CONTEXT));
 | |
|       RemainingSize -= sizeof (MEMORY_PROFILE_CONTEXT);
 | |
|       ProfileBuffer = (UINT8 *) ProfileBuffer + sizeof (MEMORY_PROFILE_CONTEXT);
 | |
|     } else {
 | |
|       goto Done;
 | |
|     }
 | |
|   }
 | |
|   Offset += sizeof (MEMORY_PROFILE_CONTEXT);
 | |
| 
 | |
|   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 (*ProfileOffset < (Offset + DriverInfoData->DriverInfo.Header.Length)) {
 | |
|       if (RemainingSize >= DriverInfoData->DriverInfo.Header.Length) {
 | |
|         DriverInfo = ProfileBuffer;
 | |
|         CopyMem (DriverInfo, &DriverInfoData->DriverInfo, sizeof (MEMORY_PROFILE_DRIVER_INFO));
 | |
|         if (DriverInfo->PdbStringOffset != 0) {
 | |
|           PdbSize = AsciiStrSize (DriverInfoData->PdbString);
 | |
|           CopyMem ((VOID *) ((UINTN) DriverInfo + DriverInfo->PdbStringOffset), DriverInfoData->PdbString, PdbSize);
 | |
|         }
 | |
|         RemainingSize -= DriverInfo->Header.Length;
 | |
|         ProfileBuffer = (UINT8 *) ProfileBuffer + DriverInfo->Header.Length;
 | |
|       } else {
 | |
|         goto Done;
 | |
|       }
 | |
|     }
 | |
|     Offset += DriverInfoData->DriverInfo.Header.Length;
 | |
| 
 | |
|     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
 | |
|                         );
 | |
|       if (*ProfileOffset < (Offset + AllocInfoData->AllocInfo.Header.Length)) {
 | |
|         if (RemainingSize >= AllocInfoData->AllocInfo.Header.Length) {
 | |
|           AllocInfo = ProfileBuffer;
 | |
|           CopyMem (AllocInfo, &AllocInfoData->AllocInfo, sizeof (MEMORY_PROFILE_ALLOC_INFO));
 | |
|           if (AllocInfo->ActionStringOffset) {
 | |
|             ActionStringSize = AsciiStrSize (AllocInfoData->ActionString);
 | |
|             CopyMem ((VOID *) ((UINTN) AllocInfo + AllocInfo->ActionStringOffset), AllocInfoData->ActionString, ActionStringSize);
 | |
|           }
 | |
|           RemainingSize -= AllocInfo->Header.Length;
 | |
|           ProfileBuffer = (UINT8 *) ProfileBuffer + AllocInfo->Header.Length;
 | |
|         } else {
 | |
|           goto Done;
 | |
|         }
 | |
|       }
 | |
|       Offset += AllocInfoData->AllocInfo.Header.Length;
 | |
|     }
 | |
|   }
 | |
| 
 | |
| 
 | |
|   if (*ProfileOffset < (Offset + sizeof (MEMORY_PROFILE_FREE_MEMORY))) {
 | |
|     if (RemainingSize >= sizeof (MEMORY_PROFILE_FREE_MEMORY)) {
 | |
|       FreeMemory = ProfileBuffer;
 | |
|       CopyMem (FreeMemory, &mSmramFreeMemory, sizeof (MEMORY_PROFILE_FREE_MEMORY));
 | |
|       Index = 0;
 | |
|       FreePageList = &mSmmMemoryMap;
 | |
|       for (Node = FreePageList->BackLink;
 | |
|            Node != FreePageList;
 | |
|            Node = Node->BackLink) {
 | |
|         Index++;
 | |
|       }
 | |
|       for (SmmPoolTypeIndex = 0; SmmPoolTypeIndex < SmmPoolTypeMax; SmmPoolTypeIndex++) {
 | |
|         for (PoolListIndex = 0; PoolListIndex < MAX_POOL_INDEX; PoolListIndex++) {
 | |
|           FreePoolList = &mSmmPoolLists[SmmPoolTypeIndex][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) {
 | |
|               Index++;
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       FreeMemory->FreeMemoryEntryCount = Index;
 | |
| 
 | |
|       RemainingSize -= sizeof (MEMORY_PROFILE_FREE_MEMORY);
 | |
|       ProfileBuffer = (UINT8 *) ProfileBuffer + sizeof (MEMORY_PROFILE_FREE_MEMORY);
 | |
|     } else {
 | |
|       goto Done;
 | |
|     }
 | |
|   }
 | |
|   Offset += sizeof (MEMORY_PROFILE_FREE_MEMORY);
 | |
|   FreePageList = &mSmmMemoryMap;
 | |
|   for (Node = FreePageList->BackLink;
 | |
|        Node != FreePageList;
 | |
|        Node = Node->BackLink) {
 | |
|     if (*ProfileOffset < (Offset + sizeof (MEMORY_PROFILE_DESCRIPTOR))) {
 | |
|       if (RemainingSize >= sizeof (MEMORY_PROFILE_DESCRIPTOR)) {
 | |
|         Pages = BASE_CR (Node, FREE_PAGE_LIST, Link);
 | |
|         MemoryProfileDescriptor = ProfileBuffer;
 | |
|         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);
 | |
| 
 | |
|         RemainingSize -= sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|         ProfileBuffer = (UINT8 *) ProfileBuffer + sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|       } else {
 | |
|         goto Done;
 | |
|       }
 | |
|     }
 | |
|     Offset += sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|   }
 | |
|   for (SmmPoolTypeIndex = 0; SmmPoolTypeIndex < SmmPoolTypeMax; SmmPoolTypeIndex++) {
 | |
|     for (PoolListIndex = 0; PoolListIndex < MAX_POOL_INDEX; PoolListIndex++) {
 | |
|       FreePoolList = &mSmmPoolLists[SmmPoolTypeIndex][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) {
 | |
|           if (*ProfileOffset < (Offset + sizeof (MEMORY_PROFILE_DESCRIPTOR))) {
 | |
|             if (RemainingSize >= sizeof (MEMORY_PROFILE_DESCRIPTOR)) {
 | |
|               MemoryProfileDescriptor = ProfileBuffer;
 | |
|               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;
 | |
| 
 | |
|               RemainingSize -= sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|               ProfileBuffer = (UINT8 *) ProfileBuffer + sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|             } else {
 | |
|               goto Done;
 | |
|             }
 | |
|           }
 | |
|           Offset += sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (*ProfileOffset < (Offset + sizeof (MEMORY_PROFILE_MEMORY_RANGE))) {
 | |
|     if (RemainingSize >= sizeof (MEMORY_PROFILE_MEMORY_RANGE)) {
 | |
|       MemoryRange = ProfileBuffer;
 | |
|       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;
 | |
| 
 | |
|       RemainingSize -= sizeof (MEMORY_PROFILE_MEMORY_RANGE);
 | |
|       ProfileBuffer = (UINT8 *) ProfileBuffer + sizeof (MEMORY_PROFILE_MEMORY_RANGE);
 | |
|     } else {
 | |
|       goto Done;
 | |
|     }
 | |
|   }
 | |
|   Offset += sizeof (MEMORY_PROFILE_MEMORY_RANGE);
 | |
|   for (Index = 0; Index < mFullSmramRangeCount; Index++) {
 | |
|     if (*ProfileOffset < (Offset + sizeof (MEMORY_PROFILE_DESCRIPTOR))) {
 | |
|       if (RemainingSize >= sizeof (MEMORY_PROFILE_DESCRIPTOR)) {
 | |
|         MemoryProfileDescriptor = ProfileBuffer;
 | |
|         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;
 | |
| 
 | |
|         RemainingSize -= sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|         ProfileBuffer = (UINT8 *) ProfileBuffer + sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|       } else {
 | |
|         goto Done;
 | |
|       }
 | |
|     }
 | |
|     Offset += sizeof (MEMORY_PROFILE_DESCRIPTOR);
 | |
|   }
 | |
| 
 | |
| Done:
 | |
|   //
 | |
|   // On output, actual profile data size copied.
 | |
|   //
 | |
|   *ProfileSize -= RemainingSize;
 | |
|   //
 | |
|   // On output, next time profile buffer offset to copy.
 | |
|   //
 | |
|   *ProfileOffset = Offset;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get memory profile data.
 | |
| 
 | |
|   @param[in]      This              The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in, out] ProfileSize       On entry, points to the size in bytes of the ProfileBuffer.
 | |
|                                     On return, points to the size of the data returned in ProfileBuffer.
 | |
|   @param[out]     ProfileBuffer     Profile buffer.
 | |
|                       
 | |
|   @return EFI_SUCCESS               Get the memory profile data successfully.
 | |
|   @return EFI_UNSUPPORTED           Memory profile is unsupported.
 | |
|   @return EFI_BUFFER_TO_SMALL       The ProfileSize is too small for the resulting data. 
 | |
|                                     ProfileSize is updated with the size required.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolGetData (
 | |
|   IN     EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN OUT UINT64                             *ProfileSize,
 | |
|      OUT VOID                               *ProfileBuffer
 | |
|   )
 | |
| {
 | |
|   UINT64                                Size;
 | |
|   UINT64                                Offset;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA           *ContextData;
 | |
|   BOOLEAN                               SmramProfileGettingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   SmramProfileGettingStatus = mSmramProfileGettingStatus;
 | |
|   mSmramProfileGettingStatus = TRUE;
 | |
| 
 | |
|   Size = SmramProfileGetDataSize ();
 | |
| 
 | |
|   if (*ProfileSize < Size) {
 | |
|     *ProfileSize = Size;
 | |
|     mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
|     return EFI_BUFFER_TOO_SMALL;
 | |
|   }
 | |
| 
 | |
|   Offset = 0;
 | |
|   SmramProfileCopyData (ProfileBuffer, &Size, &Offset);
 | |
|   *ProfileSize = Size;
 | |
| 
 | |
|   mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Register image to memory profile.
 | |
| 
 | |
|   @param[in] This               The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in] FilePath           File path of the image.
 | |
|   @param[in] ImageBase          Image base address.
 | |
|   @param[in] ImageSize          Image size.
 | |
|   @param[in] FileType           File type of the image.
 | |
| 
 | |
|   @return EFI_SUCCESS           Register successfully.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required.
 | |
|   @return EFI_OUT_OF_RESOURCES  No enough resource for this register.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolRegisterImage (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN EFI_DEVICE_PATH_PROTOCOL           *FilePath,
 | |
|   IN PHYSICAL_ADDRESS                   ImageBase,
 | |
|   IN UINT64                             ImageSize,
 | |
|   IN EFI_FV_FILETYPE                    FileType
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   EFI_SMM_DRIVER_ENTRY              DriverEntry;
 | |
|   VOID                              *EntryPointInImage;
 | |
|   EFI_GUID                          *Name;
 | |
|   
 | |
|   ZeroMem (&DriverEntry, sizeof (DriverEntry));
 | |
|   Name = GetFileNameFromFilePath (FilePath);
 | |
|   if (Name != NULL) {
 | |
|     CopyMem (&DriverEntry.FileName, Name, sizeof (EFI_GUID));
 | |
|   }
 | |
|   DriverEntry.ImageBuffer = ImageBase;
 | |
|   DriverEntry.NumberOfPage = EFI_SIZE_TO_PAGES ((UINTN) ImageSize);
 | |
|   Status = InternalPeCoffGetEntryPoint ((VOID *) (UINTN) DriverEntry.ImageBuffer, &EntryPointInImage);
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
|   DriverEntry.ImageEntryPoint = (PHYSICAL_ADDRESS) (UINTN) EntryPointInImage;
 | |
| 
 | |
|   return RegisterSmramProfileImage (&DriverEntry, FALSE);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Unregister image from memory profile.
 | |
| 
 | |
|   @param[in] This               The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in] FilePath           File path of the image.
 | |
|   @param[in] ImageBase          Image base address.
 | |
|   @param[in] ImageSize          Image size.
 | |
| 
 | |
|   @return EFI_SUCCESS           Unregister successfully.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required.
 | |
|   @return EFI_NOT_FOUND         The image is not found.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolUnregisterImage (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN EFI_DEVICE_PATH_PROTOCOL           *FilePath,
 | |
|   IN PHYSICAL_ADDRESS                   ImageBase,
 | |
|   IN UINT64                             ImageSize
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS                        Status;
 | |
|   EFI_SMM_DRIVER_ENTRY              DriverEntry;
 | |
|   VOID                              *EntryPointInImage;
 | |
|   EFI_GUID                          *Name;
 | |
| 
 | |
|   ZeroMem (&DriverEntry, sizeof (DriverEntry));
 | |
|   Name = GetFileNameFromFilePath (FilePath);
 | |
|   if (Name != NULL) {
 | |
|     CopyMem (&DriverEntry.FileName, Name, sizeof (EFI_GUID));
 | |
|   }
 | |
|   DriverEntry.ImageBuffer = ImageBase;
 | |
|   DriverEntry.NumberOfPage = EFI_SIZE_TO_PAGES ((UINTN) ImageSize);
 | |
|   Status = InternalPeCoffGetEntryPoint ((VOID *) (UINTN) DriverEntry.ImageBuffer, &EntryPointInImage);
 | |
|   ASSERT_EFI_ERROR (Status);
 | |
|   DriverEntry.ImageEntryPoint = (PHYSICAL_ADDRESS) (UINTN) EntryPointInImage;
 | |
| 
 | |
|   return UnregisterSmramProfileImage (&DriverEntry, FALSE);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get memory profile recording state.
 | |
| 
 | |
|   @param[in]  This              The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[out] RecordingState    Recording state.
 | |
| 
 | |
|   @return EFI_SUCCESS           Memory profile recording state is returned.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported.
 | |
|   @return EFI_INVALID_PARAMETER RecordingState is NULL.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolGetRecordingState (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   OUT BOOLEAN                           *RecordingState
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT_DATA           *ContextData;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   if (RecordingState == NULL) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
|   *RecordingState = mSmramProfileRecordingEnable;
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Set memory profile recording state.
 | |
| 
 | |
|   @param[in] This               The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in] RecordingState     Recording state.
 | |
| 
 | |
|   @return EFI_SUCCESS           Set memory profile recording state successfully.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolSetRecordingState (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN BOOLEAN                            RecordingState
 | |
|   )
 | |
| {
 | |
|   MEMORY_PROFILE_CONTEXT_DATA           *ContextData;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   mSmramProfileRecordingEnable = RecordingState;
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Record memory profile of multilevel caller.
 | |
| 
 | |
|   @param[in] This               The EDKII_SMM_MEMORY_PROFILE_PROTOCOL instance.
 | |
|   @param[in] CallerAddress      Address of caller.
 | |
|   @param[in] Action             Memory profile action.
 | |
|   @param[in] MemoryType         Memory type.
 | |
|                                 EfiMaxMemoryType means the MemoryType is unknown.
 | |
|   @param[in] Buffer             Buffer address.
 | |
|   @param[in] Size               Buffer size.
 | |
|   @param[in] ActionString       String for memory profile action.
 | |
|                                 Only needed for user defined allocate action.
 | |
| 
 | |
|   @return EFI_SUCCESS           Memory profile is updated.
 | |
|   @return EFI_UNSUPPORTED       Memory profile is unsupported,
 | |
|                                 or memory profile for the image is not required,
 | |
|                                 or memory profile for the memory type is not required.
 | |
|   @return EFI_ACCESS_DENIED     It is during memory profile data getting.
 | |
|   @return EFI_ABORTED           Memory profile recording is not enabled.
 | |
|   @return EFI_OUT_OF_RESOURCES  No enough resource to update memory profile for allocate action.
 | |
|   @return EFI_NOT_FOUND         No matched allocate info found for free action.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| EFIAPI
 | |
| SmramProfileProtocolRecord (
 | |
|   IN EDKII_SMM_MEMORY_PROFILE_PROTOCOL  *This,
 | |
|   IN PHYSICAL_ADDRESS                   CallerAddress,
 | |
|   IN MEMORY_PROFILE_ACTION              Action,
 | |
|   IN EFI_MEMORY_TYPE                    MemoryType,
 | |
|   IN VOID                               *Buffer,
 | |
|   IN UINTN                              Size,
 | |
|   IN CHAR8                              *ActionString OPTIONAL
 | |
|   )
 | |
| {
 | |
|   return SmmCoreUpdateProfile (CallerAddress, Action, MemoryType, Size, Buffer, ActionString);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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                       SmramProfileGettingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileGettingStatus = mSmramProfileGettingStatus;
 | |
|   mSmramProfileGettingStatus = TRUE;
 | |
| 
 | |
|   SmramProfileParameterGetInfo->ProfileSize = SmramProfileGetDataSize();
 | |
|   SmramProfileParameterGetInfo->Header.ReturnStatus = 0;
 | |
| 
 | |
|   mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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;
 | |
|   UINT64                                    ProfileOffset;
 | |
|   SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA  SmramProfileGetData;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA               *ContextData;
 | |
|   BOOLEAN                                   SmramProfileGettingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileGettingStatus = mSmramProfileGettingStatus;
 | |
|   mSmramProfileGettingStatus = TRUE;
 | |
| 
 | |
| 
 | |
|   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;
 | |
|   }
 | |
| 
 | |
|   ProfileOffset = 0;
 | |
|   SmramProfileCopyData ((VOID *) (UINTN) SmramProfileGetData.ProfileBuffer, &ProfileSize, &ProfileOffset);
 | |
|   SmramProfileParameterGetData->ProfileSize = ProfileSize;
 | |
|   SmramProfileParameterGetData->Header.ReturnStatus = 0;
 | |
| 
 | |
| Done:
 | |
|   mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   SMRAM profile handler to get profile data by offset.
 | |
| 
 | |
|   @param SmramProfileParameterGetDataByOffset   The parameter of SMM profile get data by offset.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| SmramProfileHandlerGetDataByOffset (
 | |
|   IN SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA_BY_OFFSET     *SmramProfileParameterGetDataByOffset
 | |
|   )
 | |
| {
 | |
|   SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA_BY_OFFSET    SmramProfileGetDataByOffset;
 | |
|   MEMORY_PROFILE_CONTEXT_DATA                           *ContextData;
 | |
|   BOOLEAN                                               SmramProfileGettingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileGettingStatus = mSmramProfileGettingStatus;
 | |
|   mSmramProfileGettingStatus = TRUE;
 | |
| 
 | |
| 
 | |
|   CopyMem (&SmramProfileGetDataByOffset, SmramProfileParameterGetDataByOffset, sizeof (SmramProfileGetDataByOffset));
 | |
| 
 | |
|   //
 | |
|   // Sanity check
 | |
|   //
 | |
|   if (!SmmIsBufferOutsideSmmValid ((UINTN) SmramProfileGetDataByOffset.ProfileBuffer, (UINTN) SmramProfileGetDataByOffset.ProfileSize)) {
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerGetDataByOffset: SMM ProfileBuffer in SMRAM or overflow!\n"));
 | |
|     SmramProfileParameterGetDataByOffset->Header.ReturnStatus = (UINT64) (INT64) (INTN) EFI_ACCESS_DENIED;
 | |
|     goto Done;
 | |
|   }
 | |
| 
 | |
|   SmramProfileCopyData ((VOID *) (UINTN) SmramProfileGetDataByOffset.ProfileBuffer, &SmramProfileGetDataByOffset.ProfileSize, &SmramProfileGetDataByOffset.ProfileOffset);
 | |
|   CopyMem (SmramProfileParameterGetDataByOffset, &SmramProfileGetDataByOffset, sizeof (SmramProfileGetDataByOffset));
 | |
|   SmramProfileParameterGetDataByOffset->Header.ReturnStatus = 0;
 | |
| 
 | |
| Done:
 | |
|   mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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;
 | |
| 
 | |
|   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;
 | |
| 
 | |
|   Status = RegisterSmramProfileImage (&DriverEntry, FALSE);
 | |
|   if (!EFI_ERROR (Status)) {
 | |
|     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;
 | |
| 
 | |
|   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;
 | |
| 
 | |
|   Status = UnregisterSmramProfileImage (&DriverEntry, FALSE);
 | |
|   if (!EFI_ERROR (Status)) {
 | |
|     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;
 | |
|   SMRAM_PROFILE_PARAMETER_RECORDING_STATE  *ParameterRecordingState;
 | |
| 
 | |
|   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_GET_PROFILE_DATA_BY_OFFSET:
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerGetDataByOffset\n"));
 | |
|     if (TempCommBufferSize != sizeof (SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA_BY_OFFSET)) {
 | |
|       DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     SmramProfileHandlerGetDataByOffset ((SMRAM_PROFILE_PARAMETER_GET_PROFILE_DATA_BY_OFFSET *) (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;
 | |
|   case SMRAM_PROFILE_COMMAND_GET_RECORDING_STATE:
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerGetRecordingState\n"));
 | |
|     if (TempCommBufferSize != sizeof (SMRAM_PROFILE_PARAMETER_RECORDING_STATE)) {
 | |
|       DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     ParameterRecordingState = (SMRAM_PROFILE_PARAMETER_RECORDING_STATE *) (UINTN) CommBuffer;
 | |
|     ParameterRecordingState->RecordingState = mSmramProfileRecordingEnable;
 | |
|     ParameterRecordingState->Header.ReturnStatus = 0;
 | |
|     break;
 | |
|   case SMRAM_PROFILE_COMMAND_SET_RECORDING_STATE:
 | |
|     DEBUG ((EFI_D_ERROR, "SmramProfileHandlerSetRecordingState\n"));
 | |
|     if (TempCommBufferSize != sizeof (SMRAM_PROFILE_PARAMETER_RECORDING_STATE)) {
 | |
|       DEBUG ((EFI_D_ERROR, "SmramProfileHandler: SMM communication buffer size invalid!\n"));
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
|     ParameterRecordingState = (SMRAM_PROFILE_PARAMETER_RECORDING_STATE *) (UINTN) CommBuffer;
 | |
|     mSmramProfileRecordingEnable = ParameterRecordingState->RecordingState;
 | |
|     ParameterRecordingState->Header.ReturnStatus = 0;
 | |
|     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                       SmramProfileGettingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileGettingStatus = mSmramProfileGettingStatus;
 | |
|   mSmramProfileGettingStatus = TRUE;
 | |
| 
 | |
|   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"));
 | |
| 
 | |
|   mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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                       SmramProfileGettingStatus;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileGettingStatus = mSmramProfileGettingStatus;
 | |
|   mSmramProfileGettingStatus = TRUE;
 | |
| 
 | |
|   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"));
 | |
| 
 | |
|   mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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                       SmramProfileGettingStatus;
 | |
|   UINTN                         SmmPoolTypeIndex;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileGettingStatus = mSmramProfileGettingStatus;
 | |
|   mSmramProfileGettingStatus = TRUE;
 | |
| 
 | |
|   DEBUG ((DEBUG_INFO, "======= SmramProfile begin =======\n"));
 | |
| 
 | |
|   for (SmmPoolTypeIndex = 0; SmmPoolTypeIndex < SmmPoolTypeMax; SmmPoolTypeIndex++) {
 | |
|     for (PoolListIndex = 0; PoolListIndex < MAX_POOL_INDEX; PoolListIndex++) {
 | |
|       DEBUG ((DEBUG_INFO, "FreePoolList(%d)(%d):\n", SmmPoolTypeIndex, PoolListIndex));
 | |
|       FreePoolList = &mSmmPoolLists[SmmPoolTypeIndex][PoolListIndex];
 | |
|       for (Node = FreePoolList->BackLink, Index = 0;
 | |
|            Node != FreePoolList;
 | |
|            Node = Node->BackLink, Index++) {
 | |
|         Pool = BASE_CR (Node, FREE_POOL_HEADER, Link);
 | |
|         DEBUG ((DEBUG_INFO, "  Index - 0x%x\n", Index));
 | |
|         DEBUG ((DEBUG_INFO, "    PhysicalStart - 0x%016lx\n", (PHYSICAL_ADDRESS) (UINTN) Pool));
 | |
|         DEBUG ((DEBUG_INFO, "    Size          - 0x%08x\n", Pool->Header.Size));
 | |
|         DEBUG ((DEBUG_INFO, "    Available     - 0x%02x\n", Pool->Header.Available));
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   DEBUG ((DEBUG_INFO, "======= SmramProfile end =======\n"));
 | |
| 
 | |
|   mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
| }
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED CHAR8 *mSmmActionString[] = {
 | |
|   "SmmUnknown",
 | |
|   "gSmst->SmmAllocatePages",
 | |
|   "gSmst->SmmFreePages",
 | |
|   "gSmst->SmmAllocatePool",
 | |
|   "gSmst->SmmFreePool",
 | |
| };
 | |
| 
 | |
| typedef struct {
 | |
|   MEMORY_PROFILE_ACTION  Action;
 | |
|   CHAR8                 *String;
 | |
| } ACTION_STRING;
 | |
| 
 | |
| ACTION_STRING mExtActionString[] = {
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_PAGES,                    "Lib:AllocatePages"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_RUNTIME_PAGES,            "Lib:AllocateRuntimePages"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_RESERVED_PAGES,           "Lib:AllocateReservedPages"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_FREE_PAGES,                        "Lib:FreePages"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_ALIGNED_PAGES,            "Lib:AllocateAlignedPages"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_ALIGNED_RUNTIME_PAGES,    "Lib:AllocateAlignedRuntimePages"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_ALIGNED_RESERVED_PAGES,   "Lib:AllocateAlignedReservedPages"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_FREE_ALIGNED_PAGES,                "Lib:FreeAlignedPages"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_POOL,                     "Lib:AllocatePool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_RUNTIME_POOL,             "Lib:AllocateRuntimePool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_RESERVED_POOL,            "Lib:AllocateReservedPool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_FREE_POOL,                         "Lib:FreePool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_ZERO_POOL,                "Lib:AllocateZeroPool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_RUNTIME_ZERO_POOL,        "Lib:AllocateRuntimeZeroPool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_RESERVED_ZERO_POOL,       "Lib:AllocateReservedZeroPool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_COPY_POOL,                "Lib:AllocateCopyPool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_RUNTIME_COPY_POOL,        "Lib:AllocateRuntimeCopyPool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_ALLOCATE_RESERVED_COPY_POOL,       "Lib:AllocateReservedCopyPool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_REALLOCATE_POOL,                   "Lib:ReallocatePool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_REALLOCATE_RUNTIME_POOL,           "Lib:ReallocateRuntimePool"},
 | |
|   {MEMORY_PROFILE_ACTION_LIB_REALLOCATE_RESERVED_POOL,          "Lib:ReallocateReservedPool"},
 | |
| };
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED CHAR8 mUserDefinedActionString[] = {"UserDefined-0x80000000"};
 | |
| 
 | |
| typedef struct {
 | |
|   EFI_MEMORY_TYPE   MemoryType;
 | |
|   CHAR8             *MemoryTypeStr;
 | |
| } PROFILE_MEMORY_TYPE_STRING;
 | |
| 
 | |
| GLOBAL_REMOVE_IF_UNREFERENCED PROFILE_MEMORY_TYPE_STRING mMemoryTypeString[] = {
 | |
|   {EfiRuntimeServicesCode, "EfiRuntimeServicesCode"},
 | |
|   {EfiRuntimeServicesData, "EfiRuntimeServicesData"}
 | |
| };
 | |
| 
 | |
| /**
 | |
|   Memory type to string.
 | |
| 
 | |
|   @param[in] MemoryType Memory type.
 | |
| 
 | |
|   @return Pointer to string.
 | |
| 
 | |
| **/
 | |
| CHAR8 *
 | |
| ProfileMemoryTypeToStr (
 | |
|   IN EFI_MEMORY_TYPE    MemoryType
 | |
|   )
 | |
| {
 | |
|   UINTN     Index;
 | |
|   for (Index = 0; Index < ARRAY_SIZE (mMemoryTypeString); Index++) {
 | |
|     if (mMemoryTypeString[Index].MemoryType == MemoryType) {
 | |
|       return mMemoryTypeString[Index].MemoryTypeStr;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return "UnexpectedMemoryType";
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Action to string.
 | |
| 
 | |
|   @param[in] Action                     Profile action.
 | |
| 
 | |
|   @return Pointer to string.
 | |
| 
 | |
| **/
 | |
| CHAR8 *
 | |
| ProfileActionToStr (
 | |
|   IN MEMORY_PROFILE_ACTION  Action
 | |
|   )
 | |
| {
 | |
|   UINTN     Index;
 | |
|   UINTN     ActionStringCount;
 | |
|   CHAR8     **ActionString;
 | |
| 
 | |
|   ActionString = mSmmActionString;
 | |
|   ActionStringCount = ARRAY_SIZE (mSmmActionString);
 | |
| 
 | |
|   if ((UINTN) (UINT32) Action < ActionStringCount) {
 | |
|     return ActionString[Action];
 | |
|   }
 | |
|   for (Index = 0; Index < ARRAY_SIZE (mExtActionString); Index++) {
 | |
|     if (mExtActionString[Index].Action == Action) {
 | |
|       return mExtActionString[Index].String;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return ActionString[0];
 | |
| }
 | |
| 
 | |
| /**
 | |
|   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                           SmramProfileGettingStatus;
 | |
|   UINTN                             TypeIndex;
 | |
| 
 | |
|   ContextData = GetSmramProfileContext ();
 | |
|   if (ContextData == NULL) {
 | |
|     return ;
 | |
|   }
 | |
| 
 | |
|   SmramProfileGettingStatus = mSmramProfileGettingStatus;
 | |
|   mSmramProfileGettingStatus = TRUE;
 | |
| 
 | |
|   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 (%a)\n", TypeIndex, Context->CurrentTotalUsageByType[TypeIndex], ProfileMemoryTypeToStr (TypeIndex)));
 | |
|       DEBUG ((EFI_D_INFO, "  PeakTotalUsage[0x%02x]     - 0x%016lx (%a)\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 (%a)\n", TypeIndex, DriverInfo->CurrentUsageByType[TypeIndex], ProfileMemoryTypeToStr (TypeIndex)));
 | |
|         DEBUG ((EFI_D_INFO, "    PeakUsage[0x%02x]        - 0x%016lx (%a)\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));
 | |
|       if ((AllocInfo->Action & MEMORY_PROFILE_ACTION_USER_DEFINED_MASK) != 0) {
 | |
|         if (AllocInfoData->ActionString != NULL) {
 | |
|           DEBUG ((EFI_D_INFO, "      Action         - 0x%08x (%a)\n", AllocInfo->Action, AllocInfoData->ActionString));
 | |
|         } else {
 | |
|           DEBUG ((EFI_D_INFO, "      Action         - 0x%08x (UserDefined-0x%08x)\n", AllocInfo->Action, AllocInfo->Action));
 | |
|         }
 | |
|       } else {
 | |
|         DEBUG ((EFI_D_INFO, "      Action         - 0x%08x (%a)\n", AllocInfo->Action, ProfileActionToStr (AllocInfo->Action)));
 | |
|       }
 | |
|       DEBUG ((EFI_D_INFO, "      MemoryType     - 0x%08x (%a)\n", AllocInfo->MemoryType, ProfileMemoryTypeToStr (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"));
 | |
| 
 | |
|   mSmramProfileGettingStatus = SmramProfileGettingStatus;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Dump SMRAM infromation.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| DumpSmramInfo (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   DEBUG_CODE (
 | |
|     if (IS_SMRAM_PROFILE_ENABLED) {
 | |
|       DumpSmramProfile ();
 | |
|       DumpFreePagesList ();
 | |
|       DumpFreePoolList ();
 | |
|       DumpSmramRange ();
 | |
|     }
 | |
|   );
 | |
| }
 | |
| 
 |