eigen/unsupported/Eigen/src/NonLinearOptimization/chkder.h

63 lines
1.8 KiB
C
Raw Normal View History

#define chkder_log10e 0.43429448190325182765
#define chkder_factor 100.
namespace Eigen {
namespace internal {
2009-08-21 05:26:40 +08:00
template<typename Scalar>
void chkder(
2010-01-28 11:19:39 +08:00
const Matrix< Scalar, Dynamic, 1 > &x,
const Matrix< Scalar, Dynamic, 1 > &fvec,
const Matrix< Scalar, Dynamic, Dynamic > &fjac,
2009-08-21 05:26:40 +08:00
Matrix< Scalar, Dynamic, 1 > &xp,
2010-01-28 11:19:39 +08:00
const Matrix< Scalar, Dynamic, 1 > &fvecp,
2009-08-21 05:26:40 +08:00
int mode,
Matrix< Scalar, Dynamic, 1 > &err
)
{
typedef DenseIndex Index;
const Scalar eps = sqrt(NumTraits<Scalar>::epsilon());
const Scalar epsf = chkder_factor * NumTraits<Scalar>::epsilon();
const Scalar epslog = chkder_log10e * log(eps);
Scalar temp;
const Index m = fvec.size(), n = x.size();
if (mode != 2) {
2010-01-26 20:20:24 +08:00
/* mode = 1. */
2009-08-29 08:46:19 +08:00
xp.resize(n);
for (Index j = 0; j < n; ++j) {
temp = eps * abs(x[j]);
2009-08-21 05:26:40 +08:00
if (temp == 0.)
temp = eps;
xp[j] = x[j] + temp;
}
}
2009-08-21 05:26:40 +08:00
else {
2010-01-26 20:20:24 +08:00
/* mode = 2. */
2009-08-23 12:16:05 +08:00
err.setZero(m);
for (Index j = 0; j < n; ++j) {
temp = abs(x[j]);
2009-08-21 05:26:40 +08:00
if (temp == 0.)
temp = 1.;
err += temp * fjac.col(j);
}
for (Index i = 0; i < m; ++i) {
2009-08-21 05:26:40 +08:00
temp = 1.;
if (fvec[i] != 0. && fvecp[i] != 0. && abs(fvecp[i] - fvec[i]) >= epsf * abs(fvec[i]))
temp = eps * abs((fvecp[i] - fvec[i]) / eps - err[i]) / (abs(fvec[i]) + abs(fvecp[i]));
2009-08-21 05:26:40 +08:00
err[i] = 1.;
if (temp > NumTraits<Scalar>::epsilon() && temp < eps)
err[i] = (chkder_log10e * log(temp) - epslog) / epslog;
2009-08-23 12:16:05 +08:00
if (temp >= eps)
2009-08-21 05:26:40 +08:00
err[i] = 0.;
}
}
2010-01-26 20:20:24 +08:00
}
2011-02-04 20:55:12 +08:00
} // end namespace internal
} // end namespace Eigen