forked from pytorch/pytorch
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpiecewise_linear_transform_op.cc
100 lines (93 loc) · 3.85 KB
/
piecewise_linear_transform_op.cc
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
99
100
#include "caffe2/operators/piecewise_linear_transform_op.h"
namespace caffe2 {
REGISTER_CPU_OPERATOR(
PiecewiseLinearTransform,
PiecewiseLinearTransformOp<float, CPUContext>);
OPERATOR_SCHEMA(PiecewiseLinearTransform)
.NumInputs(1, 4)
.NumOutputs(1)
.SetDoc(R"DOC(
PiecewiseLinearTransform takes inputs -- predictions, a 2-D or 1-D tensor
(Tensor) of size (batch_size x prediction_dimensions). The piecewise
linear functions are stored in bounds, slopes and intercepts. The output tensor
has the same shape of input `predictions` and contains the predictions
transformed by the piecewise linear functions. Each column of predictions has
its own piecewise linear transformation functions. Therefore the size of
piecewise function parameters are pieces x prediction_dimensions, except for
binary predictions where only the positive prediction needs them. Note that in
each piece, low bound is excluded while high bound is included. Also the
piecewise linear function must be continuous.
Notes
- If the input is binary predictions (Nx2 or Nx1 tensor), set the binary arg
to true so that one group of piecewise linear functions is needed (see
details below).
- The transform parameters (bounds, slopes, intercepts) can be passed either
through args or through input blobs.
- If we have multiple groups of piecewise linear functions, each group has the
same number of pieces.
- If a prediction is out of the bounds, it is capped to the smallest or largest
bound.
)DOC")
.Arg(
"bounds",
"1-D vector of size (prediction_dimensions x (pieces+1)) contain the "
"upper bounds of each piece of linear function. One special case is "
"the first bound is the lower bound of whole piecewise function and we "
"treat it the same as the left most functions. (bounds, slopes, "
"intercepts) can be passed through either arg or input blobs.")
.Arg(
"slopes",
"1-D vector of size (prediction_dimensions x pieces) containing the "
"slopes of linear function")
.Arg(
"intercepts",
"1-D vector of size (prediction_dimensions x pieces) containing the "
"intercepts of linear function")
.Arg(
"binary",
"If set true, we assume the input is a Nx1 or Nx2 tensor. If it is Nx1 "
"tensor, it is positive predictions. If the input is Nx2 tensor, its "
"first column is negative predictions and second column is positive "
"and negative + positive = 1. We just need one group of piecewise "
"linear functions for the positive predictions.")
.Input(
0,
"predictions",
"2-D tensor (Tensor) of size "
"(num_batches x num_classes) containing scores")
.Input(
1,
"bounds (optional)",
"See bounds in Arg. (bounds, slopes, intercepts) can be passed through "
"either arg or input blobs.")
.Input(
2,
"slopes (optional)",
"See slopes in Arg. (bounds, slopes, intercepts) can be passed through "
"either arg or input blobs.")
.Input(
3,
"intercepts (optional)",
"See intercepts in Arg. (bounds, slopes, intercepts) can be passed "
"through either arg or input blobs.")
.Output(
0,
"transforms",
"2-D tensor (Tensor) of size (num_batches x num_classes) "
"containing transformed predictions");
SHOULD_NOT_DO_GRADIENT(PiecewiseLinearTransform);
} // namespace caffe2
using PiecewiseLinearTransformOpFloatCPU =
caffe2::PiecewiseLinearTransformOp<float, caffe2::CPUContext>;
// clang-format off
C10_EXPORT_CAFFE2_OP_TO_C10_CPU(
PiecewiseLinearTransform,
"_caffe2::PiecewiseLinearTransform("
"Tensor predictions, "
"float[] bounds, "
"float[] slopes, "
"float[] intercepts, "
"bool binary"
") -> (Tensor output_0)",
PiecewiseLinearTransformOpFloatCPU);
// clang-format on