| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * @file    smallExample.cpp | 
					
						
							|  |  |  |  * @brief   Create small example with two poses and one landmark | 
					
						
							|  |  |  |  * @brief   smallExample | 
					
						
							|  |  |  |  * @author  Carlos Nieto | 
					
						
							|  |  |  |  * @author  Frank dellaert | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <iostream>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | #include <boost/optional.hpp>
 | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | using namespace std; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-18 03:34:57 +08:00
										 |  |  | #define GTSAM_MAGIC_KEY
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-20 01:23:19 +08:00
										 |  |  | #include <gtsam/inference/Ordering.h>
 | 
					
						
							|  |  |  | #include <gtsam/base/Matrix.h>
 | 
					
						
							|  |  |  | #include <gtsam/nonlinear/NonlinearFactor.h>
 | 
					
						
							|  |  |  | #include <gtsam/slam/smallExample.h>
 | 
					
						
							| 
									
										
										
										
											2009-10-27 21:34:36 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // template definitions
 | 
					
						
							| 
									
										
										
										
											2010-08-20 01:23:19 +08:00
										 |  |  | #include <gtsam/inference/FactorGraph-inl.h>
 | 
					
						
							|  |  |  | #include <gtsam/nonlinear/TupleConfig-inl.h>
 | 
					
						
							|  |  |  | #include <gtsam/nonlinear/NonlinearFactorGraph-inl.h>
 | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace gtsam { | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | namespace example { | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	typedef boost::shared_ptr<NonlinearFactor<Config> > shared; | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 	static SharedDiagonal sigma1_0 = noiseModel::Isotropic::Sigma(2,1.0); | 
					
						
							|  |  |  | 	static SharedDiagonal sigma0_1 = noiseModel::Isotropic::Sigma(2,0.1); | 
					
						
							|  |  |  | 	static SharedDiagonal sigma0_2 = noiseModel::Isotropic::Sigma(2,0.2); | 
					
						
							|  |  |  | 	static SharedDiagonal constraintModel = noiseModel::Constrained::All(2); | 
					
						
							| 
									
										
										
										
											2010-01-21 02:32:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	boost::shared_ptr<const Graph> sharedNonlinearFactorGraph() { | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		// Create
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		boost::shared_ptr<Graph> nlfg( | 
					
						
							|  |  |  | 				new Graph); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// prior on x1
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Point2 mu; | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 		shared f1(new simulated2D::Prior(mu, sigma0_1, 1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		nlfg->push_back(f1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// odometry between x1 and x2
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Point2 z2(1.5, 0); | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 		shared f2(new simulated2D::Odometry(z2, sigma0_1, 1, 2)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		nlfg->push_back(f2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// measurement between x1 and l1
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Point2 z3(0, -1); | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 		shared f3(new simulated2D::Measurement(z3, sigma0_2, 1, 1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		nlfg->push_back(f3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// measurement between x2 and l1
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Point2 z4(-1.5, -1.); | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 		shared f4(new simulated2D::Measurement(z4, sigma0_2, 2, 1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		nlfg->push_back(f4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return nlfg; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	Graph createNonlinearFactorGraph() { | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		return *sharedNonlinearFactorGraph(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-09-09 12:43:04 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	Config createConfig() { | 
					
						
							|  |  |  | 		Config c; | 
					
						
							|  |  |  | 		c.insert(simulated2D::PoseKey(1), Point2(0.0, 0.0)); | 
					
						
							|  |  |  | 		c.insert(simulated2D::PoseKey(2), Point2(1.5, 0.0)); | 
					
						
							|  |  |  | 		c.insert(simulated2D::PointKey(1), Point2(0.0, -1.0)); | 
					
						
							|  |  |  | 		return c; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	VectorConfig createVectorConfig() { | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		VectorConfig c; | 
					
						
							| 
									
										
										
										
											2010-02-16 07:54:10 +08:00
										 |  |  | 		c.insert("l1", Vector_(2, 0.0, -1.0)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		c.insert("x1", Vector_(2, 0.0, 0.0)); | 
					
						
							|  |  |  | 		c.insert("x2", Vector_(2, 1.5, 0.0)); | 
					
						
							|  |  |  | 		return c; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	boost::shared_ptr<const Config> sharedNoisyConfig() { | 
					
						
							|  |  |  | 		boost::shared_ptr<Config> c(new Config); | 
					
						
							|  |  |  | 		c->insert(simulated2D::PoseKey(1), Point2(0.1, 0.1)); | 
					
						
							|  |  |  | 		c->insert(simulated2D::PoseKey(2), Point2(1.4, 0.2)); | 
					
						
							|  |  |  | 		c->insert(simulated2D::PointKey(1), Point2(0.1, -1.1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		return c; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	Config createNoisyConfig() { | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		return *sharedNoisyConfig(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	VectorConfig createCorrectDelta() { | 
					
						
							|  |  |  | 		VectorConfig c; | 
					
						
							| 
									
										
										
										
											2010-02-16 07:54:10 +08:00
										 |  |  | 		c.insert("l1", Vector_(2, -0.1, 0.1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		c.insert("x1", Vector_(2, -0.1, -0.1)); | 
					
						
							|  |  |  | 		c.insert("x2", Vector_(2, 0.1, -0.2)); | 
					
						
							|  |  |  | 		return c; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-09-09 12:43:04 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	VectorConfig createZeroDelta() { | 
					
						
							|  |  |  | 		VectorConfig c; | 
					
						
							| 
									
										
										
										
											2010-02-16 07:54:10 +08:00
										 |  |  | 		c.insert("l1", zero(2)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		c.insert("x1", zero(2)); | 
					
						
							|  |  |  | 		c.insert("x2", zero(2)); | 
					
						
							|  |  |  | 		return c; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	GaussianFactorGraph createGaussianFactorGraph() { | 
					
						
							|  |  |  | 		Matrix I = eye(2); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		// Create empty graph
 | 
					
						
							|  |  |  | 		GaussianFactorGraph fg; | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-22 07:50:28 +08:00
										 |  |  | 		SharedDiagonal unit2 = noiseModel::Unit::Create(2); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		// linearized prior on x1: c["x1"]+x1=0 i.e. x1=-c["x1"]
 | 
					
						
							| 
									
										
										
										
											2010-02-22 07:50:28 +08:00
										 |  |  | 		fg.add("x1", 10*eye(2), -1.0*ones(2), unit2); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		// odometry between x1 and x2: x2-x1=[0.2;-0.1]
 | 
					
						
							| 
									
										
										
										
											2010-02-22 07:50:28 +08:00
										 |  |  | 		 fg.add("x1", -10*eye(2),"x2", 10*eye(2), Vector_(2, 2.0, -1.0), unit2); | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		// measurement between x1 and l1: l1-x1=[0.0;0.2]
 | 
					
						
							| 
									
										
										
										
											2010-02-22 07:50:28 +08:00
										 |  |  | 		fg.add("x1", -5*eye(2), "l1", 5*eye(2), Vector_(2, 0.0, 1.0), unit2); | 
					
						
							| 
									
										
										
										
											2009-09-09 12:43:04 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		// measurement between x2 and l1: l1-x2=[-0.2;0.3]
 | 
					
						
							| 
									
										
										
										
											2010-02-22 07:50:28 +08:00
										 |  |  | 		fg.add("x2", -5*eye(2), "l1", 5*eye(2), Vector_(2, -1.0, 1.5), unit2); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return fg; | 
					
						
							| 
									
										
										
										
											2009-10-27 21:34:36 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	/** create small Chordal Bayes Net x <- y
 | 
					
						
							|  |  |  | 	 * x y d | 
					
						
							|  |  |  | 	 * 1 1 9 | 
					
						
							|  |  |  | 	 *   1 5 | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	GaussianBayesNet createSmallGaussianBayesNet() { | 
					
						
							|  |  |  | 		Matrix R11 = Matrix_(1, 1, 1.0), S12 = Matrix_(1, 1, 1.0); | 
					
						
							|  |  |  | 		Matrix R22 = Matrix_(1, 1, 1.0); | 
					
						
							|  |  |  | 		Vector d1(1), d2(1); | 
					
						
							|  |  |  | 		d1(0) = 9; | 
					
						
							|  |  |  | 		d2(0) = 5; | 
					
						
							|  |  |  | 		Vector tau(1); | 
					
						
							|  |  |  | 		tau(0) = 1.0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// define nodes and specify in reverse topological sort (i.e. parents last)
 | 
					
						
							|  |  |  | 		GaussianConditional::shared_ptr Px_y(new GaussianConditional("x", d1, R11, | 
					
						
							|  |  |  | 				"y", S12, tau)), Py(new GaussianConditional("y", d2, R22, tau)); | 
					
						
							|  |  |  | 		GaussianBayesNet cbn; | 
					
						
							|  |  |  | 		cbn.push_back(Px_y); | 
					
						
							|  |  |  | 		cbn.push_back(Py); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return cbn; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	// Some nonlinear functions to optimize
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	namespace smallOptimize { | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Point2 h(const Point2& v) { | 
					
						
							|  |  |  | 			return Point2(cos(v.x()), sin(v.y())); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Matrix H(const Point2& v) { | 
					
						
							|  |  |  | 			return Matrix_(2, 2, | 
					
						
							|  |  |  | 					-sin(v.x()), 0.0, | 
					
						
							|  |  |  | 					 0.0, cos(v.y())); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		struct UnaryFactor: public gtsam::NonlinearFactor1<Config, | 
					
						
							|  |  |  | 		simulated2D::PoseKey, Point2> { | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 			Point2 z_; | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 			UnaryFactor(const Point2& z, const SharedGaussian& model, | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 					const simulated2D::PoseKey& key) : | 
					
						
							|  |  |  | 				gtsam::NonlinearFactor1<Config, simulated2D::PoseKey, | 
					
						
							|  |  |  | 						Point2>(model, key), z_(z) { | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 			Vector evaluateError(const Point2& x, boost::optional<Matrix&> A = | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | 					boost::none) const { | 
					
						
							|  |  |  | 				if (A) *A = H(x); | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 				return (h(x) - z_).vector(); | 
					
						
							| 
									
										
											  
											
												Large gtsam refactoring
To support faster development *and* better performance Richard and I pushed through a large refactoring of NonlinearFactors.
The following are the biggest changes:
1) NonLinearFactor1 and NonLinearFactor2 are now templated on Config, Key type, and X type, where X is the argument to the measurement function.
2) The measurement itself is no longer kept in the nonlinear factor. Instead, a derived class (see testVSLAMFactor, testNonlinearEquality, testPose3Factor etc...) has to implement a function to compute the errors, "evaluateErrors". Instead of (h(x)-z), it needs to return (z-h(x)), so Ax-b is an approximation of the error. IMPORTANT: evaluateErrors needs - if asked - *combine* the calculation of the function value h(x) and the derivatives dh(x)/dx. This was a major performance issue. To do this, boost::optional<Matrix&> arguments are provided, and tin EvaluateErrors you just  says something like
	if (H) *H = Matrix_(3,6,....);
3) We are no longer using int or strings for nonlinear factors. Instead, the preferred key type is now Symbol, defined in Key.h. This is both fast and cool: you can construct it from an int, and cast it to a strong. It also does type checking: a Symbol<Pose3,'x'> will not match a Symbol<Pose2,'x'>
4) minor: take a look at LieConfig.h: it help you avoid writing a lot of code bu automatically creating configs for a certain type. See e.g. Pose3Config.h. A "double" LieConfig is on the way - Thanks Richard and Manohar !
											
										 
											2010-01-14 06:25:03 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	boost::shared_ptr<const Graph> sharedReallyNonlinearFactorGraph() { | 
					
						
							|  |  |  | 		boost::shared_ptr<Graph> fg( | 
					
						
							|  |  |  | 				new Graph); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		Vector z = Vector_(2, 1.0, 0.0); | 
					
						
							|  |  |  | 		double sigma = 0.1; | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		boost::shared_ptr<smallOptimize::UnaryFactor> factor( | 
					
						
							|  |  |  | 				new smallOptimize::UnaryFactor(z, noiseModel::Isotropic::Sigma(2,sigma), 1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		fg->push_back(factor); | 
					
						
							|  |  |  | 		return fg; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-27 21:34:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	Graph createReallyNonlinearFactorGraph() { | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		return *sharedReallyNonlinearFactorGraph(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-10 05:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	pair<Graph, Config> createNonlinearSmoother(int T) { | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Create
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Graph nlfg; | 
					
						
							|  |  |  | 		Config poses; | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// prior on x1
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Point2 x1(1.0, 0.0); | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 		shared prior(new simulated2D::Prior(x1, sigma1_0, 1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		nlfg.push_back(prior); | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		poses.insert(simulated2D::PoseKey(1), x1); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		for (int t = 2; t <= T; t++) { | 
					
						
							|  |  |  | 			// odometry between x_t and x_{t-1}
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 			Point2 odo(1.0, 0.0); | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 			shared odometry(new simulated2D::Odometry(odo, sigma1_0, t - 1, t)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 			nlfg.push_back(odometry); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// measurement on x_t is like perfect GPS
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 			Point2 xt(t, 0); | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 			shared measurement(new simulated2D::Prior(xt, sigma1_0, t)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 			nlfg.push_back(measurement); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// initial estimate
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 			poses.insert(simulated2D::PoseKey(t), xt); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return make_pair(nlfg, poses); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-10 05:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	GaussianFactorGraph createSmoother(int T) { | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Graph nlfg; | 
					
						
							|  |  |  | 		Config poses; | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		boost::tie(nlfg, poses) = createNonlinearSmoother(T); | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-22 05:17:47 +08:00
										 |  |  | 		return *nlfg.linearize(poses); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-10 05:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	GaussianFactorGraph createSimpleConstraintGraph() { | 
					
						
							|  |  |  | 		// create unary factor
 | 
					
						
							|  |  |  | 		// prior on "x", mean = [1,-1], sigma=0.1
 | 
					
						
							|  |  |  | 		Matrix Ax = eye(2); | 
					
						
							|  |  |  | 		Vector b1(2); | 
					
						
							|  |  |  | 		b1(0) = 1.0; | 
					
						
							|  |  |  | 		b1(1) = -1.0; | 
					
						
							| 
									
										
										
										
											2010-01-21 02:32:48 +08:00
										 |  |  | 		GaussianFactor::shared_ptr f1(new GaussianFactor("x", Ax, b1, sigma0_1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// create binary constraint factor
 | 
					
						
							|  |  |  | 		// between "x" and "y", that is going to be the only factor on "y"
 | 
					
						
							|  |  |  | 		// |1 0||x_1| + |-1  0||y_1| = |0|
 | 
					
						
							|  |  |  | 		// |0 1||x_2|   | 0 -1||y_2|   |0|
 | 
					
						
							|  |  |  | 		Matrix Ax1 = eye(2); | 
					
						
							|  |  |  | 		Matrix Ay1 = eye(2) * -1; | 
					
						
							|  |  |  | 		Vector b2 = Vector_(2, 0.0, 0.0); | 
					
						
							|  |  |  | 		GaussianFactor::shared_ptr f2(new GaussianFactor("x", Ax1, "y", Ay1, b2, | 
					
						
							| 
									
										
										
										
											2010-01-21 02:32:48 +08:00
										 |  |  | 				constraintModel)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// construct the graph
 | 
					
						
							|  |  |  | 		GaussianFactorGraph fg; | 
					
						
							|  |  |  | 		fg.push_back(f1); | 
					
						
							|  |  |  | 		fg.push_back(f2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return fg; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	VectorConfig createSimpleConstraintConfig() { | 
					
						
							|  |  |  | 		VectorConfig config; | 
					
						
							|  |  |  | 		Vector v = Vector_(2, 1.0, -1.0); | 
					
						
							|  |  |  | 		config.insert("x", v); | 
					
						
							|  |  |  | 		config.insert("y", v); | 
					
						
							|  |  |  | 		return config; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-10 05:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	GaussianFactorGraph createSingleConstraintGraph() { | 
					
						
							|  |  |  | 		// create unary factor
 | 
					
						
							|  |  |  | 		// prior on "x", mean = [1,-1], sigma=0.1
 | 
					
						
							|  |  |  | 		Matrix Ax = eye(2); | 
					
						
							|  |  |  | 		Vector b1(2); | 
					
						
							|  |  |  | 		b1(0) = 1.0; | 
					
						
							|  |  |  | 		b1(1) = -1.0; | 
					
						
							| 
									
										
										
										
											2010-01-27 12:39:35 +08:00
										 |  |  | 		//GaussianFactor::shared_ptr f1(new GaussianFactor("x", sigma0_1->Whiten(Ax), sigma0_1->whiten(b1), sigma0_1));
 | 
					
						
							| 
									
										
										
										
											2010-01-21 02:32:48 +08:00
										 |  |  | 		GaussianFactor::shared_ptr f1(new GaussianFactor("x", Ax, b1, sigma0_1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// create binary constraint factor
 | 
					
						
							|  |  |  | 		// between "x" and "y", that is going to be the only factor on "y"
 | 
					
						
							|  |  |  | 		// |1 2||x_1| + |10 0||y_1| = |1|
 | 
					
						
							|  |  |  | 		// |2 1||x_2|   |0 10||y_2|   |2|
 | 
					
						
							|  |  |  | 		Matrix Ax1(2, 2); | 
					
						
							|  |  |  | 		Ax1(0, 0) = 1.0; | 
					
						
							|  |  |  | 		Ax1(0, 1) = 2.0; | 
					
						
							|  |  |  | 		Ax1(1, 0) = 2.0; | 
					
						
							|  |  |  | 		Ax1(1, 1) = 1.0; | 
					
						
							|  |  |  | 		Matrix Ay1 = eye(2) * 10; | 
					
						
							|  |  |  | 		Vector b2 = Vector_(2, 1.0, 2.0); | 
					
						
							|  |  |  | 		GaussianFactor::shared_ptr f2(new GaussianFactor("x", Ax1, "y", Ay1, b2, | 
					
						
							| 
									
										
										
										
											2010-01-21 02:32:48 +08:00
										 |  |  | 				constraintModel)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// construct the graph
 | 
					
						
							|  |  |  | 		GaussianFactorGraph fg; | 
					
						
							|  |  |  | 		fg.push_back(f1); | 
					
						
							|  |  |  | 		fg.push_back(f2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return fg; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	VectorConfig createSingleConstraintConfig() { | 
					
						
							|  |  |  | 		VectorConfig config; | 
					
						
							|  |  |  | 		config.insert("x", Vector_(2, 1.0, -1.0)); | 
					
						
							|  |  |  | 		config.insert("y", Vector_(2, 0.2, 0.1)); | 
					
						
							|  |  |  | 		return config; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	GaussianFactorGraph createMultiConstraintGraph() { | 
					
						
							|  |  |  | 		// unary factor 1
 | 
					
						
							|  |  |  | 		Matrix A = eye(2); | 
					
						
							|  |  |  | 		Vector b = Vector_(2, -2.0, 2.0); | 
					
						
							| 
									
										
										
										
											2010-01-21 02:32:48 +08:00
										 |  |  | 		GaussianFactor::shared_ptr lf1(new GaussianFactor("x", A, b, sigma0_1)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// constraint 1
 | 
					
						
							|  |  |  | 		Matrix A11(2, 2); | 
					
						
							|  |  |  | 		A11(0, 0) = 1.0; | 
					
						
							|  |  |  | 		A11(0, 1) = 2.0; | 
					
						
							|  |  |  | 		A11(1, 0) = 2.0; | 
					
						
							|  |  |  | 		A11(1, 1) = 1.0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Matrix A12(2, 2); | 
					
						
							|  |  |  | 		A12(0, 0) = 10.0; | 
					
						
							|  |  |  | 		A12(0, 1) = 0.0; | 
					
						
							|  |  |  | 		A12(1, 0) = 0.0; | 
					
						
							|  |  |  | 		A12(1, 1) = 10.0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Vector b1(2); | 
					
						
							|  |  |  | 		b1(0) = 1.0; | 
					
						
							|  |  |  | 		b1(1) = 2.0; | 
					
						
							|  |  |  | 		GaussianFactor::shared_ptr lc1(new GaussianFactor("x", A11, "y", A12, b1, | 
					
						
							| 
									
										
										
										
											2010-01-21 02:32:48 +08:00
										 |  |  | 				constraintModel)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// constraint 2
 | 
					
						
							|  |  |  | 		Matrix A21(2, 2); | 
					
						
							|  |  |  | 		A21(0, 0) = 3.0; | 
					
						
							|  |  |  | 		A21(0, 1) = 4.0; | 
					
						
							|  |  |  | 		A21(1, 0) = -1.0; | 
					
						
							|  |  |  | 		A21(1, 1) = -2.0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Matrix A22(2, 2); | 
					
						
							|  |  |  | 		A22(0, 0) = 1.0; | 
					
						
							|  |  |  | 		A22(0, 1) = 1.0; | 
					
						
							|  |  |  | 		A22(1, 0) = 1.0; | 
					
						
							|  |  |  | 		A22(1, 1) = 2.0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Vector b2(2); | 
					
						
							|  |  |  | 		b2(0) = 3.0; | 
					
						
							|  |  |  | 		b2(1) = 4.0; | 
					
						
							|  |  |  | 		GaussianFactor::shared_ptr lc2(new GaussianFactor("x", A21, "z", A22, b2, | 
					
						
							| 
									
										
										
										
											2010-01-21 02:32:48 +08:00
										 |  |  | 				constraintModel)); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// construct the graph
 | 
					
						
							|  |  |  | 		GaussianFactorGraph fg; | 
					
						
							|  |  |  | 		fg.push_back(lf1); | 
					
						
							|  |  |  | 		fg.push_back(lc1); | 
					
						
							|  |  |  | 		fg.push_back(lc2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return fg; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	VectorConfig createMultiConstraintConfig() { | 
					
						
							|  |  |  | 		VectorConfig config; | 
					
						
							|  |  |  | 		config.insert("x", Vector_(2, -2.0, 2.0)); | 
					
						
							|  |  |  | 		config.insert("y", Vector_(2, -0.1, 0.4)); | 
					
						
							|  |  |  | 		config.insert("z", Vector_(2, -4.0, 5.0)); | 
					
						
							|  |  |  | 		return config; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	//GaussianFactorGraph createConstrainedGaussianFactorGraph()
 | 
					
						
							|  |  |  | 	//{
 | 
					
						
							|  |  |  | 	//	GaussianFactorGraph graph;
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	// add an equality factor
 | 
					
						
							|  |  |  | 	//	Vector v1(2); v1(0)=1.;v1(1)=2.;
 | 
					
						
							|  |  |  | 	//	GaussianFactor::shared_ptr f1(new GaussianFactor(v1, "x0"));
 | 
					
						
							|  |  |  | 	//	graph.push_back_eq(f1);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	// add a normal linear factor
 | 
					
						
							|  |  |  | 	//	Matrix A21 = -1 * eye(2);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Matrix A22 = eye(2);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Vector b(2);
 | 
					
						
							|  |  |  | 	//	b(0) = 2 ; b(1) = 3;
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	double sigma = 0.1;
 | 
					
						
							|  |  |  | 	//	GaussianFactor::shared_ptr f2(new GaussianFactor("x0", A21/sigma,  "x1", A22/sigma, b/sigma));
 | 
					
						
							|  |  |  | 	//	graph.push_back(f2);
 | 
					
						
							|  |  |  | 	//	return graph;
 | 
					
						
							|  |  |  | 	//}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	//	ConstrainedNonlinearFactorGraph<NonlinearFactor<VectorConfig> , VectorConfig> createConstrainedNonlinearFactorGraph() {
 | 
					
						
							|  |  |  | 	//		ConstrainedNonlinearFactorGraph<NonlinearFactor<VectorConfig> , VectorConfig> graph;
 | 
					
						
							|  |  |  | 	//		VectorConfig c = createConstrainedConfig();
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//		// equality constraint for initial pose
 | 
					
						
							|  |  |  | 	//		GaussianFactor::shared_ptr f1(new GaussianFactor(c["x0"], "x0"));
 | 
					
						
							|  |  |  | 	//		graph.push_back_eq(f1);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//		// odometry between x0 and x1
 | 
					
						
							|  |  |  | 	//		double sigma = 0.1;
 | 
					
						
							|  |  |  | 	//		shared f2(new Simulated2DOdometry(c["x1"] - c["x0"], sigma, "x0", "x1"));
 | 
					
						
							|  |  |  | 	//		graph.push_back(f2); // TODO
 | 
					
						
							|  |  |  | 	//		return graph;
 | 
					
						
							|  |  |  | 	//	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	//VectorConfig createConstrainedConfig()
 | 
					
						
							|  |  |  | 	//{
 | 
					
						
							|  |  |  | 	//	VectorConfig config;
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Vector x0(2); x0(0)=1.0; x0(1)=2.0;
 | 
					
						
							|  |  |  | 	//	config.insert("x0", x0);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Vector x1(2); x1(0)=3.0; x1(1)=5.0;
 | 
					
						
							|  |  |  | 	//	config.insert("x1", x1);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	return config;
 | 
					
						
							|  |  |  | 	//}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	//VectorConfig createConstrainedLinConfig()
 | 
					
						
							|  |  |  | 	//{
 | 
					
						
							|  |  |  | 	//	VectorConfig config;
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Vector x0(2); x0(0)=1.0; x0(1)=2.0; // value doesn't actually matter
 | 
					
						
							|  |  |  | 	//	config.insert("x0", x0);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Vector x1(2); x1(0)=2.3; x1(1)=5.3;
 | 
					
						
							|  |  |  | 	//	config.insert("x1", x1);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	return config;
 | 
					
						
							|  |  |  | 	//}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	//VectorConfig createConstrainedCorrectDelta()
 | 
					
						
							|  |  |  | 	//{
 | 
					
						
							|  |  |  | 	//	VectorConfig config;
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Vector x0(2); x0(0)=0.; x0(1)=0.;
 | 
					
						
							|  |  |  | 	//	config.insert("x0", x0);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Vector x1(2); x1(0)= 0.7; x1(1)= -0.3;
 | 
					
						
							|  |  |  | 	//	config.insert("x1", x1);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	return config;
 | 
					
						
							|  |  |  | 	//}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	//ConstrainedGaussianBayesNet createConstrainedGaussianBayesNet()
 | 
					
						
							|  |  |  | 	//{
 | 
					
						
							|  |  |  | 	//	ConstrainedGaussianBayesNet cbn;
 | 
					
						
							|  |  |  | 	//	VectorConfig c = createConstrainedConfig();
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	// add regular conditional gaussian - no parent
 | 
					
						
							|  |  |  | 	//	Matrix R = eye(2);
 | 
					
						
							|  |  |  | 	//	Vector d = c["x1"];
 | 
					
						
							|  |  |  | 	//	double sigma = 0.1;
 | 
					
						
							|  |  |  | 	//	GaussianConditional::shared_ptr f1(new GaussianConditional(d/sigma, R/sigma));
 | 
					
						
							|  |  |  | 	//	cbn.insert("x1", f1);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	// add a delta function to the cbn
 | 
					
						
							|  |  |  | 	//	ConstrainedGaussianConditional::shared_ptr f2(new ConstrainedGaussianConditional); //(c["x0"], "x0"));
 | 
					
						
							|  |  |  | 	//	cbn.insert_df("x0", f2);
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	return cbn;
 | 
					
						
							|  |  |  | 	//}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	// Create key for simulated planar graph
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 	simulated2D::PoseKey key(int x, int y) { | 
					
						
							|  |  |  | 		return simulated2D::PoseKey(1000*x+y); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	pair<GaussianFactorGraph, VectorConfig> planarGraph(size_t N) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// create empty graph
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		NonlinearFactorGraph<Config> nlfg; | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Create almost hard constraint on x11, sigma=0 will work for PCG not for normal
 | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 		shared constraint(new simulated2D::Prior(Point2(1.0, 1.0), sharedSigma(2,1e-3), key(1,1))); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		nlfg.push_back(constraint); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Create horizontal constraints, 1...N*(N-1)
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Point2 z1(1.0, 0.0); // move right
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		for (size_t x = 1; x < N; x++) | 
					
						
							|  |  |  | 			for (size_t y = 1; y <= N; y++) { | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 				shared f(new simulated2D::Odometry(z1, sharedSigma(2,0.01), key(x, y), key(x + 1, y))); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 				nlfg.push_back(f); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Create vertical constraints, N*(N-1)+1..2*N*(N-1)
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Point2 z2(0.0, 1.0); // move up
 | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		for (size_t x = 1; x <= N; x++) | 
					
						
							|  |  |  | 			for (size_t y = 1; y < N; y++) { | 
					
						
							| 
									
										
										
										
											2010-01-23 01:36:57 +08:00
										 |  |  | 				shared f(new simulated2D::Odometry(z2, sharedSigma(2,0.01), key(x, y), key(x, y + 1))); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 				nlfg.push_back(f); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Create linearization and ground xtrue config
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 		Config zeros; | 
					
						
							|  |  |  | 		VectorConfig xtrue; | 
					
						
							|  |  |  | 		Point2 zero; | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 		for (size_t x = 1; x <= N; x++) | 
					
						
							|  |  |  | 			for (size_t y = 1; y <= N; y++) { | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | 				zeros.insert(key(x, y), zero); | 
					
						
							| 
									
										
										
										
											2010-02-16 07:54:10 +08:00
										 |  |  | 				xtrue.insert((Symbol)key(x, y), Point2(x,y).vector()); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// linearize around zero
 | 
					
						
							| 
									
										
										
										
											2010-02-22 05:17:47 +08:00
										 |  |  | 		return make_pair(*nlfg.linearize(zeros), xtrue); | 
					
						
							| 
									
										
										
										
											2009-12-31 20:55:51 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	Ordering planarOrdering(size_t N) { | 
					
						
							|  |  |  | 		Ordering ordering; | 
					
						
							|  |  |  | 		for (size_t y = N; y >= 1; y--) | 
					
						
							|  |  |  | 			for (size_t x = N; x >= 1; x--) | 
					
						
							|  |  |  | 				ordering.push_back(key(x, y)); | 
					
						
							|  |  |  | 		return ordering; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	pair<GaussianFactorGraph, GaussianFactorGraph> splitOffPlanarTree(size_t N, | 
					
						
							|  |  |  | 			const GaussianFactorGraph& original) { | 
					
						
							|  |  |  | 		GaussianFactorGraph T, C; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Add the x11 constraint to the tree
 | 
					
						
							|  |  |  | 		T.push_back(original[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Add all horizontal constraints to the tree
 | 
					
						
							|  |  |  | 		size_t i = 1; | 
					
						
							|  |  |  | 		for (size_t x = 1; x < N; x++) | 
					
						
							|  |  |  | 			for (size_t y = 1; y <= N; y++, i++) | 
					
						
							|  |  |  | 				T.push_back(original[i]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Add first vertical column of constraints to T, others to C
 | 
					
						
							|  |  |  | 		for (size_t x = 1; x <= N; x++) | 
					
						
							|  |  |  | 			for (size_t y = 1; y < N; y++, i++) | 
					
						
							|  |  |  | 				if (x == 1) | 
					
						
							|  |  |  | 					T.push_back(original[i]); | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					C.push_back(original[i]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return make_pair(T, C); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-14 23:32:05 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 13:33:44 +08:00
										 |  |  | } // example
 | 
					
						
							| 
									
										
										
										
											2009-08-22 06:23:24 +08:00
										 |  |  | } // namespace gtsam
 |