// 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/phi/kernels/funcs/common_shape.h" #include "paddle/phi/kernels/funcs/eigen/common.h" namespace phi { template static void LerpGradFunction(const Context& ctx, const DenseTensor& x, const DenseTensor& y, const DenseTensor& weight, const DenseTensor& out, const DenseTensor& out_grad, DenseTensor* x_grad, DenseTensor* y_grad) { auto& w = weight; auto& dout = out_grad; auto* dx = x_grad; auto* dy = y_grad; auto dout_dims = dout.dims(); DDim dx_dims; DDim dy_dims; auto w_dims = phi::funcs::ExtendDims2Rank(w.dims(), D); Eigen::DSizes dx_bcast_dims; Eigen::DSizes dy_bcast_dims; Eigen::DSizes w_bcast_dims; if (dx) { dx_dims = phi::funcs::ExtendDims2Rank(dx->dims(), D); phi::funcs::GetBroadcastDims(dx_dims, dout_dims, &dx_bcast_dims); } if (dy) { dy_dims = phi::funcs::ExtendDims2Rank(dy->dims(), D); phi::funcs::GetBroadcastDims(dy_dims, dout_dims, &dy_bcast_dims); } phi::funcs::GetBroadcastDims(w_dims, dout_dims, &w_bcast_dims); auto eigen_w = phi::EigenTensor::From(w, w_dims); auto eigen_dout = phi::EigenTensor::From(dout); Eigen::DSizes dx_reshape_dims; Eigen::DSizes dy_reshape_dims; Eigen::DSizes reduce_dims; for (int i = 0; i < dout_dims.size(); ++i) { if (dx) { dx_reshape_dims[2 * i] = dx_bcast_dims[i]; dx_reshape_dims[2 * i + 1] = dx_dims[i]; } if (dy) { dy_reshape_dims[2 * i] = dy_bcast_dims[i]; dy_reshape_dims[2 * i + 1] = dy_dims[i]; } reduce_dims[i] = 2 * i; } auto& place = *ctx.eigen_device(); if (dx) { ctx.template Alloc(dx); auto eigen_dx = phi::EigenTensor::From(*dx, dx_dims); auto eigen_expr = (1 - eigen_w.broadcast(w_bcast_dims)) * eigen_dout; eigen_dx.device(place) = eigen_expr.reshape(dx_reshape_dims) .sum(reduce_dims) .reshape(eigen_dx.dimensions()); } if (dy) { ctx.template Alloc(dy); auto eigen_dy = phi::EigenTensor::From(*dy, dy_dims); auto eigen_expr = eigen_w.broadcast(w_bcast_dims) * eigen_dout; eigen_dy.device(place) = eigen_expr.reshape(dy_reshape_dims) .sum(reduce_dims) .reshape(eigen_dy.dimensions()); } } template void LerpGradKernel(const Context& ctx, const DenseTensor& x, const DenseTensor& y, const DenseTensor& weight, const DenseTensor& out, const DenseTensor& out_grad, DenseTensor* x_grad, DenseTensor* y_grad) { int rank = out.dims().size(); PADDLE_ENFORCE_GE( rank, 1, phi::errors::InvalidArgument( "The number of dimensions for LerpGradOp must be " "greater than or equal to 1, but the value received is %d.", rank)); PADDLE_ENFORCE_LE( rank, 6, phi::errors::InvalidArgument( "The number of dimensions for LerpGradOp must be " "less than or equal to 6, but the value received is %d.", rank)); switch (rank) { case 1: LerpGradFunction( ctx, x, y, weight, out, out_grad, x_grad, y_grad); break; case 2: LerpGradFunction( ctx, x, y, weight, out, out_grad, x_grad, y_grad); break; case 3: LerpGradFunction( ctx, x, y, weight, out, out_grad, x_grad, y_grad); break; case 4: LerpGradFunction( ctx, x, y, weight, out, out_grad, x_grad, y_grad); break; case 5: LerpGradFunction( ctx, x, y, weight, out, out_grad, x_grad, y_grad); break; case 6: LerpGradFunction( ctx, x, y, weight, out, out_grad, x_grad, y_grad); break; } } } // namespace phi