From 02c8b6af821fddf1976141448977dfb12c71ec30 Mon Sep 17 00:00:00 2001 From: Gael Guennebaud Date: Wed, 27 Oct 2010 15:13:03 +0200 Subject: [PATCH] fix sparse rankUpdate and triangularView iterator --- Eigen/src/Sparse/SparseSelfAdjointView.h | 10 +++++----- Eigen/src/Sparse/SparseTriangularView.h | 14 ++++++++------ unsupported/test/sparse_llt.cpp | 13 +++++++++++-- 3 files changed, 24 insertions(+), 13 deletions(-) diff --git a/Eigen/src/Sparse/SparseSelfAdjointView.h b/Eigen/src/Sparse/SparseSelfAdjointView.h index 769a05cf7..2a7101082 100644 --- a/Eigen/src/Sparse/SparseSelfAdjointView.h +++ b/Eigen/src/Sparse/SparseSelfAdjointView.h @@ -92,7 +92,7 @@ template class SparseSelfAdjointView * call this function with u.adjoint(). */ template - SparseSelfAdjointView& rankUpdate(const MatrixBase& u, Scalar alpha = Scalar(1)); + SparseSelfAdjointView& rankUpdate(const SparseMatrixBase& u, Scalar alpha = Scalar(1)); // const SparseLLT llt() const; // const SparseLDLT ldlt() const; @@ -127,15 +127,15 @@ SparseSelfAdjointView SparseMatrixBase::selfadjointView( template template SparseSelfAdjointView& -SparseSelfAdjointView::rankUpdate(const MatrixBase& u, Scalar alpha) +SparseSelfAdjointView::rankUpdate(const SparseMatrixBase& u, Scalar alpha) { SparseMatrix tmp = u * u.adjoint(); if(alpha==Scalar(0)) - m_matrix = tmp.template triangularView(); + m_matrix.const_cast_derived() = tmp.template triangularView(); else - m_matrix += alpha * tmp.template triangularView(); + m_matrix.const_cast_derived() /*+*/= alpha * tmp.template triangularView(); - return this; + return *this; } /*************************************************************************** diff --git a/Eigen/src/Sparse/SparseTriangularView.h b/Eigen/src/Sparse/SparseTriangularView.h index 8d0256599..319eaf066 100644 --- a/Eigen/src/Sparse/SparseTriangularView.h +++ b/Eigen/src/Sparse/SparseTriangularView.h @@ -26,11 +26,13 @@ #define EIGEN_SPARSE_TRIANGULARVIEW_H namespace internal { + template struct traits > : public traits {}; -} + +} // namespace internal template class SparseTriangularView : public SparseMatrixBase > @@ -38,13 +40,13 @@ template class SparseTriangularView enum { SkipFirst = (Mode==Lower && !(MatrixType::Flags&RowMajorBit)) || (Mode==Upper && (MatrixType::Flags&RowMajorBit)) }; public: + + EIGEN_SPARSE_PUBLIC_INTERFACE(SparseTriangularView) class InnerIterator; - typedef typename MatrixType::Scalar Scalar; - typedef typename MatrixType::Index Index; - inline Index rows() { return m_matrix.rows(); } - inline Index cols() { return m_matrix.cols(); } + inline Index rows() const { return m_matrix.rows(); } + inline Index cols() const { return m_matrix.cols(); } typedef typename internal::conditional::ret, MatrixType, const MatrixType&>::type MatrixTypeNested; @@ -83,7 +85,7 @@ class SparseTriangularView::InnerIterator : public MatrixType:: EIGEN_STRONG_INLINE operator bool() const { - return SkipFirst ? Base::operator bool() : (Base::operator bool() && this->index() < this->outer()); + return SkipFirst ? Base::operator bool() : (Base::operator bool() && this->index() <= this->outer()); } }; diff --git a/unsupported/test/sparse_llt.cpp b/unsupported/test/sparse_llt.cpp index 7b1cad4c5..2fcd8b279 100644 --- a/unsupported/test/sparse_llt.cpp +++ b/unsupported/test/sparse_llt.cpp @@ -77,8 +77,11 @@ template void sparse_llt(int rows, int cols) // new API { // Cholmod, as configured in CholmodSupport.h, only supports self-adjoint matrices - SparseMatrix m3 = m2.adjoint()*m2; - DenseMatrix refMat3 = refMat2.adjoint()*refMat2; + SparseMatrix m3 = m2 * m2.adjoint(), m3_lo(rows,rows), m3_up(rows,rows); + DenseMatrix refMat3 = refMat2 * refMat2.adjoint(); + + m3_lo.template selfadjointView().rankUpdate(m2,0); + m3_up.template selfadjointView().rankUpdate(m2,0); // with a single vector as the rhs ref_x = refMat3.template selfadjointView().llt().solve(b); @@ -89,6 +92,12 @@ template void sparse_llt(int rows, int cols) x = CholmodDecomposition, Upper>(m3).solve(b); VERIFY(ref_x.isApprox(x,test_precision()) && "LLT: cholmod solve, single dense rhs"); + x = CholmodDecomposition, Lower>(m3_lo).solve(b); + VERIFY(ref_x.isApprox(x,test_precision()) && "LLT: cholmod solve, single dense rhs"); + + x = CholmodDecomposition, Upper>(m3_up).solve(b); + VERIFY(ref_x.isApprox(x,test_precision()) && "LLT: cholmod solve, single dense rhs"); + // with multiple rhs ref_X = refMat3.template selfadjointView().llt().solve(B);