// 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/concat_op.h" #include "lite/core/op_lite.h" #include "lite/core/op_registry.h" namespace paddle { namespace lite { namespace operators { bool ConcatOpLite::CheckShape() const { CHECK_GE_OR_FALSE(param_.x.size(), 1UL); CHECK_OR_FALSE(param_.output); return true; } bool ConcatOpLite::InferShape() const { const std::vector &inputs = param_.x; const size_t n = inputs.size(); CHECK_GT_OR_FALSE(n, 0); int axis = 0; if (param_.axis_tensor == nullptr) { axis = param_.axis; } else { auto *axis_tensor_val = param_.axis_tensor->data(); axis = axis_tensor_val[0]; } if (axis < 0) { axis += inputs[0]->dims().size(); } auto out_dims = inputs[0]->dims(); size_t in_zero_dims_size = out_dims.size(); for (size_t i = 1; i < n; i++) { const auto &input_dims_i = inputs[i]->dims(); for (size_t j = 0; j < in_zero_dims_size; j++) { if (j == static_cast(axis)) { out_dims[axis] += input_dims_i[j]; } else { CHECK_EQ_OR_FALSE(out_dims[j], input_dims_i[j]); } } } if (out_dims[axis] < 0) { out_dims[axis] = -1; } // Set output dims param_.output->Resize(out_dims); auto out_lod = param_.output->mutable_lod(); *out_lod = param_.x[0]->lod(); return true; } // TODO(Superjomn) replace framework::OpDesc with a lite one. bool ConcatOpLite::AttachImpl(const cpp::OpDesc &op_desc, lite::Scope *scope) { auto inputs = op_desc.Input("X"); auto out = op_desc.Output("Out").front(); param_.x.clear(); for (auto var : inputs) { param_.x.push_back(scope->FindVar(var)->GetMutable()); } CHECK(scope->FindVar(out)); param_.output = scope->FindVar(out)->GetMutable(); param_.axis = op_desc.GetAttr("axis"); std::vector input_arg_names = op_desc.InputArgumentNames(); if (std::find(input_arg_names.begin(), input_arg_names.end(), "AxisTensor") != input_arg_names.end()) { auto arguments = op_desc.Input("AxisTensor"); if (arguments.size() > 0) { auto var = scope->FindVar(arguments.front()); if (var != nullptr) { param_.axis_tensor = var->GetMutable(); } } } return true; } } // namespace operators } // namespace lite } // namespace paddle REGISTER_LITE_OP(concat, paddle::lite::operators::ConcatOpLite);