Re: [RFC PATCH] x86/topo: Unify srat_detect_node among amd/intel/hygon
From: Nikola Z. Ivanov
Date: Wed Apr 01 2026 - 11:47:43 EST
On 3/30/26 7:57 AM, K Prateek Nayak wrote:
Hello Nikola,
On 3/29/2026 5:38 PM, Nikola Z. Ivanov wrote:
This change is provoked by an observed warning afterYou can also say:
commit 717b64d58cff ("x86/topo: Replace x86_has_numa_in_package")
when faking numa nodes on intel.
For example:
qemu-system-x86_64 \
-kernel arch/x86/boot/bzImage \
-append "console=ttyS0 root=/dev/sda debug numa=fake=2" \
-hda $IMAGES/unstable.img \
-cpu qemu64,vendor=GenuineIntel \
-nographic \
-m 2G \
-smp 2 \
-smp 2,sockets=2
and that fixes the warning but that is not a valid solution? Why?
Will trigger:To me, that looks like a broken topology from a virtualization use case
[ 0.066755][ T0] ------------[ cut here ]------------
[ 0.066755][ T0] WARNING: arch/x86/kernel/smpboot.c:698 at
set_cpu_sibling_map+0xe41/0x1f90, CPU#1: swapper/1/0
[ 0.066755][ T0] Call Trace:
[ 0.066755][ T0] <TASK>
[ 0.066755][ T0] ap_starting+0x9e/0x140
[ 0.066755][ T0] ? __pfx_ap_starting+0x10/0x10
[ 0.066755][ T0] ? fpu__init_cpu_xstate+0x5c/0x320
[ 0.066755][ T0] start_secondary+0x66/0x110
[ 0.066755][ T0] common_startup_64+0x13e/0x147
[ 0.066755][ T0] </TASK>
smpboot.c suggests that the topology is invalid as
the CPUs are in the same package but different nodes.
and the user can easily go fix their QEMU cmdlline if they care. I'm
pretty sure folks using NUMA emulation in production know what they are
doing.
Fix this by unifying the srat_detect_node functionAs far as the AMD, Hygon unification goes, I don't mind that but
among amd/intel/hygon and taking the amd/hygon approach
of falling back to LLC when SRAT is not detected.
someone has to confirm if nearby_node() holds for all APICID
distribution on Intel.
Place the function inside common.c and expose it in topology.hThere is no need to make it visible out of arch/x86/kernel/cpu/
Perhaps arch/x86/kernel/cpu/cpu/cpu.h?
Yes, I have made a pretty bad mistake here
also because topology.h is not x86 specific
and will cause build warnings.
The hygon code is already basically identical to amdSo prior to you patch, If I launch:
except for the way it obtains the LLC ID.
We can reuse that from the hygon code since we
already have the struct cpuinfo_x86 passed to us.
Signed-off-by: Nikola Z. Ivanov <zlatistiv@xxxxxxxxx>
---
This is marked RFC as I lack the context for the reason
why the intel code looks the way it does. I can see
it went through a few changes in the 2008-2010 year range,
which makes be believe that the comment regarding
"not doing AMD heuristics for now" is long overdue.
-smp 4,sockets=2,cores=2
and "numa=fake=2", the srat_detect_node() for an Intel VM maps:
CPU#0 -> Node#0
CPU#1 -> Node#1
CPU#2 -> Node#0
CPU#3 -> Node#1
This is not necessarily correct either.
Those qemu parameters do not produce
an interleaved topology, but the cpu to node
map will end up interleaved, this ties back to the warning in smpboot.c
This is what it looks like pre-patch:
cpu to node mapping (interleaved):
# ls -l /sys/devices/system/cpu/cpu*/node*
lrwxrwxrwx 1 root root 0 Apr 1 13:57 /sys/devices/system/cpu/cpu0/node0 -> ../../node/node0
lrwxrwxrwx 1 root root 0 Apr 1 13:57 /sys/devices/system/cpu/cpu1/node1 -> ../../node/node1
lrwxrwxrwx 1 root root 0 Apr 1 13:57 /sys/devices/system/cpu/cpu2/node0 -> ../../node/node0
lrwxrwxrwx 1 root root 0 Apr 1 13:57 /sys/devices/system/cpu/cpu3/node1 -> ../../node/node1
#
cpu to socket (not interleaved):
# lscpu -e
CPU NODE SOCKET CORE L1d:L1i:L2:L3 ONLINE
0 0 0 0 0:0:0:0 yes
1 0 0 1 1:1:1:0 yes
2 0 1 2 2:2:2:1 yes
3 0 1 3 3:3:3:1 yes
#
The NODE output of "lscpu -e" is kind of bogus,
as it doesn't look up the cpu to node mapping, but instead
reads the node to cpu map and figures out the reverse,
which caused me to make a lot of false assumptions earlier...
However, this is an unrelated matter and the SOCKET is correct.
The early initialization code first assigns nodes in round-robin fashion,
which is later overridden on AMD by srat_detect_node,
but persists for Intel.
Which resembles Intel baremetal node assignments where the CPUs
are interleaved. After your patch, it does:
CPU#0 -> Node#0
CPU#1 -> Node#0
CPU#2 -> Node#0
CPU#3 -> Node#0
This happens because we take an interesting path in srat_detect_node
when the patch is applied.
The cpuid to llc_id map looks like this:
0 -> 0
1 -> 0
2 -> 2
3 -> 2
Since node 2 does not exist, we enter the if(!node_online(node))
path and our final mapping ends up like this:
0 -> 0
1 -> 0
2 -> 0
3 -> 0
So despite there being 2 LLCs, the Node assignments all go to Node#0
which may have other unintended consequences.
The statement "falling back to LLC when SRAT is not detected." isn't
accurate right? We have 2 LLCs and 2 Nodes but topology bits associate
both LLCs to the same node.
I'm all for unifying the AMD and Hygon's srat_detect_node() but unifying
all three for an obviously broken use-case isn't a good motivation.
I'll let others comment since they are more familiar with the NUMA
emulation bits and maybe all this is acceptable.
Hi Prateek,
Thank you for the feedback!
I have tried to dig a bit deeper
and left my findings in response.
I suppose I will wait to see if someone
chimes in with some more details, if not
I will do as you suggest and try to come up with
a good way to unify amd/hygon and leave intel as is.