zoukankan      html  css  js  c++  java
  • CoRR 2018 | Horovod: Fast and Easy Distributed Deep Learning in Tensorflow

    将深度学习模型的训练从单GPU扩展到多GPU主要面临以下问题:(1)训练框架必须支持GPU间的通信,(2)用户必须更改大量代码以使用多GPU进行训练。为了克服这些问题,本文提出了Horovod,它通过Ring Allreduce实现高效的GPU间通信,而且仅仅更改少量代码就可以实现多GPU训练。

    TensorFlow中提供了一些分布式训练的API,这些API适用于不同的环境。这就导致用户往往不知道如何更改代码以进行分布式训练,而且debug也很困难。再者,TensorFlow的分布式训练性能与理想的性能相差甚远,尤其是在大规模GPU环境下。如图1所示,随着GPU数量的增加,分布式TensorFlow的吞吐量与理想的吞吐量的差距逐渐增加,加速比逐渐降低。

    因为目前单GPU可以容纳大部分深度学习模型,所以本文主要针对数据并行进行优化。首先来看一下数据并行的训练过程:

    1. 运行多个模型副本
      (a) 读取一部分数据
      (b) 把数据喂给模型,进行前向传播
      (c) 反向传播,计算梯度
    2. 将多个模型的梯度进行平均
    3. 更新模型
    4. 重复上述步骤直到模型收敛

    在标准的TensorFlow中,分布式训练使用参数服务器架构,如图3所示。在参数服务器架构中,主要有worker和server两种角色。worker负责处理数据,计算梯度然后把梯度传给server;server负责聚合梯度,更新模型,然后把模型传回worker。

    在这上述两种模式下,主要有以下两个挑战:

    1. 如何确定worker和server的数量。如果只使用1台server,那么这台server可能成为计算和网络瓶颈;如果使用多台server,那么通信模式就类似于all-to-all,这样就不能完整利用网络带宽。
    2. 处理愈加复杂的TensorFlow程序。在TensorFlow中,必须显式地启动worker和server,传递一堆参数然后更新代码,这就使得分布式训练变得非常繁琐复杂。

    所幸的是,2017年百度提出了一种名为Ring Allreduce的算法。在该算法中,所有worker组成一个环,每台worker只和相邻的两台worker通信,如图4所示。

    在Ring Allreduce中,如果有(N)个节点,那么每个节点会通信(2 imes (N -1))次:前(N-1)次接收值并把它加到对应的buffer中,后(N-1)次接收并替换对应buffer中的值。Ring Allreduce算法是带宽最优的,也就是说,当buffer足够大时,它会最大限度地利用网络带宽。

    综上所述,本文取长补短,使用Ring Allreduce算法优化TensorFlow的分布式训练过程。本文的实现流程如下:

    1. 将代码转换成独立的Python包,名为Horovod
    2. 将百度的Ring Allreduce实现替换为NCCL
    3. 增加了对单机多GPU训练的支持
    4. 根据反馈更新了部分API,还实现了一个广播操作,以在所有worker上进行强制一致性初始化
    import tensorflow as tf
    import horovod.tensorflow as hvd
    
    # Initialize Horovod
    hvd.init()
    
    # Pin GPU to be used to process local rank (one GPU per process)
    config = tf.ConfigProto()
    config.gpu_options.visible_device_list = str(hvd.local_rank())
    
    # Build model...
    loss = ...
    opt = tf.train.AdagradOptimizer(0.01)
    
    # Add Horovod Distributed Optimizer
    opt = hvd.DistributedOptimizer(opt)
    
    # Add hook to broadcast variables from rank 0 to all other process
    # during initialization.
    hooks = [hvd.BroadcastGlobalVariablesHook(0)]
    train_op = opt.minimize(loss)
    
    # The MonitoredTrainingSession takes care of session initialization,
    # restoring from a checkpoint, saving to a checkpoint, and closing
    # when done or an error occurs.
    with tf.train.MonitoredTrainingSession(checkpoint="/tmp/train_logs", 
                                           config=config, hooks=hooks) as mon_sess:
        while not mon_sess.should_stop():
        # Perform synchronous training
        mon_sess.run(trian_op)
    

    此外,Horovod还提供了一个名为Timeline的分析工具,它可以让用户每个节点在每次迭代时做了什么,效果如图5所示。

    使用Timeline对一些模型进行分析后,发现当张量较小时,Ring Allreduce的效率并不高。因此,本文提出一种名为张量融合的技术来解决上述问题。

    1. 检测哪些张量将会被规约,选择适合缓冲区并具有相同数据类型的前几个张量
    2. 申请张量融合所需的缓冲区(如果之前没有申请的话),默认大小为64M
    3. 将选择的张量拷贝到融合缓冲区
    4. 在融合缓冲区执行allreduce操作
    5. 将数据从融合缓冲区拷贝到输出张量
    6. 重复上述步骤直到环中没有要被规约的向量

    使用Horovod之后,Inception V3和ResNet-101模型的性能提升了约88%,如图6所示。

    如图7,RDMA网络并没有比传统的TCP提升多少性能,只提升了约4%。

    未来的工作主要包括:

    1. 让MPI的安装变得更容易
    2. 分布式深度学习模型调参经验的收集与分享
    3. 增加大型模型的示例
  • 相关阅读:
    MDK(keil)4.7中文注释乱码解决
    小型功率放大器的设计与制作
    增强输出的电路
    晶体管电路设计学习笔记(一)
    MOSFET学习
    sysTick系统定时器
    C#面向对象 什么是面向对象
    JS基础 超链接、数列的用法,行内元素和块级元素
    JS基础 常用函数、事件、阻止事件冒泡
    JS基础 定时器【setTimeout、setInterval、clearInterval 】
  • 原文地址:https://www.cnblogs.com/littleorange/p/12586583.html
Copyright © 2011-2022 走看看