/* Copyright (c) 2021 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. */ #ifndef _WIN32 #include #endif #include #include #include #include // NOLINT #include #include "gtest/gtest.h" #include "paddle/fluid/framework/op_registry.h" #include "paddle/fluid/framework/operator.h" #include "paddle/fluid/framework/program_desc.h" #include "paddle/fluid/operators/copy_cross_scope_op.cc" #include "paddle/fluid/string/printf.h" #define Conn(x, y) x##y namespace f = paddle::framework; namespace p = paddle::platform; USE_NO_KERNEL_OP(copy_cross_scope); template void Compare1(f::Scope* scope, const p::DeviceContext& ctx, std::string op_type) { // init auto var_x = scope->Var("tmp"); auto x = var_x->GetMutable(); std::vector main_x = {1.0}; TensorFromVector(main_x, ctx, x); auto var_id = scope->Var("Id"); auto id = var_id->GetMutable(); std::vector main_id = {1}; TensorFromVector(main_id, ctx, id); for (int i = 0; i < 3; i++) { auto& child_scope = scope->NewScope(); auto child_var = child_scope.Var("tmp"); auto tensor_x = child_var->GetMutable(); std::vector init_x = {static_cast(i)}; TensorFromVector(init_x, ctx, tensor_x); } ctx.Wait(); // run f::AttributeMap attrs = {{"to_main_scope", false}, {"num_micro_batches", 3}}; std::map> output; auto op = f::OpRegistry::CreateOp(op_type, {{"X", {"tmp"}}, {"Id", {"Id"}}}, output, attrs); auto place = ctx.GetPlace(); op->Run(*scope, place); ctx.Wait(); std::list::const_iterator iter = scope->kids().begin(); iter++; iter++; auto* kid_scope = *iter; auto* dst_var = kid_scope->FindVar("tmp"); auto* tensor_out = dst_var->GetMutable(); std::vector out_vec; TensorToVector(*tensor_out, ctx, &out_vec); int expected = 1; EXPECT_EQ(static_cast(out_vec[0]), expected); } template void Compare2(f::Scope* scope, const p::DeviceContext& ctx, std::string op_type) { // init auto var_x = scope->Var("tmp"); auto x = var_x->GetMutable(); std::vector main_x = {1.0}; TensorFromVector(main_x, ctx, x); auto var_id = scope->Var("Id"); auto id = var_id->GetMutable(); std::vector main_id = {0}; TensorFromVector(main_id, ctx, id); for (int i = 0; i < 3; i++) { auto& child_scope = scope->NewScope(); auto child_var = child_scope.Var("tmp"); auto tensor_x = child_var->GetMutable(); std::vector init_x = {static_cast(i)}; TensorFromVector(init_x, ctx, tensor_x); } ctx.Wait(); // run f::AttributeMap attrs = {{"to_main_scope", true}, {"num_micro_batches", 3}}; std::map> output; auto op = f::OpRegistry::CreateOp(op_type, {{"X", {"tmp"}}, {"Id", {"Id"}}}, output, attrs); auto place = ctx.GetPlace(); op->Run(*scope, place); ctx.Wait(); auto* dst_var = scope->FindVar("tmp"); auto* tensor_out = dst_var->GetMutable(); std::vector out_vec; TensorToVector(*tensor_out, ctx, &out_vec); int expected = 0; EXPECT_EQ(static_cast(out_vec[0]), expected); } #ifdef PADDLE_WITH_CUDA TEST(copy_cross_scope, CUDA_fp32) { f::Scope scope; p::CUDADeviceContext ctx(p::CUDAPlace(0)); Compare1(&scope, ctx, "copy_cross_scope"); } TEST(copy_cross_scope_to_main_scope, CUDA_fp32) { f::Scope scope; p::CUDADeviceContext ctx(p::CUDAPlace(0)); Compare2(&scope, ctx, "copy_cross_scope"); } #elif PADDLE_WITH_ASCEND_CL TEST(copy_cross_scope, NPU_fp32) { f::Scope scope; p::NPUDeviceContext ctx(p::NPUPlace(0)); Compare1(&scope, ctx, "copy_cross_scope"); } TEST(copy_cross_scope_to_main_scope, NPU_fp32) { f::Scope scope; p::NPUDeviceContext ctx(p::NPUPlace(0)); Compare2(&scope, ctx, "copy_cross_scope"); } #endif