NUMA相关配置介绍

7260阅读 0评论2020-08-01 lvyilong316
分类:LINUX

NUMA相关配置介绍

什么是NUMA

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

之后的x86平台经历了一场从“拼频率”到“拼核心数”的转变,越来越多的核心被尽可能地塞进了同一块芯片上,各个核心对于内存带宽的争抢访问成为了瓶颈;此时软件、OS方面对于SMP多核心CPU的支持也愈发成熟;再加上各种商业上的考量,x86平台也顺水推舟的搞了NUMANon-uniform memory access, 非一致性内存访问)。在这种架构之下,每个Socket都会有一个独立的内存控制器IMCintegrated 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/sQPI link互访。而每个Socket事实上有2个内存控制器。双通道的缘故,每个控制器又有两个内存通道(channel),每个通道最多支持3根内存条(DIMM)。理论上最大单socket支持76.8GB/s的内存带宽,而两个QPI link,每个QPI link9.6GT/s的速率(~57.6GB/s)事实上QPI link已经出现瓶颈了。

内核的NUMA的默认行为

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

一个进程一旦被启用,它所在的NUMA node就不会被迁移,为了尽可能的优化性能,在正常的调度之中,CPUcore也会尽可能的使用可以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之间的距离远小于23的距离。这种距离方便系统在较复杂的情况下选择最合适的NUMA设定。

NUMA相关配置

查看 NUMA node 信息

点击(此处)折叠或打开

  1. numactl --hardware

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

查看 NUMA 绑定信息

点击(此处)折叠或打开

  1. numactl --show

  2.     # NUMA 关闭时:
  3.     # policy: default
  4.     # preferred node: current
  5.     # physcpubind: 0 1 ... # 核绑定
  6.     # cpubind: 0 # CPU 绑定
  7.     # nodebind: 0 # node 绑定
  8.     # membind: 0 # 内存绑定

查看 NUMA 统计信息

点击(此处)折叠或打开

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

NUMA 的 内存 分配策略

选项:

localalloc, -l

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

membind=nodes, -m nodes

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

preferred=node

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

interleave=nodes, -i nodes

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

NUMA的分配策略例子   

点击(此处)折叠或打开

  1. # Unset default cpuset awareness.
  2. numactl --all

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

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

  7. # Run ${process} with its memory interleaved on all CPUs
  8. numactl --interleave=all ${process} ${process_arguments}

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

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

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

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

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

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

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


查看设备NUMA

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

点击(此处)折叠或打开

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

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

点击(此处)折叠或打开

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

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

NUMA的开启和关闭

   NUMA可以在BIOSOS两个层面开关,其实关于两个层面开关的区别网上并没有找到详细资料。有一种说法是:BIOSOS的关闭NUMAinterleave的粒度上有区别,BIOS应该是在cache line64B)的粒度,而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

上一篇:IOMMU和VFIO概述
下一篇:“RDMA for VPC”方案之MasQ分析