| // Ceres Solver - A fast non-linear least squares minimizer | 
 | // Copyright 2015 Google Inc. All rights reserved. | 
 | // http://ceres-solver.org/ | 
 | // | 
 | // Redistribution and use in source and binary forms, with or without | 
 | // modification, are permitted provided that the following conditions are met: | 
 | // | 
 | // * Redistributions of source code must retain the above copyright notice, | 
 | //   this list of conditions and the following disclaimer. | 
 | // * Redistributions in binary form must reproduce the above copyright notice, | 
 | //   this list of conditions and the following disclaimer in the documentation | 
 | //   and/or other materials provided with the distribution. | 
 | // * Neither the name of Google Inc. nor the names of its contributors may be | 
 | //   used to endorse or promote products derived from this software without | 
 | //   specific prior written permission. | 
 | // | 
 | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
 | // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
 | // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | 
 | // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE | 
 | // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | 
 | // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | 
 | // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | 
 | // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | 
 | // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | 
 | // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | 
 | // POSSIBILITY OF SUCH DAMAGE. | 
 | // | 
 | // Author: sameeragarwal@google.com (Sameer Agarwal) | 
 |  | 
 | #include "ceres/implicit_schur_complement.h" | 
 |  | 
 | #include "Eigen/Dense" | 
 | #include "ceres/block_sparse_matrix.h" | 
 | #include "ceres/block_structure.h" | 
 | #include "ceres/internal/eigen.h" | 
 | #include "ceres/internal/scoped_ptr.h" | 
 | #include "ceres/linear_solver.h" | 
 | #include "ceres/types.h" | 
 | #include "glog/logging.h" | 
 |  | 
 | namespace ceres { | 
 | namespace internal { | 
 |  | 
 | ImplicitSchurComplement::ImplicitSchurComplement( | 
 |     const LinearSolver::Options& options) | 
 |     : options_(options), | 
 |       D_(NULL), | 
 |       b_(NULL) { | 
 | } | 
 |  | 
 | ImplicitSchurComplement::~ImplicitSchurComplement() { | 
 | } | 
 |  | 
 | void ImplicitSchurComplement::Init(const BlockSparseMatrix& A, | 
 |                                    const double* D, | 
 |                                    const double* b) { | 
 |   // Since initialization is reasonably heavy, perhaps we can save on | 
 |   // constructing a new object everytime. | 
 |   if (A_ == NULL) { | 
 |     A_.reset(PartitionedMatrixViewBase::Create(options_, A)); | 
 |   } | 
 |  | 
 |   D_ = D; | 
 |   b_ = b; | 
 |  | 
 |   // Initialize temporary storage and compute the block diagonals of | 
 |   // E'E and F'E. | 
 |   if (block_diagonal_EtE_inverse_ == NULL) { | 
 |     block_diagonal_EtE_inverse_.reset(A_->CreateBlockDiagonalEtE()); | 
 |     if (options_.preconditioner_type == JACOBI) { | 
 |       block_diagonal_FtF_inverse_.reset(A_->CreateBlockDiagonalFtF()); | 
 |     } | 
 |     rhs_.resize(A_->num_cols_f()); | 
 |     rhs_.setZero(); | 
 |     tmp_rows_.resize(A_->num_rows()); | 
 |     tmp_e_cols_.resize(A_->num_cols_e()); | 
 |     tmp_e_cols_2_.resize(A_->num_cols_e()); | 
 |     tmp_f_cols_.resize(A_->num_cols_f()); | 
 |   } else { | 
 |     A_->UpdateBlockDiagonalEtE(block_diagonal_EtE_inverse_.get()); | 
 |     if (options_.preconditioner_type == JACOBI) { | 
 |       A_->UpdateBlockDiagonalFtF(block_diagonal_FtF_inverse_.get()); | 
 |     } | 
 |   } | 
 |  | 
 |   // The block diagonals of the augmented linear system contain | 
 |   // contributions from the diagonal D if it is non-null. Add that to | 
 |   // the block diagonals and invert them. | 
 |   AddDiagonalAndInvert(D_, block_diagonal_EtE_inverse_.get()); | 
 |   if (options_.preconditioner_type == JACOBI) { | 
 |     AddDiagonalAndInvert((D_ ==  NULL) ? NULL : D_ + A_->num_cols_e(), | 
 |                          block_diagonal_FtF_inverse_.get()); | 
 |   } | 
 |  | 
 |   // Compute the RHS of the Schur complement system. | 
 |   UpdateRhs(); | 
 | } | 
 |  | 
 | // Evaluate the product | 
 | // | 
 | //   Sx = [F'F - F'E (E'E)^-1 E'F]x | 
 | // | 
 | // By breaking it down into individual matrix vector products | 
 | // involving the matrices E and F. This is implemented using a | 
 | // PartitionedMatrixView of the input matrix A. | 
 | void ImplicitSchurComplement::RightMultiply(const double* x, double* y) const { | 
 |   // y1 = F x | 
 |   tmp_rows_.setZero(); | 
 |   A_->RightMultiplyF(x, tmp_rows_.data()); | 
 |  | 
 |   // y2 = E' y1 | 
 |   tmp_e_cols_.setZero(); | 
 |   A_->LeftMultiplyE(tmp_rows_.data(), tmp_e_cols_.data()); | 
 |  | 
 |   // y3 = -(E'E)^-1 y2 | 
 |   tmp_e_cols_2_.setZero(); | 
 |   block_diagonal_EtE_inverse_->RightMultiply(tmp_e_cols_.data(), | 
 |                                              tmp_e_cols_2_.data()); | 
 |   tmp_e_cols_2_ *= -1.0; | 
 |  | 
 |   // y1 = y1 + E y3 | 
 |   A_->RightMultiplyE(tmp_e_cols_2_.data(), tmp_rows_.data()); | 
 |  | 
 |   // y5 = D * x | 
 |   if (D_ != NULL) { | 
 |     ConstVectorRef Dref(D_ + A_->num_cols_e(), num_cols()); | 
 |     VectorRef(y, num_cols()) = | 
 |         (Dref.array().square() * | 
 |          ConstVectorRef(x, num_cols()).array()).matrix(); | 
 |   } else { | 
 |     VectorRef(y, num_cols()).setZero(); | 
 |   } | 
 |  | 
 |   // y = y5 + F' y1 | 
 |   A_->LeftMultiplyF(tmp_rows_.data(), y); | 
 | } | 
 |  | 
 | // Given a block diagonal matrix and an optional array of diagonal | 
 | // entries D, add them to the diagonal of the matrix and compute the | 
 | // inverse of each diagonal block. | 
 | void ImplicitSchurComplement::AddDiagonalAndInvert( | 
 |     const double* D, | 
 |     BlockSparseMatrix* block_diagonal) { | 
 |   const CompressedRowBlockStructure* block_diagonal_structure = | 
 |       block_diagonal->block_structure(); | 
 |   for (int r = 0; r < block_diagonal_structure->rows.size(); ++r) { | 
 |     const int row_block_pos = block_diagonal_structure->rows[r].block.position; | 
 |     const int row_block_size = block_diagonal_structure->rows[r].block.size; | 
 |     const Cell& cell = block_diagonal_structure->rows[r].cells[0]; | 
 |     MatrixRef m(block_diagonal->mutable_values() + cell.position, | 
 |                 row_block_size, row_block_size); | 
 |  | 
 |     if (D != NULL) { | 
 |       ConstVectorRef d(D + row_block_pos, row_block_size); | 
 |       m += d.array().square().matrix().asDiagonal(); | 
 |     } | 
 |  | 
 |     m = m | 
 |         .selfadjointView<Eigen::Upper>() | 
 |         .llt() | 
 |         .solve(Matrix::Identity(row_block_size, row_block_size)); | 
 |   } | 
 | } | 
 |  | 
 | // Similar to RightMultiply, use the block structure of the matrix A | 
 | // to compute y = (E'E)^-1 (E'b - E'F x). | 
 | void ImplicitSchurComplement::BackSubstitute(const double* x, double* y) { | 
 |   const int num_cols_e = A_->num_cols_e(); | 
 |   const int num_cols_f = A_->num_cols_f(); | 
 |   const int num_cols =  A_->num_cols(); | 
 |   const int num_rows = A_->num_rows(); | 
 |  | 
 |   // y1 = F x | 
 |   tmp_rows_.setZero(); | 
 |   A_->RightMultiplyF(x, tmp_rows_.data()); | 
 |  | 
 |   // y2 = b - y1 | 
 |   tmp_rows_ = ConstVectorRef(b_, num_rows) - tmp_rows_; | 
 |  | 
 |   // y3 = E' y2 | 
 |   tmp_e_cols_.setZero(); | 
 |   A_->LeftMultiplyE(tmp_rows_.data(), tmp_e_cols_.data()); | 
 |  | 
 |   // y = (E'E)^-1 y3 | 
 |   VectorRef(y, num_cols).setZero(); | 
 |   block_diagonal_EtE_inverse_->RightMultiply(tmp_e_cols_.data(), y); | 
 |  | 
 |   // The full solution vector y has two blocks. The first block of | 
 |   // variables corresponds to the eliminated variables, which we just | 
 |   // computed via back substitution. The second block of variables | 
 |   // corresponds to the Schur complement system, so we just copy those | 
 |   // values from the solution to the Schur complement. | 
 |   VectorRef(y + num_cols_e, num_cols_f) =  ConstVectorRef(x, num_cols_f); | 
 | } | 
 |  | 
 | // Compute the RHS of the Schur complement system. | 
 | // | 
 | // rhs = F'b - F'E (E'E)^-1 E'b | 
 | // | 
 | // Like BackSubstitute, we use the block structure of A to implement | 
 | // this using a series of matrix vector products. | 
 | void ImplicitSchurComplement::UpdateRhs() { | 
 |   // y1 = E'b | 
 |   tmp_e_cols_.setZero(); | 
 |   A_->LeftMultiplyE(b_, tmp_e_cols_.data()); | 
 |  | 
 |   // y2 = (E'E)^-1 y1 | 
 |   Vector y2 = Vector::Zero(A_->num_cols_e()); | 
 |   block_diagonal_EtE_inverse_->RightMultiply(tmp_e_cols_.data(), y2.data()); | 
 |  | 
 |   // y3 = E y2 | 
 |   tmp_rows_.setZero(); | 
 |   A_->RightMultiplyE(y2.data(), tmp_rows_.data()); | 
 |  | 
 |   // y3 = b - y3 | 
 |   tmp_rows_ = ConstVectorRef(b_, A_->num_rows()) - tmp_rows_; | 
 |  | 
 |   // rhs = F' y3 | 
 |   rhs_.setZero(); | 
 |   A_->LeftMultiplyF(tmp_rows_.data(), rhs_.data()); | 
 | } | 
 |  | 
 | }  // namespace internal | 
 | }  // namespace ceres |