diff options
author | Daniel Gomez Ferro <dgomezferro@gmail.com> | 2008-09-02 15:28:49 +0000 |
---|---|---|
committer | Daniel Gomez Ferro <dgomezferro@gmail.com> | 2008-09-02 15:28:49 +0000 |
commit | 8fb1678f0f174e85f6550e14f349841e406c8f53 (patch) | |
tree | 8eef447ce588be1dd450cca201f56fe32e36e4b5 | |
parent | 46fe7a3d9ec14ea56a879c48ba7f15e78342c8cb (diff) |
Extended sparse unit-test: nested blocks and InnerIterators.
Block specialization for sparse matrices.
InnerIterators for Blocks and fixes in CoreIterators.
-rw-r--r-- | Eigen/Sparse | 1 | ||||
-rw-r--r-- | Eigen/src/Core/Block.h | 5 | ||||
-rw-r--r-- | Eigen/src/Core/MatrixBase.h | 2 | ||||
-rw-r--r-- | Eigen/src/Core/util/Constants.h | 3 | ||||
-rw-r--r-- | Eigen/src/Core/util/ForwardDeclarations.h | 3 | ||||
-rw-r--r-- | Eigen/src/Sparse/CoreIterators.h | 80 | ||||
-rw-r--r-- | Eigen/src/Sparse/SparseBlock.h | 122 | ||||
-rw-r--r-- | Eigen/src/Sparse/SparseMatrix.h | 4 | ||||
-rw-r--r-- | Eigen/src/Sparse/SparseMatrixBase.h | 19 | ||||
-rw-r--r-- | test/sparse.cpp | 50 |
10 files changed, 276 insertions, 13 deletions
diff --git a/Eigen/Sparse b/Eigen/Sparse index 53bc4eccc..89fa387ba 100644 --- a/Eigen/Sparse +++ b/Eigen/Sparse @@ -13,6 +13,7 @@ namespace Eigen { #include "src/Sparse/SparseUtil.h" #include "src/Sparse/SparseMatrixBase.h" #include "src/Sparse/SparseArray.h" +#include "src/Sparse/SparseBlock.h" #include "src/Sparse/SparseMatrix.h" #include "src/Sparse/HashMatrix.h" #include "src/Sparse/LinkedVectorMatrix.h" diff --git a/Eigen/src/Core/Block.h b/Eigen/src/Core/Block.h index 11583d042..18fcdbb2b 100644 --- a/Eigen/src/Core/Block.h +++ b/Eigen/src/Core/Block.h @@ -65,6 +65,8 @@ template<typename MatrixType, int BlockRows, int BlockCols, int _PacketAccess, i struct ei_traits<Block<MatrixType, BlockRows, BlockCols, _PacketAccess, _DirectAccessStatus> > { typedef typename MatrixType::Scalar Scalar; + typedef typename MatrixType::Nested MatrixTypeNested; + typedef typename ei_unref<MatrixTypeNested>::type _MatrixTypeNested; enum{ RowsAtCompileTime = MatrixType::RowsAtCompileTime == 1 ? 1 : BlockRows, ColsAtCompileTime = MatrixType::ColsAtCompileTime == 1 ? 1 : BlockCols, @@ -94,6 +96,8 @@ template<typename MatrixType, int BlockRows, int BlockCols, int PacketAccess, in EIGEN_GENERIC_PUBLIC_INTERFACE(Block) + class InnerIterator; + /** Column or Row constructor */ inline Block(const MatrixType& matrix, int i) @@ -217,6 +221,7 @@ class Block<MatrixType,BlockRows,BlockCols,PacketAccess,HasDirectAccess> _EIGEN_GENERIC_PUBLIC_INTERFACE(Block, MapBase<Block>) + class InnerIterator; typedef typename ei_traits<Block>::AlignedDerivedType AlignedDerivedType; EIGEN_INHERIT_ASSIGNMENT_OPERATORS(Block) diff --git a/Eigen/src/Core/MatrixBase.h b/Eigen/src/Core/MatrixBase.h index dbb481914..4248ae523 100644 --- a/Eigen/src/Core/MatrixBase.h +++ b/Eigen/src/Core/MatrixBase.h @@ -155,7 +155,7 @@ template<typename Derived> class MatrixBase /** \returns the number of rows. \sa cols(), RowsAtCompileTime */ inline int rows() const { return derived().rows(); } - /** \returns the number of columns. \sa row(), ColsAtCompileTime*/ + /** \returns the number of columns. \sa rows(), ColsAtCompileTime*/ inline int cols() const { return derived().cols(); } /** \returns the number of coefficients, which is \a rows()*cols(). * \sa rows(), cols(), SizeAtCompileTime. */ diff --git a/Eigen/src/Core/util/Constants.h b/Eigen/src/Core/util/Constants.h index 9576ac6d8..d08af60e0 100644 --- a/Eigen/src/Core/util/Constants.h +++ b/Eigen/src/Core/util/Constants.h @@ -211,7 +211,8 @@ enum { enum { NoDirectAccess = 0, - HasDirectAccess = DirectAccessBit + HasDirectAccess = DirectAccessBit, + IsSparse = SparseBit }; const int FullyCoherentAccessPattern = 0x1; diff --git a/Eigen/src/Core/util/ForwardDeclarations.h b/Eigen/src/Core/util/ForwardDeclarations.h index fca47bad4..92852cbfa 100644 --- a/Eigen/src/Core/util/ForwardDeclarations.h +++ b/Eigen/src/Core/util/ForwardDeclarations.h @@ -36,7 +36,8 @@ template<typename ExpressionType> class NestByValue; template<typename ExpressionType> class SwapWrapper; template<typename MatrixType> class Minor; template<typename MatrixType, int BlockRows=Dynamic, int BlockCols=Dynamic, int PacketAccess=AsRequested, - int _DirectAccessStatus = ei_traits<MatrixType>::Flags&DirectAccessBit> class Block; + int _DirectAccessStatus = ei_traits<MatrixType>::Flags&DirectAccessBit ? DirectAccessBit + : ei_traits<MatrixType>::Flags&SparseBit> class Block; template<typename MatrixType> class Transpose; template<typename MatrixType> class Conjugate; template<typename NullaryOp, typename MatrixType> class CwiseNullaryOp; diff --git a/Eigen/src/Sparse/CoreIterators.h b/Eigen/src/Sparse/CoreIterators.h index f44db0c1f..ebe0bf8bd 100644 --- a/Eigen/src/Sparse/CoreIterators.h +++ b/Eigen/src/Sparse/CoreIterators.h @@ -37,7 +37,7 @@ class MatrixBase<Derived>::InnerIterator : m_matrix(mat), m_inner(0), m_outer(outer), m_end(mat.rows()) {} - Scalar value() + Scalar value() const { return (Derived::Flags&RowMajorBit) ? m_matrix.coeff(m_outer, m_inner) : m_matrix.coeff(m_inner, m_outer); @@ -66,6 +66,80 @@ class Transpose<MatrixType>::InnerIterator : public MatrixType::InnerIterator {} }; +template<typename MatrixType, int BlockRows, int BlockCols, int PacketAccess, int _DirectAccessStatus> +class Block<MatrixType, BlockRows, BlockCols, PacketAccess, _DirectAccessStatus>::InnerIterator +{ + typedef typename Block::Scalar Scalar; + typedef typename ei_traits<Block>::_MatrixTypeNested _MatrixTypeNested; + typedef typename _MatrixTypeNested::InnerIterator MatrixTypeIterator; + public: + + InnerIterator(const Block& block, int outer) + : m_iter(block.m_matrix,(Block::Flags&RowMajor) ? block.m_startRow.value() + outer : block.m_startCol.value() + outer), + m_start( (Block::Flags&RowMajor) ? block.m_startCol.value() : block.m_startRow.value()), + m_end(m_start + ((Block::Flags&RowMajor) ? block.m_blockCols.value() : block.m_blockRows.value())), + m_offset( (Block::Flags&RowMajor) ? block.m_startCol.value() : block.m_startRow.value()) + { + while (m_iter.index()>=0 && m_iter.index()<m_start) + ++m_iter; + } + + InnerIterator& operator++() + { + ++m_iter; + return *this; + } + + Scalar value() const { return m_iter.value(); } + + int index() const { return m_iter.index() - m_offset; } + + operator bool() const { return m_iter && m_iter.index()<m_end; } + + protected: + MatrixTypeIterator m_iter; + int m_start; + int m_end; + int m_offset; +}; + +template<typename MatrixType, int BlockRows, int BlockCols, int PacketAccess> +class Block<MatrixType, BlockRows, BlockCols, PacketAccess, IsSparse>::InnerIterator +{ + typedef typename Block::Scalar Scalar; + typedef typename ei_traits<Block>::_MatrixTypeNested _MatrixTypeNested; + typedef typename _MatrixTypeNested::InnerIterator MatrixTypeIterator; + public: + + InnerIterator(const Block& block, int outer) + : m_iter(block.m_matrix,(Block::Flags&RowMajor) ? block.m_startRow.value() + outer : block.m_startCol.value() + outer), + m_start( (Block::Flags&RowMajor) ? block.m_startCol.value() : block.m_startRow.value()), + m_end(m_start + ((Block::Flags&RowMajor) ? block.m_blockCols.value() : block.m_blockRows.value())), + m_offset( (Block::Flags&RowMajor) ? block.m_startCol.value() : block.m_startRow.value()) + { + while (m_iter.index()>=0 && m_iter.index()<m_start) + ++m_iter; + } + + InnerIterator& operator++() + { + ++m_iter; + return *this; + } + + Scalar value() const { return m_iter.value(); } + + int index() const { return m_iter.index() - m_offset; } + + operator bool() const { return m_iter && m_iter.index()<m_end; } + + protected: + MatrixTypeIterator m_iter; + int m_start; + int m_end; + int m_offset; +}; + template<typename UnaryOp, typename MatrixType> class CwiseUnaryOp<UnaryOp,MatrixType>::InnerIterator { @@ -133,13 +207,13 @@ class CwiseBinaryOp<BinaryOp,Lhs,Rhs>::InnerIterator ++m_lhsIter; ++m_rhsIter; } - else if (m_lhsIter && ((!m_rhsIter) || m_lhsIter.index() < m_rhsIter.index())) + else if (m_lhsIter && (!m_rhsIter || (m_lhsIter.index() < m_rhsIter.index()))) { m_id = m_lhsIter.index(); m_value = m_functor(m_lhsIter.value(), Scalar(0)); ++m_lhsIter; } - else if (m_rhsIter && ((!m_lhsIter) || m_lhsIter.index() > m_rhsIter.index())) + else if (m_rhsIter && (!m_lhsIter || (m_lhsIter.index() > m_rhsIter.index()))) { m_id = m_rhsIter.index(); m_value = m_functor(Scalar(0), m_rhsIter.value()); diff --git a/Eigen/src/Sparse/SparseBlock.h b/Eigen/src/Sparse/SparseBlock.h new file mode 100644 index 000000000..dd952714d --- /dev/null +++ b/Eigen/src/Sparse/SparseBlock.h @@ -0,0 +1,122 @@ +// 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> +// Copyright (C) 2008 Daniel Gomez Ferro <dgomezferro@gmail.com> +// +// 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_SPARSEBLOCK_H +#define EIGEN_SPARSEBLOCK_H + +template<typename MatrixType, int BlockRows, int BlockCols, int PacketAccess> +class Block<MatrixType,BlockRows,BlockCols,PacketAccess,IsSparse> + : public SparseMatrixBase<Block<MatrixType,BlockRows,BlockCols,PacketAccess,IsSparse> > +{ +public: + + _EIGEN_GENERIC_PUBLIC_INTERFACE(Block, SparseMatrixBase<Block>) + class InnerIterator; + + /** Column or Row constructor + */ + inline Block(const MatrixType& matrix, int i) + : m_matrix(matrix), + // It is a row if and only if BlockRows==1 and BlockCols==MatrixType::ColsAtCompileTime, + // and it is a column if and only if BlockRows==MatrixType::RowsAtCompileTime and BlockCols==1, + // all other cases are invalid. + // The case a 1x1 matrix seems ambiguous, but the result is the same anyway. + m_startRow( (BlockRows==1) && (BlockCols==MatrixType::ColsAtCompileTime) ? i : 0), + m_startCol( (BlockRows==MatrixType::RowsAtCompileTime) && (BlockCols==1) ? i : 0), + m_blockRows(matrix.rows()), // if it is a row, then m_blockRows has a fixed-size of 1, so no pb to try to overwrite it + m_blockCols(matrix.cols()) // same for m_blockCols + { + ei_assert( (i>=0) && ( + ((BlockRows==1) && (BlockCols==MatrixType::ColsAtCompileTime) && i<matrix.rows()) + ||((BlockRows==MatrixType::RowsAtCompileTime) && (BlockCols==1) && i<matrix.cols()))); + } + + /** Fixed-size constructor + */ + inline Block(const MatrixType& matrix, int startRow, int startCol) + : m_matrix(matrix), m_startRow(startRow), m_startCol(startCol), + m_blockRows(matrix.rows()), m_blockCols(matrix.cols()) + { + EIGEN_STATIC_ASSERT(RowsAtCompileTime!=Dynamic && RowsAtCompileTime!=Dynamic,this_method_is_only_for_fixed_size); + ei_assert(startRow >= 0 && BlockRows >= 1 && startRow + BlockRows <= matrix.rows() + && startCol >= 0 && BlockCols >= 1 && startCol + BlockCols <= matrix.cols()); + } + + /** Dynamic-size constructor + */ + inline Block(const MatrixType& matrix, + int startRow, int startCol, + int blockRows, int blockCols) + : m_matrix(matrix), m_startRow(startRow), m_startCol(startCol), + m_blockRows(blockRows), m_blockCols(blockCols) + { + ei_assert((RowsAtCompileTime==Dynamic || RowsAtCompileTime==blockRows) + && (ColsAtCompileTime==Dynamic || ColsAtCompileTime==blockCols)); + ei_assert(startRow >= 0 && blockRows >= 1 && startRow + blockRows <= matrix.rows() + && startCol >= 0 && blockCols >= 1 && startCol + blockCols <= matrix.cols()); + } + + inline int rows() const { return m_blockRows.value(); } + inline int cols() const { return m_blockCols.value(); } + + inline int stride(void) const { return m_matrix.stride(); } + + inline Scalar& coeffRef(int row, int col) + { + return m_matrix.const_cast_derived() + .coeffRef(row + m_startRow.value(), col + m_startCol.value()); + } + + inline const Scalar coeff(int row, int col) const + { + return m_matrix.coeff(row + m_startRow.value(), col + m_startCol.value()); + } + + inline Scalar& coeffRef(int index) + { + return m_matrix.const_cast_derived() + .coeffRef(m_startRow.value() + (RowsAtCompileTime == 1 ? 0 : index), + m_startCol.value() + (RowsAtCompileTime == 1 ? index : 0)); + } + + inline const Scalar coeff(int index) const + { + return m_matrix + .coeff(m_startRow.value() + (RowsAtCompileTime == 1 ? 0 : index), + m_startCol.value() + (RowsAtCompileTime == 1 ? index : 0)); + } + + protected: + + const typename MatrixType::Nested m_matrix; + const ei_int_if_dynamic<MatrixType::RowsAtCompileTime == 1 ? 0 : Dynamic> m_startRow; + const ei_int_if_dynamic<MatrixType::ColsAtCompileTime == 1 ? 0 : Dynamic> m_startCol; + const ei_int_if_dynamic<RowsAtCompileTime> m_blockRows; + const ei_int_if_dynamic<ColsAtCompileTime> m_blockCols; + +}; + + +#endif // EIGEN_SPARSEBLOCK_H diff --git a/Eigen/src/Sparse/SparseMatrix.h b/Eigen/src/Sparse/SparseMatrix.h index 9d4f325a3..d62718491 100644 --- a/Eigen/src/Sparse/SparseMatrix.h +++ b/Eigen/src/Sparse/SparseMatrix.h @@ -94,7 +94,7 @@ class SparseMatrix // ^^ optimization: let's first check if it is the last coefficient // (very common in high level algorithms) - const int* r = std::lower_bound(&m_data.index(start),&m_data.index(end),inner); + const int* r = std::lower_bound(&m_data.index(start),&m_data.index(end-1),inner); const int id = r-&m_data.index(0); return ((*r==inner) && (id<end)) ? m_data.value(id) : Scalar(0); } @@ -263,7 +263,7 @@ class SparseMatrix<Scalar,_Flags>::InnerIterator InnerIterator& operator++() { m_id++; return *this; } - Scalar value() { return m_matrix.m_data.value(m_id); } + Scalar value() const { return m_matrix.m_data.value(m_id); } int index() const { return m_matrix.m_data.index(m_id); } diff --git a/Eigen/src/Sparse/SparseMatrixBase.h b/Eigen/src/Sparse/SparseMatrixBase.h index 051d1cbc2..83dcefe06 100644 --- a/Eigen/src/Sparse/SparseMatrixBase.h +++ b/Eigen/src/Sparse/SparseMatrixBase.h @@ -139,8 +139,23 @@ class SparseMatrixBase : public MatrixBase<Derived> } else { - LinkedVectorMatrix<Scalar, RowMajorBit> trans = m.derived(); - s << trans; + if (m.cols() == 1) { + int row = 0; + for (typename Derived::InnerIterator it(m.derived(), 0); it; ++it) + { + for ( ; row<it.index(); ++row) + s << "0" << std::endl; + s << it.value() << std::endl; + ++row; + } + for ( ; row<m.rows(); ++row) + s << "0" << std::endl; + } + else + { + LinkedVectorMatrix<Scalar, RowMajorBit> trans = m.derived(); + s << trans; + } } return s; } diff --git a/test/sparse.cpp b/test/sparse.cpp index bac1091ce..7decb4111 100644 --- a/test/sparse.cpp +++ b/test/sparse.cpp @@ -25,9 +25,8 @@ #include "main.h" #include <Eigen/Sparse> -template<typename Scalar> void sparse() +template<typename Scalar> void sparse(int rows, int cols) { - int rows = 8, cols = 8; double density = std::max(8./(rows*cols), 0.01); typedef Matrix<Scalar,Dynamic,Dynamic> DenseMatrix; Scalar eps = 1e-6; @@ -73,6 +72,49 @@ template<typename Scalar> void sparse() VERIFY_IS_APPROX(m, refMat); + // test InnerIterators and Block expressions + for(int j=0; j<cols; j++) + { + for(int i=0; i<rows; i++) + { + for(int w=1; w<cols-j; w++) + { + for(int h=1; h<rows-i; h++) + { + VERIFY_IS_APPROX(m.block(i,j,h,w), refMat.block(i,j,h,w)); + for(int c=0; c<w; c++) + { + VERIFY_IS_APPROX(m.block(i,j,h,w).col(c), refMat.block(i,j,h,w).col(c)); + for(int r=0; r<h; r++) + { + VERIFY_IS_APPROX(m.block(i,j,h,w).col(c).coeff(r), refMat.block(i,j,h,w).col(c).coeff(r)); + } + } + for(int r=0; r<h; r++) + { + VERIFY_IS_APPROX(m.block(i,j,h,w).row(r), refMat.block(i,j,h,w).row(r)); + for(int c=0; c<w; c++) + { + VERIFY_IS_APPROX(m.block(i,j,h,w).row(r).coeff(c), refMat.block(i,j,h,w).row(r).coeff(c)); + } + } + } + } + } + } + + for(int c=0; c<cols; c++) + { + VERIFY_IS_APPROX(m.col(c) + m.col(c), (m + m).col(c)); + VERIFY_IS_APPROX(m.col(c) + m.col(c), refMat.col(c) + refMat.col(c)); + } + + for(int r=0; r<rows; r++) + { + VERIFY_IS_APPROX(m.row(r) + m.row(r), (m + m).row(r)); + VERIFY_IS_APPROX(m.row(r) + m.row(r), refMat.row(r) + refMat.row(r)); + } + // test SparseSetters // coherent setter // TODO extend the MatrixSetter @@ -102,9 +144,11 @@ template<typename Scalar> void sparse() } } VERIFY_IS_APPROX(m, refMat); + } void test_sparse() { - sparse<double>(); + sparse<double>(8, 8); + sparse<double>(16, 16); } |