|  | // 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_ |