2017-08-04 03:59:08 +02:00
|
|
|
/** @file
|
|
|
|
Intel Processor Trace feature.
|
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
Copyright (c) 2017 - 2023, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:07:22 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2017-08-04 03:59:08 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "CpuCommonFeatures.h"
|
|
|
|
|
|
|
|
///
|
2017-08-18 05:19:03 +02:00
|
|
|
/// This macro define the max entries in the Topa table.
|
2018-06-27 15:14:20 +02:00
|
|
|
/// Each entry in the table contains some attribute bits, a pointer to an output region, and the size of the region.
|
|
|
|
/// The last entry in the table may hold a pointer to the next table. This pointer can either point to the top of the
|
|
|
|
/// current table (for circular array) or to the base of another table.
|
|
|
|
/// At least 2 entries are needed because the list of entries must
|
|
|
|
/// be terminated by an entry with the END bit set to 1, so 2
|
2017-08-18 05:19:03 +02:00
|
|
|
/// entries are required to use a single valid entry.
|
2017-08-04 03:59:08 +02:00
|
|
|
///
|
2017-08-18 05:19:03 +02:00
|
|
|
#define MAX_TOPA_ENTRY_COUNT 2
|
2017-08-04 03:59:08 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// Processor trace output scheme selection.
|
|
|
|
///
|
|
|
|
typedef enum {
|
2017-08-23 04:24:58 +02:00
|
|
|
RtitOutputSchemeSingleRange = 0,
|
|
|
|
RtitOutputSchemeToPA
|
|
|
|
} RTIT_OUTPUT_SCHEME;
|
2017-08-04 03:59:08 +02:00
|
|
|
|
|
|
|
typedef struct {
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
BOOLEAN TopaSupported;
|
|
|
|
BOOLEAN SingleRangeSupported;
|
|
|
|
MSR_IA32_RTIT_CTL_REGISTER RtitCtrl;
|
|
|
|
MSR_IA32_RTIT_OUTPUT_BASE_REGISTER RtitOutputBase;
|
|
|
|
MSR_IA32_RTIT_OUTPUT_MASK_PTRS_REGISTER RtitOutputMaskPtrs;
|
2023-04-25 10:33:30 +02:00
|
|
|
BOOLEAN CycPacketSupported;
|
2017-08-04 03:59:08 +02:00
|
|
|
} PROC_TRACE_PROCESSOR_DATA;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
UINT32 NumberOfProcessors;
|
|
|
|
|
2018-06-27 15:14:20 +02:00
|
|
|
UINT8 ProcTraceOutputScheme;
|
2017-08-04 03:59:08 +02:00
|
|
|
UINT32 ProcTraceMemSize;
|
|
|
|
|
|
|
|
UINTN *ThreadMemRegionTable;
|
|
|
|
UINTN AllocatedThreads;
|
|
|
|
|
|
|
|
UINTN *TopaMemArray;
|
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
BOOLEAN EnableOnBspOnly;
|
2023-04-25 10:33:30 +02:00
|
|
|
BOOLEAN EnablePerformanceCollecting;
|
2023-04-25 06:47:05 +02:00
|
|
|
|
2017-08-04 03:59:08 +02:00
|
|
|
PROC_TRACE_PROCESSOR_DATA *ProcessorData;
|
|
|
|
} PROC_TRACE_DATA;
|
|
|
|
|
|
|
|
typedef struct {
|
2017-08-18 05:19:03 +02:00
|
|
|
RTIT_TOPA_TABLE_ENTRY TopaEntry[MAX_TOPA_ENTRY_COUNT];
|
2017-08-04 03:59:08 +02:00
|
|
|
} PROC_TRACE_TOPA_TABLE;
|
|
|
|
|
|
|
|
/**
|
|
|
|
Prepares for the data used by CPU feature detection and initialization.
|
|
|
|
|
|
|
|
@param[in] NumberOfProcessors The number of CPUs in the platform.
|
|
|
|
|
|
|
|
@return Pointer to a buffer of CPU related configuration data.
|
|
|
|
|
|
|
|
@note This service could be called by BSP only.
|
|
|
|
**/
|
|
|
|
VOID *
|
|
|
|
EFIAPI
|
|
|
|
ProcTraceGetConfigData (
|
|
|
|
IN UINTN NumberOfProcessors
|
|
|
|
)
|
|
|
|
{
|
|
|
|
PROC_TRACE_DATA *ConfigData;
|
|
|
|
|
|
|
|
ConfigData = AllocateZeroPool (sizeof (PROC_TRACE_DATA) + sizeof (PROC_TRACE_PROCESSOR_DATA) * NumberOfProcessors);
|
|
|
|
ASSERT (ConfigData != NULL);
|
|
|
|
ConfigData->ProcessorData = (PROC_TRACE_PROCESSOR_DATA *)((UINT8 *)ConfigData + sizeof (PROC_TRACE_DATA));
|
|
|
|
|
2023-04-25 10:33:30 +02:00
|
|
|
ConfigData->NumberOfProcessors = (UINT32)NumberOfProcessors;
|
|
|
|
ConfigData->ProcTraceMemSize = PcdGet32 (PcdCpuProcTraceMemSize);
|
|
|
|
ConfigData->ProcTraceOutputScheme = PcdGet8 (PcdCpuProcTraceOutputScheme);
|
|
|
|
ConfigData->EnableOnBspOnly = PcdGetBool (PcdCpuProcTraceBspOnly);
|
|
|
|
ConfigData->EnablePerformanceCollecting = PcdGetBool (PcdCpuProcTracePerformanceCollecting);
|
2017-08-04 03:59:08 +02:00
|
|
|
|
|
|
|
return ConfigData;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-06-27 15:14:20 +02:00
|
|
|
Detects if Intel Processor Trace feature supported on current
|
2017-08-04 03:59:08 +02:00
|
|
|
processor.
|
|
|
|
|
|
|
|
@param[in] ProcessorNumber The index of the CPU executing this function.
|
|
|
|
@param[in] CpuInfo A pointer to the REGISTER_CPU_FEATURE_INFORMATION
|
|
|
|
structure for the CPU executing this function.
|
|
|
|
@param[in] ConfigData A pointer to the configuration buffer returned
|
|
|
|
by CPU_FEATURE_GET_CONFIG_DATA. NULL if
|
|
|
|
CPU_FEATURE_GET_CONFIG_DATA was not provided in
|
|
|
|
RegisterCpuFeature().
|
|
|
|
|
|
|
|
@retval TRUE Processor Trace feature is supported.
|
|
|
|
@retval FALSE Processor Trace feature is not supported.
|
|
|
|
|
|
|
|
@note This service could be called by BSP/APs.
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EFIAPI
|
|
|
|
ProcTraceSupport (
|
|
|
|
IN UINTN ProcessorNumber,
|
|
|
|
IN REGISTER_CPU_FEATURE_INFORMATION *CpuInfo,
|
|
|
|
IN VOID *ConfigData OPTIONAL
|
|
|
|
)
|
|
|
|
{
|
|
|
|
PROC_TRACE_DATA *ProcTraceData;
|
|
|
|
CPUID_STRUCTURED_EXTENDED_FEATURE_FLAGS_EBX Ebx;
|
2023-04-25 10:33:30 +02:00
|
|
|
CPUID_INTEL_PROCESSOR_TRACE_MAIN_LEAF_ECX ProcTraceEcx;
|
|
|
|
CPUID_INTEL_PROCESSOR_TRACE_MAIN_LEAF_EBX ProcTraceEbx;
|
2017-08-04 03:59:08 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Check if ProcTraceMemorySize option is enabled (0xFF means disable by user)
|
|
|
|
//
|
|
|
|
ProcTraceData = (PROC_TRACE_DATA *)ConfigData;
|
2017-09-26 04:43:42 +02:00
|
|
|
ASSERT (ProcTraceData != NULL);
|
2017-08-18 05:19:03 +02:00
|
|
|
if ((ProcTraceData->ProcTraceMemSize > RtitTopaMemorySize128M) ||
|
2017-08-23 04:24:58 +02:00
|
|
|
(ProcTraceData->ProcTraceOutputScheme > RtitOutputSchemeToPA))
|
|
|
|
{
|
2017-08-04 03:59:08 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Check if Processor Trace is supported
|
|
|
|
//
|
|
|
|
AsmCpuidEx (CPUID_STRUCTURED_EXTENDED_FEATURE_FLAGS, 0, NULL, &Ebx.Uint32, NULL, NULL);
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
if (Ebx.Bits.IntelProcessorTrace == 0) {
|
2017-08-04 03:59:08 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2023-04-25 10:33:30 +02:00
|
|
|
AsmCpuidEx (CPUID_INTEL_PROCESSOR_TRACE, CPUID_INTEL_PROCESSOR_TRACE_MAIN_LEAF, NULL, &ProcTraceEbx.Uint32, &ProcTraceEcx.Uint32, NULL);
|
|
|
|
ProcTraceData->ProcessorData[ProcessorNumber].TopaSupported = (BOOLEAN)(ProcTraceEcx.Bits.RTIT == 1);
|
|
|
|
ProcTraceData->ProcessorData[ProcessorNumber].SingleRangeSupported = (BOOLEAN)(ProcTraceEcx.Bits.SingleRangeOutput == 1);
|
2017-08-23 04:24:58 +02:00
|
|
|
if ((ProcTraceData->ProcessorData[ProcessorNumber].TopaSupported && (ProcTraceData->ProcTraceOutputScheme == RtitOutputSchemeToPA)) ||
|
|
|
|
(ProcTraceData->ProcessorData[ProcessorNumber].SingleRangeSupported && (ProcTraceData->ProcTraceOutputScheme == RtitOutputSchemeSingleRange)))
|
|
|
|
{
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
ProcTraceData->ProcessorData[ProcessorNumber].RtitCtrl.Uint64 = AsmReadMsr64 (MSR_IA32_RTIT_CTL);
|
|
|
|
ProcTraceData->ProcessorData[ProcessorNumber].RtitOutputBase.Uint64 = AsmReadMsr64 (MSR_IA32_RTIT_OUTPUT_BASE);
|
|
|
|
ProcTraceData->ProcessorData[ProcessorNumber].RtitOutputMaskPtrs.Uint64 = AsmReadMsr64 (MSR_IA32_RTIT_OUTPUT_MASK_PTRS);
|
2023-04-25 10:33:30 +02:00
|
|
|
ProcTraceData->ProcessorData[ProcessorNumber].CycPacketSupported = (BOOLEAN)(ProcTraceEbx.Bits.ConfigurablePsb == 1);
|
|
|
|
|
2017-08-04 03:59:08 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Initializes Intel Processor Trace feature to specific state.
|
|
|
|
|
|
|
|
@param[in] ProcessorNumber The index of the CPU executing this function.
|
|
|
|
@param[in] CpuInfo A pointer to the REGISTER_CPU_FEATURE_INFORMATION
|
|
|
|
structure for the CPU executing this function.
|
|
|
|
@param[in] ConfigData A pointer to the configuration buffer returned
|
|
|
|
by CPU_FEATURE_GET_CONFIG_DATA. NULL if
|
|
|
|
CPU_FEATURE_GET_CONFIG_DATA was not provided in
|
|
|
|
RegisterCpuFeature().
|
|
|
|
@param[in] State If TRUE, then the Processor Trace feature must be
|
|
|
|
enabled.
|
|
|
|
If FALSE, then the Processor Trace feature must be
|
|
|
|
disabled.
|
|
|
|
|
|
|
|
@retval RETURN_SUCCESS Intel Processor Trace feature is initialized.
|
|
|
|
|
|
|
|
**/
|
|
|
|
RETURN_STATUS
|
|
|
|
EFIAPI
|
|
|
|
ProcTraceInitialize (
|
|
|
|
IN UINTN ProcessorNumber,
|
|
|
|
IN REGISTER_CPU_FEATURE_INFORMATION *CpuInfo,
|
2021-12-03 03:01:00 +01:00
|
|
|
IN VOID *ConfigData OPTIONAL,
|
2017-08-04 03:59:08 +02:00
|
|
|
IN BOOLEAN State
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 MemRegionSize;
|
|
|
|
UINTN Pages;
|
|
|
|
UINTN Alignment;
|
|
|
|
UINTN MemRegionBaseAddr;
|
|
|
|
UINTN *ThreadMemRegionTable;
|
|
|
|
UINTN Index;
|
|
|
|
UINTN TopaTableBaseAddr;
|
|
|
|
UINTN AlignedAddress;
|
|
|
|
UINTN *TopaMemArray;
|
|
|
|
PROC_TRACE_TOPA_TABLE *TopaTable;
|
|
|
|
PROC_TRACE_DATA *ProcTraceData;
|
|
|
|
BOOLEAN FirstIn;
|
2017-08-18 05:19:03 +02:00
|
|
|
MSR_IA32_RTIT_CTL_REGISTER CtrlReg;
|
|
|
|
MSR_IA32_RTIT_STATUS_REGISTER StatusReg;
|
|
|
|
MSR_IA32_RTIT_OUTPUT_BASE_REGISTER OutputBaseReg;
|
|
|
|
MSR_IA32_RTIT_OUTPUT_MASK_PTRS_REGISTER OutputMaskPtrsReg;
|
|
|
|
RTIT_TOPA_TABLE_ENTRY *TopaEntryPtr;
|
2023-04-25 06:47:05 +02:00
|
|
|
BOOLEAN IsBsp;
|
2017-08-04 03:59:08 +02:00
|
|
|
|
2018-10-17 03:24:05 +02:00
|
|
|
//
|
|
|
|
// The scope of the MSR_IA32_RTIT_* is core for below processor type, only program
|
|
|
|
// MSR_IA32_RTIT_* for thread 0 in each core.
|
|
|
|
//
|
|
|
|
if (IS_GOLDMONT_PROCESSOR (CpuInfo->DisplayFamily, CpuInfo->DisplayModel) ||
|
|
|
|
IS_GOLDMONT_PLUS_PROCESSOR (CpuInfo->DisplayFamily, CpuInfo->DisplayModel))
|
|
|
|
{
|
|
|
|
if (CpuInfo->ProcessorInfo.Location.Thread != 0) {
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-04 03:59:08 +02:00
|
|
|
ProcTraceData = (PROC_TRACE_DATA *)ConfigData;
|
2017-09-26 04:43:42 +02:00
|
|
|
ASSERT (ProcTraceData != NULL);
|
2017-08-04 03:59:08 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Clear MSR_IA32_RTIT_CTL[0] and IA32_RTIT_STS only if MSR_IA32_RTIT_CTL[0]==1b
|
|
|
|
//
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
CtrlReg.Uint64 = ProcTraceData->ProcessorData[ProcessorNumber].RtitCtrl.Uint64;
|
2017-08-18 05:19:03 +02:00
|
|
|
if (CtrlReg.Bits.TraceEn != 0) {
|
2017-08-04 03:59:08 +02:00
|
|
|
///
|
|
|
|
/// Clear bit 0 in MSR IA32_RTIT_CTL (570)
|
|
|
|
///
|
2017-08-18 05:19:03 +02:00
|
|
|
CtrlReg.Bits.TraceEn = 0;
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_CTL,
|
2017-08-18 05:19:03 +02:00
|
|
|
CtrlReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Clear MSR IA32_RTIT_STS (571h) to all zeros
|
|
|
|
///
|
2017-08-18 05:19:03 +02:00
|
|
|
StatusReg.Uint64 = 0x0;
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_STATUS,
|
2017-08-18 05:19:03 +02:00
|
|
|
StatusReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-05-25 09:05:47 +02:00
|
|
|
if (!State) {
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
IsBsp = (CpuInfo->ProcessorInfo.StatusFlag & PROCESSOR_AS_BSP_BIT) ? TRUE : FALSE;
|
|
|
|
|
|
|
|
if (ProcTraceData->EnableOnBspOnly && !IsBsp) {
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-05-25 09:05:47 +02:00
|
|
|
MemRegionBaseAddr = 0;
|
|
|
|
FirstIn = FALSE;
|
|
|
|
|
|
|
|
if (ProcTraceData->ThreadMemRegionTable == NULL) {
|
|
|
|
FirstIn = TRUE;
|
|
|
|
DEBUG ((DEBUG_INFO, "Initialize Processor Trace\n"));
|
|
|
|
}
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Refer to PROC_TRACE_MEM_SIZE Table for Size Encoding
|
|
|
|
///
|
|
|
|
MemRegionSize = (UINT32)(1 << (ProcTraceData->ProcTraceMemSize + 12));
|
|
|
|
if (FirstIn) {
|
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: MemSize requested: 0x%X \n", MemRegionSize));
|
|
|
|
}
|
|
|
|
|
2017-08-04 03:59:08 +02:00
|
|
|
if (FirstIn) {
|
|
|
|
//
|
|
|
|
// Let BSP allocate and create the necessary memory region (Aligned to the size of
|
|
|
|
// the memory region from setup option(ProcTraceMemSize) which is an integral multiple of 4kB)
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
// for all the enabled threads to store Processor Trace debug data. Then Configure the trace
|
2017-08-04 03:59:08 +02:00
|
|
|
// address base in MSR, IA32_RTIT_OUTPUT_BASE (560h) bits 47:12. Note that all regions must be
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
// aligned based on their size, not just 4K. Thus a 2M region must have bits 20:12 cleared.
|
2017-08-04 03:59:08 +02:00
|
|
|
//
|
2021-12-05 23:54:17 +01:00
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
Pages = EFI_SIZE_TO_PAGES (MemRegionSize);
|
|
|
|
Alignment = MemRegionSize;
|
|
|
|
if (ProcTraceData->EnableOnBspOnly) {
|
|
|
|
//
|
|
|
|
// When only enable ProcTrace on BSP, this is the first and only time ProcTraceInitialize() runs.
|
|
|
|
//
|
|
|
|
MemRegionBaseAddr = (UINTN)AllocateAlignedReservedPages (Pages, Alignment);
|
|
|
|
if (MemRegionBaseAddr == 0) {
|
|
|
|
//
|
|
|
|
// Could not allocate for BSP even
|
|
|
|
//
|
|
|
|
DEBUG ((DEBUG_ERROR, "ProcTrace: Out of mem, failed to allocate buffer for BSP\n"));
|
|
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: Allocated PT MemRegionBaseAddr(aligned) for BSP only: 0x%llX.\n", (UINT64)MemRegionBaseAddr));
|
|
|
|
} else {
|
|
|
|
ThreadMemRegionTable = (UINTN *)AllocatePool (ProcTraceData->NumberOfProcessors * sizeof (UINTN *));
|
|
|
|
if (ThreadMemRegionTable == NULL) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "Allocate ProcTrace ThreadMemRegionTable Failed\n"));
|
|
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProcTraceData->ThreadMemRegionTable = ThreadMemRegionTable;
|
|
|
|
|
|
|
|
for (Index = 0; Index < ProcTraceData->NumberOfProcessors; Index++, ProcTraceData->AllocatedThreads++) {
|
|
|
|
AlignedAddress = (UINTN)AllocateAlignedReservedPages (Pages, Alignment);
|
|
|
|
if (AlignedAddress == 0) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "ProcTrace: Out of mem, allocated only for %d threads\n", ProcTraceData->AllocatedThreads));
|
|
|
|
if (Index == 0) {
|
|
|
|
//
|
|
|
|
// Could not allocate for BSP even
|
|
|
|
//
|
|
|
|
FreePool ((VOID *)ThreadMemRegionTable);
|
|
|
|
ThreadMemRegionTable = NULL;
|
|
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2017-08-04 03:59:08 +02:00
|
|
|
}
|
2021-12-05 23:54:17 +01:00
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
ThreadMemRegionTable[Index] = AlignedAddress;
|
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: PT MemRegionBaseAddr(aligned) for thread %d: 0x%llX \n", Index, (UINT64)ThreadMemRegionTable[Index]));
|
2017-08-04 03:59:08 +02:00
|
|
|
}
|
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: Allocated PT mem for %d thread \n", ProcTraceData->AllocatedThreads));
|
2017-08-04 03:59:08 +02:00
|
|
|
}
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
}
|
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
if (!ProcTraceData->EnableOnBspOnly) {
|
|
|
|
if (ProcessorNumber < ProcTraceData->AllocatedThreads) {
|
|
|
|
MemRegionBaseAddr = ProcTraceData->ThreadMemRegionTable[ProcessorNumber];
|
|
|
|
} else {
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
2017-08-04 03:59:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Check Processor Trace output scheme: Single Range output or ToPA table
|
|
|
|
///
|
|
|
|
|
|
|
|
//
|
|
|
|
// Single Range output scheme
|
|
|
|
//
|
2018-06-27 15:14:20 +02:00
|
|
|
if (ProcTraceData->ProcessorData[ProcessorNumber].SingleRangeSupported &&
|
2017-08-23 04:24:58 +02:00
|
|
|
(ProcTraceData->ProcTraceOutputScheme == RtitOutputSchemeSingleRange))
|
|
|
|
{
|
2017-08-04 03:59:08 +02:00
|
|
|
if (FirstIn) {
|
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: Enabling Single Range Output scheme \n"));
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Clear MSR IA32_RTIT_CTL (0x570) ToPA (Bit 8)
|
|
|
|
//
|
2017-08-18 05:19:03 +02:00
|
|
|
CtrlReg.Bits.ToPA = 0;
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_CTL,
|
2017-08-18 05:19:03 +02:00
|
|
|
CtrlReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
//
|
2017-08-18 05:19:03 +02:00
|
|
|
// Program MSR IA32_RTIT_OUTPUT_BASE (0x560) bits[63:7] with the allocated Memory Region
|
2017-08-04 03:59:08 +02:00
|
|
|
//
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
OutputBaseReg.Uint64 = ProcTraceData->ProcessorData[ProcessorNumber].RtitOutputBase.Uint64;
|
2017-08-18 05:19:03 +02:00
|
|
|
OutputBaseReg.Bits.Base = (MemRegionBaseAddr >> 7) & 0x01FFFFFF;
|
|
|
|
OutputBaseReg.Bits.BaseHi = RShiftU64 ((UINT64)MemRegionBaseAddr, 32) & 0xFFFFFFFF;
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_OUTPUT_BASE,
|
2017-08-18 05:19:03 +02:00
|
|
|
OutputBaseReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Program the Mask bits for the Memory Region to MSR IA32_RTIT_OUTPUT_MASK_PTRS (561h)
|
|
|
|
//
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
OutputMaskPtrsReg.Uint64 = ProcTraceData->ProcessorData[ProcessorNumber].RtitOutputMaskPtrs.Uint64;
|
2017-08-18 05:19:03 +02:00
|
|
|
OutputMaskPtrsReg.Bits.MaskOrTableOffset = ((MemRegionSize - 1) >> 7) & 0x01FFFFFF;
|
2017-08-29 04:46:17 +02:00
|
|
|
OutputMaskPtrsReg.Bits.OutputOffset = RShiftU64 (MemRegionSize - 1, 32) & 0xFFFFFFFF;
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_OUTPUT_MASK_PTRS,
|
2017-08-18 05:19:03 +02:00
|
|
|
OutputMaskPtrsReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// ToPA(Table of physical address) scheme
|
|
|
|
//
|
2018-06-27 15:14:20 +02:00
|
|
|
if (ProcTraceData->ProcessorData[ProcessorNumber].TopaSupported &&
|
2017-08-23 04:24:58 +02:00
|
|
|
(ProcTraceData->ProcTraceOutputScheme == RtitOutputSchemeToPA))
|
|
|
|
{
|
2017-08-04 03:59:08 +02:00
|
|
|
//
|
|
|
|
// Create ToPA structure aligned at 4KB for each logical thread
|
|
|
|
// with at least 2 entries by 8 bytes size each. The first entry
|
|
|
|
// should have the trace output base address in bits 47:12, 6:9
|
|
|
|
// for Size, bits 4,2 and 0 must be cleared. The second entry
|
|
|
|
// should have the base address of the table location in bits
|
|
|
|
// 47:12, bits 4 and 2 must be cleared and bit 0 must be set.
|
|
|
|
//
|
|
|
|
if (FirstIn) {
|
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: Enabling ToPA scheme \n"));
|
2021-12-05 23:54:17 +01:00
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
Pages = EFI_SIZE_TO_PAGES (sizeof (PROC_TRACE_TOPA_TABLE));
|
|
|
|
Alignment = 0x1000;
|
2017-08-04 03:59:08 +02:00
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
if (ProcTraceData->EnableOnBspOnly) {
|
|
|
|
//
|
|
|
|
// When only enable ProcTrace on BSP, this is the first and only time ProcTraceInitialize() runs.
|
|
|
|
//
|
|
|
|
TopaTableBaseAddr = (UINTN)AllocateAlignedReservedPages (Pages, Alignment);
|
|
|
|
if (TopaTableBaseAddr == 0) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "ProcTrace: Out of mem, failed to allocate ToPA mem for BSP"));
|
|
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
|
|
}
|
2021-12-05 23:54:17 +01:00
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: Topa table address(aligned) for BSP only: 0x%llX \n", (UINT64)TopaTableBaseAddr));
|
|
|
|
} else {
|
|
|
|
//
|
|
|
|
// Let BSP allocate ToPA table mem for all threads
|
|
|
|
//
|
|
|
|
TopaMemArray = (UINTN *)AllocatePool (ProcTraceData->AllocatedThreads * sizeof (UINTN *));
|
|
|
|
if (TopaMemArray == NULL) {
|
|
|
|
DEBUG ((DEBUG_ERROR, "ProcTrace: Allocate mem for ToPA Failed\n"));
|
|
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProcTraceData->TopaMemArray = TopaMemArray;
|
|
|
|
|
|
|
|
for (Index = 0; Index < ProcTraceData->AllocatedThreads; Index++) {
|
|
|
|
AlignedAddress = (UINTN)AllocateAlignedReservedPages (Pages, Alignment);
|
|
|
|
if (AlignedAddress == 0) {
|
|
|
|
if (Index < ProcTraceData->AllocatedThreads) {
|
|
|
|
ProcTraceData->AllocatedThreads = Index;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG ((DEBUG_ERROR, "ProcTrace: Out of mem, allocated ToPA mem only for %d threads\n", ProcTraceData->AllocatedThreads));
|
|
|
|
if (Index == 0) {
|
|
|
|
//
|
|
|
|
// Could not allocate for BSP even
|
|
|
|
//
|
|
|
|
FreePool ((VOID *)TopaMemArray);
|
|
|
|
TopaMemArray = NULL;
|
|
|
|
return RETURN_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2017-08-04 03:59:08 +02:00
|
|
|
}
|
2021-12-05 23:54:17 +01:00
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
TopaMemArray[Index] = AlignedAddress;
|
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: Topa table address(aligned) for thread %d is 0x%llX \n", Index, (UINT64)TopaMemArray[Index]));
|
2017-08-04 03:59:08 +02:00
|
|
|
}
|
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
DEBUG ((DEBUG_INFO, "ProcTrace: Allocated ToPA mem for %d thread \n", ProcTraceData->AllocatedThreads));
|
2017-08-04 03:59:08 +02:00
|
|
|
}
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
}
|
|
|
|
|
2023-04-25 06:47:05 +02:00
|
|
|
if (!ProcTraceData->EnableOnBspOnly) {
|
|
|
|
if (ProcessorNumber < ProcTraceData->AllocatedThreads) {
|
|
|
|
TopaTableBaseAddr = ProcTraceData->TopaMemArray[ProcessorNumber];
|
|
|
|
} else {
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|
2017-08-04 03:59:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TopaTable = (PROC_TRACE_TOPA_TABLE *)TopaTableBaseAddr;
|
2017-08-18 05:19:03 +02:00
|
|
|
TopaEntryPtr = &TopaTable->TopaEntry[0];
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
TopaEntryPtr->Uint64 = 0;
|
2017-08-18 05:19:03 +02:00
|
|
|
TopaEntryPtr->Bits.Base = (MemRegionBaseAddr >> 12) & 0x000FFFFF;
|
|
|
|
TopaEntryPtr->Bits.BaseHi = RShiftU64 ((UINT64)MemRegionBaseAddr, 32) & 0xFFFFFFFF;
|
|
|
|
TopaEntryPtr->Bits.Size = ProcTraceData->ProcTraceMemSize;
|
|
|
|
TopaEntryPtr->Bits.END = 0;
|
|
|
|
|
|
|
|
TopaEntryPtr = &TopaTable->TopaEntry[1];
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
TopaEntryPtr->Uint64 = 0;
|
2017-08-18 05:19:03 +02:00
|
|
|
TopaEntryPtr->Bits.Base = (TopaTableBaseAddr >> 12) & 0x000FFFFF;
|
|
|
|
TopaEntryPtr->Bits.BaseHi = RShiftU64 ((UINT64)TopaTableBaseAddr, 32) & 0xFFFFFFFF;
|
|
|
|
TopaEntryPtr->Bits.END = 1;
|
2017-08-04 03:59:08 +02:00
|
|
|
|
|
|
|
//
|
2017-08-18 05:19:03 +02:00
|
|
|
// Program the MSR IA32_RTIT_OUTPUT_BASE (0x560) bits[63:7] with ToPA base
|
2017-08-04 03:59:08 +02:00
|
|
|
//
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
OutputBaseReg.Uint64 = ProcTraceData->ProcessorData[ProcessorNumber].RtitOutputBase.Uint64;
|
2017-08-18 05:19:03 +02:00
|
|
|
OutputBaseReg.Bits.Base = (TopaTableBaseAddr >> 7) & 0x01FFFFFF;
|
|
|
|
OutputBaseReg.Bits.BaseHi = RShiftU64 ((UINT64)TopaTableBaseAddr, 32) & 0xFFFFFFFF;
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_OUTPUT_BASE,
|
2017-08-18 05:19:03 +02:00
|
|
|
OutputBaseReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Set the MSR IA32_RTIT_OUTPUT_MASK (0x561) bits[63:7] to 0
|
|
|
|
//
|
UefiCpuPkg CpuCommFeaturesLib: Fix GP fault issue about ProcTrace
BZ: https://bugzilla.tianocore.org/show_bug.cgi?id=1808
In current code, the values of TopaEntryPtr->Uint64 for TopaTable
and the values of OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64
to register table write for RTIT_OUTPUT_BASE and RTIT_OUTPUT_MASK_PTRS
are not been initialized in whole. For example, the reserved bits in
OutputBaseReg.Uint64 are random that will cause GP fault like below
when SetProcessorRegister (in CpuFeaturesInitialize.c) sets register
based on register table.
!!!! X64 Exception Type - 0D(#GP - General Protection)
CPU Apic ID - 00000000 !!!!
ExceptionData - 0000000000000000
RIP -0000000064D69576, CS -0000000000000038, RFLAGS -0000000000010246
RAX -000000006B9F1001, RCX -0000000000000560, RDX -0000000000000000
RBX -0000000064EECA18, RSP -000000006CB82BA0, RBP -0000000000000008
RSI -0000000080000000, RDI -0000000000000011
R8 -000000006B9493D0, R9 -0000000000000010, R10 -00000000000000FF
R11 -000000006CB82A50, R12 -0000000064D70F50, R13 -0000000066547050
R14 -0000000064E3E198, R15 -0000000000000000
DS -0000000000000030, ES -0000000000000030, FS -0000000000000030
GS -0000000000000030, SS -0000000000000030
CR0 -0000000080010013, CR2 -0000000000000000, CR3 -000000006C601000
CR4 -0000000000000628, CR8 -0000000000000000
DR0 -0000000000000000, DR1 -0000000000000000, DR2 -0000000000000000
DR3 -0000000000000000, DR6 -00000000FFFF0FF0, DR7 -0000000000000400
GDTR -000000006B8CCF18 0000000000000047, LDTR -0000000000000000
IDTR -000000006687E018 0000000000000FFF, TR -0000000000000000
FXSAVE_STATE -000000006CB82800
And current code gets MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceInitialize() and uses their
values for all processors. But ProcTraceInitialize() is only executed
by BSP, that means the values just for BSP. For good practice, the code
should get MSR_IA32_RTIT_CTL, MSR_IA32_RTIT_OUTPUT_BASE and
MSR_IA32_RTIT_OUTPUT_MASK_PTRS in ProcTraceSupport (executed by all
processors), and then use them in ProcTraceInitialize() for all
processors. This can also resolve the issue that the values of
OutputBaseReg.Uint64 and OutputMaskPtrsReg.Uint64 are not been
initialized in whole.
For TopaEntryPtr->Uint64, this patch updates code to initialize it
in whole explicitly by TopaEntryPtr->Uint64 = 0 before updating its
fields.
At the same time, this patch also eliminates the ProcTraceSupported
field in PROC_TRACE_PROCESSOR_DATA and the TopaMemArrayCount field in
PROC_TRACE_DATA.
Cc: Laszlo Ersek <lersek@redhat.com>
Cc: Eric Dong <eric.dong@intel.com>
Cc: Ruiyu Ni <ruiyu.ni@intel.com>
Cc: Chandana Kumar <chandana.c.kumar@intel.com>
Cc: Kevin Li <kevin.y.li@intel.com>
Signed-off-by: Star Zeng <star.zeng@intel.com>
Reviewed-by: Eric Dong <eric.dong@intel.com>
2019-05-25 09:39:23 +02:00
|
|
|
OutputMaskPtrsReg.Uint64 = ProcTraceData->ProcessorData[ProcessorNumber].RtitOutputMaskPtrs.Uint64;
|
2017-08-18 05:19:03 +02:00
|
|
|
OutputMaskPtrsReg.Bits.MaskOrTableOffset = 0;
|
|
|
|
OutputMaskPtrsReg.Bits.OutputOffset = 0;
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_OUTPUT_MASK_PTRS,
|
2017-08-18 05:19:03 +02:00
|
|
|
OutputMaskPtrsReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
//
|
|
|
|
// Enable ToPA output scheme by enabling MSR IA32_RTIT_CTL (0x570) ToPA (Bit 8)
|
|
|
|
//
|
2017-08-18 05:19:03 +02:00
|
|
|
CtrlReg.Bits.ToPA = 1;
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_CTL,
|
2017-08-18 05:19:03 +02:00
|
|
|
CtrlReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Enable the Processor Trace feature from MSR IA32_RTIT_CTL (570h)
|
|
|
|
///
|
2017-08-18 05:19:03 +02:00
|
|
|
CtrlReg.Bits.OS = 1;
|
|
|
|
CtrlReg.Bits.User = 1;
|
|
|
|
CtrlReg.Bits.BranchEn = 1;
|
2019-05-25 09:05:47 +02:00
|
|
|
CtrlReg.Bits.TraceEn = 1;
|
2023-04-25 10:33:30 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Generate CYC/TSC timing packets to collect performance data.
|
|
|
|
//
|
|
|
|
if (ProcTraceData->EnablePerformanceCollecting) {
|
|
|
|
if (ProcTraceData->ProcessorData[ProcessorNumber].CycPacketSupported) {
|
|
|
|
CtrlReg.Bits.CYCEn = 1;
|
|
|
|
CtrlReg.Bits.CYCThresh = 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Write to TSCEn is always supported
|
|
|
|
//
|
|
|
|
CtrlReg.Bits.TSCEn = 1;
|
|
|
|
}
|
|
|
|
|
2017-08-04 03:59:08 +02:00
|
|
|
CPU_REGISTER_TABLE_WRITE64 (
|
|
|
|
ProcessorNumber,
|
|
|
|
Msr,
|
|
|
|
MSR_IA32_RTIT_CTL,
|
2017-08-18 05:19:03 +02:00
|
|
|
CtrlReg.Uint64
|
2017-08-04 03:59:08 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
return RETURN_SUCCESS;
|
|
|
|
}
|