| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * @file ConstrainedConditionalGaussian.cpp | 
					
						
							|  |  |  |  * @brief Implements the constrained version of the conditional gaussian class, | 
					
						
							|  |  |  |  * primarily handling the possible solutions | 
					
						
							|  |  |  |  * @author Alex Cunningham | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <boost/numeric/ublas/lu.hpp>
 | 
					
						
							|  |  |  | #include "ConstrainedConditionalGaussian.h"
 | 
					
						
							|  |  |  | #include "Matrix.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using namespace gtsam; | 
					
						
							|  |  |  | using namespace std; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-02 11:50:30 +08:00
										 |  |  | ConstrainedConditionalGaussian::ConstrainedConditionalGaussian( | 
					
						
							|  |  |  | 		const string& key) : | 
					
						
							|  |  |  | 	ConditionalGaussian(key) { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-02 11:50:30 +08:00
										 |  |  | ConstrainedConditionalGaussian::ConstrainedConditionalGaussian( | 
					
						
							|  |  |  | 		const string& key, const Vector& v) : | 
					
						
							| 
									
										
										
										
											2009-11-05 04:59:16 +08:00
										 |  |  | 	ConditionalGaussian(key, v, eye(v.size()), ones(v.size())*INFINITY) { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-02 11:50:30 +08:00
										 |  |  | ConstrainedConditionalGaussian::ConstrainedConditionalGaussian( | 
					
						
							|  |  |  | 		const string& key, const Vector& b, const Matrix& A) : | 
					
						
							| 
									
										
										
										
											2009-11-05 04:59:16 +08:00
										 |  |  | 	ConditionalGaussian(key, b, A, ones(b.size())*INFINITY) { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-02 11:50:30 +08:00
										 |  |  | ConstrainedConditionalGaussian::ConstrainedConditionalGaussian( | 
					
						
							|  |  |  | 		const string& key, const Vector& b, const Matrix& A1, | 
					
						
							|  |  |  | 		const std::string& parent, const Matrix& A2) : | 
					
						
							| 
									
										
										
										
											2009-11-05 04:59:16 +08:00
										 |  |  | 	ConditionalGaussian(key, b, A1, parent, A2, ones(b.size())*INFINITY) { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-05 04:59:16 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-02 11:50:30 +08:00
										 |  |  | ConstrainedConditionalGaussian::ConstrainedConditionalGaussian( | 
					
						
							|  |  |  | 		const string& key, const Vector& b, const Matrix& A1, | 
					
						
							|  |  |  | 		const std::string& parentY, const Matrix& A2, const std::string& parentZ, | 
					
						
							|  |  |  | 		const Matrix& A3) : | 
					
						
							| 
									
										
										
										
											2009-11-05 04:59:16 +08:00
										 |  |  | 	ConditionalGaussian(key, b, A1, parentY, A2, parentZ, A3, ones(b.size())*INFINITY) { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ConstrainedConditionalGaussian::ConstrainedConditionalGaussian( | 
					
						
							| 
									
										
										
										
											2009-11-02 11:50:30 +08:00
										 |  |  | 		const string& key, const Matrix& A1, | 
					
						
							|  |  |  | 		const std::map<std::string, Matrix>& parents, const Vector& b) : | 
					
						
							| 
									
										
										
										
											2009-11-05 04:59:16 +08:00
										 |  |  | 	ConditionalGaussian(key, b, A1, parents, ones(b.size())*INFINITY) { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-15 04:39:59 +08:00
										 |  |  | Vector ConstrainedConditionalGaussian::solve(const VectorConfig& x) const { | 
					
						
							| 
									
										
										
										
											2009-10-08 21:57:22 +08:00
										 |  |  | 	// sum the RHS
 | 
					
						
							|  |  |  | 	Vector rhs = d_; | 
					
						
							|  |  |  | 	for (map<string, Matrix>::const_iterator it = parents_.begin(); it | 
					
						
							|  |  |  | 			!= parents_.end(); it++) { | 
					
						
							|  |  |  | 		const string& j = it->first; | 
					
						
							|  |  |  | 		const Matrix& Aj = it->second; | 
					
						
							|  |  |  | 		rhs -= Aj * x[j]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// verify invertibility of A matrix
 | 
					
						
							|  |  |  | 	Matrix A = R_; | 
					
						
							|  |  |  | 	Matrix b = Matrix_(rhs.size(), 1, rhs); | 
					
						
							|  |  |  | 	if (A.size1() != A.size2()) | 
					
						
							|  |  |  | 		throw invalid_argument("Matrix A is not invertible - non-square matrix"); | 
					
						
							|  |  |  | 	using namespace boost::numeric::ublas; | 
					
						
							|  |  |  | 	if (lu_factorize(A)) | 
					
						
							|  |  |  | 		throw invalid_argument("Matrix A is singular"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// get the actual solution
 | 
					
						
							|  |  |  | 	//FIXME: This is just the Matrix::solve() function, but due to name conflicts
 | 
					
						
							|  |  |  | 	// the compiler won't find the real version in Matrix.h
 | 
					
						
							|  |  |  | 	lu_substitute<const Matrix, Matrix> (A, b); | 
					
						
							|  |  |  | 	return Vector_(b); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//TODO: Handle overdetermined case
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//TODO: Handle underdetermined case
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |