| // Ceres Solver - A fast non-linear least squares minimizer |
| // Copyright 2010, 2011, 2012 Google Inc. All rights reserved. |
| // http://code.google.com/p/ceres-solver/ |
| // |
| // 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) |
| |
| #ifndef CERES_INTERNAL_SCHUR_COMPLEMENT_SOLVER_H_ |
| #define CERES_INTERNAL_SCHUR_COMPLEMENT_SOLVER_H_ |
| |
| #include <set> |
| #include <utility> |
| #include <vector> |
| |
| #include "ceres/block_random_access_matrix.h" |
| #include "ceres/block_sparse_matrix.h" |
| #include "ceres/block_structure.h" |
| #include "ceres/cxsparse.h" |
| #include "ceres/linear_solver.h" |
| #include "ceres/schur_eliminator.h" |
| #include "ceres/suitesparse.h" |
| #include "ceres/internal/scoped_ptr.h" |
| #include "ceres/types.h" |
| |
| namespace ceres { |
| namespace internal { |
| |
| class BlockSparseMatrix; |
| |
| // Base class for Schur complement based linear least squares |
| // solvers. It assumes that the input linear system Ax = b can be |
| // partitioned into |
| // |
| // E y + F z = b |
| // |
| // Where x = [y;z] is a partition of the variables. The paritioning |
| // of the variables is such that, E'E is a block diagonal |
| // matrix. Further, the rows of A are ordered so that for every |
| // variable block in y, all the rows containing that variable block |
| // occur as a vertically contiguous block. i.e the matrix A looks like |
| // |
| // E F |
| // A = [ y1 0 0 0 | z1 0 0 0 z5] |
| // [ y1 0 0 0 | z1 z2 0 0 0] |
| // [ 0 y2 0 0 | 0 0 z3 0 0] |
| // [ 0 0 y3 0 | z1 z2 z3 z4 z5] |
| // [ 0 0 y3 0 | z1 0 0 0 z5] |
| // [ 0 0 0 y4 | 0 0 0 0 z5] |
| // [ 0 0 0 y4 | 0 z2 0 0 0] |
| // [ 0 0 0 y4 | 0 0 0 0 0] |
| // [ 0 0 0 0 | z1 0 0 0 0] |
| // [ 0 0 0 0 | 0 0 z3 z4 z5] |
| // |
| // This structure should be reflected in the corresponding |
| // CompressedRowBlockStructure object associated with A. The linear |
| // system Ax = b should either be well posed or the array D below |
| // should be non-null and the diagonal matrix corresponding to it |
| // should be non-singular. |
| // |
| // SchurComplementSolver has two sub-classes. |
| // |
| // DenseSchurComplementSolver: For problems where the Schur complement |
| // matrix is small and dense, or if CHOLMOD/SuiteSparse is not |
| // installed. For structure from motion problems, this is solver can |
| // be used for problems with upto a few hundred cameras. |
| // |
| // SparseSchurComplementSolver: For problems where the Schur |
| // complement matrix is large and sparse. It requires that |
| // CHOLMOD/SuiteSparse be installed, as it uses CHOLMOD to find a |
| // sparse Cholesky factorization of the Schur complement. This solver |
| // can be used for solving structure from motion problems with tens of |
| // thousands of cameras, though depending on the exact sparsity |
| // structure, it maybe better to use an iterative solver. |
| // |
| // The two solvers can be instantiated by calling |
| // LinearSolver::CreateLinearSolver with LinearSolver::Options::type |
| // set to DENSE_SCHUR and SPARSE_SCHUR |
| // respectively. LinearSolver::Options::elimination_groups[0] should be |
| // at least 1. |
| class SchurComplementSolver : public BlockSparseMatrixSolver { |
| public: |
| explicit SchurComplementSolver(const LinearSolver::Options& options) |
| : options_(options) { |
| CHECK_GT(options.elimination_groups.size(), 1); |
| CHECK_GT(options.elimination_groups[0], 0); |
| } |
| |
| // LinearSolver methods |
| virtual ~SchurComplementSolver() {} |
| virtual LinearSolver::Summary SolveImpl( |
| BlockSparseMatrix* A, |
| const double* b, |
| const LinearSolver::PerSolveOptions& per_solve_options, |
| double* x); |
| |
| protected: |
| const LinearSolver::Options& options() const { return options_; } |
| |
| const BlockRandomAccessMatrix* lhs() const { return lhs_.get(); } |
| void set_lhs(BlockRandomAccessMatrix* lhs) { lhs_.reset(lhs); } |
| const double* rhs() const { return rhs_.get(); } |
| void set_rhs(double* rhs) { rhs_.reset(rhs); } |
| |
| private: |
| virtual void InitStorage(const CompressedRowBlockStructure* bs) = 0; |
| virtual bool SolveReducedLinearSystem(double* solution) = 0; |
| |
| LinearSolver::Options options_; |
| |
| scoped_ptr<SchurEliminatorBase> eliminator_; |
| scoped_ptr<BlockRandomAccessMatrix> lhs_; |
| scoped_array<double> rhs_; |
| |
| CERES_DISALLOW_COPY_AND_ASSIGN(SchurComplementSolver); |
| }; |
| |
| // Dense Cholesky factorization based solver. |
| class DenseSchurComplementSolver : public SchurComplementSolver { |
| public: |
| explicit DenseSchurComplementSolver(const LinearSolver::Options& options) |
| : SchurComplementSolver(options) {} |
| virtual ~DenseSchurComplementSolver() {} |
| |
| private: |
| virtual void InitStorage(const CompressedRowBlockStructure* bs); |
| virtual bool SolveReducedLinearSystem(double* solution); |
| |
| CERES_DISALLOW_COPY_AND_ASSIGN(DenseSchurComplementSolver); |
| }; |
| |
| #if !defined(CERES_NO_SUITESPARSE) || !defined(CERES_NO_CXSPARE) |
| // Sparse Cholesky factorization based solver. |
| class SparseSchurComplementSolver : public SchurComplementSolver { |
| public: |
| explicit SparseSchurComplementSolver(const LinearSolver::Options& options); |
| virtual ~SparseSchurComplementSolver(); |
| |
| private: |
| virtual void InitStorage(const CompressedRowBlockStructure* bs); |
| virtual bool SolveReducedLinearSystem(double* solution); |
| bool SolveReducedLinearSystemUsingSuiteSparse(double* solution); |
| bool SolveReducedLinearSystemUsingCXSparse(double* solution); |
| |
| // Size of the blocks in the Schur complement. |
| vector<int> blocks_; |
| |
| #ifndef CERES_NO_SUITESPARSE |
| SuiteSparse ss_; |
| // Symbolic factorization of the reduced linear system. Precomputed |
| // once and reused in subsequent calls. |
| cholmod_factor* factor_; |
| #endif // CERES_NO_SUITESPARSE |
| |
| #ifndef CERES_NO_CXSPARSE |
| CXSparse cxsparse_; |
| // Cached factorization |
| cs_dis* cxsparse_factor_; |
| #endif // CERES_NO_CXSPARSE |
| CERES_DISALLOW_COPY_AND_ASSIGN(SparseSchurComplementSolver); |
| }; |
| |
| #endif // !defined(CERES_NO_SUITESPARSE) || !defined(CERES_NO_CXSPARE) |
| } // namespace internal |
| } // namespace ceres |
| |
| #endif // CERES_INTERNAL_SCHUR_COMPLEMENT_SOLVER_H_ |