5 Z( p. q7 o$ ?: P# f: H1 I0 L
作者:Pieterluitjens
|: L5 j4 q- H9 @ 参与:一鸣、嘉明、思
* e. X6 i4 S* K& o* | ]% m 你的数据处理影响整个训练速度,如果加上英伟达 DALI 库,处理速度比原生 PyTorch 也能快上四倍。
7 P) Y4 Q w9 u" y& a! { ~
4 i3 h" T) w$ T% Z S% E+ I, X
+ E! ~# ?; c1 U. V4 U; E% p+ s 打开凤凰新闻,查看更多高清图片
+ H! n' I1 L# N/ O. |5 ]1 L* c0 M ! u, Z& [2 T, o' G
: K$ Z1 O6 j; Z3 A5 F- y 深度学习的加速上,除了对训练和推理过程的加速之外,还有对数据加载和预处理阶段的加速。这一阶段也尤为关键,因为数据处理 pipeline 的处理速度也影响着整体的流程效率。
; j d% V& z4 z" z 近日,有开发者介绍了如何使用 NVIDIA Data Loading Library(DALI),以及使用这个库进行数据预处理加速的方法。DALI 和 TensorFlow 自带的 DataLoader 类似,是一个专门用于加速数据预处理过程的库。
5 T# A1 V- l" I
在一篇博客中,开发者表示,该库不仅可以进行 GPU 加速,也可以在 CPU 上构建一个完整的数据预处理流程。如果在 Tesla V100 上做测试,PyTorch+DALI 的处理速度能达到 4000 images/s,比原版 PyTorch 要快近 4 倍。
6 Q9 |! A# f2 E x0 i/ B0 a% ] 支持多个框架,针对预处理
! {% d" ^' J6 m1 k7 I 英伟达数据加载库 DALI 是一个便捷式开源库,用于图像或视频的解码及增强,从而加速深度学习应用。通过并行训练和预处理过程,减少了延迟及训练时间,并为当下流行的深度学习框架中的内置数据加载器及数据迭代器提供了一个嵌入式替代器,便于集成或重定向至不同框架。
* ?& n, a* K1 E7 }8 x9 Z5 W% ] 开发人员可以在 GPU 上运行他们的数据处理通道,从而减少训练神经网络的时间。而 DALI 实现了数据处理 pipeline 可移植,因为可以轻松地重定向至 TensorFlow,PyTorch 和 MXNet。
1 p; F+ U0 Q9 F- x

' r$ r+ ?% _" J2 F. e5 B8 z DALI 设计之初就是用来帮助用户突破数据预处理的瓶颈,使得模型的训练和推理能够达到最高的效率。其主要设计是用于在 GPU 上的预处理,但大多数操作也可在 CPU 上实现。
- R, Q; Z* M8 s7 k! Q2 a
主要特征
) f+ @ F4 v% D6 J; m2 @
易于使用的 Python API
2 g1 A) _- H5 u: v& \' @6 f 在多个 GPU 之间显式地缩放
. f& K" l) }' d6 }8 J 加快图像分类(ResNet-50)和目标检测分类(SSD)工作流
) \5 v. u$ J& m: D
灵活的计算图使得开发人员可创建自定义的数据处理 pipeline
' w& I3 [5 J, Z 支持多种格式 - LMDB、RecordIO、TFRecord、COCO、JPEG、H.264 以及 HEVC
/ O/ B; _3 D8 a) e7 H; ]" ^ 开发人员可添加自定义的图像或视频处理 operator
# ]4 Q& X, b8 t7 I

9 C2 O7 P8 _, K
DALI 在 MXNet 上训练 ResNet50 的性能表现。
$ f7 d6 \9 d# B0 L- j8 E0 l6 O 使用 DALI 重构数据预处理 pipeline
( z! _; E+ X* |3 W/ O 内存占用大是 DALI 在加速数据预处理中面对的一个重要问题。随着批大小的上升,DALI 的内存占用会变得更大。这一问题现在还没有得到修复。
6 G% l$ ^+ J9 \- D# Z! }

8 p+ \1 W" d! H n" j# @) l* ]0 a0 P
DALI 的内存占用问题。
0 Q5 Y/ ?0 _8 \
而和其他的数据预处理工具相比,DALI 能够最大处理的批大小相比 TorchVision 少了一半。
' R! _/ Q0 @6 I, [2 R* l 
c: v5 U$ N/ D, ? 因此,使用 DALI 加速数据预处理需要重新构建整个 pipeline。在教程中,作者分别介绍了构建 CPU pipeline 和 GPU pipeline 的方法。
0 X8 F9 r9 ~6 b( [ ` 构建 CPU pipepline
r1 J! U# a/ R; [ 使用 CPU 进行数据预处理的时候,常用的做法是让 CPU 处理解码和 resize 的工作,而 CropMirrorNormalize 这样的工作放到 GPU 上。由于使用 DALI 将输出传入 GPU 中就会占用很大的内存,因此作者构建了一个完全基于 CPU 的 pipeline。
1 ^; w) X: l* s {9 U! D8 @
首先,重新加载并导入 DALI。
0 Z- Y( z3 R) G; R 
) \! F2 |1 p5 ^; I2 p2 y& B. r
然后构建基于 CPU 的 pipeline。
$ p" G4 }7 a! b! {2 N/ i

& Y' w0 J$ w' x# f$ B- r9 \ 在这里设置一下哪些操作应当由 CPU 运行。
& [2 k/ n9 ]/ y$ B7 ? 
$ ]' P) t# ] b) c4 {0 W2 c/ { 然后开始相应的操作。如解码:
6 U1 X1 v0 i5 t: \( U0 ] 
: s8 L3 E8 ?% t' u4 O% v 以及 resize:
( w0 I( [. m- ^

1 |& Y0 U4 H) [- E! t. O0 K X, I! b CropMirrorNormalize 的操作由 GPU 进行。
c( N {8 {3 L8 t$ @; C& q 
4 Y4 \ Y( k6 k# U* l; [1 q% u+ O
当然,还需要从 CPU 到 GPU 的转移过程,这一过程使用 PyTorch 进行,作者在教程中也给出了相应的代码。
# v. y h" \/ R% L: o4 w' S. [* i
构建基于 GPU 的 pipeline
8 o* j. y9 _% |% A; [) l6 h: a 当模型较大(如使用 ResNet50)的时候,基于 CPU 的 pipepline 效果较好,而在类似于 AlexNet 或 ResNet18 这样的小模型上,CPU 的处理还是跟不上 GPU 的数据处理速度。其主要原因在于,GPU pipeline 会降低近一半的批大小。
9 N0 x( L. V' c) J( w
为了解决这个问题,作者建议将验证 pipepline 从 GPU 中关掉,直到每个 epoch 的最后需要的时候再开启。
p2 c9 J) M; h$ [# \
效果评价
1 B2 Q2 J J$ w3 Y; ~1 V, Z9 j7 m 以下是作者在 ResNet18 中使用的最大批量大小:
5 g' K* U& x, v/ ~' F( U7 } 
: A4 e# x7 b. i2 o% x
因此通过合理应用这一系列修改,DALI 可以使得最大批量大小在 CPU 或 GPU 模式下提升 50%
7 K! E& U! B, I) v* d4 J
而在 Shufflenet V2 0.5 上,如果批大小为 512,则 DALI GPU 和 CPU 能够处理的批大小如下:
8 y, T6 }* S: G; K8 K6 Q

" L& m2 f! f v2 t3 o* D
可以看到,即使是 CPU,其处理速度也达到了每秒 1800 张图像,速度超过了 TorchVision。
; B% [4 Q* Q) U/ _+ d# q
所有的测试都在 Google Cloud V100 实例下运行:包含 12 个 vCPU(6 个物理内核),78GB 内存,以及使用 ApexFP16 进行训练。
/ }# a. B b N9 M2 q
因此,通过 DALI,一个 Tesla V100 GPU 就可以达到将近每秒 4000 张图像。接近英伟达有 8 个 V100 GPU 的 DGX-1 的性能,尽管使用的是小模型。
) L' w, M8 K% s9 c; J2 B https://towardsdatascience.com/nvidia-dali-speeding-up-pytorch-876c80182440
$ x9 U# z, z( h
加入机器之心(全职记者 / 实习生):hr@jiqizhixin.com
$ g$ M8 n+ L: o0 `) q% O( W4 |7 _ 投稿或寻求报道:content @jiqizhixin.com
+ P7 _6 a* N) S5 e$ q 广告 & 商务合作:bd@jiqizhixin.com
/ i$ o$ F8 [- J) U