README.MD 6.3 KB
Newer Older
T
tensor-tang 已提交
1 2 3 4
# Intel® MKL-DNN on PaddlePaddle: Design Doc

我们计划将Intel深度神经网络数学库(**MKL-DNN**\[[1](#references)\])集成到PaddlePaddle,充分展现英特尔平台的优势,有效提升PaddlePaddle在英特尔架构上的性能。

T
tensor-tang 已提交
5
我们短期内的基本目标是:
T
tensor-tang 已提交
6

T
tensor-tang 已提交
7 8
- 完成常用layer的MKL-DNN实现。
- 完成常见深度神经网络VGG,GoogLeNet 和 ResNet的MKL-DNN实现。
T
tensor-tang 已提交
9 10 11 12


## Contents

T
tensor-tang 已提交
13 14 15 16 17 18 19
- [Overview](#overview)
- [Actions](#actions)
 	- [CMake](#cmake)
	- [Layers](#layers)
	- [Activations](#activations)
	- [Unit Tests](#unit-tests)
	- [Protobuf Messages](#protobuf-messages)
T
tensor-tang 已提交
20
	- [Python API](#python-api)
T
tensor-tang 已提交
21 22
	- [Demos](#demos)
	- [Benchmarking](#benchmarking)
T
tensor-tang 已提交
23
	- [Others](#others)
T
tensor-tang 已提交
24
- [Design Concerns](#design-concerns)
T
tensor-tang 已提交
25

T
tensor-tang 已提交
26
## Overview
T
tensor-tang 已提交
27

T
tensor-tang 已提交
28
我们会把MKL-DNN作为第三方库集成进PaddlePaddle,整体框架图
T
tensor-tang 已提交
29 30 31 32 33
<div align="center">
<img src="image/overview.png" width=350><br/>
Figure 1. PaddlePaddle on IA.
</div>

T
tensor-tang 已提交
34
## Actions
T
tensor-tang 已提交
35
我们把集成方案大致分为了如下几个方面。
T
tensor-tang 已提交
36

T
tensor-tang 已提交
37 38
### CMake
我们会在`CMakeLists.txt`中会添加`WITH_MKLDNN`的选项,当设置这个值为`ON`的时候会启用编译MKL-DNN功能。同时会自动开启OpenMP用于提高MKL-DNN的性能。
T
tensor-tang 已提交
39

T
tensor-tang 已提交
40
同时,我们会引入`WITH_MKLML`选项,用于选择是否使用MKL-DNN自带的MKLML安装包。这个安装包可以独立于MKL-DNN使用,但是建议在开启MKL-DNN的同时也打开MKLML的开关,这样才能发挥最好的性能。
T
tensor-tang 已提交
41

T
tensor-tang 已提交
42
所以,我们会在`cmake/external`目录新建`mkldnn.cmake``mklml.cmake`文件,它们会在编译PaddlePaddle的时候下载对应的软件包,并放到PaddlePaddle的third party目录中。
T
tensor-tang 已提交
43

T
tensor-tang 已提交
44
**备注**:当`WITH_MKLML=ON`的时候,会优先使用这个包作为PaddlePaddle的CBLAS和LAPACK库,所以会稍微改动`cmake/cblas.cmake`中的逻辑。
T
tensor-tang 已提交
45

T
tensor-tang 已提交
46 47 48
### Layers
所有MKL-DNN相关的C++ layers,都会按照PaddlePaddle的目录结构存放在
`paddle/gserver/layers`中,并且文件名都会一以*Mkldnn*开头。
T
tensor-tang 已提交
49

T
tensor-tang 已提交
50
所有MKL-DNN的layers都会继承于一个叫做`MkldnnLayer`的父类,该父类继承于PaddlePaddle的基类`Layer`
T
tensor-tang 已提交
51

T
tensor-tang 已提交
52 53
### Activations
由于在PaddlePaddle中,激活函数是独立于layer概念的,所以会在`paddle/gserver/activations`目录下添加一个`MkldnnActivation.h`文件定义一些用于MKL-DNN的接口,实现方法还是会在`ActivationFunction.cpp`文件。
T
tensor-tang 已提交
54

T
tensor-tang 已提交
55 56
### Unit Tests
会在`paddle/gserver/test`目录下添加`test_Mkldnn.cpp``MkldnnTester.*`用于MKL-DNN的测试。
T
tensor-tang 已提交
57

T
tensor-tang 已提交
58
Activation的测试,计划在PaddlePaddle原有的测试文件上直接添加新的测试type。
T
tensor-tang 已提交
59

T
tensor-tang 已提交
60 61
### Protobuf Messages
根据具体layer的需求可能会在`proto/ModelConfig.proto`里面添加必要的选项。
T
tensor-tang 已提交
62 63 64 65

### Python API
目前只考虑**v1 API**

T
tensor-tang 已提交
66
计划在`python/paddle/trainer/config_parser.py`里面添加`use_mkldnn`这个选择,方便用户选择使用MKL-DNN的layers。
T
tensor-tang 已提交
67 68 69

具体实现方式比如:

T
tensor-tang 已提交
70 71 72 73 74
```python
use_mkldnn = bool(int(g_command_config_args.get("use_mkldnn", 0)))
if use_mkldnn
    self.layer_type = mkldnn_*
```
T
tensor-tang 已提交
75

T
tensor-tang 已提交
76
所有MKL-DNN的layer type会以*mkldnn_*开头,以示区分。 
T
tensor-tang 已提交
77

T
tensor-tang 已提交
78
并且可能在`python/paddle/trainer_config_helper`目录下的`activations.py ``layers.py`里面添加必要的MKL-DNN的接口。
T
tensor-tang 已提交
79

T
tensor-tang 已提交
80
### Demos
T
tensor-tang 已提交
81

T
tensor-tang 已提交
82
会在`v1_api_demo`目录下添加一个`mkldnn`的文件夹,里面放入一些用于MKL-DNN测试的demo脚本。
T
tensor-tang 已提交
83

T
tensor-tang 已提交
84 85
### Benchmarking
会考虑添加部分逻辑在`benchmark/paddle/image/run.sh`,添加使用MKL-DNN的测试。
T
tensor-tang 已提交
86 87

### Others
T
tensor-tang 已提交
88 89
1. 如果在使用MKL-DNN的情况下,会把CPU的Buffer对齐为64。
2. 深入PaddlePaddle,寻找有没有其他可以优化的可能,进一步优化。比如可能会用OpenMP改进SGD的更新性能。
T
tensor-tang 已提交
90

T
tensor-tang 已提交
91
## Design Concerns
T
tensor-tang 已提交
92

T
tensor-tang 已提交
93
为了更好的符合PaddlePaddle的代码风格\[[2](#references)\],同时又尽可能少的牺牲MKL-DNN的性能\[[3](#references)\]
T
tensor-tang 已提交
94

T
tensor-tang 已提交
95
我们总结出一些特别需要注意的点:
T
tensor-tang 已提交
96

T
tensor-tang 已提交
97 98 99 100 101 102 103
1. 使用**deviceId_**。为了尽可能少的在父类Layer中添加变量或者函数,我们决定使用已有的`deviceId_`变量来区分layer的属性,定义`-2``MkldnnLayer`特有的设备ID。
2. 重写父类Layer的**init**函数,修改`deviceId_``-2`,代表这个layer是用于跑在MKL-DNN的环境下。
3. 创建`MkldnnMatrix`,用于管理MKL-DNN会用到的相关memory函数、接口以及会用的到格式信息。
4. 创建`MkldnnBase`,定义一些除了layer和memory相关的类和函数。包括MKL-DNN会用到`MkldnnStream``CpuEngine`,和未来可能还会用到`FPGAEngine`等。
5.**Argument**里添加两个`MkldnnMatrixPtr`,取名为`mkldnnValue``mkldnnGrad`,用于存放`MkldnnLayer`会用到的memory buffer。 并且添加函数cvt(会修改为一个更加合适的函数名),用于处理"CPU device"和"MKL-DNN device"之间memory的相互转化。
6. 在父类`Layer`中的`getOutput`函数中添加一段逻辑,用于判断`deviceId`,并针对device在MKL-DNN和CPU之间不统一的情况,做一个前期转换。 也就是调用`Argument`的cvt函数把output统一到需要的device上。
7. 在原来的`FLAGS`中添加一个`use_mkldnn`的flag,用于选择是否使用MKL-DNN的相关功能。
T
update  
tensor-tang 已提交
104
8. 关于MKLDNN参数的保存。由于MKLDNN参数的格式与PaddlePaddle原有的格式存在不一样的情况,所以需要在保存参数时同时保存该格式信息。目前准备扩展[Header](https://github.com/PaddlePaddle/Paddle/blob/develop/paddle/parameter/Parameter.h#L247)里面的`int32_t version`。这个值不管是在v1还是在v2里面,一直保存的是0,所以可以充分利用这个信息,定义一个枚举处理所有MKLDNN的参数格式,从而`MKLDNNLayer`就可以从输入的参数中获取需要的格式信息。
T
tensor-tang 已提交
105

T
tensor-tang 已提交
106 107
## References

T
tensor-tang 已提交
108
1. [Intel Math Kernel Library for Deep Neural Networks (Intel MKL-DNN)](https://github.com/01org/mkl-dnn "Intel MKL-DNN")
T
tensor-tang 已提交
109
2. [原来的方案](https://github.com/PaddlePaddle/Paddle/pull/3096)会引入**nextLayer**的信息。但是在PaddlePaddle中,无论是重构前的layer还是重构后的op,都不会想要知道next layer/op的信息。
T
tensor-tang 已提交
110
3. MKL-DNN的高性能格式与PaddlePaddle原有的`NCHW`不同(PaddlePaddle中的CUDNN部分使用的也是`NCHW`,所以不存在这个问题),所以需要引入一个转换方法,并且只需要在必要的时候转换这种格式,才能更好的发挥MKL-DNN的性能。
T
tensor-tang 已提交
111