* [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library @ 2016-10-28 17:38 Leo Duran 2016-10-28 18:42 ` Laszlo Ersek 2016-10-28 21:12 ` Kinney, Michael D 0 siblings, 2 replies; 7+ messages in thread From: Leo Duran @ 2016-10-28 17:38 UTC (permalink / raw) To: edk2-devel; +Cc: liming.gao, lersek, jeff.fan, michael.d.kinney, Leo Duran 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. 2) Remove ExtractProcessorLocation() from MpInitLib library. 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Leo Duran <leo.duran@amd.com> --- UefiCpuPkg/Include/Library/LocalApicLib.h | 18 +++ UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 130 +++++++++++++++++++++ .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 130 +++++++++++++++++++++ UefiCpuPkg/Library/MpInitLib/MpLib.c | 128 +------------------- UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 121 +------------------ 5 files changed, 280 insertions(+), 247 deletions(-) diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h b/UefiCpuPkg/Include/Library/LocalApicLib.h index cd4e613..4abf64c 100644 --- a/UefiCpuPkg/Include/Library/LocalApicLib.h +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h @@ -21,6 +21,9 @@ #define LOCAL_APIC_MODE_XAPIC 0x1 ///< xAPIC mode. #define LOCAL_APIC_MODE_X2APIC 0x2 ///< x2APIC mode. +#include <PiPei.h> +#include <Protocol/MpService.h> + /** Retrieve the base address of local APIC. @@ -410,6 +413,21 @@ GetApicMsiValue ( IN BOOLEAN LevelTriggered, IN BOOLEAN AssertionLevel ); + +/** +Get Package ID/Core ID/Thread ID of a processor. + +The algorithm assumes the target system has symmetry across physical package boundaries +with respect to the number of logical processors per package, number of cores per package. + +@param InitialApicId Must be the initial APIC ID of the target logical processor. +@param Location Returns the processor location information. +**/ +VOID +GetProcessorLocation( + IN UINT32 InitialApicId, + OUT EFI_CPU_PHYSICAL_LOCATION *Location +); #endif diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c index 8d0fb02..219f99f 100644 --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c @@ -941,3 +941,133 @@ GetApicMsiValue ( } return MsiData.Uint64; } + +/** +Get Package ID/Core ID/Thread ID of a processor. + +The algorithm assumes the target system has symmetry across physical package boundaries +with respect to the number of logical processors per package, number of cores per package. + +@param InitialApicId Must be the initial APIC ID of the target logical processor. +@param Location Returns the processor location information. +**/ +VOID +GetProcessorLocation( +IN UINT32 InitialApicId, +OUT EFI_CPU_PHYSICAL_LOCATION *Location +) +{ + BOOLEAN TopologyLeafSupported; + UINTN ThreadBits; + UINTN CoreBits; + CPUID_VERSION_INFO_EBX VersionInfoEbx; + CPUID_VERSION_INFO_EDX VersionInfoEdx; + CPUID_CACHE_PARAMS_EAX CacheParamsEax; + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; + UINT32 MaxCpuIdIndex; + UINT32 SubIndex; + UINTN LevelType; + UINT32 MaxLogicProcessorsPerPackage; + UINT32 MaxCoresPerPackage; + + // + // Check if the processor is capable of supporting more than one logical processor. + // + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); + if (VersionInfoEdx.Bits.HTT == 0) { + Location->Thread = 0; + Location->Core = 0; + Location->Package = 0; + return; + } + + ThreadBits = 0; + CoreBits = 0; + + // + // Assume three-level mapping of APIC ID: Package:Core:SMT. + // + + TopologyLeafSupported = FALSE; + // + // Get the max index of basic CPUID + // + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); + + // + // If the extended topology enumeration leaf is available, it + // is the preferred mechanism for enumerating topology. + // + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { + AsmCpuidEx( + CPUID_EXTENDED_TOPOLOGY, + 0, + &ExtendedTopologyEax.Uint32, + &ExtendedTopologyEbx.Uint32, + &ExtendedTopologyEcx.Uint32, + NULL + ); + // + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not + // supported on that processor. + // + if (ExtendedTopologyEbx.Uint32 != 0) { + TopologyLeafSupported = TRUE; + + // + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract + // the SMT sub-field of x2APIC ID. + // + LevelType = ExtendedTopologyEcx.Bits.LevelType; + ASSERT(LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; + + // + // Software must not assume any "level type" encoding + // value to be related to any sub-leaf index, except sub-leaf 0. + // + SubIndex = 1; + do { + AsmCpuidEx( + CPUID_EXTENDED_TOPOLOGY, + SubIndex, + &ExtendedTopologyEax.Uint32, + NULL, + &ExtendedTopologyEcx.Uint32, + NULL + ); + LevelType = ExtendedTopologyEcx.Bits.LevelType; + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; + break; + } + SubIndex++; + } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); + } + } + + if (!TopologyLeafSupported) { + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); + MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); + MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; + } + else { + // + // Must be a single-core processor. + // + MaxCoresPerPackage = 1; + } + + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); + } + + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); +} diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c index 4c42696..16395a3 100644 --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c @@ -1036,3 +1036,133 @@ GetApicMsiValue ( } return MsiData.Uint64; } + +/** +Get Package ID/Core ID/Thread ID of a processor. + +The algorithm assumes the target system has symmetry across physical package boundaries +with respect to the number of logical processors per package, number of cores per package. + +@param InitialApicId Must be the initial APIC ID of the target logical processor. +@param Location Returns the processor location information. +**/ +VOID +GetProcessorLocation( +IN UINT32 InitialApicId, +OUT EFI_CPU_PHYSICAL_LOCATION *Location +) +{ + BOOLEAN TopologyLeafSupported; + UINTN ThreadBits; + UINTN CoreBits; + CPUID_VERSION_INFO_EBX VersionInfoEbx; + CPUID_VERSION_INFO_EDX VersionInfoEdx; + CPUID_CACHE_PARAMS_EAX CacheParamsEax; + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; + UINT32 MaxCpuIdIndex; + UINT32 SubIndex; + UINTN LevelType; + UINT32 MaxLogicProcessorsPerPackage; + UINT32 MaxCoresPerPackage; + + // + // Check if the processor is capable of supporting more than one logical processor. + // + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); + if (VersionInfoEdx.Bits.HTT == 0) { + Location->Thread = 0; + Location->Core = 0; + Location->Package = 0; + return; + } + + ThreadBits = 0; + CoreBits = 0; + + // + // Assume three-level mapping of APIC ID: Package:Core:SMT. + // + + TopologyLeafSupported = FALSE; + // + // Get the max index of basic CPUID + // + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); + + // + // If the extended topology enumeration leaf is available, it + // is the preferred mechanism for enumerating topology. + // + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { + AsmCpuidEx( + CPUID_EXTENDED_TOPOLOGY, + 0, + &ExtendedTopologyEax.Uint32, + &ExtendedTopologyEbx.Uint32, + &ExtendedTopologyEcx.Uint32, + NULL + ); + // + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not + // supported on that processor. + // + if (ExtendedTopologyEbx.Uint32 != 0) { + TopologyLeafSupported = TRUE; + + // + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract + // the SMT sub-field of x2APIC ID. + // + LevelType = ExtendedTopologyEcx.Bits.LevelType; + ASSERT(LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; + + // + // Software must not assume any "level type" encoding + // value to be related to any sub-leaf index, except sub-leaf 0. + // + SubIndex = 1; + do { + AsmCpuidEx( + CPUID_EXTENDED_TOPOLOGY, + SubIndex, + &ExtendedTopologyEax.Uint32, + NULL, + &ExtendedTopologyEcx.Uint32, + NULL + ); + LevelType = ExtendedTopologyEcx.Bits.LevelType; + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; + break; + } + SubIndex++; + } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); + } + } + + if (!TopologyLeafSupported) { + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); + MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); + MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; + } + else { + // + // Must be a single-core processor. + // + MaxCoresPerPackage = 1; + } + + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); + } + + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); +} diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c b/UefiCpuPkg/Library/MpInitLib/MpLib.c index c3fe721..f3380bb 100644 --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( } /** - Get CPU Package/Core/Thread location information. - - @param[in] InitialApicId CPU APIC ID - @param[out] Location Pointer to CPU location information -**/ -VOID -ExtractProcessorLocation ( - IN UINT32 InitialApicId, - OUT EFI_CPU_PHYSICAL_LOCATION *Location - ) -{ - BOOLEAN TopologyLeafSupported; - UINTN ThreadBits; - UINTN CoreBits; - CPUID_VERSION_INFO_EBX VersionInfoEbx; - CPUID_VERSION_INFO_EDX VersionInfoEdx; - CPUID_CACHE_PARAMS_EAX CacheParamsEax; - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; - UINT32 MaxCpuIdIndex; - UINT32 SubIndex; - UINTN LevelType; - UINT32 MaxLogicProcessorsPerPackage; - UINT32 MaxCoresPerPackage; - - // - // Check if the processor is capable of supporting more than one logical processor. - // - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); - if (VersionInfoEdx.Bits.HTT == 0) { - Location->Thread = 0; - Location->Core = 0; - Location->Package = 0; - return; - } - - ThreadBits = 0; - CoreBits = 0; - - // - // Assume three-level mapping of APIC ID: Package:Core:SMT. - // - - TopologyLeafSupported = FALSE; - // - // Get the max index of basic CPUID - // - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); - - // - // If the extended topology enumeration leaf is available, it - // is the preferred mechanism for enumerating topology. - // - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { - AsmCpuidEx ( - CPUID_EXTENDED_TOPOLOGY, - 0, - &ExtendedTopologyEax.Uint32, - &ExtendedTopologyEbx.Uint32, - &ExtendedTopologyEcx.Uint32, - NULL - ); - // - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not - // supported on that processor. - // - if (ExtendedTopologyEbx.Uint32 != 0) { - TopologyLeafSupported = TRUE; - - // - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract - // the SMT sub-field of x2APIC ID. - // - LevelType = ExtendedTopologyEcx.Bits.LevelType; - ASSERT (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; - - // - // Software must not assume any "level type" encoding - // value to be related to any sub-leaf index, except sub-leaf 0. - // - SubIndex = 1; - do { - AsmCpuidEx ( - CPUID_EXTENDED_TOPOLOGY, - SubIndex, - &ExtendedTopologyEax.Uint32, - NULL, - &ExtendedTopologyEcx.Uint32, - NULL - ); - LevelType = ExtendedTopologyEcx.Bits.LevelType; - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; - break; - } - SubIndex++; - } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); - } - } - - if (!TopologyLeafSupported) { - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); - MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); - MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; - } else { - // - // Must be a single-core processor. - // - MaxCoresPerPackage = 1; - } - - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); - } - - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); -} - -/** Worker function for SwitchBSP(). Worker function for SwitchBSP(), assigned to the AP which is intended @@ -1451,7 +1325,7 @@ MpInitLibGetProcessorInfo ( // // Get processor location information // - ExtractProcessorLocation (CpuMpData->CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); + GetProcessorLocation (CpuMpData->CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); if (HealthData != NULL) { HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c index 40f2a17..67cd0a0 100644 --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL mSmmCpuService = { }; /** - Get Package ID/Core ID/Thread ID of a processor. - - APIC ID must be an initial APIC ID. - - The algorithm below assumes the target system has symmetry across physical package boundaries - with respect to the number of logical processors per package, number of cores per package. - - @param ApicId APIC ID of the target logical processor. - @param Location Returns the processor location information. -**/ -VOID -SmmGetProcessorLocation ( - IN UINT32 ApicId, - OUT EFI_CPU_PHYSICAL_LOCATION *Location - ) -{ - UINTN ThreadBits; - UINTN CoreBits; - UINT32 RegEax; - UINT32 RegEbx; - UINT32 RegEcx; - UINT32 RegEdx; - UINT32 MaxCpuIdIndex; - UINT32 SubIndex; - UINTN LevelType; - UINT32 MaxLogicProcessorsPerPackage; - UINT32 MaxCoresPerPackage; - BOOLEAN TopologyLeafSupported; - - ASSERT (Location != NULL); - - ThreadBits = 0; - CoreBits = 0; - TopologyLeafSupported = FALSE; - - // - // Check if the processor is capable of supporting more than one logical processor. - // - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); - ASSERT ((RegEdx & BIT28) != 0); - - // - // Assume three-level mapping of APIC ID: Package:Core:SMT. - // - - // - // Get the max index of basic CPUID - // - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); - - // - // If the extended topology enumeration leaf is available, it - // is the preferred mechanism for enumerating topology. - // - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, &RegEcx, NULL); - // - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not - // supported on that processor. - // - if ((RegEbx & 0xffff) != 0) { - TopologyLeafSupported = TRUE; - - // - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract - // the SMT sub-field of x2APIC ID. - // - LevelType = (RegEcx >> 8) & 0xff; - ASSERT (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); - if ((RegEbx & 0xffff) > 1 ) { - ThreadBits = RegEax & 0x1f; - } else { - // - // HT is not supported - // - ThreadBits = 0; - } - - // - // Software must not assume any "level type" encoding - // value to be related to any sub-leaf index, except sub-leaf 0. - // - SubIndex = 1; - do { - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, NULL, &RegEcx, NULL); - LevelType = (RegEcx >> 8) & 0xff; - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { - CoreBits = (RegEax & 0x1f) - ThreadBits; - break; - } - SubIndex++; - } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); - } - } - - if (!TopologyLeafSupported) { - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); - MaxCoresPerPackage = (RegEax >> 26) + 1; - } else { - // - // Must be a single-core processor. - // - MaxCoresPerPackage = 1; - } - - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); - } - - Location->Thread = ApicId & ~((-1) << ThreadBits); - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); -} - -/** Gets processor information on the requested processor at the instant this call is made. @param[in] This A pointer to the EFI_SMM_CPU_SERVICE_PROTOCOL instance. @@ -280,7 +161,7 @@ SmmAddProcessor ( gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == INVALID_APIC_ID) { gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate->ProcessorInfo[Index].Location); + GetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate->ProcessorInfo[Index].Location); *ProcessorNumber = Index; gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; -- 1.9.1 ^ permalink raw reply related [flat|nested] 7+ messages in thread
* Re: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library 2016-10-28 17:38 [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library Leo Duran @ 2016-10-28 18:42 ` Laszlo Ersek 2016-10-28 21:12 ` Kinney, Michael D 1 sibling, 0 replies; 7+ messages in thread From: Laszlo Ersek @ 2016-10-28 18:42 UTC (permalink / raw) To: Leo Duran, edk2-devel; +Cc: michael.d.kinney, jeff.fan, liming.gao On 10/28/16 19:38, Leo Duran wrote: > 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. > 2) Remove ExtractProcessorLocation() from MpInitLib library. > 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. > > Contributed-under: TianoCore Contribution Agreement 1.0 > Signed-off-by: Leo Duran <leo.duran@amd.com> > --- > UefiCpuPkg/Include/Library/LocalApicLib.h | 18 +++ > UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 130 +++++++++++++++++++++ > .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 130 +++++++++++++++++++++ > UefiCpuPkg/Library/MpInitLib/MpLib.c | 128 +------------------- > UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 121 +------------------ > 5 files changed, 280 insertions(+), 247 deletions(-) > > diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h b/UefiCpuPkg/Include/Library/LocalApicLib.h > index cd4e613..4abf64c 100644 > --- a/UefiCpuPkg/Include/Library/LocalApicLib.h > +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h > @@ -21,6 +21,9 @@ > #define LOCAL_APIC_MODE_XAPIC 0x1 ///< xAPIC mode. > #define LOCAL_APIC_MODE_X2APIC 0x2 ///< x2APIC mode. > > +#include <PiPei.h> > +#include <Protocol/MpService.h> > + > /** > Retrieve the base address of local APIC. > > @@ -410,6 +413,21 @@ GetApicMsiValue ( > IN BOOLEAN LevelTriggered, > IN BOOLEAN AssertionLevel > ); > + > +/** > +Get Package ID/Core ID/Thread ID of a processor. > + > +The algorithm assumes the target system has symmetry across physical package boundaries > +with respect to the number of logical processors per package, number of cores per package. > + > +@param InitialApicId Must be the initial APIC ID of the target logical processor. > +@param Location Returns the processor location information. > +**/ > +VOID > +GetProcessorLocation( > + IN UINT32 InitialApicId, > + OUT EFI_CPU_PHYSICAL_LOCATION *Location > +); > > #endif > > diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > index 8d0fb02..219f99f 100644 > --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > @@ -941,3 +941,133 @@ GetApicMsiValue ( > } > return MsiData.Uint64; > } > + > +/** > +Get Package ID/Core ID/Thread ID of a processor. > + > +The algorithm assumes the target system has symmetry across physical package boundaries > +with respect to the number of logical processors per package, number of cores per package. > + > +@param InitialApicId Must be the initial APIC ID of the target logical processor. > +@param Location Returns the processor location information. > +**/ > +VOID > +GetProcessorLocation( > +IN UINT32 InitialApicId, > +OUT EFI_CPU_PHYSICAL_LOCATION *Location > +) > +{ > + BOOLEAN TopologyLeafSupported; > + UINTN ThreadBits; > + UINTN CoreBits; > + CPUID_VERSION_INFO_EBX VersionInfoEbx; > + CPUID_VERSION_INFO_EDX VersionInfoEdx; > + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > + UINT32 MaxCpuIdIndex; > + UINT32 SubIndex; > + UINTN LevelType; > + UINT32 MaxLogicProcessorsPerPackage; > + UINT32 MaxCoresPerPackage; > + > + // > + // Check if the processor is capable of supporting more than one logical processor. > + // > + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); > + if (VersionInfoEdx.Bits.HTT == 0) { > + Location->Thread = 0; > + Location->Core = 0; > + Location->Package = 0; > + return; > + } > + > + ThreadBits = 0; > + CoreBits = 0; > + > + // > + // Assume three-level mapping of APIC ID: Package:Core:SMT. > + // > + > + TopologyLeafSupported = FALSE; > + // > + // Get the max index of basic CPUID > + // > + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > + > + // > + // If the extended topology enumeration leaf is available, it > + // is the preferred mechanism for enumerating topology. > + // > + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + 0, > + &ExtendedTopologyEax.Uint32, > + &ExtendedTopologyEbx.Uint32, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + // > + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for > + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > + // supported on that processor. > + // > + if (ExtendedTopologyEbx.Uint32 != 0) { > + TopologyLeafSupported = TRUE; > + > + // > + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract > + // the SMT sub-field of x2APIC ID. > + // > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + ASSERT(LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > + > + // > + // Software must not assume any "level type" encoding > + // value to be related to any sub-leaf index, except sub-leaf 0. > + // > + SubIndex = 1; > + do { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + SubIndex, > + &ExtendedTopologyEax.Uint32, > + NULL, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > + break; > + } > + SubIndex++; > + } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > + } > + } > + > + if (!TopologyLeafSupported) { > + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); > + MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); > + MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > + } > + else { > + // > + // Must be a single-core processor. > + // > + MaxCoresPerPackage = 1; > + } > + > + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); > + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); > + } > + > + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > +} > diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > index 4c42696..16395a3 100644 > --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > @@ -1036,3 +1036,133 @@ GetApicMsiValue ( > } > return MsiData.Uint64; > } > + > +/** > +Get Package ID/Core ID/Thread ID of a processor. > + > +The algorithm assumes the target system has symmetry across physical package boundaries > +with respect to the number of logical processors per package, number of cores per package. > + > +@param InitialApicId Must be the initial APIC ID of the target logical processor. > +@param Location Returns the processor location information. > +**/ > +VOID > +GetProcessorLocation( > +IN UINT32 InitialApicId, > +OUT EFI_CPU_PHYSICAL_LOCATION *Location > +) > +{ > + BOOLEAN TopologyLeafSupported; > + UINTN ThreadBits; > + UINTN CoreBits; > + CPUID_VERSION_INFO_EBX VersionInfoEbx; > + CPUID_VERSION_INFO_EDX VersionInfoEdx; > + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > + UINT32 MaxCpuIdIndex; > + UINT32 SubIndex; > + UINTN LevelType; > + UINT32 MaxLogicProcessorsPerPackage; > + UINT32 MaxCoresPerPackage; > + > + // > + // Check if the processor is capable of supporting more than one logical processor. > + // > + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); > + if (VersionInfoEdx.Bits.HTT == 0) { > + Location->Thread = 0; > + Location->Core = 0; > + Location->Package = 0; > + return; > + } > + > + ThreadBits = 0; > + CoreBits = 0; > + > + // > + // Assume three-level mapping of APIC ID: Package:Core:SMT. > + // > + > + TopologyLeafSupported = FALSE; > + // > + // Get the max index of basic CPUID > + // > + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > + > + // > + // If the extended topology enumeration leaf is available, it > + // is the preferred mechanism for enumerating topology. > + // > + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + 0, > + &ExtendedTopologyEax.Uint32, > + &ExtendedTopologyEbx.Uint32, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + // > + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for > + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > + // supported on that processor. > + // > + if (ExtendedTopologyEbx.Uint32 != 0) { > + TopologyLeafSupported = TRUE; > + > + // > + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract > + // the SMT sub-field of x2APIC ID. > + // > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + ASSERT(LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > + > + // > + // Software must not assume any "level type" encoding > + // value to be related to any sub-leaf index, except sub-leaf 0. > + // > + SubIndex = 1; > + do { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + SubIndex, > + &ExtendedTopologyEax.Uint32, > + NULL, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > + break; > + } > + SubIndex++; > + } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > + } > + } > + > + if (!TopologyLeafSupported) { > + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); > + MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); > + MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > + } > + else { > + // > + // Must be a single-core processor. > + // > + MaxCoresPerPackage = 1; > + } > + > + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); > + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); > + } > + > + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > +} > diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c b/UefiCpuPkg/Library/MpInitLib/MpLib.c > index c3fe721..f3380bb 100644 > --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c > +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c > @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( > } > > /** > - Get CPU Package/Core/Thread location information. > - > - @param[in] InitialApicId CPU APIC ID > - @param[out] Location Pointer to CPU location information > -**/ > -VOID > -ExtractProcessorLocation ( > - IN UINT32 InitialApicId, > - OUT EFI_CPU_PHYSICAL_LOCATION *Location > - ) > -{ > - BOOLEAN TopologyLeafSupported; > - UINTN ThreadBits; > - UINTN CoreBits; > - CPUID_VERSION_INFO_EBX VersionInfoEbx; > - CPUID_VERSION_INFO_EDX VersionInfoEdx; > - CPUID_CACHE_PARAMS_EAX CacheParamsEax; > - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > - UINT32 MaxCpuIdIndex; > - UINT32 SubIndex; > - UINTN LevelType; > - UINT32 MaxLogicProcessorsPerPackage; > - UINT32 MaxCoresPerPackage; > - > - // > - // Check if the processor is capable of supporting more than one logical processor. > - // > - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); > - if (VersionInfoEdx.Bits.HTT == 0) { > - Location->Thread = 0; > - Location->Core = 0; > - Location->Package = 0; > - return; > - } > - > - ThreadBits = 0; > - CoreBits = 0; > - > - // > - // Assume three-level mapping of APIC ID: Package:Core:SMT. > - // > - > - TopologyLeafSupported = FALSE; > - // > - // Get the max index of basic CPUID > - // > - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > - > - // > - // If the extended topology enumeration leaf is available, it > - // is the preferred mechanism for enumerating topology. > - // > - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > - AsmCpuidEx ( > - CPUID_EXTENDED_TOPOLOGY, > - 0, > - &ExtendedTopologyEax.Uint32, > - &ExtendedTopologyEbx.Uint32, > - &ExtendedTopologyEcx.Uint32, > - NULL > - ); > - // > - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for > - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > - // supported on that processor. > - // > - if (ExtendedTopologyEbx.Uint32 != 0) { > - TopologyLeafSupported = TRUE; > - > - // > - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract > - // the SMT sub-field of x2APIC ID. > - // > - LevelType = ExtendedTopologyEcx.Bits.LevelType; > - ASSERT (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > - > - // > - // Software must not assume any "level type" encoding > - // value to be related to any sub-leaf index, except sub-leaf 0. > - // > - SubIndex = 1; > - do { > - AsmCpuidEx ( > - CPUID_EXTENDED_TOPOLOGY, > - SubIndex, > - &ExtendedTopologyEax.Uint32, > - NULL, > - &ExtendedTopologyEcx.Uint32, > - NULL > - ); > - LevelType = ExtendedTopologyEcx.Bits.LevelType; > - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > - break; > - } > - SubIndex++; > - } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > - } > - } > - > - if (!TopologyLeafSupported) { > - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); > - MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); > - MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > - } else { > - // > - // Must be a single-core processor. > - // > - MaxCoresPerPackage = 1; > - } > - > - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); > - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > - } > - > - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > -} > - > -/** > Worker function for SwitchBSP(). > > Worker function for SwitchBSP(), assigned to the AP which is intended > @@ -1451,7 +1325,7 @@ MpInitLibGetProcessorInfo ( > // > // Get processor location information > // > - ExtractProcessorLocation (CpuMpData->CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); > + GetProcessorLocation (CpuMpData->CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); > > if (HealthData != NULL) { > HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; > diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > index 40f2a17..67cd0a0 100644 > --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL mSmmCpuService = { > }; > > /** > - Get Package ID/Core ID/Thread ID of a processor. > - > - APIC ID must be an initial APIC ID. > - > - The algorithm below assumes the target system has symmetry across physical package boundaries > - with respect to the number of logical processors per package, number of cores per package. > - > - @param ApicId APIC ID of the target logical processor. > - @param Location Returns the processor location information. > -**/ > -VOID > -SmmGetProcessorLocation ( > - IN UINT32 ApicId, > - OUT EFI_CPU_PHYSICAL_LOCATION *Location > - ) > -{ > - UINTN ThreadBits; > - UINTN CoreBits; > - UINT32 RegEax; > - UINT32 RegEbx; > - UINT32 RegEcx; > - UINT32 RegEdx; > - UINT32 MaxCpuIdIndex; > - UINT32 SubIndex; > - UINTN LevelType; > - UINT32 MaxLogicProcessorsPerPackage; > - UINT32 MaxCoresPerPackage; > - BOOLEAN TopologyLeafSupported; > - > - ASSERT (Location != NULL); > - > - ThreadBits = 0; > - CoreBits = 0; > - TopologyLeafSupported = FALSE; > - > - // > - // Check if the processor is capable of supporting more than one logical processor. > - // > - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); > - ASSERT ((RegEdx & BIT28) != 0); > - > - // > - // Assume three-level mapping of APIC ID: Package:Core:SMT. > - // > - > - // > - // Get the max index of basic CPUID > - // > - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > - > - // > - // If the extended topology enumeration leaf is available, it > - // is the preferred mechanism for enumerating topology. > - // > - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, &RegEcx, NULL); > - // > - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for > - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > - // supported on that processor. > - // > - if ((RegEbx & 0xffff) != 0) { > - TopologyLeafSupported = TRUE; > - > - // > - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract > - // the SMT sub-field of x2APIC ID. > - // > - LevelType = (RegEcx >> 8) & 0xff; > - ASSERT (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > - if ((RegEbx & 0xffff) > 1 ) { > - ThreadBits = RegEax & 0x1f; > - } else { > - // > - // HT is not supported > - // > - ThreadBits = 0; > - } > - > - // > - // Software must not assume any "level type" encoding > - // value to be related to any sub-leaf index, except sub-leaf 0. > - // > - SubIndex = 1; > - do { > - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, NULL, &RegEcx, NULL); > - LevelType = (RegEcx >> 8) & 0xff; > - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > - CoreBits = (RegEax & 0x1f) - ThreadBits; > - break; > - } > - SubIndex++; > - } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > - } > - } > - > - if (!TopologyLeafSupported) { > - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); > - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; > - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); > - MaxCoresPerPackage = (RegEax >> 26) + 1; > - } else { > - // > - // Must be a single-core processor. > - // > - MaxCoresPerPackage = 1; > - } > - > - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); > - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > - } > - > - Location->Thread = ApicId & ~((-1) << ThreadBits); > - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); > - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); > -} > - > -/** > Gets processor information on the requested processor at the instant this call is made. > > @param[in] This A pointer to the EFI_SMM_CPU_SERVICE_PROTOCOL instance. > @@ -280,7 +161,7 @@ SmmAddProcessor ( > gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == INVALID_APIC_ID) { > gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; > gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; > - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate->ProcessorInfo[Index].Location); > + GetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate->ProcessorInfo[Index].Location); > > *ProcessorNumber = Index; > gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; > I tested this as follows: * X64 OVMF, build without -D SMM_REQUIRE. I retested the MSR_IA32_FEATURE_CONTROL setting feature at <https://bugzilla.tianocore.org/show_bug.cgi?id=86> (commit range 51773d4..dbab994), with S3 suspend/resume, using QEMU's i440fx machine type. This test is relevant because it exercises EFI_PEI_MPSERVICES_PPI, provided by CpuMpPei, both at cold boot and at S3 resume. CpuMpPei in turn depends on MpInitLib. Here's the log snippet: > Loading PEIM at 0x000BFD29000 EntryPoint=0x000BFD29240 CpuMpPei.efi > AP Loop Mode is 1 > WakeupBufferStart = 9F000, WakeupBufferSize = 1000 > APIC MODE is 1 > MpInitLib: Find 8 processors in system. > Register PPI Notify: [EfiEndOfPeiSignalPpi] > Does not find any stored CPU BIST information from PPI! > APICID - 0x00000000, BIST - 0x00000000 > APICID - 0x00000001, BIST - 0x00000000 > APICID - 0x00000002, BIST - 0x00000000 > APICID - 0x00000003, BIST - 0x00000000 > APICID - 0x00000004, BIST - 0x00000000 > APICID - 0x00000005, BIST - 0x00000000 > APICID - 0x00000006, BIST - 0x00000000 > APICID - 0x00000007, BIST - 0x00000000 > Install PPI: [EfiSecPlatformInformation2Ppi] > Install PPI: [EfiPeiMpServicesPpi] > Notify: PPI Guid: [EfiPeiMpServicesPpi], Peim notify entry point: 83BAC5 > PlatformPei: OnMpServicesAvailable * Ia32X64 OVMF, built with -D SMM_REQUIRE, using the Q35 machine type. S3 suspend/resume works. SMM variable access works after resumption too, when bound to CPU#0 (BSP) or to CPU#1 (AP) alike (with "taskset -c 0/1" in the Fedora guest). > Loading SMM driver at 0x0007FFCE000 EntryPoint=0x0007FFCE271 PiSmmCpuDxeSmm.efi > SMRR Base: 0x7F800000, SMRR Size: 0x800000 > PcdCpuSmmCodeAccessCheckEnable = 1 > SMRAM TileSize = 0x00002000 (0x00001000, 0x00001000) > SMRAM SaveState Buffer (0x7FFC0000, 0x0000E000) > CPU[000] APIC ID=0000 SMBASE=7FFB8000 SaveState=7FFC7C00 Size=00000400 > CPU[001] APIC ID=0001 SMBASE=7FFBA000 SaveState=7FFC9C00 Size=00000400 > CPU[002] APIC ID=0002 SMBASE=7FFBC000 SaveState=7FFCBC00 Size=00000400 > CPU[003] APIC ID=0003 SMBASE=7FFBE000 SaveState=7FFCDC00 Size=00000400 > One Semaphore Size = 0x40 > Total Semaphores Size = 0x840 > InstallProtocolInterface: [gEfiSmmConfigurationProtocolGuid] 7FFDEFD0 > SMM IPL registered SMM Entry Point address 7FFF0A27 > SmmInstallProtocolInterface: [EfiSmmCpuProtocol] 7FFDEF10 > SmmInstallProtocolInterface: [EfiSmmCpuServiceProtocol] 7FFDEFF0 > SMM S3 SMRAM Structure = 7F0AF9C0 > SMM S3 Structure = 7F800000 > SMM CPU Module exit from SMRAM with EFI_SUCCESS * Same as the previous point, just with the Ia32 build. Tested-by: Laszlo Ersek <lersek@redhat.com> Thanks Laszlo ^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library 2016-10-28 17:38 [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library Leo Duran 2016-10-28 18:42 ` Laszlo Ersek @ 2016-10-28 21:12 ` Kinney, Michael D 2016-10-28 21:17 ` Duran, Leo 1 sibling, 1 reply; 7+ messages in thread From: Kinney, Michael D @ 2016-10-28 21:12 UTC (permalink / raw) To: Leo Duran, edk2-devel@lists.01.org Cc: Gao, Liming, lersek@redhat.com, Fan, Jeff Leo, I tested your patch on Quark and it builds and boots. Here is a slightly updated version of the patch that makes the following changes: * Follow Jeff Fan suggestion to pass in Package, Core, Thread as 3 parameters. This removes the dependency on PiPei.h and MpService.h from LocalApicLib.h so LocalApicLib can be used from more than just PEI/DXE phases. I also made these three OUT parameters OPTIONAL. * Change tabs to spaces and fix other minor format issues to pass checks from BaseTools/Scripts/PatchCheck.py I have also tested this updated patch on Quark and it also builds and boots. >From cc932ca811e052a0a3ae103a9db9614b3807d4ed Mon Sep 17 00:00:00 2001 From: Leo Duran <leo.duran@amd.com> Date: Fri, 28 Oct 2016 10:38:45 -0700 Subject: [Patch] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. 2) Remove ExtractProcessorLocation() from MpInitLib library. 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. Cc: Jeff Fan <jeff.fan@intel.com> Cc: Laszlo Ersek <lersek@redhat.com> Cc: Michael D Kinney <michael.d.kinney@intel.com> Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Leo Duran <leo.duran@amd.com> --- UefiCpuPkg/Include/Library/LocalApicLib.h | 21 +++ UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 148 +++++++++++++++++++++ .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 148 +++++++++++++++++++++ UefiCpuPkg/Library/MpInitLib/MpLib.c | 133 +----------------- UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 126 +----------------- 5 files changed, 329 insertions(+), 247 deletions(-) diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h b/UefiCpuPkg/Include/Library/LocalApicLib.h index cd4e613..d73198f 100644 --- a/UefiCpuPkg/Include/Library/LocalApicLib.h +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h @@ -410,6 +410,27 @@ GetApicMsiValue ( IN BOOLEAN LevelTriggered, IN BOOLEAN AssertionLevel ); + +/** + Get Package ID/Core ID/Thread ID of a processor. + + The algorithm assumes the target system has symmetry across physical package + boundaries with respect to the number of logical processors per package, + number of cores per package. + + @param[in] InitialApicId Must be the initial APIC ID of the target logical + processor. + @param[out] Package Returns the processor package ID. + @param[out] Core Returns the processor core ID. + @param[out] Thread Returns the processor thread ID. +**/ +VOID +GetProcessorLocation( + IN UINT32 InitialApicId, + OUT UINT32 *Package OPTIONAL, + OUT UINT32 *Core OPTIONAL, + OUT UINT32 *Thread OPTIONAL + ); #endif diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c index 8d0fb02..7ddee1b 100644 --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c @@ -941,3 +941,151 @@ GetApicMsiValue ( } return MsiData.Uint64; } + +/** + Get Package ID/Core ID/Thread ID of a processor. + + The algorithm assumes the target system has symmetry across physical package + boundaries with respect to the number of logical processors per package, + number of cores per package. + + @param[in] InitialApicId Must be the initial APIC ID of the target logical + processor. + @param[out] Package Returns the processor package ID. + @param[out] Core Returns the processor core ID. + @param[out] Thread Returns the processor thread ID. +**/ +VOID +GetProcessorLocation( + IN UINT32 InitialApicId, + OUT UINT32 *Package OPTIONAL, + OUT UINT32 *Core OPTIONAL, + OUT UINT32 *Thread OPTIONAL + ) +{ + BOOLEAN TopologyLeafSupported; + UINTN ThreadBits; + UINTN CoreBits; + CPUID_VERSION_INFO_EBX VersionInfoEbx; + CPUID_VERSION_INFO_EDX VersionInfoEdx; + CPUID_CACHE_PARAMS_EAX CacheParamsEax; + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; + UINT32 MaxCpuIdIndex; + UINT32 SubIndex; + UINTN LevelType; + UINT32 MaxLogicProcessorsPerPackage; + UINT32 MaxCoresPerPackage; + + // + // Check if the processor is capable of supporting more than one logical processor. + // + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); + if (VersionInfoEdx.Bits.HTT == 0) { + if (Thread != NULL) { + *Thread = 0; + } + if (Core != NULL) { + *Core = 0; + } + if (Package != NULL) { + *Package = 0; + } + return; + } + + ThreadBits = 0; + CoreBits = 0; + + // + // Assume three-level mapping of APIC ID: Package:Core:SMT. + // + + TopologyLeafSupported = FALSE; + // + // Get the max index of basic CPUID + // + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); + + // + // If the extended topology enumeration leaf is available, it + // is the preferred mechanism for enumerating topology. + // + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { + AsmCpuidEx( + CPUID_EXTENDED_TOPOLOGY, + 0, + &ExtendedTopologyEax.Uint32, + &ExtendedTopologyEbx.Uint32, + &ExtendedTopologyEcx.Uint32, + NULL + ); + // + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not + // supported on that processor. + // + if (ExtendedTopologyEbx.Uint32 != 0) { + TopologyLeafSupported = TRUE; + + // + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract + // the SMT sub-field of x2APIC ID. + // + LevelType = ExtendedTopologyEcx.Bits.LevelType; + ASSERT(LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; + + // + // Software must not assume any "level type" encoding + // value to be related to any sub-leaf index, except sub-leaf 0. + // + SubIndex = 1; + do { + AsmCpuidEx( + CPUID_EXTENDED_TOPOLOGY, + SubIndex, + &ExtendedTopologyEax.Uint32, + NULL, + &ExtendedTopologyEcx.Uint32, + NULL + ); + LevelType = ExtendedTopologyEcx.Bits.LevelType; + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; + break; + } + SubIndex++; + } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); + } + } + + if (!TopologyLeafSupported) { + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); + MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); + MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; + } + else { + // + // Must be a single-core processor. + // + MaxCoresPerPackage = 1; + } + + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); + } + + if (Thread != NULL) { + *Thread = InitialApicId & ((1 << ThreadBits) - 1); + } + if (Core != NULL) { + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); + } + if (Package != NULL) { + *Package = (InitialApicId >> (ThreadBits + CoreBits)); + } +} diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c index 4c42696..c620211 100644 --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c @@ -1036,3 +1036,151 @@ GetApicMsiValue ( } return MsiData.Uint64; } + +/** + Get Package ID/Core ID/Thread ID of a processor. + + The algorithm assumes the target system has symmetry across physical package + boundaries with respect to the number of logical processors per package, + number of cores per package. + + @param[in] InitialApicId Must be the initial APIC ID of the target logical + processor. + @param[out] Package Returns the processor package ID. + @param[out] Core Returns the processor core ID. + @param[out] Thread Returns the processor thread ID. +**/ +VOID +GetProcessorLocation( + IN UINT32 InitialApicId, + OUT UINT32 *Package OPTIONAL, + OUT UINT32 *Core OPTIONAL, + OUT UINT32 *Thread OPTIONAL + ) +{ + BOOLEAN TopologyLeafSupported; + UINTN ThreadBits; + UINTN CoreBits; + CPUID_VERSION_INFO_EBX VersionInfoEbx; + CPUID_VERSION_INFO_EDX VersionInfoEdx; + CPUID_CACHE_PARAMS_EAX CacheParamsEax; + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; + UINT32 MaxCpuIdIndex; + UINT32 SubIndex; + UINTN LevelType; + UINT32 MaxLogicProcessorsPerPackage; + UINT32 MaxCoresPerPackage; + + // + // Check if the processor is capable of supporting more than one logical processor. + // + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); + if (VersionInfoEdx.Bits.HTT == 0) { + if (Thread != NULL) { + *Thread = 0; + } + if (Core != NULL) { + *Core = 0; + } + if (Package != NULL) { + *Package = 0; + } + return; + } + + ThreadBits = 0; + CoreBits = 0; + + // + // Assume three-level mapping of APIC ID: Package:Core:SMT. + // + + TopologyLeafSupported = FALSE; + // + // Get the max index of basic CPUID + // + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); + + // + // If the extended topology enumeration leaf is available, it + // is the preferred mechanism for enumerating topology. + // + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { + AsmCpuidEx( + CPUID_EXTENDED_TOPOLOGY, + 0, + &ExtendedTopologyEax.Uint32, + &ExtendedTopologyEbx.Uint32, + &ExtendedTopologyEcx.Uint32, + NULL + ); + // + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not + // supported on that processor. + // + if (ExtendedTopologyEbx.Uint32 != 0) { + TopologyLeafSupported = TRUE; + + // + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract + // the SMT sub-field of x2APIC ID. + // + LevelType = ExtendedTopologyEcx.Bits.LevelType; + ASSERT(LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; + + // + // Software must not assume any "level type" encoding + // value to be related to any sub-leaf index, except sub-leaf 0. + // + SubIndex = 1; + do { + AsmCpuidEx( + CPUID_EXTENDED_TOPOLOGY, + SubIndex, + &ExtendedTopologyEax.Uint32, + NULL, + &ExtendedTopologyEcx.Uint32, + NULL + ); + LevelType = ExtendedTopologyEcx.Bits.LevelType; + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; + break; + } + SubIndex++; + } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); + } + } + + if (!TopologyLeafSupported) { + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); + MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); + MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; + } + else { + // + // Must be a single-core processor. + // + MaxCoresPerPackage = 1; + } + + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); + } + + if (Thread != NULL) { + *Thread = InitialApicId & ((1 << ThreadBits) - 1); + } + if (Core != NULL) { + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); + } + if (Package != NULL) { + *Package = (InitialApicId >> (ThreadBits + CoreBits)); + } +} diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c b/UefiCpuPkg/Library/MpInitLib/MpLib.c index c3fe721..f205b6b 100644 --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( } /** - Get CPU Package/Core/Thread location information. - - @param[in] InitialApicId CPU APIC ID - @param[out] Location Pointer to CPU location information -**/ -VOID -ExtractProcessorLocation ( - IN UINT32 InitialApicId, - OUT EFI_CPU_PHYSICAL_LOCATION *Location - ) -{ - BOOLEAN TopologyLeafSupported; - UINTN ThreadBits; - UINTN CoreBits; - CPUID_VERSION_INFO_EBX VersionInfoEbx; - CPUID_VERSION_INFO_EDX VersionInfoEdx; - CPUID_CACHE_PARAMS_EAX CacheParamsEax; - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; - UINT32 MaxCpuIdIndex; - UINT32 SubIndex; - UINTN LevelType; - UINT32 MaxLogicProcessorsPerPackage; - UINT32 MaxCoresPerPackage; - - // - // Check if the processor is capable of supporting more than one logical processor. - // - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); - if (VersionInfoEdx.Bits.HTT == 0) { - Location->Thread = 0; - Location->Core = 0; - Location->Package = 0; - return; - } - - ThreadBits = 0; - CoreBits = 0; - - // - // Assume three-level mapping of APIC ID: Package:Core:SMT. - // - - TopologyLeafSupported = FALSE; - // - // Get the max index of basic CPUID - // - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); - - // - // If the extended topology enumeration leaf is available, it - // is the preferred mechanism for enumerating topology. - // - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { - AsmCpuidEx ( - CPUID_EXTENDED_TOPOLOGY, - 0, - &ExtendedTopologyEax.Uint32, - &ExtendedTopologyEbx.Uint32, - &ExtendedTopologyEcx.Uint32, - NULL - ); - // - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not - // supported on that processor. - // - if (ExtendedTopologyEbx.Uint32 != 0) { - TopologyLeafSupported = TRUE; - - // - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract - // the SMT sub-field of x2APIC ID. - // - LevelType = ExtendedTopologyEcx.Bits.LevelType; - ASSERT (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; - - // - // Software must not assume any "level type" encoding - // value to be related to any sub-leaf index, except sub-leaf 0. - // - SubIndex = 1; - do { - AsmCpuidEx ( - CPUID_EXTENDED_TOPOLOGY, - SubIndex, - &ExtendedTopologyEax.Uint32, - NULL, - &ExtendedTopologyEcx.Uint32, - NULL - ); - LevelType = ExtendedTopologyEcx.Bits.LevelType; - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; - break; - } - SubIndex++; - } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); - } - } - - if (!TopologyLeafSupported) { - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); - MaxLogicProcessorsPerPackage = VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); - MaxCoresPerPackage = CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; - } else { - // - // Must be a single-core processor. - // - MaxCoresPerPackage = 1; - } - - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); - } - - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); -} - -/** Worker function for SwitchBSP(). Worker function for SwitchBSP(), assigned to the AP which is intended @@ -1451,7 +1325,12 @@ MpInitLibGetProcessorInfo ( // // Get processor location information // - ExtractProcessorLocation (CpuMpData->CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); + GetProcessorLocation ( + CpuMpData->CpuData[ProcessorNumber].ApicId, + &ProcessorInfoBuffer->Location.Package, + &ProcessorInfoBuffer->Location.Core, + &ProcessorInfoBuffer->Location.Thread + ); if (HealthData != NULL) { HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c index 40f2a17..93ebb9e 100644 --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL mSmmCpuService = { }; /** - Get Package ID/Core ID/Thread ID of a processor. - - APIC ID must be an initial APIC ID. - - The algorithm below assumes the target system has symmetry across physical package boundaries - with respect to the number of logical processors per package, number of cores per package. - - @param ApicId APIC ID of the target logical processor. - @param Location Returns the processor location information. -**/ -VOID -SmmGetProcessorLocation ( - IN UINT32 ApicId, - OUT EFI_CPU_PHYSICAL_LOCATION *Location - ) -{ - UINTN ThreadBits; - UINTN CoreBits; - UINT32 RegEax; - UINT32 RegEbx; - UINT32 RegEcx; - UINT32 RegEdx; - UINT32 MaxCpuIdIndex; - UINT32 SubIndex; - UINTN LevelType; - UINT32 MaxLogicProcessorsPerPackage; - UINT32 MaxCoresPerPackage; - BOOLEAN TopologyLeafSupported; - - ASSERT (Location != NULL); - - ThreadBits = 0; - CoreBits = 0; - TopologyLeafSupported = FALSE; - - // - // Check if the processor is capable of supporting more than one logical processor. - // - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); - ASSERT ((RegEdx & BIT28) != 0); - - // - // Assume three-level mapping of APIC ID: Package:Core:SMT. - // - - // - // Get the max index of basic CPUID - // - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); - - // - // If the extended topology enumeration leaf is available, it - // is the preferred mechanism for enumerating topology. - // - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, &RegEcx, NULL); - // - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not - // supported on that processor. - // - if ((RegEbx & 0xffff) != 0) { - TopologyLeafSupported = TRUE; - - // - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract - // the SMT sub-field of x2APIC ID. - // - LevelType = (RegEcx >> 8) & 0xff; - ASSERT (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); - if ((RegEbx & 0xffff) > 1 ) { - ThreadBits = RegEax & 0x1f; - } else { - // - // HT is not supported - // - ThreadBits = 0; - } - - // - // Software must not assume any "level type" encoding - // value to be related to any sub-leaf index, except sub-leaf 0. - // - SubIndex = 1; - do { - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, NULL, &RegEcx, NULL); - LevelType = (RegEcx >> 8) & 0xff; - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { - CoreBits = (RegEax & 0x1f) - ThreadBits; - break; - } - SubIndex++; - } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); - } - } - - if (!TopologyLeafSupported) { - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); - MaxCoresPerPackage = (RegEax >> 26) + 1; - } else { - // - // Must be a single-core processor. - // - MaxCoresPerPackage = 1; - } - - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / MaxCoresPerPackage - 1) + 1); - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); - } - - Location->Thread = ApicId & ~((-1) << ThreadBits); - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); -} - -/** Gets processor information on the requested processor at the instant this call is made. @param[in] This A pointer to the EFI_SMM_CPU_SERVICE_PROTOCOL instance. @@ -280,7 +161,12 @@ SmmAddProcessor ( gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == INVALID_APIC_ID) { gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate->ProcessorInfo[Index].Location); + GetProcessorLocation ( + (UINT32)ProcessorId, + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Package, + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Core, + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Thread + ); *ProcessorNumber = Index; gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; -- 2.6.3.windows.1 Best regards, Mike > -----Original Message----- > From: Leo Duran [mailto:leo.duran@amd.com] > Sent: Friday, October 28, 2016 10:39 AM > To: edk2-devel@lists.01.org > Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff > <jeff.fan@intel.com>; Kinney, Michael D <michael.d.kinney@intel.com>; Leo Duran > <leo.duran@amd.com> > Subject: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library > > 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. > 2) Remove ExtractProcessorLocation() from MpInitLib library. > 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. > > Contributed-under: TianoCore Contribution Agreement 1.0 > Signed-off-by: Leo Duran <leo.duran@amd.com> > --- > UefiCpuPkg/Include/Library/LocalApicLib.h | 18 +++ > UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 130 +++++++++++++++++++++ > .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 130 +++++++++++++++++++++ > UefiCpuPkg/Library/MpInitLib/MpLib.c | 128 +------------------- > UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 121 +------------------ > 5 files changed, 280 insertions(+), 247 deletions(-) > > diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h > b/UefiCpuPkg/Include/Library/LocalApicLib.h > index cd4e613..4abf64c 100644 > --- a/UefiCpuPkg/Include/Library/LocalApicLib.h > +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h > @@ -21,6 +21,9 @@ > #define LOCAL_APIC_MODE_XAPIC 0x1 ///< xAPIC mode. > #define LOCAL_APIC_MODE_X2APIC 0x2 ///< x2APIC mode. > > +#include <PiPei.h> > +#include <Protocol/MpService.h> > + > /** > Retrieve the base address of local APIC. > > @@ -410,6 +413,21 @@ GetApicMsiValue ( > IN BOOLEAN LevelTriggered, > IN BOOLEAN AssertionLevel > ); > + > +/** > +Get Package ID/Core ID/Thread ID of a processor. > + > +The algorithm assumes the target system has symmetry across physical package > boundaries > +with respect to the number of logical processors per package, number of cores per > package. > + > +@param InitialApicId Must be the initial APIC ID of the target logical processor. > +@param Location Returns the processor location information. > +**/ > +VOID > +GetProcessorLocation( > + IN UINT32 InitialApicId, > + OUT EFI_CPU_PHYSICAL_LOCATION *Location > +); > > #endif > > diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > index 8d0fb02..219f99f 100644 > --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > @@ -941,3 +941,133 @@ GetApicMsiValue ( > } > return MsiData.Uint64; > } > + > +/** > +Get Package ID/Core ID/Thread ID of a processor. > + > +The algorithm assumes the target system has symmetry across physical package > boundaries > +with respect to the number of logical processors per package, number of cores per > package. > + > +@param InitialApicId Must be the initial APIC ID of the target logical processor. > +@param Location Returns the processor location information. > +**/ > +VOID > +GetProcessorLocation( > +IN UINT32 InitialApicId, > +OUT EFI_CPU_PHYSICAL_LOCATION *Location > +) > +{ > + BOOLEAN TopologyLeafSupported; > + UINTN ThreadBits; > + UINTN CoreBits; > + CPUID_VERSION_INFO_EBX VersionInfoEbx; > + CPUID_VERSION_INFO_EDX VersionInfoEdx; > + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > + UINT32 MaxCpuIdIndex; > + UINT32 SubIndex; > + UINTN LevelType; > + UINT32 MaxLogicProcessorsPerPackage; > + UINT32 MaxCoresPerPackage; > + > + // > + // Check if the processor is capable of supporting more than one logical > processor. > + // > + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); > + if (VersionInfoEdx.Bits.HTT == 0) { > + Location->Thread = 0; > + Location->Core = 0; > + Location->Package = 0; > + return; > + } > + > + ThreadBits = 0; > + CoreBits = 0; > + > + // > + // Assume three-level mapping of APIC ID: Package:Core:SMT. > + // > + > + TopologyLeafSupported = FALSE; > + // > + // Get the max index of basic CPUID > + // > + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > + > + // > + // If the extended topology enumeration leaf is available, it > + // is the preferred mechanism for enumerating topology. > + // > + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + 0, > + &ExtendedTopologyEax.Uint32, > + &ExtendedTopologyEbx.Uint32, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + // > + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for > + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > + // supported on that processor. > + // > + if (ExtendedTopologyEbx.Uint32 != 0) { > + TopologyLeafSupported = TRUE; > + > + // > + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to > extract > + // the SMT sub-field of x2APIC ID. > + // > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + ASSERT(LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > + > + // > + // Software must not assume any "level type" encoding > + // value to be related to any sub-leaf index, except sub-leaf 0. > + // > + SubIndex = 1; > + do { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + SubIndex, > + &ExtendedTopologyEax.Uint32, > + NULL, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > + break; > + } > + SubIndex++; > + } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > + } > + } > + > + if (!TopologyLeafSupported) { > + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); > + MaxLogicProcessorsPerPackage = > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, > NULL); > + MaxCoresPerPackage = > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > + } > + else { > + // > + // Must be a single-core processor. > + // > + MaxCoresPerPackage = 1; > + } > + > + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > MaxCoresPerPackage - 1) + 1); > + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); > + } > + > + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > +} > diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > index 4c42696..16395a3 100644 > --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > @@ -1036,3 +1036,133 @@ GetApicMsiValue ( > } > return MsiData.Uint64; > } > + > +/** > +Get Package ID/Core ID/Thread ID of a processor. > + > +The algorithm assumes the target system has symmetry across physical package > boundaries > +with respect to the number of logical processors per package, number of cores per > package. > + > +@param InitialApicId Must be the initial APIC ID of the target logical processor. > +@param Location Returns the processor location information. > +**/ > +VOID > +GetProcessorLocation( > +IN UINT32 InitialApicId, > +OUT EFI_CPU_PHYSICAL_LOCATION *Location > +) > +{ > + BOOLEAN TopologyLeafSupported; > + UINTN ThreadBits; > + UINTN CoreBits; > + CPUID_VERSION_INFO_EBX VersionInfoEbx; > + CPUID_VERSION_INFO_EDX VersionInfoEdx; > + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > + UINT32 MaxCpuIdIndex; > + UINT32 SubIndex; > + UINTN LevelType; > + UINT32 MaxLogicProcessorsPerPackage; > + UINT32 MaxCoresPerPackage; > + > + // > + // Check if the processor is capable of supporting more than one logical > processor. > + // > + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); > + if (VersionInfoEdx.Bits.HTT == 0) { > + Location->Thread = 0; > + Location->Core = 0; > + Location->Package = 0; > + return; > + } > + > + ThreadBits = 0; > + CoreBits = 0; > + > + // > + // Assume three-level mapping of APIC ID: Package:Core:SMT. > + // > + > + TopologyLeafSupported = FALSE; > + // > + // Get the max index of basic CPUID > + // > + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > + > + // > + // If the extended topology enumeration leaf is available, it > + // is the preferred mechanism for enumerating topology. > + // > + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + 0, > + &ExtendedTopologyEax.Uint32, > + &ExtendedTopologyEbx.Uint32, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + // > + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for > + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > + // supported on that processor. > + // > + if (ExtendedTopologyEbx.Uint32 != 0) { > + TopologyLeafSupported = TRUE; > + > + // > + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to > extract > + // the SMT sub-field of x2APIC ID. > + // > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + ASSERT(LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > + > + // > + // Software must not assume any "level type" encoding > + // value to be related to any sub-leaf index, except sub-leaf 0. > + // > + SubIndex = 1; > + do { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + SubIndex, > + &ExtendedTopologyEax.Uint32, > + NULL, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > + break; > + } > + SubIndex++; > + } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > + } > + } > + > + if (!TopologyLeafSupported) { > + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); > + MaxLogicProcessorsPerPackage = > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, > NULL); > + MaxCoresPerPackage = > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > + } > + else { > + // > + // Must be a single-core processor. > + // > + MaxCoresPerPackage = 1; > + } > + > + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > MaxCoresPerPackage - 1) + 1); > + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); > + } > + > + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > +} > diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c > b/UefiCpuPkg/Library/MpInitLib/MpLib.c > index c3fe721..f3380bb 100644 > --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c > +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c > @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( > } > > /** > - Get CPU Package/Core/Thread location information. > - > - @param[in] InitialApicId CPU APIC ID > - @param[out] Location Pointer to CPU location information > -**/ > -VOID > -ExtractProcessorLocation ( > - IN UINT32 InitialApicId, > - OUT EFI_CPU_PHYSICAL_LOCATION *Location > - ) > -{ > - BOOLEAN TopologyLeafSupported; > - UINTN ThreadBits; > - UINTN CoreBits; > - CPUID_VERSION_INFO_EBX VersionInfoEbx; > - CPUID_VERSION_INFO_EDX VersionInfoEdx; > - CPUID_CACHE_PARAMS_EAX CacheParamsEax; > - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > - UINT32 MaxCpuIdIndex; > - UINT32 SubIndex; > - UINTN LevelType; > - UINT32 MaxLogicProcessorsPerPackage; > - UINT32 MaxCoresPerPackage; > - > - // > - // Check if the processor is capable of supporting more than one logical processor. > - // > - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &VersionInfoEdx.Uint32); > - if (VersionInfoEdx.Bits.HTT == 0) { > - Location->Thread = 0; > - Location->Core = 0; > - Location->Package = 0; > - return; > - } > - > - ThreadBits = 0; > - CoreBits = 0; > - > - // > - // Assume three-level mapping of APIC ID: Package:Core:SMT. > - // > - > - TopologyLeafSupported = FALSE; > - // > - // Get the max index of basic CPUID > - // > - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > - > - // > - // If the extended topology enumeration leaf is available, it > - // is the preferred mechanism for enumerating topology. > - // > - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > - AsmCpuidEx ( > - CPUID_EXTENDED_TOPOLOGY, > - 0, > - &ExtendedTopologyEax.Uint32, > - &ExtendedTopologyEbx.Uint32, > - &ExtendedTopologyEcx.Uint32, > - NULL > - ); > - // > - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for > - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > - // supported on that processor. > - // > - if (ExtendedTopologyEbx.Uint32 != 0) { > - TopologyLeafSupported = TRUE; > - > - // > - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract > - // the SMT sub-field of x2APIC ID. > - // > - LevelType = ExtendedTopologyEcx.Bits.LevelType; > - ASSERT (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > - > - // > - // Software must not assume any "level type" encoding > - // value to be related to any sub-leaf index, except sub-leaf 0. > - // > - SubIndex = 1; > - do { > - AsmCpuidEx ( > - CPUID_EXTENDED_TOPOLOGY, > - SubIndex, > - &ExtendedTopologyEax.Uint32, > - NULL, > - &ExtendedTopologyEcx.Uint32, > - NULL > - ); > - LevelType = ExtendedTopologyEcx.Bits.LevelType; > - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > - break; > - } > - SubIndex++; > - } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > - } > - } > - > - if (!TopologyLeafSupported) { > - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, NULL); > - MaxLogicProcessorsPerPackage = > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, NULL, NULL, NULL); > - MaxCoresPerPackage = > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > - } else { > - // > - // Must be a single-core processor. > - // > - MaxCoresPerPackage = 1; > - } > - > - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > MaxCoresPerPackage - 1) + 1); > - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > - } > - > - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > -} > - > -/** > Worker function for SwitchBSP(). > > Worker function for SwitchBSP(), assigned to the AP which is intended > @@ -1451,7 +1325,7 @@ MpInitLibGetProcessorInfo ( > // > // Get processor location information > // > - ExtractProcessorLocation (CpuMpData->CpuData[ProcessorNumber].ApicId, > &ProcessorInfoBuffer->Location); > + GetProcessorLocation (CpuMpData->CpuData[ProcessorNumber].ApicId, > &ProcessorInfoBuffer->Location); > > if (HealthData != NULL) { > HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; > diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > index 40f2a17..67cd0a0 100644 > --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL mSmmCpuService = { > }; > > /** > - Get Package ID/Core ID/Thread ID of a processor. > - > - APIC ID must be an initial APIC ID. > - > - The algorithm below assumes the target system has symmetry across physical package > boundaries > - with respect to the number of logical processors per package, number of cores per > package. > - > - @param ApicId APIC ID of the target logical processor. > - @param Location Returns the processor location information. > -**/ > -VOID > -SmmGetProcessorLocation ( > - IN UINT32 ApicId, > - OUT EFI_CPU_PHYSICAL_LOCATION *Location > - ) > -{ > - UINTN ThreadBits; > - UINTN CoreBits; > - UINT32 RegEax; > - UINT32 RegEbx; > - UINT32 RegEcx; > - UINT32 RegEdx; > - UINT32 MaxCpuIdIndex; > - UINT32 SubIndex; > - UINTN LevelType; > - UINT32 MaxLogicProcessorsPerPackage; > - UINT32 MaxCoresPerPackage; > - BOOLEAN TopologyLeafSupported; > - > - ASSERT (Location != NULL); > - > - ThreadBits = 0; > - CoreBits = 0; > - TopologyLeafSupported = FALSE; > - > - // > - // Check if the processor is capable of supporting more than one logical processor. > - // > - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); > - ASSERT ((RegEdx & BIT28) != 0); > - > - // > - // Assume three-level mapping of APIC ID: Package:Core:SMT. > - // > - > - // > - // Get the max index of basic CPUID > - // > - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > - > - // > - // If the extended topology enumeration leaf is available, it > - // is the preferred mechanism for enumerating topology. > - // > - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, &RegEcx, NULL); > - // > - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input value for > - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > - // supported on that processor. > - // > - if ((RegEbx & 0xffff) != 0) { > - TopologyLeafSupported = TRUE; > - > - // > - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to extract > - // the SMT sub-field of x2APIC ID. > - // > - LevelType = (RegEcx >> 8) & 0xff; > - ASSERT (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > - if ((RegEbx & 0xffff) > 1 ) { > - ThreadBits = RegEax & 0x1f; > - } else { > - // > - // HT is not supported > - // > - ThreadBits = 0; > - } > - > - // > - // Software must not assume any "level type" encoding > - // value to be related to any sub-leaf index, except sub-leaf 0. > - // > - SubIndex = 1; > - do { > - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, NULL, &RegEcx, NULL); > - LevelType = (RegEcx >> 8) & 0xff; > - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > - CoreBits = (RegEax & 0x1f) - ThreadBits; > - break; > - } > - SubIndex++; > - } while (LevelType != CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > - } > - } > - > - if (!TopologyLeafSupported) { > - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); > - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; > - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); > - MaxCoresPerPackage = (RegEax >> 26) + 1; > - } else { > - // > - // Must be a single-core processor. > - // > - MaxCoresPerPackage = 1; > - } > - > - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > MaxCoresPerPackage - 1) + 1); > - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > - } > - > - Location->Thread = ApicId & ~((-1) << ThreadBits); > - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); > - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); > -} > - > -/** > Gets processor information on the requested processor at the instant this call is > made. > > @param[in] This A pointer to the EFI_SMM_CPU_SERVICE_PROTOCOL > instance. > @@ -280,7 +161,7 @@ SmmAddProcessor ( > gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == INVALID_APIC_ID) { > gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; > gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; > - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- > >ProcessorInfo[Index].Location); > + GetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- > >ProcessorInfo[Index].Location); > > *ProcessorNumber = Index; > gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; > -- > 1.9.1 ^ permalink raw reply related [flat|nested] 7+ messages in thread
* Re: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library 2016-10-28 21:12 ` Kinney, Michael D @ 2016-10-28 21:17 ` Duran, Leo 2016-10-28 21:48 ` Laszlo Ersek 0 siblings, 1 reply; 7+ messages in thread From: Duran, Leo @ 2016-10-28 21:17 UTC (permalink / raw) To: 'Kinney, Michael D', edk2-devel@lists.01.org Cc: Gao, Liming, lersek@redhat.com, Fan, Jeff Replies below. Thanks. > -----Original Message----- > From: Kinney, Michael D [mailto:michael.d.kinney@intel.com] > Sent: Friday, October 28, 2016 4:12 PM > To: Duran, Leo <leo.duran@amd.com>; edk2-devel@lists.01.org > Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff > <jeff.fan@intel.com> > Subject: RE: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib > library > > Leo, > > I tested your patch on Quark and it builds and boots. > > Here is a slightly updated version of the patch that makes the following > changes: > > * Follow Jeff Fan suggestion to pass in Package, Core, Thread as 3 > parameters. > This removes the dependency on PiPei.h and MpService.h from > LocalApicLib.h > so LocalApicLib can be used from more than just PEI/DXE phases. I also > made > these three OUT parameters OPTIONAL. > * Change tabs to spaces and fix other minor format issues to pass checks > from > BaseTools/Scripts/PatchCheck.py > > I have also tested this updated patch on Quark and it also builds and boots. > > [Duran, Leo] Sounds a good thing! (I assume Lazlo will probably test this as well) Thanks Mike. Leo. > > From cc932ca811e052a0a3ae103a9db9614b3807d4ed Mon Sep 17 00:00:00 > 2001 > From: Leo Duran <leo.duran@amd.com> > Date: Fri, 28 Oct 2016 10:38:45 -0700 > Subject: [Patch] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib > library > > 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. > 2) Remove ExtractProcessorLocation() from MpInitLib library. > 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. > > Cc: Jeff Fan <jeff.fan@intel.com> > Cc: Laszlo Ersek <lersek@redhat.com> > Cc: Michael D Kinney <michael.d.kinney@intel.com> > Contributed-under: TianoCore Contribution Agreement 1.0 > Signed-off-by: Leo Duran <leo.duran@amd.com> > --- > UefiCpuPkg/Include/Library/LocalApicLib.h | 21 +++ > UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 148 > +++++++++++++++++++++ > .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 148 > +++++++++++++++++++++ > UefiCpuPkg/Library/MpInitLib/MpLib.c | 133 +----------------- > UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 126 +----------------- > 5 files changed, 329 insertions(+), 247 deletions(-) > > diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h > b/UefiCpuPkg/Include/Library/LocalApicLib.h > index cd4e613..d73198f 100644 > --- a/UefiCpuPkg/Include/Library/LocalApicLib.h > +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h > @@ -410,6 +410,27 @@ GetApicMsiValue ( > IN BOOLEAN LevelTriggered, > IN BOOLEAN AssertionLevel > ); > + > +/** > + Get Package ID/Core ID/Thread ID of a processor. > + > + The algorithm assumes the target system has symmetry across physical > + package boundaries with respect to the number of logical processors > + per package, number of cores per package. > + > + @param[in] InitialApicId Must be the initial APIC ID of the target logical > + processor. > + @param[out] Package Returns the processor package ID. > + @param[out] Core Returns the processor core ID. > + @param[out] Thread Returns the processor thread ID. > +**/ > +VOID > +GetProcessorLocation( > + IN UINT32 InitialApicId, > + OUT UINT32 *Package OPTIONAL, > + OUT UINT32 *Core OPTIONAL, > + OUT UINT32 *Thread OPTIONAL > + ); > > #endif > > diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > index 8d0fb02..7ddee1b 100644 > --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > @@ -941,3 +941,151 @@ GetApicMsiValue ( > } > return MsiData.Uint64; > } > + > +/** > + Get Package ID/Core ID/Thread ID of a processor. > + > + The algorithm assumes the target system has symmetry across physical > + package boundaries with respect to the number of logical processors > + per package, number of cores per package. > + > + @param[in] InitialApicId Must be the initial APIC ID of the target logical > + processor. > + @param[out] Package Returns the processor package ID. > + @param[out] Core Returns the processor core ID. > + @param[out] Thread Returns the processor thread ID. > +**/ > +VOID > +GetProcessorLocation( > + IN UINT32 InitialApicId, > + OUT UINT32 *Package OPTIONAL, > + OUT UINT32 *Core OPTIONAL, > + OUT UINT32 *Thread OPTIONAL > + ) > +{ > + BOOLEAN TopologyLeafSupported; > + UINTN ThreadBits; > + UINTN CoreBits; > + CPUID_VERSION_INFO_EBX VersionInfoEbx; > + CPUID_VERSION_INFO_EDX VersionInfoEdx; > + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > + UINT32 MaxCpuIdIndex; > + UINT32 SubIndex; > + UINTN LevelType; > + UINT32 MaxLogicProcessorsPerPackage; > + UINT32 MaxCoresPerPackage; > + > + // > + // Check if the processor is capable of supporting more than one logical > processor. > + // > + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, > + &VersionInfoEdx.Uint32); if (VersionInfoEdx.Bits.HTT == 0) { > + if (Thread != NULL) { > + *Thread = 0; > + } > + if (Core != NULL) { > + *Core = 0; > + } > + if (Package != NULL) { > + *Package = 0; > + } > + return; > + } > + > + ThreadBits = 0; > + CoreBits = 0; > + > + // > + // Assume three-level mapping of APIC ID: Package:Core:SMT. > + // > + > + TopologyLeafSupported = FALSE; > + // > + // Get the max index of basic CPUID > + // > + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > + > + // > + // If the extended topology enumeration leaf is available, it // is > + the preferred mechanism for enumerating topology. > + // > + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + 0, > + &ExtendedTopologyEax.Uint32, > + &ExtendedTopologyEbx.Uint32, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + // > + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > value for > + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > + // supported on that processor. > + // > + if (ExtendedTopologyEbx.Uint32 != 0) { > + TopologyLeafSupported = TRUE; > + > + // > + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters > to extract > + // the SMT sub-field of x2APIC ID. > + // > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + ASSERT(LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > + > + // > + // Software must not assume any "level type" encoding > + // value to be related to any sub-leaf index, except sub-leaf 0. > + // > + SubIndex = 1; > + do { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + SubIndex, > + &ExtendedTopologyEax.Uint32, > + NULL, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > + break; > + } > + SubIndex++; > + } while (LevelType != > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > + } > + } > + > + if (!TopologyLeafSupported) { > + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, > NULL); > + MaxLogicProcessorsPerPackage = > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, > NULL, NULL, NULL); > + MaxCoresPerPackage = > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > + } > + else { > + // > + // Must be a single-core processor. > + // > + MaxCoresPerPackage = 1; > + } > + > + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > MaxCoresPerPackage - 1) + 1); > + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); } > + > + if (Thread != NULL) { > + *Thread = InitialApicId & ((1 << ThreadBits) - 1); > + } > + if (Core != NULL) { > + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > + } > + if (Package != NULL) { > + *Package = (InitialApicId >> (ThreadBits + CoreBits)); > + } > +} > diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > index 4c42696..c620211 100644 > --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > @@ -1036,3 +1036,151 @@ GetApicMsiValue ( > } > return MsiData.Uint64; > } > + > +/** > + Get Package ID/Core ID/Thread ID of a processor. > + > + The algorithm assumes the target system has symmetry across physical > + package boundaries with respect to the number of logical processors > + per package, number of cores per package. > + > + @param[in] InitialApicId Must be the initial APIC ID of the target logical > + processor. > + @param[out] Package Returns the processor package ID. > + @param[out] Core Returns the processor core ID. > + @param[out] Thread Returns the processor thread ID. > +**/ > +VOID > +GetProcessorLocation( > + IN UINT32 InitialApicId, > + OUT UINT32 *Package OPTIONAL, > + OUT UINT32 *Core OPTIONAL, > + OUT UINT32 *Thread OPTIONAL > + ) > +{ > + BOOLEAN TopologyLeafSupported; > + UINTN ThreadBits; > + UINTN CoreBits; > + CPUID_VERSION_INFO_EBX VersionInfoEbx; > + CPUID_VERSION_INFO_EDX VersionInfoEdx; > + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > + UINT32 MaxCpuIdIndex; > + UINT32 SubIndex; > + UINTN LevelType; > + UINT32 MaxLogicProcessorsPerPackage; > + UINT32 MaxCoresPerPackage; > + > + // > + // Check if the processor is capable of supporting more than one logical > processor. > + // > + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, > + &VersionInfoEdx.Uint32); if (VersionInfoEdx.Bits.HTT == 0) { > + if (Thread != NULL) { > + *Thread = 0; > + } > + if (Core != NULL) { > + *Core = 0; > + } > + if (Package != NULL) { > + *Package = 0; > + } > + return; > + } > + > + ThreadBits = 0; > + CoreBits = 0; > + > + // > + // Assume three-level mapping of APIC ID: Package:Core:SMT. > + // > + > + TopologyLeafSupported = FALSE; > + // > + // Get the max index of basic CPUID > + // > + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > + > + // > + // If the extended topology enumeration leaf is available, it // is > + the preferred mechanism for enumerating topology. > + // > + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + 0, > + &ExtendedTopologyEax.Uint32, > + &ExtendedTopologyEbx.Uint32, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + // > + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > value for > + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > + // supported on that processor. > + // > + if (ExtendedTopologyEbx.Uint32 != 0) { > + TopologyLeafSupported = TRUE; > + > + // > + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters > to extract > + // the SMT sub-field of x2APIC ID. > + // > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + ASSERT(LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > + > + // > + // Software must not assume any "level type" encoding > + // value to be related to any sub-leaf index, except sub-leaf 0. > + // > + SubIndex = 1; > + do { > + AsmCpuidEx( > + CPUID_EXTENDED_TOPOLOGY, > + SubIndex, > + &ExtendedTopologyEax.Uint32, > + NULL, > + &ExtendedTopologyEcx.Uint32, > + NULL > + ); > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > + break; > + } > + SubIndex++; > + } while (LevelType != > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > + } > + } > + > + if (!TopologyLeafSupported) { > + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, > NULL); > + MaxLogicProcessorsPerPackage = > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, > NULL, NULL, NULL); > + MaxCoresPerPackage = > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > + } > + else { > + // > + // Must be a single-core processor. > + // > + MaxCoresPerPackage = 1; > + } > + > + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > MaxCoresPerPackage - 1) + 1); > + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); } > + > + if (Thread != NULL) { > + *Thread = InitialApicId & ((1 << ThreadBits) - 1); > + } > + if (Core != NULL) { > + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > + } > + if (Package != NULL) { > + *Package = (InitialApicId >> (ThreadBits + CoreBits)); > + } > +} > diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c > b/UefiCpuPkg/Library/MpInitLib/MpLib.c > index c3fe721..f205b6b 100644 > --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c > +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c > @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( } > > /** > - Get CPU Package/Core/Thread location information. > - > - @param[in] InitialApicId CPU APIC ID > - @param[out] Location Pointer to CPU location information > -**/ > -VOID > -ExtractProcessorLocation ( > - IN UINT32 InitialApicId, > - OUT EFI_CPU_PHYSICAL_LOCATION *Location > - ) > -{ > - BOOLEAN TopologyLeafSupported; > - UINTN ThreadBits; > - UINTN CoreBits; > - CPUID_VERSION_INFO_EBX VersionInfoEbx; > - CPUID_VERSION_INFO_EDX VersionInfoEdx; > - CPUID_CACHE_PARAMS_EAX CacheParamsEax; > - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > - UINT32 MaxCpuIdIndex; > - UINT32 SubIndex; > - UINTN LevelType; > - UINT32 MaxLogicProcessorsPerPackage; > - UINT32 MaxCoresPerPackage; > - > - // > - // Check if the processor is capable of supporting more than one logical > processor. > - // > - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, > &VersionInfoEdx.Uint32); > - if (VersionInfoEdx.Bits.HTT == 0) { > - Location->Thread = 0; > - Location->Core = 0; > - Location->Package = 0; > - return; > - } > - > - ThreadBits = 0; > - CoreBits = 0; > - > - // > - // Assume three-level mapping of APIC ID: Package:Core:SMT. > - // > - > - TopologyLeafSupported = FALSE; > - // > - // Get the max index of basic CPUID > - // > - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > - > - // > - // If the extended topology enumeration leaf is available, it > - // is the preferred mechanism for enumerating topology. > - // > - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > - AsmCpuidEx ( > - CPUID_EXTENDED_TOPOLOGY, > - 0, > - &ExtendedTopologyEax.Uint32, > - &ExtendedTopologyEbx.Uint32, > - &ExtendedTopologyEcx.Uint32, > - NULL > - ); > - // > - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > value for > - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > - // supported on that processor. > - // > - if (ExtendedTopologyEbx.Uint32 != 0) { > - TopologyLeafSupported = TRUE; > - > - // > - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to > extract > - // the SMT sub-field of x2APIC ID. > - // > - LevelType = ExtendedTopologyEcx.Bits.LevelType; > - ASSERT (LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > - > - // > - // Software must not assume any "level type" encoding > - // value to be related to any sub-leaf index, except sub-leaf 0. > - // > - SubIndex = 1; > - do { > - AsmCpuidEx ( > - CPUID_EXTENDED_TOPOLOGY, > - SubIndex, > - &ExtendedTopologyEax.Uint32, > - NULL, > - &ExtendedTopologyEcx.Uint32, > - NULL > - ); > - LevelType = ExtendedTopologyEcx.Bits.LevelType; > - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > - break; > - } > - SubIndex++; > - } while (LevelType != > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > - } > - } > - > - if (!TopologyLeafSupported) { > - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, > NULL); > - MaxLogicProcessorsPerPackage = > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, > NULL, NULL, NULL); > - MaxCoresPerPackage = > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > - } else { > - // > - // Must be a single-core processor. > - // > - MaxCoresPerPackage = 1; > - } > - > - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > MaxCoresPerPackage - 1) + 1); > - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > - } > - > - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); -} > - > -/** > Worker function for SwitchBSP(). > > Worker function for SwitchBSP(), assigned to the AP which is intended @@ > -1451,7 +1325,12 @@ MpInitLibGetProcessorInfo ( > // > // Get processor location information > // > - ExtractProcessorLocation (CpuMpData- > >CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); > + GetProcessorLocation ( > + CpuMpData->CpuData[ProcessorNumber].ApicId, > + &ProcessorInfoBuffer->Location.Package, > + &ProcessorInfoBuffer->Location.Core, > + &ProcessorInfoBuffer->Location.Thread > + ); > > if (HealthData != NULL) { > HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; > diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > index 40f2a17..93ebb9e 100644 > --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL > mSmmCpuService = { }; > > /** > - Get Package ID/Core ID/Thread ID of a processor. > - > - APIC ID must be an initial APIC ID. > - > - The algorithm below assumes the target system has symmetry across > physical package boundaries > - with respect to the number of logical processors per package, number of > cores per package. > - > - @param ApicId APIC ID of the target logical processor. > - @param Location Returns the processor location information. > -**/ > -VOID > -SmmGetProcessorLocation ( > - IN UINT32 ApicId, > - OUT EFI_CPU_PHYSICAL_LOCATION *Location > - ) > -{ > - UINTN ThreadBits; > - UINTN CoreBits; > - UINT32 RegEax; > - UINT32 RegEbx; > - UINT32 RegEcx; > - UINT32 RegEdx; > - UINT32 MaxCpuIdIndex; > - UINT32 SubIndex; > - UINTN LevelType; > - UINT32 MaxLogicProcessorsPerPackage; > - UINT32 MaxCoresPerPackage; > - BOOLEAN TopologyLeafSupported; > - > - ASSERT (Location != NULL); > - > - ThreadBits = 0; > - CoreBits = 0; > - TopologyLeafSupported = FALSE; > - > - // > - // Check if the processor is capable of supporting more than one logical > processor. > - // > - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); > - ASSERT ((RegEdx & BIT28) != 0); > - > - // > - // Assume three-level mapping of APIC ID: Package:Core:SMT. > - // > - > - // > - // Get the max index of basic CPUID > - // > - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > - > - // > - // If the extended topology enumeration leaf is available, it > - // is the preferred mechanism for enumerating topology. > - // > - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, > &RegEcx, NULL); > - // > - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > value for > - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not > - // supported on that processor. > - // > - if ((RegEbx & 0xffff) != 0) { > - TopologyLeafSupported = TRUE; > - > - // > - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to > extract > - // the SMT sub-field of x2APIC ID. > - // > - LevelType = (RegEcx >> 8) & 0xff; > - ASSERT (LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > - if ((RegEbx & 0xffff) > 1 ) { > - ThreadBits = RegEax & 0x1f; > - } else { > - // > - // HT is not supported > - // > - ThreadBits = 0; > - } > - > - // > - // Software must not assume any "level type" encoding > - // value to be related to any sub-leaf index, except sub-leaf 0. > - // > - SubIndex = 1; > - do { > - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, NULL, > &RegEcx, NULL); > - LevelType = (RegEcx >> 8) & 0xff; > - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > - CoreBits = (RegEax & 0x1f) - ThreadBits; > - break; > - } > - SubIndex++; > - } while (LevelType != > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > - } > - } > - > - if (!TopologyLeafSupported) { > - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); > - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; > - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); > - MaxCoresPerPackage = (RegEax >> 26) + 1; > - } else { > - // > - // Must be a single-core processor. > - // > - MaxCoresPerPackage = 1; > - } > - > - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > MaxCoresPerPackage - 1) + 1); > - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > - } > - > - Location->Thread = ApicId & ~((-1) << ThreadBits); > - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); > - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); -} > - > -/** > Gets processor information on the requested processor at the instant this > call is made. > > @param[in] This A pointer to the > EFI_SMM_CPU_SERVICE_PROTOCOL instance. > @@ -280,7 +161,12 @@ SmmAddProcessor ( > gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == > INVALID_APIC_ID) { > gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; > gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; > - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- > >ProcessorInfo[Index].Location); > + GetProcessorLocation ( > + (UINT32)ProcessorId, > + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Package, > + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Core, > + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Thread > + ); > > *ProcessorNumber = Index; > gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; > -- > 2.6.3.windows.1 > > > Best regards, > > Mike > > > -----Original Message----- > > From: Leo Duran [mailto:leo.duran@amd.com] > > Sent: Friday, October 28, 2016 10:39 AM > > To: edk2-devel@lists.01.org > > Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff > > <jeff.fan@intel.com>; Kinney, Michael D <michael.d.kinney@intel.com>; > Leo Duran > > <leo.duran@amd.com> > > Subject: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib > library > > > > 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. > > 2) Remove ExtractProcessorLocation() from MpInitLib library. > > 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. > > > > Contributed-under: TianoCore Contribution Agreement 1.0 > > Signed-off-by: Leo Duran <leo.duran@amd.com> > > --- > > UefiCpuPkg/Include/Library/LocalApicLib.h | 18 +++ > > UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 130 > +++++++++++++++++++++ > > .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 130 > +++++++++++++++++++++ > > UefiCpuPkg/Library/MpInitLib/MpLib.c | 128 +------------------- > > UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 121 +------------------ > > 5 files changed, 280 insertions(+), 247 deletions(-) > > > > diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h > > b/UefiCpuPkg/Include/Library/LocalApicLib.h > > index cd4e613..4abf64c 100644 > > --- a/UefiCpuPkg/Include/Library/LocalApicLib.h > > +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h > > @@ -21,6 +21,9 @@ > > #define LOCAL_APIC_MODE_XAPIC 0x1 ///< xAPIC mode. > > #define LOCAL_APIC_MODE_X2APIC 0x2 ///< x2APIC mode. > > > > +#include <PiPei.h> > > +#include <Protocol/MpService.h> > > + > > /** > > Retrieve the base address of local APIC. > > > > @@ -410,6 +413,21 @@ GetApicMsiValue ( > > IN BOOLEAN LevelTriggered, > > IN BOOLEAN AssertionLevel > > ); > > + > > +/** > > +Get Package ID/Core ID/Thread ID of a processor. > > + > > +The algorithm assumes the target system has symmetry across physical > package > > boundaries > > +with respect to the number of logical processors per package, number of > cores per > > package. > > + > > +@param InitialApicId Must be the initial APIC ID of the target logical > processor. > > +@param Location Returns the processor location information. > > +**/ > > +VOID > > +GetProcessorLocation( > > + IN UINT32 InitialApicId, > > + OUT EFI_CPU_PHYSICAL_LOCATION *Location > > +); > > > > #endif > > > > diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > > b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > > index 8d0fb02..219f99f 100644 > > --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > > +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > > @@ -941,3 +941,133 @@ GetApicMsiValue ( > > } > > return MsiData.Uint64; > > } > > + > > +/** > > +Get Package ID/Core ID/Thread ID of a processor. > > + > > +The algorithm assumes the target system has symmetry across physical > package > > boundaries > > +with respect to the number of logical processors per package, number of > cores per > > package. > > + > > +@param InitialApicId Must be the initial APIC ID of the target logical > processor. > > +@param Location Returns the processor location information. > > +**/ > > +VOID > > +GetProcessorLocation( > > +IN UINT32 InitialApicId, > > +OUT EFI_CPU_PHYSICAL_LOCATION *Location > > +) > > +{ > > + BOOLEAN TopologyLeafSupported; > > + UINTN ThreadBits; > > + UINTN CoreBits; > > + CPUID_VERSION_INFO_EBX VersionInfoEbx; > > + CPUID_VERSION_INFO_EDX VersionInfoEdx; > > + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > > + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > > + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > > + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > > + UINT32 MaxCpuIdIndex; > > + UINT32 SubIndex; > > + UINTN LevelType; > > + UINT32 MaxLogicProcessorsPerPackage; > > + UINT32 MaxCoresPerPackage; > > + > > + // > > + // Check if the processor is capable of supporting more than one > logical > > processor. > > + // > > + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, > &VersionInfoEdx.Uint32); > > + if (VersionInfoEdx.Bits.HTT == 0) { > > + Location->Thread = 0; > > + Location->Core = 0; > > + Location->Package = 0; > > + return; > > + } > > + > > + ThreadBits = 0; > > + CoreBits = 0; > > + > > + // > > + // Assume three-level mapping of APIC ID: Package:Core:SMT. > > + // > > + > > + TopologyLeafSupported = FALSE; > > + // > > + // Get the max index of basic CPUID > > + // > > + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, > NULL); > > + > > + // > > + // If the extended topology enumeration leaf is available, it > > + // is the preferred mechanism for enumerating topology. > > + // > > + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > > + AsmCpuidEx( > > + CPUID_EXTENDED_TOPOLOGY, > > + 0, > > + &ExtendedTopologyEax.Uint32, > > + &ExtendedTopologyEbx.Uint32, > > + &ExtendedTopologyEcx.Uint32, > > + NULL > > + ); > > + // > > + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and > maximum input value for > > + // basic CPUID information is greater than 0BH, then > CPUID.0BH leaf is not > > + // supported on that processor. > > + // > > + if (ExtendedTopologyEbx.Uint32 != 0) { > > + TopologyLeafSupported = TRUE; > > + > > + // > > + // Sub-leaf index 0 (ECX= 0 as input) provides > enumeration parameters to > > extract > > + // the SMT sub-field of x2APIC ID. > > + // > > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > > + ASSERT(LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > > + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > > + > > + // > > + // Software must not assume any "level type" > encoding > > + // value to be related to any sub-leaf index, except > sub-leaf 0. > > + // > > + SubIndex = 1; > > + do { > > + AsmCpuidEx( > > + CPUID_EXTENDED_TOPOLOGY, > > + SubIndex, > > + &ExtendedTopologyEax.Uint32, > > + NULL, > > + &ExtendedTopologyEcx.Uint32, > > + NULL > > + ); > > + LevelType = > ExtendedTopologyEcx.Bits.LevelType; > > + if (LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > > + CoreBits = > ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > > + break; > > + } > > + SubIndex++; > > + } while (LevelType != > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > > + } > > + } > > + > > + if (!TopologyLeafSupported) { > > + AsmCpuid(CPUID_VERSION_INFO, NULL, > &VersionInfoEbx.Uint32, NULL, NULL); > > + MaxLogicProcessorsPerPackage = > > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > > + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > > + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, > &CacheParamsEax.Uint32, NULL, NULL, > > NULL); > > + MaxCoresPerPackage = > > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > > + } > > + else { > > + // > > + // Must be a single-core processor. > > + // > > + MaxCoresPerPackage = 1; > > + } > > + > > + ThreadBits = > (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > > MaxCoresPerPackage - 1) + 1); > > + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + > 1); > > + } > > + > > + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > > + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > > + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > > +} > > diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > > b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > > index 4c42696..16395a3 100644 > > --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > > +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > > @@ -1036,3 +1036,133 @@ GetApicMsiValue ( > > } > > return MsiData.Uint64; > > } > > + > > +/** > > +Get Package ID/Core ID/Thread ID of a processor. > > + > > +The algorithm assumes the target system has symmetry across physical > package > > boundaries > > +with respect to the number of logical processors per package, number of > cores per > > package. > > + > > +@param InitialApicId Must be the initial APIC ID of the target logical > processor. > > +@param Location Returns the processor location information. > > +**/ > > +VOID > > +GetProcessorLocation( > > +IN UINT32 InitialApicId, > > +OUT EFI_CPU_PHYSICAL_LOCATION *Location > > +) > > +{ > > + BOOLEAN TopologyLeafSupported; > > + UINTN ThreadBits; > > + UINTN CoreBits; > > + CPUID_VERSION_INFO_EBX VersionInfoEbx; > > + CPUID_VERSION_INFO_EDX VersionInfoEdx; > > + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > > + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > > + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > > + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > > + UINT32 MaxCpuIdIndex; > > + UINT32 SubIndex; > > + UINTN LevelType; > > + UINT32 MaxLogicProcessorsPerPackage; > > + UINT32 MaxCoresPerPackage; > > + > > + // > > + // Check if the processor is capable of supporting more than one > logical > > processor. > > + // > > + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, > &VersionInfoEdx.Uint32); > > + if (VersionInfoEdx.Bits.HTT == 0) { > > + Location->Thread = 0; > > + Location->Core = 0; > > + Location->Package = 0; > > + return; > > + } > > + > > + ThreadBits = 0; > > + CoreBits = 0; > > + > > + // > > + // Assume three-level mapping of APIC ID: Package:Core:SMT. > > + // > > + > > + TopologyLeafSupported = FALSE; > > + // > > + // Get the max index of basic CPUID > > + // > > + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, > NULL); > > + > > + // > > + // If the extended topology enumeration leaf is available, it > > + // is the preferred mechanism for enumerating topology. > > + // > > + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > > + AsmCpuidEx( > > + CPUID_EXTENDED_TOPOLOGY, > > + 0, > > + &ExtendedTopologyEax.Uint32, > > + &ExtendedTopologyEbx.Uint32, > > + &ExtendedTopologyEcx.Uint32, > > + NULL > > + ); > > + // > > + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and > maximum input value for > > + // basic CPUID information is greater than 0BH, then > CPUID.0BH leaf is not > > + // supported on that processor. > > + // > > + if (ExtendedTopologyEbx.Uint32 != 0) { > > + TopologyLeafSupported = TRUE; > > + > > + // > > + // Sub-leaf index 0 (ECX= 0 as input) provides > enumeration parameters to > > extract > > + // the SMT sub-field of x2APIC ID. > > + // > > + LevelType = ExtendedTopologyEcx.Bits.LevelType; > > + ASSERT(LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > > + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > > + > > + // > > + // Software must not assume any "level type" > encoding > > + // value to be related to any sub-leaf index, except > sub-leaf 0. > > + // > > + SubIndex = 1; > > + do { > > + AsmCpuidEx( > > + CPUID_EXTENDED_TOPOLOGY, > > + SubIndex, > > + &ExtendedTopologyEax.Uint32, > > + NULL, > > + &ExtendedTopologyEcx.Uint32, > > + NULL > > + ); > > + LevelType = > ExtendedTopologyEcx.Bits.LevelType; > > + if (LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > > + CoreBits = > ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > > + break; > > + } > > + SubIndex++; > > + } while (LevelType != > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > > + } > > + } > > + > > + if (!TopologyLeafSupported) { > > + AsmCpuid(CPUID_VERSION_INFO, NULL, > &VersionInfoEbx.Uint32, NULL, NULL); > > + MaxLogicProcessorsPerPackage = > > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > > + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > > + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, > &CacheParamsEax.Uint32, NULL, NULL, > > NULL); > > + MaxCoresPerPackage = > > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > > + } > > + else { > > + // > > + // Must be a single-core processor. > > + // > > + MaxCoresPerPackage = 1; > > + } > > + > > + ThreadBits = > (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > > MaxCoresPerPackage - 1) + 1); > > + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + > 1); > > + } > > + > > + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > > + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > > + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > > +} > > diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c > > b/UefiCpuPkg/Library/MpInitLib/MpLib.c > > index c3fe721..f3380bb 100644 > > --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c > > +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c > > @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( > > } > > > > /** > > - Get CPU Package/Core/Thread location information. > > - > > - @param[in] InitialApicId CPU APIC ID > > - @param[out] Location Pointer to CPU location information > > -**/ > > -VOID > > -ExtractProcessorLocation ( > > - IN UINT32 InitialApicId, > > - OUT EFI_CPU_PHYSICAL_LOCATION *Location > > - ) > > -{ > > - BOOLEAN TopologyLeafSupported; > > - UINTN ThreadBits; > > - UINTN CoreBits; > > - CPUID_VERSION_INFO_EBX VersionInfoEbx; > > - CPUID_VERSION_INFO_EDX VersionInfoEdx; > > - CPUID_CACHE_PARAMS_EAX CacheParamsEax; > > - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > > - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > > - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > > - UINT32 MaxCpuIdIndex; > > - UINT32 SubIndex; > > - UINTN LevelType; > > - UINT32 MaxLogicProcessorsPerPackage; > > - UINT32 MaxCoresPerPackage; > > - > > - // > > - // Check if the processor is capable of supporting more than one logical > processor. > > - // > > - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, > &VersionInfoEdx.Uint32); > > - if (VersionInfoEdx.Bits.HTT == 0) { > > - Location->Thread = 0; > > - Location->Core = 0; > > - Location->Package = 0; > > - return; > > - } > > - > > - ThreadBits = 0; > > - CoreBits = 0; > > - > > - // > > - // Assume three-level mapping of APIC ID: Package:Core:SMT. > > - // > > - > > - TopologyLeafSupported = FALSE; > > - // > > - // Get the max index of basic CPUID > > - // > > - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > > - > > - // > > - // If the extended topology enumeration leaf is available, it > > - // is the preferred mechanism for enumerating topology. > > - // > > - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > > - AsmCpuidEx ( > > - CPUID_EXTENDED_TOPOLOGY, > > - 0, > > - &ExtendedTopologyEax.Uint32, > > - &ExtendedTopologyEbx.Uint32, > > - &ExtendedTopologyEcx.Uint32, > > - NULL > > - ); > > - // > > - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > value for > > - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is > not > > - // supported on that processor. > > - // > > - if (ExtendedTopologyEbx.Uint32 != 0) { > > - TopologyLeafSupported = TRUE; > > - > > - // > > - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters > to extract > > - // the SMT sub-field of x2APIC ID. > > - // > > - LevelType = ExtendedTopologyEcx.Bits.LevelType; > > - ASSERT (LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > > - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > > - > > - // > > - // Software must not assume any "level type" encoding > > - // value to be related to any sub-leaf index, except sub-leaf 0. > > - // > > - SubIndex = 1; > > - do { > > - AsmCpuidEx ( > > - CPUID_EXTENDED_TOPOLOGY, > > - SubIndex, > > - &ExtendedTopologyEax.Uint32, > > - NULL, > > - &ExtendedTopologyEcx.Uint32, > > - NULL > > - ); > > - LevelType = ExtendedTopologyEcx.Bits.LevelType; > > - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > > - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > > - break; > > - } > > - SubIndex++; > > - } while (LevelType != > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > > - } > > - } > > - > > - if (!TopologyLeafSupported) { > > - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, > NULL, NULL); > > - MaxLogicProcessorsPerPackage = > > VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > > - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > > - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, > NULL, NULL, NULL); > > - MaxCoresPerPackage = > > CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > > - } else { > > - // > > - // Must be a single-core processor. > > - // > > - MaxCoresPerPackage = 1; > > - } > > - > > - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > > MaxCoresPerPackage - 1) + 1); > > - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > > - } > > - > > - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > > - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > > - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > > -} > > - > > -/** > > Worker function for SwitchBSP(). > > > > Worker function for SwitchBSP(), assigned to the AP which is intended > > @@ -1451,7 +1325,7 @@ MpInitLibGetProcessorInfo ( > > // > > // Get processor location information > > // > > - ExtractProcessorLocation (CpuMpData- > >CpuData[ProcessorNumber].ApicId, > > &ProcessorInfoBuffer->Location); > > + GetProcessorLocation (CpuMpData- > >CpuData[ProcessorNumber].ApicId, > > &ProcessorInfoBuffer->Location); > > > > if (HealthData != NULL) { > > HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; > > diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > > b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > > index 40f2a17..67cd0a0 100644 > > --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > > +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > > @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL > mSmmCpuService = { > > }; > > > > /** > > - Get Package ID/Core ID/Thread ID of a processor. > > - > > - APIC ID must be an initial APIC ID. > > - > > - The algorithm below assumes the target system has symmetry across > physical package > > boundaries > > - with respect to the number of logical processors per package, number of > cores per > > package. > > - > > - @param ApicId APIC ID of the target logical processor. > > - @param Location Returns the processor location information. > > -**/ > > -VOID > > -SmmGetProcessorLocation ( > > - IN UINT32 ApicId, > > - OUT EFI_CPU_PHYSICAL_LOCATION *Location > > - ) > > -{ > > - UINTN ThreadBits; > > - UINTN CoreBits; > > - UINT32 RegEax; > > - UINT32 RegEbx; > > - UINT32 RegEcx; > > - UINT32 RegEdx; > > - UINT32 MaxCpuIdIndex; > > - UINT32 SubIndex; > > - UINTN LevelType; > > - UINT32 MaxLogicProcessorsPerPackage; > > - UINT32 MaxCoresPerPackage; > > - BOOLEAN TopologyLeafSupported; > > - > > - ASSERT (Location != NULL); > > - > > - ThreadBits = 0; > > - CoreBits = 0; > > - TopologyLeafSupported = FALSE; > > - > > - // > > - // Check if the processor is capable of supporting more than one logical > processor. > > - // > > - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); > > - ASSERT ((RegEdx & BIT28) != 0); > > - > > - // > > - // Assume three-level mapping of APIC ID: Package:Core:SMT. > > - // > > - > > - // > > - // Get the max index of basic CPUID > > - // > > - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > > - > > - // > > - // If the extended topology enumeration leaf is available, it > > - // is the preferred mechanism for enumerating topology. > > - // > > - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > > - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, > &RegEcx, NULL); > > - // > > - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > value for > > - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is > not > > - // supported on that processor. > > - // > > - if ((RegEbx & 0xffff) != 0) { > > - TopologyLeafSupported = TRUE; > > - > > - // > > - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters > to extract > > - // the SMT sub-field of x2APIC ID. > > - // > > - LevelType = (RegEcx >> 8) & 0xff; > > - ASSERT (LevelType == > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > > - if ((RegEbx & 0xffff) > 1 ) { > > - ThreadBits = RegEax & 0x1f; > > - } else { > > - // > > - // HT is not supported > > - // > > - ThreadBits = 0; > > - } > > - > > - // > > - // Software must not assume any "level type" encoding > > - // value to be related to any sub-leaf index, except sub-leaf 0. > > - // > > - SubIndex = 1; > > - do { > > - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, > NULL, &RegEcx, NULL); > > - LevelType = (RegEcx >> 8) & 0xff; > > - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > > - CoreBits = (RegEax & 0x1f) - ThreadBits; > > - break; > > - } > > - SubIndex++; > > - } while (LevelType != > CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > > - } > > - } > > - > > - if (!TopologyLeafSupported) { > > - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); > > - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; > > - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > > - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); > > - MaxCoresPerPackage = (RegEax >> 26) + 1; > > - } else { > > - // > > - // Must be a single-core processor. > > - // > > - MaxCoresPerPackage = 1; > > - } > > - > > - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > > MaxCoresPerPackage - 1) + 1); > > - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > > - } > > - > > - Location->Thread = ApicId & ~((-1) << ThreadBits); > > - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); > > - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); > > -} > > - > > -/** > > Gets processor information on the requested processor at the instant this > call is > > made. > > > > @param[in] This A pointer to the > EFI_SMM_CPU_SERVICE_PROTOCOL > > instance. > > @@ -280,7 +161,7 @@ SmmAddProcessor ( > > gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == > INVALID_APIC_ID) { > > gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; > > gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; > > - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- > > >ProcessorInfo[Index].Location); > > + GetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- > > >ProcessorInfo[Index].Location); > > > > *ProcessorNumber = Index; > > gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; > > -- > > 1.9.1 ^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library 2016-10-28 21:17 ` Duran, Leo @ 2016-10-28 21:48 ` Laszlo Ersek 2016-10-31 6:40 ` Fan, Jeff 0 siblings, 1 reply; 7+ messages in thread From: Laszlo Ersek @ 2016-10-28 21:48 UTC (permalink / raw) To: Duran, Leo, 'Kinney, Michael D', edk2-devel@lists.01.org Cc: Gao, Liming, Fan, Jeff On 10/28/16 23:17, Duran, Leo wrote: > Replies below. > Thanks. > >> -----Original Message----- >> From: Kinney, Michael D [mailto:michael.d.kinney@intel.com] >> Sent: Friday, October 28, 2016 4:12 PM >> To: Duran, Leo <leo.duran@amd.com>; edk2-devel@lists.01.org >> Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff >> <jeff.fan@intel.com> >> Subject: RE: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib >> library >> >> Leo, >> >> I tested your patch on Quark and it builds and boots. >> >> Here is a slightly updated version of the patch that makes the following >> changes: >> >> * Follow Jeff Fan suggestion to pass in Package, Core, Thread as 3 >> parameters. >> This removes the dependency on PiPei.h and MpService.h from >> LocalApicLib.h >> so LocalApicLib can be used from more than just PEI/DXE phases. I also >> made >> these three OUT parameters OPTIONAL. >> * Change tabs to spaces and fix other minor format issues to pass checks >> from >> BaseTools/Scripts/PatchCheck.py >> >> I have also tested this updated patch on Quark and it also builds and boots. >> >> > [Duran, Leo] > Sounds a good thing! > (I assume Lazlo will probably test this as well) If Mike thinks the changes warrant it, I certainly can (could be next week), but for that, I'd need a regularly formatted & mailed patch, one that I can apply with "git am" from the list. Thanks Laszlo > Thanks Mike. > Leo. > >> >> From cc932ca811e052a0a3ae103a9db9614b3807d4ed Mon Sep 17 00:00:00 >> 2001 >> From: Leo Duran <leo.duran@amd.com> >> Date: Fri, 28 Oct 2016 10:38:45 -0700 >> Subject: [Patch] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib >> library >> >> 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. >> 2) Remove ExtractProcessorLocation() from MpInitLib library. >> 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. >> >> Cc: Jeff Fan <jeff.fan@intel.com> >> Cc: Laszlo Ersek <lersek@redhat.com> >> Cc: Michael D Kinney <michael.d.kinney@intel.com> >> Contributed-under: TianoCore Contribution Agreement 1.0 >> Signed-off-by: Leo Duran <leo.duran@amd.com> >> --- >> UefiCpuPkg/Include/Library/LocalApicLib.h | 21 +++ >> UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 148 >> +++++++++++++++++++++ >> .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 148 >> +++++++++++++++++++++ >> UefiCpuPkg/Library/MpInitLib/MpLib.c | 133 +----------------- >> UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 126 +----------------- >> 5 files changed, 329 insertions(+), 247 deletions(-) >> >> diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h >> b/UefiCpuPkg/Include/Library/LocalApicLib.h >> index cd4e613..d73198f 100644 >> --- a/UefiCpuPkg/Include/Library/LocalApicLib.h >> +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h >> @@ -410,6 +410,27 @@ GetApicMsiValue ( >> IN BOOLEAN LevelTriggered, >> IN BOOLEAN AssertionLevel >> ); >> + >> +/** >> + Get Package ID/Core ID/Thread ID of a processor. >> + >> + The algorithm assumes the target system has symmetry across physical >> + package boundaries with respect to the number of logical processors >> + per package, number of cores per package. >> + >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical >> + processor. >> + @param[out] Package Returns the processor package ID. >> + @param[out] Core Returns the processor core ID. >> + @param[out] Thread Returns the processor thread ID. >> +**/ >> +VOID >> +GetProcessorLocation( >> + IN UINT32 InitialApicId, >> + OUT UINT32 *Package OPTIONAL, >> + OUT UINT32 *Core OPTIONAL, >> + OUT UINT32 *Thread OPTIONAL >> + ); >> >> #endif >> >> diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >> b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >> index 8d0fb02..7ddee1b 100644 >> --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >> +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >> @@ -941,3 +941,151 @@ GetApicMsiValue ( >> } >> return MsiData.Uint64; >> } >> + >> +/** >> + Get Package ID/Core ID/Thread ID of a processor. >> + >> + The algorithm assumes the target system has symmetry across physical >> + package boundaries with respect to the number of logical processors >> + per package, number of cores per package. >> + >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical >> + processor. >> + @param[out] Package Returns the processor package ID. >> + @param[out] Core Returns the processor core ID. >> + @param[out] Thread Returns the processor thread ID. >> +**/ >> +VOID >> +GetProcessorLocation( >> + IN UINT32 InitialApicId, >> + OUT UINT32 *Package OPTIONAL, >> + OUT UINT32 *Core OPTIONAL, >> + OUT UINT32 *Thread OPTIONAL >> + ) >> +{ >> + BOOLEAN TopologyLeafSupported; >> + UINTN ThreadBits; >> + UINTN CoreBits; >> + CPUID_VERSION_INFO_EBX VersionInfoEbx; >> + CPUID_VERSION_INFO_EDX VersionInfoEdx; >> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; >> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >> + UINT32 MaxCpuIdIndex; >> + UINT32 SubIndex; >> + UINTN LevelType; >> + UINT32 MaxLogicProcessorsPerPackage; >> + UINT32 MaxCoresPerPackage; >> + >> + // >> + // Check if the processor is capable of supporting more than one logical >> processor. >> + // >> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, >> + &VersionInfoEdx.Uint32); if (VersionInfoEdx.Bits.HTT == 0) { >> + if (Thread != NULL) { >> + *Thread = 0; >> + } >> + if (Core != NULL) { >> + *Core = 0; >> + } >> + if (Package != NULL) { >> + *Package = 0; >> + } >> + return; >> + } >> + >> + ThreadBits = 0; >> + CoreBits = 0; >> + >> + // >> + // Assume three-level mapping of APIC ID: Package:Core:SMT. >> + // >> + >> + TopologyLeafSupported = FALSE; >> + // >> + // Get the max index of basic CPUID >> + // >> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >> + >> + // >> + // If the extended topology enumeration leaf is available, it // is >> + the preferred mechanism for enumerating topology. >> + // >> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >> + AsmCpuidEx( >> + CPUID_EXTENDED_TOPOLOGY, >> + 0, >> + &ExtendedTopologyEax.Uint32, >> + &ExtendedTopologyEbx.Uint32, >> + &ExtendedTopologyEcx.Uint32, >> + NULL >> + ); >> + // >> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >> + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not >> + // supported on that processor. >> + // >> + if (ExtendedTopologyEbx.Uint32 != 0) { >> + TopologyLeafSupported = TRUE; >> + >> + // >> + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters >> to extract >> + // the SMT sub-field of x2APIC ID. >> + // >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >> + ASSERT(LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >> + >> + // >> + // Software must not assume any "level type" encoding >> + // value to be related to any sub-leaf index, except sub-leaf 0. >> + // >> + SubIndex = 1; >> + do { >> + AsmCpuidEx( >> + CPUID_EXTENDED_TOPOLOGY, >> + SubIndex, >> + &ExtendedTopologyEax.Uint32, >> + NULL, >> + &ExtendedTopologyEcx.Uint32, >> + NULL >> + ); >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >> + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >> + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >> + break; >> + } >> + SubIndex++; >> + } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >> + } >> + } >> + >> + if (!TopologyLeafSupported) { >> + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, >> NULL); >> + MaxLogicProcessorsPerPackage = >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, >> NULL, NULL, NULL); >> + MaxCoresPerPackage = >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >> + } >> + else { >> + // >> + // Must be a single-core processor. >> + // >> + MaxCoresPerPackage = 1; >> + } >> + >> + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / >> MaxCoresPerPackage - 1) + 1); >> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); } >> + >> + if (Thread != NULL) { >> + *Thread = InitialApicId & ((1 << ThreadBits) - 1); >> + } >> + if (Core != NULL) { >> + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >> + } >> + if (Package != NULL) { >> + *Package = (InitialApicId >> (ThreadBits + CoreBits)); >> + } >> +} >> diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >> b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >> index 4c42696..c620211 100644 >> --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >> +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >> @@ -1036,3 +1036,151 @@ GetApicMsiValue ( >> } >> return MsiData.Uint64; >> } >> + >> +/** >> + Get Package ID/Core ID/Thread ID of a processor. >> + >> + The algorithm assumes the target system has symmetry across physical >> + package boundaries with respect to the number of logical processors >> + per package, number of cores per package. >> + >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical >> + processor. >> + @param[out] Package Returns the processor package ID. >> + @param[out] Core Returns the processor core ID. >> + @param[out] Thread Returns the processor thread ID. >> +**/ >> +VOID >> +GetProcessorLocation( >> + IN UINT32 InitialApicId, >> + OUT UINT32 *Package OPTIONAL, >> + OUT UINT32 *Core OPTIONAL, >> + OUT UINT32 *Thread OPTIONAL >> + ) >> +{ >> + BOOLEAN TopologyLeafSupported; >> + UINTN ThreadBits; >> + UINTN CoreBits; >> + CPUID_VERSION_INFO_EBX VersionInfoEbx; >> + CPUID_VERSION_INFO_EDX VersionInfoEdx; >> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; >> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >> + UINT32 MaxCpuIdIndex; >> + UINT32 SubIndex; >> + UINTN LevelType; >> + UINT32 MaxLogicProcessorsPerPackage; >> + UINT32 MaxCoresPerPackage; >> + >> + // >> + // Check if the processor is capable of supporting more than one logical >> processor. >> + // >> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, >> + &VersionInfoEdx.Uint32); if (VersionInfoEdx.Bits.HTT == 0) { >> + if (Thread != NULL) { >> + *Thread = 0; >> + } >> + if (Core != NULL) { >> + *Core = 0; >> + } >> + if (Package != NULL) { >> + *Package = 0; >> + } >> + return; >> + } >> + >> + ThreadBits = 0; >> + CoreBits = 0; >> + >> + // >> + // Assume three-level mapping of APIC ID: Package:Core:SMT. >> + // >> + >> + TopologyLeafSupported = FALSE; >> + // >> + // Get the max index of basic CPUID >> + // >> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >> + >> + // >> + // If the extended topology enumeration leaf is available, it // is >> + the preferred mechanism for enumerating topology. >> + // >> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >> + AsmCpuidEx( >> + CPUID_EXTENDED_TOPOLOGY, >> + 0, >> + &ExtendedTopologyEax.Uint32, >> + &ExtendedTopologyEbx.Uint32, >> + &ExtendedTopologyEcx.Uint32, >> + NULL >> + ); >> + // >> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >> + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not >> + // supported on that processor. >> + // >> + if (ExtendedTopologyEbx.Uint32 != 0) { >> + TopologyLeafSupported = TRUE; >> + >> + // >> + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters >> to extract >> + // the SMT sub-field of x2APIC ID. >> + // >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >> + ASSERT(LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >> + >> + // >> + // Software must not assume any "level type" encoding >> + // value to be related to any sub-leaf index, except sub-leaf 0. >> + // >> + SubIndex = 1; >> + do { >> + AsmCpuidEx( >> + CPUID_EXTENDED_TOPOLOGY, >> + SubIndex, >> + &ExtendedTopologyEax.Uint32, >> + NULL, >> + &ExtendedTopologyEcx.Uint32, >> + NULL >> + ); >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >> + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >> + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >> + break; >> + } >> + SubIndex++; >> + } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >> + } >> + } >> + >> + if (!TopologyLeafSupported) { >> + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, >> NULL); >> + MaxLogicProcessorsPerPackage = >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, >> NULL, NULL, NULL); >> + MaxCoresPerPackage = >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >> + } >> + else { >> + // >> + // Must be a single-core processor. >> + // >> + MaxCoresPerPackage = 1; >> + } >> + >> + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / >> MaxCoresPerPackage - 1) + 1); >> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); } >> + >> + if (Thread != NULL) { >> + *Thread = InitialApicId & ((1 << ThreadBits) - 1); >> + } >> + if (Core != NULL) { >> + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >> + } >> + if (Package != NULL) { >> + *Package = (InitialApicId >> (ThreadBits + CoreBits)); >> + } >> +} >> diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c >> b/UefiCpuPkg/Library/MpInitLib/MpLib.c >> index c3fe721..f205b6b 100644 >> --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c >> +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c >> @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( } >> >> /** >> - Get CPU Package/Core/Thread location information. >> - >> - @param[in] InitialApicId CPU APIC ID >> - @param[out] Location Pointer to CPU location information >> -**/ >> -VOID >> -ExtractProcessorLocation ( >> - IN UINT32 InitialApicId, >> - OUT EFI_CPU_PHYSICAL_LOCATION *Location >> - ) >> -{ >> - BOOLEAN TopologyLeafSupported; >> - UINTN ThreadBits; >> - UINTN CoreBits; >> - CPUID_VERSION_INFO_EBX VersionInfoEbx; >> - CPUID_VERSION_INFO_EDX VersionInfoEdx; >> - CPUID_CACHE_PARAMS_EAX CacheParamsEax; >> - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >> - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >> - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >> - UINT32 MaxCpuIdIndex; >> - UINT32 SubIndex; >> - UINTN LevelType; >> - UINT32 MaxLogicProcessorsPerPackage; >> - UINT32 MaxCoresPerPackage; >> - >> - // >> - // Check if the processor is capable of supporting more than one logical >> processor. >> - // >> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, >> &VersionInfoEdx.Uint32); >> - if (VersionInfoEdx.Bits.HTT == 0) { >> - Location->Thread = 0; >> - Location->Core = 0; >> - Location->Package = 0; >> - return; >> - } >> - >> - ThreadBits = 0; >> - CoreBits = 0; >> - >> - // >> - // Assume three-level mapping of APIC ID: Package:Core:SMT. >> - // >> - >> - TopologyLeafSupported = FALSE; >> - // >> - // Get the max index of basic CPUID >> - // >> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >> - >> - // >> - // If the extended topology enumeration leaf is available, it >> - // is the preferred mechanism for enumerating topology. >> - // >> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >> - AsmCpuidEx ( >> - CPUID_EXTENDED_TOPOLOGY, >> - 0, >> - &ExtendedTopologyEax.Uint32, >> - &ExtendedTopologyEbx.Uint32, >> - &ExtendedTopologyEcx.Uint32, >> - NULL >> - ); >> - // >> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not >> - // supported on that processor. >> - // >> - if (ExtendedTopologyEbx.Uint32 != 0) { >> - TopologyLeafSupported = TRUE; >> - >> - // >> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to >> extract >> - // the SMT sub-field of x2APIC ID. >> - // >> - LevelType = ExtendedTopologyEcx.Bits.LevelType; >> - ASSERT (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >> - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >> - >> - // >> - // Software must not assume any "level type" encoding >> - // value to be related to any sub-leaf index, except sub-leaf 0. >> - // >> - SubIndex = 1; >> - do { >> - AsmCpuidEx ( >> - CPUID_EXTENDED_TOPOLOGY, >> - SubIndex, >> - &ExtendedTopologyEax.Uint32, >> - NULL, >> - &ExtendedTopologyEcx.Uint32, >> - NULL >> - ); >> - LevelType = ExtendedTopologyEcx.Bits.LevelType; >> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >> - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >> - break; >> - } >> - SubIndex++; >> - } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >> - } >> - } >> - >> - if (!TopologyLeafSupported) { >> - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, >> NULL); >> - MaxLogicProcessorsPerPackage = >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, >> NULL, NULL, NULL); >> - MaxCoresPerPackage = >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >> - } else { >> - // >> - // Must be a single-core processor. >> - // >> - MaxCoresPerPackage = 1; >> - } >> - >> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / >> MaxCoresPerPackage - 1) + 1); >> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); >> - } >> - >> - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); >> - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >> - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); -} >> - >> -/** >> Worker function for SwitchBSP(). >> >> Worker function for SwitchBSP(), assigned to the AP which is intended @@ >> -1451,7 +1325,12 @@ MpInitLibGetProcessorInfo ( >> // >> // Get processor location information >> // >> - ExtractProcessorLocation (CpuMpData- >>> CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); >> + GetProcessorLocation ( >> + CpuMpData->CpuData[ProcessorNumber].ApicId, >> + &ProcessorInfoBuffer->Location.Package, >> + &ProcessorInfoBuffer->Location.Core, >> + &ProcessorInfoBuffer->Location.Thread >> + ); >> >> if (HealthData != NULL) { >> HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; >> diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >> b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >> index 40f2a17..93ebb9e 100644 >> --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >> +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >> @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL >> mSmmCpuService = { }; >> >> /** >> - Get Package ID/Core ID/Thread ID of a processor. >> - >> - APIC ID must be an initial APIC ID. >> - >> - The algorithm below assumes the target system has symmetry across >> physical package boundaries >> - with respect to the number of logical processors per package, number of >> cores per package. >> - >> - @param ApicId APIC ID of the target logical processor. >> - @param Location Returns the processor location information. >> -**/ >> -VOID >> -SmmGetProcessorLocation ( >> - IN UINT32 ApicId, >> - OUT EFI_CPU_PHYSICAL_LOCATION *Location >> - ) >> -{ >> - UINTN ThreadBits; >> - UINTN CoreBits; >> - UINT32 RegEax; >> - UINT32 RegEbx; >> - UINT32 RegEcx; >> - UINT32 RegEdx; >> - UINT32 MaxCpuIdIndex; >> - UINT32 SubIndex; >> - UINTN LevelType; >> - UINT32 MaxLogicProcessorsPerPackage; >> - UINT32 MaxCoresPerPackage; >> - BOOLEAN TopologyLeafSupported; >> - >> - ASSERT (Location != NULL); >> - >> - ThreadBits = 0; >> - CoreBits = 0; >> - TopologyLeafSupported = FALSE; >> - >> - // >> - // Check if the processor is capable of supporting more than one logical >> processor. >> - // >> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); >> - ASSERT ((RegEdx & BIT28) != 0); >> - >> - // >> - // Assume three-level mapping of APIC ID: Package:Core:SMT. >> - // >> - >> - // >> - // Get the max index of basic CPUID >> - // >> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >> - >> - // >> - // If the extended topology enumeration leaf is available, it >> - // is the preferred mechanism for enumerating topology. >> - // >> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, >> &RegEcx, NULL); >> - // >> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not >> - // supported on that processor. >> - // >> - if ((RegEbx & 0xffff) != 0) { >> - TopologyLeafSupported = TRUE; >> - >> - // >> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to >> extract >> - // the SMT sub-field of x2APIC ID. >> - // >> - LevelType = (RegEcx >> 8) & 0xff; >> - ASSERT (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >> - if ((RegEbx & 0xffff) > 1 ) { >> - ThreadBits = RegEax & 0x1f; >> - } else { >> - // >> - // HT is not supported >> - // >> - ThreadBits = 0; >> - } >> - >> - // >> - // Software must not assume any "level type" encoding >> - // value to be related to any sub-leaf index, except sub-leaf 0. >> - // >> - SubIndex = 1; >> - do { >> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, NULL, >> &RegEcx, NULL); >> - LevelType = (RegEcx >> 8) & 0xff; >> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >> - CoreBits = (RegEax & 0x1f) - ThreadBits; >> - break; >> - } >> - SubIndex++; >> - } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >> - } >> - } >> - >> - if (!TopologyLeafSupported) { >> - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); >> - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; >> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); >> - MaxCoresPerPackage = (RegEax >> 26) + 1; >> - } else { >> - // >> - // Must be a single-core processor. >> - // >> - MaxCoresPerPackage = 1; >> - } >> - >> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / >> MaxCoresPerPackage - 1) + 1); >> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); >> - } >> - >> - Location->Thread = ApicId & ~((-1) << ThreadBits); >> - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); >> - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); -} >> - >> -/** >> Gets processor information on the requested processor at the instant this >> call is made. >> >> @param[in] This A pointer to the >> EFI_SMM_CPU_SERVICE_PROTOCOL instance. >> @@ -280,7 +161,12 @@ SmmAddProcessor ( >> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == >> INVALID_APIC_ID) { >> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; >> gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; >> - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- >>> ProcessorInfo[Index].Location); >> + GetProcessorLocation ( >> + (UINT32)ProcessorId, >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Package, >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Core, >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Thread >> + ); >> >> *ProcessorNumber = Index; >> gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; >> -- >> 2.6.3.windows.1 >> >> >> Best regards, >> >> Mike >> >>> -----Original Message----- >>> From: Leo Duran [mailto:leo.duran@amd.com] >>> Sent: Friday, October 28, 2016 10:39 AM >>> To: edk2-devel@lists.01.org >>> Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff >>> <jeff.fan@intel.com>; Kinney, Michael D <michael.d.kinney@intel.com>; >> Leo Duran >>> <leo.duran@amd.com> >>> Subject: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib >> library >>> >>> 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. >>> 2) Remove ExtractProcessorLocation() from MpInitLib library. >>> 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. >>> >>> Contributed-under: TianoCore Contribution Agreement 1.0 >>> Signed-off-by: Leo Duran <leo.duran@amd.com> >>> --- >>> UefiCpuPkg/Include/Library/LocalApicLib.h | 18 +++ >>> UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 130 >> +++++++++++++++++++++ >>> .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 130 >> +++++++++++++++++++++ >>> UefiCpuPkg/Library/MpInitLib/MpLib.c | 128 +------------------- >>> UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 121 +------------------ >>> 5 files changed, 280 insertions(+), 247 deletions(-) >>> >>> diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h >>> b/UefiCpuPkg/Include/Library/LocalApicLib.h >>> index cd4e613..4abf64c 100644 >>> --- a/UefiCpuPkg/Include/Library/LocalApicLib.h >>> +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h >>> @@ -21,6 +21,9 @@ >>> #define LOCAL_APIC_MODE_XAPIC 0x1 ///< xAPIC mode. >>> #define LOCAL_APIC_MODE_X2APIC 0x2 ///< x2APIC mode. >>> >>> +#include <PiPei.h> >>> +#include <Protocol/MpService.h> >>> + >>> /** >>> Retrieve the base address of local APIC. >>> >>> @@ -410,6 +413,21 @@ GetApicMsiValue ( >>> IN BOOLEAN LevelTriggered, >>> IN BOOLEAN AssertionLevel >>> ); >>> + >>> +/** >>> +Get Package ID/Core ID/Thread ID of a processor. >>> + >>> +The algorithm assumes the target system has symmetry across physical >> package >>> boundaries >>> +with respect to the number of logical processors per package, number of >> cores per >>> package. >>> + >>> +@param InitialApicId Must be the initial APIC ID of the target logical >> processor. >>> +@param Location Returns the processor location information. >>> +**/ >>> +VOID >>> +GetProcessorLocation( >>> + IN UINT32 InitialApicId, >>> + OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> +); >>> >>> #endif >>> >>> diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >>> b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >>> index 8d0fb02..219f99f 100644 >>> --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >>> +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >>> @@ -941,3 +941,133 @@ GetApicMsiValue ( >>> } >>> return MsiData.Uint64; >>> } >>> + >>> +/** >>> +Get Package ID/Core ID/Thread ID of a processor. >>> + >>> +The algorithm assumes the target system has symmetry across physical >> package >>> boundaries >>> +with respect to the number of logical processors per package, number of >> cores per >>> package. >>> + >>> +@param InitialApicId Must be the initial APIC ID of the target logical >> processor. >>> +@param Location Returns the processor location information. >>> +**/ >>> +VOID >>> +GetProcessorLocation( >>> +IN UINT32 InitialApicId, >>> +OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> +) >>> +{ >>> + BOOLEAN TopologyLeafSupported; >>> + UINTN ThreadBits; >>> + UINTN CoreBits; >>> + CPUID_VERSION_INFO_EBX VersionInfoEbx; >>> + CPUID_VERSION_INFO_EDX VersionInfoEdx; >>> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; >>> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >>> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >>> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >>> + UINT32 MaxCpuIdIndex; >>> + UINT32 SubIndex; >>> + UINTN LevelType; >>> + UINT32 MaxLogicProcessorsPerPackage; >>> + UINT32 MaxCoresPerPackage; >>> + >>> + // >>> + // Check if the processor is capable of supporting more than one >> logical >>> processor. >>> + // >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, >> &VersionInfoEdx.Uint32); >>> + if (VersionInfoEdx.Bits.HTT == 0) { >>> + Location->Thread = 0; >>> + Location->Core = 0; >>> + Location->Package = 0; >>> + return; >>> + } >>> + >>> + ThreadBits = 0; >>> + CoreBits = 0; >>> + >>> + // >>> + // Assume three-level mapping of APIC ID: Package:Core:SMT. >>> + // >>> + >>> + TopologyLeafSupported = FALSE; >>> + // >>> + // Get the max index of basic CPUID >>> + // >>> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, >> NULL); >>> + >>> + // >>> + // If the extended topology enumeration leaf is available, it >>> + // is the preferred mechanism for enumerating topology. >>> + // >>> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >>> + AsmCpuidEx( >>> + CPUID_EXTENDED_TOPOLOGY, >>> + 0, >>> + &ExtendedTopologyEax.Uint32, >>> + &ExtendedTopologyEbx.Uint32, >>> + &ExtendedTopologyEcx.Uint32, >>> + NULL >>> + ); >>> + // >>> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and >> maximum input value for >>> + // basic CPUID information is greater than 0BH, then >> CPUID.0BH leaf is not >>> + // supported on that processor. >>> + // >>> + if (ExtendedTopologyEbx.Uint32 != 0) { >>> + TopologyLeafSupported = TRUE; >>> + >>> + // >>> + // Sub-leaf index 0 (ECX= 0 as input) provides >> enumeration parameters to >>> extract >>> + // the SMT sub-field of x2APIC ID. >>> + // >>> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >>> + ASSERT(LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >>> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >>> + >>> + // >>> + // Software must not assume any "level type" >> encoding >>> + // value to be related to any sub-leaf index, except >> sub-leaf 0. >>> + // >>> + SubIndex = 1; >>> + do { >>> + AsmCpuidEx( >>> + CPUID_EXTENDED_TOPOLOGY, >>> + SubIndex, >>> + &ExtendedTopologyEax.Uint32, >>> + NULL, >>> + &ExtendedTopologyEcx.Uint32, >>> + NULL >>> + ); >>> + LevelType = >> ExtendedTopologyEcx.Bits.LevelType; >>> + if (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >>> + CoreBits = >> ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >>> + break; >>> + } >>> + SubIndex++; >>> + } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >>> + } >>> + } >>> + >>> + if (!TopologyLeafSupported) { >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, >> &VersionInfoEbx.Uint32, NULL, NULL); >>> + MaxLogicProcessorsPerPackage = >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >>> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >>> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, >> &CacheParamsEax.Uint32, NULL, NULL, >>> NULL); >>> + MaxCoresPerPackage = >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >>> + } >>> + else { >>> + // >>> + // Must be a single-core processor. >>> + // >>> + MaxCoresPerPackage = 1; >>> + } >>> + >>> + ThreadBits = >> (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / >>> MaxCoresPerPackage - 1) + 1); >>> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + >> 1); >>> + } >>> + >>> + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); >>> + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >>> + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); >>> +} >>> diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >>> b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >>> index 4c42696..16395a3 100644 >>> --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >>> +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >>> @@ -1036,3 +1036,133 @@ GetApicMsiValue ( >>> } >>> return MsiData.Uint64; >>> } >>> + >>> +/** >>> +Get Package ID/Core ID/Thread ID of a processor. >>> + >>> +The algorithm assumes the target system has symmetry across physical >> package >>> boundaries >>> +with respect to the number of logical processors per package, number of >> cores per >>> package. >>> + >>> +@param InitialApicId Must be the initial APIC ID of the target logical >> processor. >>> +@param Location Returns the processor location information. >>> +**/ >>> +VOID >>> +GetProcessorLocation( >>> +IN UINT32 InitialApicId, >>> +OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> +) >>> +{ >>> + BOOLEAN TopologyLeafSupported; >>> + UINTN ThreadBits; >>> + UINTN CoreBits; >>> + CPUID_VERSION_INFO_EBX VersionInfoEbx; >>> + CPUID_VERSION_INFO_EDX VersionInfoEdx; >>> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; >>> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >>> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >>> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >>> + UINT32 MaxCpuIdIndex; >>> + UINT32 SubIndex; >>> + UINTN LevelType; >>> + UINT32 MaxLogicProcessorsPerPackage; >>> + UINT32 MaxCoresPerPackage; >>> + >>> + // >>> + // Check if the processor is capable of supporting more than one >> logical >>> processor. >>> + // >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, >> &VersionInfoEdx.Uint32); >>> + if (VersionInfoEdx.Bits.HTT == 0) { >>> + Location->Thread = 0; >>> + Location->Core = 0; >>> + Location->Package = 0; >>> + return; >>> + } >>> + >>> + ThreadBits = 0; >>> + CoreBits = 0; >>> + >>> + // >>> + // Assume three-level mapping of APIC ID: Package:Core:SMT. >>> + // >>> + >>> + TopologyLeafSupported = FALSE; >>> + // >>> + // Get the max index of basic CPUID >>> + // >>> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, >> NULL); >>> + >>> + // >>> + // If the extended topology enumeration leaf is available, it >>> + // is the preferred mechanism for enumerating topology. >>> + // >>> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >>> + AsmCpuidEx( >>> + CPUID_EXTENDED_TOPOLOGY, >>> + 0, >>> + &ExtendedTopologyEax.Uint32, >>> + &ExtendedTopologyEbx.Uint32, >>> + &ExtendedTopologyEcx.Uint32, >>> + NULL >>> + ); >>> + // >>> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and >> maximum input value for >>> + // basic CPUID information is greater than 0BH, then >> CPUID.0BH leaf is not >>> + // supported on that processor. >>> + // >>> + if (ExtendedTopologyEbx.Uint32 != 0) { >>> + TopologyLeafSupported = TRUE; >>> + >>> + // >>> + // Sub-leaf index 0 (ECX= 0 as input) provides >> enumeration parameters to >>> extract >>> + // the SMT sub-field of x2APIC ID. >>> + // >>> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >>> + ASSERT(LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >>> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >>> + >>> + // >>> + // Software must not assume any "level type" >> encoding >>> + // value to be related to any sub-leaf index, except >> sub-leaf 0. >>> + // >>> + SubIndex = 1; >>> + do { >>> + AsmCpuidEx( >>> + CPUID_EXTENDED_TOPOLOGY, >>> + SubIndex, >>> + &ExtendedTopologyEax.Uint32, >>> + NULL, >>> + &ExtendedTopologyEcx.Uint32, >>> + NULL >>> + ); >>> + LevelType = >> ExtendedTopologyEcx.Bits.LevelType; >>> + if (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >>> + CoreBits = >> ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >>> + break; >>> + } >>> + SubIndex++; >>> + } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >>> + } >>> + } >>> + >>> + if (!TopologyLeafSupported) { >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, >> &VersionInfoEbx.Uint32, NULL, NULL); >>> + MaxLogicProcessorsPerPackage = >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >>> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >>> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, >> &CacheParamsEax.Uint32, NULL, NULL, >>> NULL); >>> + MaxCoresPerPackage = >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >>> + } >>> + else { >>> + // >>> + // Must be a single-core processor. >>> + // >>> + MaxCoresPerPackage = 1; >>> + } >>> + >>> + ThreadBits = >> (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / >>> MaxCoresPerPackage - 1) + 1); >>> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + >> 1); >>> + } >>> + >>> + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); >>> + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >>> + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); >>> +} >>> diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c >>> b/UefiCpuPkg/Library/MpInitLib/MpLib.c >>> index c3fe721..f3380bb 100644 >>> --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c >>> +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c >>> @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( >>> } >>> >>> /** >>> - Get CPU Package/Core/Thread location information. >>> - >>> - @param[in] InitialApicId CPU APIC ID >>> - @param[out] Location Pointer to CPU location information >>> -**/ >>> -VOID >>> -ExtractProcessorLocation ( >>> - IN UINT32 InitialApicId, >>> - OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> - ) >>> -{ >>> - BOOLEAN TopologyLeafSupported; >>> - UINTN ThreadBits; >>> - UINTN CoreBits; >>> - CPUID_VERSION_INFO_EBX VersionInfoEbx; >>> - CPUID_VERSION_INFO_EDX VersionInfoEdx; >>> - CPUID_CACHE_PARAMS_EAX CacheParamsEax; >>> - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >>> - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >>> - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >>> - UINT32 MaxCpuIdIndex; >>> - UINT32 SubIndex; >>> - UINTN LevelType; >>> - UINT32 MaxLogicProcessorsPerPackage; >>> - UINT32 MaxCoresPerPackage; >>> - >>> - // >>> - // Check if the processor is capable of supporting more than one logical >> processor. >>> - // >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, >> &VersionInfoEdx.Uint32); >>> - if (VersionInfoEdx.Bits.HTT == 0) { >>> - Location->Thread = 0; >>> - Location->Core = 0; >>> - Location->Package = 0; >>> - return; >>> - } >>> - >>> - ThreadBits = 0; >>> - CoreBits = 0; >>> - >>> - // >>> - // Assume three-level mapping of APIC ID: Package:Core:SMT. >>> - // >>> - >>> - TopologyLeafSupported = FALSE; >>> - // >>> - // Get the max index of basic CPUID >>> - // >>> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >>> - >>> - // >>> - // If the extended topology enumeration leaf is available, it >>> - // is the preferred mechanism for enumerating topology. >>> - // >>> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >>> - AsmCpuidEx ( >>> - CPUID_EXTENDED_TOPOLOGY, >>> - 0, >>> - &ExtendedTopologyEax.Uint32, >>> - &ExtendedTopologyEbx.Uint32, >>> - &ExtendedTopologyEcx.Uint32, >>> - NULL >>> - ); >>> - // >>> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >>> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is >> not >>> - // supported on that processor. >>> - // >>> - if (ExtendedTopologyEbx.Uint32 != 0) { >>> - TopologyLeafSupported = TRUE; >>> - >>> - // >>> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters >> to extract >>> - // the SMT sub-field of x2APIC ID. >>> - // >>> - LevelType = ExtendedTopologyEcx.Bits.LevelType; >>> - ASSERT (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >>> - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >>> - >>> - // >>> - // Software must not assume any "level type" encoding >>> - // value to be related to any sub-leaf index, except sub-leaf 0. >>> - // >>> - SubIndex = 1; >>> - do { >>> - AsmCpuidEx ( >>> - CPUID_EXTENDED_TOPOLOGY, >>> - SubIndex, >>> - &ExtendedTopologyEax.Uint32, >>> - NULL, >>> - &ExtendedTopologyEcx.Uint32, >>> - NULL >>> - ); >>> - LevelType = ExtendedTopologyEcx.Bits.LevelType; >>> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >>> - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >>> - break; >>> - } >>> - SubIndex++; >>> - } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >>> - } >>> - } >>> - >>> - if (!TopologyLeafSupported) { >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, >> NULL, NULL); >>> - MaxLogicProcessorsPerPackage = >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >>> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >>> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, >> NULL, NULL, NULL); >>> - MaxCoresPerPackage = >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >>> - } else { >>> - // >>> - // Must be a single-core processor. >>> - // >>> - MaxCoresPerPackage = 1; >>> - } >>> - >>> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / >>> MaxCoresPerPackage - 1) + 1); >>> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); >>> - } >>> - >>> - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); >>> - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >>> - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); >>> -} >>> - >>> -/** >>> Worker function for SwitchBSP(). >>> >>> Worker function for SwitchBSP(), assigned to the AP which is intended >>> @@ -1451,7 +1325,7 @@ MpInitLibGetProcessorInfo ( >>> // >>> // Get processor location information >>> // >>> - ExtractProcessorLocation (CpuMpData- >>> CpuData[ProcessorNumber].ApicId, >>> &ProcessorInfoBuffer->Location); >>> + GetProcessorLocation (CpuMpData- >>> CpuData[ProcessorNumber].ApicId, >>> &ProcessorInfoBuffer->Location); >>> >>> if (HealthData != NULL) { >>> HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; >>> diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >>> b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >>> index 40f2a17..67cd0a0 100644 >>> --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >>> +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >>> @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL >> mSmmCpuService = { >>> }; >>> >>> /** >>> - Get Package ID/Core ID/Thread ID of a processor. >>> - >>> - APIC ID must be an initial APIC ID. >>> - >>> - The algorithm below assumes the target system has symmetry across >> physical package >>> boundaries >>> - with respect to the number of logical processors per package, number of >> cores per >>> package. >>> - >>> - @param ApicId APIC ID of the target logical processor. >>> - @param Location Returns the processor location information. >>> -**/ >>> -VOID >>> -SmmGetProcessorLocation ( >>> - IN UINT32 ApicId, >>> - OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> - ) >>> -{ >>> - UINTN ThreadBits; >>> - UINTN CoreBits; >>> - UINT32 RegEax; >>> - UINT32 RegEbx; >>> - UINT32 RegEcx; >>> - UINT32 RegEdx; >>> - UINT32 MaxCpuIdIndex; >>> - UINT32 SubIndex; >>> - UINTN LevelType; >>> - UINT32 MaxLogicProcessorsPerPackage; >>> - UINT32 MaxCoresPerPackage; >>> - BOOLEAN TopologyLeafSupported; >>> - >>> - ASSERT (Location != NULL); >>> - >>> - ThreadBits = 0; >>> - CoreBits = 0; >>> - TopologyLeafSupported = FALSE; >>> - >>> - // >>> - // Check if the processor is capable of supporting more than one logical >> processor. >>> - // >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); >>> - ASSERT ((RegEdx & BIT28) != 0); >>> - >>> - // >>> - // Assume three-level mapping of APIC ID: Package:Core:SMT. >>> - // >>> - >>> - // >>> - // Get the max index of basic CPUID >>> - // >>> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >>> - >>> - // >>> - // If the extended topology enumeration leaf is available, it >>> - // is the preferred mechanism for enumerating topology. >>> - // >>> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >>> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, >> &RegEcx, NULL); >>> - // >>> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >>> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is >> not >>> - // supported on that processor. >>> - // >>> - if ((RegEbx & 0xffff) != 0) { >>> - TopologyLeafSupported = TRUE; >>> - >>> - // >>> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters >> to extract >>> - // the SMT sub-field of x2APIC ID. >>> - // >>> - LevelType = (RegEcx >> 8) & 0xff; >>> - ASSERT (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >>> - if ((RegEbx & 0xffff) > 1 ) { >>> - ThreadBits = RegEax & 0x1f; >>> - } else { >>> - // >>> - // HT is not supported >>> - // >>> - ThreadBits = 0; >>> - } >>> - >>> - // >>> - // Software must not assume any "level type" encoding >>> - // value to be related to any sub-leaf index, except sub-leaf 0. >>> - // >>> - SubIndex = 1; >>> - do { >>> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, >> NULL, &RegEcx, NULL); >>> - LevelType = (RegEcx >> 8) & 0xff; >>> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >>> - CoreBits = (RegEax & 0x1f) - ThreadBits; >>> - break; >>> - } >>> - SubIndex++; >>> - } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >>> - } >>> - } >>> - >>> - if (!TopologyLeafSupported) { >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); >>> - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; >>> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >>> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); >>> - MaxCoresPerPackage = (RegEax >> 26) + 1; >>> - } else { >>> - // >>> - // Must be a single-core processor. >>> - // >>> - MaxCoresPerPackage = 1; >>> - } >>> - >>> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / >>> MaxCoresPerPackage - 1) + 1); >>> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); >>> - } >>> - >>> - Location->Thread = ApicId & ~((-1) << ThreadBits); >>> - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); >>> - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); >>> -} >>> - >>> -/** >>> Gets processor information on the requested processor at the instant this >> call is >>> made. >>> >>> @param[in] This A pointer to the >> EFI_SMM_CPU_SERVICE_PROTOCOL >>> instance. >>> @@ -280,7 +161,7 @@ SmmAddProcessor ( >>> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == >> INVALID_APIC_ID) { >>> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; >>> gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; >>> - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- >>>> ProcessorInfo[Index].Location); >>> + GetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- >>>> ProcessorInfo[Index].Location); >>> >>> *ProcessorNumber = Index; >>> gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; >>> -- >>> 1.9.1 > ^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library 2016-10-28 21:48 ` Laszlo Ersek @ 2016-10-31 6:40 ` Fan, Jeff 2016-10-31 16:45 ` Duran, Leo 0 siblings, 1 reply; 7+ messages in thread From: Fan, Jeff @ 2016-10-31 6:40 UTC (permalink / raw) To: Laszlo Ersek, Duran, Leo, Kinney, Michael D, edk2-devel@lists.01.org Cc: Gao, Liming Leo, Reviewed-by: Jeff Fan <jeff.fan@intel.com> based on Mike's updating. Could you refine the final patch and send out for testing(requested by Laszlo)? And then I could check-in it also. Thanks! Jeff -----Original Message----- From: edk2-devel [mailto:edk2-devel-bounces@lists.01.org] On Behalf Of Laszlo Ersek Sent: Saturday, October 29, 2016 5:48 AM To: Duran, Leo; Kinney, Michael D; edk2-devel@lists.01.org Cc: Fan, Jeff; Gao, Liming Subject: Re: [edk2] [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library On 10/28/16 23:17, Duran, Leo wrote: > Replies below. > Thanks. > >> -----Original Message----- >> From: Kinney, Michael D [mailto:michael.d.kinney@intel.com] >> Sent: Friday, October 28, 2016 4:12 PM >> To: Duran, Leo <leo.duran@amd.com>; edk2-devel@lists.01.org >> Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff >> <jeff.fan@intel.com> >> Subject: RE: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib >> library >> >> Leo, >> >> I tested your patch on Quark and it builds and boots. >> >> Here is a slightly updated version of the patch that makes the following >> changes: >> >> * Follow Jeff Fan suggestion to pass in Package, Core, Thread as 3 >> parameters. >> This removes the dependency on PiPei.h and MpService.h from >> LocalApicLib.h >> so LocalApicLib can be used from more than just PEI/DXE phases. I also >> made >> these three OUT parameters OPTIONAL. >> * Change tabs to spaces and fix other minor format issues to pass checks >> from >> BaseTools/Scripts/PatchCheck.py >> >> I have also tested this updated patch on Quark and it also builds and boots. >> >> > [Duran, Leo] > Sounds a good thing! > (I assume Lazlo will probably test this as well) If Mike thinks the changes warrant it, I certainly can (could be next week), but for that, I'd need a regularly formatted & mailed patch, one that I can apply with "git am" from the list. Thanks Laszlo > Thanks Mike. > Leo. > >> >> From cc932ca811e052a0a3ae103a9db9614b3807d4ed Mon Sep 17 00:00:00 >> 2001 >> From: Leo Duran <leo.duran@amd.com> >> Date: Fri, 28 Oct 2016 10:38:45 -0700 >> Subject: [Patch] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib >> library >> >> 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. >> 2) Remove ExtractProcessorLocation() from MpInitLib library. >> 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. >> >> Cc: Jeff Fan <jeff.fan@intel.com> >> Cc: Laszlo Ersek <lersek@redhat.com> >> Cc: Michael D Kinney <michael.d.kinney@intel.com> >> Contributed-under: TianoCore Contribution Agreement 1.0 >> Signed-off-by: Leo Duran <leo.duran@amd.com> >> --- >> UefiCpuPkg/Include/Library/LocalApicLib.h | 21 +++ >> UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 148 >> +++++++++++++++++++++ >> .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 148 >> +++++++++++++++++++++ >> UefiCpuPkg/Library/MpInitLib/MpLib.c | 133 +----------------- >> UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 126 +----------------- >> 5 files changed, 329 insertions(+), 247 deletions(-) >> >> diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h >> b/UefiCpuPkg/Include/Library/LocalApicLib.h >> index cd4e613..d73198f 100644 >> --- a/UefiCpuPkg/Include/Library/LocalApicLib.h >> +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h >> @@ -410,6 +410,27 @@ GetApicMsiValue ( >> IN BOOLEAN LevelTriggered, >> IN BOOLEAN AssertionLevel >> ); >> + >> +/** >> + Get Package ID/Core ID/Thread ID of a processor. >> + >> + The algorithm assumes the target system has symmetry across physical >> + package boundaries with respect to the number of logical processors >> + per package, number of cores per package. >> + >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical >> + processor. >> + @param[out] Package Returns the processor package ID. >> + @param[out] Core Returns the processor core ID. >> + @param[out] Thread Returns the processor thread ID. >> +**/ >> +VOID >> +GetProcessorLocation( >> + IN UINT32 InitialApicId, >> + OUT UINT32 *Package OPTIONAL, >> + OUT UINT32 *Core OPTIONAL, >> + OUT UINT32 *Thread OPTIONAL >> + ); >> >> #endif >> >> diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >> b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >> index 8d0fb02..7ddee1b 100644 >> --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >> +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >> @@ -941,3 +941,151 @@ GetApicMsiValue ( >> } >> return MsiData.Uint64; >> } >> + >> +/** >> + Get Package ID/Core ID/Thread ID of a processor. >> + >> + The algorithm assumes the target system has symmetry across physical >> + package boundaries with respect to the number of logical processors >> + per package, number of cores per package. >> + >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical >> + processor. >> + @param[out] Package Returns the processor package ID. >> + @param[out] Core Returns the processor core ID. >> + @param[out] Thread Returns the processor thread ID. >> +**/ >> +VOID >> +GetProcessorLocation( >> + IN UINT32 InitialApicId, >> + OUT UINT32 *Package OPTIONAL, >> + OUT UINT32 *Core OPTIONAL, >> + OUT UINT32 *Thread OPTIONAL >> + ) >> +{ >> + BOOLEAN TopologyLeafSupported; >> + UINTN ThreadBits; >> + UINTN CoreBits; >> + CPUID_VERSION_INFO_EBX VersionInfoEbx; >> + CPUID_VERSION_INFO_EDX VersionInfoEdx; >> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; >> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >> + UINT32 MaxCpuIdIndex; >> + UINT32 SubIndex; >> + UINTN LevelType; >> + UINT32 MaxLogicProcessorsPerPackage; >> + UINT32 MaxCoresPerPackage; >> + >> + // >> + // Check if the processor is capable of supporting more than one logical >> processor. >> + // >> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, >> + &VersionInfoEdx.Uint32); if (VersionInfoEdx.Bits.HTT == 0) { >> + if (Thread != NULL) { >> + *Thread = 0; >> + } >> + if (Core != NULL) { >> + *Core = 0; >> + } >> + if (Package != NULL) { >> + *Package = 0; >> + } >> + return; >> + } >> + >> + ThreadBits = 0; >> + CoreBits = 0; >> + >> + // >> + // Assume three-level mapping of APIC ID: Package:Core:SMT. >> + // >> + >> + TopologyLeafSupported = FALSE; >> + // >> + // Get the max index of basic CPUID >> + // >> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >> + >> + // >> + // If the extended topology enumeration leaf is available, it // is >> + the preferred mechanism for enumerating topology. >> + // >> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >> + AsmCpuidEx( >> + CPUID_EXTENDED_TOPOLOGY, >> + 0, >> + &ExtendedTopologyEax.Uint32, >> + &ExtendedTopologyEbx.Uint32, >> + &ExtendedTopologyEcx.Uint32, >> + NULL >> + ); >> + // >> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >> + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not >> + // supported on that processor. >> + // >> + if (ExtendedTopologyEbx.Uint32 != 0) { >> + TopologyLeafSupported = TRUE; >> + >> + // >> + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters >> to extract >> + // the SMT sub-field of x2APIC ID. >> + // >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >> + ASSERT(LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >> + >> + // >> + // Software must not assume any "level type" encoding >> + // value to be related to any sub-leaf index, except sub-leaf 0. >> + // >> + SubIndex = 1; >> + do { >> + AsmCpuidEx( >> + CPUID_EXTENDED_TOPOLOGY, >> + SubIndex, >> + &ExtendedTopologyEax.Uint32, >> + NULL, >> + &ExtendedTopologyEcx.Uint32, >> + NULL >> + ); >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >> + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >> + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >> + break; >> + } >> + SubIndex++; >> + } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >> + } >> + } >> + >> + if (!TopologyLeafSupported) { >> + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, >> NULL); >> + MaxLogicProcessorsPerPackage = >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, >> NULL, NULL, NULL); >> + MaxCoresPerPackage = >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >> + } >> + else { >> + // >> + // Must be a single-core processor. >> + // >> + MaxCoresPerPackage = 1; >> + } >> + >> + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / >> MaxCoresPerPackage - 1) + 1); >> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); } >> + >> + if (Thread != NULL) { >> + *Thread = InitialApicId & ((1 << ThreadBits) - 1); >> + } >> + if (Core != NULL) { >> + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >> + } >> + if (Package != NULL) { >> + *Package = (InitialApicId >> (ThreadBits + CoreBits)); >> + } >> +} >> diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >> b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >> index 4c42696..c620211 100644 >> --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >> +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >> @@ -1036,3 +1036,151 @@ GetApicMsiValue ( >> } >> return MsiData.Uint64; >> } >> + >> +/** >> + Get Package ID/Core ID/Thread ID of a processor. >> + >> + The algorithm assumes the target system has symmetry across physical >> + package boundaries with respect to the number of logical processors >> + per package, number of cores per package. >> + >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical >> + processor. >> + @param[out] Package Returns the processor package ID. >> + @param[out] Core Returns the processor core ID. >> + @param[out] Thread Returns the processor thread ID. >> +**/ >> +VOID >> +GetProcessorLocation( >> + IN UINT32 InitialApicId, >> + OUT UINT32 *Package OPTIONAL, >> + OUT UINT32 *Core OPTIONAL, >> + OUT UINT32 *Thread OPTIONAL >> + ) >> +{ >> + BOOLEAN TopologyLeafSupported; >> + UINTN ThreadBits; >> + UINTN CoreBits; >> + CPUID_VERSION_INFO_EBX VersionInfoEbx; >> + CPUID_VERSION_INFO_EDX VersionInfoEdx; >> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; >> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >> + UINT32 MaxCpuIdIndex; >> + UINT32 SubIndex; >> + UINTN LevelType; >> + UINT32 MaxLogicProcessorsPerPackage; >> + UINT32 MaxCoresPerPackage; >> + >> + // >> + // Check if the processor is capable of supporting more than one logical >> processor. >> + // >> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, >> + &VersionInfoEdx.Uint32); if (VersionInfoEdx.Bits.HTT == 0) { >> + if (Thread != NULL) { >> + *Thread = 0; >> + } >> + if (Core != NULL) { >> + *Core = 0; >> + } >> + if (Package != NULL) { >> + *Package = 0; >> + } >> + return; >> + } >> + >> + ThreadBits = 0; >> + CoreBits = 0; >> + >> + // >> + // Assume three-level mapping of APIC ID: Package:Core:SMT. >> + // >> + >> + TopologyLeafSupported = FALSE; >> + // >> + // Get the max index of basic CPUID >> + // >> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >> + >> + // >> + // If the extended topology enumeration leaf is available, it // is >> + the preferred mechanism for enumerating topology. >> + // >> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >> + AsmCpuidEx( >> + CPUID_EXTENDED_TOPOLOGY, >> + 0, >> + &ExtendedTopologyEax.Uint32, >> + &ExtendedTopologyEbx.Uint32, >> + &ExtendedTopologyEcx.Uint32, >> + NULL >> + ); >> + // >> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >> + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not >> + // supported on that processor. >> + // >> + if (ExtendedTopologyEbx.Uint32 != 0) { >> + TopologyLeafSupported = TRUE; >> + >> + // >> + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters >> to extract >> + // the SMT sub-field of x2APIC ID. >> + // >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >> + ASSERT(LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >> + >> + // >> + // Software must not assume any "level type" encoding >> + // value to be related to any sub-leaf index, except sub-leaf 0. >> + // >> + SubIndex = 1; >> + do { >> + AsmCpuidEx( >> + CPUID_EXTENDED_TOPOLOGY, >> + SubIndex, >> + &ExtendedTopologyEax.Uint32, >> + NULL, >> + &ExtendedTopologyEcx.Uint32, >> + NULL >> + ); >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >> + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >> + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >> + break; >> + } >> + SubIndex++; >> + } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >> + } >> + } >> + >> + if (!TopologyLeafSupported) { >> + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, >> NULL); >> + MaxLogicProcessorsPerPackage = >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, >> NULL, NULL, NULL); >> + MaxCoresPerPackage = >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >> + } >> + else { >> + // >> + // Must be a single-core processor. >> + // >> + MaxCoresPerPackage = 1; >> + } >> + >> + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / >> MaxCoresPerPackage - 1) + 1); >> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); } >> + >> + if (Thread != NULL) { >> + *Thread = InitialApicId & ((1 << ThreadBits) - 1); >> + } >> + if (Core != NULL) { >> + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >> + } >> + if (Package != NULL) { >> + *Package = (InitialApicId >> (ThreadBits + CoreBits)); >> + } >> +} >> diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c >> b/UefiCpuPkg/Library/MpInitLib/MpLib.c >> index c3fe721..f205b6b 100644 >> --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c >> +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c >> @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( } >> >> /** >> - Get CPU Package/Core/Thread location information. >> - >> - @param[in] InitialApicId CPU APIC ID >> - @param[out] Location Pointer to CPU location information >> -**/ >> -VOID >> -ExtractProcessorLocation ( >> - IN UINT32 InitialApicId, >> - OUT EFI_CPU_PHYSICAL_LOCATION *Location >> - ) >> -{ >> - BOOLEAN TopologyLeafSupported; >> - UINTN ThreadBits; >> - UINTN CoreBits; >> - CPUID_VERSION_INFO_EBX VersionInfoEbx; >> - CPUID_VERSION_INFO_EDX VersionInfoEdx; >> - CPUID_CACHE_PARAMS_EAX CacheParamsEax; >> - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >> - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >> - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >> - UINT32 MaxCpuIdIndex; >> - UINT32 SubIndex; >> - UINTN LevelType; >> - UINT32 MaxLogicProcessorsPerPackage; >> - UINT32 MaxCoresPerPackage; >> - >> - // >> - // Check if the processor is capable of supporting more than one logical >> processor. >> - // >> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, >> &VersionInfoEdx.Uint32); >> - if (VersionInfoEdx.Bits.HTT == 0) { >> - Location->Thread = 0; >> - Location->Core = 0; >> - Location->Package = 0; >> - return; >> - } >> - >> - ThreadBits = 0; >> - CoreBits = 0; >> - >> - // >> - // Assume three-level mapping of APIC ID: Package:Core:SMT. >> - // >> - >> - TopologyLeafSupported = FALSE; >> - // >> - // Get the max index of basic CPUID >> - // >> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >> - >> - // >> - // If the extended topology enumeration leaf is available, it >> - // is the preferred mechanism for enumerating topology. >> - // >> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >> - AsmCpuidEx ( >> - CPUID_EXTENDED_TOPOLOGY, >> - 0, >> - &ExtendedTopologyEax.Uint32, >> - &ExtendedTopologyEbx.Uint32, >> - &ExtendedTopologyEcx.Uint32, >> - NULL >> - ); >> - // >> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not >> - // supported on that processor. >> - // >> - if (ExtendedTopologyEbx.Uint32 != 0) { >> - TopologyLeafSupported = TRUE; >> - >> - // >> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to >> extract >> - // the SMT sub-field of x2APIC ID. >> - // >> - LevelType = ExtendedTopologyEcx.Bits.LevelType; >> - ASSERT (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >> - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >> - >> - // >> - // Software must not assume any "level type" encoding >> - // value to be related to any sub-leaf index, except sub-leaf 0. >> - // >> - SubIndex = 1; >> - do { >> - AsmCpuidEx ( >> - CPUID_EXTENDED_TOPOLOGY, >> - SubIndex, >> - &ExtendedTopologyEax.Uint32, >> - NULL, >> - &ExtendedTopologyEcx.Uint32, >> - NULL >> - ); >> - LevelType = ExtendedTopologyEcx.Bits.LevelType; >> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >> - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >> - break; >> - } >> - SubIndex++; >> - } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >> - } >> - } >> - >> - if (!TopologyLeafSupported) { >> - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, NULL, >> NULL); >> - MaxLogicProcessorsPerPackage = >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, >> NULL, NULL, NULL); >> - MaxCoresPerPackage = >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >> - } else { >> - // >> - // Must be a single-core processor. >> - // >> - MaxCoresPerPackage = 1; >> - } >> - >> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / >> MaxCoresPerPackage - 1) + 1); >> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); >> - } >> - >> - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); >> - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >> - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); -} >> - >> -/** >> Worker function for SwitchBSP(). >> >> Worker function for SwitchBSP(), assigned to the AP which is intended @@ >> -1451,7 +1325,12 @@ MpInitLibGetProcessorInfo ( >> // >> // Get processor location information >> // >> - ExtractProcessorLocation (CpuMpData- >>> CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); >> + GetProcessorLocation ( >> + CpuMpData->CpuData[ProcessorNumber].ApicId, >> + &ProcessorInfoBuffer->Location.Package, >> + &ProcessorInfoBuffer->Location.Core, >> + &ProcessorInfoBuffer->Location.Thread >> + ); >> >> if (HealthData != NULL) { >> HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; >> diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >> b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >> index 40f2a17..93ebb9e 100644 >> --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >> +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >> @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL >> mSmmCpuService = { }; >> >> /** >> - Get Package ID/Core ID/Thread ID of a processor. >> - >> - APIC ID must be an initial APIC ID. >> - >> - The algorithm below assumes the target system has symmetry across >> physical package boundaries >> - with respect to the number of logical processors per package, number of >> cores per package. >> - >> - @param ApicId APIC ID of the target logical processor. >> - @param Location Returns the processor location information. >> -**/ >> -VOID >> -SmmGetProcessorLocation ( >> - IN UINT32 ApicId, >> - OUT EFI_CPU_PHYSICAL_LOCATION *Location >> - ) >> -{ >> - UINTN ThreadBits; >> - UINTN CoreBits; >> - UINT32 RegEax; >> - UINT32 RegEbx; >> - UINT32 RegEcx; >> - UINT32 RegEdx; >> - UINT32 MaxCpuIdIndex; >> - UINT32 SubIndex; >> - UINTN LevelType; >> - UINT32 MaxLogicProcessorsPerPackage; >> - UINT32 MaxCoresPerPackage; >> - BOOLEAN TopologyLeafSupported; >> - >> - ASSERT (Location != NULL); >> - >> - ThreadBits = 0; >> - CoreBits = 0; >> - TopologyLeafSupported = FALSE; >> - >> - // >> - // Check if the processor is capable of supporting more than one logical >> processor. >> - // >> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); >> - ASSERT ((RegEdx & BIT28) != 0); >> - >> - // >> - // Assume three-level mapping of APIC ID: Package:Core:SMT. >> - // >> - >> - // >> - // Get the max index of basic CPUID >> - // >> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >> - >> - // >> - // If the extended topology enumeration leaf is available, it >> - // is the preferred mechanism for enumerating topology. >> - // >> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, >> &RegEcx, NULL); >> - // >> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is not >> - // supported on that processor. >> - // >> - if ((RegEbx & 0xffff) != 0) { >> - TopologyLeafSupported = TRUE; >> - >> - // >> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters to >> extract >> - // the SMT sub-field of x2APIC ID. >> - // >> - LevelType = (RegEcx >> 8) & 0xff; >> - ASSERT (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >> - if ((RegEbx & 0xffff) > 1 ) { >> - ThreadBits = RegEax & 0x1f; >> - } else { >> - // >> - // HT is not supported >> - // >> - ThreadBits = 0; >> - } >> - >> - // >> - // Software must not assume any "level type" encoding >> - // value to be related to any sub-leaf index, except sub-leaf 0. >> - // >> - SubIndex = 1; >> - do { >> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, NULL, >> &RegEcx, NULL); >> - LevelType = (RegEcx >> 8) & 0xff; >> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >> - CoreBits = (RegEax & 0x1f) - ThreadBits; >> - break; >> - } >> - SubIndex++; >> - } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >> - } >> - } >> - >> - if (!TopologyLeafSupported) { >> - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); >> - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; >> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); >> - MaxCoresPerPackage = (RegEax >> 26) + 1; >> - } else { >> - // >> - // Must be a single-core processor. >> - // >> - MaxCoresPerPackage = 1; >> - } >> - >> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / >> MaxCoresPerPackage - 1) + 1); >> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); >> - } >> - >> - Location->Thread = ApicId & ~((-1) << ThreadBits); >> - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); >> - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); -} >> - >> -/** >> Gets processor information on the requested processor at the instant this >> call is made. >> >> @param[in] This A pointer to the >> EFI_SMM_CPU_SERVICE_PROTOCOL instance. >> @@ -280,7 +161,12 @@ SmmAddProcessor ( >> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == >> INVALID_APIC_ID) { >> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; >> gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; >> - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- >>> ProcessorInfo[Index].Location); >> + GetProcessorLocation ( >> + (UINT32)ProcessorId, >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Package, >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Core, >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Thread >> + ); >> >> *ProcessorNumber = Index; >> gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; >> -- >> 2.6.3.windows.1 >> >> >> Best regards, >> >> Mike >> >>> -----Original Message----- >>> From: Leo Duran [mailto:leo.duran@amd.com] >>> Sent: Friday, October 28, 2016 10:39 AM >>> To: edk2-devel@lists.01.org >>> Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff >>> <jeff.fan@intel.com>; Kinney, Michael D <michael.d.kinney@intel.com>; >> Leo Duran >>> <leo.duran@amd.com> >>> Subject: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib >> library >>> >>> 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. >>> 2) Remove ExtractProcessorLocation() from MpInitLib library. >>> 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. >>> >>> Contributed-under: TianoCore Contribution Agreement 1.0 >>> Signed-off-by: Leo Duran <leo.duran@amd.com> >>> --- >>> UefiCpuPkg/Include/Library/LocalApicLib.h | 18 +++ >>> UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 130 >> +++++++++++++++++++++ >>> .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 130 >> +++++++++++++++++++++ >>> UefiCpuPkg/Library/MpInitLib/MpLib.c | 128 +------------------- >>> UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 121 +------------------ >>> 5 files changed, 280 insertions(+), 247 deletions(-) >>> >>> diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h >>> b/UefiCpuPkg/Include/Library/LocalApicLib.h >>> index cd4e613..4abf64c 100644 >>> --- a/UefiCpuPkg/Include/Library/LocalApicLib.h >>> +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h >>> @@ -21,6 +21,9 @@ >>> #define LOCAL_APIC_MODE_XAPIC 0x1 ///< xAPIC mode. >>> #define LOCAL_APIC_MODE_X2APIC 0x2 ///< x2APIC mode. >>> >>> +#include <PiPei.h> >>> +#include <Protocol/MpService.h> >>> + >>> /** >>> Retrieve the base address of local APIC. >>> >>> @@ -410,6 +413,21 @@ GetApicMsiValue ( >>> IN BOOLEAN LevelTriggered, >>> IN BOOLEAN AssertionLevel >>> ); >>> + >>> +/** >>> +Get Package ID/Core ID/Thread ID of a processor. >>> + >>> +The algorithm assumes the target system has symmetry across physical >> package >>> boundaries >>> +with respect to the number of logical processors per package, number of >> cores per >>> package. >>> + >>> +@param InitialApicId Must be the initial APIC ID of the target logical >> processor. >>> +@param Location Returns the processor location information. >>> +**/ >>> +VOID >>> +GetProcessorLocation( >>> + IN UINT32 InitialApicId, >>> + OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> +); >>> >>> #endif >>> >>> diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >>> b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >>> index 8d0fb02..219f99f 100644 >>> --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >>> +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c >>> @@ -941,3 +941,133 @@ GetApicMsiValue ( >>> } >>> return MsiData.Uint64; >>> } >>> + >>> +/** >>> +Get Package ID/Core ID/Thread ID of a processor. >>> + >>> +The algorithm assumes the target system has symmetry across physical >> package >>> boundaries >>> +with respect to the number of logical processors per package, number of >> cores per >>> package. >>> + >>> +@param InitialApicId Must be the initial APIC ID of the target logical >> processor. >>> +@param Location Returns the processor location information. >>> +**/ >>> +VOID >>> +GetProcessorLocation( >>> +IN UINT32 InitialApicId, >>> +OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> +) >>> +{ >>> + BOOLEAN TopologyLeafSupported; >>> + UINTN ThreadBits; >>> + UINTN CoreBits; >>> + CPUID_VERSION_INFO_EBX VersionInfoEbx; >>> + CPUID_VERSION_INFO_EDX VersionInfoEdx; >>> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; >>> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >>> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >>> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >>> + UINT32 MaxCpuIdIndex; >>> + UINT32 SubIndex; >>> + UINTN LevelType; >>> + UINT32 MaxLogicProcessorsPerPackage; >>> + UINT32 MaxCoresPerPackage; >>> + >>> + // >>> + // Check if the processor is capable of supporting more than one >> logical >>> processor. >>> + // >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, >> &VersionInfoEdx.Uint32); >>> + if (VersionInfoEdx.Bits.HTT == 0) { >>> + Location->Thread = 0; >>> + Location->Core = 0; >>> + Location->Package = 0; >>> + return; >>> + } >>> + >>> + ThreadBits = 0; >>> + CoreBits = 0; >>> + >>> + // >>> + // Assume three-level mapping of APIC ID: Package:Core:SMT. >>> + // >>> + >>> + TopologyLeafSupported = FALSE; >>> + // >>> + // Get the max index of basic CPUID >>> + // >>> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, >> NULL); >>> + >>> + // >>> + // If the extended topology enumeration leaf is available, it >>> + // is the preferred mechanism for enumerating topology. >>> + // >>> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >>> + AsmCpuidEx( >>> + CPUID_EXTENDED_TOPOLOGY, >>> + 0, >>> + &ExtendedTopologyEax.Uint32, >>> + &ExtendedTopologyEbx.Uint32, >>> + &ExtendedTopologyEcx.Uint32, >>> + NULL >>> + ); >>> + // >>> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and >> maximum input value for >>> + // basic CPUID information is greater than 0BH, then >> CPUID.0BH leaf is not >>> + // supported on that processor. >>> + // >>> + if (ExtendedTopologyEbx.Uint32 != 0) { >>> + TopologyLeafSupported = TRUE; >>> + >>> + // >>> + // Sub-leaf index 0 (ECX= 0 as input) provides >> enumeration parameters to >>> extract >>> + // the SMT sub-field of x2APIC ID. >>> + // >>> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >>> + ASSERT(LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >>> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >>> + >>> + // >>> + // Software must not assume any "level type" >> encoding >>> + // value to be related to any sub-leaf index, except >> sub-leaf 0. >>> + // >>> + SubIndex = 1; >>> + do { >>> + AsmCpuidEx( >>> + CPUID_EXTENDED_TOPOLOGY, >>> + SubIndex, >>> + &ExtendedTopologyEax.Uint32, >>> + NULL, >>> + &ExtendedTopologyEcx.Uint32, >>> + NULL >>> + ); >>> + LevelType = >> ExtendedTopologyEcx.Bits.LevelType; >>> + if (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >>> + CoreBits = >> ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >>> + break; >>> + } >>> + SubIndex++; >>> + } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >>> + } >>> + } >>> + >>> + if (!TopologyLeafSupported) { >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, >> &VersionInfoEbx.Uint32, NULL, NULL); >>> + MaxLogicProcessorsPerPackage = >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >>> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >>> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, >> &CacheParamsEax.Uint32, NULL, NULL, >>> NULL); >>> + MaxCoresPerPackage = >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >>> + } >>> + else { >>> + // >>> + // Must be a single-core processor. >>> + // >>> + MaxCoresPerPackage = 1; >>> + } >>> + >>> + ThreadBits = >> (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / >>> MaxCoresPerPackage - 1) + 1); >>> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + >> 1); >>> + } >>> + >>> + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); >>> + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >>> + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); >>> +} >>> diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >>> b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >>> index 4c42696..16395a3 100644 >>> --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >>> +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c >>> @@ -1036,3 +1036,133 @@ GetApicMsiValue ( >>> } >>> return MsiData.Uint64; >>> } >>> + >>> +/** >>> +Get Package ID/Core ID/Thread ID of a processor. >>> + >>> +The algorithm assumes the target system has symmetry across physical >> package >>> boundaries >>> +with respect to the number of logical processors per package, number of >> cores per >>> package. >>> + >>> +@param InitialApicId Must be the initial APIC ID of the target logical >> processor. >>> +@param Location Returns the processor location information. >>> +**/ >>> +VOID >>> +GetProcessorLocation( >>> +IN UINT32 InitialApicId, >>> +OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> +) >>> +{ >>> + BOOLEAN TopologyLeafSupported; >>> + UINTN ThreadBits; >>> + UINTN CoreBits; >>> + CPUID_VERSION_INFO_EBX VersionInfoEbx; >>> + CPUID_VERSION_INFO_EDX VersionInfoEdx; >>> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; >>> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >>> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >>> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >>> + UINT32 MaxCpuIdIndex; >>> + UINT32 SubIndex; >>> + UINTN LevelType; >>> + UINT32 MaxLogicProcessorsPerPackage; >>> + UINT32 MaxCoresPerPackage; >>> + >>> + // >>> + // Check if the processor is capable of supporting more than one >> logical >>> processor. >>> + // >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, >> &VersionInfoEdx.Uint32); >>> + if (VersionInfoEdx.Bits.HTT == 0) { >>> + Location->Thread = 0; >>> + Location->Core = 0; >>> + Location->Package = 0; >>> + return; >>> + } >>> + >>> + ThreadBits = 0; >>> + CoreBits = 0; >>> + >>> + // >>> + // Assume three-level mapping of APIC ID: Package:Core:SMT. >>> + // >>> + >>> + TopologyLeafSupported = FALSE; >>> + // >>> + // Get the max index of basic CPUID >>> + // >>> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, >> NULL); >>> + >>> + // >>> + // If the extended topology enumeration leaf is available, it >>> + // is the preferred mechanism for enumerating topology. >>> + // >>> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >>> + AsmCpuidEx( >>> + CPUID_EXTENDED_TOPOLOGY, >>> + 0, >>> + &ExtendedTopologyEax.Uint32, >>> + &ExtendedTopologyEbx.Uint32, >>> + &ExtendedTopologyEcx.Uint32, >>> + NULL >>> + ); >>> + // >>> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and >> maximum input value for >>> + // basic CPUID information is greater than 0BH, then >> CPUID.0BH leaf is not >>> + // supported on that processor. >>> + // >>> + if (ExtendedTopologyEbx.Uint32 != 0) { >>> + TopologyLeafSupported = TRUE; >>> + >>> + // >>> + // Sub-leaf index 0 (ECX= 0 as input) provides >> enumeration parameters to >>> extract >>> + // the SMT sub-field of x2APIC ID. >>> + // >>> + LevelType = ExtendedTopologyEcx.Bits.LevelType; >>> + ASSERT(LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >>> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >>> + >>> + // >>> + // Software must not assume any "level type" >> encoding >>> + // value to be related to any sub-leaf index, except >> sub-leaf 0. >>> + // >>> + SubIndex = 1; >>> + do { >>> + AsmCpuidEx( >>> + CPUID_EXTENDED_TOPOLOGY, >>> + SubIndex, >>> + &ExtendedTopologyEax.Uint32, >>> + NULL, >>> + &ExtendedTopologyEcx.Uint32, >>> + NULL >>> + ); >>> + LevelType = >> ExtendedTopologyEcx.Bits.LevelType; >>> + if (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >>> + CoreBits = >> ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >>> + break; >>> + } >>> + SubIndex++; >>> + } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >>> + } >>> + } >>> + >>> + if (!TopologyLeafSupported) { >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, >> &VersionInfoEbx.Uint32, NULL, NULL); >>> + MaxLogicProcessorsPerPackage = >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >>> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >>> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, >> &CacheParamsEax.Uint32, NULL, NULL, >>> NULL); >>> + MaxCoresPerPackage = >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >>> + } >>> + else { >>> + // >>> + // Must be a single-core processor. >>> + // >>> + MaxCoresPerPackage = 1; >>> + } >>> + >>> + ThreadBits = >> (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / >>> MaxCoresPerPackage - 1) + 1); >>> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + >> 1); >>> + } >>> + >>> + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); >>> + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >>> + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); >>> +} >>> diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c >>> b/UefiCpuPkg/Library/MpInitLib/MpLib.c >>> index c3fe721..f3380bb 100644 >>> --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c >>> +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c >>> @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( >>> } >>> >>> /** >>> - Get CPU Package/Core/Thread location information. >>> - >>> - @param[in] InitialApicId CPU APIC ID >>> - @param[out] Location Pointer to CPU location information >>> -**/ >>> -VOID >>> -ExtractProcessorLocation ( >>> - IN UINT32 InitialApicId, >>> - OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> - ) >>> -{ >>> - BOOLEAN TopologyLeafSupported; >>> - UINTN ThreadBits; >>> - UINTN CoreBits; >>> - CPUID_VERSION_INFO_EBX VersionInfoEbx; >>> - CPUID_VERSION_INFO_EDX VersionInfoEdx; >>> - CPUID_CACHE_PARAMS_EAX CacheParamsEax; >>> - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; >>> - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; >>> - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; >>> - UINT32 MaxCpuIdIndex; >>> - UINT32 SubIndex; >>> - UINTN LevelType; >>> - UINT32 MaxLogicProcessorsPerPackage; >>> - UINT32 MaxCoresPerPackage; >>> - >>> - // >>> - // Check if the processor is capable of supporting more than one logical >> processor. >>> - // >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, >> &VersionInfoEdx.Uint32); >>> - if (VersionInfoEdx.Bits.HTT == 0) { >>> - Location->Thread = 0; >>> - Location->Core = 0; >>> - Location->Package = 0; >>> - return; >>> - } >>> - >>> - ThreadBits = 0; >>> - CoreBits = 0; >>> - >>> - // >>> - // Assume three-level mapping of APIC ID: Package:Core:SMT. >>> - // >>> - >>> - TopologyLeafSupported = FALSE; >>> - // >>> - // Get the max index of basic CPUID >>> - // >>> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >>> - >>> - // >>> - // If the extended topology enumeration leaf is available, it >>> - // is the preferred mechanism for enumerating topology. >>> - // >>> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >>> - AsmCpuidEx ( >>> - CPUID_EXTENDED_TOPOLOGY, >>> - 0, >>> - &ExtendedTopologyEax.Uint32, >>> - &ExtendedTopologyEbx.Uint32, >>> - &ExtendedTopologyEcx.Uint32, >>> - NULL >>> - ); >>> - // >>> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >>> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is >> not >>> - // supported on that processor. >>> - // >>> - if (ExtendedTopologyEbx.Uint32 != 0) { >>> - TopologyLeafSupported = TRUE; >>> - >>> - // >>> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters >> to extract >>> - // the SMT sub-field of x2APIC ID. >>> - // >>> - LevelType = ExtendedTopologyEcx.Bits.LevelType; >>> - ASSERT (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >>> - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; >>> - >>> - // >>> - // Software must not assume any "level type" encoding >>> - // value to be related to any sub-leaf index, except sub-leaf 0. >>> - // >>> - SubIndex = 1; >>> - do { >>> - AsmCpuidEx ( >>> - CPUID_EXTENDED_TOPOLOGY, >>> - SubIndex, >>> - &ExtendedTopologyEax.Uint32, >>> - NULL, >>> - &ExtendedTopologyEcx.Uint32, >>> - NULL >>> - ); >>> - LevelType = ExtendedTopologyEcx.Bits.LevelType; >>> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >>> - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; >>> - break; >>> - } >>> - SubIndex++; >>> - } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >>> - } >>> - } >>> - >>> - if (!TopologyLeafSupported) { >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, >> NULL, NULL); >>> - MaxLogicProcessorsPerPackage = >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; >>> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >>> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, >> NULL, NULL, NULL); >>> - MaxCoresPerPackage = >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; >>> - } else { >>> - // >>> - // Must be a single-core processor. >>> - // >>> - MaxCoresPerPackage = 1; >>> - } >>> - >>> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / >>> MaxCoresPerPackage - 1) + 1); >>> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); >>> - } >>> - >>> - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); >>> - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); >>> - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); >>> -} >>> - >>> -/** >>> Worker function for SwitchBSP(). >>> >>> Worker function for SwitchBSP(), assigned to the AP which is intended >>> @@ -1451,7 +1325,7 @@ MpInitLibGetProcessorInfo ( >>> // >>> // Get processor location information >>> // >>> - ExtractProcessorLocation (CpuMpData- >>> CpuData[ProcessorNumber].ApicId, >>> &ProcessorInfoBuffer->Location); >>> + GetProcessorLocation (CpuMpData- >>> CpuData[ProcessorNumber].ApicId, >>> &ProcessorInfoBuffer->Location); >>> >>> if (HealthData != NULL) { >>> HealthData->Uint32 = CpuMpData->CpuData[ProcessorNumber].Health; >>> diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >>> b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >>> index 40f2a17..67cd0a0 100644 >>> --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >>> +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c >>> @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL >> mSmmCpuService = { >>> }; >>> >>> /** >>> - Get Package ID/Core ID/Thread ID of a processor. >>> - >>> - APIC ID must be an initial APIC ID. >>> - >>> - The algorithm below assumes the target system has symmetry across >> physical package >>> boundaries >>> - with respect to the number of logical processors per package, number of >> cores per >>> package. >>> - >>> - @param ApicId APIC ID of the target logical processor. >>> - @param Location Returns the processor location information. >>> -**/ >>> -VOID >>> -SmmGetProcessorLocation ( >>> - IN UINT32 ApicId, >>> - OUT EFI_CPU_PHYSICAL_LOCATION *Location >>> - ) >>> -{ >>> - UINTN ThreadBits; >>> - UINTN CoreBits; >>> - UINT32 RegEax; >>> - UINT32 RegEbx; >>> - UINT32 RegEcx; >>> - UINT32 RegEdx; >>> - UINT32 MaxCpuIdIndex; >>> - UINT32 SubIndex; >>> - UINTN LevelType; >>> - UINT32 MaxLogicProcessorsPerPackage; >>> - UINT32 MaxCoresPerPackage; >>> - BOOLEAN TopologyLeafSupported; >>> - >>> - ASSERT (Location != NULL); >>> - >>> - ThreadBits = 0; >>> - CoreBits = 0; >>> - TopologyLeafSupported = FALSE; >>> - >>> - // >>> - // Check if the processor is capable of supporting more than one logical >> processor. >>> - // >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); >>> - ASSERT ((RegEdx & BIT28) != 0); >>> - >>> - // >>> - // Assume three-level mapping of APIC ID: Package:Core:SMT. >>> - // >>> - >>> - // >>> - // Get the max index of basic CPUID >>> - // >>> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); >>> - >>> - // >>> - // If the extended topology enumeration leaf is available, it >>> - // is the preferred mechanism for enumerating topology. >>> - // >>> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { >>> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, >> &RegEcx, NULL); >>> - // >>> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input >> value for >>> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is >> not >>> - // supported on that processor. >>> - // >>> - if ((RegEbx & 0xffff) != 0) { >>> - TopologyLeafSupported = TRUE; >>> - >>> - // >>> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration parameters >> to extract >>> - // the SMT sub-field of x2APIC ID. >>> - // >>> - LevelType = (RegEcx >> 8) & 0xff; >>> - ASSERT (LevelType == >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); >>> - if ((RegEbx & 0xffff) > 1 ) { >>> - ThreadBits = RegEax & 0x1f; >>> - } else { >>> - // >>> - // HT is not supported >>> - // >>> - ThreadBits = 0; >>> - } >>> - >>> - // >>> - // Software must not assume any "level type" encoding >>> - // value to be related to any sub-leaf index, except sub-leaf 0. >>> - // >>> - SubIndex = 1; >>> - do { >>> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, >> NULL, &RegEcx, NULL); >>> - LevelType = (RegEcx >> 8) & 0xff; >>> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { >>> - CoreBits = (RegEax & 0x1f) - ThreadBits; >>> - break; >>> - } >>> - SubIndex++; >>> - } while (LevelType != >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); >>> - } >>> - } >>> - >>> - if (!TopologyLeafSupported) { >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); >>> - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; >>> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { >>> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, NULL); >>> - MaxCoresPerPackage = (RegEax >> 26) + 1; >>> - } else { >>> - // >>> - // Must be a single-core processor. >>> - // >>> - MaxCoresPerPackage = 1; >>> - } >>> - >>> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / >>> MaxCoresPerPackage - 1) + 1); >>> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); >>> - } >>> - >>> - Location->Thread = ApicId & ~((-1) << ThreadBits); >>> - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); >>> - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); >>> -} >>> - >>> -/** >>> Gets processor information on the requested processor at the instant this >> call is >>> made. >>> >>> @param[in] This A pointer to the >> EFI_SMM_CPU_SERVICE_PROTOCOL >>> instance. >>> @@ -280,7 +161,7 @@ SmmAddProcessor ( >>> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == >> INVALID_APIC_ID) { >>> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; >>> gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; >>> - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- >>>> ProcessorInfo[Index].Location); >>> + GetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- >>>> ProcessorInfo[Index].Location); >>> >>> *ProcessorNumber = Index; >>> gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; >>> -- >>> 1.9.1 > _______________________________________________ edk2-devel mailing list edk2-devel@lists.01.org https://lists.01.org/mailman/listinfo/edk2-devel ^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library 2016-10-31 6:40 ` Fan, Jeff @ 2016-10-31 16:45 ` Duran, Leo 0 siblings, 0 replies; 7+ messages in thread From: Duran, Leo @ 2016-10-31 16:45 UTC (permalink / raw) To: 'Fan, Jeff', Laszlo Ersek, Kinney, Michael D, edk2-devel@lists.01.org Cc: Gao, Liming Please see my reply below. Leo. > -----Original Message----- > From: Fan, Jeff [mailto:jeff.fan@intel.com] > Sent: Monday, October 31, 2016 1:41 AM > To: Laszlo Ersek <lersek@redhat.com>; Duran, Leo <leo.duran@amd.com>; > Kinney, Michael D <michael.d.kinney@intel.com>; edk2-devel@lists.01.org > <edk2-devel@ml01.01.org> > Cc: Gao, Liming <liming.gao@intel.com> > Subject: RE: [edk2] [v3] UefiCpuPkg: Move GetProcessorLocation() to > LocalApicLib library > > Leo, > > Reviewed-by: Jeff Fan <jeff.fan@intel.com> based on Mike's updating. > > Could you refine the final patch and send out for testing(requested by > Laszlo)? And then I could check-in it also. > > Thanks! > Jeff [Duran, Leo] Sure, I'll try to send it out a bit later today. Leo. > > -----Original Message----- > From: edk2-devel [mailto:edk2-devel-bounces@lists.01.org] On Behalf Of > Laszlo Ersek > Sent: Saturday, October 29, 2016 5:48 AM > To: Duran, Leo; Kinney, Michael D; edk2-devel@lists.01.org > Cc: Fan, Jeff; Gao, Liming > Subject: Re: [edk2] [v3] UefiCpuPkg: Move GetProcessorLocation() to > LocalApicLib library > > On 10/28/16 23:17, Duran, Leo wrote: > > Replies below. > > Thanks. > > > >> -----Original Message----- > >> From: Kinney, Michael D [mailto:michael.d.kinney@intel.com] > >> Sent: Friday, October 28, 2016 4:12 PM > >> To: Duran, Leo <leo.duran@amd.com>; edk2-devel@lists.01.org > >> Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff > >> <jeff.fan@intel.com> > >> Subject: RE: [v3] UefiCpuPkg: Move GetProcessorLocation() to > LocalApicLib > >> library > >> > >> Leo, > >> > >> I tested your patch on Quark and it builds and boots. > >> > >> Here is a slightly updated version of the patch that makes the following > >> changes: > >> > >> * Follow Jeff Fan suggestion to pass in Package, Core, Thread as 3 > >> parameters. > >> This removes the dependency on PiPei.h and MpService.h from > >> LocalApicLib.h > >> so LocalApicLib can be used from more than just PEI/DXE phases. I also > >> made > >> these three OUT parameters OPTIONAL. > >> * Change tabs to spaces and fix other minor format issues to pass checks > >> from > >> BaseTools/Scripts/PatchCheck.py > >> > >> I have also tested this updated patch on Quark and it also builds and > boots. > >> > >> > > [Duran, Leo] > > Sounds a good thing! > > (I assume Lazlo will probably test this as well) > > If Mike thinks the changes warrant it, I certainly can (could be next > week), but for that, I'd need a regularly formatted & mailed patch, one > that I can apply with "git am" from the list. > > Thanks > Laszlo > > > Thanks Mike. > > Leo. > > > >> > >> From cc932ca811e052a0a3ae103a9db9614b3807d4ed Mon Sep 17 00:00:00 > >> 2001 > >> From: Leo Duran <leo.duran@amd.com> > >> Date: Fri, 28 Oct 2016 10:38:45 -0700 > >> Subject: [Patch] UefiCpuPkg: Move GetProcessorLocation() to > LocalApicLib > >> library > >> > >> 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm driver. > >> 2) Remove ExtractProcessorLocation() from MpInitLib library. > >> 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. > >> > >> Cc: Jeff Fan <jeff.fan@intel.com> > >> Cc: Laszlo Ersek <lersek@redhat.com> > >> Cc: Michael D Kinney <michael.d.kinney@intel.com> > >> Contributed-under: TianoCore Contribution Agreement 1.0 > >> Signed-off-by: Leo Duran <leo.duran@amd.com> > >> --- > >> UefiCpuPkg/Include/Library/LocalApicLib.h | 21 +++ > >> UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 148 > >> +++++++++++++++++++++ > >> .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 148 > >> +++++++++++++++++++++ > >> UefiCpuPkg/Library/MpInitLib/MpLib.c | 133 +----------------- > >> UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 126 +---------------- > - > >> 5 files changed, 329 insertions(+), 247 deletions(-) > >> > >> diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h > >> b/UefiCpuPkg/Include/Library/LocalApicLib.h > >> index cd4e613..d73198f 100644 > >> --- a/UefiCpuPkg/Include/Library/LocalApicLib.h > >> +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h > >> @@ -410,6 +410,27 @@ GetApicMsiValue ( > >> IN BOOLEAN LevelTriggered, > >> IN BOOLEAN AssertionLevel > >> ); > >> + > >> +/** > >> + Get Package ID/Core ID/Thread ID of a processor. > >> + > >> + The algorithm assumes the target system has symmetry across physical > >> + package boundaries with respect to the number of logical processors > >> + per package, number of cores per package. > >> + > >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical > >> + processor. > >> + @param[out] Package Returns the processor package ID. > >> + @param[out] Core Returns the processor core ID. > >> + @param[out] Thread Returns the processor thread ID. > >> +**/ > >> +VOID > >> +GetProcessorLocation( > >> + IN UINT32 InitialApicId, > >> + OUT UINT32 *Package OPTIONAL, > >> + OUT UINT32 *Core OPTIONAL, > >> + OUT UINT32 *Thread OPTIONAL > >> + ); > >> > >> #endif > >> > >> diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > >> b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > >> index 8d0fb02..7ddee1b 100644 > >> --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > >> +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > >> @@ -941,3 +941,151 @@ GetApicMsiValue ( > >> } > >> return MsiData.Uint64; > >> } > >> + > >> +/** > >> + Get Package ID/Core ID/Thread ID of a processor. > >> + > >> + The algorithm assumes the target system has symmetry across physical > >> + package boundaries with respect to the number of logical processors > >> + per package, number of cores per package. > >> + > >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical > >> + processor. > >> + @param[out] Package Returns the processor package ID. > >> + @param[out] Core Returns the processor core ID. > >> + @param[out] Thread Returns the processor thread ID. > >> +**/ > >> +VOID > >> +GetProcessorLocation( > >> + IN UINT32 InitialApicId, > >> + OUT UINT32 *Package OPTIONAL, > >> + OUT UINT32 *Core OPTIONAL, > >> + OUT UINT32 *Thread OPTIONAL > >> + ) > >> +{ > >> + BOOLEAN TopologyLeafSupported; > >> + UINTN ThreadBits; > >> + UINTN CoreBits; > >> + CPUID_VERSION_INFO_EBX VersionInfoEbx; > >> + CPUID_VERSION_INFO_EDX VersionInfoEdx; > >> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > >> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > >> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > >> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > >> + UINT32 MaxCpuIdIndex; > >> + UINT32 SubIndex; > >> + UINTN LevelType; > >> + UINT32 MaxLogicProcessorsPerPackage; > >> + UINT32 MaxCoresPerPackage; > >> + > >> + // > >> + // Check if the processor is capable of supporting more than one logical > >> processor. > >> + // > >> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, > >> + &VersionInfoEdx.Uint32); if (VersionInfoEdx.Bits.HTT == 0) { > >> + if (Thread != NULL) { > >> + *Thread = 0; > >> + } > >> + if (Core != NULL) { > >> + *Core = 0; > >> + } > >> + if (Package != NULL) { > >> + *Package = 0; > >> + } > >> + return; > >> + } > >> + > >> + ThreadBits = 0; > >> + CoreBits = 0; > >> + > >> + // > >> + // Assume three-level mapping of APIC ID: Package:Core:SMT. > >> + // > >> + > >> + TopologyLeafSupported = FALSE; > >> + // > >> + // Get the max index of basic CPUID > >> + // > >> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > >> + > >> + // > >> + // If the extended topology enumeration leaf is available, it // is > >> + the preferred mechanism for enumerating topology. > >> + // > >> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > >> + AsmCpuidEx( > >> + CPUID_EXTENDED_TOPOLOGY, > >> + 0, > >> + &ExtendedTopologyEax.Uint32, > >> + &ExtendedTopologyEbx.Uint32, > >> + &ExtendedTopologyEcx.Uint32, > >> + NULL > >> + ); > >> + // > >> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > >> value for > >> + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is > not > >> + // supported on that processor. > >> + // > >> + if (ExtendedTopologyEbx.Uint32 != 0) { > >> + TopologyLeafSupported = TRUE; > >> + > >> + // > >> + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration > parameters > >> to extract > >> + // the SMT sub-field of x2APIC ID. > >> + // > >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; > >> + ASSERT(LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > >> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > >> + > >> + // > >> + // Software must not assume any "level type" encoding > >> + // value to be related to any sub-leaf index, except sub-leaf 0. > >> + // > >> + SubIndex = 1; > >> + do { > >> + AsmCpuidEx( > >> + CPUID_EXTENDED_TOPOLOGY, > >> + SubIndex, > >> + &ExtendedTopologyEax.Uint32, > >> + NULL, > >> + &ExtendedTopologyEcx.Uint32, > >> + NULL > >> + ); > >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; > >> + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) > { > >> + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > >> + break; > >> + } > >> + SubIndex++; > >> + } while (LevelType != > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > >> + } > >> + } > >> + > >> + if (!TopologyLeafSupported) { > >> + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, > NULL, > >> NULL); > >> + MaxLogicProcessorsPerPackage = > >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > >> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > >> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, > >> NULL, NULL, NULL); > >> + MaxCoresPerPackage = > >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > >> + } > >> + else { > >> + // > >> + // Must be a single-core processor. > >> + // > >> + MaxCoresPerPackage = 1; > >> + } > >> + > >> + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > >> MaxCoresPerPackage - 1) + 1); > >> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); } > >> + > >> + if (Thread != NULL) { > >> + *Thread = InitialApicId & ((1 << ThreadBits) - 1); > >> + } > >> + if (Core != NULL) { > >> + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > >> + } > >> + if (Package != NULL) { > >> + *Package = (InitialApicId >> (ThreadBits + CoreBits)); > >> + } > >> +} > >> diff --git a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > >> b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > >> index 4c42696..c620211 100644 > >> --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > >> +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > >> @@ -1036,3 +1036,151 @@ GetApicMsiValue ( > >> } > >> return MsiData.Uint64; > >> } > >> + > >> +/** > >> + Get Package ID/Core ID/Thread ID of a processor. > >> + > >> + The algorithm assumes the target system has symmetry across physical > >> + package boundaries with respect to the number of logical processors > >> + per package, number of cores per package. > >> + > >> + @param[in] InitialApicId Must be the initial APIC ID of the target logical > >> + processor. > >> + @param[out] Package Returns the processor package ID. > >> + @param[out] Core Returns the processor core ID. > >> + @param[out] Thread Returns the processor thread ID. > >> +**/ > >> +VOID > >> +GetProcessorLocation( > >> + IN UINT32 InitialApicId, > >> + OUT UINT32 *Package OPTIONAL, > >> + OUT UINT32 *Core OPTIONAL, > >> + OUT UINT32 *Thread OPTIONAL > >> + ) > >> +{ > >> + BOOLEAN TopologyLeafSupported; > >> + UINTN ThreadBits; > >> + UINTN CoreBits; > >> + CPUID_VERSION_INFO_EBX VersionInfoEbx; > >> + CPUID_VERSION_INFO_EDX VersionInfoEdx; > >> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > >> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > >> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > >> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > >> + UINT32 MaxCpuIdIndex; > >> + UINT32 SubIndex; > >> + UINTN LevelType; > >> + UINT32 MaxLogicProcessorsPerPackage; > >> + UINT32 MaxCoresPerPackage; > >> + > >> + // > >> + // Check if the processor is capable of supporting more than one logical > >> processor. > >> + // > >> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, > >> + &VersionInfoEdx.Uint32); if (VersionInfoEdx.Bits.HTT == 0) { > >> + if (Thread != NULL) { > >> + *Thread = 0; > >> + } > >> + if (Core != NULL) { > >> + *Core = 0; > >> + } > >> + if (Package != NULL) { > >> + *Package = 0; > >> + } > >> + return; > >> + } > >> + > >> + ThreadBits = 0; > >> + CoreBits = 0; > >> + > >> + // > >> + // Assume three-level mapping of APIC ID: Package:Core:SMT. > >> + // > >> + > >> + TopologyLeafSupported = FALSE; > >> + // > >> + // Get the max index of basic CPUID > >> + // > >> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > >> + > >> + // > >> + // If the extended topology enumeration leaf is available, it // is > >> + the preferred mechanism for enumerating topology. > >> + // > >> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > >> + AsmCpuidEx( > >> + CPUID_EXTENDED_TOPOLOGY, > >> + 0, > >> + &ExtendedTopologyEax.Uint32, > >> + &ExtendedTopologyEbx.Uint32, > >> + &ExtendedTopologyEcx.Uint32, > >> + NULL > >> + ); > >> + // > >> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > >> value for > >> + // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is > not > >> + // supported on that processor. > >> + // > >> + if (ExtendedTopologyEbx.Uint32 != 0) { > >> + TopologyLeafSupported = TRUE; > >> + > >> + // > >> + // Sub-leaf index 0 (ECX= 0 as input) provides enumeration > parameters > >> to extract > >> + // the SMT sub-field of x2APIC ID. > >> + // > >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; > >> + ASSERT(LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > >> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > >> + > >> + // > >> + // Software must not assume any "level type" encoding > >> + // value to be related to any sub-leaf index, except sub-leaf 0. > >> + // > >> + SubIndex = 1; > >> + do { > >> + AsmCpuidEx( > >> + CPUID_EXTENDED_TOPOLOGY, > >> + SubIndex, > >> + &ExtendedTopologyEax.Uint32, > >> + NULL, > >> + &ExtendedTopologyEcx.Uint32, > >> + NULL > >> + ); > >> + LevelType = ExtendedTopologyEcx.Bits.LevelType; > >> + if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) > { > >> + CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > >> + break; > >> + } > >> + SubIndex++; > >> + } while (LevelType != > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > >> + } > >> + } > >> + > >> + if (!TopologyLeafSupported) { > >> + AsmCpuid(CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, > NULL, > >> NULL); > >> + MaxLogicProcessorsPerPackage = > >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > >> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > >> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, > >> NULL, NULL, NULL); > >> + MaxCoresPerPackage = > >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > >> + } > >> + else { > >> + // > >> + // Must be a single-core processor. > >> + // > >> + MaxCoresPerPackage = 1; > >> + } > >> + > >> + ThreadBits = (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > >> MaxCoresPerPackage - 1) + 1); > >> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + 1); } > >> + > >> + if (Thread != NULL) { > >> + *Thread = InitialApicId & ((1 << ThreadBits) - 1); > >> + } > >> + if (Core != NULL) { > >> + *Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > >> + } > >> + if (Package != NULL) { > >> + *Package = (InitialApicId >> (ThreadBits + CoreBits)); > >> + } > >> +} > >> diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c > >> b/UefiCpuPkg/Library/MpInitLib/MpLib.c > >> index c3fe721..f205b6b 100644 > >> --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c > >> +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c > >> @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( } > >> > >> /** > >> - Get CPU Package/Core/Thread location information. > >> - > >> - @param[in] InitialApicId CPU APIC ID > >> - @param[out] Location Pointer to CPU location information > >> -**/ > >> -VOID > >> -ExtractProcessorLocation ( > >> - IN UINT32 InitialApicId, > >> - OUT EFI_CPU_PHYSICAL_LOCATION *Location > >> - ) > >> -{ > >> - BOOLEAN TopologyLeafSupported; > >> - UINTN ThreadBits; > >> - UINTN CoreBits; > >> - CPUID_VERSION_INFO_EBX VersionInfoEbx; > >> - CPUID_VERSION_INFO_EDX VersionInfoEdx; > >> - CPUID_CACHE_PARAMS_EAX CacheParamsEax; > >> - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > >> - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > >> - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > >> - UINT32 MaxCpuIdIndex; > >> - UINT32 SubIndex; > >> - UINTN LevelType; > >> - UINT32 MaxLogicProcessorsPerPackage; > >> - UINT32 MaxCoresPerPackage; > >> - > >> - // > >> - // Check if the processor is capable of supporting more than one logical > >> processor. > >> - // > >> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, > >> &VersionInfoEdx.Uint32); > >> - if (VersionInfoEdx.Bits.HTT == 0) { > >> - Location->Thread = 0; > >> - Location->Core = 0; > >> - Location->Package = 0; > >> - return; > >> - } > >> - > >> - ThreadBits = 0; > >> - CoreBits = 0; > >> - > >> - // > >> - // Assume three-level mapping of APIC ID: Package:Core:SMT. > >> - // > >> - > >> - TopologyLeafSupported = FALSE; > >> - // > >> - // Get the max index of basic CPUID > >> - // > >> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > >> - > >> - // > >> - // If the extended topology enumeration leaf is available, it > >> - // is the preferred mechanism for enumerating topology. > >> - // > >> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > >> - AsmCpuidEx ( > >> - CPUID_EXTENDED_TOPOLOGY, > >> - 0, > >> - &ExtendedTopologyEax.Uint32, > >> - &ExtendedTopologyEbx.Uint32, > >> - &ExtendedTopologyEcx.Uint32, > >> - NULL > >> - ); > >> - // > >> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > >> value for > >> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is > not > >> - // supported on that processor. > >> - // > >> - if (ExtendedTopologyEbx.Uint32 != 0) { > >> - TopologyLeafSupported = TRUE; > >> - > >> - // > >> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration > parameters to > >> extract > >> - // the SMT sub-field of x2APIC ID. > >> - // > >> - LevelType = ExtendedTopologyEcx.Bits.LevelType; > >> - ASSERT (LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > >> - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > >> - > >> - // > >> - // Software must not assume any "level type" encoding > >> - // value to be related to any sub-leaf index, except sub-leaf 0. > >> - // > >> - SubIndex = 1; > >> - do { > >> - AsmCpuidEx ( > >> - CPUID_EXTENDED_TOPOLOGY, > >> - SubIndex, > >> - &ExtendedTopologyEax.Uint32, > >> - NULL, > >> - &ExtendedTopologyEcx.Uint32, > >> - NULL > >> - ); > >> - LevelType = ExtendedTopologyEcx.Bits.LevelType; > >> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > >> - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > >> - break; > >> - } > >> - SubIndex++; > >> - } while (LevelType != > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > >> - } > >> - } > >> - > >> - if (!TopologyLeafSupported) { > >> - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, > NULL, > >> NULL); > >> - MaxLogicProcessorsPerPackage = > >> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > >> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > >> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, > >> NULL, NULL, NULL); > >> - MaxCoresPerPackage = > >> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > >> - } else { > >> - // > >> - // Must be a single-core processor. > >> - // > >> - MaxCoresPerPackage = 1; > >> - } > >> - > >> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > >> MaxCoresPerPackage - 1) + 1); > >> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > >> - } > >> - > >> - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > >> - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > >> - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); -} > >> - > >> -/** > >> Worker function for SwitchBSP(). > >> > >> Worker function for SwitchBSP(), assigned to the AP which is intended > @@ > >> -1451,7 +1325,12 @@ MpInitLibGetProcessorInfo ( > >> // > >> // Get processor location information > >> // > >> - ExtractProcessorLocation (CpuMpData- > >>> CpuData[ProcessorNumber].ApicId, &ProcessorInfoBuffer->Location); > >> + GetProcessorLocation ( > >> + CpuMpData->CpuData[ProcessorNumber].ApicId, > >> + &ProcessorInfoBuffer->Location.Package, > >> + &ProcessorInfoBuffer->Location.Core, > >> + &ProcessorInfoBuffer->Location.Thread > >> + ); > >> > >> if (HealthData != NULL) { > >> HealthData->Uint32 = CpuMpData- > >CpuData[ProcessorNumber].Health; > >> diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > >> b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > >> index 40f2a17..93ebb9e 100644 > >> --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > >> +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > >> @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL > >> mSmmCpuService = { }; > >> > >> /** > >> - Get Package ID/Core ID/Thread ID of a processor. > >> - > >> - APIC ID must be an initial APIC ID. > >> - > >> - The algorithm below assumes the target system has symmetry across > >> physical package boundaries > >> - with respect to the number of logical processors per package, number > of > >> cores per package. > >> - > >> - @param ApicId APIC ID of the target logical processor. > >> - @param Location Returns the processor location information. > >> -**/ > >> -VOID > >> -SmmGetProcessorLocation ( > >> - IN UINT32 ApicId, > >> - OUT EFI_CPU_PHYSICAL_LOCATION *Location > >> - ) > >> -{ > >> - UINTN ThreadBits; > >> - UINTN CoreBits; > >> - UINT32 RegEax; > >> - UINT32 RegEbx; > >> - UINT32 RegEcx; > >> - UINT32 RegEdx; > >> - UINT32 MaxCpuIdIndex; > >> - UINT32 SubIndex; > >> - UINTN LevelType; > >> - UINT32 MaxLogicProcessorsPerPackage; > >> - UINT32 MaxCoresPerPackage; > >> - BOOLEAN TopologyLeafSupported; > >> - > >> - ASSERT (Location != NULL); > >> - > >> - ThreadBits = 0; > >> - CoreBits = 0; > >> - TopologyLeafSupported = FALSE; > >> - > >> - // > >> - // Check if the processor is capable of supporting more than one logical > >> processor. > >> - // > >> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); > >> - ASSERT ((RegEdx & BIT28) != 0); > >> - > >> - // > >> - // Assume three-level mapping of APIC ID: Package:Core:SMT. > >> - // > >> - > >> - // > >> - // Get the max index of basic CPUID > >> - // > >> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > >> - > >> - // > >> - // If the extended topology enumeration leaf is available, it > >> - // is the preferred mechanism for enumerating topology. > >> - // > >> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > >> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, > >> &RegEcx, NULL); > >> - // > >> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > >> value for > >> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is > not > >> - // supported on that processor. > >> - // > >> - if ((RegEbx & 0xffff) != 0) { > >> - TopologyLeafSupported = TRUE; > >> - > >> - // > >> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration > parameters to > >> extract > >> - // the SMT sub-field of x2APIC ID. > >> - // > >> - LevelType = (RegEcx >> 8) & 0xff; > >> - ASSERT (LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > >> - if ((RegEbx & 0xffff) > 1 ) { > >> - ThreadBits = RegEax & 0x1f; > >> - } else { > >> - // > >> - // HT is not supported > >> - // > >> - ThreadBits = 0; > >> - } > >> - > >> - // > >> - // Software must not assume any "level type" encoding > >> - // value to be related to any sub-leaf index, except sub-leaf 0. > >> - // > >> - SubIndex = 1; > >> - do { > >> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, > NULL, > >> &RegEcx, NULL); > >> - LevelType = (RegEcx >> 8) & 0xff; > >> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > >> - CoreBits = (RegEax & 0x1f) - ThreadBits; > >> - break; > >> - } > >> - SubIndex++; > >> - } while (LevelType != > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > >> - } > >> - } > >> - > >> - if (!TopologyLeafSupported) { > >> - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); > >> - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; > >> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > >> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, > NULL); > >> - MaxCoresPerPackage = (RegEax >> 26) + 1; > >> - } else { > >> - // > >> - // Must be a single-core processor. > >> - // > >> - MaxCoresPerPackage = 1; > >> - } > >> - > >> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > >> MaxCoresPerPackage - 1) + 1); > >> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > >> - } > >> - > >> - Location->Thread = ApicId & ~((-1) << ThreadBits); > >> - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); > >> - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); -} > >> - > >> -/** > >> Gets processor information on the requested processor at the instant > this > >> call is made. > >> > >> @param[in] This A pointer to the > >> EFI_SMM_CPU_SERVICE_PROTOCOL instance. > >> @@ -280,7 +161,12 @@ SmmAddProcessor ( > >> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == > >> INVALID_APIC_ID) { > >> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; > >> gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; > >> - SmmGetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- > >>> ProcessorInfo[Index].Location); > >> + GetProcessorLocation ( > >> + (UINT32)ProcessorId, > >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Package, > >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Core, > >> + &gSmmCpuPrivate->ProcessorInfo[Index].Location.Thread > >> + ); > >> > >> *ProcessorNumber = Index; > >> gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; > >> -- > >> 2.6.3.windows.1 > >> > >> > >> Best regards, > >> > >> Mike > >> > >>> -----Original Message----- > >>> From: Leo Duran [mailto:leo.duran@amd.com] > >>> Sent: Friday, October 28, 2016 10:39 AM > >>> To: edk2-devel@lists.01.org > >>> Cc: Gao, Liming <liming.gao@intel.com>; lersek@redhat.com; Fan, Jeff > >>> <jeff.fan@intel.com>; Kinney, Michael D > <michael.d.kinney@intel.com>; > >> Leo Duran > >>> <leo.duran@amd.com> > >>> Subject: [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib > >> library > >>> > >>> 1) Remove SmmGetProcessorLocation() from PiSmmCpuDxeSmm > driver. > >>> 2) Remove ExtractProcessorLocation() from MpInitLib library. > >>> 3) Add GetProcessorLocation() to BaseXApicLib and BaseXApicX2ApicLib. > >>> > >>> Contributed-under: TianoCore Contribution Agreement 1.0 > >>> Signed-off-by: Leo Duran <leo.duran@amd.com> > >>> --- > >>> UefiCpuPkg/Include/Library/LocalApicLib.h | 18 +++ > >>> UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c | 130 > >> +++++++++++++++++++++ > >>> .../BaseXApicX2ApicLib/BaseXApicX2ApicLib.c | 130 > >> +++++++++++++++++++++ > >>> UefiCpuPkg/Library/MpInitLib/MpLib.c | 128 +------------------- > >>> UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c | 121 +--------------- > --- > >>> 5 files changed, 280 insertions(+), 247 deletions(-) > >>> > >>> diff --git a/UefiCpuPkg/Include/Library/LocalApicLib.h > >>> b/UefiCpuPkg/Include/Library/LocalApicLib.h > >>> index cd4e613..4abf64c 100644 > >>> --- a/UefiCpuPkg/Include/Library/LocalApicLib.h > >>> +++ b/UefiCpuPkg/Include/Library/LocalApicLib.h > >>> @@ -21,6 +21,9 @@ > >>> #define LOCAL_APIC_MODE_XAPIC 0x1 ///< xAPIC mode. > >>> #define LOCAL_APIC_MODE_X2APIC 0x2 ///< x2APIC mode. > >>> > >>> +#include <PiPei.h> > >>> +#include <Protocol/MpService.h> > >>> + > >>> /** > >>> Retrieve the base address of local APIC. > >>> > >>> @@ -410,6 +413,21 @@ GetApicMsiValue ( > >>> IN BOOLEAN LevelTriggered, > >>> IN BOOLEAN AssertionLevel > >>> ); > >>> + > >>> +/** > >>> +Get Package ID/Core ID/Thread ID of a processor. > >>> + > >>> +The algorithm assumes the target system has symmetry across physical > >> package > >>> boundaries > >>> +with respect to the number of logical processors per package, number > of > >> cores per > >>> package. > >>> + > >>> +@param InitialApicId Must be the initial APIC ID of the target logical > >> processor. > >>> +@param Location Returns the processor location information. > >>> +**/ > >>> +VOID > >>> +GetProcessorLocation( > >>> + IN UINT32 InitialApicId, > >>> + OUT EFI_CPU_PHYSICAL_LOCATION *Location > >>> +); > >>> > >>> #endif > >>> > >>> diff --git a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > >>> b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > >>> index 8d0fb02..219f99f 100644 > >>> --- a/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > >>> +++ b/UefiCpuPkg/Library/BaseXApicLib/BaseXApicLib.c > >>> @@ -941,3 +941,133 @@ GetApicMsiValue ( > >>> } > >>> return MsiData.Uint64; > >>> } > >>> + > >>> +/** > >>> +Get Package ID/Core ID/Thread ID of a processor. > >>> + > >>> +The algorithm assumes the target system has symmetry across physical > >> package > >>> boundaries > >>> +with respect to the number of logical processors per package, number > of > >> cores per > >>> package. > >>> + > >>> +@param InitialApicId Must be the initial APIC ID of the target logical > >> processor. > >>> +@param Location Returns the processor location information. > >>> +**/ > >>> +VOID > >>> +GetProcessorLocation( > >>> +IN UINT32 InitialApicId, > >>> +OUT EFI_CPU_PHYSICAL_LOCATION *Location > >>> +) > >>> +{ > >>> + BOOLEAN TopologyLeafSupported; > >>> + UINTN ThreadBits; > >>> + UINTN CoreBits; > >>> + CPUID_VERSION_INFO_EBX VersionInfoEbx; > >>> + CPUID_VERSION_INFO_EDX VersionInfoEdx; > >>> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > >>> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > >>> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > >>> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > >>> + UINT32 MaxCpuIdIndex; > >>> + UINT32 SubIndex; > >>> + UINTN LevelType; > >>> + UINT32 MaxLogicProcessorsPerPackage; > >>> + UINT32 MaxCoresPerPackage; > >>> + > >>> + // > >>> + // Check if the processor is capable of supporting more than one > >> logical > >>> processor. > >>> + // > >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, > >> &VersionInfoEdx.Uint32); > >>> + if (VersionInfoEdx.Bits.HTT == 0) { > >>> + Location->Thread = 0; > >>> + Location->Core = 0; > >>> + Location->Package = 0; > >>> + return; > >>> + } > >>> + > >>> + ThreadBits = 0; > >>> + CoreBits = 0; > >>> + > >>> + // > >>> + // Assume three-level mapping of APIC ID: Package:Core:SMT. > >>> + // > >>> + > >>> + TopologyLeafSupported = FALSE; > >>> + // > >>> + // Get the max index of basic CPUID > >>> + // > >>> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, > >> NULL); > >>> + > >>> + // > >>> + // If the extended topology enumeration leaf is available, it > >>> + // is the preferred mechanism for enumerating topology. > >>> + // > >>> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > >>> + AsmCpuidEx( > >>> + CPUID_EXTENDED_TOPOLOGY, > >>> + 0, > >>> + &ExtendedTopologyEax.Uint32, > >>> + &ExtendedTopologyEbx.Uint32, > >>> + &ExtendedTopologyEcx.Uint32, > >>> + NULL > >>> + ); > >>> + // > >>> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and > >> maximum input value for > >>> + // basic CPUID information is greater than 0BH, then > >> CPUID.0BH leaf is not > >>> + // supported on that processor. > >>> + // > >>> + if (ExtendedTopologyEbx.Uint32 != 0) { > >>> + TopologyLeafSupported = TRUE; > >>> + > >>> + // > >>> + // Sub-leaf index 0 (ECX= 0 as input) provides > >> enumeration parameters to > >>> extract > >>> + // the SMT sub-field of x2APIC ID. > >>> + // > >>> + LevelType = ExtendedTopologyEcx.Bits.LevelType; > >>> + ASSERT(LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > >>> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > >>> + > >>> + // > >>> + // Software must not assume any "level type" > >> encoding > >>> + // value to be related to any sub-leaf index, except > >> sub-leaf 0. > >>> + // > >>> + SubIndex = 1; > >>> + do { > >>> + AsmCpuidEx( > >>> + CPUID_EXTENDED_TOPOLOGY, > >>> + SubIndex, > >>> + &ExtendedTopologyEax.Uint32, > >>> + NULL, > >>> + &ExtendedTopologyEcx.Uint32, > >>> + NULL > >>> + ); > >>> + LevelType = > >> ExtendedTopologyEcx.Bits.LevelType; > >>> + if (LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > >>> + CoreBits = > >> ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > >>> + break; > >>> + } > >>> + SubIndex++; > >>> + } while (LevelType != > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > >>> + } > >>> + } > >>> + > >>> + if (!TopologyLeafSupported) { > >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, > >> &VersionInfoEbx.Uint32, NULL, NULL); > >>> + MaxLogicProcessorsPerPackage = > >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > >>> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > >>> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, > >> &CacheParamsEax.Uint32, NULL, NULL, > >>> NULL); > >>> + MaxCoresPerPackage = > >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > >>> + } > >>> + else { > >>> + // > >>> + // Must be a single-core processor. > >>> + // > >>> + MaxCoresPerPackage = 1; > >>> + } > >>> + > >>> + ThreadBits = > >> (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > >>> MaxCoresPerPackage - 1) + 1); > >>> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + > >> 1); > >>> + } > >>> + > >>> + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > >>> + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > >>> + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > >>> +} > >>> diff --git > a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > >>> b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > >>> index 4c42696..16395a3 100644 > >>> --- a/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > >>> +++ b/UefiCpuPkg/Library/BaseXApicX2ApicLib/BaseXApicX2ApicLib.c > >>> @@ -1036,3 +1036,133 @@ GetApicMsiValue ( > >>> } > >>> return MsiData.Uint64; > >>> } > >>> + > >>> +/** > >>> +Get Package ID/Core ID/Thread ID of a processor. > >>> + > >>> +The algorithm assumes the target system has symmetry across physical > >> package > >>> boundaries > >>> +with respect to the number of logical processors per package, number > of > >> cores per > >>> package. > >>> + > >>> +@param InitialApicId Must be the initial APIC ID of the target logical > >> processor. > >>> +@param Location Returns the processor location information. > >>> +**/ > >>> +VOID > >>> +GetProcessorLocation( > >>> +IN UINT32 InitialApicId, > >>> +OUT EFI_CPU_PHYSICAL_LOCATION *Location > >>> +) > >>> +{ > >>> + BOOLEAN TopologyLeafSupported; > >>> + UINTN ThreadBits; > >>> + UINTN CoreBits; > >>> + CPUID_VERSION_INFO_EBX VersionInfoEbx; > >>> + CPUID_VERSION_INFO_EDX VersionInfoEdx; > >>> + CPUID_CACHE_PARAMS_EAX CacheParamsEax; > >>> + CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > >>> + CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > >>> + CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > >>> + UINT32 MaxCpuIdIndex; > >>> + UINT32 SubIndex; > >>> + UINTN LevelType; > >>> + UINT32 MaxLogicProcessorsPerPackage; > >>> + UINT32 MaxCoresPerPackage; > >>> + > >>> + // > >>> + // Check if the processor is capable of supporting more than one > >> logical > >>> processor. > >>> + // > >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, NULL, NULL, > >> &VersionInfoEdx.Uint32); > >>> + if (VersionInfoEdx.Bits.HTT == 0) { > >>> + Location->Thread = 0; > >>> + Location->Core = 0; > >>> + Location->Package = 0; > >>> + return; > >>> + } > >>> + > >>> + ThreadBits = 0; > >>> + CoreBits = 0; > >>> + > >>> + // > >>> + // Assume three-level mapping of APIC ID: Package:Core:SMT. > >>> + // > >>> + > >>> + TopologyLeafSupported = FALSE; > >>> + // > >>> + // Get the max index of basic CPUID > >>> + // > >>> + AsmCpuid(CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, > >> NULL); > >>> + > >>> + // > >>> + // If the extended topology enumeration leaf is available, it > >>> + // is the preferred mechanism for enumerating topology. > >>> + // > >>> + if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > >>> + AsmCpuidEx( > >>> + CPUID_EXTENDED_TOPOLOGY, > >>> + 0, > >>> + &ExtendedTopologyEax.Uint32, > >>> + &ExtendedTopologyEbx.Uint32, > >>> + &ExtendedTopologyEcx.Uint32, > >>> + NULL > >>> + ); > >>> + // > >>> + // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and > >> maximum input value for > >>> + // basic CPUID information is greater than 0BH, then > >> CPUID.0BH leaf is not > >>> + // supported on that processor. > >>> + // > >>> + if (ExtendedTopologyEbx.Uint32 != 0) { > >>> + TopologyLeafSupported = TRUE; > >>> + > >>> + // > >>> + // Sub-leaf index 0 (ECX= 0 as input) provides > >> enumeration parameters to > >>> extract > >>> + // the SMT sub-field of x2APIC ID. > >>> + // > >>> + LevelType = ExtendedTopologyEcx.Bits.LevelType; > >>> + ASSERT(LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > >>> + ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > >>> + > >>> + // > >>> + // Software must not assume any "level type" > >> encoding > >>> + // value to be related to any sub-leaf index, except > >> sub-leaf 0. > >>> + // > >>> + SubIndex = 1; > >>> + do { > >>> + AsmCpuidEx( > >>> + CPUID_EXTENDED_TOPOLOGY, > >>> + SubIndex, > >>> + &ExtendedTopologyEax.Uint32, > >>> + NULL, > >>> + &ExtendedTopologyEcx.Uint32, > >>> + NULL > >>> + ); > >>> + LevelType = > >> ExtendedTopologyEcx.Bits.LevelType; > >>> + if (LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) { > >>> + CoreBits = > >> ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > >>> + break; > >>> + } > >>> + SubIndex++; > >>> + } while (LevelType != > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > >>> + } > >>> + } > >>> + > >>> + if (!TopologyLeafSupported) { > >>> + AsmCpuid(CPUID_VERSION_INFO, NULL, > >> &VersionInfoEbx.Uint32, NULL, NULL); > >>> + MaxLogicProcessorsPerPackage = > >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > >>> + if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > >>> + AsmCpuidEx(CPUID_CACHE_PARAMS, 0, > >> &CacheParamsEax.Uint32, NULL, NULL, > >>> NULL); > >>> + MaxCoresPerPackage = > >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > >>> + } > >>> + else { > >>> + // > >>> + // Must be a single-core processor. > >>> + // > >>> + MaxCoresPerPackage = 1; > >>> + } > >>> + > >>> + ThreadBits = > >> (UINTN)(HighBitSet32(MaxLogicProcessorsPerPackage / > >>> MaxCoresPerPackage - 1) + 1); > >>> + CoreBits = (UINTN)(HighBitSet32(MaxCoresPerPackage - 1) + > >> 1); > >>> + } > >>> + > >>> + Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > >>> + Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > >>> + Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > >>> +} > >>> diff --git a/UefiCpuPkg/Library/MpInitLib/MpLib.c > >>> b/UefiCpuPkg/Library/MpInitLib/MpLib.c > >>> index c3fe721..f3380bb 100644 > >>> --- a/UefiCpuPkg/Library/MpInitLib/MpLib.c > >>> +++ b/UefiCpuPkg/Library/MpInitLib/MpLib.c > >>> @@ -58,132 +58,6 @@ IsBspExecuteDisableEnabled ( > >>> } > >>> > >>> /** > >>> - Get CPU Package/Core/Thread location information. > >>> - > >>> - @param[in] InitialApicId CPU APIC ID > >>> - @param[out] Location Pointer to CPU location information > >>> -**/ > >>> -VOID > >>> -ExtractProcessorLocation ( > >>> - IN UINT32 InitialApicId, > >>> - OUT EFI_CPU_PHYSICAL_LOCATION *Location > >>> - ) > >>> -{ > >>> - BOOLEAN TopologyLeafSupported; > >>> - UINTN ThreadBits; > >>> - UINTN CoreBits; > >>> - CPUID_VERSION_INFO_EBX VersionInfoEbx; > >>> - CPUID_VERSION_INFO_EDX VersionInfoEdx; > >>> - CPUID_CACHE_PARAMS_EAX CacheParamsEax; > >>> - CPUID_EXTENDED_TOPOLOGY_EAX ExtendedTopologyEax; > >>> - CPUID_EXTENDED_TOPOLOGY_EBX ExtendedTopologyEbx; > >>> - CPUID_EXTENDED_TOPOLOGY_ECX ExtendedTopologyEcx; > >>> - UINT32 MaxCpuIdIndex; > >>> - UINT32 SubIndex; > >>> - UINTN LevelType; > >>> - UINT32 MaxLogicProcessorsPerPackage; > >>> - UINT32 MaxCoresPerPackage; > >>> - > >>> - // > >>> - // Check if the processor is capable of supporting more than one logical > >> processor. > >>> - // > >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, > >> &VersionInfoEdx.Uint32); > >>> - if (VersionInfoEdx.Bits.HTT == 0) { > >>> - Location->Thread = 0; > >>> - Location->Core = 0; > >>> - Location->Package = 0; > >>> - return; > >>> - } > >>> - > >>> - ThreadBits = 0; > >>> - CoreBits = 0; > >>> - > >>> - // > >>> - // Assume three-level mapping of APIC ID: Package:Core:SMT. > >>> - // > >>> - > >>> - TopologyLeafSupported = FALSE; > >>> - // > >>> - // Get the max index of basic CPUID > >>> - // > >>> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > >>> - > >>> - // > >>> - // If the extended topology enumeration leaf is available, it > >>> - // is the preferred mechanism for enumerating topology. > >>> - // > >>> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > >>> - AsmCpuidEx ( > >>> - CPUID_EXTENDED_TOPOLOGY, > >>> - 0, > >>> - &ExtendedTopologyEax.Uint32, > >>> - &ExtendedTopologyEbx.Uint32, > >>> - &ExtendedTopologyEcx.Uint32, > >>> - NULL > >>> - ); > >>> - // > >>> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > >> value for > >>> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is > >> not > >>> - // supported on that processor. > >>> - // > >>> - if (ExtendedTopologyEbx.Uint32 != 0) { > >>> - TopologyLeafSupported = TRUE; > >>> - > >>> - // > >>> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration > parameters > >> to extract > >>> - // the SMT sub-field of x2APIC ID. > >>> - // > >>> - LevelType = ExtendedTopologyEcx.Bits.LevelType; > >>> - ASSERT (LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > >>> - ThreadBits = ExtendedTopologyEax.Bits.ApicIdShift; > >>> - > >>> - // > >>> - // Software must not assume any "level type" encoding > >>> - // value to be related to any sub-leaf index, except sub-leaf 0. > >>> - // > >>> - SubIndex = 1; > >>> - do { > >>> - AsmCpuidEx ( > >>> - CPUID_EXTENDED_TOPOLOGY, > >>> - SubIndex, > >>> - &ExtendedTopologyEax.Uint32, > >>> - NULL, > >>> - &ExtendedTopologyEcx.Uint32, > >>> - NULL > >>> - ); > >>> - LevelType = ExtendedTopologyEcx.Bits.LevelType; > >>> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) > { > >>> - CoreBits = ExtendedTopologyEax.Bits.ApicIdShift - ThreadBits; > >>> - break; > >>> - } > >>> - SubIndex++; > >>> - } while (LevelType != > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > >>> - } > >>> - } > >>> - > >>> - if (!TopologyLeafSupported) { > >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, &VersionInfoEbx.Uint32, > >> NULL, NULL); > >>> - MaxLogicProcessorsPerPackage = > >>> VersionInfoEbx.Bits.MaximumAddressableIdsForLogicalProcessors; > >>> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > >>> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &CacheParamsEax.Uint32, > >> NULL, NULL, NULL); > >>> - MaxCoresPerPackage = > >>> CacheParamsEax.Bits.MaximumAddressableIdsForLogicalProcessors + 1; > >>> - } else { > >>> - // > >>> - // Must be a single-core processor. > >>> - // > >>> - MaxCoresPerPackage = 1; > >>> - } > >>> - > >>> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > >>> MaxCoresPerPackage - 1) + 1); > >>> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > >>> - } > >>> - > >>> - Location->Thread = InitialApicId & ((1 << ThreadBits) - 1); > >>> - Location->Core = (InitialApicId >> ThreadBits) & ((1 << CoreBits) - 1); > >>> - Location->Package = (InitialApicId >> (ThreadBits + CoreBits)); > >>> -} > >>> - > >>> -/** > >>> Worker function for SwitchBSP(). > >>> > >>> Worker function for SwitchBSP(), assigned to the AP which is intended > >>> @@ -1451,7 +1325,7 @@ MpInitLibGetProcessorInfo ( > >>> // > >>> // Get processor location information > >>> // > >>> - ExtractProcessorLocation (CpuMpData- > >>> CpuData[ProcessorNumber].ApicId, > >>> &ProcessorInfoBuffer->Location); > >>> + GetProcessorLocation (CpuMpData- > >>> CpuData[ProcessorNumber].ApicId, > >>> &ProcessorInfoBuffer->Location); > >>> > >>> if (HealthData != NULL) { > >>> HealthData->Uint32 = CpuMpData- > >CpuData[ProcessorNumber].Health; > >>> diff --git a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > >>> b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > >>> index 40f2a17..67cd0a0 100644 > >>> --- a/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > >>> +++ b/UefiCpuPkg/PiSmmCpuDxeSmm/CpuService.c > >>> @@ -27,125 +27,6 @@ EFI_SMM_CPU_SERVICE_PROTOCOL > >> mSmmCpuService = { > >>> }; > >>> > >>> /** > >>> - Get Package ID/Core ID/Thread ID of a processor. > >>> - > >>> - APIC ID must be an initial APIC ID. > >>> - > >>> - The algorithm below assumes the target system has symmetry across > >> physical package > >>> boundaries > >>> - with respect to the number of logical processors per package, number > of > >> cores per > >>> package. > >>> - > >>> - @param ApicId APIC ID of the target logical processor. > >>> - @param Location Returns the processor location information. > >>> -**/ > >>> -VOID > >>> -SmmGetProcessorLocation ( > >>> - IN UINT32 ApicId, > >>> - OUT EFI_CPU_PHYSICAL_LOCATION *Location > >>> - ) > >>> -{ > >>> - UINTN ThreadBits; > >>> - UINTN CoreBits; > >>> - UINT32 RegEax; > >>> - UINT32 RegEbx; > >>> - UINT32 RegEcx; > >>> - UINT32 RegEdx; > >>> - UINT32 MaxCpuIdIndex; > >>> - UINT32 SubIndex; > >>> - UINTN LevelType; > >>> - UINT32 MaxLogicProcessorsPerPackage; > >>> - UINT32 MaxCoresPerPackage; > >>> - BOOLEAN TopologyLeafSupported; > >>> - > >>> - ASSERT (Location != NULL); > >>> - > >>> - ThreadBits = 0; > >>> - CoreBits = 0; > >>> - TopologyLeafSupported = FALSE; > >>> - > >>> - // > >>> - // Check if the processor is capable of supporting more than one logical > >> processor. > >>> - // > >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, NULL, NULL, &RegEdx); > >>> - ASSERT ((RegEdx & BIT28) != 0); > >>> - > >>> - // > >>> - // Assume three-level mapping of APIC ID: Package:Core:SMT. > >>> - // > >>> - > >>> - // > >>> - // Get the max index of basic CPUID > >>> - // > >>> - AsmCpuid (CPUID_SIGNATURE, &MaxCpuIdIndex, NULL, NULL, NULL); > >>> - > >>> - // > >>> - // If the extended topology enumeration leaf is available, it > >>> - // is the preferred mechanism for enumerating topology. > >>> - // > >>> - if (MaxCpuIdIndex >= CPUID_EXTENDED_TOPOLOGY) { > >>> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, 0, &RegEax, &RegEbx, > >> &RegEcx, NULL); > >>> - // > >>> - // If CPUID.(EAX=0BH, ECX=0H):EBX returns zero and maximum input > >> value for > >>> - // basic CPUID information is greater than 0BH, then CPUID.0BH leaf is > >> not > >>> - // supported on that processor. > >>> - // > >>> - if ((RegEbx & 0xffff) != 0) { > >>> - TopologyLeafSupported = TRUE; > >>> - > >>> - // > >>> - // Sub-leaf index 0 (ECX= 0 as input) provides enumeration > parameters > >> to extract > >>> - // the SMT sub-field of x2APIC ID. > >>> - // > >>> - LevelType = (RegEcx >> 8) & 0xff; > >>> - ASSERT (LevelType == > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_SMT); > >>> - if ((RegEbx & 0xffff) > 1 ) { > >>> - ThreadBits = RegEax & 0x1f; > >>> - } else { > >>> - // > >>> - // HT is not supported > >>> - // > >>> - ThreadBits = 0; > >>> - } > >>> - > >>> - // > >>> - // Software must not assume any "level type" encoding > >>> - // value to be related to any sub-leaf index, except sub-leaf 0. > >>> - // > >>> - SubIndex = 1; > >>> - do { > >>> - AsmCpuidEx (CPUID_EXTENDED_TOPOLOGY, SubIndex, &RegEax, > >> NULL, &RegEcx, NULL); > >>> - LevelType = (RegEcx >> 8) & 0xff; > >>> - if (LevelType == CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_CORE) > { > >>> - CoreBits = (RegEax & 0x1f) - ThreadBits; > >>> - break; > >>> - } > >>> - SubIndex++; > >>> - } while (LevelType != > >> CPUID_EXTENDED_TOPOLOGY_LEVEL_TYPE_INVALID); > >>> - } > >>> - } > >>> - > >>> - if (!TopologyLeafSupported) { > >>> - AsmCpuid (CPUID_VERSION_INFO, NULL, &RegEbx, NULL, NULL); > >>> - MaxLogicProcessorsPerPackage = (RegEbx >> 16) & 0xff; > >>> - if (MaxCpuIdIndex >= CPUID_CACHE_PARAMS) { > >>> - AsmCpuidEx (CPUID_CACHE_PARAMS, 0, &RegEax, NULL, NULL, > NULL); > >>> - MaxCoresPerPackage = (RegEax >> 26) + 1; > >>> - } else { > >>> - // > >>> - // Must be a single-core processor. > >>> - // > >>> - MaxCoresPerPackage = 1; > >>> - } > >>> - > >>> - ThreadBits = (UINTN) (HighBitSet32 (MaxLogicProcessorsPerPackage / > >>> MaxCoresPerPackage - 1) + 1); > >>> - CoreBits = (UINTN) (HighBitSet32 (MaxCoresPerPackage - 1) + 1); > >>> - } > >>> - > >>> - Location->Thread = ApicId & ~((-1) << ThreadBits); > >>> - Location->Core = (ApicId >> ThreadBits) & ~((-1) << CoreBits); > >>> - Location->Package = (ApicId >> (ThreadBits+ CoreBits)); > >>> -} > >>> - > >>> -/** > >>> Gets processor information on the requested processor at the instant > this > >> call is > >>> made. > >>> > >>> @param[in] This A pointer to the > >> EFI_SMM_CPU_SERVICE_PROTOCOL > >>> instance. > >>> @@ -280,7 +161,7 @@ SmmAddProcessor ( > >>> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId == > >> INVALID_APIC_ID) { > >>> gSmmCpuPrivate->ProcessorInfo[Index].ProcessorId = ProcessorId; > >>> gSmmCpuPrivate->ProcessorInfo[Index].StatusFlag = 0; > >>> - SmmGetProcessorLocation ((UINT32)ProcessorId, > &gSmmCpuPrivate- > >>>> ProcessorInfo[Index].Location); > >>> + GetProcessorLocation ((UINT32)ProcessorId, &gSmmCpuPrivate- > >>>> ProcessorInfo[Index].Location); > >>> > >>> *ProcessorNumber = Index; > >>> gSmmCpuPrivate->Operation[Index] = SmmCpuAdd; > >>> -- > >>> 1.9.1 > > > > _______________________________________________ > edk2-devel mailing list > edk2-devel@lists.01.org > https://lists.01.org/mailman/listinfo/edk2-devel ^ permalink raw reply [flat|nested] 7+ messages in thread
end of thread, other threads:[~2016-10-31 16:45 UTC | newest] Thread overview: 7+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2016-10-28 17:38 [v3] UefiCpuPkg: Move GetProcessorLocation() to LocalApicLib library Leo Duran 2016-10-28 18:42 ` Laszlo Ersek 2016-10-28 21:12 ` Kinney, Michael D 2016-10-28 21:17 ` Duran, Leo 2016-10-28 21:48 ` Laszlo Ersek 2016-10-31 6:40 ` Fan, Jeff 2016-10-31 16:45 ` Duran, Leo
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox