// Copyright (c) 2018 PaddlePaddle Authors. All Rights Reserved. // // 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. #pragma once #include #include "paddle/fluid/framework/eigen.h" #include "paddle/fluid/framework/op_registry.h" namespace paddle { namespace operators { template struct ReverseFunctor { void operator()(const DeviceContext& context, const framework::LoDTensor& in, framework::LoDTensor* out, const std::vector& axis) { Eigen::array reverse_axis; for (int i = 0; i < Rank; ++i) { reverse_axis[i] = false; } for (int a : axis) { if (a >= 0) { reverse_axis[a] = true; } else { reverse_axis[Rank + a] = true; } } auto in_eigen = framework::EigenTensor::From(in); auto out_eigen = framework::EigenTensor::From(*out); auto* dev = context.eigen_device(); out_eigen.device(*dev) = in_eigen.reverse(reverse_axis); } }; template class ReverseKernel : public framework::OpKernel { public: void Compute(const framework::ExecutionContext& context) const override { auto* x = context.Input("X"); auto* out = context.Output("Out"); out->mutable_data(context.GetPlace()); const auto& axis = context.Attr>("axis"); int rank = x->dims().size(); auto& dev_ctx = context.template device_context(); switch (rank) { case 1: ReverseFunctor functor1; functor1(dev_ctx, *x, out, axis); break; case 2: ReverseFunctor functor2; functor2(dev_ctx, *x, out, axis); break; case 3: ReverseFunctor functor3; functor3(dev_ctx, *x, out, axis); break; case 4: ReverseFunctor functor4; functor4(dev_ctx, *x, out, axis); break; case 5: ReverseFunctor functor5; functor5(dev_ctx, *x, out, axis); break; case 6: ReverseFunctor functor6; functor6(dev_ctx, *x, out, axis); break; default: PADDLE_THROW(paddle::platform::errors::OutOfRange( "The reserve operator does not support input tensors" "whose ranks are greater than 6.")); } } }; } // namespace operators } // namespace paddle