version 0.4.1
newtonraphson.hh
Go to the documentation of this file.
1// SPDX-FileCopyrightText: 2021-2025 The Ikarus Developers mueller@ibb.uni-stuttgart.de
2// SPDX-License-Identifier: LGPL-3.0-or-later
3
9#pragma once
10
20
21namespace Ikarus {
22
23template <typename F, typename LS = utils::SolverDefault, typename UF = utils::UpdateDefault>
24class NewtonRaphson;
25
27{
28 double tol{1e-8};
29 int maxIter{20};
30 int minIter{0};
31};
32
37template <typename LS = utils::SolverDefault, typename UF = utils::UpdateDefault>
39{
40 using LinearSolver = LS;
41 using UpdateFunction = UF;
45
46 template <typename UF2>
49 .parameters = parameters, .linearSolver = linearSolver, .updateFunction = std::forward<UF2>(updateFunction)};
50 return settings;
51 }
52
53 template <typename F>
55};
56
57// THE CTAD is broken for designated initializers in clang 16, when we drop support this can be simplified
58#ifndef DOXYGEN
61
62template <typename LS>
64
65template <typename LS, typename UF>
67
68template <typename UF>
70#endif
71
80template <typename F, typename NRConfig>
81requires traits::isSpecialization<NewtonRaphsonConfig, std::remove_cvref_t<NRConfig>>::value
82auto createNonlinearSolver(NRConfig&& config, F&& f) {
84 using UF = std::remove_cvref_t<NRConfig>::UpdateFunction;
85 auto solverFactory = []<class F2, class LS2, class UF2>(F2&& f2, LS2&& ls, UF2&& uf) {
86 return std::make_shared<NewtonRaphson<std::remove_cvref_t<F2>, std::remove_cvref_t<LS2>, std::remove_cvref_t<UF2>>>(
87 f2, std::forward<LS2>(ls), std::forward<UF2>(uf));
88 };
89
90 if constexpr (std::remove_cvref_t<F>::nDerivatives == 2) {
91 auto solver = solverFactory(derivative(f), std::forward<NRConfig>(config).linearSolver,
92 std::forward<NRConfig>(config).updateFunction);
93 solver->setup(config.parameters);
94 return solver;
95 } else {
96 static_assert(std::remove_cvref_t<F>::nDerivatives >= 1,
97 "The number of derivatives in the differentiable function have to be more than 0");
98 auto solver =
99 solverFactory(f, std::forward<NRConfig>(config).linearSolver, std::forward<NRConfig>(config).updateFunction);
100 ;
101
102 solver->setup(std::forward<NRConfig>(config).parameters);
103 return solver;
104 }
105}
106
116template <typename F, typename LS, typename UF>
118{
119public:
121 using SignatureTraits = typename F::Traits;
122 using CorrectionType = typename SignatureTraits::template Range<0>;
123 using JacobianType = typename SignatureTraits::template Range<1>;
126
127 using Domain = typename SignatureTraits::Domain;
128
129 using UpdateFunction = UF;
131
138 template <typename LS2 = LS, typename UF2 = UF>
139 explicit NewtonRaphson(const DifferentiableFunction& residual, LS2&& linearSolver = {}, UF2&& updateFunction = {})
140 : residualFunction_{residual},
141 jacobianFunction_{derivative(residualFunction_)},
142 linearSolver_{std::forward<LS2>(linearSolver)},
143 updateFunction_{std::forward<UF2>(updateFunction)} {}
144
149 void setup(const Settings& settings) {
150 if (settings.minIter > settings.maxIter)
151 DUNE_THROW(Dune::InvalidStateException,
152 "Minimum number of iterations cannot be greater than maximum number of iterations");
153 settings_ = settings;
154 }
155
161 [[nodiscard(
162 "The solve method returns information of the solution process. You should store this information and check if "
163 "it was successful")]] Ikarus::NonLinearSolverInformation
165 using enum NonLinearSolverMessages;
166 this->notify(INIT);
167
168 Ikarus::NonLinearSolverInformation solverInformation;
169 solverInformation.success = true;
170
171 decltype(auto) rx = residualFunction_(x);
172 decltype(auto) Ax = jacobianFunction_(x);
173 auto rNorm = norm(rx);
174 decltype(rNorm) dNorm;
175 int iter{0};
176 if constexpr (isLinearSolver)
177 linearSolver_.analyzePattern(Ax);
178
179 auto solverState = typename NewtonRaphson::State{.domain = x, .correction = correction_};
180
181 while ((rNorm > settings_.tol && iter < settings_.maxIter) or iter < settings_.minIter) {
182 this->notify(ITERATION_STARTED);
183 if constexpr (isLinearSolver) {
184 linearSolver_.factorize(Ax);
185 linearSolver_.solve(correction_, -rx);
186 dNorm = correction_.norm();
187 } else {
188 correction_ = -linearSolver_(rx, Ax);
189 dNorm = norm(correction_);
190 }
191 this->notify(CORRECTION_UPDATED, solverState);
192 updateFunction_(x, correction_);
193 this->notify(CORRECTIONNORM_UPDATED, static_cast<double>(dNorm));
194 this->notify(SOLUTION_CHANGED);
195 rx = residualFunction_(x);
196 Ax = jacobianFunction_(x);
197 rNorm = norm(rx);
198 this->notify(RESIDUALNORM_UPDATED, static_cast<double>(rNorm));
199 this->notify(ITERATION_ENDED);
200 ++iter;
201 }
202 if (iter == settings_.maxIter)
203 solverInformation.success = false;
204 solverInformation.iterations = iter;
205 solverInformation.residualNorm = static_cast<double>(rNorm);
206 solverInformation.correctionNorm = static_cast<double>(dNorm);
207 if (solverInformation.success)
208 this->notify(FINISHED_SUCESSFULLY, iter);
209 return solverInformation;
210 }
211
216 auto& residual() { return residualFunction_; }
217
218private:
219 DifferentiableFunction residualFunction_;
220 typename DifferentiableFunction::Derivative jacobianFunction_;
221 std::remove_cvref_t<CorrectionType> correction_;
222 LS linearSolver_;
223 UpdateFunction updateFunction_;
224 Settings settings_;
225};
226
237template <typename F, typename LS = utils::SolverDefault, typename UF = utils::UpdateDefault>
238auto makeNewtonRaphson(const F& f, LS&& linearSolver = {}, UF&& updateFunction = {}) {
239 return std::make_shared<NewtonRaphson<F, LS, UF>>(f, std::forward<LS>(linearSolver), std::move(updateFunction));
240}
241
242template <typename F, typename LS = utils::SolverDefault, typename UF = utils::UpdateDefault>
243NewtonRaphson(const F& f, LS&& linearSolver = {},
244 UF&& updateFunction = {}) -> NewtonRaphson<F, std::remove_cvref_t<LS>, std::remove_cvref_t<UF>>;
245
246} // namespace Ikarus
Collection of fallback default functions.
Helper for the autodiff library.
Type-erased linear solver with templated scalar type.
Base for all nonlinear solvers.
State for all nonlinear solvers.
Implementation of the solver information returned by the nonlinear solvers.
Enums for observer messages.
Implementation of the observer design pattern with broadcasters.
NonLinearSolverMessages
Enum class defining non-linear solver-related messages.
Definition: broadcastermessages.hh:22
auto norm(const Eigen::MatrixBase< Derived > &v)
Adding free norm function to Eigen types.
Definition: linearalgebrahelper.hh:259
Definition: assemblermanipulatorbuildingblocks.hh:22
auto makeNewtonRaphson(const F &f, LS &&linearSolver={}, UF &&updateFunction={})
Function to create a NewtonRaphson solver instance.
Definition: newtonraphson.hh:238
LinearSolverTemplate< double > LinearSolver
Definition: linearsolver.hh:235
::value auto createNonlinearSolver(NRConfig &&config, F &&f)
Function to create a NewtonRaphson solver instance.
Definition: newtonraphson.hh:82
NewtonRaphson(const F &f, LS &&linearSolver={}, UF &&updateFunction={}) -> NewtonRaphson< F, std::remove_cvref_t< LS >, std::remove_cvref_t< UF > >
Implementation of the Newton-Raphson method for solving nonlinear equations.
Definition: newtonraphson.hh:118
static constexpr bool isLinearSolver
Definition: newtonraphson.hh:125
NRSettings Settings
Definition: newtonraphson.hh:120
typename SignatureTraits::Domain Domain
Type representing the parameter vector of the function.
Definition: newtonraphson.hh:127
Ikarus::NonLinearSolverInformation solve(Domain &x)
Solve the nonlinear system.
Definition: newtonraphson.hh:164
F DifferentiableFunction
Type of the non-linear operator.
Definition: newtonraphson.hh:130
typename SignatureTraits::template Range< 0 > CorrectionType
Type of the correction of x += deltaX.
Definition: newtonraphson.hh:122
typename F::Traits SignatureTraits
Definition: newtonraphson.hh:121
void setup(const Settings &settings)
Set up the solver with the given settings.
Definition: newtonraphson.hh:149
typename SignatureTraits::template Range< 1 > JacobianType
Compile-time boolean indicating if the linear solver satisfies the non-linear solver concept.
Definition: newtonraphson.hh:124
NewtonRaphson(const DifferentiableFunction &residual, LS2 &&linearSolver={}, UF2 &&updateFunction={})
Constructor for NewtonRaphson.
Definition: newtonraphson.hh:139
auto & residual()
Access the function.
Definition: newtonraphson.hh:216
UF UpdateFunction
Type representing the update function.
Definition: newtonraphson.hh:129
Definition: newtonraphson.hh:27
int maxIter
Definition: newtonraphson.hh:29
int minIter
Definition: newtonraphson.hh:30
double tol
Definition: newtonraphson.hh:28
Config for the Newton-Raphson solver.
Definition: newtonraphson.hh:39
NRSettings parameters
Definition: newtonraphson.hh:42
UF updateFunction
Definition: newtonraphson.hh:44
LS LinearSolver
Definition: newtonraphson.hh:40
LS linearSolver
Definition: newtonraphson.hh:43
UF UpdateFunction
Definition: newtonraphson.hh:41
auto rebindUpdateFunction(UF2 &&updateFunction) const
Definition: newtonraphson.hh:47
Base for all nonlinear solvers. Defines the message interface that can be broadcasted to listeners.
Definition: nonlinearsolverbase.hh:27
State for nonlinear solvers.
Definition: nonlinearsolverstate.hh:23
const Domain & domain
Definition: nonlinearsolverstate.hh:27
Information about the result of a non-linear solver.
Definition: solverinfos.hh:21
double correctionNorm
Definition: solverinfos.hh:30
int iterations
Definition: solverinfos.hh:31
double residualNorm
Definition: solverinfos.hh:29
bool success
Definition: solverinfos.hh:28
Default functor for solving operations.
Definition: defaultfunctions.hh:26
Concept to check if a linear solver implements all the needed functions for given vector and matrix t...
Definition: utils/concepts.hh:223
Several concepts.