forked from pytorch/pytorch
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMKLDNNConversions.cpp
98 lines (84 loc) · 3.44 KB
/
MKLDNNConversions.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
#include <ATen/ATen.h>
#include <ATen/NativeFunctions.h>
#include <ATen/Config.h>
#include <ATen/native/mkldnn/MKLDNNCommon.h>
#include <ATen/native/utils/ParamUtils.h>
namespace at { namespace native {
#if AT_MKLDNN_ENABLED()
Tensor mkldnn_to_dense(const Tensor& mkldnn_tensor) {
ideep::tensor& stensor = itensor_from_mkldnn(mkldnn_tensor);
auto dims = stensor.get_dims();
// NOTE: int32_t dims from ideep::tensor but sizes needs int64_t
Tensor cpu_tensor = at::empty(
std::vector<int64_t>(dims.begin(), dims.end()),
mkldnn_tensor.options().layout(c10::kStrided));
stensor.to_public(cpu_tensor.template data_ptr<float>());
return cpu_tensor;
}
Tensor dense_to_mkldnn(const Tensor& cpu_tensor) {
AT_ASSERTM(cpu_tensor.device().type() == DeviceType::CPU,
"dense_to_mkldnn expects CPU tensor input");
AT_ASSERTM(cpu_tensor.layout() == Layout::Strided,
"dense_to_mkldnn expects strided tensor input");
AT_ASSERTM(cpu_tensor.scalar_type() == ScalarType::Float,
"dense_to_mkldnn expects float tensor input");
AT_ASSERTM(cpu_tensor.dim() <= 5,
"Can't convert cpu tensor with the number of dimensions > 5");
// TODO: consider to convert non-contiguous tensor to `ideep::tensor` directly.
auto cpu_tensor_cont = cpu_tensor.contiguous();
Tensor mkldnn_tensor = empty_mkldnn(cpu_tensor_cont.sizes(), cpu_tensor_cont.options());
ideep::tensor& dtensor = itensor_from_mkldnn(mkldnn_tensor);
dtensor.feed_from(dtensor.get_dims(),
ideep::tensor::data_type::f32,
(cpu_tensor_cont.template data_ptr<float>()));
return mkldnn_tensor;
}
// Mkldnn tensor has special non-public format for conv2d weights
// (dense_to_mkldnn only converts dense tensor to mkldnn tensor with
// public format). Ideep conv kernel will do implicit reorder if the
// weight is not already in this optimized format. By the time I'm
// writing this note, we are seeing ~20% perf cost of doing the
// on-the-fly reorder.
Tensor mkldnn_reorder_conv2d_weight(
const Tensor& self,
IntArrayRef padding,
IntArrayRef stride,
IntArrayRef dilation,
int64_t groups) {
auto stride_vec = expand_param_if_needed(stride, "stride", 2);
auto padding_vec = expand_param_if_needed(padding, "padding", 2);
auto dilation_vec = expand_param_if_needed(dilation, "dilation", 2);
ideep::tensor w = itensor_from_mkldnn(self).as_weights();
w.make_group(groups);
ideep::tensor::descriptor desc =
ideep::convolution_forward::expected_weights_descriptor(
w.get_dims(),
w.get_data_type(),
{stride_vec.cbegin(), stride_vec.cend()},
{padding_vec.cbegin(), padding_vec.cend()},
{padding_vec.cbegin(), padding_vec.cend()},
{dilation_vec.cbegin(), dilation_vec.cend()},
groups,
ideep::algorithm::convolution_direct);
ideep::tensor result;
result.init<AllocForMKLDNN>(desc);
result.feed_from(w);
return new_with_itensor_mkldnn(std::move(result), self.options());
}
#else
Tensor mkldnn_to_dense(const Tensor& mkldnn_tensor) {
AT_ERROR("MKL-DNN build is disabled");
}
Tensor dense_to_mkldnn(const Tensor& cpu_tensor) {
AT_ERROR("MKL-DNN build is disabled");
}
Tensor mkldnn_reorder_conv2d_weight(
const Tensor& self,
IntArrayRef padding,
IntArrayRef stride,
IntArrayRef dilation,
int64_t groups) {
AT_ERROR("mkldnn_reorder_conv2d_weight: MKL-DNN build is disabled");
}
#endif // AT_MKLDNN_ENABLED()
}}