mirror of
https://gitlab.com/libeigen/eigen.git
synced 2025-01-24 14:45:14 +08:00
cleaning, removing goto's, uniformization (try to reduce diff between
hybr[dj].h or lm[der,dif,str].h as much as possible), for future merging.
This commit is contained in:
parent
91a2145cb3
commit
63071ac968
@ -89,7 +89,7 @@ int ei_hybrd(
|
|||||||
while (true) {
|
while (true) {
|
||||||
jeval = true;
|
jeval = true;
|
||||||
|
|
||||||
/* calculate the jacobian matrix. */
|
/* calculate the jacobian matrix. */
|
||||||
|
|
||||||
iflag = ei_fdjac1<Functor,Scalar>(x, fvec, fjac,
|
iflag = ei_fdjac1<Functor,Scalar>(x, fvec, fjac,
|
||||||
nb_of_subdiagonals, nb_of_superdiagonals, epsfcn, wa1, wa2);
|
nb_of_subdiagonals, nb_of_superdiagonals, epsfcn, wa1, wa2);
|
||||||
@ -97,12 +97,12 @@ int ei_hybrd(
|
|||||||
if (iflag < 0)
|
if (iflag < 0)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* compute the qr factorization of the jacobian. */
|
/* compute the qr factorization of the jacobian. */
|
||||||
|
|
||||||
ei_qrfac<Scalar>(n, n, fjac.data(), fjac.rows(), false, iwa, 1, wa1.data(), wa2.data());
|
ei_qrfac<Scalar>(n, n, fjac.data(), fjac.rows(), false, iwa, 1, wa1.data(), wa2.data());
|
||||||
|
|
||||||
/* on the first iteration and if mode is 1, scale according */
|
/* on the first iteration and if mode is 1, scale according */
|
||||||
/* to the norms of the columns of the initial jacobian. */
|
/* to the norms of the columns of the initial jacobian. */
|
||||||
|
|
||||||
if (iter == 1) {
|
if (iter == 1) {
|
||||||
if (mode != 2)
|
if (mode != 2)
|
||||||
@ -112,8 +112,8 @@ int ei_hybrd(
|
|||||||
diag[j] = 1.;
|
diag[j] = 1.;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* on the first iteration, calculate the norm of the scaled x */
|
/* on the first iteration, calculate the norm of the scaled x */
|
||||||
/* and initialize the step bound delta. */
|
/* and initialize the step bound delta. */
|
||||||
|
|
||||||
wa3 = diag.cwise() * x;
|
wa3 = diag.cwise() * x;
|
||||||
xnorm = wa3.stableNorm();
|
xnorm = wa3.stableNorm();
|
||||||
@ -122,7 +122,7 @@ int ei_hybrd(
|
|||||||
delta = factor;
|
delta = factor;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* form (q transpose)*fvec and store in qtf. */
|
/* form (q transpose)*fvec and store in qtf. */
|
||||||
|
|
||||||
qtf = fvec;
|
qtf = fvec;
|
||||||
for (j = 0; j < n; ++j)
|
for (j = 0; j < n; ++j)
|
||||||
@ -135,7 +135,7 @@ int ei_hybrd(
|
|||||||
qtf[i] += fjac(i,j) * temp;
|
qtf[i] += fjac(i,j) * temp;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* copy the triangular factor of the qr factorization into r. */
|
/* copy the triangular factor of the qr factorization into r. */
|
||||||
|
|
||||||
sing = false;
|
sing = false;
|
||||||
for (j = 0; j < n; ++j) {
|
for (j = 0; j < n; ++j) {
|
||||||
@ -150,20 +150,20 @@ int ei_hybrd(
|
|||||||
sing = true;
|
sing = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* accumulate the orthogonal factor in fjac. */
|
/* accumulate the orthogonal factor in fjac. */
|
||||||
|
|
||||||
ei_qform<Scalar>(n, n, fjac.data(), fjac.rows(), wa1.data());
|
ei_qform<Scalar>(n, n, fjac.data(), fjac.rows(), wa1.data());
|
||||||
|
|
||||||
/* rescale if necessary. */
|
/* rescale if necessary. */
|
||||||
|
|
||||||
/* Computing MAX */
|
/* Computing MAX */
|
||||||
if (mode != 2)
|
if (mode != 2)
|
||||||
diag = diag.cwise().max(wa2);
|
diag = diag.cwise().max(wa2);
|
||||||
|
|
||||||
/* beginning of the inner loop. */
|
/* beginning of the inner loop. */
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
/* if requested, call Functor::f to enable printing of iterates. */
|
/* if requested, call Functor::f to enable printing of iterates. */
|
||||||
|
|
||||||
if (nprint > 0) {
|
if (nprint > 0) {
|
||||||
iflag = 0;
|
iflag = 0;
|
||||||
@ -173,23 +173,23 @@ int ei_hybrd(
|
|||||||
goto L300;
|
goto L300;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* determine the direction p. */
|
/* determine the direction p. */
|
||||||
|
|
||||||
ei_dogleg<Scalar>(R, diag, qtf, delta, wa1);
|
ei_dogleg<Scalar>(R, diag, qtf, delta, wa1);
|
||||||
|
|
||||||
/* store the direction p and x + p. calculate the norm of p. */
|
/* store the direction p and x + p. calculate the norm of p. */
|
||||||
|
|
||||||
wa1 = -wa1;
|
wa1 = -wa1;
|
||||||
wa2 = x + wa1;
|
wa2 = x + wa1;
|
||||||
wa3 = diag.cwise() * wa1;
|
wa3 = diag.cwise() * wa1;
|
||||||
pnorm = wa3.stableNorm();
|
pnorm = wa3.stableNorm();
|
||||||
|
|
||||||
/* on the first iteration, adjust the initial step bound. */
|
/* on the first iteration, adjust the initial step bound. */
|
||||||
|
|
||||||
if (iter == 1)
|
if (iter == 1)
|
||||||
delta = std::min(delta,pnorm);
|
delta = std::min(delta,pnorm);
|
||||||
|
|
||||||
/* evaluate the function at x + p and calculate its norm. */
|
/* evaluate the function at x + p and calculate its norm. */
|
||||||
|
|
||||||
iflag = Functor::f(wa2, wa4);
|
iflag = Functor::f(wa2, wa4);
|
||||||
++nfev;
|
++nfev;
|
||||||
@ -197,13 +197,13 @@ int ei_hybrd(
|
|||||||
goto L300;
|
goto L300;
|
||||||
fnorm1 = wa4.stableNorm();
|
fnorm1 = wa4.stableNorm();
|
||||||
|
|
||||||
/* compute the scaled actual reduction. */
|
/* compute the scaled actual reduction. */
|
||||||
|
|
||||||
actred = -1.;
|
actred = -1.;
|
||||||
if (fnorm1 < fnorm) /* Computing 2nd power */
|
if (fnorm1 < fnorm) /* Computing 2nd power */
|
||||||
actred = 1. - ei_abs2(fnorm1 / fnorm);
|
actred = 1. - ei_abs2(fnorm1 / fnorm);
|
||||||
|
|
||||||
/* compute the scaled predicted reduction. */
|
/* compute the scaled predicted reduction. */
|
||||||
|
|
||||||
l = 0;
|
l = 0;
|
||||||
for (i = 0; i < n; ++i) {
|
for (i = 0; i < n; ++i) {
|
||||||
@ -219,14 +219,14 @@ int ei_hybrd(
|
|||||||
if (temp < fnorm) /* Computing 2nd power */
|
if (temp < fnorm) /* Computing 2nd power */
|
||||||
prered = 1. - ei_abs2(temp / fnorm);
|
prered = 1. - ei_abs2(temp / fnorm);
|
||||||
|
|
||||||
/* compute the ratio of the actual to the predicted */
|
/* compute the ratio of the actual to the predicted */
|
||||||
/* reduction. */
|
/* reduction. */
|
||||||
|
|
||||||
ratio = 0.;
|
ratio = 0.;
|
||||||
if (prered > 0.)
|
if (prered > 0.)
|
||||||
ratio = actred / prered;
|
ratio = actred / prered;
|
||||||
|
|
||||||
/* update the step bound. */
|
/* update the step bound. */
|
||||||
|
|
||||||
if (ratio < Scalar(.1)) {
|
if (ratio < Scalar(.1)) {
|
||||||
ncsuc = 0;
|
ncsuc = 0;
|
||||||
@ -242,10 +242,10 @@ int ei_hybrd(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* test for successful iteration. */
|
/* test for successful iteration. */
|
||||||
|
|
||||||
if (ratio >= Scalar(1e-4)) {
|
if (ratio >= Scalar(1e-4)) {
|
||||||
/* successful iteration. update x, fvec, and their norms. */
|
/* successful iteration. update x, fvec, and their norms. */
|
||||||
x = wa2;
|
x = wa2;
|
||||||
wa2 = diag.cwise() * x;
|
wa2 = diag.cwise() * x;
|
||||||
fvec = wa4;
|
fvec = wa4;
|
||||||
@ -254,7 +254,7 @@ int ei_hybrd(
|
|||||||
++iter;
|
++iter;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* determine the progress of the iteration. */
|
/* determine the progress of the iteration. */
|
||||||
|
|
||||||
++nslow1;
|
++nslow1;
|
||||||
if (actred >= Scalar(.001))
|
if (actred >= Scalar(.001))
|
||||||
@ -264,14 +264,14 @@ int ei_hybrd(
|
|||||||
if (actred >= Scalar(.1))
|
if (actred >= Scalar(.1))
|
||||||
nslow2 = 0;
|
nslow2 = 0;
|
||||||
|
|
||||||
/* test for convergence. */
|
/* test for convergence. */
|
||||||
|
|
||||||
if (delta <= xtol * xnorm || fnorm == 0.)
|
if (delta <= xtol * xnorm || fnorm == 0.)
|
||||||
info = 1;
|
info = 1;
|
||||||
if (info != 0)
|
if (info != 0)
|
||||||
goto L300;
|
goto L300;
|
||||||
|
|
||||||
/* tests for termination and stringent tolerances. */
|
/* tests for termination and stringent tolerances. */
|
||||||
|
|
||||||
if (nfev >= maxfev)
|
if (nfev >= maxfev)
|
||||||
info = 2;
|
info = 2;
|
||||||
@ -285,14 +285,14 @@ int ei_hybrd(
|
|||||||
if (info != 0)
|
if (info != 0)
|
||||||
goto L300;
|
goto L300;
|
||||||
|
|
||||||
/* criterion for recalculating jacobian approximation */
|
/* criterion for recalculating jacobian approximation */
|
||||||
/* by forward differences. */
|
/* by forward differences. */
|
||||||
|
|
||||||
if (ncfail == 2)
|
if (ncfail == 2)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* calculate the rank one modification to the jacobian */
|
/* calculate the rank one modification to the jacobian */
|
||||||
/* and update qtf if necessary. */
|
/* and update qtf if necessary. */
|
||||||
|
|
||||||
for (j = 0; j < n; ++j) {
|
for (j = 0; j < n; ++j) {
|
||||||
sum = wa4.dot(fjac.col(j));
|
sum = wa4.dot(fjac.col(j));
|
||||||
@ -302,17 +302,17 @@ int ei_hybrd(
|
|||||||
qtf[j] = sum;
|
qtf[j] = sum;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* compute the qr factorization of the updated jacobian. */
|
/* compute the qr factorization of the updated jacobian. */
|
||||||
|
|
||||||
ei_r1updt<Scalar>(n, n, R.data(), R.size(), wa1.data(), wa2.data(), wa3.data(), &sing);
|
ei_r1updt<Scalar>(n, n, R.data(), R.size(), wa1.data(), wa2.data(), wa3.data(), &sing);
|
||||||
ei_r1mpyq<Scalar>(n, n, fjac.data(), fjac.rows(), wa2.data(), wa3.data());
|
ei_r1mpyq<Scalar>(n, n, fjac.data(), fjac.rows(), wa2.data(), wa3.data());
|
||||||
ei_r1mpyq<Scalar>(1, n, qtf.data(), 1, wa2.data(), wa3.data());
|
ei_r1mpyq<Scalar>(1, n, qtf.data(), 1, wa2.data(), wa3.data());
|
||||||
|
|
||||||
/* end of the inner loop. */
|
/* end of the inner loop. */
|
||||||
|
|
||||||
jeval = false;
|
jeval = false;
|
||||||
}
|
}
|
||||||
/* end of the outer loop. */
|
/* end of the outer loop. */
|
||||||
}
|
}
|
||||||
L300:
|
L300:
|
||||||
/* termination, either normal or user imposed. */
|
/* termination, either normal or user imposed. */
|
||||||
|
@ -78,19 +78,19 @@ int ei_hybrj(
|
|||||||
while (true) {
|
while (true) {
|
||||||
jeval = true;
|
jeval = true;
|
||||||
|
|
||||||
/* calculate the jacobian matrix. */
|
/* calculate the jacobian matrix. */
|
||||||
|
|
||||||
iflag = Functor::df(x, fjac);
|
iflag = Functor::df(x, fjac);
|
||||||
++njev;
|
++njev;
|
||||||
if (iflag < 0)
|
if (iflag < 0)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* compute the qr factorization of the jacobian. */
|
/* compute the qr factorization of the jacobian. */
|
||||||
|
|
||||||
ei_qrfac<Scalar>(n, n, fjac.data(), fjac.rows(), false, iwa, 1, wa1.data(), wa2.data());
|
ei_qrfac<Scalar>(n, n, fjac.data(), fjac.rows(), false, iwa, 1, wa1.data(), wa2.data());
|
||||||
|
|
||||||
/* on the first iteration and if mode is 1, scale according */
|
/* on the first iteration and if mode is 1, scale according */
|
||||||
/* to the norms of the columns of the initial jacobian. */
|
/* to the norms of the columns of the initial jacobian. */
|
||||||
|
|
||||||
if (iter == 1) {
|
if (iter == 1) {
|
||||||
if (mode != 2)
|
if (mode != 2)
|
||||||
@ -100,8 +100,8 @@ int ei_hybrj(
|
|||||||
diag[j] = 1.;
|
diag[j] = 1.;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* on the first iteration, calculate the norm of the scaled x */
|
/* on the first iteration, calculate the norm of the scaled x */
|
||||||
/* and initialize the step bound delta. */
|
/* and initialize the step bound delta. */
|
||||||
|
|
||||||
wa3 = diag.cwise() * x;
|
wa3 = diag.cwise() * x;
|
||||||
xnorm = wa3.stableNorm();
|
xnorm = wa3.stableNorm();
|
||||||
@ -110,7 +110,7 @@ int ei_hybrj(
|
|||||||
delta = factor;
|
delta = factor;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* form (q transpose)*fvec and store in qtf. */
|
/* form (q transpose)*fvec and store in qtf. */
|
||||||
|
|
||||||
qtf = fvec;
|
qtf = fvec;
|
||||||
for (j = 0; j < n; ++j)
|
for (j = 0; j < n; ++j)
|
||||||
@ -123,7 +123,7 @@ int ei_hybrj(
|
|||||||
qtf[i] += fjac(i,j) * temp;
|
qtf[i] += fjac(i,j) * temp;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* copy the triangular factor of the qr factorization into r. */
|
/* copy the triangular factor of the qr factorization into r. */
|
||||||
|
|
||||||
sing = false;
|
sing = false;
|
||||||
for (j = 0; j < n; ++j) {
|
for (j = 0; j < n; ++j) {
|
||||||
@ -138,20 +138,20 @@ int ei_hybrj(
|
|||||||
sing = true;
|
sing = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* accumulate the orthogonal factor in fjac. */
|
/* accumulate the orthogonal factor in fjac. */
|
||||||
|
|
||||||
ei_qform<Scalar>(n, n, fjac.data(), fjac.rows(), wa1.data());
|
ei_qform<Scalar>(n, n, fjac.data(), fjac.rows(), wa1.data());
|
||||||
|
|
||||||
/* rescale if necessary. */
|
/* rescale if necessary. */
|
||||||
|
|
||||||
/* Computing MAX */
|
/* Computing MAX */
|
||||||
if (mode != 2)
|
if (mode != 2)
|
||||||
diag = diag.cwise().max(wa2);
|
diag = diag.cwise().max(wa2);
|
||||||
|
|
||||||
/* beginning of the inner loop. */
|
/* beginning of the inner loop. */
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
/* if requested, call Functor::f to enable printing of iterates. */
|
/* if requested, call Functor::f to enable printing of iterates. */
|
||||||
|
|
||||||
if (nprint > 0) {
|
if (nprint > 0) {
|
||||||
iflag = 0;
|
iflag = 0;
|
||||||
@ -161,23 +161,23 @@ int ei_hybrj(
|
|||||||
goto L300;
|
goto L300;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* determine the direction p. */
|
/* determine the direction p. */
|
||||||
|
|
||||||
ei_dogleg<Scalar>(R, diag, qtf, delta, wa1);
|
ei_dogleg<Scalar>(R, diag, qtf, delta, wa1);
|
||||||
|
|
||||||
/* store the direction p and x + p. calculate the norm of p. */
|
/* store the direction p and x + p. calculate the norm of p. */
|
||||||
|
|
||||||
wa1 = -wa1;
|
wa1 = -wa1;
|
||||||
wa2 = x + wa1;
|
wa2 = x + wa1;
|
||||||
wa3 = diag.cwise() * wa1;
|
wa3 = diag.cwise() * wa1;
|
||||||
pnorm = wa3.stableNorm();
|
pnorm = wa3.stableNorm();
|
||||||
|
|
||||||
/* on the first iteration, adjust the initial step bound. */
|
/* on the first iteration, adjust the initial step bound. */
|
||||||
|
|
||||||
if (iter == 1)
|
if (iter == 1)
|
||||||
delta = std::min(delta,pnorm);
|
delta = std::min(delta,pnorm);
|
||||||
|
|
||||||
/* evaluate the function at x + p and calculate its norm. */
|
/* evaluate the function at x + p and calculate its norm. */
|
||||||
|
|
||||||
iflag = Functor::f(wa2, wa4);
|
iflag = Functor::f(wa2, wa4);
|
||||||
++nfev;
|
++nfev;
|
||||||
@ -185,13 +185,13 @@ int ei_hybrj(
|
|||||||
goto L300;
|
goto L300;
|
||||||
fnorm1 = wa4.stableNorm();
|
fnorm1 = wa4.stableNorm();
|
||||||
|
|
||||||
/* compute the scaled actual reduction. */
|
/* compute the scaled actual reduction. */
|
||||||
|
|
||||||
actred = -1.;
|
actred = -1.;
|
||||||
if (fnorm1 < fnorm) /* Computing 2nd power */
|
if (fnorm1 < fnorm) /* Computing 2nd power */
|
||||||
actred = 1. - ei_abs2(fnorm1 / fnorm);
|
actred = 1. - ei_abs2(fnorm1 / fnorm);
|
||||||
|
|
||||||
/* compute the scaled predicted reduction. */
|
/* compute the scaled predicted reduction. */
|
||||||
|
|
||||||
l = 0;
|
l = 0;
|
||||||
for (i = 0; i < n; ++i) {
|
for (i = 0; i < n; ++i) {
|
||||||
@ -207,14 +207,14 @@ int ei_hybrj(
|
|||||||
if (temp < fnorm) /* Computing 2nd power */
|
if (temp < fnorm) /* Computing 2nd power */
|
||||||
prered = 1. - ei_abs2(temp / fnorm);
|
prered = 1. - ei_abs2(temp / fnorm);
|
||||||
|
|
||||||
/* compute the ratio of the actual to the predicted */
|
/* compute the ratio of the actual to the predicted */
|
||||||
/* reduction. */
|
/* reduction. */
|
||||||
|
|
||||||
ratio = 0.;
|
ratio = 0.;
|
||||||
if (prered > 0.)
|
if (prered > 0.)
|
||||||
ratio = actred / prered;
|
ratio = actred / prered;
|
||||||
|
|
||||||
/* update the step bound. */
|
/* update the step bound. */
|
||||||
|
|
||||||
if (ratio < Scalar(.1)) {
|
if (ratio < Scalar(.1)) {
|
||||||
ncsuc = 0;
|
ncsuc = 0;
|
||||||
@ -230,10 +230,10 @@ int ei_hybrj(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* test for successful iteration. */
|
/* test for successful iteration. */
|
||||||
|
|
||||||
if (ratio >= Scalar(1e-4)) {
|
if (ratio >= Scalar(1e-4)) {
|
||||||
/* successful iteration. update x, fvec, and their norms. */
|
/* successful iteration. update x, fvec, and their norms. */
|
||||||
x = wa2;
|
x = wa2;
|
||||||
wa2 = diag.cwise() * x;
|
wa2 = diag.cwise() * x;
|
||||||
fvec = wa4;
|
fvec = wa4;
|
||||||
@ -242,7 +242,7 @@ int ei_hybrj(
|
|||||||
++iter;
|
++iter;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* determine the progress of the iteration. */
|
/* determine the progress of the iteration. */
|
||||||
|
|
||||||
++nslow1;
|
++nslow1;
|
||||||
if (actred >= Scalar(.001))
|
if (actred >= Scalar(.001))
|
||||||
@ -252,14 +252,14 @@ int ei_hybrj(
|
|||||||
if (actred >= Scalar(.1))
|
if (actred >= Scalar(.1))
|
||||||
nslow2 = 0;
|
nslow2 = 0;
|
||||||
|
|
||||||
/* test for convergence. */
|
/* test for convergence. */
|
||||||
|
|
||||||
if (delta <= xtol * xnorm || fnorm == 0.)
|
if (delta <= xtol * xnorm || fnorm == 0.)
|
||||||
info = 1;
|
info = 1;
|
||||||
if (info != 0)
|
if (info != 0)
|
||||||
goto L300;
|
goto L300;
|
||||||
|
|
||||||
/* tests for termination and stringent tolerances. */
|
/* tests for termination and stringent tolerances. */
|
||||||
|
|
||||||
if (nfev >= maxfev)
|
if (nfev >= maxfev)
|
||||||
info = 2;
|
info = 2;
|
||||||
@ -273,13 +273,13 @@ int ei_hybrj(
|
|||||||
if (info != 0)
|
if (info != 0)
|
||||||
goto L300;
|
goto L300;
|
||||||
|
|
||||||
/* criterion for recalculating jacobian. */
|
/* criterion for recalculating jacobian. */
|
||||||
|
|
||||||
if (ncfail == 2)
|
if (ncfail == 2)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* calculate the rank one modification to the jacobian */
|
/* calculate the rank one modification to the jacobian */
|
||||||
/* and update qtf if necessary. */
|
/* and update qtf if necessary. */
|
||||||
|
|
||||||
for (j = 0; j < n; ++j) {
|
for (j = 0; j < n; ++j) {
|
||||||
sum = wa4.dot(fjac.col(j));
|
sum = wa4.dot(fjac.col(j));
|
||||||
@ -289,17 +289,17 @@ int ei_hybrj(
|
|||||||
qtf[j] = sum;
|
qtf[j] = sum;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* compute the qr factorization of the updated jacobian. */
|
/* compute the qr factorization of the updated jacobian. */
|
||||||
|
|
||||||
ei_r1updt<Scalar>(n, n, R.data(), R.size(), wa1.data(), wa2.data(), wa3.data(), &sing);
|
ei_r1updt<Scalar>(n, n, R.data(), R.size(), wa1.data(), wa2.data(), wa3.data(), &sing);
|
||||||
ei_r1mpyq<Scalar>(n, n, fjac.data(), fjac.rows(), wa2.data(), wa3.data());
|
ei_r1mpyq<Scalar>(n, n, fjac.data(), fjac.rows(), wa2.data(), wa3.data());
|
||||||
ei_r1mpyq<Scalar>(1, n, qtf.data(), 1, wa2.data(), wa3.data());
|
ei_r1mpyq<Scalar>(1, n, qtf.data(), 1, wa2.data(), wa3.data());
|
||||||
|
|
||||||
/* end of the inner loop. */
|
/* end of the inner loop. */
|
||||||
|
|
||||||
jeval = false;
|
jeval = false;
|
||||||
}
|
}
|
||||||
/* end of the outer loop. */
|
/* end of the outer loop. */
|
||||||
}
|
}
|
||||||
L300:
|
L300:
|
||||||
/* termination, either normal or user imposed. */
|
/* termination, either normal or user imposed. */
|
||||||
|
@ -46,8 +46,7 @@ int ei_lmder(
|
|||||||
|
|
||||||
/* check the input parameters for errors. */
|
/* check the input parameters for errors. */
|
||||||
|
|
||||||
if (n <= 0 || m < n || ftol < 0. || xtol < 0. ||
|
if (n <= 0 || m < n || ftol < 0. || xtol < 0. || gtol < 0. || maxfev <= 0 || factor <= 0.)
|
||||||
gtol < 0. || maxfev <= 0 || factor <= 0.)
|
|
||||||
goto L300;
|
goto L300;
|
||||||
|
|
||||||
if (mode == 2)
|
if (mode == 2)
|
||||||
@ -70,16 +69,16 @@ int ei_lmder(
|
|||||||
|
|
||||||
/* beginning of the outer loop. */
|
/* beginning of the outer loop. */
|
||||||
|
|
||||||
while(true) {
|
while (true) {
|
||||||
|
|
||||||
/* calculate the jacobian matrix. */
|
/* calculate the jacobian matrix. */
|
||||||
|
|
||||||
iflag = Functor::df(x, fjac);
|
iflag = Functor::df(x, fjac);
|
||||||
++njev;
|
++njev;
|
||||||
if (iflag < 0)
|
if (iflag < 0)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* if requested, call Functor::f to enable printing of iterates. */
|
/* if requested, call Functor::f to enable printing of iterates. */
|
||||||
|
|
||||||
if (nprint > 0) {
|
if (nprint > 0) {
|
||||||
iflag = 0;
|
iflag = 0;
|
||||||
@ -89,13 +88,13 @@ int ei_lmder(
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* compute the qr factorization of the jacobian. */
|
/* compute the qr factorization of the jacobian. */
|
||||||
|
|
||||||
ei_qrfac<Scalar>(m, n, fjac.data(), fjac.rows(), true, ipvt.data(), n, wa1.data(), wa2.data());
|
ei_qrfac<Scalar>(m, n, fjac.data(), fjac.rows(), true, ipvt.data(), n, wa1.data(), wa2.data());
|
||||||
ipvt.cwise()-=1; // qrfac() creates ipvt with fortran convetion (1->n), convert it to c (0->n-1)
|
ipvt.cwise()-=1; // qrfac() creates ipvt with fortran convetion (1->n), convert it to c (0->n-1)
|
||||||
|
|
||||||
/* on the first iteration and if mode is 1, scale according */
|
/* on the first iteration and if mode is 1, scale according */
|
||||||
/* to the norms of the columns of the initial jacobian. */
|
/* to the norms of the columns of the initial jacobian. */
|
||||||
|
|
||||||
if (iter == 1) {
|
if (iter == 1) {
|
||||||
if (mode != 2)
|
if (mode != 2)
|
||||||
@ -104,8 +103,10 @@ int ei_lmder(
|
|||||||
if (wa2[j] == 0.)
|
if (wa2[j] == 0.)
|
||||||
diag[j] = 1.;
|
diag[j] = 1.;
|
||||||
}
|
}
|
||||||
/* on the first iteration, calculate the norm of the scaled x */
|
|
||||||
/* and initialize the step bound delta. */
|
/* on the first iteration, calculate the norm of the scaled x */
|
||||||
|
/* and initialize the step bound delta. */
|
||||||
|
|
||||||
wa3 = diag.cwise() * x;
|
wa3 = diag.cwise() * x;
|
||||||
xnorm = wa3.stableNorm();
|
xnorm = wa3.stableNorm();
|
||||||
delta = factor * xnorm;
|
delta = factor * xnorm;
|
||||||
@ -113,8 +114,8 @@ int ei_lmder(
|
|||||||
delta = factor;
|
delta = factor;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* form (q transpose)*fvec and store the first n components in */
|
/* form (q transpose)*fvec and store the first n components in */
|
||||||
/* qtf. */
|
/* qtf. */
|
||||||
|
|
||||||
wa4 = fvec;
|
wa4 = fvec;
|
||||||
for (j = 0; j < n; ++j) {
|
for (j = 0; j < n; ++j) {
|
||||||
@ -130,7 +131,7 @@ int ei_lmder(
|
|||||||
qtf[j] = wa4[j];
|
qtf[j] = wa4[j];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* compute the norm of the scaled gradient. */
|
/* compute the norm of the scaled gradient. */
|
||||||
|
|
||||||
gnorm = 0.;
|
gnorm = 0.;
|
||||||
if (fnorm != 0.)
|
if (fnorm != 0.)
|
||||||
@ -145,21 +146,21 @@ int ei_lmder(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* test for convergence of the gradient norm. */
|
/* test for convergence of the gradient norm. */
|
||||||
|
|
||||||
if (gnorm <= gtol) {
|
if (gnorm <= gtol)
|
||||||
info = 4;
|
info = 4;
|
||||||
}
|
|
||||||
if (info != 0)
|
if (info != 0)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* rescale if necessary. */
|
/* rescale if necessary. */
|
||||||
|
|
||||||
if (mode != 2) /* Computing MAX */
|
if (mode != 2) /* Computing MAX */
|
||||||
diag = diag.cwise().max(wa2);
|
diag = diag.cwise().max(wa2);
|
||||||
|
|
||||||
/* beginning of the inner loop. */
|
/* beginning of the inner loop. */
|
||||||
do {
|
do {
|
||||||
|
|
||||||
/* determine the levenberg-marquardt parameter. */
|
/* determine the levenberg-marquardt parameter. */
|
||||||
|
|
||||||
ei_lmpar<Scalar>(fjac, ipvt, diag, qtf, delta, par, wa1, wa2);
|
ei_lmpar<Scalar>(fjac, ipvt, diag, qtf, delta, par, wa1, wa2);
|
||||||
@ -173,9 +174,8 @@ int ei_lmder(
|
|||||||
|
|
||||||
/* on the first iteration, adjust the initial step bound. */
|
/* on the first iteration, adjust the initial step bound. */
|
||||||
|
|
||||||
if (iter == 1) {
|
if (iter == 1)
|
||||||
delta = std::min(delta,pnorm);
|
delta = std::min(delta,pnorm);
|
||||||
}
|
|
||||||
|
|
||||||
/* evaluate the function at x + p and calculate its norm. */
|
/* evaluate the function at x + p and calculate its norm. */
|
||||||
|
|
||||||
@ -198,9 +198,8 @@ int ei_lmder(
|
|||||||
for (j = 0; j < n; ++j) {
|
for (j = 0; j < n; ++j) {
|
||||||
l = ipvt[j];
|
l = ipvt[j];
|
||||||
temp = wa1[l];
|
temp = wa1[l];
|
||||||
for (i = 0; i <= j; ++i) {
|
for (i = 0; i <= j; ++i)
|
||||||
wa3[i] += fjac(i,j) * temp;
|
wa3[i] += fjac(i,j) * temp;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
temp1 = ei_abs2(wa3.stableNorm() / fnorm);
|
temp1 = ei_abs2(wa3.stableNorm() / fnorm);
|
||||||
temp2 = ei_abs2(ei_sqrt(par) * pnorm / fnorm);
|
temp2 = ei_abs2(ei_sqrt(par) * pnorm / fnorm);
|
||||||
@ -227,12 +226,9 @@ int ei_lmder(
|
|||||||
/* Computing MIN */
|
/* Computing MIN */
|
||||||
delta = temp * std::min(delta, pnorm / Scalar(.1));
|
delta = temp * std::min(delta, pnorm / Scalar(.1));
|
||||||
par /= temp;
|
par /= temp;
|
||||||
}
|
} else if (!(par != 0. && ratio < Scalar(.75))) {
|
||||||
else {
|
delta = pnorm / Scalar(.5);
|
||||||
if (!(par != 0. && ratio < Scalar(.75))) {
|
par = Scalar(.5) * par;
|
||||||
delta = pnorm / Scalar(.5);
|
|
||||||
par = Scalar(.5) * par;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* test for successful iteration. */
|
/* test for successful iteration. */
|
||||||
@ -272,12 +268,11 @@ int ei_lmder(
|
|||||||
goto L300;
|
goto L300;
|
||||||
/* end of the inner loop. repeat if iteration unsuccessful. */
|
/* end of the inner loop. repeat if iteration unsuccessful. */
|
||||||
} while (ratio < Scalar(1e-4));
|
} while (ratio < Scalar(1e-4));
|
||||||
/* end of the outer loop. */
|
/* end of the outer loop. */
|
||||||
}
|
}
|
||||||
L300:
|
L300:
|
||||||
|
|
||||||
/* termination, either normal or user imposed. */
|
/* termination, either normal or user imposed. */
|
||||||
|
|
||||||
if (iflag < 0)
|
if (iflag < 0)
|
||||||
info = iflag;
|
info = iflag;
|
||||||
if (nprint > 0)
|
if (nprint > 0)
|
||||||
|
@ -45,10 +45,8 @@ int ei_lmdif(
|
|||||||
|
|
||||||
/* check the input parameters for errors. */
|
/* check the input parameters for errors. */
|
||||||
|
|
||||||
if (n <= 0 || m < n || ftol < 0. || xtol < 0. ||
|
if (n <= 0 || m < n || ftol < 0. || xtol < 0. || gtol < 0. || maxfev <= 0 || factor <= 0.)
|
||||||
gtol < 0. || maxfev <= 0 || factor <= 0.) {
|
|
||||||
goto L300;
|
goto L300;
|
||||||
}
|
|
||||||
if (mode == 2)
|
if (mode == 2)
|
||||||
for (j = 0; j < n; ++j)
|
for (j = 0; j < n; ++j)
|
||||||
if (diag[j] <= 0.) goto L300;
|
if (diag[j] <= 0.) goto L300;
|
||||||
@ -58,9 +56,8 @@ int ei_lmdif(
|
|||||||
|
|
||||||
iflag = Functor::f(x, fvec);
|
iflag = Functor::f(x, fvec);
|
||||||
nfev = 1;
|
nfev = 1;
|
||||||
if (iflag < 0) {
|
if (iflag < 0)
|
||||||
goto L300;
|
goto L300;
|
||||||
}
|
|
||||||
fnorm = fvec.stableNorm();
|
fnorm = fvec.stableNorm();
|
||||||
|
|
||||||
/* initialize levenberg-marquardt parameter and iteration counter. */
|
/* initialize levenberg-marquardt parameter and iteration counter. */
|
||||||
@ -70,283 +67,214 @@ int ei_lmdif(
|
|||||||
|
|
||||||
/* beginning of the outer loop. */
|
/* beginning of the outer loop. */
|
||||||
|
|
||||||
L30:
|
while (true) {
|
||||||
|
|
||||||
/* calculate the jacobian matrix. */
|
/* calculate the jacobian matrix. */
|
||||||
|
|
||||||
iflag = ei_fdjac2<Functor,Scalar>(x, fvec, fjac, epsfcn, wa4);
|
iflag = ei_fdjac2<Functor,Scalar>(x, fvec, fjac, epsfcn, wa4);
|
||||||
nfev += n;
|
nfev += n;
|
||||||
if (iflag < 0) {
|
if (iflag < 0)
|
||||||
goto L300;
|
break;
|
||||||
}
|
|
||||||
|
|
||||||
/* if requested, call Functor::f to enable printing of iterates. */
|
/* if requested, call Functor::f to enable printing of iterates. */
|
||||||
|
|
||||||
if (nprint <= 0) {
|
if (nprint > 0) {
|
||||||
goto L40;
|
iflag = 0;
|
||||||
}
|
if ((iter - 1) % nprint == 0)
|
||||||
iflag = 0;
|
iflag = Functor::debug(x, fvec);
|
||||||
if ((iter - 1) % nprint == 0) {
|
if (iflag < 0)
|
||||||
iflag = Functor::debug(x, fvec);
|
break;
|
||||||
}
|
|
||||||
if (iflag < 0) {
|
|
||||||
goto L300;
|
|
||||||
}
|
|
||||||
L40:
|
|
||||||
|
|
||||||
/* compute the qr factorization of the jacobian. */
|
|
||||||
|
|
||||||
ei_qrfac<Scalar>(m, n, fjac.data(), fjac.rows(), true, ipvt.data(), n, wa1.data(), wa2.data());
|
|
||||||
ipvt.cwise()-=1; // qrfac() creates ipvt with fortran convetion (1->n), convert it to c (0->n-1)
|
|
||||||
|
|
||||||
/* on the first iteration and if mode is 1, scale according */
|
|
||||||
/* to the norms of the columns of the initial jacobian. */
|
|
||||||
|
|
||||||
if (iter != 1) {
|
|
||||||
goto L80;
|
|
||||||
}
|
|
||||||
if (mode == 2) {
|
|
||||||
goto L60;
|
|
||||||
}
|
|
||||||
for (j = 0; j < n; ++j) {
|
|
||||||
diag[j] = wa2[j];
|
|
||||||
if (wa2[j] == 0.) {
|
|
||||||
diag[j] = 1.;
|
|
||||||
}
|
}
|
||||||
/* L50: */
|
|
||||||
}
|
|
||||||
L60:
|
|
||||||
|
|
||||||
/* on the first iteration, calculate the norm of the scaled x */
|
/* compute the qr factorization of the jacobian. */
|
||||||
/* and initialize the step bound delta. */
|
|
||||||
|
|
||||||
wa3 = diag.cwise() * x;
|
ei_qrfac<Scalar>(m, n, fjac.data(), fjac.rows(), true, ipvt.data(), n, wa1.data(), wa2.data());
|
||||||
xnorm = wa3.stableNorm();
|
ipvt.cwise()-=1; // qrfac() creates ipvt with fortran convetion (1->n), convert it to c (0->n-1)
|
||||||
delta = factor * xnorm;
|
|
||||||
if (delta == 0.) {
|
|
||||||
delta = factor;
|
|
||||||
}
|
|
||||||
L80:
|
|
||||||
|
|
||||||
/* form (q transpose)*fvec and store the first n components in */
|
/* on the first iteration and if mode is 1, scale according */
|
||||||
/* qtf. */
|
/* to the norms of the columns of the initial jacobian. */
|
||||||
|
|
||||||
wa4 = fvec;
|
if (iter == 1) {
|
||||||
for (j = 0; j < n; ++j) {
|
if (mode != 2)
|
||||||
if (fjac(j,j) == 0.) {
|
for (j = 0; j < n; ++j) {
|
||||||
goto L120;
|
diag[j] = wa2[j];
|
||||||
|
if (wa2[j] == 0.)
|
||||||
|
diag[j] = 1.;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* on the first iteration, calculate the norm of the scaled x */
|
||||||
|
/* and initialize the step bound delta. */
|
||||||
|
|
||||||
|
wa3 = diag.cwise() * x;
|
||||||
|
xnorm = wa3.stableNorm();
|
||||||
|
delta = factor * xnorm;
|
||||||
|
if (delta == 0.)
|
||||||
|
delta = factor;
|
||||||
}
|
}
|
||||||
sum = 0.;
|
|
||||||
for (i = j; i < m; ++i) {
|
/* form (q transpose)*fvec and store the first n components in */
|
||||||
sum += fjac(i,j) * wa4[i];
|
/* qtf. */
|
||||||
/* L100: */
|
|
||||||
|
wa4 = fvec;
|
||||||
|
for (j = 0; j < n; ++j) {
|
||||||
|
if (fjac(j,j) != 0.) {
|
||||||
|
sum = 0.;
|
||||||
|
for (i = j; i < m; ++i)
|
||||||
|
sum += fjac(i,j) * wa4[i];
|
||||||
|
temp = -sum / fjac(j,j);
|
||||||
|
for (i = j; i < m; ++i)
|
||||||
|
wa4[i] += fjac(i,j) * temp;
|
||||||
|
}
|
||||||
|
fjac(j,j) = wa1[j];
|
||||||
|
qtf[j] = wa4[j];
|
||||||
}
|
}
|
||||||
temp = -sum / fjac(j,j);
|
|
||||||
for (i = j; i < m; ++i) {
|
/* compute the norm of the scaled gradient. */
|
||||||
wa4[i] += fjac(i,j) * temp;
|
|
||||||
/* L110: */
|
gnorm = 0.;
|
||||||
}
|
if (fnorm != 0.)
|
||||||
L120:
|
for (j = 0; j < n; ++j) {
|
||||||
fjac(j,j) = wa1[j];
|
l = ipvt[j];
|
||||||
qtf[j] = wa4[j];
|
if (wa2[l] != 0.) {
|
||||||
/* L130: */
|
sum = 0.;
|
||||||
|
for (i = 0; i <= j; ++i)
|
||||||
|
sum += fjac(i,j) * (qtf[i] / fnorm);
|
||||||
|
/* Computing MAX */
|
||||||
|
gnorm = std::max(gnorm, ei_abs(sum / wa2[l]));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* test for convergence of the gradient norm. */
|
||||||
|
|
||||||
|
if (gnorm <= gtol)
|
||||||
|
info = 4;
|
||||||
|
if (info != 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* rescale if necessary. */
|
||||||
|
|
||||||
|
if (mode != 2) /* Computing MAX */
|
||||||
|
diag = diag.cwise().max(wa2);
|
||||||
|
|
||||||
|
/* beginning of the inner loop. */
|
||||||
|
do {
|
||||||
|
|
||||||
|
/* determine the levenberg-marquardt parameter. */
|
||||||
|
|
||||||
|
ei_lmpar<Scalar>(fjac, ipvt, diag, qtf, delta, par, wa1, wa2);
|
||||||
|
|
||||||
|
/* store the direction p and x + p. calculate the norm of p. */
|
||||||
|
|
||||||
|
wa1 = -wa1;
|
||||||
|
wa2 = x + wa1;
|
||||||
|
wa3 = diag.cwise() * wa1;
|
||||||
|
pnorm = wa3.stableNorm();
|
||||||
|
|
||||||
|
/* on the first iteration, adjust the initial step bound. */
|
||||||
|
|
||||||
|
if (iter == 1)
|
||||||
|
delta = std::min(delta,pnorm);
|
||||||
|
|
||||||
|
/* evaluate the function at x + p and calculate its norm. */
|
||||||
|
|
||||||
|
iflag = Functor::f(wa2, wa4);
|
||||||
|
++nfev;
|
||||||
|
if (iflag < 0)
|
||||||
|
goto L300;
|
||||||
|
fnorm1 = wa4.stableNorm();
|
||||||
|
|
||||||
|
/* compute the scaled actual reduction. */
|
||||||
|
|
||||||
|
actred = -1.;
|
||||||
|
if (Scalar(.1) * fnorm1 < fnorm) /* Computing 2nd power */
|
||||||
|
actred = 1. - ei_abs2(fnorm1 / fnorm);
|
||||||
|
|
||||||
|
/* compute the scaled predicted reduction and */
|
||||||
|
/* the scaled directional derivative. */
|
||||||
|
|
||||||
|
wa3.fill(0.);
|
||||||
|
for (j = 0; j < n; ++j) {
|
||||||
|
l = ipvt[j];
|
||||||
|
temp = wa1[l];
|
||||||
|
for (i = 0; i <= j; ++i)
|
||||||
|
wa3[i] += fjac(i,j) * temp;
|
||||||
|
}
|
||||||
|
temp1 = ei_abs2(wa3.stableNorm() / fnorm);
|
||||||
|
temp2 = ei_abs2(ei_sqrt(par) * pnorm / fnorm);
|
||||||
|
/* Computing 2nd power */
|
||||||
|
prered = temp1 + temp2 / Scalar(.5);
|
||||||
|
dirder = -(temp1 + temp2);
|
||||||
|
|
||||||
|
/* compute the ratio of the actual to the predicted */
|
||||||
|
/* reduction. */
|
||||||
|
|
||||||
|
ratio = 0.;
|
||||||
|
if (prered != 0.)
|
||||||
|
ratio = actred / prered;
|
||||||
|
|
||||||
|
/* update the step bound. */
|
||||||
|
|
||||||
|
if (ratio <= Scalar(.25)) {
|
||||||
|
if (actred >= 0.)
|
||||||
|
temp = Scalar(.5);
|
||||||
|
if (actred < 0.)
|
||||||
|
temp = Scalar(.5) * dirder / (dirder + Scalar(.5) * actred);
|
||||||
|
if (Scalar(.1) * fnorm1 >= fnorm || temp < Scalar(.1))
|
||||||
|
temp = Scalar(.1);
|
||||||
|
/* Computing MIN */
|
||||||
|
delta = temp * std::min(delta, pnorm / Scalar(.1));
|
||||||
|
par /= temp;
|
||||||
|
} else if (!(par != 0. && ratio < Scalar(.75))) {
|
||||||
|
delta = pnorm / Scalar(.5);
|
||||||
|
par = Scalar(.5) * par;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* test for successful iteration. */
|
||||||
|
|
||||||
|
if (ratio >= Scalar(1e-4)) {
|
||||||
|
/* successful iteration. update x, fvec, and their norms. */
|
||||||
|
x = wa2;
|
||||||
|
wa2 = diag.cwise() * x;
|
||||||
|
fvec = wa4;
|
||||||
|
xnorm = wa2.stableNorm();
|
||||||
|
fnorm = fnorm1;
|
||||||
|
++iter;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* tests for convergence. */
|
||||||
|
|
||||||
|
if (ei_abs(actred) <= ftol && prered <= ftol && Scalar(.5) * ratio <= 1.)
|
||||||
|
info = 1;
|
||||||
|
if (delta <= xtol * xnorm)
|
||||||
|
info = 2;
|
||||||
|
if (ei_abs(actred) <= ftol && prered <= ftol && Scalar(.5) * ratio <= 1. && info == 2)
|
||||||
|
info = 3;
|
||||||
|
if (info != 0)
|
||||||
|
goto L300;
|
||||||
|
|
||||||
|
/* tests for termination and stringent tolerances. */
|
||||||
|
|
||||||
|
if (nfev >= maxfev)
|
||||||
|
info = 5;
|
||||||
|
if (ei_abs(actred) <= epsilon<Scalar>() && prered <= epsilon<Scalar>() && Scalar(.5) * ratio <= 1.)
|
||||||
|
info = 6;
|
||||||
|
if (delta <= epsilon<Scalar>() * xnorm)
|
||||||
|
info = 7;
|
||||||
|
if (gnorm <= epsilon<Scalar>())
|
||||||
|
info = 8;
|
||||||
|
if (info != 0)
|
||||||
|
goto L300;
|
||||||
|
/* end of the inner loop. repeat if iteration unsuccessful. */
|
||||||
|
} while (ratio < Scalar(1e-4));
|
||||||
|
/* end of the outer loop. */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* compute the norm of the scaled gradient. */
|
|
||||||
|
|
||||||
gnorm = 0.;
|
|
||||||
if (fnorm == 0.) {
|
|
||||||
goto L170;
|
|
||||||
}
|
|
||||||
for (j = 0; j < n; ++j) {
|
|
||||||
l = ipvt[j];
|
|
||||||
if (wa2[l] != 0.) {
|
|
||||||
sum = 0.;
|
|
||||||
for (i = 0; i <= j; ++i)
|
|
||||||
sum += fjac(i,j) * (qtf[i] / fnorm);
|
|
||||||
/* Computing MAX */
|
|
||||||
gnorm = std::max(gnorm, ei_abs(sum / wa2[l]));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
L170:
|
|
||||||
|
|
||||||
/* test for convergence of the gradient norm. */
|
|
||||||
|
|
||||||
if (gnorm <= gtol) {
|
|
||||||
info = 4;
|
|
||||||
}
|
|
||||||
if (info != 0) {
|
|
||||||
goto L300;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* rescale if necessary. */
|
|
||||||
|
|
||||||
if (mode == 2) {
|
|
||||||
goto L190;
|
|
||||||
}
|
|
||||||
/* Computing MAX */
|
|
||||||
diag = diag.cwise().max(wa2);
|
|
||||||
L190:
|
|
||||||
|
|
||||||
/* beginning of the inner loop. */
|
|
||||||
|
|
||||||
L200:
|
|
||||||
|
|
||||||
/* determine the levenberg-marquardt parameter. */
|
|
||||||
|
|
||||||
ei_lmpar<Scalar>(fjac, ipvt, diag, qtf, delta, par, wa1, wa2);
|
|
||||||
|
|
||||||
/* store the direction p and x + p. calculate the norm of p. */
|
|
||||||
|
|
||||||
wa1 = -wa1;
|
|
||||||
wa2 = x + wa1;
|
|
||||||
wa3 = diag.cwise() * wa1;
|
|
||||||
pnorm = wa3.stableNorm();
|
|
||||||
|
|
||||||
/* on the first iteration, adjust the initial step bound. */
|
|
||||||
|
|
||||||
if (iter == 1) {
|
|
||||||
delta = std::min(delta,pnorm);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* evaluate the function at x + p and calculate its norm. */
|
|
||||||
|
|
||||||
iflag = Functor::f(wa2, wa4);
|
|
||||||
++nfev;
|
|
||||||
if (iflag < 0) {
|
|
||||||
goto L300;
|
|
||||||
}
|
|
||||||
fnorm1 = wa4.stableNorm();
|
|
||||||
|
|
||||||
/* compute the scaled actual reduction. */
|
|
||||||
|
|
||||||
actred = -1.;
|
|
||||||
if (Scalar(.1) * fnorm1 < fnorm) /* Computing 2nd power */
|
|
||||||
actred = 1. - ei_abs2(fnorm1 / fnorm);
|
|
||||||
|
|
||||||
/* compute the scaled predicted reduction and */
|
|
||||||
/* the scaled directional derivative. */
|
|
||||||
|
|
||||||
wa3.fill(0.);
|
|
||||||
for (j = 0; j < n; ++j) {
|
|
||||||
l = ipvt[j];
|
|
||||||
temp = wa1[l];
|
|
||||||
for (i = 0; i <= j; ++i) {
|
|
||||||
wa3[i] += fjac(i,j) * temp;
|
|
||||||
/* L220: */
|
|
||||||
}
|
|
||||||
/* L230: */
|
|
||||||
}
|
|
||||||
temp1 = ei_abs2(wa3.stableNorm() / fnorm);
|
|
||||||
temp2 = ei_abs2(ei_sqrt(par) * pnorm / fnorm);
|
|
||||||
/* Computing 2nd power */
|
|
||||||
prered = temp1 + temp2 / Scalar(.5);
|
|
||||||
dirder = -(temp1 + temp2);
|
|
||||||
|
|
||||||
/* compute the ratio of the actual to the predicted */
|
|
||||||
/* reduction. */
|
|
||||||
|
|
||||||
ratio = 0.;
|
|
||||||
if (prered != 0.) {
|
|
||||||
ratio = actred / prered;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* update the step bound. */
|
|
||||||
|
|
||||||
if (ratio > Scalar(.25)) {
|
|
||||||
goto L240;
|
|
||||||
}
|
|
||||||
if (actred >= 0.) {
|
|
||||||
temp = Scalar(.5);
|
|
||||||
}
|
|
||||||
if (actred < 0.) {
|
|
||||||
temp = Scalar(.5) * dirder / (dirder + Scalar(.5) * actred);
|
|
||||||
}
|
|
||||||
if (Scalar(.1) * fnorm1 >= fnorm || temp < Scalar(.1))
|
|
||||||
temp = Scalar(.1);
|
|
||||||
/* Computing MIN */
|
|
||||||
delta = temp * std::min(delta, pnorm / Scalar(.1));
|
|
||||||
par /= temp;
|
|
||||||
goto L260;
|
|
||||||
L240:
|
|
||||||
if (par != 0. && ratio < Scalar(.75)) {
|
|
||||||
goto L250;
|
|
||||||
}
|
|
||||||
delta = pnorm / Scalar(.5);
|
|
||||||
par = Scalar(.5) * par;
|
|
||||||
L250:
|
|
||||||
L260:
|
|
||||||
|
|
||||||
/* test for successful iteration. */
|
|
||||||
|
|
||||||
if (ratio < Scalar(1e-4)) {
|
|
||||||
goto L290;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* successful iteration. update x, fvec, and their norms. */
|
|
||||||
|
|
||||||
x = wa2;
|
|
||||||
wa2 = diag.cwise() * x;
|
|
||||||
fvec = wa4;
|
|
||||||
xnorm = wa2.stableNorm();
|
|
||||||
fnorm = fnorm1;
|
|
||||||
++iter;
|
|
||||||
L290:
|
|
||||||
|
|
||||||
/* tests for convergence. */
|
|
||||||
|
|
||||||
if (ei_abs(actred) <= ftol && prered <= ftol && Scalar(.5) * ratio <= 1.) {
|
|
||||||
info = 1;
|
|
||||||
}
|
|
||||||
if (delta <= xtol * xnorm) {
|
|
||||||
info = 2;
|
|
||||||
}
|
|
||||||
if (ei_abs(actred) <= ftol && prered <= ftol && Scalar(.5) * ratio <= 1. && info
|
|
||||||
== 2) {
|
|
||||||
info = 3;
|
|
||||||
}
|
|
||||||
if (info != 0) {
|
|
||||||
goto L300;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* tests for termination and stringent tolerances. */
|
|
||||||
|
|
||||||
if (nfev >= maxfev) {
|
|
||||||
info = 5;
|
|
||||||
}
|
|
||||||
if (ei_abs(actred) <= epsilon<Scalar>() && prered <= epsilon<Scalar>() && Scalar(.5) * ratio <= 1.) {
|
|
||||||
info = 6;
|
|
||||||
}
|
|
||||||
if (delta <= epsilon<Scalar>() * xnorm) {
|
|
||||||
info = 7;
|
|
||||||
}
|
|
||||||
if (gnorm <= epsilon<Scalar>()) {
|
|
||||||
info = 8;
|
|
||||||
}
|
|
||||||
if (info != 0) {
|
|
||||||
goto L300;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* end of the inner loop. repeat if iteration unsuccessful. */
|
|
||||||
|
|
||||||
if (ratio < Scalar(1e-4)) {
|
|
||||||
goto L200;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* end of the outer loop. */
|
|
||||||
|
|
||||||
goto L30;
|
|
||||||
L300:
|
L300:
|
||||||
|
|
||||||
/* termination, either normal or user imposed. */
|
/* termination, either normal or user imposed. */
|
||||||
|
if (iflag < 0)
|
||||||
if (iflag < 0) {
|
|
||||||
info = iflag;
|
info = iflag;
|
||||||
}
|
if (nprint > 0)
|
||||||
iflag = 0;
|
|
||||||
if (nprint > 0) {
|
|
||||||
iflag = Functor::debug(x, fvec);
|
iflag = Functor::debug(x, fvec);
|
||||||
}
|
|
||||||
return info;
|
return info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -46,22 +46,19 @@ int ei_lmstr(
|
|||||||
|
|
||||||
/* check the input parameters for errors. */
|
/* check the input parameters for errors. */
|
||||||
|
|
||||||
if (n <= 0 || m < n || ftol < 0. || xtol < 0. ||
|
if (n <= 0 || m < n || ftol < 0. || xtol < 0. || gtol < 0. || maxfev <= 0 || factor <= 0.)
|
||||||
gtol < 0. || maxfev <= 0 || factor <= 0.) {
|
|
||||||
goto L340;
|
goto L340;
|
||||||
}
|
|
||||||
if (mode == 2)
|
if (mode == 2)
|
||||||
for (j = 0; j < n; ++j)
|
for (j = 0; j < n; ++j)
|
||||||
if (diag[j] <= 0.) goto L300;
|
if (diag[j] <= 0.) goto L340;
|
||||||
|
|
||||||
/* evaluate the function at the starting point */
|
/* evaluate the function at the starting point */
|
||||||
/* and calculate its norm. */
|
/* and calculate its norm. */
|
||||||
|
|
||||||
iflag = Functor::f(x, fvec);
|
iflag = Functor::f(x, fvec);
|
||||||
nfev = 1;
|
nfev = 1;
|
||||||
if (iflag < 0) {
|
if (iflag < 0)
|
||||||
goto L340;
|
goto L340;
|
||||||
}
|
|
||||||
fnorm = fvec.stableNorm();
|
fnorm = fvec.stableNorm();
|
||||||
|
|
||||||
/* initialize levenberg-marquardt parameter and iteration counter. */
|
/* initialize levenberg-marquardt parameter and iteration counter. */
|
||||||
@ -71,297 +68,230 @@ int ei_lmstr(
|
|||||||
|
|
||||||
/* beginning of the outer loop. */
|
/* beginning of the outer loop. */
|
||||||
|
|
||||||
L30:
|
while (true) {
|
||||||
|
|
||||||
/* if requested, call Functor::f to enable printing of iterates. */
|
/* if requested, call Functor::f to enable printing of iterates. */
|
||||||
|
|
||||||
if (nprint <= 0) {
|
if (nprint > 0) {
|
||||||
goto L40;
|
iflag = 0;
|
||||||
}
|
if ((iter - 1) % nprint == 0)
|
||||||
iflag = 0;
|
iflag = Functor::debug(x, fvec, wa3);
|
||||||
if ((iter - 1) % nprint == 0) {
|
if (iflag < 0)
|
||||||
iflag = Functor::debug(x, fvec, wa3);
|
break;
|
||||||
}
|
|
||||||
if (iflag < 0) {
|
|
||||||
goto L340;
|
|
||||||
}
|
|
||||||
L40:
|
|
||||||
|
|
||||||
/* compute the qr factorization of the jacobian matrix */
|
|
||||||
/* calculated one row at a time, while simultaneously */
|
|
||||||
/* forming (q transpose)*fvec and storing the first */
|
|
||||||
/* n components in qtf. */
|
|
||||||
|
|
||||||
qtf.fill(0.);
|
|
||||||
fjac.fill(0.);
|
|
||||||
iflag = 2;
|
|
||||||
for (i = 0; i < m; ++i) {
|
|
||||||
if (Functor::df(x, wa3, iflag) < 0)
|
|
||||||
goto L340;
|
|
||||||
temp = fvec[i];
|
|
||||||
ei_rwupdt<Scalar>(n, fjac.data(), fjac.rows(), wa3.data(), qtf.data(), &temp, wa1.data(), wa2.data());
|
|
||||||
++iflag;
|
|
||||||
}
|
|
||||||
++njev;
|
|
||||||
|
|
||||||
/* if the jacobian is rank deficient, call qrfac to */
|
|
||||||
/* reorder its columns and update the components of qtf. */
|
|
||||||
|
|
||||||
sing = false;
|
|
||||||
for (j = 0; j < n; ++j) {
|
|
||||||
if (fjac(j,j) == 0.) {
|
|
||||||
sing = true;
|
|
||||||
}
|
}
|
||||||
ipvt[j] = j;
|
|
||||||
wa2[j] = fjac.col(j).start(j).stableNorm();
|
/* compute the qr factorization of the jacobian matrix */
|
||||||
}
|
/* calculated one row at a time, while simultaneously */
|
||||||
if (! sing)
|
/* forming (q transpose)*fvec and storing the first */
|
||||||
goto L130;
|
/* n components in qtf. */
|
||||||
ipvt.cwise()+=1;
|
|
||||||
ei_qrfac<Scalar>(n, n, fjac.data(), fjac.rows(), true, ipvt.data(), n, wa1.data(), wa2.data());
|
qtf.fill(0.);
|
||||||
ipvt.cwise()-=1; // qrfac() creates ipvt with fortran convetion (1->n), convert it to c (0->n-1)
|
fjac.fill(0.);
|
||||||
for (j = 0; j < n; ++j) {
|
iflag = 2;
|
||||||
if (fjac(j,j) == 0.)
|
for (i = 0; i < m; ++i) {
|
||||||
goto L110;
|
if (Functor::df(x, wa3, iflag) < 0)
|
||||||
sum = 0.;
|
break;
|
||||||
for (i = j; i < n; ++i) {
|
temp = fvec[i];
|
||||||
sum += fjac(i,j) * qtf[i];
|
ei_rwupdt<Scalar>(n, fjac.data(), fjac.rows(), wa3.data(), qtf.data(), &temp, wa1.data(), wa2.data());
|
||||||
|
++iflag;
|
||||||
}
|
}
|
||||||
temp = -sum / fjac(j,j);
|
++njev;
|
||||||
for (i = j; i < n; ++i) {
|
|
||||||
qtf[i] += fjac(i,j) * temp;
|
/* if the jacobian is rank deficient, call qrfac to */
|
||||||
|
/* reorder its columns and update the components of qtf. */
|
||||||
|
|
||||||
|
sing = false;
|
||||||
|
for (j = 0; j < n; ++j) {
|
||||||
|
if (fjac(j,j) == 0.) {
|
||||||
|
sing = true;
|
||||||
|
}
|
||||||
|
ipvt[j] = j;
|
||||||
|
wa2[j] = fjac.col(j).start(j).stableNorm();
|
||||||
}
|
}
|
||||||
L110:
|
if (sing) {
|
||||||
fjac(j,j) = wa1[j];
|
ipvt.cwise()+=1;
|
||||||
/* L120: */
|
ei_qrfac<Scalar>(n, n, fjac.data(), fjac.rows(), true, ipvt.data(), n, wa1.data(), wa2.data());
|
||||||
}
|
ipvt.cwise()-=1; // qrfac() creates ipvt with fortran convetion (1->n), convert it to c (0->n-1)
|
||||||
L130:
|
for (j = 0; j < n; ++j) {
|
||||||
|
if (fjac(j,j) != 0.) {
|
||||||
/* on the first iteration and if mode is 1, scale according */
|
sum = 0.;
|
||||||
/* to the norms of the columns of the initial jacobian. */
|
for (i = j; i < n; ++i)
|
||||||
|
sum += fjac(i,j) * qtf[i];
|
||||||
if (iter != 1) {
|
temp = -sum / fjac(j,j);
|
||||||
goto L170;
|
for (i = j; i < n; ++i)
|
||||||
}
|
qtf[i] += fjac(i,j) * temp;
|
||||||
if (mode == 2) {
|
}
|
||||||
goto L150;
|
fjac(j,j) = wa1[j];
|
||||||
}
|
}
|
||||||
for (j = 0; j < n; ++j) {
|
|
||||||
diag[j] = wa2[j];
|
|
||||||
if (wa2[j] == 0.) {
|
|
||||||
diag[j] = 1.;
|
|
||||||
}
|
}
|
||||||
/* L140: */
|
|
||||||
}
|
|
||||||
L150:
|
|
||||||
|
|
||||||
/* on the first iteration, calculate the norm of the scaled x */
|
/* on the first iteration and if mode is 1, scale according */
|
||||||
/* and initialize the step bound delta. */
|
/* to the norms of the columns of the initial jacobian. */
|
||||||
|
|
||||||
wa3 = diag.cwise() * x;
|
if (iter == 1) {
|
||||||
xnorm = wa3.stableNorm();
|
if (mode != 2)
|
||||||
delta = factor * xnorm;
|
for (j = 0; j < n; ++j) {
|
||||||
if (delta == 0.) {
|
diag[j] = wa2[j];
|
||||||
delta = factor;
|
if (wa2[j] == 0.)
|
||||||
}
|
diag[j] = 1.;
|
||||||
L170:
|
}
|
||||||
|
|
||||||
/* compute the norm of the scaled gradient. */
|
/* on the first iteration, calculate the norm of the scaled x */
|
||||||
|
/* and initialize the step bound delta. */
|
||||||
|
|
||||||
gnorm = 0.;
|
wa3 = diag.cwise() * x;
|
||||||
if (fnorm == 0.) {
|
xnorm = wa3.stableNorm();
|
||||||
goto L210;
|
delta = factor * xnorm;
|
||||||
}
|
if (delta == 0.)
|
||||||
for (j = 0; j < n; ++j) {
|
delta = factor;
|
||||||
l = ipvt[j];
|
|
||||||
if (wa2[l] != 0.) {
|
|
||||||
sum = 0.;
|
|
||||||
for (i = 0; i <= j; ++i)
|
|
||||||
sum += fjac(i,j) * (qtf[i] / fnorm);
|
|
||||||
/* Computing MAX */
|
|
||||||
gnorm = std::max(gnorm, ei_abs(sum / wa2[l]));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* compute the norm of the scaled gradient. */
|
||||||
|
|
||||||
|
gnorm = 0.;
|
||||||
|
if (fnorm != 0.)
|
||||||
|
for (j = 0; j < n; ++j) {
|
||||||
|
l = ipvt[j];
|
||||||
|
if (wa2[l] != 0.) {
|
||||||
|
sum = 0.;
|
||||||
|
for (i = 0; i <= j; ++i)
|
||||||
|
sum += fjac(i,j) * (qtf[i] / fnorm);
|
||||||
|
/* Computing MAX */
|
||||||
|
gnorm = std::max(gnorm, ei_abs(sum / wa2[l]));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* test for convergence of the gradient norm. */
|
||||||
|
|
||||||
|
if (gnorm <= gtol)
|
||||||
|
info = 4;
|
||||||
|
if (info != 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* rescale if necessary. */
|
||||||
|
|
||||||
|
if (mode != 2) /* Computing MAX */
|
||||||
|
diag = diag.cwise().max(wa2);
|
||||||
|
|
||||||
|
/* beginning of the inner loop. */
|
||||||
|
do {
|
||||||
|
|
||||||
|
/* determine the levenberg-marquardt parameter. */
|
||||||
|
|
||||||
|
ei_lmpar<Scalar>(fjac, ipvt, diag, qtf, delta, par, wa1, wa2);
|
||||||
|
|
||||||
|
/* store the direction p and x + p. calculate the norm of p. */
|
||||||
|
|
||||||
|
wa1 = -wa1;
|
||||||
|
wa2 = x + wa1;
|
||||||
|
wa3 = diag.cwise() * wa1;
|
||||||
|
pnorm = wa3.stableNorm();
|
||||||
|
|
||||||
|
/* on the first iteration, adjust the initial step bound. */
|
||||||
|
|
||||||
|
if (iter == 1)
|
||||||
|
delta = std::min(delta,pnorm);
|
||||||
|
|
||||||
|
/* evaluate the function at x + p and calculate its norm. */
|
||||||
|
|
||||||
|
iflag = Functor::f(wa2, wa4);
|
||||||
|
++nfev;
|
||||||
|
if (iflag < 0)
|
||||||
|
goto L340;
|
||||||
|
fnorm1 = wa4.stableNorm();
|
||||||
|
|
||||||
|
/* compute the scaled actual reduction. */
|
||||||
|
|
||||||
|
actred = -1.;
|
||||||
|
if (Scalar(.1) * fnorm1 < fnorm) /* Computing 2nd power */
|
||||||
|
actred = 1. - ei_abs2(fnorm1 / fnorm);
|
||||||
|
|
||||||
|
/* compute the scaled predicted reduction and */
|
||||||
|
/* the scaled directional derivative. */
|
||||||
|
|
||||||
|
wa3.fill(0.);
|
||||||
|
for (j = 0; j < n; ++j) {
|
||||||
|
l = ipvt[j];
|
||||||
|
temp = wa1[l];
|
||||||
|
for (i = 0; i <= j; ++i)
|
||||||
|
wa3[i] += fjac(i,j) * temp;
|
||||||
|
}
|
||||||
|
temp1 = ei_abs2(wa3.stableNorm() / fnorm);
|
||||||
|
temp2 = ei_abs2(ei_sqrt(par) * pnorm / fnorm);
|
||||||
|
/* Computing 2nd power */
|
||||||
|
prered = temp1 + temp2 / Scalar(.5);
|
||||||
|
dirder = -(temp1 + temp2);
|
||||||
|
|
||||||
|
/* compute the ratio of the actual to the predicted */
|
||||||
|
/* reduction. */
|
||||||
|
|
||||||
|
ratio = 0.;
|
||||||
|
if (prered != 0.)
|
||||||
|
ratio = actred / prered;
|
||||||
|
|
||||||
|
/* update the step bound. */
|
||||||
|
|
||||||
|
if (ratio <= Scalar(.25)) {
|
||||||
|
if (actred >= 0.)
|
||||||
|
temp = Scalar(.5);
|
||||||
|
if (actred < 0.)
|
||||||
|
temp = Scalar(.5) * dirder / (dirder + Scalar(.5) * actred);
|
||||||
|
if (Scalar(.1) * fnorm1 >= fnorm || temp < Scalar(.1))
|
||||||
|
temp = Scalar(.1);
|
||||||
|
/* Computing MIN */
|
||||||
|
delta = temp * std::min(delta, pnorm / Scalar(.1));
|
||||||
|
par /= temp;
|
||||||
|
} else if (!(par != 0. && ratio < Scalar(.75))) {
|
||||||
|
delta = pnorm / Scalar(.5);
|
||||||
|
par = Scalar(.5) * par;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* test for successful iteration. */
|
||||||
|
|
||||||
|
if (ratio >= Scalar(1e-4)) {
|
||||||
|
/* successful iteration. update x, fvec, and their norms. */
|
||||||
|
x = wa2;
|
||||||
|
wa2 = diag.cwise() * x;
|
||||||
|
fvec = wa4;
|
||||||
|
xnorm = wa2.stableNorm();
|
||||||
|
fnorm = fnorm1;
|
||||||
|
++iter;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* tests for convergence. */
|
||||||
|
|
||||||
|
if (ei_abs(actred) <= ftol && prered <= ftol && Scalar(.5) * ratio <= 1.)
|
||||||
|
info = 1;
|
||||||
|
if (delta <= xtol * xnorm)
|
||||||
|
info = 2;
|
||||||
|
if (ei_abs(actred) <= ftol && prered <= ftol && Scalar(.5) * ratio <= 1. && info == 2)
|
||||||
|
info = 3;
|
||||||
|
if (info != 0)
|
||||||
|
goto L340;
|
||||||
|
|
||||||
|
/* tests for termination and stringent tolerances. */
|
||||||
|
|
||||||
|
if (nfev >= maxfev)
|
||||||
|
info = 5;
|
||||||
|
if (ei_abs(actred) <= epsilon<Scalar>() && prered <= epsilon<Scalar>() && Scalar(.5) * ratio <= 1.)
|
||||||
|
info = 6;
|
||||||
|
if (delta <= epsilon<Scalar>() * xnorm)
|
||||||
|
info = 7;
|
||||||
|
if (gnorm <= epsilon<Scalar>())
|
||||||
|
info = 8;
|
||||||
|
if (info != 0)
|
||||||
|
goto L340;
|
||||||
|
/* end of the inner loop. repeat if iteration unsuccessful. */
|
||||||
|
} while (ratio < Scalar(1e-4));
|
||||||
|
/* end of the outer loop. */
|
||||||
}
|
}
|
||||||
L210:
|
|
||||||
|
|
||||||
/* test for convergence of the gradient norm. */
|
|
||||||
|
|
||||||
if (gnorm <= gtol) {
|
|
||||||
info = 4;
|
|
||||||
}
|
|
||||||
if (info != 0) {
|
|
||||||
goto L340;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* rescale if necessary. */
|
|
||||||
|
|
||||||
if (mode == 2) {
|
|
||||||
goto L230;
|
|
||||||
}
|
|
||||||
/* Computing MAX */
|
|
||||||
diag = diag.cwise().max(wa2);
|
|
||||||
L230:
|
|
||||||
|
|
||||||
/* beginning of the inner loop. */
|
|
||||||
|
|
||||||
L240:
|
|
||||||
|
|
||||||
/* determine the levenberg-marquardt parameter. */
|
|
||||||
|
|
||||||
ei_lmpar<Scalar>(fjac, ipvt, diag, qtf, delta, par, wa1, wa2);
|
|
||||||
|
|
||||||
/* store the direction p and x + p. calculate the norm of p. */
|
|
||||||
|
|
||||||
wa1 = -wa1;
|
|
||||||
wa2 = x + wa1;
|
|
||||||
wa3 = diag.cwise() * wa1;
|
|
||||||
pnorm = wa3.stableNorm();
|
|
||||||
|
|
||||||
/* on the first iteration, adjust the initial step bound. */
|
|
||||||
|
|
||||||
if (iter == 1) {
|
|
||||||
delta = std::min(delta,pnorm);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* evaluate the function at x + p and calculate its norm. */
|
|
||||||
|
|
||||||
iflag = Functor::f(wa2, wa4);
|
|
||||||
++nfev;
|
|
||||||
if (iflag < 0) {
|
|
||||||
goto L340;
|
|
||||||
}
|
|
||||||
fnorm1 = wa4.stableNorm();
|
|
||||||
|
|
||||||
/* compute the scaled actual reduction. */
|
|
||||||
|
|
||||||
actred = -1.;
|
|
||||||
if (Scalar(.1) * fnorm1 < fnorm) /* Computing 2nd power */
|
|
||||||
actred = 1. - ei_abs2(fnorm1 / fnorm);
|
|
||||||
|
|
||||||
/* compute the scaled predicted reduction and */
|
|
||||||
/* the scaled directional derivative. */
|
|
||||||
|
|
||||||
wa3.fill(0.);
|
|
||||||
for (j = 0; j < n; ++j) {
|
|
||||||
l = ipvt[j];
|
|
||||||
temp = wa1[l];
|
|
||||||
for (i = 0; i <= j; ++i) {
|
|
||||||
wa3[i] += fjac(i,j) * temp;
|
|
||||||
/* L260: */
|
|
||||||
}
|
|
||||||
/* L270: */
|
|
||||||
}
|
|
||||||
temp1 = ei_abs2(wa3.stableNorm() / fnorm);
|
|
||||||
temp2 = ei_abs2(ei_sqrt(par) * pnorm / fnorm);
|
|
||||||
/* Computing 2nd power */
|
|
||||||
prered = temp1 + temp2 / Scalar(.5);
|
|
||||||
dirder = -(temp1 + temp2);
|
|
||||||
|
|
||||||
/* compute the ratio of the actual to the predicted */
|
|
||||||
/* reduction. */
|
|
||||||
|
|
||||||
ratio = 0.;
|
|
||||||
if (prered != 0.) {
|
|
||||||
ratio = actred / prered;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* update the step bound. */
|
|
||||||
|
|
||||||
if (ratio > Scalar(.25)) {
|
|
||||||
goto L280;
|
|
||||||
}
|
|
||||||
if (actred >= 0.) {
|
|
||||||
temp = Scalar(.5);
|
|
||||||
}
|
|
||||||
if (actred < 0.) {
|
|
||||||
temp = Scalar(.5) * dirder / (dirder + Scalar(.5) * actred);
|
|
||||||
}
|
|
||||||
if (Scalar(.1) * fnorm1 >= fnorm || temp < Scalar(.1))
|
|
||||||
temp = Scalar(.1);
|
|
||||||
/* Computing MIN */
|
|
||||||
delta = temp * std::min(delta, pnorm / Scalar(.1));
|
|
||||||
|
|
||||||
par /= temp;
|
|
||||||
goto L300;
|
|
||||||
L280:
|
|
||||||
if (par != 0. && ratio < Scalar(.75)) {
|
|
||||||
goto L290;
|
|
||||||
}
|
|
||||||
delta = pnorm / Scalar(.5);
|
|
||||||
par = Scalar(.5) * par;
|
|
||||||
L290:
|
|
||||||
L300:
|
|
||||||
|
|
||||||
/* test for successful iteration. */
|
|
||||||
|
|
||||||
if (ratio < Scalar(1e-4)) {
|
|
||||||
goto L330;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* successful iteration. update x, fvec, and their norms. */
|
|
||||||
|
|
||||||
x = wa2;
|
|
||||||
wa2 = diag.cwise() * x;
|
|
||||||
fvec = wa4;
|
|
||||||
xnorm = wa2.stableNorm();
|
|
||||||
fnorm = fnorm1;
|
|
||||||
++iter;
|
|
||||||
L330:
|
|
||||||
|
|
||||||
/* tests for convergence. */
|
|
||||||
|
|
||||||
if (ei_abs(actred) <= ftol && prered <= ftol && Scalar(.5) * ratio <= 1.) {
|
|
||||||
info = 1;
|
|
||||||
}
|
|
||||||
if (delta <= xtol * xnorm) {
|
|
||||||
info = 2;
|
|
||||||
}
|
|
||||||
if (ei_abs(actred) <= ftol && prered <= ftol && Scalar(.5) * ratio <= 1. && info
|
|
||||||
== 2) {
|
|
||||||
info = 3;
|
|
||||||
}
|
|
||||||
if (info != 0) {
|
|
||||||
goto L340;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* tests for termination and stringent tolerances. */
|
|
||||||
|
|
||||||
if (nfev >= maxfev) {
|
|
||||||
info = 5;
|
|
||||||
}
|
|
||||||
if (ei_abs(actred) <= epsilon<Scalar>() && prered <= epsilon<Scalar>() && Scalar(.5) * ratio <= 1.) {
|
|
||||||
info = 6;
|
|
||||||
}
|
|
||||||
if (delta <= epsilon<Scalar>() * xnorm) {
|
|
||||||
info = 7;
|
|
||||||
}
|
|
||||||
if (gnorm <= epsilon<Scalar>()) {
|
|
||||||
info = 8;
|
|
||||||
}
|
|
||||||
if (info != 0) {
|
|
||||||
goto L340;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* end of the inner loop. repeat if iteration unsuccessful. */
|
|
||||||
|
|
||||||
if (ratio < Scalar(1e-4)) {
|
|
||||||
goto L240;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* end of the outer loop. */
|
|
||||||
|
|
||||||
goto L30;
|
|
||||||
L340:
|
L340:
|
||||||
|
|
||||||
/* termination, either normal or user imposed. */
|
/* termination, either normal or user imposed. */
|
||||||
|
if (iflag < 0)
|
||||||
if (iflag < 0) {
|
|
||||||
info = iflag;
|
info = iflag;
|
||||||
}
|
if (nprint > 0)
|
||||||
iflag = 0;
|
|
||||||
if (nprint > 0) {
|
|
||||||
iflag = Functor::debug(x, fvec, wa3);
|
iflag = Functor::debug(x, fvec, wa3);
|
||||||
}
|
|
||||||
return info;
|
return info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user