elementwise_div_op.cc 5.4 KB
Newer Older
1
/* Copyright (c) 2016 PaddlePaddle Authors. All Rights Reserved.
G
gongweibao 已提交
2

L
Luo Tao 已提交
3 4 5
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
G
gongweibao 已提交
6

L
Luo Tao 已提交
7
    http://www.apache.org/licenses/LICENSE-2.0
G
gongweibao 已提交
8

L
Luo Tao 已提交
9 10 11 12 13
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License. */
G
gongweibao 已提交
14

W
Wu Yi 已提交
15
#include "paddle/fluid/operators/elementwise/elementwise_div_op.h"
16

17 18
#include <memory>
#include <string>
19

W
Wu Yi 已提交
20
#include "paddle/fluid/operators/elementwise/elementwise_op.h"
21
#include "paddle/fluid/platform/complex.h"
22 23 24
#include "paddle/fluid/prim/api/manual/backward/composite_backward_api.h"
#include "paddle/fluid/prim/utils/static/composite_grad_desc_maker.h"
#include "paddle/fluid/prim/utils/static/desc_tensor.h"
25 26 27 28 29 30 31
namespace paddle {
namespace operators {

class ElementwiseDivOpMaker : public ElementwiseOpMaker {
 protected:
  std::string GetName() const override { return "Div"; }
  std::string GetEquation() const override { return "Out = X / Y"; }
32 33

  void AddInputX() override {
34 35 36 37
    AddInput(
        "X",
        "(Variable), Tensor or phi::DenseTensor of any dimensions. Its dtype "
        "should be int32, int64, float32, float64.");
38 39 40
  }

  void AddInputY() override {
41 42 43 44
    AddInput(
        "Y",
        "(Variable), Tensor or phi::DenseTensor of any dimensions. Its dtype "
        "should be int32, int64, float32, float64.");
45 46 47 48 49
  }

  std::string GetOpFuntionality() const override {
    return "Divide two tensors element-wise";
  }
50 51
};

H
hong 已提交
52 53
template <typename T>
class ElementwiseDivGradOpMaker : public framework::SingleGradOpMaker<T> {
54
 public:
H
hong 已提交
55
  using framework::SingleGradOpMaker<T>::SingleGradOpMaker;
56 57

 protected:
58
  void Apply(GradOpPtr<T> op) const override {
59
    op->SetType("elementwise_div_grad");
60
    op->SetInput("X", this->Input("X"));
H
hong 已提交
61 62 63 64 65 66
    op->SetInput("Y", this->Input("Y"));
    op->SetInput("Out", this->Output("Out"));
    op->SetInput(framework::GradVarName("Out"), this->OutputGrad("Out"));
    op->SetOutput(framework::GradVarName("X"), this->InputGrad("X"));
    op->SetOutput(framework::GradVarName("Y"), this->InputGrad("Y"));
    op->SetAttrMap(this->Attrs());
67 68 69
  }
};

70
class ElementwiseDivGradCompositeOpMaker
71 72
    : public prim::CompositeGradOpMakerBase {
  using prim::CompositeGradOpMakerBase::CompositeGradOpMakerBase;
73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94

 public:
  void Apply() override {
    paddle::experimental::Tensor x = this->GetSingleForwardInput("X");
    paddle::experimental::Tensor y = this->GetSingleForwardInput("Y");
    paddle::experimental::Tensor out = this->GetSingleForwardOutput("Out");
    paddle::experimental::Tensor out_grad = this->GetSingleOutputGrad("Out");
    paddle::experimental::Tensor dx = this->GetSingleInputGrad("X");
    auto dx_ptr = this->GetOutputPtr(&dx);
    std::string dx_name = this->GetOutputName(dx);
    paddle::experimental::Tensor dy = this->GetSingleInputGrad("Y");
    auto dy_ptr = this->GetOutputPtr(&dy);
    std::string dy_name = this->GetOutputName(dy);
    int axis = static_cast<int>(this->Attr<int>("axis"));
    VLOG(3) << "Runing div_grad composite func";
    prim::divide_grad<prim::DescTensor>(
        x, y, out, out_grad, axis, dx_ptr, dy_ptr);
    this->RecoverOutputName(dx, dx_name);
    this->RecoverOutputName(dy, dy_name);
  }
};

H
hong 已提交
95 96
template <typename T>
class ElementwiseDivDoubleGradMaker : public framework::SingleGradOpMaker<T> {
97
 public:
H
hong 已提交
98
  using framework::SingleGradOpMaker<T>::SingleGradOpMaker;
99 100

 protected:
101
  void Apply(GradOpPtr<T> op) const override {
102
    op->SetType("elementwise_div_grad_grad");
H
hong 已提交
103 104 105 106 107
    op->SetInput("Y", this->Input("Y"));
    op->SetInput("Out", this->Input("Out"));
    op->SetInput("DDX", this->OutputGrad(framework::GradVarName("X")));
    op->SetInput("DDY", this->OutputGrad(framework::GradVarName("Y")));
    op->SetInput("DX", this->Output(framework::GradVarName("X")));
108

H
hong 已提交
109
    op->SetAttrMap(this->Attrs());
110

H
hong 已提交
111 112 113
    op->SetOutput(framework::GradVarName("Y"), this->InputGrad("Y"));
    op->SetOutput("DOut", this->InputGrad("Out"));
    op->SetOutput("DDOut", this->InputGrad(framework::GradVarName("Out")));
114 115 116
  }
};

117 118 119
}  // namespace operators
}  // namespace paddle

G
gongweibao 已提交
120
namespace ops = paddle::operators;
121

122 123 124 125
REGISTER_OPERATOR(elementwise_div,
                  ops::ElementwiseOp,
                  ops::ElementwiseDivOpMaker,
                  ops::ElementwiseOpInferVarType,
126
                  ops::ElementwiseDivGradCompositeOpMaker,
H
hong 已提交
127 128 129 130
                  ops::ElementwiseDivGradOpMaker<paddle::framework::OpDesc>,
                  ops::ElementwiseDivGradOpMaker<paddle::imperative::OpBase>);

REGISTER_OPERATOR(
131 132
    elementwise_div_grad,
    ops::ElementwiseOpGrad,
H
hong 已提交
133 134
    ops::ElementwiseDivDoubleGradMaker<paddle::framework::OpDesc>,
    ops::ElementwiseDivDoubleGradMaker<paddle::imperative::OpBase>);
135

136 137
REGISTER_OPERATOR(elementwise_div_grad_grad,
                  ops::ElementwiseDivOpDoubleGrad,
138
                  ops::ElementwiseDoubleGradOpInplaceInferer);
139

140 141 142 143 144 145 146 147
REGISTER_OP_VERSION(elementwise_div)
    .AddCheckpoint(
        R"ROC(Register elementwise_div for adding the attribute of Scale_y)ROC",
        paddle::framework::compatible::OpVersionDesc().NewAttr(
            "Scale_y",
            "In order to support the function of scaling the input Y when "
            "using the operator of elementwise_div.",
            1.0f));