elementwise_min_op.cc 5.9 KB
Newer Older
1
/* Copyright (c) 2016 PaddlePaddle Authors. All Rights Reserved.
F
wip  
fengjiayi 已提交
2 3 4 5 6 7 8 9 10 11 12 13 14

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

    http://www.apache.org/licenses/LICENSE-2.0

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. */

15
#include <string>
W
wanghuancoder 已提交
16

W
Wu Yi 已提交
17
#include "paddle/fluid/operators/elementwise/elementwise_op.h"
18 19 20
#include "paddle/fluid/prim/api/composite_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"
21

W
wanghuancoder 已提交
22 23 24 25 26 27 28 29 30
namespace paddle {
namespace framework {
class OpDesc;
}  // namespace framework
namespace imperative {
class OpBase;
}  // namespace imperative
}  // namespace paddle

31 32 33 34 35 36 37
namespace paddle {
namespace operators {

class ElementwiseMinOpMaker : public ElementwiseOpMaker {
 protected:
  std::string GetName() const override { return "Min"; }
  std::string GetEquation() const override { return "Out = min(X, Y)"; }
38 39

  void AddInputX() override {
N
Noel 已提交
40
    AddInput("X", "The first tensor holding the elements to be compared.");
41 42 43
  }

  void AddInputY() override {
N
Noel 已提交
44
    AddInput("Y", "The second tensor holding the elements to be compared.");
45 46
  }

C
co63oc 已提交
47
  std::string GetOpFunctionality() const override {
48 49 50
    return "Compare two tensors and returns a new tensor containing the "
           "element-wise minima.";
  }
51 52
};

L
LJQ❤️ 已提交
53 54 55 56 57 58 59 60 61 62 63 64 65
class ElementwiseFMinOpMaker : public ElementwiseOpMaker {
 protected:
  std::string GetName() const override { return "FMin"; }
  std::string GetEquation() const override { return "Out = fmin(X, Y)"; }

  void AddInputX() override {
    AddInput("X", "The first tensor holding the elements to be compared.");
  }

  void AddInputY() override {
    AddInput("Y", "The second tensor holding the elements to be compared.");
  }

C
co63oc 已提交
66
  std::string GetOpFunctionality() const override {
L
LJQ❤️ 已提交
67 68 69 70 71 72 73
    return "Compare two tensors and returns a new tensor containing the "
           "element-wise minima. If the element of one tensor is nan, "
           "return the element value of the other tensor, if both are nan, "
           "return the first nan";
  }
};

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 101 102
class ElementwiseMinCompositeGradOpMaker
    : public prim::CompositeGradOpMakerBase {
  using prim::CompositeGradOpMakerBase::CompositeGradOpMakerBase;

 public:
  void Apply() override {
    paddle::Tensor x = this->GetSingleForwardInput("X");
    paddle::Tensor y = this->GetSingleForwardInput("Y");
    paddle::Tensor out_grad = this->GetSingleOutputGrad("Out");
    paddle::Tensor dx = this->GetSingleInputGrad("X");
    auto* dx_ptr = this->GetOutputPtr(&dx);
    std::string dx_name = this->GetOutputName(dx);
    paddle::Tensor dy = this->GetSingleInputGrad("Y");
    auto* dy_ptr = this->GetOutputPtr(&dy);
    std::string dy_name = this->GetOutputName(dy);
    VLOG(6) << "Runing minimum_grad composite func";
    int axis = static_cast<int>(this->Attr<int>("axis"));
    PADDLE_ENFORCE_EQ(
        axis,
        -1,
        phi::errors::InvalidArgument(
            "We only support axis = -1 in composite minimum_grad but we got: ",
            axis));
    prim::minimum_grad<prim::DescTensor>(x, y, out_grad, dx_ptr, dy_ptr);
    this->RecoverOutputName(dx, dx_name);
    this->RecoverOutputName(dy, dy_name);
  }
};

H
hong 已提交
103 104
template <typename T>
class ElementwiseMinGradOpMaker : public framework::SingleGradOpMaker<T> {
105
 public:
H
hong 已提交
106
  using framework::SingleGradOpMaker<T>::SingleGradOpMaker;
107 108

 protected:
109
  void Apply(GradOpPtr<T> op) const override {
110
    op->SetType("elementwise_min_grad");
H
hong 已提交
111 112 113 114 115 116
    op->SetInput("X", this->Input("X"));
    op->SetInput("Y", this->Input("Y"));
    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());
117 118 119
  }
};

L
LJQ❤️ 已提交
120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136
template <typename T>
class ElementwiseFMinGradOpMaker : public framework::SingleGradOpMaker<T> {
 public:
  using framework::SingleGradOpMaker<T>::SingleGradOpMaker;

 protected:
  void Apply(GradOpPtr<T> op) const override {
    op->SetType("elementwise_fmin_grad");
    op->SetInput("X", this->Input("X"));
    op->SetInput("Y", this->Input("Y"));
    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());
  }
};

137 138 139
}  // namespace operators
}  // namespace paddle

F
wip  
fengjiayi 已提交
140
namespace ops = paddle::operators;
141

142 143 144 145
REGISTER_OPERATOR(elementwise_min,
                  ops::ElementwiseOp,
                  ops::ElementwiseMinOpMaker,
                  ops::ElementwiseOpInferVarType,
H
hong 已提交
146
                  ops::ElementwiseMinGradOpMaker<paddle::framework::OpDesc>,
147 148
                  ops::ElementwiseMinGradOpMaker<paddle::imperative::OpBase>,
                  ops::ElementwiseMinCompositeGradOpMaker);
149 150 151

REGISTER_OPERATOR(elementwise_min_grad, ops::ElementwiseOpGrad);

152 153 154 155 156 157 158 159
REGISTER_OP_VERSION(elementwise_min)
    .AddCheckpoint(
        R"ROC(Register elementwise_min 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_min.",
            1.0f));
L
LJQ❤️ 已提交
160

161 162 163 164
REGISTER_OPERATOR(elementwise_fmin,
                  ops::ElementwiseOp,
                  ops::ElementwiseFMinOpMaker,
                  ops::ElementwiseOpInferVarType,
L
LJQ❤️ 已提交
165 166 167 168
                  ops::ElementwiseFMinGradOpMaker<paddle::framework::OpDesc>,
                  ops::ElementwiseFMinGradOpMaker<paddle::imperative::OpBase>);

REGISTER_OPERATOR(elementwise_fmin_grad, ops::ElementwiseOpGrad);