Line data Source code
1 : //* This file is part of the MOOSE framework 2 : //* https://mooseframework.inl.gov 3 : //* 4 : //* All rights reserved, see COPYRIGHT for full restrictions 5 : //* https://github.com/idaholab/moose/blob/master/COPYRIGHT 6 : //* 7 : //* Licensed under LGPL 2.1, please see LICENSE for details 8 : //* https://www.gnu.org/licenses/lgpl-2.1.html 9 : 10 : #ifdef MOOSE_MFEM_ENABLED 11 : 12 : #include "MFEMCGSolver.h" 13 : #include "MFEMProblem.h" 14 : 15 : registerMooseObject("MooseApp", MFEMCGSolver); 16 : 17 : InputParameters 18 2236 : MFEMCGSolver::validParams() 19 : { 20 2236 : InputParameters params = Moose::MFEM::LinearSolverBase::validParams(); 21 4472 : params.addClassDescription("MFEM native solver for the iterative solution of MFEM equation " 22 : "systems using the conjugate gradient method."); 23 4472 : params.set<bool>("use_initial_guess", /*quiet_mode=*/true) = true; 24 8944 : params.addParam<mfem::real_t>("l_tol", 1e-5, "Set the relative tolerance."); 25 8944 : params.addParam<mfem::real_t>("l_abs_tol", 1e-50, "Set the absolute tolerance."); 26 8944 : params.addParam<int>("l_max_its", 10000, "Set the maximum number of iterations."); 27 8944 : params.addParam<int>("print_level", 2, "Set the solver verbosity."); 28 6708 : params.addParam<MFEMSolverName>("preconditioner", "Optional choice of preconditioner to use."); 29 : 30 2236 : return params; 31 0 : } 32 : 33 69 : MFEMCGSolver::MFEMCGSolver(const InputParameters & parameters) 34 69 : : Moose::MFEM::LinearSolverBase(parameters) 35 : { 36 69 : ConstructSolver(); 37 69 : } 38 : 39 : void 40 69 : MFEMCGSolver::ConstructSolver() 41 : { 42 69 : auto solver = std::make_unique<mfem::CGSolver>(getMFEMProblem().getComm()); 43 138 : solver->iterative_mode = getParam<bool>("use_initial_guess"); 44 138 : solver->SetRelTol(getParam<mfem::real_t>("l_tol")); 45 138 : solver->SetAbsTol(getParam<mfem::real_t>("l_abs_tol")); 46 138 : solver->SetMaxIter(getParam<int>("l_max_its")); 47 138 : solver->SetPrintLevel(getParam<int>("print_level")); 48 69 : SetPreconditioner(*solver); 49 69 : _solver = std::move(solver); 50 69 : } 51 : 52 : void 53 49 : MFEMCGSolver::SetupLOR(mfem::ParBilinearForm & a, mfem::Array<int> & ess_bdr_markers) 54 : { 55 49 : if (_lor && _preconditioner) 56 0 : mooseError("LOR solver cannot take a preconditioner"); 57 : 58 49 : mfem::Array<int> ess_tdofs; 59 49 : a.ParFESpace()->GetEssentialTrueDofs(ess_bdr_markers, ess_tdofs); 60 49 : if (_preconditioner) 61 : { 62 45 : _preconditioner->SetupLOR(a, ess_tdofs); 63 45 : SetPreconditioner(static_cast<mfem::CGSolver &>(*_solver)); 64 : } 65 4 : else if (_lor) 66 : { 67 2 : CheckSpectralEquivalence(a); 68 2 : auto lor_solver = new mfem::LORSolver<mfem::CGSolver>(a, ess_tdofs); 69 4 : lor_solver->GetSolver().SetRelTol(getParam<mfem::real_t>("l_tol")); 70 4 : lor_solver->GetSolver().SetAbsTol(getParam<mfem::real_t>("l_abs_tol")); 71 4 : lor_solver->GetSolver().SetMaxIter(getParam<int>("l_max_its")); 72 4 : lor_solver->GetSolver().SetPrintLevel(getParam<int>("print_level")); 73 : 74 2 : _solver.reset(lor_solver); 75 : } 76 49 : } 77 : 78 : #endif