// 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. #include "paddle/phi/kernels/gather_kernel.h" #include "paddle/phi/backends/xpu/enforce_xpu.h" #include "paddle/phi/core/kernel_registry.h" namespace phi { template void GatherKernel(const Context& dev_ctx, const DenseTensor& x, const DenseTensor& index, const Scalar& axis, DenseTensor* out) { auto axis_v = axis.to(); const auto& index_type = index.dtype(); dev_ctx.template Alloc(out); if (x.numel() == 0) return; const auto index_dims = index.dims(); if (index_dims.size() == 2) { PADDLE_ENFORCE_EQ( index_dims[1], 1, phi::errors::InvalidArgument( "The last dim of index should be 1 when it is 2D, but we get %d", index_dims[1])); } else { PADDLE_ENFORCE_EQ( index_dims.size() == 1 || index_dims.size() == 0, true, phi::errors::InvalidArgument( "The index should be 0D, 1D, when it is not 2D, but we get %d", index_dims.size())); } std::vector xshape(x.dims().size()); for (int i = 0; i < x.dims().size(); ++i) { xshape[i] = x.dims()[i]; } using XPUType = typename XPUTypeTrait::Type; int r = XPU_SUCCESS; if (index_type == DataType::INT32) { r = xpu::gather( dev_ctx.x_context(), reinterpret_cast(x.data()), index.data(), reinterpret_cast(out->data()), xshape, index.dims().size() == 0 ? 1 : index.dims()[0], axis_v); } else { r = xpu::gather( dev_ctx.x_context(), reinterpret_cast(x.data()), index.data(), reinterpret_cast(out->data()), xshape, index.dims().size() == 0 ? 1 : index.dims()[0], axis_v); } PADDLE_ENFORCE_EQ( r, xpu::Error_t::SUCCESS, phi::errors::External( "XPU gather kernel return wrong value[%d %s]", r, XPUAPIErrorMsg[r])); } } // namespace phi PD_REGISTER_KERNEL(gather, XPU, ALL_LAYOUT, phi::GatherKernel, float, phi::dtype::float16, int, int64_t, bool) {}