在 Windows pthread_setaffinity_np 上设置 CPU 关联性错误

问题描述

我有一个多线程项目。我想将 cpu 关联性设置为特定线程。我正在 Windows 上工作并使用 MinGW 编译器。但是当我初始化类型 cpu_set_t 时,出现错误“类型 cpu_set_t 未在此范围内声明”。我不明白原因。我期待你的帮助!谢谢!

#define _GNU_SOURCE
#include <iostream>
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sched.h>
int main() 
{
   int core_id = 1;
   cpu_set_t cpuset; // error here
   cpu_ZERO(&cpuset);
   cpu_SET(core_id,&cpuset);
   pthread_t current_thread = pthread_self();    
   pthread_setaffinity_np(current_thread,sizeof(cpu_set_t),&cpuset);
}

解决方法

Windows 本身不支持 pthreads 标准。 MinGW 项目使用 winpthreads 库作为原生 Windows 线程的包装器。

'_np' 后缀 - 表示不可移植。 pthreads(posix) 标准确实需要这些 *_np 函数。

当前版本的 winpthreads 没有很多在 Linux 等其他操作系统上可以看到的 *_np 函数。

您可以编写自己的函数来更改 Windows 线程关联并从您的线程调用它。 类似的东西:

#include <windows.h>

// Bind thread to CPUs
// Return previous mask value on success,0 on failure
DWORD_PTR BindThreadToCPU(
                            DWORD mask // 1  -  bind to cpu 0
                                       // 4  -  bind to cpu 2
                                       // 15 -  bind to cpu 0,1,2,3
                          )
{
    HANDLE th = GetCurrentThread();
    DWORD_PTR prev_mask = SetThreadAffinityMask(th,mask);
    return prev_mask;
}

注意:线程关联掩码必须是进程关联掩码的子集。

还有来自 Microsoft documentation 的关于 SetThreadAffinityMask() 及其调度程序的有趣引用:

线程关联强制线程在特定的处理器子集上运行。 通常应该避免设置线程关联,因为它会干扰 调度程序具有跨处理器有效调度线程的能力。 这会降低并行处理产生的性能提升。 线程亲和性的适当使用是测试每个处理器。

在现代版本的 Windows 中,SetThreadIdealProcessor() 函数对调度程序更友好,并允许您为线程设置首选 CPU(但没有任何严格的亲和性保证)。