Windows设置线程关联掩码无效

Windows SetThreadAffinityMask has no effect

本文关键字:掩码 无效 关联 线程 设置 Windows      更新时间:2023-10-16

我编写了一个小型测试程序,在该程序中,我尝试使用Windows API调用SetThreadAffinityMask将线程锁定到单个NUMA节点。我使用GetNumaNodeProcessorMask API调用检索节点的CPU位掩码,然后将该位掩码与GetCurrentThread返回的线程句柄一起传递给SetThreadAffinityMask。这是我代码的一个大大简化的版本:

// Inside a function called from a boost::thread
unsigned long long nodeMask = 0;
GetNumaNodeProcessorMask(1, &nodeMask);
HANDLE thread = GetCurrentThread();
SetThreadAffinityMask(thread, nodeMask);
DoWork(); // make-work function

当然,我检查API调用是否在我的代码中返回0,并且我还打印出了NUMA节点掩码,这正是我所期望的。我也遵循了其他地方给出的建议,打印出了第二次对SetThreadAffinityMask的相同调用返回的掩码,它与节点掩码匹配。

然而,当DoWork函数执行时,通过观察资源监视器,工作在所有核心之间进行了拆分,而不是只在表面上绑定到那些核心。在使用SetThreadAffinityMask时,我是否错过了任何失误?我运行的是Windows 7 Professional 64位,DoWork函数包含一个与OpenMP并行的循环,它对三个非常大的数组的元素执行操作(这些数组组合在一起仍然能够容纳在节点中)。

编辑:为了扩展David Schwartz给出的答案,在Windows上,任何使用OpenMP生成的线程都不会继承生成它们的线程的亲和力。问题在于此,而不是SetThreadAffinityMask。

是否确认其关联掩码的特定线程正在另一个numa节点的核心上运行?否则,它会按预期工作。您正在一个线程上设置处理器掩码,然后观察线程组的行为。