testDoglegOptimizer.cpp
Go to the documentation of this file.
1 /* ----------------------------------------------------------------------------
2 
3  * GTSAM Copyright 2010, Georgia Tech Research Corporation,
4  * Atlanta, Georgia 30332-0415
5  * All Rights Reserved
6  * Authors: Frank Dellaert, et al. (see THANKS for the full author list)
7 
8  * See LICENSE for the license information
9 
10  * -------------------------------------------------------------------------- */
11 
20 
21 #include <tests/smallExample.h>
22 #include <gtsam/geometry/Pose2.h>
27 #include <gtsam/nonlinear/ISAM2.h>
29 #include "examples/SFMdata.h"
30 
31 #include <functional>
32 
33 using namespace std;
34 using namespace gtsam;
35 
36 // Convenience for named keys
38 
39 /* ************************************************************************* */
40 TEST(DoglegOptimizer, ComputeBlend) {
41  // Create an arbitrary Bayes Net
44  0, Vector2(1.0,2.0), (Matrix(2, 2) << 3.0,4.0,0.0,6.0).finished(),
45  3, (Matrix(2, 2) << 7.0,8.0,9.0,10.0).finished(),
46  4, (Matrix(2, 2) << 11.0,12.0,13.0,14.0).finished());
48  1, Vector2(15.0,16.0), (Matrix(2, 2) << 17.0,18.0,0.0,20.0).finished(),
49  2, (Matrix(2, 2) << 21.0,22.0,23.0,24.0).finished(),
50  4, (Matrix(2, 2) << 25.0,26.0,27.0,28.0).finished());
52  2, Vector2(29.0,30.0), (Matrix(2, 2) << 31.0,32.0,0.0,34.0).finished(),
53  3, (Matrix(2, 2) << 35.0,36.0,37.0,38.0).finished());
55  3, Vector2(39.0,40.0), (Matrix(2, 2) << 41.0,42.0,0.0,44.0).finished(),
56  4, (Matrix(2, 2) << 45.0,46.0,47.0,48.0).finished());
58  4, Vector2(49.0,50.0), (Matrix(2, 2) << 51.0,52.0,0.0,54.0).finished());
59 
60  // Compute steepest descent point
62 
63  // Compute Newton's method point
64  VectorValues xn = gbn.optimize();
65 
66  // The Newton's method point should be more "adventurous", i.e. larger, than the steepest descent point
67  EXPECT(xu.vector().norm() < xn.vector().norm());
68 
69  // Compute blend
70  double Delta = 1.5;
71  VectorValues xb = DoglegOptimizerImpl::ComputeBlend(Delta, xu, xn);
72  DOUBLES_EQUAL(Delta, xb.vector().norm(), 1e-10);
73 }
74 
75 /* ************************************************************************* */
76 TEST(DoglegOptimizer, ComputeBlendEdgeCases) {
77  // Test Derived from Issue #1861
78  // Evaluate ComputeBlend Behavior for edge cases where the trust region
79  // is equal in size to that of the newton step or the gradient step.
80 
81  // Simulated Newton (n) and Gradient Descent (u) step vectors w/ ||n|| > ||u||
82  VectorValues::Dims dims;
83  dims[0] = 3;
84  VectorValues n(Vector3(0.3233546123, -0.2133456123, 0.3664345632), dims);
85  VectorValues u(Vector3(0.0023456342, -0.04535687, 0.087345661212), dims);
86 
87  // Test upper edge case where trust region is equal to magnitude of newton step
88  EXPECT(assert_equal(n, DoglegOptimizerImpl::ComputeBlend(n.norm(), u, n, false)));
89  // Test lower edge case where trust region is equal to magnitude of gradient step
90  EXPECT(assert_equal(u, DoglegOptimizerImpl::ComputeBlend(u.norm(), u, n, false)));
91 }
92 
93 /* ************************************************************************* */
94 TEST(DoglegOptimizer, ComputeDoglegPoint) {
95  // Create an arbitrary Bayes Net
98  0, Vector2(1.0,2.0), (Matrix(2, 2) << 3.0,4.0,0.0,6.0).finished(),
99  3, (Matrix(2, 2) << 7.0,8.0,9.0,10.0).finished(),
100  4, (Matrix(2, 2) << 11.0,12.0,13.0,14.0).finished());
102  1, Vector2(15.0,16.0), (Matrix(2, 2) << 17.0,18.0,0.0,20.0).finished(),
103  2, (Matrix(2, 2) << 21.0,22.0,23.0,24.0).finished(),
104  4, (Matrix(2, 2) << 25.0,26.0,27.0,28.0).finished());
106  2, Vector2(29.0,30.0), (Matrix(2, 2) << 31.0,32.0,0.0,34.0).finished(),
107  3, (Matrix(2, 2) << 35.0,36.0,37.0,38.0).finished());
109  3, Vector2(39.0,40.0), (Matrix(2, 2) << 41.0,42.0,0.0,44.0).finished(),
110  4, (Matrix(2, 2) << 45.0,46.0,47.0,48.0).finished());
112  4, Vector2(49.0,50.0), (Matrix(2, 2) << 51.0,52.0,0.0,54.0).finished());
113 
114  // Compute dogleg point for different deltas
115 
116  double Delta1 = 0.5; // Less than steepest descent
117  VectorValues actual1 = DoglegOptimizerImpl::ComputeDoglegPoint(Delta1, gbn.optimizeGradientSearch(), gbn.optimize());
118  DOUBLES_EQUAL(Delta1, actual1.vector().norm(), 1e-5);
119 
120  double Delta2 = 1.5; // Between steepest descent and Newton's method
121  VectorValues expected2 = DoglegOptimizerImpl::ComputeBlend(Delta2, gbn.optimizeGradientSearch(), gbn.optimize());
122  VectorValues actual2 = DoglegOptimizerImpl::ComputeDoglegPoint(Delta2, gbn.optimizeGradientSearch(), gbn.optimize());
123  DOUBLES_EQUAL(Delta2, actual2.vector().norm(), 1e-5);
124  EXPECT(assert_equal(expected2, actual2));
125 
126  double Delta3 = 5.0; // Larger than Newton's method point
127  VectorValues expected3 = gbn.optimize();
128  VectorValues actual3 = DoglegOptimizerImpl::ComputeDoglegPoint(Delta3, gbn.optimizeGradientSearch(), gbn.optimize());
129  EXPECT(assert_equal(expected3, actual3));
130 }
131 
132 /* ************************************************************************* */
134  // really non-linear factor graph
136 
137  // config far from minimum
138  Point2 x0(3,0);
139  Values config;
140  config.insert(X(1), x0);
141 
142  double Delta = 1.0;
143  for(size_t it=0; it<10; ++it) {
144  auto linearized = fg.linearize(config);
145 
146  // Iterate assumes that linear error = nonlinear error at the linearization point, and this should be true
147  double nonlinearError = fg.error(config);
148  double linearError = linearized->error(config.zeroVectors());
149  DOUBLES_EQUAL(nonlinearError, linearError, 1e-5);
150 
151  auto gbn = linearized->eliminateSequential();
153  VectorValues dx_n = gbn->optimize();
154  DoglegOptimizerImpl::IterationResult result = DoglegOptimizerImpl::Iterate(
155  Delta, DoglegOptimizerImpl::SEARCH_EACH_ITERATION, dx_u, dx_n, *gbn, fg,
156  config, fg.error(config));
157  Delta = result.delta;
158  EXPECT(result.f_error < fg.error(config)); // Check that error decreases
159 
160  Values newConfig(config.retract(result.dx_d));
161  config = newConfig;
162  DOUBLES_EQUAL(fg.error(config), result.f_error, 1e-5); // Check that error is correctly filled in
163  }
164 }
165 
166 /* ************************************************************************* */
168  // Create a pose-graph graph with a constraint on the first pose
170  const Pose2 origin(0, 0, 0), pose2(2, 0, 0);
172  auto model = noiseModel::Diagonal::Sigmas(Vector3(0.2, 0.2, 0.1));
174 
175  // Create feasible initial estimate
176  Values initial;
177  initial.insert(1, origin); // feasible !
178  initial.insert(2, Pose2(2.3, 0.1, -0.2));
179 
180  // Optimize the initial values using DoglegOptimizer
182  params.setVerbosityDL("VERBOSITY");
183  DoglegOptimizer optimizer(graph, initial, params);
184  Values result = optimizer.optimize();
185 
186  // Check result
188 
189  // Create infeasible initial estimate
190  Values infeasible;
191  infeasible.insert(1, Pose2(0.1, 0, 0)); // infeasible !
192  infeasible.insert(2, Pose2(2.3, 0.1, -0.2));
193 
194  // Try optimizing with infeasible initial estimate
195  DoglegOptimizer optimizer2(graph, infeasible, params);
196 
197 #ifdef GTSAM_USE_TBB
198  CHECK_EXCEPTION(optimizer2.optimize(), std::exception);
199 #else
200  CHECK_EXCEPTION(optimizer2.optimize(), std::invalid_argument);
201 #endif
202 }
203 
204 /* ************************************************************************* */
213 TEST(DogLegOptimizer, VariableUpdate) {
214  // Make the typename short so it looks much cleaner
216 
217  // create a typedef to the camera type
219  // Define the camera calibration parameters
220  Cal3_S2::shared_ptr K(new Cal3_S2(50.0, 50.0, 0.0, 50.0, 50.0));
221 
222  // Define the camera observation noise model
223  noiseModel::Isotropic::shared_ptr measurementNoise =
224  noiseModel::Isotropic::Sigma(2, 1.0); // one pixel in u and v
225 
226  // Create the set of ground-truth landmarks and poses
227  vector<Point3> points = createPoints();
228  vector<Pose3> poses = createPoses();
229 
230  // Create a factor graph
232 
233  ISAM2DoglegParams doglegparams = ISAM2DoglegParams();
234  doglegparams.verbose = false;
235  ISAM2Params isam2_params;
236  isam2_params.evaluateNonlinearError = true;
237  isam2_params.relinearizeThreshold = 0.0;
238  isam2_params.enableRelinearization = true;
239  isam2_params.optimizationParams = doglegparams;
240  isam2_params.relinearizeSkip = 1;
241  ISAM2 isam2(isam2_params);
242 
243  // Simulated measurements from each camera pose, adding them to the factor
244  // graph
245  unordered_map<int, SmartFactor::shared_ptr> smart_factors;
246  for (size_t j = 0; j < points.size(); ++j) {
247  // every landmark represent a single landmark, we use shared pointer to init
248  // the factor, and then insert measurements.
249  SmartFactor::shared_ptr smartfactor(new SmartFactor(measurementNoise, K));
250 
251  for (size_t i = 0; i < poses.size(); ++i) {
252  // generate the 2D measurement
253  Camera camera(poses[i], K);
254  Point2 measurement = camera.project(points[j]);
255 
256  // call add() function to add measurement into a single factor, here we
257  // need to add:
258  // 1. the 2D measurement
259  // 2. the corresponding camera's key
260  // 3. camera noise model
261  // 4. camera calibration
262 
263  // add only first 3 measurements and update the later measurements
264  // incrementally
265  if (i < 3) smartfactor->add(measurement, i);
266  }
267 
268  // insert the smart factor in the graph
269  smart_factors[j] = smartfactor;
270  graph.push_back(smartfactor);
271  }
272 
273  // Add a prior on pose x0. This indirectly specifies where the origin is.
274  // 30cm std on x,y,z 0.1 rad on roll,pitch,yaw
275  noiseModel::Diagonal::shared_ptr noise = noiseModel::Diagonal::Sigmas(
276  (Vector(6) << Vector3::Constant(0.3), Vector3::Constant(0.1)).finished());
277  graph.emplace_shared<PriorFactor<Pose3> >(0, poses[0], noise);
278 
279  // Because the structure-from-motion problem has a scale ambiguity, the
280  // problem is still under-constrained. Here we add a prior on the second pose
281  // x1, so this will fix the scale by indicating the distance between x0 and
282  // x1. Because these two are fixed, the rest of the poses will be also be
283  // fixed.
285  noise); // add directly to graph
286 
287  // Create the initial estimate to the solution
288  // Intentionally initialize the variables off from the ground truth
289  Values initialEstimate;
290  Pose3 delta(Rot3::Rodrigues(-0.1, 0.2, 0.25), Point3(0.05, -0.10, 0.20));
291  for (size_t i = 0; i < 3; ++i)
292  initialEstimate.insert(i, poses[i].compose(delta));
293  // initialEstimate.print("Initial Estimates:\n");
294 
295  // Optimize the graph and print results
296  isam2.update(graph, initialEstimate);
297  Values result = isam2.calculateEstimate();
298  // result.print("Results:\n");
299 
300  // we add new measurements from this pose
301  size_t pose_idx = 3;
302 
303  // Now update existing smart factors with new observations
304  for (size_t j = 0; j < points.size(); ++j) {
305  SmartFactor::shared_ptr smartfactor = smart_factors[j];
306 
307  // add the 4th measurement
308  Camera camera(poses[pose_idx], K);
309  Point2 measurement = camera.project(points[j]);
310  smartfactor->add(measurement, pose_idx);
311  }
312 
313  graph.resize(0);
314  initialEstimate.clear();
315 
316  // update initial estimate for the new pose
317  initialEstimate.insert(pose_idx, poses[pose_idx].compose(delta));
318 
319  // this should break the system
320  isam2.update(graph, initialEstimate);
321  result = isam2.calculateEstimate();
322  EXPECT(std::find(result.keys().begin(), result.keys().end(), pose_idx) !=
323  result.keys().end());
324 }
325 
326 /* ************************************************************************* */
327 int main() { TestResult tr; return TestRegistry::runAllTests(tr); }
328 /* ************************************************************************* */
TestRegistry::runAllTests
static int runAllTests(TestResult &result)
Definition: TestRegistry.cpp:27
gtsam::ISAM2Params
Definition: ISAM2Params.h:136
gtsam::VectorValues::Dims
std::map< Key, size_t > Dims
Keyed vector dimensions.
Definition: VectorValues.h:89
CHECK_EXCEPTION
#define CHECK_EXCEPTION(condition, exception_name)
Definition: Test.h:118
Pose2.h
2D Pose
gtsam::NonlinearOptimizer::optimize
virtual const Values & optimize()
Definition: NonlinearOptimizer.h:98
gtsam::ISAM2Params::enableRelinearization
bool enableRelinearization
Definition: ISAM2Params.h:175
gtsam::ISAM2
Definition: ISAM2.h:45
gtsam::Values::keys
KeyVector keys() const
Definition: Values.cpp:219
gtsam::createPoints
std::vector< Point3 > createPoints()
Create a set of ground-truth landmarks.
Definition: SFMdata.h:43
gtsam::GaussianBayesNet::optimizeGradientSearch
VectorValues optimizeGradientSearch() const
Definition: GaussianBayesNet.cpp:89
e
Array< double, 1, 3 > e(1./3., 0.5, 2.)
EXPECT
#define EXPECT(condition)
Definition: Test.h:150
gtsam::Vector2
Eigen::Vector2d Vector2
Definition: Vector.h:43
TestHarness.h
Camera
Definition: camera.h:36
TEST
TEST(DoglegOptimizer, ComputeBlend)
Definition: testDoglegOptimizer.cpp:40
initial
Values initial
Definition: OdometryOptimize.cpp:2
gtsam::PriorFactor
Definition: nonlinear/PriorFactor.h:30
gtsam::NonlinearFactorGraph::linearize
std::shared_ptr< GaussianFactorGraph > linearize(const Values &linearizationPoint) const
Linearize a nonlinear factor graph.
Definition: NonlinearFactorGraph.cpp:239
simple::pose2
static Pose3 pose2
Definition: testInitializePose3.cpp:58
X
#define X
Definition: icosphere.cpp:20
gtsam::Vector3
Eigen::Vector3d Vector3
Definition: Vector.h:44
vanilla::params
static const SmartProjectionParams params
Definition: smartFactorScenarios.h:69
result
Values result
Definition: OdometryOptimize.cpp:8
gtsam::NonlinearEquality
Definition: NonlinearEquality.h:43
gtsam::ISAM2Params::optimizationParams
OptimizationParams optimizationParams
Definition: ISAM2Params.h:151
gtsam::ISAM2DoglegParams
Definition: ISAM2Params.h:68
gtsam::DoglegOptimizer
Definition: DoglegOptimizer.h:68
n
int n
Definition: BiCGSTAB_simple.cpp:1
gtsam::NonlinearFactorGraph::error
double error(const Values &values) const
Definition: NonlinearFactorGraph.cpp:170
ceres::Matrix
Eigen::Matrix< double, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor > Matrix
Definition: gtsam/3rdparty/ceres/eigen.h:42
gtsam::Values::at
const ValueType at(Key j) const
Definition: Values-inl.h:261
gtsam::Values::retract
Values retract(const VectorValues &delta) const
Definition: Values.cpp:99
gtsam::FactorGraph::resize
virtual void resize(size_t size)
Definition: FactorGraph.h:367
gtsam::ISAM2Params::relinearizeThreshold
RelinearizationThreshold relinearizeThreshold
Definition: ISAM2Params.h:169
gtsam.examples.PlanarManipulatorExample.delta
def delta(g0, g1)
Definition: PlanarManipulatorExample.py:45
gtsam::VectorValues
Definition: VectorValues.h:74
j
std::ptrdiff_t j
Definition: tut_arithmetic_redux_minmax.cpp:2
BetweenFactor.h
gtsam::Pose3
Definition: Pose3.h:37
DOUBLES_EQUAL
#define DOUBLES_EQUAL(expected, actual, threshold)
Definition: Test.h:141
gtsam::SmartProjectionPoseFactor
Definition: SmartProjectionPoseFactor.h:45
gtsam::NonlinearFactorGraph
Definition: NonlinearFactorGraph.h:55
gtsam::GaussianConditional
Definition: GaussianConditional.h:40
x0
static Symbol x0('x', 0)
gtsam::GaussianBayesNet::optimize
VectorValues optimize() const
Definition: GaussianBayesNet.cpp:44
gtsam::Point2
Vector2 Point2
Definition: Point2.h:32
gtsam::Cal3_S2::shared_ptr
std::shared_ptr< Cal3_S2 > shared_ptr
Definition: Cal3_S2.h:39
SmartFactor
SmartProjectionPoseFactor< Cal3_S2 > SmartFactor
Definition: ISAM2Example_SmartFactor.cpp:22
NonlinearEquality.h
SmartProjectionPoseFactor.h
Smart factor on poses, assuming camera calibration is fixed.
sampling::gbn
static const GaussianBayesNet gbn
Definition: testGaussianBayesNet.cpp:171
model
noiseModel::Diagonal::shared_ptr model
Definition: doc/Code/Pose2SLAMExample.cpp:7
TestResult
Definition: TestResult.h:26
gtsam::VectorValues::norm
double norm() const
Definition: VectorValues.cpp:255
gtsam::DoglegParams
Definition: DoglegOptimizer.h:32
gtsam::ISAM2::calculateEstimate
Values calculateEstimate() const
Definition: ISAM2.cpp:786
ISAM2.h
Incremental update functionality (ISAM2) for BayesTree, with fluid relinearization.
gtsam::Values::zeroVectors
VectorValues zeroVectors() const
Definition: Values.cpp:261
gtsam::createPoses
std::vector< Pose3 > createPoses(const Pose3 &init=Pose3(Rot3::Ypr(M_PI_2, 0, -M_PI_2), {30, 0, 0}), const Pose3 &delta=Pose3(Rot3::Ypr(0, -M_PI_4, 0), {sin(M_PI_4) *30, 0, 30 *(1 - sin(M_PI_4))}), int steps=8)
Definition: SFMdata.h:62
gtsam
traits
Definition: SFMdata.h:40
gtsam::ISAM2::update
virtual ISAM2Result update(const NonlinearFactorGraph &newFactors=NonlinearFactorGraph(), const Values &newTheta=Values(), const FactorIndices &removeFactorIndices=FactorIndices(), const std::optional< FastMap< Key, int > > &constrainedKeys={}, const std::optional< FastList< Key > > &noRelinKeys={}, const std::optional< FastList< Key > > &extraReelimKeys={}, bool force_relinearize=false)
Definition: ISAM2.cpp:401
K
#define K
Definition: igam.h:8
gtsam::FactorGraph::push_back
IsDerived< DERIVEDFACTOR > push_back(std::shared_ptr< DERIVEDFACTOR > factor)
Add a factor directly using a shared_ptr.
Definition: FactorGraph.h:147
gtsam::Values
Definition: Values.h:65
gtsam::testing::compose
T compose(const T &t1, const T &t2)
Definition: lieProxies.h:39
std
Definition: BFloat16.h:88
gtsam::DoglegOptimizerImpl::IterationResult
Definition: DoglegOptimizerImpl.h:35
origin
set noclip points set clip one set noclip two set bar set border lt lw set xdata set ydata set zdata set x2data set y2data set boxwidth set dummy y set format x g set format y g set format x2 g set format y2 g set format z g set angles radians set nogrid set key title set key left top Right noreverse box linetype linewidth samplen spacing width set nolabel set noarrow set nologscale set logscale x set set pointsize set encoding default set nopolar set noparametric set set set set surface set nocontour set clabel set mapping cartesian set nohidden3d set cntrparam order set cntrparam linear set cntrparam levels auto set cntrparam points set size set origin
Definition: gnuplot_common_settings.hh:45
gtsam::noiseModel::Diagonal::shared_ptr
std::shared_ptr< Diagonal > shared_ptr
Definition: NoiseModel.h:321
gtsam::Values::insert
void insert(Key j, const Value &val)
Definition: Values.cpp:156
Camera
PinholePose< Cal3_S2 > Camera
Definition: SFMExample_SmartFactor.cpp:38
gtsam::assert_equal
bool assert_equal(const Matrix &expected, const Matrix &actual, double tol)
Definition: Matrix.cpp:41
gtsam::Cal3_S2
The most common 5DOF 3D->2D calibration.
Definition: Cal3_S2.h:34
gtsam::Point3
Vector3 Point3
Definition: Point3.h:38
gtsam::example::createReallyNonlinearFactorGraph
NonlinearFactorGraph createReallyNonlinearFactorGraph()
Definition: smallExample.h:378
DoglegOptimizerImpl.h
Nonlinear factor graph optimizer using Powell's Dogleg algorithm (detail implementation)
initial
Definition: testScenarioRunner.cpp:148
main
int main()
Definition: testDoglegOptimizer.cpp:327
gtsam::noiseModel::Isotropic::shared_ptr
std::shared_ptr< Isotropic > shared_ptr
Definition: NoiseModel.h:559
DoglegOptimizer.h
gtsam::SmartProjectionPoseFactor::shared_ptr
std::shared_ptr< This > shared_ptr
shorthand for a smart pointer to a factor
Definition: SmartProjectionPoseFactor.h:59
camera
static const CalibratedCamera camera(kDefaultPose)
gtsam::ISAM2DoglegParams::verbose
bool verbose
Whether Dogleg prints iteration and convergence information.
Definition: ISAM2Params.h:77
gtsam::Constraint
Definition: Constraint.h:35
gtsam::PinholePose
Definition: PinholePose.h:239
graph
NonlinearFactorGraph graph
Definition: doc/Code/OdometryExample.cpp:2
gtsam::ISAM2Params::relinearizeSkip
int relinearizeSkip
Definition: ISAM2Params.h:171
smallExample.h
Create small example with two poses and one landmark.
SFMdata.h
Simple example for the structure-from-motion problems.
ceres::Vector
Eigen::Matrix< double, Eigen::Dynamic, 1 > Vector
Definition: gtsam/3rdparty/ceres/eigen.h:38
i
int i
Definition: BiCGSTAB_step_by_step.cpp:9
gtsam::ISAM2Params::evaluateNonlinearError
bool evaluateNonlinearError
Definition: ISAM2Params.h:178
measurement
static Point2 measurement(323.0, 240.0)
gtsam::GaussianBayesNet
Definition: GaussianBayesNet.h:35
gtsam::CalibratedCamera::project
Point2 project(const Point3 &point, OptionalJacobian< 2, 6 > Dcamera={}, OptionalJacobian< 2, 3 > Dpoint={}) const
Definition: CalibratedCamera.cpp:188
gtsam::FactorGraph::emplace_shared
IsDerived< DERIVEDFACTOR > emplace_shared(Args &&... args)
Emplace a shared pointer to factor of given type.
Definition: FactorGraph.h:153
gtsam::Pose2
Definition: Pose2.h:39
gtsam::VectorValues::vector
Vector vector() const
Definition: VectorValues.cpp:175
gtsam::BetweenFactor
Definition: BetweenFactor.h:40


gtsam
Author(s):
autogenerated on Tue Jan 7 2025 04:07:10