GEOS
CoupledSolver.hpp
Go to the documentation of this file.
1 /*
2  * ------------------------------------------------------------------------------------------------------------
3  * SPDX-License-Identifier: LGPL-2.1-only
4  *
5  * Copyright (c) 2016-2024 Lawrence Livermore National Security LLC
6  * Copyright (c) 2018-2024 TotalEnergies
7  * Copyright (c) 2018-2024 The Board of Trustees of the Leland Stanford Junior University
8  * Copyright (c) 2023-2024 Chevron
9  * Copyright (c) 2019- GEOS/GEOSX Contributors
10  * All rights reserved
11  *
12  * See top level LICENSE, COPYRIGHT, CONTRIBUTORS, NOTICE, and ACKNOWLEDGEMENTS files for details.
13  * ------------------------------------------------------------------------------------------------------------
14  */
15 
21 #ifndef GEOS_PHYSICSSOLVERS_MULTIPHYSICS_COUPLEDSOLVER_HPP_
22 #define GEOS_PHYSICSSOLVERS_MULTIPHYSICS_COUPLEDSOLVER_HPP_
23 
26 
27 #include <tuple>
28 
29 namespace geos
30 {
31 
32 template< typename ... SOLVERS >
34 {
35 
36 public:
37 
43  CoupledSolver( const string & name,
44  Group * const parent )
45  : PhysicsSolverBase( name, parent )
46  {
47  forEachArgInTuple( m_solvers, [&]( auto solver, auto idx )
48  {
49  using SolverType = TYPEOFPTR( solver );
50  string const key = SolverType::coupledSolverAttributePrefix() + "SolverName";
51  registerWrapper( key, &m_names[idx()] ).
52  setRTTypeName( rtTypes::CustomTypes::groupNameRef ).
54  setDescription( "Name of the " + SolverType::coupledSolverAttributePrefix() + " solver used by the coupled solver" );
55  } );
56 
57  this->getWrapper< string >( PhysicsSolverBase::viewKeyStruct::discretizationString() ).
58  setInputFlag( dataRepository::InputFlags::FALSE );
59 
60  addLogLevel< logInfo::Convergence >();
61  addLogLevel< logInfo::Coupling >();
62  addLogLevel< logInfo::TimeStep >();
63  }
64 
66  CoupledSolver( CoupledSolver const & ) = delete;
67 
69  CoupledSolver( CoupledSolver && ) = default;
70 
72  CoupledSolver & operator=( CoupledSolver const & ) = delete;
73 
76 
77 
81  void
83  {
84  forEachArgInTuple( m_solvers, [&]( auto & solver, auto idx )
85  {
86  using SolverPtr = TYPEOFREF( solver );
87  using SolverType = TYPEOFPTR( SolverPtr {} );
88  auto const & solverName = m_names[idx()];
89  auto const & solverType = LvArray::system::demangleType< SolverType >();
90  solver = this->getParent().template getGroupPointer< SolverType >( solverName );
91  GEOS_THROW_IF( solver == nullptr,
92  GEOS_FMT( "{}: Could not find solver '{}' of type {}",
94  solverName, solverType ),
95  InputError );
96  GEOS_LOG_LEVEL_RANK_0( logInfo::Coupling,
97  GEOS_FMT( "{}: found {} solver named {}",
98  getName(), solver->getCatalogName(), solverName ) );
99  } );
100  }
101 
102 
108  virtual void
110  DofManager & dofManager ) const
111  { GEOS_UNUSED_VAR( domain, dofManager ); }
112 
122  virtual void
124  real64 const dt,
125  DomainPartition const & domain,
126  DofManager const & dofManager,
127  CRSMatrixView< real64, globalIndex const > const & localMatrix,
128  arrayView1d< real64 > const & localRhs )
129  { GEOS_UNUSED_VAR( time_n, dt, domain, dofManager, localMatrix, localRhs ); }
130 
138  void
139  setupDofs( DomainPartition const & domain,
140  DofManager & dofManager ) const override
141  {
142  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
143  {
144  solver->setupDofs( domain, dofManager );
145  } );
146 
147  setupCoupling( domain, dofManager );
148  }
149 
150  virtual void
151  implicitStepSetup( real64 const & time_n,
152  real64 const & dt,
153  DomainPartition & domain ) override
154  {
155  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
156  {
157  solver->implicitStepSetup( time_n, dt, domain );
158  } );
159  }
160 
161  virtual void
162  implicitStepComplete( real64 const & time_n,
163  real64 const & dt,
164  DomainPartition & domain ) override
165  {
166  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
167  {
168  solver->implicitStepComplete( time_n, dt, domain );
169  } );
170  }
171 
172  // general version of assembleSystem function, keep in mind many solvers will override it
173  virtual void
174  assembleSystem( real64 const time_n,
175  real64 const dt,
176  DomainPartition & domain,
177  DofManager const & dofManager,
178  CRSMatrixView< real64, globalIndex const > const & localMatrix,
179  arrayView1d< real64 > const & localRhs ) override
180  {
182 
183  // 1. Assemble matrix blocks of each individual solver
184  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
185  {
186  solver->assembleSystem( time_n, dt, domain, dofManager, localMatrix, localRhs );
187  } );
188 
189  // 2. Assemble coupling blocks
190  assembleCouplingTerms( time_n, dt, domain, dofManager, localMatrix, localRhs );
191  }
192 
193  virtual void
194  applySystemSolution( DofManager const & dofManager,
195  arrayView1d< real64 const > const & localSolution,
196  real64 const scalingFactor,
197  real64 const dt,
198  DomainPartition & domain ) override
199  {
200  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
201  {
202  solver->applySystemSolution( dofManager, localSolution, scalingFactor, dt, domain );
203  } );
204  }
205 
206  virtual void
207  updateState( DomainPartition & domain ) override
208  {
209  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
210  {
211  solver->updateState( domain );
212  } );
213  }
214 
215  virtual void
217  {
218  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
219  {
220  solver->resetStateToBeginningOfStep( domain );
221  } );
222  }
223 
226  real64
227  solverStep( real64 const & time_n,
228  real64 const & dt,
229  int const cycleNumber,
230  DomainPartition & domain ) override final
231  {
233 
235  {
236  return fullyCoupledSolverStep( time_n, dt, cycleNumber, domain );
237  }
239  {
240  return sequentiallyCoupledSolverStep( time_n, dt, cycleNumber, domain );
241  }
242  else
243  {
244  GEOS_ERROR( getDataContext() << ": Invalid coupling type option." );
245  return 0;
246  }
247 
248  }
249 
250 
251  virtual real64
252  calculateResidualNorm( real64 const & time_n,
253  real64 const & dt,
254  DomainPartition const & domain,
255  DofManager const & dofManager,
256  arrayView1d< real64 const > const & localRhs ) override
257  {
258  real64 norm = 0.0;
259  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
260  {
261  real64 const singlePhysicsNorm = solver->calculateResidualNorm( time_n, dt, domain, dofManager, localRhs );
262  norm += singlePhysicsNorm * singlePhysicsNorm;
263  } );
264 
265  return sqrt( norm );
266  }
267 
268  virtual void
270  real64 const dt,
271  DomainPartition & domain,
272  DofManager const & dofManager,
273  CRSMatrixView< real64, globalIndex const > const & localMatrix,
274  arrayView1d< real64 > const & localRhs ) override
275  {
276  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
277  {
278  solver->applyBoundaryConditions( time_n, dt, domain, dofManager, localMatrix, localRhs );
279  } );
280  }
281 
282  virtual bool
284  DofManager const & dofManager,
285  arrayView1d< real64 const > const & localSolution,
286  real64 const scalingFactor ) override
287  {
288  bool validSolution = true;
289  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
290  {
291  bool const validSinglePhysicsSolution = solver->checkSystemSolution( domain, dofManager, localSolution, scalingFactor );
292  if( !validSinglePhysicsSolution )
293  {
294  GEOS_LOG_RANK_0( GEOS_FMT( " {}/{}: Solution check failed. Newton loop terminated.", getName(), solver->getName()) );
295  }
296  validSolution = validSolution && validSinglePhysicsSolution;
297  } );
298  return validSolution;
299  }
300 
301  virtual real64
303  DofManager const & dofManager,
304  arrayView1d< real64 const > const & localSolution ) override
305  {
306  real64 scalingFactor = 1e9;
307  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
308  {
309  real64 const singlePhysicsScalingFactor = solver->scalingForSystemSolution( domain, dofManager, localSolution );
310  scalingFactor = LvArray::math::min( scalingFactor, singlePhysicsScalingFactor );
311  } );
312  return scalingFactor;
313  }
314 
315  virtual real64
316  setNextDt( real64 const & currentTime,
317  real64 const & currentDt,
318  DomainPartition & domain ) override
319  {
320  real64 nextDt = PhysicsSolverBase::setNextDt( currentTime, currentDt, domain );
321  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
322  {
323  real64 const singlePhysicsNextDt =
324  solver->setNextDt( currentTime, currentDt, domain );
325  nextDt = LvArray::math::min( singlePhysicsNextDt, nextDt );
326  } );
327  return nextDt;
328  }
329 
330  virtual void cleanup( real64 const time_n,
331  integer const cycleNumber,
332  integer const eventCounter,
333  real64 const eventProgress,
334  DomainPartition & domain ) override
335  {
336  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
337  {
338  solver->cleanup( time_n, cycleNumber, eventCounter, eventProgress, domain );
339  } );
340  PhysicsSolverBase::cleanup( time_n, cycleNumber, eventCounter, eventProgress, domain );
341  }
342 
345  virtual bool checkSequentialSolutionIncrements( DomainPartition & domain ) const override
346  {
347  bool isConverged = true;
348  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
349  {
350  isConverged &= solver->checkSequentialSolutionIncrements( domain );
351  } );
352  return isConverged;
353  }
354 
355  virtual bool updateConfiguration( DomainPartition & domain ) override
356  {
357  bool result = true;
358  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
359  {
360  result &= solver->updateConfiguration( domain );
361  } );
362  return result;
363  }
364 
365  virtual void outputConfigurationStatistics( DomainPartition const & domain ) const override
366  {
367  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
368  {
369  solver->outputConfigurationStatistics( domain );
370  } );
371  }
372 
373  virtual void resetConfigurationToBeginningOfStep( DomainPartition & domain ) override
374  {
375  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
376  {
377  solver->resetConfigurationToBeginningOfStep( domain );
378  } );
379  }
380 
381  virtual bool resetConfigurationToDefault( DomainPartition & domain ) const override
382  {
383  bool result = true;
384  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
385  {
386  result &= solver->resetConfigurationToDefault( domain );
387  } );
388  return result;
389  }
390 
391 protected:
392 
402  virtual real64 fullyCoupledSolverStep( real64 const & time_n,
403  real64 const & dt,
404  int const cycleNumber,
405  DomainPartition & domain )
406  {
407  return PhysicsSolverBase::solverStep( time_n, dt, cycleNumber, domain );
408  }
409 
420  virtual real64 sequentiallyCoupledSolverStep( real64 const & time_n,
421  real64 const & dt,
422  int const cycleNumber,
423  DomainPartition & domain )
424  {
426 
427  // Only build the sparsity pattern if the mesh has changed
428  Timestamp const meshModificationTimestamp = getMeshModificationTimestamp( domain );
429  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
430  {
431  if( meshModificationTimestamp > solver->getSystemSetupTimestamp() )
432  {
433  solver->setupSystem( domain,
434  solver->getDofManager(),
435  solver->getLocalMatrix(),
436  solver->getSystemRhs(),
437  solver->getSystemSolution() );
438  solver->setSystemSetupTimestamp( meshModificationTimestamp );
439  }
440  } );
441 
442  implicitStepSetup( time_n, dt, domain );
443 
445  integer const maxNumberDtCuts = solverParams.m_maxTimeStepCuts;
446  real64 const dtCutFactor = solverParams.m_timeStepCutFactor;
447  integer & dtAttempt = solverParams.m_numTimeStepAttempts;
448 
449  bool isConverged = false;
450  // dt may be cut during the course of this step, so we are keeping a local
451  // value to track the achieved dt for this step.
452  real64 stepDt = dt;
453 
454  // outer loop attempts to apply full timestep, and managed the cutting of the timestep if
455  // required.
456  for( dtAttempt = 0; dtAttempt < maxNumberDtCuts; ++dtAttempt )
457  {
458  // TODO configuration loop
459 
460  // Reset the states of all solvers if any of them had to restart
461  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
462  {
463  solver->resetStateToBeginningOfStep( domain );
464  solver->getSolverStatistics().initializeTimeStepStatistics(); // initialize counters for subsolvers
465  } );
466  resetStateToBeginningOfStep( domain );
467 
468  integer & iter = solverParams.m_numNewtonIterations;
469 
471  for( iter = 0; iter < solverParams.m_maxIterNewton; iter++ )
472  {
473  // Increment the solver statistics for reporting purposes
474  // Pass a "0" as argument (0 linear iteration) to skip the output of linear iteration stats at the end
476 
477  startSequentialIteration( iter, domain );
478 
479  // Solve the subproblems nonlinearly
480  forEachArgInTuple( m_solvers, [&]( auto & solver, auto idx )
481  {
482  GEOS_LOG_LEVEL_RANK_0( logInfo::NonlinearSolver,
483  GEOS_FMT( " Iteration {:2}: {}", iter + 1, solver->getName() ) );
484  real64 solverDt = solver->nonlinearImplicitStep( time_n,
485  stepDt,
486  cycleNumber,
487  domain );
488 
489  // save fields (e.g. pressure and temperature) after inner solve
490  if( solver->getNonlinearSolverParameters().couplingType() == NonlinearSolverParameters::CouplingType::Sequential )
491  {
492  solver->saveSequentialIterationState( domain );
493  }
494 
495  mapSolutionBetweenSolvers( domain, idx() );
496 
497  if( solverDt < stepDt ) // subsolver had to cut the time step
498  {
499  iter = 0; // restart outer loop
500  stepDt = solverDt; // sync time step
501  }
502  } );
503 
504  // Check convergence of the outer loop
505  isConverged = checkSequentialConvergence( iter,
506  time_n,
507  stepDt,
508  domain );
509 
510  if( isConverged )
511  {
512  // we still want to count current iteration
513  ++iter;
514  // exit outer loop
515  break;
516  }
517  else
518  {
519  finishSequentialIteration( iter, domain );
520  }
521  }
522 
523  if( isConverged )
524  {
525  // Save time step statistics for the subsolvers
526  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
527  {
528  solver->getSolverStatistics().saveTimeStepStatistics();
529  } );
530  // get out of the time loop
531  break;
532  }
533  else
534  {
535  // cut timestep, go back to beginning of step and restart the Newton loop
536  stepDt *= dtCutFactor;
537  GEOS_LOG_LEVEL_RANK_0( logInfo::TimeStep, GEOS_FMT( "New dt = {}", stepDt ) );
538 
539  // notify the solver statistics counter that this is a time step cut
541  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
542  {
543  solver->getSolverStatistics().logTimeStepCut();
544  } );
545  }
546  }
547 
548  if( !isConverged )
549  {
550  GEOS_LOG_RANK_0( "Convergence not achieved." );
551 
553  {
554  GEOS_LOG_RANK_0( "The accepted solution may be inaccurate." );
555  }
556  else
557  {
558  GEOS_ERROR( "Nonconverged solutions not allowed. Terminating..." );
559  }
560  }
561 
562  implicitStepComplete( time_n, stepDt, domain );
563 
564  return stepDt;
565  }
566 
574  integer const solverType )
575  {
576  GEOS_UNUSED_VAR( domain, solverType );
577  }
578 
579  virtual bool checkSequentialConvergence( int const & iter,
580  real64 const & time_n,
581  real64 const & dt,
582  DomainPartition & domain )
583  {
585  bool isConverged = true;
586 
587  if( params.m_subcyclingOption == 0 )
588  {
589  GEOS_LOG_LEVEL_RANK_0( logInfo::Convergence, "***** Single Pass solver, no subcycling *****" );
590  }
591  else
592  {
593  GEOS_LOG_LEVEL_RANK_0( logInfo::Convergence, GEOS_FMT( " Iteration {:2}: outer-loop convergence check", iter + 1 ) );
594 
596  {
597  real64 residualNorm = 0;
598 
599  // loop over all the single-physics solvers
600  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
601  {
602 
603  solver->getLocalMatrix().toViewConstSizes().zero();
604  solver->getSystemRhs().zero();
605  arrayView1d< real64 > const localRhs = solver->getSystemRhs().open();
606 
607  // for each solver, we have to recompute the residual (and Jacobian, although not necessary)
608  solver->assembleSystem( time_n,
609  dt,
610  domain,
611  solver->getDofManager(),
612  solver->getLocalMatrix().toViewConstSizes(),
613  localRhs );
614  solver->applyBoundaryConditions( time_n,
615  dt,
616  domain,
617  solver->getDofManager(),
618  solver->getLocalMatrix().toViewConstSizes(),
619  localRhs );
620  solver->getSystemRhs().close();
621 
622  // once this is done, we recompute the single-physics residual
623  real64 const singlePhysicsNorm =
624  solver->calculateResidualNorm( time_n,
625  dt,
626  domain,
627  solver->getDofManager(),
628  solver->getSystemRhs().values() );
629  residualNorm += singlePhysicsNorm * singlePhysicsNorm;
630  } );
631 
632  // finally, we perform the convergence check on the multiphysics residual
633  residualNorm = sqrt( residualNorm );
634  GEOS_LOG_LEVEL_RANK_0( logInfo::Convergence,
635  GEOS_FMT( " ( R ) = ( {:4.2e} )", residualNorm ) );
636  isConverged = ( residualNorm < params.m_newtonTol );
637 
638  }
640  {
641  // TODO also make recursive?
642  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
643  {
644  NonlinearSolverParameters const & singlePhysicsParams = solver->getNonlinearSolverParameters();
645  if( singlePhysicsParams.m_numNewtonIterations > singlePhysicsParams.m_minIterNewton )
646  {
647  isConverged = false;
648  }
649  } );
650  }
652  {
653  isConverged = checkSequentialSolutionIncrements( domain );
654  }
655  else
656  {
657  GEOS_ERROR( getDataContext() << ": Invalid sequential convergence criterion." );
658  }
659 
660  if( isConverged )
661  {
662  GEOS_LOG_LEVEL_RANK_0( logInfo::Convergence,
663  GEOS_FMT( "***** The iterative coupling has converged in {} iteration(s) *****", iter + 1 ) );
664  }
665  }
666  return isConverged;
667  }
668 
669  virtual void
671  {
672  setSubSolvers();
673 
676  GEOS_THROW_IF( isSequential && usesLineSearch,
677  GEOS_FMT( "{}: line search is not supported by the coupled solver when {} is set to `{}`. Please set {} to `{}` to remove this error",
678  getNonlinearSolverParameters().getWrapperDataContext( NonlinearSolverParameters::viewKeysStruct::couplingTypeString() ),
679  NonlinearSolverParameters::viewKeysStruct::couplingTypeString(),
681  NonlinearSolverParameters::viewKeysStruct::lineSearchActionString(),
683  InputError );
684 
685  if( !isSequential )
686  {
688  }
689 
691  validateNonlinearAcceleration();
692  }
693 
694  virtual void validateNonlinearAcceleration()
695  {
696  GEOS_THROW ( GEOS_FMT( "{}: Nonlinear acceleration {} is not supported by {} solver '{}'",
697  getWrapperDataContext( NonlinearSolverParameters::viewKeysStruct::nonlinearAccelerationTypeString() ),
699  getCatalogName(), getName()),
700  InputError );
701  }
702 
703  virtual void
705  {
706  forEachArgInTuple( m_solvers, [&]( auto & solver, auto )
707  {
708  solver->getNonlinearSolverParameters() = getNonlinearSolverParameters();
709  } );
710  }
711 
712  virtual void startSequentialIteration( integer const & iter,
713  DomainPartition & domain )
714  {
715  GEOS_UNUSED_VAR( iter, domain );
716  }
717 
718  virtual void finishSequentialIteration( integer const & iter,
719  DomainPartition & domain )
720  {
721  GEOS_UNUSED_VAR( iter, domain );
722  }
723 
724 protected:
725 
727  std::tuple< SOLVERS *... > m_solvers;
728 
730  std::array< string, sizeof...( SOLVERS ) > m_names;
731 };
732 
733 } /* namespace geos */
734 
735 #endif /* GEOS_PHYSICSSOLVERS_MULTIPHYSICS_COUPLEDSOLVER_HPP_ */
#define GEOS_UNUSED_VAR(...)
Mark an unused variable and silence compiler warnings.
Definition: GeosxMacros.hpp:84
#define GEOS_THROW(msg, TYPE)
Throw an exception.
Definition: Logger.hpp:164
#define GEOS_ERROR(msg)
Raise a hard error and terminate the program.
Definition: Logger.hpp:157
#define GEOS_LOG_RANK_0(msg)
Log a message on screen on rank 0.
Definition: Logger.hpp:101
#define GEOS_THROW_IF(EXP, msg, TYPE)
Conditionally throw an exception.
Definition: Logger.hpp:151
#define GEOS_MARK_FUNCTION
Mark function with both Caliper and NVTX if enabled.
virtual void assembleCouplingTerms(real64 const time_n, real64 const dt, DomainPartition const &domain, DofManager const &dofManager, CRSMatrixView< real64, globalIndex const > const &localMatrix, arrayView1d< real64 > const &localRhs)
Utility function to compute coupling terms.
CoupledSolver & operator=(CoupledSolver const &)=delete
deleted assignment operator
virtual real64 fullyCoupledSolverStep(real64 const &time_n, real64 const &dt, int const cycleNumber, DomainPartition &domain)
Fully coupled solution approach solution step.
virtual bool resetConfigurationToDefault(DomainPartition &domain) const override
resets the configuration to the default value.
CoupledSolver(const string &name, Group *const parent)
main constructor for CoupledSolver Objects
virtual void outputConfigurationStatistics(DomainPartition const &domain) const override
CoupledSolver(CoupledSolver &&)=default
default move constructor
virtual void resetConfigurationToBeginningOfStep(DomainPartition &domain) override
resets the configuration to the beginning of the time-step.
CoupledSolver(CoupledSolver const &)=delete
deleted copy constructor
virtual void synchronizeNonlinearSolverParameters() override
syncronize the nonlinear solver parameters.
virtual void setupCoupling(DomainPartition const &domain, DofManager &dofManager) const
Utility function to set the coupling between degrees of freedom.
virtual real64 sequentiallyCoupledSolverStep(real64 const &time_n, real64 const &dt, int const cycleNumber, DomainPartition &domain)
Sequentially coupled solver step. It solves a nonlinear system of equations using a sequential approa...
virtual bool checkSequentialSolutionIncrements(DomainPartition &domain) const override
Check if the solution increments are ok to use.
virtual void postInputInitialization() override
virtual bool updateConfiguration(DomainPartition &domain) override
updates the configuration (if needed) based on the state after a converged Newton loop.
std::array< string, sizeof...(SOLVERS) > m_names
Names of the single-physics solvers.
std::tuple< SOLVERS *... > m_solvers
Pointers of the single-physics solvers.
CoupledSolver & operator=(CoupledSolver &&)=delete
deleted move operator
void setSubSolvers()
Utility function to set the subsolvers pointers using the names provided by the user.
virtual void mapSolutionBetweenSolvers(DomainPartition &domain, integer const solverType)
Maps the solution obtained from one solver to the fields used by the other solver(s)
The DoFManager is responsible for allocating global dofs, constructing sparsity patterns,...
Definition: DofManager.hpp:44
Partition of the decomposed physical domain. It also manages the connexion information to its neighbo...
SequentialConvergenceCriterion sequentialConvergenceCriterion() const
Getter for the sequential convergence criterion.
integer m_allowNonConverged
Flag to allow for a non-converged nonlinear solution and continue with the problem.
real64 m_newtonTol
The tolerance for the nonlinear convergence check.
NonlinearAccelerationType m_nonlinearAccelerationType
Type of nonlinear acceleration for sequential solver.
integer m_maxIterNewton
The maximum number of nonlinear iterations that are allowed.
real64 m_timeStepCutFactor
Factor by which the time step will be cut if a timestep cut is required.
integer m_numNewtonIterations
The number of nonlinear iterations that have been exectued.
integer m_numTimeStepAttempts
Number of times that the time-step had to be cut.
integer m_maxTimeStepCuts
Max number of time step cuts.
CouplingType couplingType() const
Getter for the coupling type.
LineSearchAction m_lineSearchAction
Flag to apply a line search.
@ ResidualNorm
convergence achieved when the residual drops below a given norm
@ NumberOfNonlinearIterations
convergence achieved when the subproblems convergence is achieved in less than minNewtonIteration
@ SolutionIncrements
convergence achieved when the solution increments are small enough
integer m_subcyclingOption
Flag to specify whether subcycling is allowed or not in sequential schemes.
Base class for all physics solvers.
virtual string getCatalogName() const =0
SolverStatistics m_solverStatistics
Solver statistics.
virtual void cleanup(real64 const time_n, integer const cycleNumber, integer const eventCounter, real64 const eventProgress, DomainPartition &domain) override
Called as the code exits the main run loop.
Timestamp getMeshModificationTimestamp(DomainPartition &domain) const
getter for the timestamp of the mesh modification on the mesh levels
NonlinearSolverParameters & getNonlinearSolverParameters()
accessor for the nonlinear solver parameters.
NonlinearSolverParameters m_nonlinearSolverParameters
Nonlinear solver parameters.
void logTimeStepCut()
Tell the solverStatistics that there is a time step cut.
void logNonlinearIteration(integer const numLinearIterations)
Tell the solverStatistics that we are doing a nonlinear iteration.
Wrapper< TBASE > & registerWrapper(string const &name, wrapperMap::KeyIndex::index_type *const rkey=nullptr)
Create and register a Wrapper around a new object.
DataContext const & getDataContext() const
Definition: Group.hpp:1345
string const & getName() const
Get group name.
Definition: Group.hpp:1331
Group & getParent()
Access the group's parent.
Definition: Group.hpp:1364
DataContext const & getWrapperDataContext(KEY key) const
Definition: Group.hpp:1356
#define GEOS_LOG_LEVEL_RANK_0(logInfoStruct, msg)
Output messages (only on rank 0) based on current Group's log level.
virtual void implicitStepSetup(real64 const &time_n, real64 const &dt, DomainPartition &domain) override
function to perform setup for implicit timestep
virtual void cleanup(real64 const time_n, integer const cycleNumber, integer const eventCounter, real64 const eventProgress, DomainPartition &domain) override
Called as the code exits the main run loop.
virtual real64 scalingForSystemSolution(DomainPartition &domain, DofManager const &dofManager, arrayView1d< real64 const > const &localSolution) override
Function to determine if the solution vector should be scaled back in order to maintain a known const...
virtual void updateState(DomainPartition &domain) override
Recompute all dependent quantities from primary variables (including constitutive models)
virtual void implicitStepComplete(real64 const &time_n, real64 const &dt, DomainPartition &domain) override
perform cleanup for implicit timestep
virtual void applyBoundaryConditions(real64 const time_n, real64 const dt, DomainPartition &domain, DofManager const &dofManager, CRSMatrixView< real64, globalIndex const > const &localMatrix, arrayView1d< real64 > const &localRhs) override
apply boundary condition to system
virtual real64 calculateResidualNorm(real64 const &time_n, real64 const &dt, DomainPartition const &domain, DofManager const &dofManager, arrayView1d< real64 const > const &localRhs) override
calculate the norm of the global system residual
real64 solverStep(real64 const &time_n, real64 const &dt, int const cycleNumber, DomainPartition &domain) override final
virtual real64 setNextDt(real64 const &currentTime, real64 const &currentDt, DomainPartition &domain) override
function to set the next time step size
virtual real64 solverStep(real64 const &time_n, real64 const &dt, integer const cycleNumber, DomainPartition &domain)
entry function to perform a solver step
void setupDofs(DomainPartition const &domain, DofManager &dofManager) const override
Populate degree-of-freedom manager with fields relevant to this solver.
virtual void applySystemSolution(DofManager const &dofManager, arrayView1d< real64 const > const &localSolution, real64 const scalingFactor, real64 const dt, DomainPartition &domain) override
Function to apply the solution vector to the state.
virtual real64 setNextDt(real64 const &currentTime, real64 const &currentDt, DomainPartition &domain)
function to set the next time step size
virtual void assembleSystem(real64 const time_n, real64 const dt, DomainPartition &domain, DofManager const &dofManager, CRSMatrixView< real64, globalIndex const > const &localMatrix, arrayView1d< real64 > const &localRhs) override
function to assemble the linear system matrix and rhs
virtual bool checkSystemSolution(DomainPartition &domain, DofManager const &dofManager, arrayView1d< real64 const > const &localSolution, real64 const scalingFactor) override
Function to check system solution for physical consistency and constraint violation.
virtual void resetStateToBeginningOfStep(DomainPartition &domain) override
reset state of physics back to the beginning of the step.
@ FALSE
Not read from input.
@ REQUIRED
Required in input.
ArrayView< T, 1 > arrayView1d
Alias for 1D array view.
Definition: DataTypes.hpp:180
unsigned long long int Timestamp
Timestamp type (used to perform actions such a sparsity pattern computation after mesh modifications)
Definition: DataTypes.hpp:127
LvArray::CRSMatrixView< T, COL_INDEX, localIndex const, LvArray::ChaiBuffer > CRSMatrixView
Alias for CRS Matrix View.
Definition: DataTypes.hpp:310
std::string string
String type.
Definition: DataTypes.hpp:91
double real64
64-bit floating point type.
Definition: DataTypes.hpp:99
std::int32_t integer
Signed integer type.
Definition: DataTypes.hpp:82
Provides enum <-> string conversion facilities.
Exception class used to report errors in user input.
Definition: Logger.hpp:464
static constexpr char const * discretizationString()