/* 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. */ #pragma once #include "paddle/fluid/framework/op_registry.h" #include "paddle/fluid/operators/solve_op.h" #include "paddle/fluid/operators/triangular_solve_op.h" #include "paddle/fluid/platform/complex.h" #include "paddle/phi/kernels/funcs/lapack/lapack_function.h" #include "paddle/phi/kernels/math_kernel.h" #include "paddle/phi/kernels/transpose_kernel.h" namespace paddle { namespace operators { // namespace operators template class CholeskySolveFunctor { public: void operator()(const platform::DeviceContext &dev_ctx, bool upper, int n, int nrhs, T *Adata, int lda, T *Bdata, int *devInfo); }; template class CholeskySolveFunctor { public: void operator()(const platform::CPUDeviceContext &dev_ctx, bool upper, int n, int nrhs, T *Adata, int lda, T *Bdata, int *devInfo) { char uplo = upper ? 'U' : 'L'; phi::funcs::lapackCholeskySolve(uplo, n, nrhs, Adata, lda, Bdata, lda, devInfo); } }; template void cholesky_solve_fn(const paddle::framework::ExecutionContext &ctx, const framework::Tensor &uin, const framework::Tensor &bin, framework::Tensor *out, bool upper) { const auto &dev_ctx = ctx.template device_context(); // framework::Tensor broadcast std::vector u_bst_dims_vec; std::vector b_bst_dims_vec; std::tie(u_bst_dims_vec, b_bst_dims_vec) = get_broadcast_dims(uin, bin); framework::Tensor u_bst(uin.type()); TensorExpand(dev_ctx, uin, &u_bst, u_bst_dims_vec); framework::Tensor b_bst(bin.type()); TensorExpand(dev_ctx, bin, &b_bst, b_bst_dims_vec); auto &phi_dev_ctx = static_cast< const typename framework::ConvertToPhiContext::TYPE &>( dev_ctx); // calculate u's conjugate for complex framework::Tensor u_conj(u_bst.type()); platform::ForRange u_for_range(dev_ctx, u_bst.numel()); phi::funcs::ConjFunctor u_functor( u_bst.data(), u_bst.numel(), u_conj.mutable_data(u_bst.dims(), dev_ctx.GetPlace())); u_for_range(u_functor); u_conj = phi::TransposeLast2Dim(phi_dev_ctx, u_conj); // calculate b's conjugate for complex framework::Tensor b_conj(b_bst.type()); platform::ForRange b_for_range(dev_ctx, b_bst.numel()); phi::funcs::ConjFunctor b_functor( b_bst.data(), b_bst.numel(), b_conj.mutable_data(b_bst.dims(), dev_ctx.GetPlace())); b_for_range(b_functor); b_conj = phi::TransposeLast2Dim(phi_dev_ctx, b_conj); auto ut_data = u_conj.mutable_data(dev_ctx.GetPlace()); auto uindims = u_bst.dims(); auto bindims = b_bst.dims(); int uinrank = uindims.size(); int binrank = bindims.size(); int n = uindims[uinrank - 2]; int nrhs = bindims[binrank - 1]; int ldab = std::max(1, n); // framework::Tensor out_copy(b_conj.type()); // out_copy.Resize(b_conj.dims()); framework::TensorCopy(b_conj, dev_ctx.GetPlace(), out); T *out_data = out->mutable_data(dev_ctx.GetPlace()); auto info_dims = phi::slice_ddim(bindims, 0, binrank - 2); auto batchsize = product(info_dims); framework::Tensor tmp; std::vector tmpdim(1, batchsize); tmp.Resize(phi::make_ddim(tmpdim)); int *info = tmp.mutable_data(dev_ctx.GetPlace()); CholeskySolveFunctor functor; for (int b = 0; b < batchsize; b++) { auto uin_data_item = &ut_data[b * n * n]; auto out_data_item = &out_data[b * n * nrhs]; auto info_item = &info[b]; functor(dev_ctx, upper, n, nrhs, uin_data_item, ldab, out_data_item, info_item); } // calculate out's conjugate for complex platform::ForRange out_for_range(dev_ctx, out->numel()); phi::funcs::ConjFunctor out_functor( out->data(), out->numel(), out->mutable_data(out->dims(), dev_ctx.GetPlace())); out_for_range(out_functor); *out = phi::TransposeLast2Dim(phi_dev_ctx, *out); } template class CholeskySolveKernel : public framework::OpKernel { public: void Compute(const paddle::framework::ExecutionContext &ctx) const override { auto *uin = ctx.Input("Y"); auto *bin = ctx.Input("X"); auto *out = ctx.Output("Out"); auto upper = ctx.Attr("upper"); cholesky_solve_fn(ctx, *uin, *bin, out, upper); } }; template class CholeskySolveGradKernel : public framework::OpKernel { public: void Compute(const framework::ExecutionContext &ctx) const override { auto *bin = ctx.Input("X"); auto *uin = ctx.Input("Y"); auto *out = ctx.Input("Out"); auto *dout = ctx.Input(framework::GradVarName("Out")); auto *db = ctx.Output(framework::GradVarName("X")); auto *du = ctx.Output(framework::GradVarName("Y")); auto upper = ctx.Attr("upper"); const auto &dev_ctx = ctx.template device_context(); auto &phi_dev_ctx = static_cast< const typename framework::ConvertToPhiContext::TYPE &>( dev_ctx); std::vector u_bst_dims_vec; std::vector b_bst_dims_vec; std::tie(u_bst_dims_vec, b_bst_dims_vec) = get_broadcast_dims(*uin, *bin); framework::Tensor u_bst(uin->type()); TensorExpand(dev_ctx, *uin, &u_bst, u_bst_dims_vec); framework::Tensor db_bst(bin->type()); TensorExpand(dev_ctx, *bin, &db_bst, b_bst_dims_vec); if (dout) { db->mutable_data(dev_ctx.GetPlace()); cholesky_solve_fn(ctx, u_bst, *dout, &db_bst, upper); if (db_bst.dims() == db->dims()) { framework::TensorCopy(db_bst, dev_ctx.GetPlace(), dev_ctx, db); } else { MatrixReduceSumFunctor functor; functor(db_bst, db, ctx); db->Resize(bin->dims()); } auto blas = phi::funcs::GetBlas(ctx); // calculate out's conjugate for complex framework::Tensor out_conj(out->type()); platform::ForRange out_for_range(dev_ctx, out->numel()); phi::funcs::ConjFunctor out_functor( out->data(), out->numel(), out_conj.mutable_data(out->dims(), dev_ctx.GetPlace())); out_for_range(out_functor); out_conj = phi::TransposeLast2Dim(phi_dev_ctx, out_conj); framework::Tensor commonterm(out->type()); auto outdims = out_conj.dims(); auto dbdims = db_bst.dims(); auto mat_dim_a = phi::funcs::CreateMatrixDescriptor(outdims, 0, false); auto mat_dim_b = phi::funcs::CreateMatrixDescriptor(dbdims, 0, false); auto cmtdim = outdims; cmtdim[cmtdim.size() - 2] = dbdims[dbdims.size() - 2]; commonterm.Resize(cmtdim); commonterm.mutable_data(dev_ctx.GetPlace()); blas.MatMul(db_bst, mat_dim_b, out_conj, mat_dim_a, static_cast(1), &commonterm, static_cast(0)); // calculate commonterm's conjugate for complex framework::Tensor commonterm_conj(commonterm.type()); platform::ForRange commonterm_for_range( dev_ctx, commonterm.numel()); phi::funcs::ConjFunctor commonterm_functor( commonterm.data(), commonterm.numel(), commonterm_conj.mutable_data(commonterm.dims(), dev_ctx.GetPlace())); commonterm_for_range(commonterm_functor); commonterm_conj = phi::TransposeLast2Dim(phi_dev_ctx, commonterm_conj); phi::AddRawKernel( static_cast::TYPE &>(dev_ctx), commonterm, commonterm_conj, -1, &commonterm); auto mat_dim_u = phi::funcs::CreateMatrixDescriptor(u_bst.dims(), 0, false); auto mat_dim_c = phi::funcs::CreateMatrixDescriptor(commonterm.dims(), 0, false); Tensor du_bst(uin->type()); // get upper or lower triangular du_bst.Resize(u_bst.dims()); du_bst.mutable_data(dev_ctx.GetPlace()); if (upper) { blas.MatMul(u_bst, mat_dim_u, commonterm, mat_dim_c, static_cast(-1), &du_bst, static_cast(0)); } else { blas.MatMul(commonterm, mat_dim_c, u_bst, mat_dim_u, static_cast(-1), &du_bst, static_cast(0)); } const auto &udims = u_bst.dims(); const auto H = udims[udims.size() - 2]; const auto W = udims[udims.size() - 1]; platform::ForRange x_for_range(dev_ctx, u_bst.numel()); TrilTriuCompute tril_triu_computer(du_bst.data(), 0, !upper, H, W, u_bst.data()); x_for_range(tril_triu_computer); du->mutable_data(dev_ctx.GetPlace()); if (u_bst.dims() == du->dims()) { framework::TensorCopy(u_bst, dev_ctx.GetPlace(), dev_ctx, du); } else { MatrixReduceSumFunctor functor; functor(u_bst, du, ctx); du->Resize(uin->dims()); } } } }; } // namespace operators } // namespace paddle