// 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 #include "lite/api/paddle_use_kernels.h" #include "lite/api/paddle_use_ops.h" #include "lite/core/arena/framework.h" namespace paddle { namespace lite { class FillConstantComputeTester : public arena::TestCase { protected: // common attributes for this op. std::string out_ = "out"; std::string shape_tensor_ = "shape_tensor"; std::vector shape_tensor_list_{}; std::vector shape_{}; float value_{0.0f}; int dtype_{static_cast(VarDescAPI::VarDataType::FP32)}; bool is_use_shape_tensor_{false}; bool is_use_shape_tensor_list_{false}; // useless for x86, keep it for compatibility bool force_cpu_{false}; public: FillConstantComputeTester(const Place& place, const std::string& alias, std::vector shape, float value, int dtype, const bool is_use_shape_tensor = false, const bool is_use_shape_tensor_list = false) : TestCase(place, alias), shape_(shape), value_(value), dtype_(dtype), is_use_shape_tensor_(is_use_shape_tensor), is_use_shape_tensor_list_(is_use_shape_tensor_list) { if (is_use_shape_tensor_list) { for (int i = 0; i < shape.size(); i++) { shape_tensor_list_.push_back(shape_tensor_ + std::to_string(i)); } } } void RunBaseline(Scope* scope) override { auto* out = scope->NewTensor(out_); std::vector out_shape; if (is_use_shape_tensor_) { auto* shape_tensor = scope->FindTensor(shape_tensor_); auto* shape_tensor_data = shape_tensor->data(); out_shape = std::vector( shape_tensor_data, shape_tensor_data + shape_tensor->numel()); } else if (is_use_shape_tensor_list_) { for (int i = 0; i < shape_tensor_list_.size(); i++) { auto* shape_tensor = scope->FindTensor(shape_tensor_list_[i]); out_shape.push_back(shape_tensor->data()[0]); } } else { out_shape = shape_; } out->Resize(out_shape); auto* output_data = out->mutable_data(); for (int i = 0; i < out->numel(); i++) { output_data[i] = value_; } } void PrepareOpDesc(cpp::OpDesc* op_desc) { op_desc->SetType("fill_constant"); if (is_use_shape_tensor_) { op_desc->SetInput("ShapeTensor", {shape_tensor_}); } else if (is_use_shape_tensor_list_) { op_desc->SetInput("ShapeTensorList", shape_tensor_list_); } else { op_desc->SetAttr("shape", shape_); } op_desc->SetOutput("Out", {out_}); op_desc->SetAttr("dtype", dtype_); op_desc->SetAttr("value", value_); op_desc->SetAttr("force_cpu", force_cpu_); } void PrepareData() override { if (is_use_shape_tensor_) { std::vector dshape_tensor(shape_.begin(), shape_.end()); SetCommonTensor(shape_tensor_, DDim({static_cast(shape_.size())}), dshape_tensor.data()); } if (is_use_shape_tensor_list_) { for (int i = 0; i < shape_.size(); ++i) { std::vector dshape_tensor{static_cast(shape_[i])}; SetCommonTensor(shape_tensor_list_[i], DDim({1}), dshape_tensor.data()); } } } }; void TestFillConstantShape(Place place, float abs_error) { std::vector> out_shapes{ {2, 3, 4, 5}, {2, 3, 4}, {3, 4}, {4}}; for (auto out_shape : out_shapes) { std::unique_ptr tester(new FillConstantComputeTester( place, "def", out_shape, 1.f, static_cast(VarDescAPI::VarDataType::FP32))); arena::Arena arena(std::move(tester), place, abs_error); arena.TestPrecision(); } } void TestFillConstantValue(Place place, float abs_error) { std::vector values{-1., 0., 3.5}; for (auto value : values) { std::unique_ptr tester(new FillConstantComputeTester( place, "def", {2, 3}, value, static_cast(VarDescAPI::VarDataType::FP32))); arena::Arena arena(std::move(tester), place, abs_error); arena.TestPrecision(); } } void TestFillConstantShapeTensor(Place place, float abs_error) { std::unique_ptr tester(new FillConstantComputeTester( place, "def", {2, 3, 4}, 1.f, static_cast(VarDescAPI::VarDataType::FP32), true)); arena::Arena arena(std::move(tester), place, abs_error); arena.TestPrecision(); } void TestFillConstantShapeTensorList(Place place, float abs_error) { std::unique_ptr tester(new FillConstantComputeTester( place, "def", {2, 3, 4}, 1.f, static_cast(VarDescAPI::VarDataType::FP32), false, true)); arena::Arena arena(std::move(tester), place, abs_error); arena.TestPrecision(); } TEST(fill_constant, precision) { LOG(INFO) << "test fill_constant op"; Place place; float abs_error = 1e-5; #if defined(LITE_WITH_NPU) place = TARGET(kNPU); abs_error = 1e-2; // use fp16 in npu #elif defined(LITE_WITH_ARM) place = TARGET(kARM); #else return; #endif TestFillConstantShape(place, abs_error); TestFillConstantValue(place, abs_error); TestFillConstantShapeTensor(place, abs_error); TestFillConstantShapeTensorList(place, abs_error); } } // namespace lite } // namespace paddle