有时候需要将程序绑定到固定 CPU 的某个核心上运行。
我们知道多核 CPU 系统中,进程和线程的运行在哪个核心是由操作系统内核根据一定的调度算法进行调度的。但是实际软件开发过程中,我们出于一些目的,想要进程或者线程稳定运行在某个 CPU 核心上。比如我想测试两种算法的性能,因为服务器上有一些其他的进程干扰,测试的时间总是有波动,此时就需要将测试程序稳定在某个核心上测试。
Linux 中有 CPU 亲和性这种说法(Windows 有没有我不知道,也不关心)。引用一下维基百科的说法:
CPU 亲和性就是绑定某一进程(或线程)到特定的 CPU(或 CPU 集合),从而使得该进程(或线程)只能运行在绑定的 CPU(或 CPU 集合)上。CPU 亲和性利用了这样一个事实:进程上一次运行后的残余信息会保留在 CPU 的状态中(也就是指 CPU 的缓存)。如果下一次仍然将该进程调度到同一个 CPU 上,就能避免缓存未命中等对 CPU 处理性能不利的情况,从而使得进程的运行更加高效。
Linux 系统中每个进程的 task_struct
结构中有一个 cpus_allowed
位掩码,该掩码的位数与系统CPU 核数相同(若 CPU 启用了超线程则为核数乘以 2),通过修改该位掩码可以控制进程可运行在哪些特定 CPU 上。Linux 系统为我们提供了 CPU 亲和性相关的调用函数和一些操作的宏定义。
Linux 提供了一些宏定义来修改掩码,如 CPU_ZERO()
(将位掩码全部设置为 0)和CPU_SET()
(设置特定掩码位为 1)。CPU 的亲合力掩码用一个 cpu_set_t
结构体来表示一个 CPU 集合,下面的几个宏分别对这个掩码集进行操作:
CPU_ZERO()
:清空一个集合。CPU_SET()
与CPU_CLR()
分别对将一个给定的 CPU 号加到一个集合或者从一个集合中去掉。CPU_ISSET()
检查一个 CPU 号是否在这个集合中。
然后还有两个接口帮助我们绑定进程到某个 CPU 或者 CPU 集合上。
sched_setaffinity(pid_t pid, unsigned int cpusetsize, cpu_set_t *mask)
- 该函数设置进程为
pid
的这个进程,让它运行在mask
所设定的 CPU 上。如果pid
的值为 0,则表示指定的是当前进程,使当前进程运行在mask
所设定的那些 CPU 上。第二个参数cpusetsize
是 mask 所指定的数的长度。通常设定为sizeof(cpu_set_t)
。如果当前pid
所指定的进程此时没有运行在mask
所指定的任意一个 CPU 上,则该指定的进程会从其它 CPU 上迁移到mask
的指定的一个 CPU 上运行。
- 该函数设置进程为
sched_getaffinity(pid_t pid, unsigned int cpusetsize, cpu_set_t *mask)
- 该函数获得
pid
所指示的进程的 CPU 位掩码,并将该掩码返回到mask
所指向的结构中。即获得指定pid
当前可以运行在哪些 CPU 上。同样,如果pid
值为0,也表示的是当前进程。
- 该函数获得
因此,一个简易常见的将当前进程绑定到 CPU 某个核心(比如 6,CPU ID 从 0 开始)的示例如下:
1 | cpu_set_t mask; |
Linux 还提供了线程绑定核心的接口:
1 | int threadBindCPU(std::thread &thread, int cpuID) |
查看某个线程绑定的 CPU 核心,需要在线程内部调用:
1 | sched_getcpu() |