Commit 1b3108f6 authored by Thomas Gleixner's avatar Thomas Gleixner Committed by Ingo Molnar

x86/cpu/amd: Make the CPUID 0x80000008 parser correct

CPUID 0x80000008 ECX.cpu_nthreads describes the number of threads in the
package. The parser uses this value to initialize the SMT domain level.

That's wrong because cpu_nthreads does not describe the number of threads
per physical core. So this needs to set the CORE domain level and let the
later parsers set the SMT shift if available.

Preset the SMT domain level with the assumption of one thread per core,
which is correct ifrt here are no other CPUID leafs to parse, and propagate
cpu_nthreads and the core level APIC bitwidth into the CORE domain.

Fixes: f7fb3b2d ("x86/cpu: Provide an AMD/HYGON specific topology parser")
Reported-by: default avatar"kernelci.org bot" <bot@kernelci.org>
Reported-by: default avatarLaura Nao <laura.nao@collabora.com>
Signed-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
Tested-by: default avatarLaura Nao <laura.nao@collabora.com>
Link: https://lore.kernel.org/r/20240410194311.535206450@linutronix.de
parent 4f511739
...@@ -29,11 +29,21 @@ static bool parse_8000_0008(struct topo_scan *tscan) ...@@ -29,11 +29,21 @@ static bool parse_8000_0008(struct topo_scan *tscan)
if (!sft) if (!sft)
sft = get_count_order(ecx.cpu_nthreads + 1); sft = get_count_order(ecx.cpu_nthreads + 1);
topology_set_dom(tscan, TOPO_SMT_DOMAIN, sft, ecx.cpu_nthreads + 1); /*
* cpu_nthreads describes the number of threads in the package
* sft is the number of APIC ID bits per package
*
* As the number of actual threads per core is not described in
* this leaf, just set the CORE domain shift and let the later
* parsers set SMT shift. Assume one thread per core by default
* which is correct if there are no other CPUID leafs to parse.
*/
topology_update_dom(tscan, TOPO_SMT_DOMAIN, 0, 1);
topology_set_dom(tscan, TOPO_CORE_DOMAIN, sft, ecx.cpu_nthreads + 1);
return true; return true;
} }
static void store_node(struct topo_scan *tscan, unsigned int nr_nodes, u16 node_id) static void store_node(struct topo_scan *tscan, u16 nr_nodes, u16 node_id)
{ {
/* /*
* Starting with Fam 17h the DIE domain could probably be used to * Starting with Fam 17h the DIE domain could probably be used to
...@@ -73,12 +83,14 @@ static bool parse_8000_001e(struct topo_scan *tscan, bool has_0xb) ...@@ -73,12 +83,14 @@ static bool parse_8000_001e(struct topo_scan *tscan, bool has_0xb)
tscan->c->topo.initial_apicid = leaf.ext_apic_id; tscan->c->topo.initial_apicid = leaf.ext_apic_id;
/* /*
* If leaf 0xb is available, then SMT shift is set already. If not * If leaf 0xb is available, then the domain shifts are set
* take it from ecx.threads_per_core and use topo_update_dom() - * already and nothing to do here.
* topology_set_dom() would propagate and overwrite the already
* propagated CORE level.
*/ */
if (!has_0xb) { if (!has_0xb) {
/*
* Leaf 0x80000008 set the CORE domain shift already.
* Update the SMT domain, but do not propagate it.
*/
unsigned int nthreads = leaf.core_nthreads + 1; unsigned int nthreads = leaf.core_nthreads + 1;
topology_update_dom(tscan, TOPO_SMT_DOMAIN, get_count_order(nthreads), nthreads); topology_update_dom(tscan, TOPO_SMT_DOMAIN, get_count_order(nthreads), nthreads);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment