// Copyright (c) 2019 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. #include "lite/operators/reduce_ops.h" #include #include "lite/core/op_registry.h" namespace paddle { namespace lite { namespace operators { bool ReduceOp::CheckShape() const { CHECK_OR_FALSE(param_.x); CHECK_OR_FALSE(param_.output); auto x_dims = param_.x->dims(); auto x_rank = x_dims.size(); CHECK_LE(x_rank, 6UL) << "Tensors with rank at most 6 are supported."; return true; } bool ReduceOp::InferShape() const { auto x_dims = param_.x->dims(); auto x_rank = x_dims.size(); auto dims = param_.dim; for (size_t i = 0; i < dims.size(); ++i) { if (dims[i] < 0) dims[i] = x_rank + dims[i]; CHECK_LT(dims[i], x_rank) << "The dim should be in the range [-rank(input), rank(input)."; } sort(dims.begin(), dims.end()); bool reduce_all = param_.reduce_all; bool keep_dim = param_.keep_dim; if (reduce_all) { if (keep_dim) param_.output->Resize(lite::DDim(std::vector(x_rank, 1))); else param_.output->Resize(lite::DDim(std::vector{1})); } else { auto dims_vector = x_dims.Vectorize(); if (keep_dim) { for (size_t i = 0; i < dims.size(); ++i) { dims_vector[dims[i]] = 1; } } else { const int kDelFlag = -2; for (size_t i = 0; i < dims.size(); ++i) { dims_vector[dims[i]] = kDelFlag; } dims_vector.erase( remove(dims_vector.begin(), dims_vector.end(), kDelFlag), dims_vector.end()); } auto out_dims = lite::DDim(dims_vector); param_.output->Resize(out_dims); if (dims[0] != 0) { param_.output->set_lod(param_.x->lod()); } } return true; } bool ReduceOp::AttachImpl(const cpp::OpDesc &opdesc, lite::Scope *scope) { param_.x = scope->FindVar(opdesc.Input("X").front())->GetMutable(); param_.output = scope->FindVar(opdesc.Output("Out").front())->GetMutable(); param_.dim = opdesc.GetAttr>("dim"); param_.reduce_all = opdesc.GetAttr("reduce_all"); param_.keep_dim = opdesc.GetAttr("keep_dim"); return true; } } // namespace operators } // namespace lite } // namespace paddle REGISTER_LITE_OP(reduce_sum, paddle::lite::operators::ReduceOp);