version 0.4.1
newtonraphsonwithscalarsubsidiaryfunction.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
11#include <type_traits>
12#include <utility>
13
21
22namespace Ikarus {
23
24template <typename F, typename LS = utils::SolverDefault, typename UF = utils::UpdateDefault>
26
28{
29 double tol{1e-8};
30 int maxIter{20};
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
64
65template <typename LS>
68
69template <typename LS, typename UF>
72
73template <typename UF>
76#endif
85template <typename F, typename NRConfig>
86requires traits::isSpecialization<NewtonRaphsonWithSubsidiaryFunctionConfig, std::remove_cvref_t<NRConfig>>::value
87auto createNonlinearSolver(NRConfig&& config, F&& f) {
89 using UF = std::remove_cvref_t<NRConfig>::UpdateFunction;
90 auto solverFactory = []<class F2, class LS2, class UF2>(F2&& f2, LS2&& ls, UF2&& uf) {
91 return std::make_shared<NewtonRaphsonWithSubsidiaryFunction<std::remove_cvref_t<F2>, std::remove_cvref_t<LS2>,
92 std::remove_cvref_t<UF2>>>(f2, std::forward<LS2>(ls),
93 std::forward<UF2>(uf));
94 };
95
96 if constexpr (std::remove_cvref_t<F>::nDerivatives == 2) {
97 auto solver = solverFactory(derivative(f), std::forward<NRConfig>(config).linearSolver,
98 std::forward<NRConfig>(config).updateFunction);
99 solver->setup(config.parameters);
100 return solver;
101 } else {
102 static_assert(std::remove_cvref_t<F>::nDerivatives >= 1,
103 "The number of derivatives in the differentiable function have to be more than 0");
104 auto solver =
105 solverFactory(f, std::forward<NRConfig>(config).linearSolver, std::forward<NRConfig>(config).updateFunction);
106 ;
107
108 solver->setup(std::forward<NRConfig>(config).parameters);
109 return solver;
110 }
111}
112
123template <typename F, typename LS, typename UF>
125{
126public:
128 using SignatureTraits = typename F::Traits;
129
130 using Domain = typename SignatureTraits::Domain;
131 using CorrectionType = typename SignatureTraits::template Range<0>;
132 using JacobianType = typename SignatureTraits::template Range<1>;
135
139
146 template <typename LS2 = LS, typename UF2 = UF>
148 UF2&& updateFunction = {})
149 : residualFunction_{residual},
150 jacobianFunction_{derivative(residualFunction_)},
151 linearSolver_{std::forward<LS2>(linearSolver)},
152 updateFunction_{std::forward<UF2>(updateFunction)} {}
153
159 void setup(const Settings& settings) { settings_ = settings; }
160
171 template <typename SubsidiaryType>
172 [[nodiscard(
173 "The solve method returns information of the solution process. You should store this information and check if "
174 "it was successful")]] NonLinearSolverInformation
175 solve(Domain& req, SubsidiaryType&& subsidiaryFunction, SubsidiaryArgs& subsidiaryArgs) {
176 using enum NonLinearSolverMessages;
177 this->notify(INIT);
178
180 Ikarus::NonLinearSolverInformation solverInformation;
181 solverInformation.success = true;
182
183 auto& lambda = req.parameter();
184 auto& x = req.globalSolution();
185
189
190 auto lambdaDummy = lambda;
191 auto DDummy = x;
192 x.setZero();
193 lambda = 1.0;
194 decltype(auto) rx = residualFunction_(req);
195 const auto Fext0 = (-rx).eval(); // dRdlambda(lambda)
196 lambda = lambdaDummy;
197 x = DDummy;
198
199 rx = residualFunction_(req);
200 decltype(auto) Ax = jacobianFunction_(req);
201
202 Eigen::VectorXd deltaD;
203 deltaD.resizeLike(rx);
204 deltaD.setZero();
205
206 subsidiaryArgs.dfdDD.resizeLike(Fext0);
207
208 subsidiaryFunction(subsidiaryArgs);
209 auto rNorm = sqrt(rx.dot(rx));
210 decltype(rNorm) dNorm;
211 int iter{0};
212 if constexpr (isLinearSolver)
213 linearSolver_.analyzePattern(Ax);
214
215 auto solverState = typename NewtonRaphsonWithSubsidiaryFunction::State{.domain = req, .correction = correction_};
216
217 Eigen::MatrixX2<double> residual2d, sol2d;
218
220 while (rNorm > settings_.tol && iter < settings_.maxIter) {
221 this->notify(ITERATION_STARTED);
222
224 residual2d.resize(rx.rows(), 2);
225 residual2d << -rx, Fext0;
226 sol2d.resize(rx.rows(), 2);
227
228 if constexpr (isLinearSolver) {
229 linearSolver_.factorize(Ax);
230 linearSolver_.solve(sol2d, residual2d);
231 } else {
232 sol2d = linearSolver_(residual2d, Ax);
233 }
234
235 subsidiaryFunction(subsidiaryArgs);
236
237 const double deltalambda = (-subsidiaryArgs.f - subsidiaryArgs.dfdDD.dot(sol2d.col(0))) /
238 (subsidiaryArgs.dfdDD.dot(sol2d.col(1)) + subsidiaryArgs.dfdDlambda);
239 deltaD = sol2d.col(0) + deltalambda * sol2d.col(1);
240
241 solverState.dNorm = static_cast<double>(dNorm);
242 solverState.rNorm = static_cast<double>(rNorm);
243 solverState.iteration = iter;
244 this->notify(CORRECTION_UPDATED, solverState);
245
246 updateFunction_(x, deltaD);
247 updateFunction_(subsidiaryArgs.DD, deltaD);
248
249 lambda += deltalambda;
250 subsidiaryArgs.Dlambda += deltalambda;
251
252 dNorm = sqrt(deltaD.dot(deltaD) + deltalambda * deltalambda);
253 rx = residualFunction_(req);
254 Ax = jacobianFunction_(req);
255 rNorm = sqrt(rx.dot(rx) + subsidiaryArgs.f * subsidiaryArgs.f);
256
257 this->notify(SOLUTION_CHANGED, static_cast<double>(lambda));
258 this->notify(CORRECTIONNORM_UPDATED, static_cast<double>(dNorm));
259 this->notify(RESIDUALNORM_UPDATED, static_cast<double>(rNorm));
260 this->notify(ITERATION_ENDED);
261
262 ++iter;
263 }
264
265 if (iter == settings_.maxIter)
266 solverInformation.success = false;
267 solverInformation.iterations = iter;
268 solverInformation.residualNorm = rNorm;
269 solverInformation.correctionNorm = dNorm;
270 if (solverInformation.success)
271 this->notify(FINISHED_SUCESSFULLY, iter);
272
273 return solverInformation;
274 }
275
280 auto& residual() { return residualFunction_; }
281
282private:
283 DifferentiableFunction residualFunction_;
284 typename DifferentiableFunction::Derivative jacobianFunction_;
285 std::remove_cvref_t<CorrectionType> correction_;
286 LS linearSolver_;
287 UF updateFunction_;
288 Settings settings_;
289};
300template <typename F, typename LS = utils::SolverDefault, typename UF = utils::UpdateDefault>
301auto makeNewtonRaphsonWithSubsidiaryFunction(const F& f, LS&& linearSolver = {}, UF&& updateFunction = {}) {
302 return std::make_shared<NewtonRaphsonWithSubsidiaryFunction<F, LS, UF>>(f, std::forward<LS>(linearSolver),
303 std::move(updateFunction));
304}
305
306template <typename F, typename LS = utils::SolverDefault, typename UF = utils::UpdateDefault>
307NewtonRaphsonWithSubsidiaryFunction(const F& f, LS&& linearSolver = {}, UF&& updateFunction = {})
308 -> NewtonRaphsonWithSubsidiaryFunction<F, std::remove_cvref_t<LS>, std::remove_cvref_t<UF>>;
309
310} // namespace Ikarus
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.
Defines structures and methods related to subsidiary functions for control routines.
NonLinearSolverMessages
Enum class defining non-linear solver-related messages.
Definition: broadcastermessages.hh:22
Definition: assemblermanipulatorbuildingblocks.hh:22
NewtonRaphsonWithSubsidiaryFunction(const F &f, LS &&linearSolver={}, UF &&updateFunction={}) -> NewtonRaphsonWithSubsidiaryFunction< F, std::remove_cvref_t< LS >, std::remove_cvref_t< UF > >
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
auto makeNewtonRaphsonWithSubsidiaryFunction(const F &f, LS &&linearSolver={}, UF &&updateFunction={})
Function to create a NewtonRaphson with subsidiary function solver instance.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:301
Structure containing arguments for subsidiary functions.
Definition: pathfollowingfunctions.hh:39
double Dlambda
The increment in the load factor.
Definition: pathfollowingfunctions.hh:42
double dfdDlambda
The derivative of the subsidiary function with respect to Dlambda.
Definition: pathfollowingfunctions.hh:45
double f
The value of the subsidiary function.
Definition: pathfollowingfunctions.hh:43
Eigen::VectorX< double > dfdDD
The derivative of the subsidiary function with respect to DD.
Definition: pathfollowingfunctions.hh:44
Eigen::VectorX< double > DD
The vector representing the solution increment.
Definition: pathfollowingfunctions.hh:41
Newton-Raphson solver with subsidiary function.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:125
NewtonRaphsonWithSubsidiaryFunctionSettings Settings
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:127
auto & residual()
Access the residual function.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:280
void setup(const Settings &settings)
Setup the Newton-Raphson solver with subsidiary function.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:159
typename F::Traits SignatureTraits
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:128
NonLinearSolverInformation solve(Domain &req, SubsidiaryType &&subsidiaryFunction, SubsidiaryArgs &subsidiaryArgs)
Solve the nonlinear system using the Newton-Raphson method with subsidiary function.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:175
static constexpr bool isLinearSolver
Type representing the update function.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:134
typename SignatureTraits::template Range< 0 > CorrectionType
Type of the correction of x += deltaX.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:131
UF UpdateFunctionType
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:137
typename SignatureTraits::template Range< 1 > JacobianType
Compile-time boolean indicating if the linear solver satisfies the non-linear solver concept.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:133
typename SignatureTraits::Domain Domain
Type representing the parameter vector of the Function.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:130
NewtonRaphsonWithSubsidiaryFunction(const DifferentiableFunction &residual, LS2 &&linearSolver={}, UF2 &&updateFunction={})
Constructor for NewtonRaphsonWithSubsidiaryFunction.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:147
F DifferentiableFunction
Type of the non-linear operator.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:138
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:28
double tol
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:29
int maxIter
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:30
Settings for the Newton-Raphson solver with subsidiary function.
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:39
LS linearSolver
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:43
UF updateFunction
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:44
LS LinearSolver
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:40
NewtonRaphsonWithSubsidiaryFunctionSettings parameters
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:42
auto rebindUpdateFunction(UF2 &&updateFunction) const
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:47
UF UpdateFunction
Definition: newtonraphsonwithscalarsubsidiaryfunction.hh:41
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.