LCOV - code coverage report
Current view: top level - src/convergence - DefaultNonlinearConvergence.C (source / functions) Hit Total Coverage
Test: idaholab/moose framework: #33187 (5aa0b2) with base d7c4bd Lines: 107 119 89.9 %
Date: 2026-06-30 12:18:20 Functions: 5 5 100.0 %
Legend: Lines: hit not hit

          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             : // MOOSE includes
      11             : #include "DefaultNonlinearConvergence.h"
      12             : #include "FEProblemBase.h"
      13             : #include "PetscSupport.h"
      14             : #include "NonlinearSystemBase.h"
      15             : #include "ConvergenceIterationTypes.h"
      16             : 
      17             : #include "libmesh/equation_systems.h"
      18             : 
      19             : // PETSc includes
      20             : #include <petsc.h>
      21             : #include <petscmat.h>
      22             : #include <petscsnes.h>
      23             : 
      24             : registerMooseObject("MooseApp", DefaultNonlinearConvergence);
      25             : 
      26             : InputParameters
      27      131933 : DefaultNonlinearConvergence::validParams()
      28             : {
      29      131933 :   InputParameters params = DefaultConvergenceBase::validParams();
      30      131933 :   params += FEProblemSolve::feProblemDefaultConvergenceParams();
      31             : 
      32      395799 :   params.addPrivateParam<bool>("added_as_default", false);
      33             : 
      34      131933 :   params.addClassDescription("Default convergence criteria for FEProblem.");
      35             : 
      36      131933 :   return params;
      37           0 : }
      38             : 
      39       65810 : DefaultNonlinearConvergence::DefaultNonlinearConvergence(const InputParameters & parameters)
      40             :   : DefaultConvergenceBase(parameters),
      41      197430 :     _fe_problem(*getCheckedPointerParam<FEProblemBase *>("_fe_problem_base")),
      42      131620 :     _nl_abs_div_tol(getSharedExecutionerParam<Real>("nl_abs_div_tol")),
      43      131620 :     _nl_rel_div_tol(getSharedExecutionerParam<Real>("nl_div_tol")),
      44       65810 :     _div_threshold(std::numeric_limits<Real>::max()),
      45      131620 :     _nl_forced_its(getSharedExecutionerParam<unsigned int>("nl_forced_its")),
      46      131620 :     _nl_max_pingpong(getSharedExecutionerParam<unsigned int>("n_max_nonlinear_pingpong")),
      47       65810 :     _nl_current_pingpong(0)
      48             : {
      49       65810 :   EquationSystems & es = _fe_problem.es();
      50             : 
      51       65810 :   es.parameters.set<unsigned int>("nonlinear solver maximum iterations") =
      52      197430 :       getSharedExecutionerParam<unsigned int>("nl_max_its");
      53       65810 :   es.parameters.set<unsigned int>("nonlinear solver maximum function evaluations") =
      54      197430 :       getSharedExecutionerParam<unsigned int>("nl_max_funcs");
      55       65810 :   es.parameters.set<Real>("nonlinear solver absolute residual tolerance") =
      56      197430 :       getSharedExecutionerParam<Real>("nl_abs_tol");
      57       65810 :   es.parameters.set<Real>("nonlinear solver relative residual tolerance") =
      58      197430 :       getSharedExecutionerParam<Real>("nl_rel_tol");
      59       65810 :   es.parameters.set<Real>("nonlinear solver divergence tolerance") =
      60      197430 :       getSharedExecutionerParam<Real>("nl_div_tol");
      61       65810 :   es.parameters.set<Real>("nonlinear solver absolute step tolerance") =
      62      197430 :       getSharedExecutionerParam<Real>("nl_abs_step_tol");
      63       65810 :   es.parameters.set<Real>("nonlinear solver relative step tolerance") =
      64      197430 :       getSharedExecutionerParam<Real>("nl_rel_step_tol");
      65       65810 : }
      66             : 
      67             : void
      68       56162 : DefaultNonlinearConvergence::checkIterationType(IterationType it_type) const
      69             : {
      70       56162 :   DefaultConvergenceBase::checkIterationType(it_type);
      71             : 
      72       56162 :   if (it_type != ConvergenceIterationTypes::NONLINEAR)
      73           3 :     mooseError("DefaultNonlinearConvergence can only be used with nonlinear solves.");
      74       56159 : }
      75             : 
      76             : bool
      77      729746 : DefaultNonlinearConvergence::checkResidualConvergence(const unsigned int iter,
      78             :                                                       const Real fnorm,
      79             :                                                       const Real ref_norm,
      80             :                                                       const Real rel_tol,
      81             :                                                       const Real abs_tol,
      82             :                                                       std::ostringstream & oss)
      83             : {
      84      729746 :   if (fnorm < abs_tol)
      85             :   {
      86      143850 :     oss << "Converged due to absolute residual " << fnorm << " < absolute tolerance (" << abs_tol
      87      143850 :         << ")\n";
      88      143850 :     return true;
      89             :   }
      90      585896 :   else if (iter && fnorm <= ref_norm * rel_tol)
      91             :   {
      92      139484 :     oss << "Converged due to relative/normalized residual norm " << fnorm / ref_norm
      93      139484 :         << " < relative tolerance (" << rel_tol << ")\n";
      94      139484 :     return true;
      95             :   }
      96             :   else
      97      446412 :     return false;
      98             : }
      99             : 
     100             : Convergence::MooseConvergenceStatus
     101      748785 : DefaultNonlinearConvergence::checkConvergence(unsigned int iter)
     102             : {
     103      748785 :   TIME_SECTION(_perfid_check_convergence);
     104             : 
     105      748785 :   NonlinearSystemBase & system = _fe_problem.currentNonlinearSystem();
     106      748785 :   MooseConvergenceStatus status = MooseConvergenceStatus::ITERATING;
     107             : 
     108             :   // Needed by ResidualReferenceConvergence
     109      748785 :   nonlinearConvergenceSetup();
     110             : 
     111      748785 :   SNES snes = system.getSNES();
     112             : 
     113             :   // ||u||
     114             :   PetscReal xnorm;
     115      748785 :   LibmeshPetscCallA(_fe_problem.comm().get(), SNESGetSolutionNorm(snes, &xnorm));
     116             : 
     117             :   // ||r||
     118             :   PetscReal fnorm;
     119      748785 :   LibmeshPetscCallA(_fe_problem.comm().get(), SNESGetFunctionNorm(snes, &fnorm));
     120             : 
     121             :   // ||du||
     122             :   PetscReal snorm;
     123      748785 :   LibmeshPetscCallA(_fe_problem.comm().get(), SNESGetUpdateNorm(snes, &snorm));
     124             : 
     125             :   // Get current number of function evaluations done by SNES
     126             :   PetscInt nfuncs;
     127      748785 :   LibmeshPetscCallA(_fe_problem.comm().get(), SNESGetNumberFunctionEvals(snes, &nfuncs));
     128             : 
     129             :   // Get tolerances from SNES
     130             :   PetscReal abs_tol, rel_tol, rel_step_tol;
     131             :   PetscInt max_its, max_funcs;
     132      748785 :   LibmeshPetscCallA(
     133             :       _fe_problem.comm().get(),
     134             :       SNESGetTolerances(snes, &abs_tol, &rel_tol, &rel_step_tol, &max_its, &max_funcs));
     135             : 
     136             : #if !PETSC_VERSION_LESS_THAN(3, 8, 4)
     137      748785 :   PetscBool force_iteration = PETSC_FALSE;
     138      748785 :   LibmeshPetscCallA(_fe_problem.comm().get(), SNESGetForceIteration(snes, &force_iteration));
     139             : 
     140             :   // if PETSc says to force iteration, then force at least one iteration
     141      748785 :   if (force_iteration && !(_nl_forced_its))
     142         734 :     _nl_forced_its = 1;
     143             : 
     144             :   // if specified here to force iteration, but PETSc doesn't know, tell it
     145      748785 :   if (!force_iteration && (_nl_forced_its))
     146             :   {
     147         164 :     LibmeshPetscCallA(_fe_problem.comm().get(), SNESSetForceIteration(snes, PETSC_TRUE));
     148             :   }
     149             : #endif
     150             : 
     151             :   // See if SNESSetFunctionDomainError() has been called.
     152             :   // Note: SNESSetFunctionDomainError() and SNESGetFunctionDomainError() were added
     153             :   // in different releases of PETSc. The latter was removed in PETSc 3.25.0.
     154             : #if PETSC_VERSION_LESS_THAN(3, 25, 0)
     155             :   PetscBool domainerror;
     156             :   LibmeshPetscCallA(_fe_problem.comm().get(), SNESGetFunctionDomainError(snes, &domainerror));
     157             :   if (domainerror)
     158             :     status = MooseConvergenceStatus::DIVERGED;
     159             : #else
     160             :   SNESConvergedReason reason;
     161      748785 :   LibmeshPetscCallA(_fe_problem.comm().get(), SNESGetConvergedReason(snes, &reason));
     162      748785 :   if (reason == SNES_DIVERGED_FUNCTION_DOMAIN)
     163           0 :     status = MooseConvergenceStatus::DIVERGED;
     164             : #endif
     165             : 
     166             :   Real fnorm_old;
     167             :   // This is the first residual before any iterations have been done, but after
     168             :   // solution-modifying objects (if any) have been imposed on the solution vector.
     169             :   // We save it, and use it to detect convergence if system.usePreSMOResidual() == false.
     170      748785 :   if (iter == 0)
     171             :   {
     172      285602 :     system.setInitialResidual(fnorm);
     173      285602 :     fnorm_old = fnorm;
     174      285602 :     _nl_current_pingpong = 0;
     175             :   }
     176             :   else
     177      463183 :     fnorm_old = system._last_nl_rnorm;
     178             : 
     179             :   // Check for nonlinear residual ping-pong.
     180             :   // Ping-pong will always start from a residual increase
     181      748785 :   if ((_nl_current_pingpong % 2 == 1 && !(fnorm > fnorm_old)) ||
     182      744784 :       (_nl_current_pingpong % 2 == 0 && fnorm > fnorm_old))
     183        8616 :     _nl_current_pingpong += 1;
     184             :   else
     185      740169 :     _nl_current_pingpong = 0;
     186             : 
     187      748785 :   const auto ref_residual = system.referenceResidual();
     188      748785 :   std::ostringstream oss;
     189      748785 :   if (iter < _nl_forced_its)
     190        1536 :     oss << "Number of forced iterations not yet reached: " << iter << " < " << _nl_forced_its
     191        1536 :         << '\n';
     192      747249 :   else if (fnorm != fnorm)
     193             :   {
     194           0 :     oss << "Failed to converge, residual norm is NaN\n";
     195           0 :     status = MooseConvergenceStatus::DIVERGED;
     196             :   }
     197      747249 :   else if (checkResidualConvergence(iter, fnorm, ref_residual, rel_tol, abs_tol, oss))
     198      284977 :     status = MooseConvergenceStatus::CONVERGED;
     199      462272 :   else if (nfuncs >= max_funcs)
     200             :   {
     201           0 :     oss << "Exceeded maximum number of residual evaluations: " << nfuncs << " > " << max_funcs
     202           0 :         << '\n';
     203           0 :     status = MooseConvergenceStatus::DIVERGED;
     204             :   }
     205      462272 :   else if ((iter >= _nl_forced_its) && iter && fnorm > system._last_nl_rnorm &&
     206        5095 :            fnorm >= _div_threshold)
     207             :   {
     208           0 :     oss << "Nonlinear solve was blowing up!\n";
     209           0 :     status = MooseConvergenceStatus::DIVERGED;
     210             :   }
     211      462272 :   else if (snorm < rel_step_tol * xnorm)
     212             :   {
     213           0 :     oss << "Converged due to small update length: " << snorm << " < " << rel_step_tol << " * "
     214           0 :         << xnorm << '\n';
     215           0 :     status = MooseConvergenceStatus::CONVERGED;
     216             :   }
     217      462272 :   else if (_nl_rel_div_tol > 0 && fnorm > ref_residual * _nl_rel_div_tol)
     218             :   {
     219           2 :     oss << "Diverged due to relative residual " << ref_residual << " > divergence tolerance "
     220           2 :         << _nl_rel_div_tol << " * relative residual " << ref_residual << '\n';
     221           2 :     status = MooseConvergenceStatus::DIVERGED;
     222             :   }
     223      462270 :   else if (_nl_abs_div_tol > 0 && fnorm > _nl_abs_div_tol)
     224             :   {
     225           9 :     oss << "Diverged due to residual " << fnorm << " > absolute divergence tolerance "
     226           9 :         << _nl_abs_div_tol << '\n';
     227           9 :     status = MooseConvergenceStatus::DIVERGED;
     228             :   }
     229      462261 :   else if (_nl_current_pingpong > _nl_max_pingpong)
     230             :   {
     231           9 :     oss << "Diverged due to maximum nonlinear residual pingpong achieved" << '\n';
     232           9 :     status = MooseConvergenceStatus::DIVERGED;
     233             :   }
     234             : 
     235      748785 :   system._last_nl_rnorm = fnorm;
     236      748785 :   system._current_nl_its = static_cast<unsigned int>(iter);
     237             : 
     238      748785 :   std::string msg;
     239      748785 :   msg = oss.str();
     240      748785 :   if (_app.multiAppLevel() > 0)
     241      383940 :     MooseUtils::indentMessage(_app.name(), msg);
     242      748785 :   if (msg.length() > 0)
     243             : #if !PETSC_VERSION_LESS_THAN(3, 17, 0)
     244      286533 :     LibmeshPetscCallA(_fe_problem.comm().get(), PetscInfo(snes, "%s", msg.c_str()));
     245             : #else
     246             :     LibmeshPetscCallA(_fe_problem.comm().get(), PetscInfo(snes, msg.c_str()));
     247             : #endif
     248             : 
     249      748785 :   verboseOutput(oss);
     250             : 
     251      748785 :   return status;
     252      748785 : }

Generated by: LCOV version 1.14