// This file is part of Eigen, a lightweight C++ template library // for linear algebra. // // Copyright (C) 2008 Gael Guennebaud // Copyright (C) 2006-2008 Benoit Jacob // // 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 . #ifndef EIGEN_BLOCK_H #define EIGEN_BLOCK_H /** \class Block * * \brief Expression of a fixed-size or dynamic-size block * * \param MatrixType the type of the object in which we are taking a block * \param BlockRows the number of rows of the block we are taking at compile time (optional) * \param BlockCols the number of columns of the block we are taking at compile time (optional) * \param _PacketAccess \internal used to enforce aligned loads in expressions such as * \code mat.block() += other; \endcode. Possible values are * \c AsRequested (default) and \c EnforceAlignedAccess. * See class MapBase for more details. * \param _DirectAccessStatus \internal used for partial specialization * * This class represents an expression of either a fixed-size or dynamic-size block. It is the return * type of MatrixBase::block(int,int,int,int) and MatrixBase::block(int,int) and * most of the time this is the only way it is used. * * However, if you want to directly maniputate block expressions, * for instance if you want to write a function returning such an expression, you * will need to use this class. * * Here is an example illustrating the dynamic case: * \include class_Block.cpp * Output: \verbinclude class_Block.out * * \note Even though this expression has dynamic size, in the case where \a MatrixType * has fixed size, this expression inherits a fixed maximal size which means that evaluating * it does not cause a dynamic memory allocation. * * Here is an example illustrating the fixed-size case: * \include class_FixedBlock.cpp * Output: \verbinclude class_FixedBlock.out * * \sa MatrixBase::block(int,int,int,int), MatrixBase::block(int,int), class VectorBlock */ template struct ei_traits > { typedef typename ei_traits::Scalar Scalar; typedef typename ei_nested::type MatrixTypeNested; typedef typename ei_unref::type _MatrixTypeNested; typedef typename ei_traits::StorageType StorageType; enum{ RowsAtCompileTime = BlockRows, ColsAtCompileTime = BlockCols, MaxRowsAtCompileTime = RowsAtCompileTime == 1 ? 1 : (BlockRows==Dynamic ? int(ei_traits::MaxRowsAtCompileTime) : BlockRows), MaxColsAtCompileTime = ColsAtCompileTime == 1 ? 1 : (BlockCols==Dynamic ? int(ei_traits::MaxColsAtCompileTime) : BlockCols), RowMajor = int(ei_traits::Flags)&RowMajorBit, InnerSize = RowMajor ? int(ColsAtCompileTime) : int(RowsAtCompileTime), InnerMaxSize = RowMajor ? int(MaxColsAtCompileTime) : int(MaxRowsAtCompileTime), MaskPacketAccessBit = (InnerMaxSize == Dynamic || (InnerSize >= ei_packet_traits::size)) ? PacketAccessBit : 0, FlagsLinearAccessBit = (RowsAtCompileTime == 1 || ColsAtCompileTime == 1) ? LinearAccessBit : 0, Flags = (ei_traits::Flags & (HereditaryBits | MaskPacketAccessBit | DirectAccessBit)) | FlagsLinearAccessBit, CoeffReadCost = ei_traits::CoeffReadCost, PacketAccess = _PacketAccess }; typedef typename ei_meta_if&, Block >::ret AlignedDerivedType; }; template class Block : public MatrixBase > { public: EIGEN_GENERIC_PUBLIC_INTERFACE(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= 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 >= 0 && startRow + blockRows <= matrix.rows() && startCol >= 0 && blockCols >= 0 && startCol + blockCols <= matrix.cols()); } EIGEN_INHERIT_ASSIGNMENT_OPERATORS(Block) inline int rows() const { return m_blockRows.value(); } inline int cols() const { return m_blockCols.value(); } inline Scalar& coeffRef(int row, int col) { return m_matrix.const_cast_derived() .coeffRef(row + m_startRow.value(), col + m_startCol.value()); } inline const CoeffReturnType 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 CoeffReturnType coeff(int index) const { return m_matrix .coeff(m_startRow.value() + (RowsAtCompileTime == 1 ? 0 : index), m_startCol.value() + (RowsAtCompileTime == 1 ? index : 0)); } template inline PacketScalar packet(int row, int col) const { return m_matrix.template packet (row + m_startRow.value(), col + m_startCol.value()); } template inline void writePacket(int row, int col, const PacketScalar& x) { m_matrix.const_cast_derived().template writePacket (row + m_startRow.value(), col + m_startCol.value(), x); } template inline PacketScalar packet(int index) const { return m_matrix.template packet (m_startRow.value() + (RowsAtCompileTime == 1 ? 0 : index), m_startCol.value() + (RowsAtCompileTime == 1 ? index : 0)); } template inline void writePacket(int index, const PacketScalar& x) { m_matrix.const_cast_derived().template writePacket (m_startRow.value() + (RowsAtCompileTime == 1 ? 0 : index), m_startCol.value() + (RowsAtCompileTime == 1 ? index : 0), x); } #ifdef EIGEN_PARSED_BY_DOXYGEN /** \sa MapBase::data() */ inline const Scalar* data() const; /** \sa MapBase::stride() */ inline int stride() const; #endif protected: const typename MatrixType::Nested m_matrix; const ei_int_if_dynamic m_startRow; const ei_int_if_dynamic m_startCol; const ei_int_if_dynamic m_blockRows; const ei_int_if_dynamic m_blockCols; }; /** \internal */ template class Block : public MapBase > { public: _EIGEN_GENERIC_PUBLIC_INTERFACE(Block, MapBase) class InnerIterator; typedef typename ei_traits::AlignedDerivedType AlignedDerivedType; friend class Block; EIGEN_INHERIT_ASSIGNMENT_OPERATORS(Block) AlignedDerivedType _convertToEnforceAlignedAccess() { return Block (m_matrix, Base::m_data, Base::m_rows.value(), Base::m_cols.value()); } /** Column or Row constructor */ inline Block(const MatrixType& matrix, int i) : Base(&matrix.const_cast_derived().coeffRef( (BlockRows==1) && (BlockCols==MatrixType::ColsAtCompileTime) ? i : 0, (BlockRows==MatrixType::RowsAtCompileTime) && (BlockCols==1) ? i : 0), BlockRows==1 ? 1 : matrix.rows(), BlockCols==1 ? 1 : matrix.cols()), m_matrix(matrix) { ei_assert( (i>=0) && ( ((BlockRows==1) && (BlockCols==MatrixType::ColsAtCompileTime) && i= 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) : Base(&matrix.const_cast_derived().coeffRef(startRow,startCol), blockRows, blockCols), m_matrix(matrix) { ei_assert((RowsAtCompileTime==Dynamic || RowsAtCompileTime==blockRows) && (ColsAtCompileTime==Dynamic || ColsAtCompileTime==blockCols)); ei_assert(startRow >= 0 && blockRows >= 0 && startRow + blockRows <= matrix.rows() && startCol >= 0 && blockCols >= 0 && startCol + blockCols <= matrix.cols()); } /** \sa MapBase::stride() */ inline int stride() const { return ((!Base::IsVectorAtCompileTime) || (BlockRows==1 && ((Flags&RowMajorBit)==0)) || (BlockCols==1 && ((Flags&RowMajorBit)==RowMajorBit))) ? m_matrix.stride() : 1; } #ifndef __SUNPRO_CC // FIXME sunstudio is not friendly with the above friend... protected: #endif #ifndef EIGEN_PARSED_BY_DOXYGEN /** \internal used by allowAligned() */ inline Block(const MatrixType& matrix, const Scalar* data, int blockRows, int blockCols) : Base(data, blockRows, blockCols), m_matrix(matrix) {} #endif protected: const typename MatrixType::Nested m_matrix; }; /** \returns a dynamic-size expression of a block in *this. * * \param startRow the first row in the block * \param startCol the first column in the block * \param blockRows the number of rows in the block * \param blockCols the number of columns in the block * * Example: \include MatrixBase_block_int_int_int_int.cpp * Output: \verbinclude MatrixBase_block_int_int_int_int.out * * \note Even though the returned expression has dynamic size, in the case * when it is applied to a fixed-size matrix, it inherits a fixed maximal size, * which means that evaluating it does not cause a dynamic memory allocation. * * \sa class Block, block(int,int) */ template inline typename BlockReturnType::Type MatrixBase ::block(int startRow, int startCol, int blockRows, int blockCols) { return typename BlockReturnType::Type(derived(), startRow, startCol, blockRows, blockCols); } /** This is the const version of block(int,int,int,int). */ template inline const typename BlockReturnType::Type MatrixBase ::block(int startRow, int startCol, int blockRows, int blockCols) const { return typename BlockReturnType::Type(derived(), startRow, startCol, blockRows, blockCols); } /** \returns a dynamic-size expression of a corner of *this. * * \param type the type of corner. Can be \a Eigen::TopLeft, \a Eigen::TopRight, * \a Eigen::BottomLeft, \a Eigen::BottomRight. * \param cRows the number of rows in the corner * \param cCols the number of columns in the corner * * Example: \include MatrixBase_corner_enum_int_int.cpp * Output: \verbinclude MatrixBase_corner_enum_int_int.out * * \note Even though the returned expression has dynamic size, in the case * when it is applied to a fixed-size matrix, it inherits a fixed maximal size, * which means that evaluating it does not cause a dynamic memory allocation. * * \sa class Block, block(int,int,int,int) */ template inline typename BlockReturnType::Type MatrixBase ::corner(CornerType type, int cRows, int cCols) { switch(type) { default: ei_assert(false && "Bad corner type."); case TopLeft: return typename BlockReturnType::Type(derived(), 0, 0, cRows, cCols); case TopRight: return typename BlockReturnType::Type(derived(), 0, cols() - cCols, cRows, cCols); case BottomLeft: return typename BlockReturnType::Type(derived(), rows() - cRows, 0, cRows, cCols); case BottomRight: return typename BlockReturnType::Type(derived(), rows() - cRows, cols() - cCols, cRows, cCols); } } /** This is the const version of corner(CornerType, int, int).*/ template inline const typename BlockReturnType::Type MatrixBase::corner(CornerType type, int cRows, int cCols) const { switch(type) { default: ei_assert(false && "Bad corner type."); case TopLeft: return typename BlockReturnType::Type(derived(), 0, 0, cRows, cCols); case TopRight: return typename BlockReturnType::Type(derived(), 0, cols() - cCols, cRows, cCols); case BottomLeft: return typename BlockReturnType::Type(derived(), rows() - cRows, 0, cRows, cCols); case BottomRight: return typename BlockReturnType::Type(derived(), rows() - cRows, cols() - cCols, cRows, cCols); } } /** \returns a fixed-size expression of a corner of *this. * * \param type the type of corner. Can be \a Eigen::TopLeft, \a Eigen::TopRight, * \a Eigen::BottomLeft, \a Eigen::BottomRight. * * The template parameters CRows and CCols arethe number of rows and columns in the corner. * * Example: \include MatrixBase_template_int_int_corner_enum.cpp * Output: \verbinclude MatrixBase_template_int_int_corner_enum.out * * \sa class Block, block(int,int,int,int) */ template template inline typename BlockReturnType::Type MatrixBase::corner(CornerType type) { switch(type) { default: ei_assert(false && "Bad corner type."); case TopLeft: return Block(derived(), 0, 0); case TopRight: return Block(derived(), 0, cols() - CCols); case BottomLeft: return Block(derived(), rows() - CRows, 0); case BottomRight: return Block(derived(), rows() - CRows, cols() - CCols); } } /** This is the const version of corner(CornerType).*/ template template inline const typename BlockReturnType::Type MatrixBase::corner(CornerType type) const { switch(type) { default: ei_assert(false && "Bad corner type."); case TopLeft: return Block(derived(), 0, 0); case TopRight: return Block(derived(), 0, cols() - CCols); case BottomLeft: return Block(derived(), rows() - CRows, 0); case BottomRight: return Block(derived(), rows() - CRows, cols() - CCols); } } /** \returns a fixed-size expression of a block in *this. * * The template parameters \a BlockRows and \a BlockCols are the number of * rows and columns in the block. * * \param startRow the first row in the block * \param startCol the first column in the block * * Example: \include MatrixBase_block_int_int.cpp * Output: \verbinclude MatrixBase_block_int_int.out * * \note since block is a templated member, the keyword template has to be used * if the matrix type is also a template parameter: \code m.template block<3,3>(1,1); \endcode * * \sa class Block, block(int,int,int,int) */ template template inline typename BlockReturnType::Type MatrixBase::block(int startRow, int startCol) { return Block(derived(), startRow, startCol); } /** This is the const version of block<>(int, int). */ template template inline const typename BlockReturnType::Type MatrixBase::block(int startRow, int startCol) const { return Block(derived(), startRow, startCol); } /** \returns an expression of the \a i-th column of *this. Note that the numbering starts at 0. * * Example: \include MatrixBase_col.cpp * Output: \verbinclude MatrixBase_col.out * * \sa row(), class Block */ template inline typename MatrixBase::ColXpr MatrixBase::col(int i) { return ColXpr(derived(), i); } /** This is the const version of col(). */ template inline const typename MatrixBase::ColXpr MatrixBase::col(int i) const { return ColXpr(derived(), i); } /** \returns an expression of the \a i-th row of *this. Note that the numbering starts at 0. * * Example: \include MatrixBase_row.cpp * Output: \verbinclude MatrixBase_row.out * * \sa col(), class Block */ template inline typename MatrixBase::RowXpr MatrixBase::row(int i) { return RowXpr(derived(), i); } /** This is the const version of row(). */ template inline const typename MatrixBase::RowXpr MatrixBase::row(int i) const { return RowXpr(derived(), i); } #endif // EIGEN_BLOCK_H