// Copyright (c) 2010-2023, Lawrence Livermore National Security, LLC. Produced // at the Lawrence Livermore National Laboratory. All Rights reserved. See files // LICENSE and NOTICE for details. LLNL-CODE-806117. // // This file is part of the MFEM library. For more information and source code // availability visit https://mfem.org. // // MFEM is free software; you can redistribute it and/or modify it under the // terms of the BSD-3 license. We welcome feedback and contributions, see file // CONTRIBUTING.md for details. #include "../tmop.hpp" #include "tmop_pa.hpp" #include "../../general/forall.hpp" #include "../../linalg/kernels.hpp" namespace mfem { MFEM_REGISTER_TMOP_KERNELS(void, AssembleDiagonalPA_Kernel_C0_2D, const int NE, const Array &b, const Vector &h0, Vector &diagonal, const int d1d, const int q1d) { constexpr int DIM = 2; const int D1D = T_D1D ? T_D1D : d1d; const int Q1D = T_Q1D ? T_Q1D : q1d; const auto B = Reshape(b.Read(), Q1D, D1D); const auto H0 = Reshape(h0.Read(), DIM, DIM, Q1D, Q1D, NE); auto D = Reshape(diagonal.ReadWrite(), D1D, D1D, DIM, NE); mfem::forall_2D(NE, Q1D, Q1D, [=] MFEM_HOST_DEVICE (int e) { constexpr int DIM = 2; const int D1D = T_D1D ? T_D1D : d1d; const int Q1D = T_Q1D ? T_Q1D : q1d; constexpr int MD1 = T_D1D ? T_D1D : DofQuadLimits::MAX_D1D; constexpr int MQ1 = T_Q1D ? T_Q1D : DofQuadLimits::MAX_Q1D; MFEM_SHARED double qd[MQ1*MD1]; DeviceTensor<2,double> QD(qd, MQ1, MD1); for (int v = 0; v < DIM; v++) { MFEM_FOREACH_THREAD(qx,x,Q1D) { MFEM_FOREACH_THREAD(dy,y,D1D) { QD(qx,dy) = 0.0; MFEM_UNROLL(MQ1) for (int qy = 0; qy < Q1D; ++qy) { const double bb = B(qy,dy) * B(qy,dy); QD(qx,dy) += bb * H0(v,v,qx,qy,e); } } } MFEM_SYNC_THREAD; MFEM_FOREACH_THREAD(dy,y,D1D) { MFEM_FOREACH_THREAD(dx,x,D1D) { double d = 0.0; MFEM_UNROLL(MQ1) for (int qx = 0; qx < Q1D; ++qx) { const double bb = B(qx,dx) * B(qx,dx); d += bb * QD(qx,dy); } D(dx,dy,v,e) += d; } } MFEM_SYNC_THREAD; } }); } void TMOP_Integrator::AssembleDiagonalPA_C0_2D(Vector &D) const { const int N = PA.ne; const int D1D = PA.maps->ndof; const int Q1D = PA.maps->nqpt; const int id = (D1D << 4 ) | Q1D; const Array &B = PA.maps->B; const Vector &H0 = PA.H0; MFEM_LAUNCH_TMOP_KERNEL(AssembleDiagonalPA_Kernel_C0_2D,id,N,B,H0,D); } } // namespace mfem