14#ifndef EIGEN_LMONESTEP_H
15#define EIGEN_LMONESTEP_H
18#include "./InternalHeaderCheck.h"
22template <
typename FunctorType>
23LevenbergMarquardtSpace::Status LevenbergMarquardt<FunctorType>::minimizeOneStep(FVectorType &x) {
26 RealScalar temp, temp1, temp2;
28 RealScalar pnorm, xnorm, fnorm1, actred, dirder, prered;
29 eigen_assert(x.size() == n);
34 Index df_ret = m_functor.df(x, m_fjac);
35 if (df_ret < 0)
return LevenbergMarquardtSpace::UserAsked;
43 for (
int j = 0; j < x.size(); ++j) m_wa2(j) = m_fjac.col(j).blueNorm();
44 QRSolver qrfac(m_fjac);
47 return LevenbergMarquardtSpace::ImproperInputParameters;
50 m_rfactor = qrfac.matrixR();
51 m_permutation = (qrfac.colsPermutation());
56 if (!m_useExternalScaling)
57 for (
Index j = 0; j < n; ++j) m_diag[j] = (m_wa2[j] == 0.) ? 1. : m_wa2[j];
61 xnorm = m_diag.cwiseProduct(x).stableNorm();
62 m_delta = m_factor * xnorm;
63 if (m_delta == 0.) m_delta = m_factor;
69 m_wa4 = qrfac.matrixQ().adjoint() * m_fvec;
70 m_qtf = m_wa4.head(n);
75 for (
Index j = 0; j < n; ++j)
76 if (m_wa2[m_permutation.indices()[j]] != 0.)
77 m_gnorm = (std::max)(m_gnorm,
abs(m_rfactor.col(j).head(j + 1).dot(m_qtf.head(j + 1) / m_fnorm) /
78 m_wa2[m_permutation.indices()[j]]));
81 if (m_gnorm <= m_gtol) {
83 return LevenbergMarquardtSpace::CosinusTooSmall;
87 if (!m_useExternalScaling) m_diag = m_diag.cwiseMax(m_wa2);
91 internal::lmpar2(qrfac, m_diag, m_qtf, m_delta, m_par, m_wa1);
96 pnorm = m_diag.cwiseProduct(m_wa1).stableNorm();
99 if (m_iter == 1) m_delta = (std::min)(m_delta, pnorm);
102 if (m_functor(m_wa2, m_wa4) < 0)
return LevenbergMarquardtSpace::UserAsked;
104 fnorm1 = m_wa4.stableNorm();
108 if (
Scalar(.1) * fnorm1 < m_fnorm) actred = 1. - numext::abs2(fnorm1 / m_fnorm);
112 m_wa3 = m_rfactor.template triangularView<Upper>() * (m_permutation.inverse() * m_wa1);
113 temp1 = numext::abs2(m_wa3.stableNorm() / m_fnorm);
114 temp2 = numext::abs2(
sqrt(m_par) * pnorm / m_fnorm);
115 prered = temp1 + temp2 /
Scalar(.5);
116 dirder = -(temp1 + temp2);
121 if (prered != 0.) ratio = actred / prered;
124 if (ratio <=
Scalar(.25)) {
125 if (actred >= 0.) temp = RealScalar(.5);
126 if (actred < 0.) temp = RealScalar(.5) * dirder / (dirder + RealScalar(.5) * actred);
127 if (RealScalar(.1) * fnorm1 >= m_fnorm || temp < RealScalar(.1)) temp =
Scalar(.1);
129 m_delta = temp * (std::min)(m_delta, pnorm / RealScalar(.1));
131 }
else if (!(m_par != 0. && ratio < RealScalar(.75))) {
132 m_delta = pnorm / RealScalar(.5);
133 m_par = RealScalar(.5) * m_par;
137 if (ratio >= RealScalar(1e-4)) {
140 m_wa2 = m_diag.cwiseProduct(x);
142 xnorm = m_wa2.stableNorm();
148 if (
abs(actred) <= m_ftol && prered <= m_ftol &&
Scalar(.5) * ratio <= 1. && m_delta <= m_xtol * xnorm) {
150 return LevenbergMarquardtSpace::RelativeErrorAndReductionTooSmall;
152 if (
abs(actred) <= m_ftol && prered <= m_ftol &&
Scalar(.5) * ratio <= 1.) {
154 return LevenbergMarquardtSpace::RelativeReductionTooSmall;
156 if (m_delta <= m_xtol * xnorm) {
158 return LevenbergMarquardtSpace::RelativeErrorTooSmall;
162 if (m_nfev >= m_maxfev) {
164 return LevenbergMarquardtSpace::TooManyFunctionEvaluation;
167 Scalar(.5) * ratio <= 1.) {
169 return LevenbergMarquardtSpace::FtolTooSmall;
173 return LevenbergMarquardtSpace::XtolTooSmall;
177 return LevenbergMarquardtSpace::GtolTooSmall;
180 }
while (ratio <
Scalar(1e-4));
182 return LevenbergMarquardtSpace::Running;
Namespace containing all symbols from the Eigen library.
const Eigen::CwiseUnaryOp< Eigen::internal::scalar_sqrt_op< typename Derived::Scalar >, const Derived > sqrt(const Eigen::ArrayBase< Derived > &x)
const Eigen::CwiseUnaryOp< Eigen::internal::scalar_abs_op< typename Derived::Scalar >, const Derived > abs(const Eigen::ArrayBase< Derived > &x)
EIGEN_DEFAULT_DENSE_INDEX_TYPE Index