c++ - 用于检测 APIC id 的代码为不同的逻辑处理器返回相同的 id

标签 c++ windows assembly multicore affinity

我在基于 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/

相关文章:

c++ - 在 C++ 中重载显式构造函数

python - 如何在windows中安装pytorch?

c++ - 如何使用 CryptoApi 以使用 AES-128 加密超过一个 block 大小的缓冲区

assembly - 中间语言的最小汇编指令集?

c++ - 我的 park_car 功能有什么问题?

c++ - 删除特定字符后的所有内容

c++ - vector 的 vector push_back

windows - 如何使用 Windows 命令获取剪贴板内容?

assembly - 如何将浮点舍入到最接近的偶数汇编

c++ - C++ 中的怪异性能(VC 2010)