Skip to content

Latest commit

 

History

History

resnet50v1.5

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 

【DLPerf】TensorFlow 2.x-ResNet50V1.5测评

Overview

本次复现采用了TensorFlow官方仓库中的TensorFlow2.3版的ResNet50 v1.5,目的在于速度测评,同时根据测速结果给出1机、2机器、4机情况下的加速比,评判框架在分布式多机训练情况下的横向拓展能力。

目前,该测试覆盖了FP32精度、FP16混合精度,后续将持续维护,增加更多方式的测评。

Environment

系统

  • 系统:Ubuntu 16.04.4 LTS (GNU/Linux 4.4.0-116-generic x86_64)
  • 显卡:Tesla V100-SXM2-16GB x 8
  • 驱动:NVIDIA 440.33.01
  • CUDA:10.2
  • cuDNN:7.6.5
  • NCCL:2.7.3

框架

  • TensorFlow 2.3.0

Quick Start

框架安装

python -m pip install tensorflow==2.3.0 -i https://mirror.baidu.com/pypi/simple

项目代码

下载官方源码:

git clone https://github.com/tensorflow/models.git && checkout r2.3.0
cd models/official/vision/image_classification
# 根据项目official目录下的 requirements.txt安装依赖(需要指定tensorflow-datasets==3.0.0,否则可能报错)
python -m pip install -r requirements.txt

将本页面scripts文件夹中的.yaml配置文件放入image_classification/configs/examples/resnet/imagenet/下,其余脚本全部放入image_classification/目录下。

NCCL

TensorFlow的分布式训练底层依赖NCCL库,需要从NVIDIA-NCCL官网下载并安装和操作系统、CUDA版本适配的NCCL。

本次测试中安装2.7.3版本的NCCL:

sudo dpkg -i nccl-repo-ubuntu1604-2.7.3-ga-cuda10.2_1-1_amd64.deb
sudo apt update
sudo apt install libnccl2=2.7.3-1+cuda10.2 libnccl-dev=2.7.3-1+cuda10.2

数据集

本次训练使用了ImageNet2012的一个子集:train-00000-of-01024,train-00001-of-01024...train-0000511-of-01024共640512张图片,数据集格式为TFRecord。具体制作方法可参考Tensorflow官方提供的脚本:download_and_preprocess_imagenet.sh

Training

集群中有4台节点:

  • NODE1=10.11.0.2
  • NODE2=10.11.0.3
  • NODE3=10.11.0.4
  • NODE4=10.11.0.5

每个节点有8张显卡,这里默认设置batch_size=128,从1机1卡~4机32卡进行了多组训练,每组测试5次。也可通过脚本及yaml文件里的参数指定其他的batch_size或是否开启混合精度(fp16)训练等。

单机

models/official/vision/image_classification目录下,执行脚本:

bash run_single_node.sh

默认对单机1卡、2卡、4卡、8卡分别做5组测试(fp32,batch_size=128)

混合精度

可以通过在脚本中指定变量:DTYPE="fp16" 来开启混合精度训练,或者直接在运行脚本时传递如下参数:

bash run_single_node.sh 256 fp16

fp16混合精度条件下,默认测试batch_size=256,可以在gpu_fp16.yaml的line:18行指定其他batch_size大小(如320),然后执行:

bash run_single_node.sh 320 fp16

2机16卡

2机、4机等多机情况下,需要在所有机器节点上相同路径准备同样的数据集、以完成分布式训练。需要注意的是,多机时各个节点需要在对应的.yam配置文件中,设置task_index。如:2机FP32时,NODE1节点two_node_gpu.yaml的task_index设置为0,NODE2节点的task_index设置为1。

如2机:NODE1='10.11.0.2' NODE2='10.11.0.3' 的训练,需在NODE1节点models/official/vision/image_classification/目录下,执行脚本:

bash run_two_node.sh

NODE2节点相同目录下,执行同样的脚本bash run_two_node.sh即可运行2机16卡的训练,同样默认进行5次测试。

混合精度

同单机脚本一样,多机的混合精度训练方式类似,如指定batch size 256的fp16混合精度2机训练:

bash run_two_node.sh 256 fp16

fp16混合精度条件下,默认测试的batch_size=256,可以修改two_node_gpu_fp16.yaml的line:20指定其他batch size大小。

4机32卡

流程同上,NODE1至NODE4节点需要分别在.yam配置文件中,设置task_index为0至3,然后在4个机器节点上分别执行:

bash run_multi_node.sh

以运行4机32卡的训练,默认进行5次测试。

混合精度

与上述类似,指定batch size 256的fp16混合精度4机训练:

bash run_multi_node.sh 256 fp16

fp16混合精度条件下,默认测试的batch_size=256,可以修改multi_node_gpu_fp16.yaml的line:21指定其他batch size大小。

Result

吞吐率及加速比

执行以下命令,即可计算各种测试配置下的吞吐率及加速比:

python extract_tensorflow_logs.py --log_dir=logs/tensorflow/resnet50/bz128

输出:

logs/tensorflow/resnet50/bz128/4n8g/rn50_b128_fp32_1.log {1: 9369.06}
logs/tensorflow/resnet50/bz128/4n8g/rn50_b128_fp32_3.log {1: 9369.06, 3: 9418.44}
logs/tensorflow/resnet50/bz128/4n8g/rn50_b128_fp32_5.log {1: 9369.06, 3: 9418.44, 5: 9454.31}
logs/tensorflow/resnet50/bz128/4n8g/rn50_b128_fp32_4.log {1: 9369.06, 3: 9418.44, 5: 9454.31, 4: 9420.23}
logs/tensorflow/resnet50/bz128/4n8g/rn50_b128_fp32_2.log {1: 9369.06, 3: 9418.44, 5: 9454.31, 4: 9420.23, 2: 9402.51}
logs/tensorflow/resnet50/bz128/1n8g/rn50_b128_fp32_1.log {1: 2494.22}
logs/tensorflow/resnet50/bz128/1n8g/rn50_b128_fp32_3.log {1: 2494.22, 3: 2458.74}
logs/tensorflow/resnet50/bz128/1n8g/rn50_b128_fp32_5.log {1: 2494.22, 3: 2458.74, 5: 2455.28}
logs/tensorflow/resnet50/bz128/1n8g/rn50_b128_fp32_4.log {1: 2494.22, 3: 2458.74, 5: 2455.28, 4: 2480.44}
logs/tensorflow/resnet50/bz128/1n8g/rn50_b128_fp32_2.log {1: 2494.22, 3: 2458.74, 5: 2455.28, 4: 2480.44, 2: 2441.07}
logs/tensorflow/resnet50/bz128/1n4g/rn50_b128_fp32_1.log {1: 1145.76}
logs/tensorflow/resnet50/bz128/1n4g/rn50_b128_fp32_3.log {1: 1145.76, 3: 1167.06}
logs/tensorflow/resnet50/bz128/1n4g/rn50_b128_fp32_5.log {1: 1145.76, 3: 1167.06, 5: 1128.34}
logs/tensorflow/resnet50/bz128/1n4g/rn50_b128_fp32_4.log {1: 1145.76, 3: 1167.06, 5: 1128.34, 4: 1122.51}
logs/tensorflow/resnet50/bz128/1n4g/rn50_b128_fp32_2.log {1: 1145.76, 3: 1167.06, 5: 1128.34, 4: 1122.51, 2: 1143.46}
logs/tensorflow/resnet50/bz128/1n1g/rn50_b128_fp32_1.log {1: 323.9}
logs/tensorflow/resnet50/bz128/1n1g/rn50_b128_fp32_3.log {1: 323.9, 3: 320.18}
logs/tensorflow/resnet50/bz128/1n1g/rn50_b128_fp32_5.log {1: 323.9, 3: 320.18, 5: 321.22}
logs/tensorflow/resnet50/bz128/1n1g/rn50_b128_fp32_4.log {1: 323.9, 3: 320.18, 5: 321.22, 4: 322.25}
logs/tensorflow/resnet50/bz128/1n1g/rn50_b128_fp32_2.log {1: 323.9, 3: 320.18, 5: 321.22, 4: 322.25, 2: 321.8}
logs/tensorflow/resnet50/bz128/1n2g/rn50_b128_fp32_1.log {1: 575.76}
logs/tensorflow/resnet50/bz128/1n2g/rn50_b128_fp32_3.log {1: 575.76, 3: 574.66}
logs/tensorflow/resnet50/bz128/1n2g/rn50_b128_fp32_5.log {1: 575.76, 3: 574.66, 5: 563.93}
logs/tensorflow/resnet50/bz128/1n2g/rn50_b128_fp32_4.log {1: 575.76, 3: 574.66, 5: 563.93, 4: 571.71}
logs/tensorflow/resnet50/bz128/1n2g/rn50_b128_fp32_2.log {1: 575.76, 3: 574.66, 5: 563.93, 4: 571.71, 2: 580.55}
logs/tensorflow/resnet50/bz128/2n8g/rn50_b128_fp32_1.log {1: 4864.58}
logs/tensorflow/resnet50/bz128/2n8g/rn50_b128_fp32_3.log {1: 4864.58, 3: 4829.93}
logs/tensorflow/resnet50/bz128/2n8g/rn50_b128_fp32_5.log {1: 4864.58, 3: 4829.93, 5: 4789.87}
logs/tensorflow/resnet50/bz128/2n8g/rn50_b128_fp32_4.log {1: 4864.58, 3: 4829.93, 5: 4789.87, 4: 4849.68}
logs/tensorflow/resnet50/bz128/2n8g/rn50_b128_fp32_2.log {1: 4864.58, 3: 4829.93, 5: 4789.87, 4: 4849.68, 2: 4918.45}
{'rn50': {'1n1g': {'average_speed': 321.87,
                   'batch_size_per_device': 128,
                   'median_speed': 321.8,
                   'speedup': 1.0},
          '1n2g': {'average_speed': 573.32,
                   'batch_size_per_device': 128,
                   'median_speed': 574.66,
                   'speedup': 1.79},
          '1n4g': {'average_speed': 1141.43,
                   'batch_size_per_device': 128,
                   'median_speed': 1143.46,
                   'speedup': 3.55},
          '1n8g': {'average_speed': 2465.95,
                   'batch_size_per_device': 128,
                   'median_speed': 2458.74,
                   'speedup': 7.64},
          '2n8g': {'average_speed': 4850.5,
                   'batch_size_per_device': 128,
                   'median_speed': 4849.68,
                   'speedup': 15.07},
          '4n8g': {'average_speed': 9412.91,
                   'batch_size_per_device': 128,
                   'median_speed': 9418.44,
                   'speedup': 29.27}}}
Saving result to ./result/bz128_result.json

计算规则

1.测速脚本

  • extract_tensorflow_logs.py
  • extract_tensorflow_logs_time.py

两个脚本略有不同,得到的结果稍有误差:

extract_tensorflow_logs.py根据官方在log中打印的速度,在600个iter中,排除前100iter,取后500个iter的速度做平均;

extract_paddle_logs_time.py则根据log中打印出的时间,排除前100iter取后500个iter的实际运行时间计算速度。

README展示的是extract_tensorflow_logs.py的计算结果。

2.均值速度和中值速度

  • average_speed均值速度

  • median_speed中值速度

    每个batch size进行5次训练测试,记为一组,每一组取average_speed为均值速度,median_speed为中值速度

3.加速比以中值速度计算

脚本和表格中的 加速比 是以单机单卡下的中值速度为基准进行计算的。例如:

单机单卡情况下速度为200(samples/s),单机2卡速度为400,单机4卡速度为700,则加速比分别为:1.0、2.0、3.5

ResNet50 v1.5 FP32

batch size = 128 & without xla

node_num gpu_num samples/s speedup
1 1 321.8 1
1 2 574.66 1.79
1 4 1143.46 3.55
1 8 2458.74 7.64
2 16 4849.68 15.07
4 32 9418.44 29.27

ResNet50 v1.5 FP16

batch size = 128 & without xla

node_num gpu_num samples/s speedup
1 1 938.09 1
1 2 1572.3 1.68
1 4 3061.66 3.26
1 8 5734.83 6.11
2 16 8418.2 8.97
4 32 16250.19 17.32

batch size = 256 & without xla

node_num gpu_num samples/s speedup
1 1 1075.27 1
1 4 3560.81 3.31
1 8 6898.0 6.42
2 16 9870.15 9.18
4 32 19314.31 17.96

batch size = 320 & without xla

node_num gpu_num samples/s speedup
1 1 1014.13 1
1 4 3616.97 3.57
1 8 7114.79 7.02
2 16 10210.45 10.07
4 32 20311.14 20.03

完整日志