NUMA相关配置介绍

Posted 一只独角兽

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了NUMA相关配置介绍相关的知识,希望对你有一定的参考价值。

NUMA相关配置介绍

什么是NUMA

在早期,对于x86架构的计算机,那时的内存控制器还没有整合进CPU,所有内存的访问都需要通过北桥芯片来完成。此时的内存访问如下图所示,被称为UMA(uniform memory access, 一致性内存访问 )。这样的访问对于软件层面来说非常容易实现:总线模型保证了所有的内存访问是一致的,不必考虑由不同内存地址之前的差异。

之后的x86平台经历了一场从“拼频率”到“拼核心数”的转变,越来越多的核心被尽可能地塞进了同一块芯片上,各个核心对于内存带宽的争抢访问成为了瓶颈;此时软件、OS方面对于SMP多核心CPU的支持也愈发成熟;再加上各种商业上的考量,x86平台也顺水推舟的搞了NUMA(Non-uniform memory access, 非一致性内存访问)。在这种架构之下,每个Socket都会有一个独立的内存控制器IMC(integrated memory controllers, 集成内存控制器),分属于不同的socket之内的IMC之间通过QPI link通讯。

然后就是进一步的架构演进,由于每个socket上都会有多个core进行内存访问,这就会在每个core的内部出现一个类似最早SMP架构相似的内存访问总线,这个总线被称为IMC bus。

于是,很明显的,在这种架构之下,两个socket各自管理1/2的内存插槽,如果要访问不属于本socket的内存则必须通过QPI link。也就是说内存的访问出现了本地/远程(local/remote)的概念,内存的延时是会有显著的区别的。这也是为什么在NUMA架构下有些应用性能反而更差的原因。

回到当前世面上的CPU,工程上的实现其实更加复杂了。以来看,两个Socket之之间通过各自的一条9.6GT/s的QPI link互访。而每个Socket事实上有2个内存控制器。双通道的缘故,每个控制器又有两个内存通道(channel),每个通道最多支持3根内存条(DIMM)。理论上最大单socket支持76.8GB/s的内存带宽,而两个QPI link,每个QPI link有9.6GT/s的速率(~57.6GB/s)事实上QPI link已经出现瓶颈了。

内核的NUMA的默认行为

Linux内核中/ , //文件定义了NUMA的数据结构和操作方式。在一个启用了NUMA支持的Linux中,Kernel不会将任务内存从一个NUMA node搬迁到另一个NUMA node。

一个进程一旦被启用,它所在的NUMA node就不会被迁移,为了尽可能的优化性能,在正常的调度之中,CPU的core也会尽可能的使用可以local访问的本地core,在进程的整个生命周期之中,NUMA node保持不变。

一旦当某个NUMA node的负载超出了另一个node一个阈值(默认25%),则认为需要在此node上减少负载,不同的NUMA结构和不同的负载状况,系统见给予一个延时任务的迁移——类似于漏杯算法。在这种情况下将会产生内存的remote访问。

NUMA node之间有不同的拓扑结构,各个 node 之间的访问会有一个距离(node distances)的概念,如numactl -H命令的结果有这样的描述:node distances:
node 0 1 2 3
0: 10 11 21 21
1: 11 10 21 21
2: 21 21 10 11
3: 21 21 11 10

可以看出:0 node 到0 node之间距离为10,这肯定的最近的距离,不提。0-1之间的距离远小于2或3的距离。这种距离方便系统在较复杂的情况下选择最合适的NUMA设定。
NUMA相关配置
查看 NUMA node 信息

点击(此处)折叠或打开

numactl --hardware

    # NUMA 关闭时:
    # available: 1 nodes (0) # node 列表
    # node 0 cpus: 0 1 .. # node0 核列表
    # node 0 size: 65523 MB # node0 总内存大小
    # node 0 free: 51796 MB # node0 空闲内存大小
    # node distances: # node 之间 distances 大小
    # node 0 # distances 会由矩阵来表示
    # 0: 10

查看 NUMA 绑定信息

点击(此处)折叠或打开

numactl --show

    # NUMA 关闭时:
    # policy: default
    # preferred node: current
    # physcpubind: 0 1 ... # 核绑定
    # cpubind: 0 # CPU 绑定
    # nodebind: 0 # node 绑定
    # membind: 0 # 内存绑定

查看 NUMA 统计信息

点击(此处)折叠或打开

numastat 等同于 cat /sys/devices/system/node/node[0…n]/numastat。
numastat
# NUMA 关闭时:
# node0
# numa_hit 5571187 //打算在本节点上分配内存,最后从本节点分配的次数
# numa_miss 0 //打算在本节点分配内存,最后却从其他节点分配的次数
# numa_foreign 0 //打算在其他节点分配内存,最后却从这个节点分配的次数
# interleave_hit 53465 //采用interleave策略最后从本节点分配的次数
# local_node 5571187//本节点上的进程,在本节点上分配的次数
# other_node 0 //其他节点进程,在本节点上分配的次数

NUMA 的 内存 分配策略

选项:

–localalloc, -l

规定进程从当前 node 上请求分配内存。

–membind=nodes, -m nodes

规定进程只能从指定的 nodes 上请求分配内存。

–preferred=node

指定了 一个 推荐的 node 来获取内存,如果失败,则尝试别的 node。

–interleave=nodes, -i nodes

规定进程从指定的 nodes 上,以 round robin 算法 交织地 请求分配内存。
NUMA的分配策略例子

点击(此处)折叠或打开

# Unset default cpuset awareness.
numactl --all

# Run $process on node 0 with memory allocated on node 0 and 1.
numactl --cpunodebind=0 --membind=0,1 $process $process_arguments

# Run $process on cpus 0-4 and 8-12 of the current cpuset.
numactl --physcpubind=+0-4,8-12 $process $process_arguments

# Run $process with its memory interleaved on all CPUs
numactl --interleave=all $process $process_arguments

# Run process as above, but with an option (-l) that would be confused with a numactl option.
numactl --cpunodebind=0 --membind=0,1 -- $process -l

# Run $network-server on the node of network device eth0 with its memory also in the same node.
numactl --cpunodebind=netdev:eth0 --membind=netdev:eth0 $network-server

# Set preferred node 1 and show the resulting state.
numactl --preferred=1 numactl --show

# Interleave all of the sysv shared memory region specified by /tmp/shmkey over all nodes.
numactl --interleave=all --shm /tmp/shmkey

# Place a tmpfs file on 2 nodes.
numactl --membind=2 dd if=/dev/zero of=/dev/shm/A bs=1M count=1024

numactl --membind=3 dd if=/dev/zero of=/dev/shm/A seek=1024 bs=1M count=1024

# Reset the policy for the shared memory file file to the default localalloc policy.
numactl --localalloc /dev/shm/file

查看设备NUMA

事实上,在PCIe channel上也是有NUMA亲和性的。比如:查看网卡enp0s20f0u5的NUMA,用到了netdev:<设备名称>这种方式。

点击(此处)折叠或打开

[root@local ~]# numactl --prefer netdev:enp0s20f0u5 --show
policy: preferred
preferred node: 0
physcpubind: 0
cpubind: 0
nodebind: 0
membind: 0 1 2 3

或者一个PCI address 为00:17的SATA控制器,用到了pci:

点击(此处)折叠或打开

[root@local~]# numactl --prefer pci:00:17 --show
policy: preferred
preferred node: 0
physcpubind: 0
cpubind: 0
nodebind: 0
membind: 0 1 2 3

还有block/ip/file的方式查看NUMA affinity,这里也就不再累述了。

NUMA的开启和关闭

NUMA可以在Bios和OS两个层面开关,其实关于两个层面开关的区别网上并没有找到详细资料。有一种说法是:BIOS和OS的关闭NUMA在interleave的粒度上有区别,BIOS应该是在cache line(64B)的粒度,而OS利用内核页表所以是页(4kB)的粒度。效果上BIOS表现应该更稳定一些,但OS的配置相对更方便。
BIOS 层 的 NUMA 设置

  1. 查看 BIOS 层是否开启 NUMA

    grep -i numa /var/log/dmesg

    #1. 如果为 “No NUMA configuration found”

    #则说明 NUMA 为 disable。

    #2. 如果不是 “No NUMA configuration found”

    #则说明 NUMA 为 enable。

  2. 修改 BIOS interleave

注意,由于 BIOS 种类繁多,请以实际情况为准。

参数路径:

BIOS: interleave

设定值:

Disable                # interleave 关闭,开启 NUMA。

Enable                # interleave 开启,关闭 NUMA。

OS 层 的 NUMA 设置

  1. 查询 当前内核启动参数

    cat /porc/cmdline

  2. 修改 内核启动参数

    vim /etc/grub.conf

    修改 kernel 或 linuxe 或 linuxefi 行

    #添加 numa=off,关闭 NUMA。

    #去除 numa=off,开启 NUMA。

    例子:

    linuxefi /vmlinuz-3.10.0-123.el7.x86_64 … numa=off

以上是关于NUMA相关配置介绍的主要内容,如果未能解决你的问题,请参考以下文章

NUMA相关配置介绍

ceph 14 后 ceph adm支持自动配置numa

OpenStack中的CPU绑核、NUMA亲和、大页内存

Linux 操作系统原理 — NUMA 架构中的多线程调度开销与性能优化

如何在 Windows 上将 c++11 线程关联设置为 NUMA 节点?

cpu亲和性绑定