TrustRegionMinimier tests for Dogleg strategy.
Also a fix for a minor segfault in trust_region_minimizer.cc
which was discovered while writing this test.
Change-Id: I50353d0292fd37495bf73de3824c430912ef221d
diff --git a/internal/ceres/trust_region_minimizer.cc b/internal/ceres/trust_region_minimizer.cc
index dd4a8a7..4d0c91e 100644
--- a/internal/ceres/trust_region_minimizer.cc
+++ b/internal/ceres/trust_region_minimizer.cc
@@ -196,15 +196,18 @@
scale.setOnes();
}
- const double absolute_gradient_tolerance =
- options_.gradient_tolerance *
+ // The initial gradient max_norm is bounded from below so that we do
+ // not divide by zero.
+ const double gradient_max_norm_0 =
max(iteration_summary.gradient_max_norm, kEpsilon);
+ const double absolute_gradient_tolerance =
+ options_.gradient_tolerance * gradient_max_norm_0;
+
if (iteration_summary.gradient_max_norm <= absolute_gradient_tolerance) {
summary->termination_type = GRADIENT_TOLERANCE;
VLOG(1) << "Terminating: Gradient tolerance reached."
<< "Relative gradient max norm: "
- << iteration_summary.gradient_max_norm /
- summary->iterations[0].gradient_max_norm
+ << iteration_summary.gradient_max_norm / gradient_max_norm_0
<< " <= " << options_.gradient_tolerance;
return;
}
@@ -406,8 +409,7 @@
summary->termination_type = GRADIENT_TOLERANCE;
VLOG(1) << "Terminating: Gradient tolerance reached."
<< "Relative gradient max norm: "
- << iteration_summary.gradient_max_norm /
- summary->iterations[0].gradient_max_norm
+ << iteration_summary.gradient_max_norm / gradient_max_norm_0
<< " <= " << options_.gradient_tolerance;
return;
}
diff --git a/internal/ceres/trust_region_minimizer_test.cc b/internal/ceres/trust_region_minimizer_test.cc
index b4ef601..d87c963 100644
--- a/internal/ceres/trust_region_minimizer_test.cc
+++ b/internal/ceres/trust_region_minimizer_test.cc
@@ -38,10 +38,10 @@
#include "ceres/dense_sparse_matrix.h"
#include "ceres/evaluator.h"
#include "ceres/internal/port.h"
-#include "ceres/levenberg_marquardt_strategy.h"
#include "ceres/linear_solver.h"
#include "ceres/minimizer.h"
#include "ceres/trust_region_minimizer.h"
+#include "ceres/trust_region_strategy.h"
#include "gtest/gtest.h"
namespace ceres {
@@ -52,7 +52,7 @@
// active. This is equivalent to constructing a problem and using the
// SubsetLocalParameterization. This allows us to test the support for
// the Evaluator::Plus operation besides checking for the basic
-// performance of the LevenbergMarquardt algorithm.
+// performance of the trust region algorithm.
template <bool col1, bool col2, bool col3, bool col4>
class PowellEvaluator2 : public Evaluator {
public:
@@ -183,7 +183,7 @@
// Templated function to hold a subset of the columns fixed and check
// if the solver converges to the optimal values or not.
template<bool col1, bool col2, bool col3, bool col4>
-void IsLevenbergMarquardtSolveSuccessful() {
+void IsTrustRegionSolveSuccessful(TrustRegionStrategyType strategy_type) {
Solver::Options solver_options;
LinearSolver::Options linear_solver_options;
DenseQRSolver linear_solver(linear_solver_options);
@@ -208,6 +208,7 @@
minimizer_options.jacobian = jacobian.get();
TrustRegionStrategy::Options trust_region_strategy_options;
+ trust_region_strategy_options.trust_region_strategy_type = strategy_type;
trust_region_strategy_options.linear_solver = &linear_solver;
trust_region_strategy_options.initial_radius = 1e4;
trust_region_strategy_options.max_radius = 1e20;
@@ -228,7 +229,7 @@
EXPECT_NEAR(0.0, parameters[3], 0.001);
};
-TEST(TrustRegionMinimizer, PowellsSingularFunction) {
+TEST(TrustRegionMinimizer, PowellsSingularFunctionUsingLevenbergMarquardt) {
// This case is excluded because this has a local minimum and does
// not find the optimum. This should not affect the correctness of
// this test since we are testing all the other 14 combinations of
@@ -236,20 +237,43 @@
//
// IsSolveSuccessful<true, true, false, true>();
- IsLevenbergMarquardtSolveSuccessful<true, true, true, true>();
- IsLevenbergMarquardtSolveSuccessful<true, true, true, false>();
- IsLevenbergMarquardtSolveSuccessful<true, false, true, true>();
- IsLevenbergMarquardtSolveSuccessful<false, true, true, true>();
- IsLevenbergMarquardtSolveSuccessful<true, true, false, false>();
- IsLevenbergMarquardtSolveSuccessful<true, false, true, false>();
- IsLevenbergMarquardtSolveSuccessful<false, true, true, false>();
- IsLevenbergMarquardtSolveSuccessful<true, false, false, true>();
- IsLevenbergMarquardtSolveSuccessful<false, true, false, true>();
- IsLevenbergMarquardtSolveSuccessful<false, false, true, true>();
- IsLevenbergMarquardtSolveSuccessful<true, false, false, false>();
- IsLevenbergMarquardtSolveSuccessful<false, true, false, false>();
- IsLevenbergMarquardtSolveSuccessful<false, false, true, false>();
- IsLevenbergMarquardtSolveSuccessful<false, false, false, true>();
+ const TrustRegionStrategyType kStrategy = LEVENBERG_MARQUARDT;
+ IsTrustRegionSolveSuccessful<true, true, true, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<true, true, true, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<true, false, true, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<false, true, true, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<true, true, false, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<true, false, true, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<false, true, true, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<true, false, false, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<false, true, false, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<false, false, true, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<true, false, false, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<false, true, false, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<false, false, true, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<false, false, false, true >(kStrategy);
+}
+
+TEST(TrustRegionMinimizer, PowellsSingularFunctionUsingDogleg) {
+ // The following two cases are excluded because they encounter a local minimum.
+ //
+ // IsTrustRegionSolveSuccessful<true, true, false, true >(kStrategy);
+ // IsTrustRegionSolveSuccessful<true, true, true, true >(kStrategy);
+
+ const TrustRegionStrategyType kStrategy = DOGLEG;
+ IsTrustRegionSolveSuccessful<true, true, true, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<true, false, true, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<false, true, true, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<true, true, false, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<true, false, true, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<false, true, true, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<true, false, false, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<false, true, false, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<false, false, true, true >(kStrategy);
+ IsTrustRegionSolveSuccessful<true, false, false, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<false, true, false, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<false, false, true, false>(kStrategy);
+ IsTrustRegionSolveSuccessful<false, false, false, true >(kStrategy);
}
} // namespace internal