| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  *  @file   testIterative.cpp | 
					
						
							|  |  |  |  *  @brief  Unit tests for iterative methods | 
					
						
							|  |  |  |  *  @author Frank Dellaert | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <boost/assign/std/list.hpp> // for operator +=
 | 
					
						
							|  |  |  | using namespace boost::assign; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <CppUnitLite/TestHarness.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-18 13:38:53 +08:00
										 |  |  | // TODO: DANGEROUS, create shared pointers
 | 
					
						
							| 
									
										
										
										
											2010-01-27 12:39:35 +08:00
										 |  |  | #define GTSAM_MAGIC_GAUSSIAN 3
 | 
					
						
							| 
									
										
										
										
											2010-01-18 03:34:57 +08:00
										 |  |  | #define GTSAM_MAGIC_KEY
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | #include "Ordering.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-17 11:29:12 +08:00
										 |  |  | #include "VectorConfig.h"
 | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | #include "smallExample.h"
 | 
					
						
							| 
									
										
										
										
											2010-01-17 02:01:16 +08:00
										 |  |  | #include "pose2SLAM.h"
 | 
					
						
							| 
									
										
										
										
											2010-01-18 13:51:19 +08:00
										 |  |  | #include "SubgraphPreconditioner.h"
 | 
					
						
							| 
									
										
										
										
											2010-01-18 03:34:57 +08:00
										 |  |  | #include "FactorGraph-inl.h"
 | 
					
						
							|  |  |  | #include "NonlinearFactorGraph-inl.h"
 | 
					
						
							| 
									
										
										
										
											2010-01-18 13:51:19 +08:00
										 |  |  | #include "iterative-inl.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | using namespace std; | 
					
						
							|  |  |  | using namespace gtsam; | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | using namespace example; | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-14 13:52:20 +08:00
										 |  |  | static bool verbose = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-16 13:10:07 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | TEST( Iterative, steepestDescent ) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	// Expected solution
 | 
					
						
							|  |  |  | 	Ordering ord; | 
					
						
							|  |  |  | 	ord += "l1", "x1", "x2"; | 
					
						
							|  |  |  | 	GaussianFactorGraph fg = createGaussianFactorGraph(); | 
					
						
							|  |  |  | 	VectorConfig expected = fg.optimize(ord); // destructive
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Do gradient descent
 | 
					
						
							|  |  |  | 	GaussianFactorGraph fg2 = createGaussianFactorGraph(); | 
					
						
							|  |  |  | 	VectorConfig zero = createZeroDelta(); | 
					
						
							|  |  |  | 	VectorConfig actual = steepestDescent(fg2, zero, verbose); | 
					
						
							|  |  |  | 	CHECK(assert_equal(expected,actual,1e-2)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | TEST( Iterative, conjugateGradientDescent ) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	// Expected solution
 | 
					
						
							|  |  |  | 	Ordering ord; | 
					
						
							|  |  |  | 	ord += "l1", "x1", "x2"; | 
					
						
							|  |  |  | 	GaussianFactorGraph fg = createGaussianFactorGraph(); | 
					
						
							|  |  |  | 	VectorConfig expected = fg.optimize(ord); // destructive
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// create graph and get matrices
 | 
					
						
							|  |  |  | 	GaussianFactorGraph fg2 = createGaussianFactorGraph(); | 
					
						
							|  |  |  | 	Matrix A; | 
					
						
							|  |  |  | 	Vector b; | 
					
						
							|  |  |  | 	Vector x0 = gtsam::zero(6); | 
					
						
							|  |  |  | 	boost::tie(A, b) = fg2.matrix(ord); | 
					
						
							|  |  |  | 	Vector expectedX = Vector_(6, -0.1, 0.1, -0.1, -0.1, 0.1, -0.2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Do conjugate gradient descent, System version
 | 
					
						
							|  |  |  | 	System Ab(A, b); | 
					
						
							| 
									
										
										
										
											2010-02-14 13:52:20 +08:00
										 |  |  | 	Vector actualX = conjugateGradientDescent(Ab, x0, verbose); | 
					
						
							| 
									
										
										
										
											2010-01-16 13:10:07 +08:00
										 |  |  | 	CHECK(assert_equal(expectedX,actualX,1e-9)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Do conjugate gradient descent, Matrix version
 | 
					
						
							| 
									
										
										
										
											2010-02-14 13:52:20 +08:00
										 |  |  | 	Vector actualX2 = conjugateGradientDescent(A, b, x0, verbose); | 
					
						
							| 
									
										
										
										
											2010-01-16 13:10:07 +08:00
										 |  |  | 	CHECK(assert_equal(expectedX,actualX2,1e-9)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Do conjugate gradient descent on factor graph
 | 
					
						
							|  |  |  | 	VectorConfig zero = createZeroDelta(); | 
					
						
							| 
									
										
										
										
											2010-02-14 13:52:20 +08:00
										 |  |  | 	VectorConfig actual = conjugateGradientDescent(fg2, zero, verbose); | 
					
						
							| 
									
										
										
										
											2010-01-16 13:10:07 +08:00
										 |  |  | 	CHECK(assert_equal(expected,actual,1e-2)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Test method
 | 
					
						
							| 
									
										
										
										
											2010-02-14 13:52:20 +08:00
										 |  |  | 	VectorConfig actual2 = fg2.conjugateGradientDescent(zero, verbose); | 
					
						
							| 
									
										
										
										
											2010-01-16 13:10:07 +08:00
										 |  |  | 	CHECK(assert_equal(expected,actual2,1e-2)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-01-16 13:08:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* *
 | 
					
						
							|  |  |  | TEST( Iterative, conjugateGradientDescent_hard_constraint ) | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-16 13:08:29 +08:00
										 |  |  | 	typedef Pose2Config::Key Key; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Pose2Config config; | 
					
						
							|  |  |  | 	config.insert(1, Pose2(0.,0.,0.)); | 
					
						
							|  |  |  | 	config.insert(2, Pose2(1.5,0.,0.)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Pose2Graph graph; | 
					
						
							|  |  |  | 	Matrix cov = eye(3); | 
					
						
							|  |  |  | 	graph.push_back(Pose2Graph::sharedFactor(new Pose2Factor(Key(1), Key(2), Pose2(1.,0.,0.), cov))); | 
					
						
							| 
									
										
										
										
											2010-02-17 11:29:12 +08:00
										 |  |  | 	graph.addHardConstraint(1, config[1]); | 
					
						
							| 
									
										
										
										
											2010-01-16 13:08:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	VectorConfig zeros; | 
					
						
							|  |  |  | 	zeros.insert("x1",zero(3)); | 
					
						
							|  |  |  | 	zeros.insert("x2",zero(3)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GaussianFactorGraph fg = graph.linearize(config); | 
					
						
							|  |  |  | 	VectorConfig actual = conjugateGradientDescent(fg, zeros, true, 1e-3, 1e-5, 10); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	VectorConfig expected; | 
					
						
							|  |  |  | 	expected.insert("x1", zero(3)); | 
					
						
							|  |  |  | 	expected.insert("x2", Vector_(-0.5,0.,0.)); | 
					
						
							|  |  |  | 	CHECK(assert_equal(expected, actual)); | 
					
						
							| 
									
										
										
										
											2009-12-28 18:48:48 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-28 18:48:48 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2010-01-16 13:08:29 +08:00
										 |  |  | TEST( Iterative, conjugateGradientDescent_soft_constraint ) | 
					
						
							| 
									
										
										
										
											2009-12-28 18:48:48 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-16 13:08:29 +08:00
										 |  |  | 	Pose2Config config; | 
					
						
							|  |  |  | 	config.insert(1, Pose2(0.,0.,0.)); | 
					
						
							|  |  |  | 	config.insert(2, Pose2(1.5,0.,0.)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Pose2Graph graph; | 
					
						
							| 
									
										
										
										
											2010-01-27 12:39:35 +08:00
										 |  |  | 	graph.addPrior(1, Pose2(0.,0.,0.), Isotropic::Sigma(3, 1e-10)); | 
					
						
							|  |  |  | 	graph.addConstraint(1,2, Pose2(1.,0.,0.), Isotropic::Sigma(3, 1)); | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-16 13:08:29 +08:00
										 |  |  | 	VectorConfig zeros; | 
					
						
							|  |  |  | 	zeros.insert("x1",zero(3)); | 
					
						
							|  |  |  | 	zeros.insert("x2",zero(3)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GaussianFactorGraph fg = graph.linearize(config); | 
					
						
							| 
									
										
										
										
											2010-02-14 13:52:20 +08:00
										 |  |  | 	VectorConfig actual = conjugateGradientDescent(fg, zeros, verbose, 1e-3, 1e-5, 100); | 
					
						
							| 
									
										
										
										
											2010-01-16 13:08:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	VectorConfig expected; | 
					
						
							|  |  |  | 	expected.insert("x1", zero(3)); | 
					
						
							|  |  |  | 	expected.insert("x2", Vector_(3,-0.5,0.,0.)); | 
					
						
							|  |  |  | 	CHECK(assert_equal(expected, actual)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-01-18 13:51:19 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | TEST( Iterative, subgraphPCG ) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	typedef Pose2Config::Key Key; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Pose2Config theta_bar; | 
					
						
							|  |  |  | 	theta_bar.insert(1, Pose2(0.,0.,0.)); | 
					
						
							|  |  |  | 	theta_bar.insert(2, Pose2(1.5,0.,0.)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Pose2Graph graph; | 
					
						
							| 
									
										
										
										
											2010-01-27 12:39:35 +08:00
										 |  |  | 	graph.addPrior(1, Pose2(0.,0.,0.), Isotropic::Sigma(3, 1e-10)); | 
					
						
							|  |  |  | 	graph.addConstraint(1,2, Pose2(1.,0.,0.), Isotropic::Sigma(3, 1)); | 
					
						
							| 
									
										
										
										
											2010-01-18 13:51:19 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// generate spanning tree and create ordering
 | 
					
						
							|  |  |  | 	PredecessorMap<Key> tree = graph.findMinimumSpanningTree<Key, Pose2Factor>(); | 
					
						
							|  |  |  | 	list<Key> keys = predecessorMap2Keys(tree); | 
					
						
							|  |  |  | 	list<Symbol> symbols; | 
					
						
							|  |  |  | 	symbols.resize(keys.size()); | 
					
						
							|  |  |  | 	std::transform(keys.begin(), keys.end(), symbols.begin(), key2symbol<Key>); | 
					
						
							|  |  |  | 	Ordering ordering(symbols); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Key root = keys.back(); | 
					
						
							|  |  |  | 	Pose2Graph T, C; | 
					
						
							|  |  |  | 	graph.split<Key, Pose2Factor>(tree, T, C); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// build the subgraph PCG system
 | 
					
						
							| 
									
										
										
										
											2010-01-23 09:03:47 +08:00
										 |  |  | 	GaussianFactorGraph Ab1_ = T.linearize(theta_bar); | 
					
						
							|  |  |  | 	SubgraphPreconditioner::sharedFG Ab1 = T.linearize_(theta_bar); | 
					
						
							| 
									
										
										
										
											2010-01-19 18:46:12 +08:00
										 |  |  | 	SubgraphPreconditioner::sharedFG Ab2 = C.linearize_(theta_bar); | 
					
						
							| 
									
										
										
										
											2010-01-23 09:03:47 +08:00
										 |  |  | 	SubgraphPreconditioner::sharedBayesNet Rc1 = Ab1_.eliminate_(ordering); | 
					
						
							| 
									
										
										
										
											2010-01-19 18:46:12 +08:00
										 |  |  | 	SubgraphPreconditioner::sharedConfig xbar = optimize_(*Rc1); | 
					
						
							| 
									
										
										
										
											2010-01-23 09:03:47 +08:00
										 |  |  | 	SubgraphPreconditioner system(Ab1, Ab2, Rc1, xbar); | 
					
						
							| 
									
										
										
										
											2010-01-18 13:51:19 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 11:29:12 +08:00
										 |  |  | 	VectorConfig zeros = VectorConfig::zero(*xbar); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-18 13:51:19 +08:00
										 |  |  | 	// Solve the subgraph PCG
 | 
					
						
							|  |  |  | 	VectorConfig ybar = conjugateGradients<SubgraphPreconditioner, VectorConfig, | 
					
						
							| 
									
										
										
										
											2010-02-14 13:52:20 +08:00
										 |  |  | 			Errors> (system, zeros, verbose, 1e-5, 1e-5, 100); | 
					
						
							| 
									
										
										
										
											2010-01-18 13:51:19 +08:00
										 |  |  | 	VectorConfig actual = system.x(ybar); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	VectorConfig expected; | 
					
						
							|  |  |  | 	expected.insert("x1", zero(3)); | 
					
						
							|  |  |  | 	expected.insert("x2", Vector_(3, -0.5, 0., 0.)); | 
					
						
							|  |  |  | 	CHECK(assert_equal(expected, actual)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-02-17 11:29:12 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-12-28 18:48:48 +08:00
										 |  |  | int main() { | 
					
						
							|  |  |  | 	TestResult tr; | 
					
						
							|  |  |  | 	return TestRegistry::runAllTests(tr); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-12-28 17:44:30 +08:00
										 |  |  | /* ************************************************************************* */ |