我在基于 Intel Core2 的 Win2k3 机器上运行我的 NT 服务,我需要在其中迭代所有逻辑 CPU(进程关联中的所有位)。为此,我调用 GetProcessAffinityMask() 来检索系统关联掩码,然后依次将进程切换到每个处理器:
DWORD systemMask;
GetProcessAffinityMask( ... &systemMask );
DWORD processorId = 1;
while( systemMask != 0 ) {
SetProcessAffinityMask(... processorId );
Sleep( 1 ); // to be sure that it shifts to that processor
systemMask >>= 1;
processorId <<= 1;
}
在每次迭代中我调用 code from here检索当前处理器 APIC ID。问题是对于不同的处理器,它有时会返回相同的 APIC id。根据文档,系统中的每个处理器都必须具有相同的 ID。
我尝试对此进行调试 - 检查 Windows 是否真的改变了亲和性:
while( systemMask != 0 ) {
SetProcessAffinityMask(... processorId );
Sleep( 1 ); // to be sure that it shifts to that processor
DWORD tempAffinity;
GetProcessAffinityMask( ... &tempAffinity );
// run APIC id detection code here
systemMask >>= 1;
processorId <<= 1;
}
它返回的正是我所期望的亲和掩码,但 APIC id 对于不同的处理器仍然可以相同。
对于这种奇怪的情况有解释吗?
最佳答案
您不能使用 Windows API 调用来确定这一点。
首先,在使用 EAX 调用 CPUID 后,Intel CPU(不确定 AMD 或其他)上的 APCID 最初编码在 EBX 寄存器的 8 位部分(位 24 - 31)又名 EBX[31:24] = 0x1。
考虑以下带有内联汇编的 C/C++ 代码:
unsigned int cpu_eax;
unsigned int cpu_ebx;
unsigned int cpu_ecx;
unsigned int cpu_edx;
unsigned int apic_id;
__asm
{
mov eax, 0x1
cpuid
mov [cpu_eax], eax
mov [cpu_ebx], ebx
mov [cpu_ecx], ecx
mov [cpu_edx], edx
}
apic_id = ( cpu_ebx & 0xFF000000) >> 24;
GetProcessAffinityMask 只是枚举所有可用内核、CPU 中的内核以及每个 CPU 的超线程内核,并返回这些组合的总数。它没有物理 CPU 与 CPU 中的内核与支持超线程的内核之间的概念。至少它没有这样报告。
关于c++ - 用于检测 APIC id 的代码为不同的逻辑处理器返回相同的 id,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1421056/