solver
RtMatrix.hpp
Go to the documentation of this file.
1 
10 #pragma once
11 
12 #include <Eigen/Dense>
13 
14 namespace rt_solver {
15 
16  // Class for creating a dynamic matrix with a pre-allocated maximum size
17  template<int _MaxRows, int _MaxCols>
18  struct RtMatrix
19  {
20  #ifdef RTEIG_DYN_ALLOCS
21  typedef Eigen::MatrixXd d;
22  #else
23  static const int min_possible_elements_ = 2;
24  typedef Eigen::Matrix<double, Eigen::Dynamic,
25  Eigen::Dynamic, Eigen::AutoAlign,
26  (_MaxRows>=min_possible_elements_)?_MaxRows:min_possible_elements_,
27  (_MaxCols>=min_possible_elements_)?_MaxCols:min_possible_elements_> d;
28  typedef Eigen::Matrix<float, Eigen::Dynamic,
29  Eigen::Dynamic, Eigen::AutoAlign,
30  (_MaxRows>=min_possible_elements_)?_MaxRows:min_possible_elements_,
31  (_MaxCols>=min_possible_elements_)?_MaxCols:min_possible_elements_> f;
32  typedef Eigen::Matrix<int, Eigen::Dynamic,
33  Eigen::Dynamic, Eigen::AutoAlign,
34  (_MaxRows>=min_possible_elements_)?_MaxRows:min_possible_elements_,
35  (_MaxCols>=min_possible_elements_)?_MaxCols:min_possible_elements_> i;
36  #endif
37  };
38 
39  template<int _MaxRows, int _MaxCols>
41 
42  // Class for creating a dynamic vector with a pre-allocated maximum size
43  template<int _MaxRows>
44  struct RtVector
45  {
46  #ifdef RTEIG_DYN_ALLOCS
47  typedef Eigen::VectorXd d;
48  #else
49  static const int min_possible_elements_ = 2;
50  typedef Eigen::Matrix<double, Eigen::Dynamic, 1, Eigen::AutoAlign,
51  (_MaxRows>=min_possible_elements_)?_MaxRows:min_possible_elements_, 1> d;
52  typedef Eigen::Matrix<float, Eigen::Dynamic, 1, Eigen::AutoAlign,
53  (_MaxRows>=min_possible_elements_)?_MaxRows:min_possible_elements_, 1> f;
54  typedef Eigen::Matrix<int, Eigen::Dynamic, 1, Eigen::AutoAlign,
55  (_MaxRows>=min_possible_elements_)?_MaxRows:min_possible_elements_, 1> i;
56  typedef Eigen::Matrix<bool, Eigen::Dynamic, 1, Eigen::AutoAlign,
57  (_MaxRows>=min_possible_elements_)?_MaxRows:min_possible_elements_, 1> b;
58  #endif
59  };
60 
61  template<int _MaxRows>
63 
64  // Class with utilities for Rt elements
66  {
67  private:
68  RtMatrixUtils(){}
69  virtual ~RtMatrixUtils(){}
70 
71  public:
72 
73  // creates a selector matrix based on entries of a selector vector which are more than specified threshold
74  template<int MaxRows, int MaxCols>
75  static inline void createSelectorMat(typename RtMatrix<MaxRows, MaxCols>::d& selector_mat,
76  const typename RtVector<MaxRows>::d& row_selector, int num_valids, double threshold)
77  {
78  RtMatrixUtils::setZero(selector_mat, num_valids, row_selector.size());
79  int row = 0;
80  for (int col=0; col<row_selector.size(); col++)
81  if (row_selector[col] > threshold)
82  selector_mat(row++,col) = 1.0;
83  }
84 
85  // pruning rows of matrix which are close to zero (squared norm row < threshold)
86  template<int MaxRows, int MaxCols>
87  static inline void removeZeroRows(typename RtMatrix<MaxRows, MaxCols>::d& mat, double threshold = remove_zero_thresh_)
88  {
89  if(mat.rows() == 0)
90  return;
91 
92  typename RtVector<MaxRows>::d squared_norms;
93  squared_norms = mat.rowwise().squaredNorm();
94  const double normed_threash = squared_norms.size() * threshold * threshold;
95  int num_valids = 0;
96  for (int row_id=1; row_id<squared_norms.size(); row_id++)
97  if(squared_norms[row_id] > normed_threash)
98  ++num_valids;
99 
100  if(num_valids == mat.rows()) { return; }
101  else if (num_valids == 0) { RtMatrixUtils::resize(mat, 0, mat.cols()); }
102  else
103  {
104  typename RtMatrix<MaxRows, MaxCols>::d selector_mat;
105  createSelectorMat(selector_mat, squared_norms, num_valids, normed_threash); // create selector matrix to prune zero rows
106  mat = selector_mat * mat;
107  }
108  }
109 
117  template <typename Derived>
118  static inline void conservativeResize(Eigen::MatrixBase<Derived>& mat, int rows, int cols)
119  {
120  #ifndef RTEIG_NO_ASSERTS
121  assert(rows <= mat.MaxRowsAtCompileTime && cols <= mat.MaxColsAtCompileTime);
122  #endif
123  mat.derived().conservativeResize(rows, cols);
124  }
125 
132  template <typename Derived>
133  static inline void resize(Eigen::MatrixBase<Derived>& mat, int rows, int cols)
134  {
135  #ifndef RTEIG_NO_ASSERTS
136  assert(rows <= mat.MaxRowsAtCompileTime && cols <= mat.MaxColsAtCompileTime);
137  #endif
138  mat.derived().resize(rows, cols);
139  }
140 
141  template <typename Derived>
142  static inline void setZero(Eigen::MatrixBase<Derived>& mat)
143  {
144  #ifdef RTEIG_DYN_ALLOCS
145  resize(mat,1, 1);
146  #else
147  resize(mat, mat.MaxRowsAtCompileTime, mat.MaxColsAtCompileTime);
148  #endif
149  mat.derived().setZero();
150  }
151 
152  template <typename Derived>
153  static inline void setZero(Eigen::MatrixBase<Derived>& mat, int rows, int cols)
154  {
155  resize(mat, rows, cols);
156  mat.derived().setZero();
157  }
158 
159  template <typename Derived>
160  static inline void setOnes(Eigen::MatrixBase<Derived>& mat)
161  {
162  #ifdef RTEIG_DYN_ALLOCS
163  resize(mat,1, 1);
164  #else
165  resize(mat, mat.MaxRowsAtCompileTime, mat.MaxColsAtCompileTime);
166  #endif
167  mat.derived().setOnes();
168  }
169 
170  template <typename Derived>
171  static inline void setOnes(Eigen::MatrixBase<Derived>& mat, int rows, int cols)
172  {
173  resize(mat, rows, cols);
174  mat.derived().setOnes();
175  }
176 
177  template <typename Derived>
178  static inline void setConstant(Eigen::MatrixBase<Derived>& mat, double scalar)
179  {
180  #ifdef RTEIG_DYN_ALLOCS
181  resize(mat,1, 1);
182  #else
183  resize(mat, mat.MaxRowsAtCompileTime, mat.MaxColsAtCompileTime);
184  #endif
185  mat.derived().setConstant(scalar);
186  }
187 
188  template <typename Derived>
189  static inline void setConstant(Eigen::MatrixBase<Derived>& mat, int rows, int cols, double scalar)
190  {
191  resize(mat, rows, cols);
192  mat.derived().setConstant(scalar);
193  }
194 
195  template <typename Derived>
196  static inline void setIdentity(Eigen::MatrixBase<Derived>& mat)
197  {
198  #ifndef RTEIG_NO_ASSERTS
199  assert(mat.MaxRowsAtCompileTime == mat.MaxColsAtCompileTime);
200  #endif
201  #ifdef RTEIG_DYN_ALLOCS
202  resize(mat,1, 1);
203  #else
204  resize(mat, mat.MaxRowsAtCompileTime, mat.MaxRowsAtCompileTime);
205  #endif
206  mat.derived().setIdentity();
207  }
208 
209  template <typename Derived>
210  static inline void setIdentity(Eigen::MatrixBase<Derived>& mat, int rows)
211  {
212  resize(mat, rows, rows);
213  mat.derived().setIdentity();
214  }
215 
216  // Append a matrix to another one
217  template <typename AffMatDerived, typename SubstMatDerived>
218  static inline void append(Eigen::MatrixBase<AffMatDerived>& aff_mat, const Eigen::MatrixBase<SubstMatDerived>& subst_mat)
219  {
220  #ifndef RTEIG_NO_ASSERTS
221  assert(subst_mat.cols() == aff_mat.cols() && aff_mat.rows() + subst_mat.rows() <= aff_mat.MaxRowsAtCompileTime);
222  #endif
223  const int last_row = aff_mat.rows();
224  if (last_row == 0) { aff_mat = subst_mat; }
225  else {
226  RtMatrixUtils::conservativeResize(aff_mat, aff_mat.rows() + subst_mat.rows(), aff_mat.cols());
227  aff_mat.block(last_row, 0, subst_mat.rows(), subst_mat.cols()) = subst_mat;
228  }
229  }
230 
235  template <typename AffMatDerived, typename SubstMatDerived>
236  static inline void appendAtColumn(Eigen::MatrixBase<AffMatDerived>& aff_mat, const Eigen::MatrixBase<SubstMatDerived>& subst_mat, int starting_column)
237  {
238  #ifndef RTEIG_NO_ASSERTS
239  assert(subst_mat.cols() + starting_column <= aff_mat.cols() && aff_mat.rows() + subst_mat.rows() <= aff_mat.MaxRowsAtCompileTime);
240  #endif
241  const int last_row = aff_mat.rows();
242  if (last_row == 0) { RtMatrixUtils::resize(aff_mat, aff_mat.rows() + subst_mat.rows(), aff_mat.cols()); }
243  else { RtMatrixUtils::conservativeResize(aff_mat, aff_mat.rows() + subst_mat.rows(), aff_mat.cols()); }
244  aff_mat.block(last_row, 0, subst_mat.rows(), aff_mat.cols()).setZero();
245  aff_mat.block(last_row, starting_column, subst_mat.rows(), subst_mat.cols()) = subst_mat;
246  }
247 
248  template<typename sigmaPinvDer, typename svdMatDer>
249  static void computeSigmaPinv(Eigen::MatrixBase<sigmaPinvDer>& sigma_pinv_vec, const Eigen::JacobiSVD<svdMatDer>& mat_svd);
250 
251  template<int _mat_rows, int _mat_cols, int _mat_align, int max_mat_rows, int max_mat_cols, typename nullDer>
252  static int computeNullspaceMap(
253  const Eigen::Matrix<double, _mat_rows, _mat_cols, _mat_align, max_mat_rows, max_mat_cols>& mat, Eigen::MatrixBase<nullDer>& null_map,
254  const Eigen::JacobiSVD<typename RtMatrix<max_mat_rows, max_mat_cols>::d >& mat_svd, const double mat_cond_threas = mat_condition_thresh_);
255 
256  template<typename Mat>
257  static inline double conditionNumber(const Mat& mat)
258  {
259  Eigen::JacobiSVD<Mat> mat_svd;
260  mat_svd.compute(mat);
261  return conditionNumberFromSvd(mat_svd);
262  }
263 
264  template<typename Mat, int QRPreconditioner>
265  static inline double conditionNumberFromSvd(const Eigen::JacobiSVD<Mat, QRPreconditioner>& mat_svd)
266  { return mat_svd.singularValues()[0]/mat_svd.singularValues()[mat_svd.singularValues().size()-1]; }
267 
268  static const double remove_zero_thresh_;
269  static const double mat_condition_thresh_;
270 
271  };
272 
273  template<typename sigmaPinvDer, typename svdMatDer>
274  void RtMatrixUtils::computeSigmaPinv(Eigen::MatrixBase<sigmaPinvDer>& sigma_pinv_vec, const Eigen::JacobiSVD<svdMatDer>& mat_svd)
275  {
276  #ifndef RTEIG_NO_ASSERTS
277  assert(sigma_pinv_vec.rows() == mat_svd.singularValues().size() && sigma_pinv_vec.cols() == 1 && "sigma_pinv_vec wrongly sized");
278  #endif
279 
280  const double min_sing_val = mat_svd.singularValues()[0]/mat_condition_thresh_;
281  int id;
282  for (id=0; id<mat_svd.singularValues().size(); id++)
283  if (mat_svd.singularValues()[id]>min_sing_val) { sigma_pinv_vec[id] = 1.0/mat_svd.singularValues()[id]; }
284  else { sigma_pinv_vec[id] = 0.0; break; }
285  sigma_pinv_vec.segment(id+1, sigma_pinv_vec.size()-id);
286  }
287 
288  template<int _mat_rows, int _mat_cols, int _mat_align, int max_mat_rows, int max_mat_cols, typename nullDer>
289  int RtMatrixUtils::computeNullspaceMap(
290  const Eigen::Matrix<double, _mat_rows, _mat_cols, _mat_align, max_mat_rows, max_mat_cols>& mat, Eigen::MatrixBase<nullDer>& null_map,
291  const Eigen::JacobiSVD<typename RtMatrix<max_mat_rows, max_mat_cols>::d >& mat_svd, const double mat_cond_threas)
292  {
293  int dim_range = 0;
294 
295  const double first_sv = mat_svd.singularValues()[0];
296  for(dim_range=0; dim_range<mat_svd.singularValues().size(); ++dim_range)
297  if (mat_svd.singularValues()[dim_range] <= 0.0 || first_sv / mat_svd.singularValues()[dim_range] > mat_cond_threas)
298  break;
299 
300  const int dim_null = mat.cols() - dim_range;
301  RtMatrixUtils::resize(null_map, mat.cols(), dim_null); //asserts that null_map has enough space
302 
303  if (null_map.cols() > 0)
304  null_map = mat_svd.matrixV().rightCols(dim_null);
305 
306  #ifndef RTEIG_NO_ASSERTS
307  assert((mat * null_map).norm() < 0.0001 && "Nullspace map is not correctly computed");
308  typename RtMatrix<max_mat_cols, max_mat_cols>::d kernel_id;
309  RtMatrixUtils::setIdentity(kernel_id, dim_null);
310  assert((null_map.transpose() * null_map - kernel_id).norm() < 0.0001 && "Nullspace map is not correctly computed");
311  #endif
312  return dim_range;
313  }
314 
315 
317  {
318  private:
319  RtVectorUtils(){}
320  virtual ~RtVectorUtils(){}
321 
322  public:
323  template <typename Derived>
324  static inline void conservativeResize(Eigen::MatrixBase<Derived>& vec, int rows)
325  {
326  #ifndef RTEIG_NO_ASSERTS
327  assert(rows <= vec.MaxSizeAtCompileTime);
328  #endif
329  vec.derived().conservativeResize(rows);
330  }
331 
332  template <typename Derived>
333  static inline void resize(Eigen::MatrixBase<Derived>& vec, int rows)
334  {
335  #ifndef RTEIG_NO_ASSERTS
336  assert(rows <= vec.MaxSizeAtCompileTime);
337  #endif
338  vec.derived().resize(rows);
339  }
340 
341  template <typename Derived>
342  static inline void setZero(Eigen::MatrixBase<Derived>& vec)
343  {
344  #ifdef RTEIG_DYN_ALLOCS
345  resize(vec, 1);
346  #else
347  resize(vec, vec.MaxSizeAtCompileTime);
348  #endif
349  vec.derived().setZero();
350  }
351 
352  template <typename Derived>
353  static inline void setZero(Eigen::MatrixBase<Derived>& vec, int size)
354  {
355  resize(vec, size);
356  vec.derived().setZero();
357  }
358 
359  template <typename Derived>
360  static inline void setOnes(Eigen::MatrixBase<Derived>& vec)
361  {
362  #ifdef RTEIG_DYN_ALLOCS
363  resize(vec, 1);
364  #else
365  resize(vec, vec.MaxSizeAtCompileTime);
366  #endif
367  vec.derived().setOnes();
368  }
369 
370  template <typename Derived>
371  static inline void setOnes(Eigen::MatrixBase<Derived>& vec, int size)
372  {
373  resize(vec, size);
374  vec.derived().setOnes();
375  }
376 
377  template <typename Derived>
378  static inline void setConstant(Eigen::MatrixBase<Derived>& vec, double scalar)
379  {
380  #ifdef RTEIG_DYN_ALLOCS
381  resize(vec, 1);
382  #else
383  resize(vec, vec.MaxSizeAtCompileTime);
384  #endif
385  vec.derived().setConstant(scalar);
386  }
387 
388  template <typename Derived>
389  static inline void setConstant(Eigen::MatrixBase<Derived>& vec, int size, double scalar)
390  {
391  resize(vec, size);
392  vec.derived().setConstant(scalar);
393  }
394 
395  // Append a vector to another one
396  template <typename AffVecDerived, typename SubstVecDerived>
397  static inline void append(Eigen::MatrixBase<AffVecDerived>& aff_vec, const Eigen::MatrixBase<SubstVecDerived>& subst_vec)
398  {
399  #ifndef RTEIG_NO_ASSERTS
400  assert(aff_vec.rows() + subst_vec.rows() <= aff_vec.MaxSizeAtCompileTime);
401  #endif
402  const int last_row = aff_vec.rows();
403  RtVectorUtils::conservativeResize(aff_vec, aff_vec.rows() + subst_vec.rows());
404  aff_vec.block(last_row, 0, subst_vec.rows(), 1) = subst_vec;
405  }
406  };
407 
409  {
410  private:
411  RtAffineUtils(){}
412  virtual ~ RtAffineUtils(){}
413 
414  public:
415 
416  // Append an affine mapping at the end of another one
417  template <typename AffMatDerived, typename AffVecDerived, typename SubstMatDerived>
418  static inline void append(Eigen::MatrixBase<AffMatDerived>& aff_mat,
419  Eigen::MatrixBase<AffVecDerived>& aff_vec, const Eigen::MatrixBase<SubstMatDerived>& subst_mat)
420  {
421  #ifndef RTEIG_NO_ASSERTS
422  assertMatchingDimensions(aff_mat, aff_vec);
423  assert(aff_vec.rows() + subst_mat.rows() <= aff_vec.MaxSizeAtCompileTime);
424  #endif
425  const int last_row = aff_vec.rows();
426  RtVectorUtils::conservativeResize(aff_vec, aff_vec.rows() + subst_mat.rows());
427  RtMatrixUtils::append(aff_mat, subst_mat);
428  aff_vec.block(last_row, 0, subst_mat.rows(),1).setZero();
429  }
430 
431  template <typename AffMatDerived, typename AffVecDerived, typename SubstMatDerived, typename SubstVecDerived>
432  static inline void append(Eigen::MatrixBase<AffMatDerived>& aff_mat, Eigen::MatrixBase<AffVecDerived>& aff_vec,
433  const Eigen::MatrixBase<SubstMatDerived>& subst_mat, const Eigen::MatrixBase<SubstVecDerived>& subst_vec)
434  {
435  #ifndef RTEIG_NO_ASSERTS
436  assertMatchingDimensions(aff_mat, aff_vec);
437  assertMatchingDimensions(subst_mat, subst_vec);
438  #endif
439 
440  const int last_row = aff_vec.rows();
441  append(aff_mat, aff_vec, subst_mat);
442  aff_vec.block(last_row, 0, subst_mat.rows(),1) = subst_vec;
443  }
444 
449  template <typename AffMatDerived, typename AffVecDerived, typename SubstMatDerived>
450  static inline void appendAtColumn(Eigen::MatrixBase<AffMatDerived>& aff_mat, Eigen::MatrixBase<AffVecDerived>& aff_vec,
451  const Eigen::MatrixBase<SubstMatDerived>& subst_mat, int starting_column)
452  {
453  #ifndef RTEIG_NO_ASSERTS
454  assertMatchingDimensions(aff_mat, aff_vec);
455  assert(aff_vec.rows() + subst_mat.rows() <= aff_vec.MaxSizeAtCompileTime);
456  #endif
457  const int last_row = aff_vec.rows();
458  RtVectorUtils::conservativeResize(aff_vec, aff_vec.rows() + subst_mat.rows());
459  RtMatrixUtils::appendAtColumn(aff_mat, subst_mat, starting_column);
460  aff_vec.block(last_row, 0, subst_mat.rows(),1).setZero();
461  }
462 
463  template <typename AffMatDerived, typename AffVecDerived, typename SubstMatDerived, typename SubstVecDerived>
464  static inline void appendAtColumn(Eigen::MatrixBase<AffMatDerived>& aff_mat, Eigen::MatrixBase<AffVecDerived>& aff_vec,
465  const Eigen::MatrixBase<SubstMatDerived>& subst_mat, const Eigen::MatrixBase<SubstVecDerived>& subst_vec, int starting_column)
466  {
467  #ifndef RTEIG_NO_ASSERTS
468  assertMatchingDimensions(aff_mat, aff_vec);
469  assertMatchingDimensions(subst_mat, subst_vec);
470  #endif
471 
472  const int last_row = aff_vec.rows();
473  appendAtColumn(aff_mat, aff_vec, subst_mat, starting_column);
474  aff_vec.block(last_row, 0, subst_mat.rows(),1) = subst_vec;
475  }
476 
477  template<int MaxRows, int MaxCols>
478  static inline void removeZeroRows(typename RtMatrix<MaxRows, MaxCols>::d& mat,
479  typename RtVector<MaxRows>::d& vec, double threshold = RtMatrixUtils::remove_zero_thresh_)
480  {
481  if (mat.rows() == 0) { return; }
482 
483  typename RtVector<MaxRows>::d squared_norms;
484  squared_norms = mat.rowwise().squaredNorm();
485  const double normed_thresh = std::sqrt(double(squared_norms.size())) * threshold * threshold;
486  int num_valids = 0;
487  for (int row_id=0; row_id<squared_norms.size(); row_id++)
488  if(squared_norms[row_id] > normed_thresh)
489  ++num_valids;
490 
491  if (num_valids == 0) {
492  RtVectorUtils::resize(vec, 0);
493  RtMatrixUtils::resize(mat, 0, mat.cols());
494  } else {
495  typename RtMatrix<MaxRows, MaxCols>::d selector_mat;
496  RtMatrixUtils::createSelectorMat<MaxRows, MaxCols>(selector_mat, squared_norms, num_valids, normed_thresh);
497  mat = selector_mat * mat;
498  vec = selector_mat * vec;
499  }
500  }
501 
502  template <typename Mat, typename Vec>
503  static inline void removeZeroRows(Eigen::EigenBase<Mat>& mat, Eigen::EigenBase<Vec>& vec)
504  {
505  const double length_thresh = 1.e-12;
506  int first_empty_i=0;
507  for (; first_empty_i < mat.rows(); ++first_empty_i)
508  if (mat.derived().block(first_empty_i, 0, 1, mat.cols()).array().abs().sum() < length_thresh)
509  break;
510 
511  // found an empty row, now find the next nonempty row
512  int block_start_i = first_empty_i+1;
513  while (true) {
514  for (; block_start_i < mat.rows(); ++block_start_i)
515  if (mat.derived().block(block_start_i, 0, 1, mat.cols()).array().abs().sum() > length_thresh)
516  break;
517 
518  // if the rest of the matrix is zero, stop
519  if (block_start_i >= mat.rows())
520  break;
521 
522  // now find the end of the block
523  int block_end_i=block_start_i+1;
524  for (; block_end_i < mat.rows(); ++block_end_i)
525  if (mat.derived().block(block_end_i, 0, 1, mat.cols()).array().abs().sum() < length_thresh)
526  break;
527 
528  //now shift the block left
529  vec.derived().block(first_empty_i, 0, block_end_i-block_start_i, 1) =
530  vec.derived().block(block_start_i, 0, block_end_i-block_start_i,1);
531  mat.derived().block(first_empty_i, 0, block_end_i-block_start_i, mat.cols()) =
532  mat.derived().block(block_start_i, 0, block_end_i-block_start_i, mat.cols());
533 
534  first_empty_i += block_end_i-block_start_i;
535  block_start_i = block_end_i;
536  }
537  RtVectorUtils::conservativeResize(vec.derived(), first_empty_i);
538  RtMatrixUtils::conservativeResize(mat.derived(), first_empty_i, mat.cols());
539  }
540 
544  template <typename AffMatDerived, typename AffVecDerived, typename SubstMatDerived, typename SubstVecDerived>
545  static inline void substitute(Eigen::MatrixBase<AffMatDerived>& aff_mat, Eigen::MatrixBase<AffVecDerived>& aff_vec,
546  const Eigen::MatrixBase<SubstMatDerived>& subst_mat, const Eigen::MatrixBase<SubstVecDerived>& subst_vec)
547  {
548  #ifndef RTEIG_NO_ASSERTS
549  assertMatchingDimensions(aff_mat, aff_vec);
550  assertMatchingDimensions(subst_mat, subst_vec);
551  #endif
552  aff_vec += aff_mat*subst_vec;
553  substitute(aff_mat, subst_mat);
554  }
555 
556  template <typename AffMatDerived, typename SubstMatDerived>
557  static inline void substitute(Eigen::MatrixBase<AffMatDerived>& aff_mat, const Eigen::MatrixBase<SubstMatDerived>& subst_mat)
558  {
559  #ifndef RTEIG_NO_ASSERTS
560  assert(aff_mat.rows()*subst_mat.cols() <= aff_mat.MaxSizeAtCompileTime);
561  #endif
562  aff_mat = aff_mat*subst_mat;
563  }
564 
565  #ifndef RTEIG_NO_ASSERTS
566  template <typename AffMatDerived, typename AffVecDerived>
567  static inline void assertMatchingDimensions(const Eigen::MatrixBase<AffMatDerived>& aff_mat, const Eigen::MatrixBase<AffVecDerived>& aff_vec)
568  {
569  assert(aff_mat.rows() == aff_vec.rows());
570  assert(aff_vec.cols() == 1);
571  }
572  #endif
573  };
574 
575 
577  {
578  private:
579  RtQuadraticUtils(){}
580  virtual ~ RtQuadraticUtils(){}
581 
582  public:
583  // CARE with const correctness: quad_mat is NOT const! const will be casted away!
584  template <typename QuadMatDerived, typename AffMatDerived>
585  static inline void addFromNorm(const Eigen::MatrixBase<QuadMatDerived>& quad_mat, const Eigen::MatrixBase<AffMatDerived>& aff_mat)
586  {
587  const_cast<Eigen::MatrixBase<QuadMatDerived>&>(quad_mat) += aff_mat.transpose() * aff_mat;
588  }
589 
590  // CARE with const correctness: quad_mat is NOT const! const will be casted away!
591  template <typename QuadMatDerived, typename QuadVecDerived, typename AffMatDerived, typename AffVecDerived>
592  static inline void addFromNorm(const Eigen::MatrixBase<QuadMatDerived>& quad_mat, const Eigen::MatrixBase<QuadVecDerived>& quad_vec,
593  const Eigen::MatrixBase<AffMatDerived>& aff_mat, const Eigen::MatrixBase<AffVecDerived>& aff_vec)
594  {
595  #ifndef RTEIG_NO_ASSERTS
596  assertMatchingDimensions(quad_mat, quad_vec);
597  RtAffineUtils::assertMatchingDimensions(aff_mat, aff_vec);
598  #endif
599  addFromNorm(quad_mat, aff_mat);
600  const_cast<Eigen::MatrixBase<QuadVecDerived>&>(quad_vec) += aff_mat.transpose() * aff_vec;
601  }
602 
603  template <typename QuadMatDerived, typename AffMatDerived, typename WeightMatDerived>
604  static inline void addFromWeightedNorm(Eigen::MatrixBase<QuadMatDerived>& quad_mat,
605  const Eigen::MatrixBase<AffMatDerived>& aff_mat, const Eigen::MatrixBase<WeightMatDerived>& weight_mat)
606  {
607  quad_mat += aff_mat.transpose() * weight_mat * aff_mat;
608  }
609 
610  template <typename QuadMatDerived, typename QuadVecDerived, typename AffMatDerived, typename AffVecDerived, typename WeightMatDerived>
611  static inline void addFromWeightedNorm(Eigen::MatrixBase<QuadMatDerived>& quad_mat,
612  Eigen::MatrixBase<QuadVecDerived>& quad_vec, const Eigen::MatrixBase<AffMatDerived>& aff_mat,
613  const Eigen::MatrixBase<AffVecDerived>& aff_vec, const Eigen::MatrixBase<WeightMatDerived>& weight_mat)
614  {
615  #ifndef RTEIG_NO_ASSERTS
616  assertMatchingDimensions(quad_mat, quad_vec);
617  RtAffineUtils::assertMatchingDimensions(aff_mat, aff_vec);
618  #endif
619  addFromWeightedNorm(quad_mat, aff_mat, weight_mat);
620  quad_vec += aff_mat.transpose() * weight_mat * aff_vec;
621  }
622 
623  template <typename QuadMatDerived, typename SubstMatDerived>
624  static inline void substitute(Eigen::MatrixBase<QuadMatDerived>& quad_mat, const Eigen::MatrixBase<SubstMatDerived>& subst_mat)
625  {
626  #ifndef RTEIG_NO_ASSERTS
627  assert(subst_mat.cols()*subst_mat.cols() <= quad_mat.MaxSizeAtCompileTime);
628  #endif
629  quad_mat = subst_mat.transpose() * quad_mat *subst_mat;
630  }
631 
632  template <typename QuadMatDerived, typename QuadVecDerived, typename SubstMatDerived, typename SubstVecDerived>
633  static inline void substitute(Eigen::MatrixBase<QuadMatDerived>& quad_mat, Eigen::MatrixBase<QuadVecDerived>& quad_vec,
634  const Eigen::MatrixBase<SubstMatDerived>& subst_mat, const Eigen::MatrixBase<SubstVecDerived>& subst_vec)
635  {
636  #ifndef RTEIG_NO_ASSERTS
637  assertMatchingDimensions(quad_mat, quad_vec);
638  RtAffineUtils::assertMatchingDimensions(subst_mat, subst_vec);
639  assert(subst_mat.cols() <= quad_vec.MaxSizeAtCompileTime);
640  #endif
641  quad_vec += quad_mat*subst_vec;
642  quad_vec = subst_mat.transpose() * quad_vec;
643  substitute(quad_mat, subst_mat);
644  }
645 
646  #ifndef RTEIG_NO_ASSERTS
647  template <typename QuadMatDerived, typename QuadVecDerived>
648  static inline void assertMatchingDimensions(const Eigen::MatrixBase<QuadMatDerived>& quad_mat, const Eigen::MatrixBase<QuadVecDerived>& quad_vec)
649  {
650  assert(quad_mat.rows() == quad_mat.cols() && quad_mat.cols() == quad_vec.rows());
651  assert(quad_vec.cols() == 1);
652  }
653  #endif
654 
655  };
656 
657 }
Definition: RtMatrix.hpp:408
Definition: RtMatrix.hpp:18
Definition: RtMatrix.hpp:65
static void resize(Eigen::MatrixBase< Derived > &mat, int rows, int cols)
! Resize matrix without memory allocation.
Definition: RtMatrix.hpp:133
static void appendAtColumn(Eigen::MatrixBase< AffMatDerived > &aff_mat, Eigen::MatrixBase< AffVecDerived > &aff_vec, const Eigen::MatrixBase< SubstMatDerived > &subst_mat, int starting_column)
! Append an affine mapping at the end of another one starting from a certain column.
Definition: RtMatrix.hpp:450
Definition: Var.hpp:14
Definition: RtMatrix.hpp:576
Definition: RtMatrix.hpp:316
Definition: RtMatrix.hpp:44
static void appendAtColumn(Eigen::MatrixBase< AffMatDerived > &aff_mat, const Eigen::MatrixBase< SubstMatDerived > &subst_mat, int starting_column)
! Append a matrix at the end of another one starting from a certain column.
Definition: RtMatrix.hpp:236
static void substitute(Eigen::MatrixBase< AffMatDerived > &aff_mat, Eigen::MatrixBase< AffVecDerived > &aff_vec, const Eigen::MatrixBase< SubstMatDerived > &subst_mat, const Eigen::MatrixBase< SubstVecDerived > &subst_vec)
! Given aff_mat*x + aff_vec, substitute x = subst_mat * y + subst_vec into x.
Definition: RtMatrix.hpp:545
static void conservativeResize(Eigen::MatrixBase< Derived > &mat, int rows, int cols)
! Resize matrix by keeping data untouched without memory allocation.
Definition: RtMatrix.hpp:118