// 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 "../config/config.hpp" #ifdef MFEM_USE_MPI #include "fem.hpp" #include "../general/forall.hpp" namespace mfem { ParNonlinearForm::ParNonlinearForm(ParFiniteElementSpace *pf) : NonlinearForm(pf), pGrad(Operator::Hypre_ParCSR) { X.MakeRef(pf, NULL); Y.MakeRef(pf, NULL); MFEM_VERIFY(!Serial(), "internal MFEM error"); } double ParNonlinearForm::GetParGridFunctionEnergy(const Vector &x) const { double loc_energy, glob_energy; loc_energy = GetGridFunctionEnergy(x); if (fnfi.Size()) { MFEM_ABORT("TODO: add energy contribution from shared faces"); } MPI_Allreduce(&loc_energy, &glob_energy, 1, MPI_DOUBLE, MPI_SUM, ParFESpace()->GetComm()); return glob_energy; } void ParNonlinearForm::Mult(const Vector &x, Vector &y) const { NonlinearForm::Mult(x, y); // x --(P)--> aux1 --(A_local)--> aux2 if (fnfi.Size()) { MFEM_VERIFY(!NonlinearForm::ext, "Not implemented (extensions + faces"); // Terms over shared interior faces in parallel. ParFiniteElementSpace *pfes = ParFESpace(); ParMesh *pmesh = pfes->GetParMesh(); FaceElementTransformations *tr; const FiniteElement *fe1, *fe2; Array vdofs1, vdofs2; Vector el_x, el_y; aux1.HostReadWrite(); X.MakeRef(aux1, 0); // aux1 contains P.x X.ExchangeFaceNbrData(); const int n_shared_faces = pmesh->GetNSharedFaces(); for (int i = 0; i < n_shared_faces; i++) { tr = pmesh->GetSharedFaceTransformations(i, true); int Elem2NbrNo = tr->Elem2No - pmesh->GetNE(); fe1 = pfes->GetFE(tr->Elem1No); fe2 = pfes->GetFaceNbrFE(Elem2NbrNo); pfes->GetElementVDofs(tr->Elem1No, vdofs1); pfes->GetFaceNbrElementVDofs(Elem2NbrNo, vdofs2); el_x.SetSize(vdofs1.Size() + vdofs2.Size()); X.GetSubVector(vdofs1, el_x.GetData()); X.FaceNbrData().GetSubVector(vdofs2, el_x.GetData() + vdofs1.Size()); for (int k = 0; k < fnfi.Size(); k++) { fnfi[k]->AssembleFaceVector(*fe1, *fe2, *tr, el_x, el_y); aux2.AddElementVector(vdofs1, el_y.GetData()); } } } P->MultTranspose(aux2, y); const int N = ess_tdof_list.Size(); const auto idx = ess_tdof_list.Read(); auto Y_RW = y.ReadWrite(); mfem::forall(N, [=] MFEM_HOST_DEVICE (int i) { Y_RW[idx[i]] = 0.0; }); } const SparseMatrix &ParNonlinearForm::GetLocalGradient(const Vector &x) const { MFEM_VERIFY(NonlinearForm::ext == nullptr, "this method is not supported yet with partial assembly"); NonlinearForm::GetGradient(x); // (re)assemble Grad, no b.c. return *Grad; } Operator &ParNonlinearForm::GetGradient(const Vector &x) const { if (NonlinearForm::ext) { return NonlinearForm::GetGradient(x); } ParFiniteElementSpace *pfes = ParFESpace(); pGrad.Clear(); NonlinearForm::GetGradient(x); // (re)assemble Grad, no b.c. OperatorHandle dA(pGrad.Type()), Ph(pGrad.Type()); if (fnfi.Size() == 0) { dA.MakeSquareBlockDiag(pfes->GetComm(), pfes->GlobalVSize(), pfes->GetDofOffsets(), Grad); } else { MFEM_ABORT("TODO: assemble contributions from shared face terms"); } // RAP the local gradient dA. // TODO - construct Dof_TrueDof_Matrix directly in the pGrad format Ph.ConvertFrom(pfes->Dof_TrueDof_Matrix()); pGrad.MakePtAP(dA, Ph); // Impose b.c. on pGrad OperatorHandle pGrad_e; pGrad_e.EliminateRowsCols(pGrad, ess_tdof_list); return *pGrad.Ptr(); } void ParNonlinearForm::Update() { Y.MakeRef(ParFESpace(), NULL); X.MakeRef(ParFESpace(), NULL); pGrad.Clear(); NonlinearForm::Update(); } ParBlockNonlinearForm::ParBlockNonlinearForm(Array &pf) : BlockNonlinearForm() { pBlockGrad = NULL; SetParSpaces(pf); } void ParBlockNonlinearForm::SetParSpaces(Array &pf) { delete pBlockGrad; pBlockGrad = NULL; for (int s1=0; s1 serialSpaces(pf.Size()); for (int s=0; s *>&bdr_attr_is_ess, Array &rhs) { Array nullarray(fes.Size()); nullarray = NULL; BlockNonlinearForm::SetEssentialBC(bdr_attr_is_ess, nullarray); for (int s = 0; s < fes.Size(); ++s) { if (rhs[s]) { rhs[s]->SetSubVector(*ess_tdofs[s], 0.0); } } } double ParBlockNonlinearForm::GetEnergy(const Vector &x) const { // xs_true is not modified, so const_cast is okay xs_true.Update(const_cast(x), block_trueOffsets); xs.Update(block_offsets); for (int s = 0; s < fes.Size(); ++s) { fes[s]->GetProlongationMatrix()->Mult(xs_true.GetBlock(s), xs.GetBlock(s)); } double enloc = BlockNonlinearForm::GetEnergyBlocked(xs); double englo = 0.0; MPI_Allreduce(&enloc, &englo, 1, MPI_DOUBLE, MPI_SUM, ParFESpace(0)->GetComm()); return englo; } void ParBlockNonlinearForm::Mult(const Vector &x, Vector &y) const { // xs_true is not modified, so const_cast is okay xs_true.Update(const_cast(x), block_trueOffsets); ys_true.Update(y, block_trueOffsets); xs.Update(block_offsets); ys.Update(block_offsets); for (int s=0; sGetProlongationMatrix()->Mult( xs_true.GetBlock(s), xs.GetBlock(s)); } BlockNonlinearForm::MultBlocked(xs, ys); if (fnfi.Size() > 0) { MFEM_ABORT("TODO: assemble contributions from shared face terms"); } for (int s=0; sGetProlongationMatrix()->MultTranspose( ys.GetBlock(s), ys_true.GetBlock(s)); ys_true.GetBlock(s).SetSubVector(*ess_tdofs[s], 0.0); } ys_true.SyncFromBlocks(); y.SyncMemory(ys_true); } /// Return the local gradient matrix for the given true-dof vector x const BlockOperator & ParBlockNonlinearForm::GetLocalGradient( const Vector &x) const { // xs_true is not modified, so const_cast is okay xs_true.Update(const_cast(x), block_trueOffsets); xs.Update(block_offsets); for (int s=0; sGetProlongationMatrix()->Mult( xs_true.GetBlock(s), xs.GetBlock(s)); } // (re)assemble Grad without b.c. into 'Grads' BlockNonlinearForm::ComputeGradientBlocked(xs); delete BlockGrad; BlockGrad = new BlockOperator(block_offsets); for (int i = 0; i < fes.Size(); ++i) { for (int j = 0; j < fes.Size(); ++j) { BlockGrad->SetBlock(i, j, Grads(i, j)); } } return *BlockGrad; } // Set the operator type id for the parallel gradient matrix/operator. void ParBlockNonlinearForm::SetGradientType(Operator::Type tid) { for (int s1=0; s1SetType(tid); } } } BlockOperator & ParBlockNonlinearForm::GetGradient(const Vector &x) const { if (pBlockGrad == NULL) { pBlockGrad = new BlockOperator(block_trueOffsets); } Array pfes(fes.Size()); for (int s1=0; s1Clear(); } } GetLocalGradient(x); // gradients are stored in 'Grads' if (fnfi.Size() > 0) { MFEM_ABORT("TODO: assemble contributions from shared face terms"); } for (int s1=0; s1Type()), Ph(phBlockGrad(s1,s2)->Type()), Rh(phBlockGrad(s1,s2)->Type()); if (s1 == s2) { dA.MakeSquareBlockDiag(pfes[s1]->GetComm(), pfes[s1]->GlobalVSize(), pfes[s1]->GetDofOffsets(), Grads(s1,s1)); Ph.ConvertFrom(pfes[s1]->Dof_TrueDof_Matrix()); phBlockGrad(s1,s1)->MakePtAP(dA, Ph); OperatorHandle Ae; Ae.EliminateRowsCols(*phBlockGrad(s1,s1), *ess_tdofs[s1]); } else { dA.MakeRectangularBlockDiag(pfes[s1]->GetComm(), pfes[s1]->GlobalVSize(), pfes[s2]->GlobalVSize(), pfes[s1]->GetDofOffsets(), pfes[s2]->GetDofOffsets(), Grads(s1,s2)); Rh.ConvertFrom(pfes[s1]->Dof_TrueDof_Matrix()); Ph.ConvertFrom(pfes[s2]->Dof_TrueDof_Matrix()); phBlockGrad(s1,s2)->MakeRAP(Rh, dA, Ph); phBlockGrad(s1,s2)->EliminateRows(*ess_tdofs[s1]); phBlockGrad(s1,s2)->EliminateCols(*ess_tdofs[s2]); } pBlockGrad->SetBlock(s1, s2, phBlockGrad(s1,s2)->Ptr()); } } return *pBlockGrad; } ParBlockNonlinearForm::~ParBlockNonlinearForm() { delete pBlockGrad; for (int s1=0; s1