// Copyright (c) 2022 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. #pragma once #include "paddle/fluid/framework/tensor_util.h" #include "paddle/phi/core/dense_tensor.h" #include "paddle/phi/kernels/copy_kernel.h" #include "paddle/phi/kernels/funcs/eigen/common.h" #include "paddle/phi/kernels/funcs/eigen/eigen_function.h" #include "paddle/phi/kernels/meshgrid_kernel.h" namespace phi { template void MeshgridForward(const Context& ctx, const std::vector& ins, std::vector outs) { PADDLE_ENFORCE_EQ( ins.size() > 1, true, phi::errors::InvalidArgument( "Expected at least 2 input tensors, but only received d%.", ins.size())); int64_t size = ins.size(); std::vector shape(size); for (int64_t i = 0; i < size; i++) { switch (ins[i]->dims().size()) { case 0: shape[i] = 1; break; case 1: shape[i] = ins[i]->dims()[0]; break; default: PADDLE_THROW(phi::errors::InvalidArgument( "Expected scalar or 1D tensor in the tensor list but got tensor " "%d: ", i)); } } for (int64_t i = 0; i < size; i++) { std::vector view_shape(size, 1); view_shape[i] = shape[i]; DenseTensor reshape_ins_tensor; paddle::framework::TensorCopy( *ins[i], ctx.GetPlace(), ctx, &reshape_ins_tensor); DDim out_dims_reshape = phi::make_ddim(view_shape); reshape_ins_tensor.Resize(out_dims_reshape); DDim out_dims = phi::make_ddim(shape); Eigen::DSizes bcast_dims; for (int64_t j = 0; j < size; j++) { bcast_dims[j] = shape[j]; } bcast_dims[i] = 1; outs[i]->Resize(out_dims); auto x = EigenTensor::From( static_cast(reshape_ins_tensor)); ctx.template Alloc(outs[i]); auto y = EigenTensor::From(*outs[i]); auto& place = *ctx.eigen_device(); funcs::EigenBroadcast, T, Rank>::Eval( place, y, x, bcast_dims); } } template void MeshgridKernel(const Context& ctx, const std::vector& inputs, std::vector outputs) { int rank = inputs.size(); switch (rank) { case 1: MeshgridForward(ctx, inputs, outputs); break; case 2: MeshgridForward(ctx, inputs, outputs); break; case 3: MeshgridForward(ctx, inputs, outputs); break; case 4: MeshgridForward(ctx, inputs, outputs); break; case 5: MeshgridForward(ctx, inputs, outputs); break; case 6: MeshgridForward(ctx, inputs, outputs); break; default: PADDLE_THROW(phi::errors::InvalidArgument( "Excepted Tensor numbers between 1 and 6, but only received d% .", rank)); } } } // namespace phi