// 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::InferShapeImpl() const { 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)."; } bool reduce_all = param_.reduce_all; bool keep_dim = param_.keep_dim; if (reduce_all) { if (keep_dim) param_.output->Resize(std::vector(x_rank, 1)); else param_.output->Resize(std::vector{1}); } else { size_t out_rank = keep_dim ? x_rank : x_rank - dims.size(); std::vector out_dims(out_rank); std::stable_sort(dims.begin(), dims.end()); int dim_index = 0; int out_index = 0; for (size_t i = 0; i < x_rank; ++i) { if (dim_index < dims.size() && dims[dim_index] == static_cast(i)) { if (keep_dim) { out_dims[out_index++] = 1; } dim_index++; } else { out_dims[out_index++] = x_dims[i]; } } 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);