2011-12-01 02:57:27 +01:00
/** @file
This module install ACPI Firmware Performance Data Table ( FPDT ) .
This module register report status code listener to collect performance data
2018-06-27 15:08:52 +02:00
for Firmware Basic Boot Performance Record and other boot performance records ,
2012-05-08 05:09:54 +02:00
and install FPDT to ACPI table .
2011-12-01 02:57:27 +01:00
2021-06-28 13:50:22 +02:00
Copyright ( c ) 2011 - 2021 , Intel Corporation . All rights reserved . < BR >
2019-04-04 01:05:13 +02:00
SPDX - License - Identifier : BSD - 2 - Clause - Patent
2011-12-01 02:57:27 +01:00
* */
# include <PiDxe.h>
# include <Protocol/ReportStatusCodeHandler.h>
# include <Protocol/AcpiTable.h>
2013-05-16 09:35:30 +02:00
# include <Protocol/LockBox.h>
# include <Protocol/Variable.h>
2021-06-28 13:50:22 +02:00
# include <Protocol/VariablePolicy.h>
2011-12-01 02:57:27 +01:00
# include <Guid/Acpi.h>
# include <Guid/FirmwarePerformance.h>
# include <Library/UefiBootServicesTableLib.h>
# include <Library/UefiRuntimeServicesTableLib.h>
# include <Library/BaseLib.h>
# include <Library/DebugLib.h>
2018-05-22 15:37:58 +02:00
# include <Library/DxeServicesLib.h>
2011-12-01 02:57:27 +01:00
# include <Library/TimerLib.h>
# include <Library/BaseMemoryLib.h>
# include <Library/MemoryAllocationLib.h>
# include <Library/PcdLib.h>
# include <Library/HobLib.h>
2013-05-16 09:35:30 +02:00
# include <Library/LockBoxLib.h>
# include <Library/UefiLib.h>
2021-06-28 13:50:22 +02:00
# include <Library/VariablePolicyHelperLib.h>
# include <Library/PerformanceLib.h>
2011-12-01 02:57:27 +01:00
2021-12-05 23:54:02 +01:00
# define SMM_BOOT_RECORD_COMM_SIZE (OFFSET_OF (EFI_SMM_COMMUNICATE_HEADER, Data) + sizeof(SMM_BOOT_RECORD_COMMUNICATE))
2011-12-01 02:57:27 +01:00
2021-12-05 23:54:02 +01:00
EFI_RSC_HANDLER_PROTOCOL * mRscHandlerProtocol = NULL ;
2011-12-01 02:57:27 +01:00
2021-12-05 23:54:02 +01:00
BOOLEAN mLockBoxReady = FALSE ;
EFI_EVENT mReadyToBootEvent ;
EFI_EVENT mLegacyBootEvent ;
static EFI_EVENT mExitBootServicesEvent ;
UINTN mFirmwarePerformanceTableTemplateKey = 0 ;
BOOLEAN mDxeCoreReportStatusCodeEnable = FALSE ;
2011-12-01 02:57:27 +01:00
2021-12-05 23:54:02 +01:00
BOOT_PERFORMANCE_TABLE * mAcpiBootPerformanceTable = NULL ;
BOOT_PERFORMANCE_TABLE * mReceivedAcpiBootPerformanceTable = NULL ;
S3_PERFORMANCE_TABLE * mAcpiS3PerformanceTable = NULL ;
2011-12-01 02:57:27 +01:00
FIRMWARE_PERFORMANCE_TABLE mFirmwarePerformanceTableTemplate = {
{
EFI_ACPI_5_0_FIRMWARE_PERFORMANCE_DATA_TABLE_SIGNATURE ,
sizeof ( FIRMWARE_PERFORMANCE_TABLE ) ,
2021-12-05 23:54:02 +01:00
EFI_ACPI_5_0_FIRMWARE_PERFORMANCE_DATA_TABLE_REVISION , // Revision
0x00 , // Checksum will be updated at runtime
2011-12-01 02:57:27 +01:00
//
2013-09-16 04:20:52 +02:00
// It is expected that these values will be updated at EntryPoint.
2011-12-01 02:57:27 +01:00
//
2021-12-05 23:54:02 +01:00
{ 0x00 } , // OEM ID is a 6 bytes long field
2013-09-16 04:20:52 +02:00
0x00 , // OEM Table ID(8 bytes long)
0x00 , // OEM Revision
0x00 , // Creator ID
0x00 , // Creator Revision
2011-12-01 02:57:27 +01:00
} ,
//
// Firmware Basic Boot Performance Table Pointer Record.
//
{
{
2021-12-05 23:54:02 +01:00
EFI_ACPI_5_0_FPDT_RECORD_TYPE_FIRMWARE_BASIC_BOOT_POINTER , // Type
2011-12-01 02:57:27 +01:00
sizeof ( EFI_ACPI_5_0_FPDT_BOOT_PERFORMANCE_TABLE_POINTER_RECORD ) , // Length
EFI_ACPI_5_0_FPDT_RECORD_REVISION_FIRMWARE_BASIC_BOOT_POINTER // Revision
} ,
0 , // Reserved
0 // BootPerformanceTablePointer will be updated at runtime.
} ,
//
// S3 Performance Table Pointer Record.
//
{
{
EFI_ACPI_5_0_FPDT_RECORD_TYPE_S3_PERFORMANCE_TABLE_POINTER , // Type
sizeof ( EFI_ACPI_5_0_FPDT_S3_PERFORMANCE_TABLE_POINTER_RECORD ) , // Length
EFI_ACPI_5_0_FPDT_RECORD_REVISION_S3_PERFORMANCE_TABLE_POINTER // Revision
} ,
0 , // Reserved
0 // S3PerformanceTablePointer will be updated at runtime.
}
} ;
2021-12-05 23:54:02 +01:00
BOOT_PERFORMANCE_TABLE mBootPerformanceTableTemplate = {
2011-12-01 02:57:27 +01:00
{
EFI_ACPI_5_0_FPDT_BOOT_PERFORMANCE_TABLE_SIGNATURE ,
sizeof ( BOOT_PERFORMANCE_TABLE )
} ,
{
{
EFI_ACPI_5_0_FPDT_RUNTIME_RECORD_TYPE_FIRMWARE_BASIC_BOOT , // Type
sizeof ( EFI_ACPI_5_0_FPDT_FIRMWARE_BASIC_BOOT_RECORD ) , // Length
EFI_ACPI_5_0_FPDT_RUNTIME_RECORD_REVISION_FIRMWARE_BASIC_BOOT // Revision
} ,
0 , // Reserved
//
// These values will be updated at runtime.
//
0 , // ResetEnd
0 , // OsLoaderLoadImageStart
0 , // OsLoaderStartImageStart
0 , // ExitBootServicesEntry
0 // ExitBootServicesExit
}
} ;
2021-12-05 23:54:02 +01:00
S3_PERFORMANCE_TABLE mS3PerformanceTableTemplate = {
2011-12-01 02:57:27 +01:00
{
EFI_ACPI_5_0_FPDT_S3_PERFORMANCE_TABLE_SIGNATURE ,
sizeof ( S3_PERFORMANCE_TABLE )
} ,
{
{
EFI_ACPI_5_0_FPDT_RUNTIME_RECORD_TYPE_S3_RESUME , // Type
sizeof ( EFI_ACPI_5_0_FPDT_S3_RESUME_RECORD ) , // Length
EFI_ACPI_5_0_FPDT_RUNTIME_RECORD_REVISION_S3_RESUME // Revision
} ,
//
// These values will be updated by Firmware Performance PEIM.
//
0 , // ResumeCount
0 , // FullResume
0 // AverageResume
} ,
{
{
EFI_ACPI_5_0_FPDT_RUNTIME_RECORD_TYPE_S3_SUSPEND , // Type
sizeof ( EFI_ACPI_5_0_FPDT_S3_SUSPEND_RECORD ) , // Length
EFI_ACPI_5_0_FPDT_RUNTIME_RECORD_REVISION_S3_SUSPEND // Revision
} ,
//
// These values will be updated bye Firmware Performance SMM driver.
//
0 , // SuspendStart
0 // SuspendEnd
}
} ;
/**
This function calculates and updates an UINT8 checksum .
@ param [ in ] Buffer Pointer to buffer to checksum
@ param [ in ] Size Number of bytes to checksum
* */
VOID
FpdtAcpiTableChecksum (
2021-12-05 23:54:02 +01:00
IN UINT8 * Buffer ,
IN UINTN Size
2011-12-01 02:57:27 +01:00
)
{
2021-12-05 23:54:02 +01:00
UINTN ChecksumOffset ;
2011-12-01 02:57:27 +01:00
ChecksumOffset = OFFSET_OF ( EFI_ACPI_DESCRIPTION_HEADER , Checksum ) ;
//
// Set checksum to 0 first.
//
Buffer [ ChecksumOffset ] = 0 ;
//
// Update checksum value.
//
Buffer [ ChecksumOffset ] = CalculateCheckSum8 ( Buffer , Size ) ;
}
2013-05-16 09:35:30 +02:00
/**
Callback function upon VariableArchProtocol and LockBoxProtocol
to allocate S3 performance table memory and save the pointer to LockBox .
@ param [ in ] Event Event whose notification function is being invoked .
@ param [ in ] Context Pointer to the notification function ' s context .
* */
VOID
EFIAPI
FpdtAllocateS3PerformanceTableMemory (
2021-12-05 23:54:02 +01:00
IN EFI_EVENT Event ,
IN VOID * Context
2013-05-16 09:35:30 +02:00
)
{
2021-12-05 23:54:02 +01:00
EFI_STATUS Status ;
VOID * Interface ;
FIRMWARE_PERFORMANCE_VARIABLE PerformanceVariable ;
UINTN Size ;
EFI_PHYSICAL_ADDRESS S3PerformanceTablePointer ;
2013-05-16 09:35:30 +02:00
if ( mLockBoxReady & & ( mAcpiS3PerformanceTable ! = NULL ) ) {
//
// The memory for S3 performance table should have been ready,
// and the pointer should have been saved to LockBox, just return.
//
return ;
}
if ( ! mLockBoxReady ) {
Status = gBS - > LocateProtocol ( & gEfiLockBoxProtocolGuid , NULL , & Interface ) ;
if ( ! EFI_ERROR ( Status ) ) {
//
// LockBox services has been ready.
//
mLockBoxReady = TRUE ;
}
}
if ( mAcpiS3PerformanceTable = = NULL ) {
Status = gBS - > LocateProtocol ( & gEfiVariableArchProtocolGuid , NULL , & Interface ) ;
if ( ! EFI_ERROR ( Status ) ) {
//
// Try to allocate the same runtime buffer as last time boot.
//
ZeroMem ( & PerformanceVariable , sizeof ( PerformanceVariable ) ) ;
2021-12-05 23:54:02 +01:00
Size = sizeof ( PerformanceVariable ) ;
2013-05-16 09:35:30 +02:00
Status = gRT - > GetVariable (
EFI_FIRMWARE_PERFORMANCE_VARIABLE_NAME ,
& gEfiFirmwarePerformanceGuid ,
NULL ,
& Size ,
& PerformanceVariable
) ;
if ( ! EFI_ERROR ( Status ) ) {
Status = gBS - > AllocatePages (
AllocateAddress ,
EfiReservedMemoryType ,
EFI_SIZE_TO_PAGES ( sizeof ( S3_PERFORMANCE_TABLE ) ) ,
& PerformanceVariable . S3PerformanceTablePointer
) ;
if ( ! EFI_ERROR ( Status ) ) {
2021-12-05 23:54:02 +01:00
mAcpiS3PerformanceTable = ( S3_PERFORMANCE_TABLE * ) ( UINTN ) PerformanceVariable . S3PerformanceTablePointer ;
2013-05-16 09:35:30 +02:00
}
}
2021-12-05 23:54:02 +01:00
2013-05-16 09:35:30 +02:00
if ( mAcpiS3PerformanceTable = = NULL ) {
//
// Fail to allocate at specified address, continue to allocate at any address.
//
2021-12-05 23:54:02 +01:00
mAcpiS3PerformanceTable = ( S3_PERFORMANCE_TABLE * ) AllocatePeiAccessiblePages (
EfiReservedMemoryType ,
EFI_SIZE_TO_PAGES ( sizeof ( S3_PERFORMANCE_TABLE ) )
) ;
2013-05-16 09:35:30 +02:00
}
2021-12-05 23:54:02 +01:00
2021-11-17 04:21:29 +01:00
DEBUG ( ( DEBUG_INFO , " FPDT: ACPI S3 Performance Table address = 0x%x \n " , mAcpiS3PerformanceTable ) ) ;
2013-05-16 09:35:30 +02:00
if ( mAcpiS3PerformanceTable ! = NULL ) {
CopyMem ( mAcpiS3PerformanceTable , & mS3PerformanceTableTemplate , sizeof ( mS3PerformanceTableTemplate ) ) ;
}
}
}
if ( mLockBoxReady & & ( mAcpiS3PerformanceTable ! = NULL ) ) {
//
// If LockBox services has been ready and memory for FPDT S3 performance table has been allocated,
// save the pointer to LockBox for use in S3 resume.
//
2021-12-05 23:54:02 +01:00
S3PerformanceTablePointer = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) mAcpiS3PerformanceTable ;
Status = SaveLockBox (
& gFirmwarePerformanceS3PointerGuid ,
& S3PerformanceTablePointer ,
sizeof ( EFI_PHYSICAL_ADDRESS )
) ;
2013-05-16 09:35:30 +02:00
ASSERT_EFI_ERROR ( Status ) ;
}
}
2011-12-01 02:57:27 +01:00
/**
Install ACPI Firmware Performance Data Table ( FPDT ) .
@ return Status code .
* */
EFI_STATUS
InstallFirmwarePerformanceDataTable (
VOID
)
{
2021-06-28 13:50:22 +02:00
EFI_STATUS Status ;
EFI_ACPI_TABLE_PROTOCOL * AcpiTableProtocol ;
UINTN BootPerformanceDataSize ;
FIRMWARE_PERFORMANCE_VARIABLE PerformanceVariable ;
UINTN Size ;
EDKII_VARIABLE_POLICY_PROTOCOL * VariablePolicyProtocol ;
2011-12-01 02:57:27 +01:00
//
// Get AcpiTable Protocol.
//
2021-12-05 23:54:02 +01:00
Status = gBS - > LocateProtocol ( & gEfiAcpiTableProtocolGuid , NULL , ( VOID * * ) & AcpiTableProtocol ) ;
2011-12-01 02:57:27 +01:00
if ( EFI_ERROR ( Status ) ) {
return Status ;
}
2021-06-28 13:50:22 +02:00
//
// Get VariablePolicy Protocol.
//
2021-12-05 23:54:02 +01:00
Status = gBS - > LocateProtocol ( & gEdkiiVariablePolicyProtocolGuid , NULL , ( VOID * * ) & VariablePolicyProtocol ) ;
2021-06-28 13:50:22 +02:00
if ( EFI_ERROR ( Status ) ) {
return Status ;
}
2018-01-17 02:47:58 +01:00
if ( mReceivedAcpiBootPerformanceTable ! = NULL ) {
2021-12-05 23:54:02 +01:00
mAcpiBootPerformanceTable = mReceivedAcpiBootPerformanceTable ;
2018-01-17 02:47:58 +01:00
mAcpiBootPerformanceTable - > BasicBoot . ResetEnd = mBootPerformanceTableTemplate . BasicBoot . ResetEnd ;
} else {
2012-05-08 05:09:54 +02:00
//
2018-01-17 02:47:58 +01:00
// Try to allocate the same runtime buffer as last time boot.
//
BootPerformanceDataSize = sizeof ( BOOT_PERFORMANCE_TABLE ) ;
ZeroMem ( & PerformanceVariable , sizeof ( PerformanceVariable ) ) ;
2021-12-05 23:54:02 +01:00
Size = sizeof ( PerformanceVariable ) ;
2018-01-17 02:47:58 +01:00
Status = gRT - > GetVariable (
EFI_FIRMWARE_PERFORMANCE_VARIABLE_NAME ,
& gEfiFirmwarePerformanceGuid ,
NULL ,
& Size ,
& PerformanceVariable
) ;
2016-04-22 09:31:07 +02:00
if ( ! EFI_ERROR ( Status ) ) {
2018-01-17 02:47:58 +01:00
Status = gBS - > AllocatePages (
AllocateAddress ,
EfiReservedMemoryType ,
EFI_SIZE_TO_PAGES ( BootPerformanceDataSize ) ,
& PerformanceVariable . BootPerformanceTablePointer
) ;
if ( ! EFI_ERROR ( Status ) ) {
2021-12-05 23:54:02 +01:00
mAcpiBootPerformanceTable = ( BOOT_PERFORMANCE_TABLE * ) ( UINTN ) PerformanceVariable . BootPerformanceTablePointer ;
2016-04-22 09:31:07 +02:00
}
2018-01-17 02:47:58 +01:00
}
2021-12-05 23:54:02 +01:00
2018-01-17 02:47:58 +01:00
if ( mAcpiBootPerformanceTable = = NULL ) {
2012-05-08 05:09:54 +02:00
//
2018-01-17 02:47:58 +01:00
// Fail to allocate at specified address, continue to allocate at any address.
2012-05-08 05:09:54 +02:00
//
2021-12-05 23:54:02 +01:00
mAcpiBootPerformanceTable = ( BOOT_PERFORMANCE_TABLE * ) AllocatePeiAccessiblePages (
EfiReservedMemoryType ,
EFI_SIZE_TO_PAGES ( BootPerformanceDataSize )
) ;
2012-05-08 05:09:54 +02:00
}
2021-12-05 23:54:02 +01:00
2018-01-17 02:47:58 +01:00
DEBUG ( ( DEBUG_INFO , " FPDT: ACPI Boot Performance Table address = 0x%x \n " , mAcpiBootPerformanceTable ) ) ;
if ( mAcpiBootPerformanceTable = = NULL ) {
return EFI_OUT_OF_RESOURCES ;
2013-05-16 09:35:30 +02:00
}
2021-12-05 23:54:02 +01:00
2012-05-08 05:09:54 +02:00
//
2018-01-17 02:47:58 +01:00
// Fill Basic Boot record to Boot Performance Table.
2012-05-08 05:09:54 +02:00
//
2018-01-17 02:47:58 +01:00
CopyMem ( mAcpiBootPerformanceTable , & mBootPerformanceTableTemplate , sizeof ( mBootPerformanceTableTemplate ) ) ;
2013-05-27 07:45:35 +02:00
}
2021-12-05 23:54:02 +01:00
BootPerformanceDataSize = mAcpiBootPerformanceTable - > Header . Length ;
2013-05-27 07:45:35 +02:00
2012-05-08 05:09:54 +02:00
//
// Save Boot Performance Table address to Variable for use in S4 resume.
//
2021-12-05 23:54:02 +01:00
PerformanceVariable . BootPerformanceTablePointer = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) mAcpiBootPerformanceTable ;
2012-05-08 05:09:54 +02:00
//
// Update Boot Performance Table Pointer in template.
//
2021-12-05 23:54:02 +01:00
mFirmwarePerformanceTableTemplate . BootPointerRecord . BootPerformanceTablePointer = ( UINT64 ) ( UINTN ) mAcpiBootPerformanceTable ;
2012-05-08 05:09:54 +02:00
2013-05-16 09:35:30 +02:00
//
// Save S3 Performance Table address to Variable for use in S4 resume.
//
2021-12-05 23:54:02 +01:00
PerformanceVariable . S3PerformanceTablePointer = ( EFI_PHYSICAL_ADDRESS ) ( UINTN ) mAcpiS3PerformanceTable ;
2013-05-16 09:35:30 +02:00
//
// Update S3 Performance Table Pointer in template.
//
2021-12-05 23:54:02 +01:00
mFirmwarePerformanceTableTemplate . S3PointerRecord . S3PerformanceTablePointer = ( UINT64 ) ( UINTN ) mAcpiS3PerformanceTable ;
2011-12-01 02:57:27 +01:00
//
// Save Runtime Performance Table pointers to Variable.
2018-01-17 02:47:58 +01:00
// Don't check SetVariable return status. It doesn't impact FPDT table generation.
2014-03-17 08:22:50 +01:00
//
gRT - > SetVariable (
2021-12-05 23:54:02 +01:00
EFI_FIRMWARE_PERFORMANCE_VARIABLE_NAME ,
& gEfiFirmwarePerformanceGuid ,
EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS ,
sizeof ( PerformanceVariable ) ,
& PerformanceVariable
) ;
2011-12-01 02:57:27 +01:00
2021-06-28 13:50:22 +02:00
//
// Lock the variable which stores the Performance Table pointers.
//
Status = RegisterBasicVariablePolicy (
VariablePolicyProtocol ,
& gEfiFirmwarePerformanceGuid ,
EFI_FIRMWARE_PERFORMANCE_VARIABLE_NAME ,
VARIABLE_POLICY_NO_MIN_SIZE ,
VARIABLE_POLICY_NO_MAX_SIZE ,
VARIABLE_POLICY_NO_MUST_ATTR ,
VARIABLE_POLICY_NO_CANT_ATTR ,
VARIABLE_POLICY_TYPE_LOCK_NOW
) ;
2021-12-05 23:54:02 +01:00
if ( EFI_ERROR ( Status ) ) {
DEBUG ( ( DEBUG_ERROR , " [FirmwarePerformanceDxe] Error when lock variable %s, Status = %r \n " , EFI_FIRMWARE_PERFORMANCE_VARIABLE_NAME , Status ) ) ;
ASSERT_EFI_ERROR ( Status ) ;
2021-06-28 13:50:22 +02:00
}
2011-12-01 02:57:27 +01:00
//
// Publish Firmware Performance Data Table.
//
2021-12-05 23:54:02 +01:00
FpdtAcpiTableChecksum ( ( UINT8 * ) & mFirmwarePerformanceTableTemplate , mFirmwarePerformanceTableTemplate . Header . Length ) ;
2011-12-01 02:57:27 +01:00
Status = AcpiTableProtocol - > InstallAcpiTable (
AcpiTableProtocol ,
& mFirmwarePerformanceTableTemplate ,
mFirmwarePerformanceTableTemplate . Header . Length ,
& mFirmwarePerformanceTableTemplateKey
) ;
if ( EFI_ERROR ( Status ) ) {
2018-01-17 02:47:58 +01:00
if ( mAcpiBootPerformanceTable ! = NULL ) {
FreePages ( mAcpiBootPerformanceTable , EFI_SIZE_TO_PAGES ( BootPerformanceDataSize ) ) ;
}
2021-12-05 23:54:02 +01:00
2013-05-16 09:35:30 +02:00
if ( mAcpiS3PerformanceTable ! = NULL ) {
FreePages ( mAcpiS3PerformanceTable , EFI_SIZE_TO_PAGES ( sizeof ( S3_PERFORMANCE_TABLE ) ) ) ;
}
2021-12-05 23:54:02 +01:00
2011-12-01 02:57:27 +01:00
mAcpiBootPerformanceTable = NULL ;
2021-12-05 23:54:02 +01:00
mAcpiS3PerformanceTable = NULL ;
2011-12-01 02:57:27 +01:00
return Status ;
}
2021-12-05 23:54:02 +01:00
2011-12-01 02:57:27 +01:00
return EFI_SUCCESS ;
}
/**
Report status code listener of FPDT . This is used to collect performance data
for OsLoaderLoadImageStart and OsLoaderStartImageStart in FPDT .
@ param [ in ] CodeType Indicates the type of status code being reported .
@ param [ in ] Value Describes the current status of a hardware or software entity .
This included information about the class and subclass that is used to
classify the entity as well as an operation .
@ param [ in ] Instance The enumeration of a hardware or software entity within
the system . Valid instance numbers start with 1.
@ param [ in ] CallerId This optional parameter may be used to identify the caller .
This parameter allows the status code driver to apply different rules to
different callers .
@ param [ in ] Data This optional parameter may be used to pass additional data .
@ retval EFI_SUCCESS Status code is what we expected .
@ retval EFI_UNSUPPORTED Status code not supported .
* */
EFI_STATUS
EFIAPI
FpdtStatusCodeListenerDxe (
2021-12-05 23:54:02 +01:00
IN EFI_STATUS_CODE_TYPE CodeType ,
IN EFI_STATUS_CODE_VALUE Value ,
IN UINT32 Instance ,
IN EFI_GUID * CallerId ,
IN EFI_STATUS_CODE_DATA * Data
2011-12-01 02:57:27 +01:00
)
{
EFI_STATUS Status ;
//
// Check whether status code is what we are interested in.
//
if ( ( CodeType & EFI_STATUS_CODE_TYPE_MASK ) ! = EFI_PROGRESS_CODE ) {
return EFI_UNSUPPORTED ;
}
2018-06-27 15:08:52 +02:00
2014-12-03 16:34:08 +01:00
if ( Value = = ( EFI_SOFTWARE_DXE_CORE | EFI_SW_DXE_CORE_PC_HANDOFF_TO_NEXT ) ) {
//
// DxeCore ReportStatusCode Enable so that the capability can be supported.
//
mDxeCoreReportStatusCodeEnable = TRUE ;
}
2011-12-01 02:57:27 +01:00
Status = EFI_SUCCESS ;
if ( Value = = PcdGet32 ( PcdProgressCodeOsLoaderLoad ) ) {
//
// Progress code for OS Loader LoadImage.
//
if ( mAcpiBootPerformanceTable = = NULL ) {
return Status ;
}
//
// Update OS Loader LoadImage Start for UEFI boot.
//
mAcpiBootPerformanceTable - > BasicBoot . OsLoaderLoadImageStart = GetTimeInNanoSecond ( GetPerformanceCounter ( ) ) ;
} else if ( Value = = PcdGet32 ( PcdProgressCodeOsLoaderStart ) ) {
//
// Progress code for OS Loader StartImage.
//
if ( mAcpiBootPerformanceTable = = NULL ) {
return Status ;
}
//
// Update OS Loader StartImage Start for UEFI boot.
//
mAcpiBootPerformanceTable - > BasicBoot . OsLoaderStartImageStart = GetTimeInNanoSecond ( GetPerformanceCounter ( ) ) ;
} else if ( Value = = ( EFI_SOFTWARE_EFI_BOOT_SERVICE | EFI_SW_BS_PC_EXIT_BOOT_SERVICES ) ) {
2014-12-03 16:34:08 +01:00
//
// Unregister boot time report status code listener.
//
mRscHandlerProtocol - > Unregister ( FpdtStatusCodeListenerDxe ) ;
2011-12-01 02:57:27 +01:00
//
// Progress code for ExitBootServices.
//
if ( mAcpiBootPerformanceTable = = NULL ) {
return Status ;
}
//
// Update ExitBootServicesExit for UEFI boot.
//
mAcpiBootPerformanceTable - > BasicBoot . ExitBootServicesExit = GetTimeInNanoSecond ( GetPerformanceCounter ( ) ) ;
2015-05-08 09:44:59 +02:00
} else if ( Value = = ( EFI_SOFTWARE_DXE_BS_DRIVER | EFI_SW_DXE_BS_PC_LEGACY_BOOT_EVENT ) ) {
if ( mAcpiBootPerformanceTable = = NULL ) {
//
// Firmware Performance Data Table not installed, do nothing.
//
return Status ;
}
//
// Update Firmware Basic Boot Performance Record for legacy boot.
//
mAcpiBootPerformanceTable - > BasicBoot . OsLoaderStartImageStart = GetTimeInNanoSecond ( GetPerformanceCounter ( ) ) ;
//
// Dump FPDT Boot Performance record.
//
2021-11-17 04:21:29 +01:00
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - ResetEnd = %ld \n " , mAcpiBootPerformanceTable - > BasicBoot . ResetEnd ) ) ;
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - OsLoaderLoadImageStart = 0 \n " ) ) ;
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - OsLoaderStartImageStart = %ld \n " , mAcpiBootPerformanceTable - > BasicBoot . OsLoaderStartImageStart ) ) ;
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - ExitBootServicesEntry = 0 \n " ) ) ;
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - ExitBootServicesExit = 0 \n " ) ) ;
2021-12-05 23:54:02 +01:00
} else if ( ( Data ! = NULL ) & & CompareGuid ( & Data - > Type , & gEdkiiFpdtExtendedFirmwarePerformanceGuid ) ) {
2018-01-17 02:47:58 +01:00
//
// Get the Boot performance table and then install it to ACPI table.
//
CopyMem ( & mReceivedAcpiBootPerformanceTable , Data + 1 , Data - > Size ) ;
2021-06-28 13:50:22 +02:00
InstallFirmwarePerformanceDataTable ( ) ;
2021-12-05 23:54:02 +01:00
} else if ( ( Data ! = NULL ) & & CompareGuid ( & Data - > Type , & gEfiFirmwarePerformanceGuid ) ) {
2018-04-10 07:51:08 +02:00
DEBUG ( ( DEBUG_ERROR , " FpdtStatusCodeListenerDxe: Performance data reported through gEfiFirmwarePerformanceGuid will not be collected by FirmwarePerformanceDataTableDxe \n " ) ) ;
Status = EFI_UNSUPPORTED ;
2011-12-01 02:57:27 +01:00
} else {
//
// Ignore else progress code.
//
Status = EFI_UNSUPPORTED ;
}
return Status ;
}
2021-06-28 13:50:22 +02:00
/**
Notify function for event EndOfDxe .
This is used to install ACPI Firmware Performance Data Table for basic boot records .
@ param [ in ] Event The Event that is being processed .
@ param [ in ] Context The Event Context .
* */
VOID
EFIAPI
FpdtEndOfDxeEventNotify (
2021-12-05 23:54:02 +01:00
IN EFI_EVENT Event ,
IN VOID * Context
2021-06-28 13:50:22 +02:00
)
{
//
// When performance is enabled, the FPDT will be installed when DxeCorePerformanceLib report the data to FimwarePerformanceDxe.
// This is used to install the FPDT for the basic boot recods when performance infrastructure is not enabled.
//
2021-12-05 23:54:02 +01:00
if ( ( PcdGet8 ( PcdPerformanceLibraryPropertyMask ) & PERFORMANCE_LIBRARY_PROPERTY_MEASUREMENT_ENABLED ) ! = 0 ) {
2021-06-28 13:50:22 +02:00
return ;
}
2021-12-05 23:54:02 +01:00
2021-06-28 13:50:22 +02:00
ASSERT ( mReceivedAcpiBootPerformanceTable = = NULL ) ;
InstallFirmwarePerformanceDataTable ( ) ;
}
2014-12-03 16:34:08 +01:00
/**
Notify function for event EVT_SIGNAL_EXIT_BOOT_SERVICES . This is used to record
performance data for ExitBootServicesEntry in FPDT .
@ param [ in ] Event The Event that is being processed .
@ param [ in ] Context The Event Context .
* */
VOID
EFIAPI
FpdtExitBootServicesEventNotify (
2021-12-05 23:54:02 +01:00
IN EFI_EVENT Event ,
IN VOID * Context
2014-12-03 16:34:08 +01:00
)
{
2014-12-12 04:17:55 +01:00
if ( ! mDxeCoreReportStatusCodeEnable ) {
2014-12-03 16:34:08 +01:00
//
2018-06-27 15:08:52 +02:00
// When DxeCore Report Status Code is disabled,
2014-12-03 16:34:08 +01:00
// Unregister boot time report status code listener at ExitBootService Event.
//
mRscHandlerProtocol - > Unregister ( FpdtStatusCodeListenerDxe ) ;
}
if ( mAcpiBootPerformanceTable = = NULL ) {
//
// Firmware Performance Data Table not installed, do nothing.
//
2021-12-05 23:54:02 +01:00
return ;
2014-12-03 16:34:08 +01:00
}
//
// Update Firmware Basic Boot Performance Record for UEFI boot.
//
mAcpiBootPerformanceTable - > BasicBoot . ExitBootServicesEntry = GetTimeInNanoSecond ( GetPerformanceCounter ( ) ) ;
//
// Dump FPDT Boot Performance record.
//
2021-11-17 04:21:29 +01:00
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - ResetEnd = %ld \n " , mAcpiBootPerformanceTable - > BasicBoot . ResetEnd ) ) ;
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - OsLoaderLoadImageStart = %ld \n " , mAcpiBootPerformanceTable - > BasicBoot . OsLoaderLoadImageStart ) ) ;
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - OsLoaderStartImageStart = %ld \n " , mAcpiBootPerformanceTable - > BasicBoot . OsLoaderStartImageStart ) ) ;
DEBUG ( ( DEBUG_INFO , " FPDT: Boot Performance - ExitBootServicesEntry = %ld \n " , mAcpiBootPerformanceTable - > BasicBoot . ExitBootServicesEntry ) ) ;
2014-12-03 16:34:08 +01:00
//
// ExitBootServicesExit will be updated later, so don't dump it here.
//
}
2011-12-01 02:57:27 +01:00
/**
The module Entry Point of the Firmware Performance Data Table DXE driver .
@ param [ in ] ImageHandle The firmware allocated handle for the EFI image .
@ param [ in ] SystemTable A pointer to the EFI System Table .
@ retval EFI_SUCCESS The entry point is executed successfully .
@ retval Other Some error occurs when executing this entry point .
* */
EFI_STATUS
EFIAPI
FirmwarePerformanceDxeEntryPoint (
2021-12-05 23:54:02 +01:00
IN EFI_HANDLE ImageHandle ,
IN EFI_SYSTEM_TABLE * SystemTable
2011-12-01 02:57:27 +01:00
)
{
2021-12-05 23:54:02 +01:00
EFI_STATUS Status ;
EFI_HOB_GUID_TYPE * GuidHob ;
FIRMWARE_SEC_PERFORMANCE * Performance ;
VOID * Registration ;
UINT64 OemTableId ;
EFI_EVENT EndOfDxeEvent ;
2013-09-16 04:20:52 +02:00
CopyMem (
mFirmwarePerformanceTableTemplate . Header . OemId ,
PcdGetPtr ( PcdAcpiDefaultOemId ) ,
sizeof ( mFirmwarePerformanceTableTemplate . Header . OemId )
) ;
OemTableId = PcdGet64 ( PcdAcpiDefaultOemTableId ) ;
CopyMem ( & mFirmwarePerformanceTableTemplate . Header . OemTableId , & OemTableId , sizeof ( UINT64 ) ) ;
2021-12-05 23:54:02 +01:00
mFirmwarePerformanceTableTemplate . Header . OemRevision = PcdGet32 ( PcdAcpiDefaultOemRevision ) ;
mFirmwarePerformanceTableTemplate . Header . CreatorId = PcdGet32 ( PcdAcpiDefaultCreatorId ) ;
mFirmwarePerformanceTableTemplate . Header . CreatorRevision = PcdGet32 ( PcdAcpiDefaultCreatorRevision ) ;
2011-12-01 02:57:27 +01:00
//
// Get Report Status Code Handler Protocol.
//
2021-12-05 23:54:02 +01:00
Status = gBS - > LocateProtocol ( & gEfiRscHandlerProtocolGuid , NULL , ( VOID * * ) & mRscHandlerProtocol ) ;
2011-12-01 02:57:27 +01:00
ASSERT_EFI_ERROR ( Status ) ;
//
// Register report status code listener for OS Loader load and start.
//
Status = mRscHandlerProtocol - > Register ( FpdtStatusCodeListenerDxe , TPL_HIGH_LEVEL ) ;
ASSERT_EFI_ERROR ( Status ) ;
2021-06-28 13:50:22 +02:00
//
// Register the notify function to install FPDT at EndOfDxe.
//
Status = gBS - > CreateEventEx (
EVT_NOTIFY_SIGNAL ,
TPL_NOTIFY ,
FpdtEndOfDxeEventNotify ,
NULL ,
& gEfiEndOfDxeEventGroupGuid ,
& EndOfDxeEvent
) ;
ASSERT_EFI_ERROR ( Status ) ;
2011-12-01 02:57:27 +01:00
//
// Register the notify function to update FPDT on ExitBootServices Event.
//
Status = gBS - > CreateEventEx (
EVT_NOTIFY_SIGNAL ,
TPL_NOTIFY ,
FpdtExitBootServicesEventNotify ,
NULL ,
& gEfiEventExitBootServicesGuid ,
& mExitBootServicesEvent
) ;
ASSERT_EFI_ERROR ( Status ) ;
//
// Retrieve GUID HOB data that contains the ResetEnd.
//
GuidHob = GetFirstGuidHob ( & gEfiFirmwarePerformanceGuid ) ;
if ( GuidHob ! = NULL ) {
2021-12-05 23:54:02 +01:00
Performance = ( FIRMWARE_SEC_PERFORMANCE * ) GET_GUID_HOB_DATA ( GuidHob ) ;
2011-12-01 02:57:27 +01:00
mBootPerformanceTableTemplate . BasicBoot . ResetEnd = Performance - > ResetEnd ;
} else {
//
// SEC Performance Data Hob not found, ResetEnd in ACPI FPDT table will be 0.
//
2017-04-20 08:08:53 +02:00
DEBUG ( ( DEBUG_WARN , " FPDT: WARNING: SEC Performance Data Hob not found, ResetEnd will be set to 0! \n " ) ) ;
2011-12-01 02:57:27 +01:00
}
2013-05-16 09:35:30 +02:00
if ( FeaturePcdGet ( PcdFirmwarePerformanceDataTableS3Support ) ) {
//
// Register callback function upon VariableArchProtocol and LockBoxProtocol
// to allocate S3 performance table memory and save the pointer to LockBox.
//
EfiCreateProtocolNotifyEvent (
& gEfiVariableArchProtocolGuid ,
TPL_CALLBACK ,
FpdtAllocateS3PerformanceTableMemory ,
NULL ,
& Registration
) ;
EfiCreateProtocolNotifyEvent (
& gEfiLockBoxProtocolGuid ,
TPL_CALLBACK ,
FpdtAllocateS3PerformanceTableMemory ,
NULL ,
& Registration
) ;
} else {
//
// Exclude S3 Performance Table Pointer from FPDT table template.
//
mFirmwarePerformanceTableTemplate . Header . Length - = sizeof ( EFI_ACPI_5_0_FPDT_S3_PERFORMANCE_TABLE_POINTER_RECORD ) ;
}
2011-12-01 02:57:27 +01:00
return EFI_SUCCESS ;
}