| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * @file    ISAM2-inl.h | 
					
						
							| 
									
										
										
										
											2009-12-30 12:27:14 +08:00
										 |  |  |  * @brief   Incremental update functionality (ISAM2) for BayesTree, with fluid relinearization. | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  |  * @author  Michael Kaess | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <boost/foreach.hpp>
 | 
					
						
							|  |  |  | #include <boost/assign/std/list.hpp> // for operator +=
 | 
					
						
							|  |  |  | using namespace boost::assign; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | #include <set>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-16 09:16:59 +08:00
										 |  |  | #include "NonlinearFactorGraph-inl.h"
 | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | #include "GaussianFactor.h"
 | 
					
						
							|  |  |  | #include "VectorConfig.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Conditional.h"
 | 
					
						
							|  |  |  | #include "BayesTree-inl.h"
 | 
					
						
							|  |  |  | #include "ISAM2.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace gtsam { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	using namespace std; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 	// from inference-inl.h - need to additionally return the newly created factor for caching
 | 
					
						
							|  |  |  | 	boost::shared_ptr<GaussianConditional> _eliminateOne(FactorGraph<GaussianFactor>& graph, cachedFactors& cached, const string& key) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// combine the factors of all nodes connected to the variable to be eliminated
 | 
					
						
							|  |  |  | 		// if no factors are connected to key, returns an empty factor
 | 
					
						
							|  |  |  | 		boost::shared_ptr<GaussianFactor> joint_factor = removeAndCombineFactors(graph,key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// eliminate that joint factor
 | 
					
						
							|  |  |  | 		boost::shared_ptr<GaussianFactor> factor; | 
					
						
							|  |  |  | 		boost::shared_ptr<GaussianConditional> conditional; | 
					
						
							|  |  |  | 		boost::tie(conditional, factor) = joint_factor->eliminate(key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// remember the intermediate result to be able to later restart computation in the middle
 | 
					
						
							|  |  |  | 		cached[key] = factor; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// add new factor on separator back into the graph
 | 
					
						
							|  |  |  | 		if (!factor->empty()) graph.push_back(factor); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// return the conditional Gaussian
 | 
					
						
							|  |  |  | 		return conditional; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// from GaussianFactorGraph.cpp, see _eliminateOne above
 | 
					
						
							|  |  |  | 	GaussianBayesNet _eliminate(FactorGraph<GaussianFactor>& graph, cachedFactors& cached, const Ordering& ordering) { | 
					
						
							|  |  |  | 		GaussianBayesNet chordalBayesNet; // empty
 | 
					
						
							|  |  |  | 		BOOST_FOREACH(string key, ordering) { | 
					
						
							|  |  |  | 			GaussianConditional::shared_ptr cg = _eliminateOne(graph, cached, key); | 
					
						
							|  |  |  | 			chordalBayesNet.push_back(cg); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return chordalBayesNet; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GaussianBayesNet _eliminate_const(const FactorGraph<GaussianFactor>& graph, cachedFactors& cached, const Ordering& ordering) { | 
					
						
							|  |  |  | 		// make a copy that can be modified locally
 | 
					
						
							|  |  |  | 		FactorGraph<GaussianFactor> graph_ignored = graph; | 
					
						
							|  |  |  | 		return _eliminate(graph_ignored, cached, ordering); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 	/** Create an empty Bayes Tree */ | 
					
						
							|  |  |  | 	template<class Conditional, class Config> | 
					
						
							|  |  |  | 	ISAM2<Conditional, Config>::ISAM2() : BayesTree<Conditional>() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-30 12:27:14 +08:00
										 |  |  | 	/** Create a Bayes Tree from a nonlinear factor graph */ | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 	template<class Conditional, class Config> | 
					
						
							| 
									
										
										
										
											2009-12-31 13:35:08 +08:00
										 |  |  | 	ISAM2<Conditional, Config>::ISAM2(const NonlinearFactorGraph<Config>& nlfg, const Ordering& ordering, const Config& config) | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 	: BayesTree<Conditional>(nlfg.linearize(config).eliminate(ordering)), nonlinearFactors_(nlfg), config_(config) { | 
					
						
							|  |  |  | 		// todo: repeats calculation above, just to set "cached"
 | 
					
						
							|  |  |  | 		_eliminate_const(nlfg.linearize(config), cached, ordering); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* ************************************************************************* */ | 
					
						
							|  |  |  | 	template<class Conditional, class Config> | 
					
						
							| 
									
										
										
										
											2009-12-30 12:27:14 +08:00
										 |  |  | 	void ISAM2<Conditional, Config>::update_internal(const NonlinearFactorGraph<Config>& newFactors, | 
					
						
							|  |  |  | 			const Config& config, Cliques& orphans) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// copy variables into config_, but don't overwrite existing entries (current linearization point!)
 | 
					
						
							|  |  |  | 		for (typename Config::const_iterator it = config.begin(); it!=config.end(); it++) { | 
					
						
							|  |  |  | 			if (!config_.contains(it->first)) { | 
					
						
							|  |  |  | 				config_.insert(it->first, it->second); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-30 12:27:14 +08:00
										 |  |  | 		FactorGraph<GaussianFactor> newFactorsLinearized = newFactors.linearize(config_); | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Remove the contaminated part of the Bayes tree
 | 
					
						
							| 
									
										
										
										
											2009-12-30 12:27:14 +08:00
										 |  |  | 		FactorGraph<GaussianFactor> affectedFactors; | 
					
						
							|  |  |  | 		boost::tie(affectedFactors, orphans) = this->removeTop(newFactorsLinearized); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | #if 1
 | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2009-12-30 12:27:14 +08:00
										 |  |  | 		// find the corresponding original nonlinear factors, and relinearize them
 | 
					
						
							|  |  |  | 		NonlinearFactorGraph<Config> nonlinearAffectedFactors; | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | 		set<int> idxs; // avoid duplicates by putting index into set
 | 
					
						
							| 
									
										
										
										
											2009-12-31 13:35:08 +08:00
										 |  |  | 		BOOST_FOREACH(FactorGraph<GaussianFactor>::sharedFactor fac, affectedFactors) { | 
					
						
							|  |  |  | 			// retrieve correspondent factor from nonlinearFactors_
 | 
					
						
							|  |  |  | 			Ordering keys = fac->keys(); | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | 			BOOST_FOREACH(string key, keys) { | 
					
						
							|  |  |  | 				list<int> indices = nonlinearFactors_.factors(key); | 
					
						
							|  |  |  | 				BOOST_FOREACH(int idx, indices) { | 
					
						
							|  |  |  | 					// todo - only insert index if factor is subset of keys... not needed once we do relinearization - but then how to deal with overlap with orphans?
 | 
					
						
							|  |  |  | 					bool subset = true; | 
					
						
							|  |  |  | 					BOOST_FOREACH(string k, nonlinearFactors_[idx]->keys()) { | 
					
						
							|  |  |  | 						if (find(keys.begin(), keys.end(), k)==keys.end()) subset = false; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					if (subset) { | 
					
						
							|  |  |  | 						idxs.insert(idx); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-12-31 13:35:08 +08:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-12-30 12:27:14 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | 		BOOST_FOREACH(int idx, idxs) { | 
					
						
							|  |  |  | 			nonlinearAffectedFactors.push_back(nonlinearFactors_[idx]); | 
					
						
							| 
									
										
										
										
											2009-12-31 13:35:08 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		FactorGraph<GaussianFactor> factors = nonlinearAffectedFactors.linearize(config_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		NonlinearFactorGraph<Config> nonlinearAffectedFactors; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// retrieve all factors that ONLY contain the affected variables
 | 
					
						
							|  |  |  | 		// (note that the remaining stuff is summarized in the cached factors)
 | 
					
						
							|  |  |  | 		list<string> affectedKeys = affectedFactors.keys(); | 
					
						
							|  |  |  | 		typename FactorGraph<NonlinearFactor<Config> >::iterator it; | 
					
						
							|  |  |  | 		for(it = nonlinearFactors_.begin(); it != nonlinearFactors_.end(); it++) { | 
					
						
							|  |  |  | 			bool inside = true; | 
					
						
							|  |  |  | 			BOOST_FOREACH(string key, (*it)->keys()) { | 
					
						
							|  |  |  | 				if (find(affectedKeys.begin(), affectedKeys.end(), key) == affectedKeys.end()) | 
					
						
							|  |  |  | 					inside = false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (inside) | 
					
						
							|  |  |  | 				nonlinearAffectedFactors.push_back(*it); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | 		FactorGraph<GaussianFactor> factors = nonlinearAffectedFactors.linearize(config_); | 
					
						
							| 
									
										
										
										
											2009-12-31 13:35:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 		// recover intermediate factors from cache that are passed into the affected area
 | 
					
						
							|  |  |  | 		FactorGraph<GaussianFactor> cachedBoundary; | 
					
						
							|  |  |  | 		BOOST_FOREACH(sharedClique orphan, orphans) { | 
					
						
							|  |  |  | 			// find the last variable that is not part of the separator
 | 
					
						
							|  |  |  | 			string oneTooFar = orphan->separator_.front(); | 
					
						
							|  |  |  | 			list<string> keys = orphan->keys(); | 
					
						
							|  |  |  | 			list<string>::iterator it = find(keys.begin(), keys.end(), oneTooFar); | 
					
						
							|  |  |  | 			it--; | 
					
						
							|  |  |  | 			string key = *it; | 
					
						
							|  |  |  | 			// retrieve the cached factor and add to boundary
 | 
					
						
							|  |  |  | 			cachedBoundary.push_back(cached[key]); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		factors.push_back(cachedBoundary); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 		printf("**************\n"); | 
					
						
							|  |  |  | 		nonlinearFactors_.linearize(config).print("all factors"); | 
					
						
							|  |  |  | 		printf("--------------\n"); | 
					
						
							|  |  |  | 		newFactorsLinearized.print("newFactorsLinearized"); | 
					
						
							|  |  |  | 		printf("--------------\n"); | 
					
						
							|  |  |  | 		factors.print("factors"); | 
					
						
							|  |  |  | 		printf("--------------\n"); | 
					
						
							|  |  |  | 		affectedFactors.print("affectedFactors"); | 
					
						
							|  |  |  | 		printf("--------------\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-30 12:27:14 +08:00
										 |  |  | 		// add the new factors themselves
 | 
					
						
							|  |  |  | 		factors.push_back(newFactorsLinearized); | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 		// create an ordering for the new and contaminated factors
 | 
					
						
							|  |  |  | 		Ordering ordering; | 
					
						
							|  |  |  | 		if (true) { | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 			ordering = factors.getOrdering(); | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 			list<string> keys = factors.keys(); | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 			keys.sort(); // todo: correct sorting order?
 | 
					
						
							|  |  |  | 			ordering = keys; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | 		ordering.print(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		factors.print("factors BEFORE"); | 
					
						
							|  |  |  | 		printf("--------------\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 		// eliminate into a Bayes net
 | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 		BayesNet<Conditional> bayesNet = _eliminate(factors, cached, ordering); | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if 1
 | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 		// check if relinearized agrees with correct solution
 | 
					
						
							|  |  |  | 		affectedFactors.push_back(newFactorsLinearized); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 		affectedFactors.print("affectedFactors BEFORE"); | 
					
						
							|  |  |  | 		printf("--------------\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BayesNet<Conditional> bayesNetTest = eliminate<GaussianFactor, GaussianConditional>(affectedFactors, ordering); | 
					
						
							| 
									
										
										
										
											2010-01-03 12:57:35 +08:00
										 |  |  | 		if (!bayesNet.equals(bayesNetTest)) { | 
					
						
							|  |  |  | 			printf("differ\n"); | 
					
						
							|  |  |  | 			bayesNet.print(); | 
					
						
							|  |  |  | 			bayesNetTest.print(); | 
					
						
							|  |  |  | 			exit(42); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 		nonlinearFactors_.push_back(newFactors); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 		// insert conditionals back in, straight into the topless bayesTree
 | 
					
						
							|  |  |  | 		typename BayesNet<Conditional>::const_reverse_iterator rit; | 
					
						
							|  |  |  | 		for ( rit=bayesNet.rbegin(); rit != bayesNet.rend(); ++rit ) | 
					
						
							|  |  |  | 			this->insert(*rit); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		int count = 0; | 
					
						
							|  |  |  | 		// add orphans to the bottom of the new tree
 | 
					
						
							|  |  |  | 		BOOST_FOREACH(sharedClique orphan, orphans) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			string key = orphan->separator_.front(); | 
					
						
							|  |  |  | 			sharedClique parent = (*this)[key]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			parent->children_ += orphan; | 
					
						
							|  |  |  | 			orphan->parent_ = parent; // set new parent!
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	template<class Conditional, class Config> | 
					
						
							| 
									
										
										
										
											2009-12-29 22:54:45 +08:00
										 |  |  | 	void ISAM2<Conditional, Config>::update(const NonlinearFactorGraph<Config>& newFactors, const Config& config) { | 
					
						
							| 
									
										
										
										
											2010-01-17 14:06:20 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 		printf("8888888888888888\n"); | 
					
						
							|  |  |  | 		try { | 
					
						
							|  |  |  | 		this->print("BayesTree"); | 
					
						
							|  |  |  | 		} catch (char * c) {}; | 
					
						
							|  |  |  | 		printf("8888888888888888\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 		Cliques orphans; | 
					
						
							| 
									
										
										
										
											2009-12-29 22:54:45 +08:00
										 |  |  | 		this->update_internal(newFactors, config, orphans); | 
					
						
							| 
									
										
										
										
											2009-12-29 13:57:05 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************************************************************************* */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | /// namespace gtsam
 |