aboutsummaryrefslogtreecommitdiffhomepage
path: root/Eigen/src/Sparse/TriangularSolver.h
diff options
context:
space:
mode:
Diffstat (limited to 'Eigen/src/Sparse/TriangularSolver.h')
-rw-r--r--Eigen/src/Sparse/TriangularSolver.h170
1 files changed, 170 insertions, 0 deletions
diff --git a/Eigen/src/Sparse/TriangularSolver.h b/Eigen/src/Sparse/TriangularSolver.h
new file mode 100644
index 000000000..8634e114c
--- /dev/null
+++ b/Eigen/src/Sparse/TriangularSolver.h
@@ -0,0 +1,170 @@
+// This file is part of Eigen, a lightweight C++ template library
+// for linear algebra. Eigen itself is part of the KDE project.
+//
+// Copyright (C) 2008 Gael Guennebaud <g.gael@free.fr>
+//
+// Eigen is free software; you can redistribute it and/or
+// modify it under the terms of the GNU Lesser General Public
+// License as published by the Free Software Foundation; either
+// version 3 of the License, or (at your option) any later version.
+//
+// Alternatively, you can redistribute it and/or
+// modify it under the terms of the GNU General Public License as
+// published by the Free Software Foundation; either version 2 of
+// the License, or (at your option) any later version.
+//
+// Eigen is distributed in the hope that it will be useful, but WITHOUT ANY
+// WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
+// FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License or the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU Lesser General Public
+// License and a copy of the GNU General Public License along with
+// Eigen. If not, see <http://www.gnu.org/licenses/>.
+
+#ifndef EIGEN_SPARSETRIANGULARSOLVER_H
+#define EIGEN_SPARSETRIANGULARSOLVER_H
+
+template<typename Lhs, typename Rhs,
+ int TriangularPart = (int(Lhs::Flags) & LowerTriangularBit)
+ ? Lower
+ : (int(Lhs::Flags) & UpperTriangularBit)
+ ? Upper
+ : -1,
+ int StorageOrder = int(Lhs::Flags) & RowMajorBit ? RowMajor : ColMajor
+ >
+struct ei_inverse_product_selector;
+
+// forward substitution, row-major
+template<typename Lhs, typename Rhs>
+struct ei_inverse_product_selector<Lhs,Rhs,Lower,RowMajor>
+{
+ typedef typename Rhs::Scalar Scalar;
+ static void run(const Lhs& lhs, const Rhs& rhs, Rhs& res)
+ {
+ for(int col=0 ; col<rhs.cols() ; ++col)
+ {
+ for(int i=0; i<lhs.rows(); ++i)
+ {
+ Scalar tmp = rhs.coeff(i,col);
+ Scalar lastVal = 0;
+ int lastIndex = 0;
+ for(typename Lhs::InnerIterator it(lhs, i); it; ++it)
+ {
+ lastVal = it.value();
+ lastIndex = it.index();
+ tmp -= lastVal * res.coeff(lastIndex,col);
+ }
+ if (Lhs::Flags & UnitDiagBit)
+ res.coeffRef(i,col) = tmp;
+ else
+ {
+ ei_assert(lastIndex==i);
+ res.coeffRef(i,col) = tmp/lastVal;
+ }
+ }
+ }
+ }
+};
+
+// backward substitution, row-major
+template<typename Lhs, typename Rhs>
+struct ei_inverse_product_selector<Lhs,Rhs,Upper,RowMajor>
+{
+ typedef typename Rhs::Scalar Scalar;
+ static void run(const Lhs& lhs, const Rhs& rhs, Rhs& res)
+ {
+ for(int col=0 ; col<rhs.cols() ; ++col)
+ {
+ for(int i=lhs.rows()-1 ; i>=0 ; --i)
+ {
+ Scalar tmp = rhs.coeff(i,col);
+ typename Lhs::InnerIterator it(lhs, i);
+ for(++it; it; ++it)
+ {
+ tmp -= it.value() * res.coeff(it.index(),col);
+ }
+
+ if (Lhs::Flags & UnitDiagBit)
+ res.coeffRef(i,col) = tmp;
+ else
+ {
+ typename Lhs::InnerIterator it(lhs, i);
+ ei_assert(it.index() == i);
+ res.coeffRef(i,col) = tmp/it.value();
+ }
+ }
+ }
+ }
+};
+
+// forward substitution, col-major
+template<typename Lhs, typename Rhs>
+struct ei_inverse_product_selector<Lhs,Rhs,Lower,ColMajor>
+{
+ typedef typename Rhs::Scalar Scalar;
+ static void run(const Lhs& lhs, const Rhs& rhs, Rhs& res)
+ {
+ // NOTE we could avoid this copy using an in-place API
+ res = rhs;
+ for(int col=0 ; col<rhs.cols() ; ++col)
+ {
+ for(int i=0; i<lhs.cols(); ++i)
+ {
+ typename Lhs::InnerIterator it(lhs, i);
+ if(!(Lhs::Flags & UnitDiagBit))
+ {
+ ei_assert(it.index()==i);
+ res.coeffRef(i,col) /= it.value();
+ }
+ Scalar tmp = res.coeffRef(i,col);
+ for(++it; it; ++it)
+ res.coeffRef(it.index(), col) -= tmp * it.value();
+ }
+ }
+ }
+};
+
+// backward substitution, col-major
+template<typename Lhs, typename Rhs>
+struct ei_inverse_product_selector<Lhs,Rhs,Upper,ColMajor>
+{
+ typedef typename Rhs::Scalar Scalar;
+ static void run(const Lhs& lhs, const Rhs& rhs, Rhs& res)
+ {
+ // NOTE we could avoid this copy using an in-place API
+ res = rhs;
+ for(int col=0 ; col<rhs.cols() ; ++col)
+ {
+ for(int i=lhs.cols()-1; i>=0; --i)
+ {
+ if(!(Lhs::Flags & UnitDiagBit))
+ {
+ // FIXME lhs.coeff(i,i) might not be always efficient while it must simply be the
+ // last element of the column !
+ res.coeffRef(i,col) /= lhs.coeff(i,i);
+ }
+ Scalar tmp = res.coeffRef(i,col);
+ typename Lhs::InnerIterator it(lhs, i);
+ for(; it && it.index()<i; ++it)
+ res.coeffRef(it.index(), col) -= tmp * it.value();
+ }
+ }
+ }
+};
+
+template<typename Derived>
+template<typename OtherDerived>
+OtherDerived
+SparseMatrixBase<Derived>::inverseProduct(const MatrixBase<OtherDerived>& other) const
+{
+ ei_assert(derived().cols() == other.rows());
+ ei_assert(!(Flags & ZeroDiagBit));
+ ei_assert(Flags & (UpperTriangularBit|LowerTriangularBit));
+
+ OtherDerived res(other.rows(), other.cols());
+ ei_inverse_product_selector<Derived, OtherDerived>::run(derived(), other.derived(), res);
+ return res;
+}
+
+#endif // EIGEN_SPARSETRIANGULARSOLVER_H