|  | // This file is part of Eigen, a lightweight C++ template library | 
|  | // for linear algebra. | 
|  | // | 
|  | // Copyright (C) 2009 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_SPARSE_DIAGONAL_PRODUCT_H | 
|  | #define EIGEN_SPARSE_DIAGONAL_PRODUCT_H | 
|  |  | 
|  | // The product of a diagonal matrix with a sparse matrix can be easily | 
|  | // implemented using expression template. | 
|  | // We have two consider very different cases: | 
|  | // 1 - diag * row-major sparse | 
|  | //     => each inner vector <=> scalar * sparse vector product | 
|  | //     => so we can reuse CwiseUnaryOp::InnerIterator | 
|  | // 2 - diag * col-major sparse | 
|  | //     => each inner vector <=> densevector * sparse vector cwise product | 
|  | //     => again, we can reuse specialization of CwiseBinaryOp::InnerIterator | 
|  | //        for that particular case | 
|  | // The two other cases are symmetric. | 
|  |  | 
|  | template<typename Lhs, typename Rhs> | 
|  | struct ei_traits<SparseDiagonalProduct<Lhs, Rhs> > | 
|  | { | 
|  | typedef typename ei_cleantype<Lhs>::type _Lhs; | 
|  | typedef typename ei_cleantype<Rhs>::type _Rhs; | 
|  | typedef typename _Lhs::Scalar Scalar; | 
|  | typedef typename ei_promote_index_type<typename ei_traits<Lhs>::Index, | 
|  | typename ei_traits<Rhs>::Index>::type Index; | 
|  | typedef Sparse StorageKind; | 
|  | typedef MatrixXpr XprKind; | 
|  | enum { | 
|  | RowsAtCompileTime = _Lhs::RowsAtCompileTime, | 
|  | ColsAtCompileTime = _Rhs::ColsAtCompileTime, | 
|  |  | 
|  | MaxRowsAtCompileTime = _Lhs::MaxRowsAtCompileTime, | 
|  | MaxColsAtCompileTime = _Rhs::MaxColsAtCompileTime, | 
|  |  | 
|  | SparseFlags = ei_is_diagonal<_Lhs>::ret ? int(_Rhs::Flags) : int(_Lhs::Flags), | 
|  | Flags = (SparseFlags&RowMajorBit), | 
|  | CoeffReadCost = Dynamic | 
|  | }; | 
|  | }; | 
|  |  | 
|  | enum {SDP_IsDiagonal, SDP_IsSparseRowMajor, SDP_IsSparseColMajor}; | 
|  | template<typename Lhs, typename Rhs, typename SparseDiagonalProductType, int RhsMode, int LhsMode> | 
|  | class ei_sparse_diagonal_product_inner_iterator_selector; | 
|  |  | 
|  | template<typename Lhs, typename Rhs> | 
|  | class SparseDiagonalProduct | 
|  | : public SparseMatrixBase<SparseDiagonalProduct<Lhs,Rhs> >, | 
|  | ei_no_assignment_operator | 
|  | { | 
|  | typedef typename Lhs::Nested LhsNested; | 
|  | typedef typename Rhs::Nested RhsNested; | 
|  |  | 
|  | typedef typename ei_cleantype<LhsNested>::type _LhsNested; | 
|  | typedef typename ei_cleantype<RhsNested>::type _RhsNested; | 
|  |  | 
|  | enum { | 
|  | LhsMode = ei_is_diagonal<_LhsNested>::ret ? SDP_IsDiagonal | 
|  | : (_LhsNested::Flags&RowMajorBit) ? SDP_IsSparseRowMajor : SDP_IsSparseColMajor, | 
|  | RhsMode = ei_is_diagonal<_RhsNested>::ret ? SDP_IsDiagonal | 
|  | : (_RhsNested::Flags&RowMajorBit) ? SDP_IsSparseRowMajor : SDP_IsSparseColMajor | 
|  | }; | 
|  |  | 
|  | public: | 
|  |  | 
|  | EIGEN_SPARSE_PUBLIC_INTERFACE(SparseDiagonalProduct) | 
|  |  | 
|  | typedef ei_sparse_diagonal_product_inner_iterator_selector | 
|  | <_LhsNested,_RhsNested,SparseDiagonalProduct,LhsMode,RhsMode> InnerIterator; | 
|  |  | 
|  | EIGEN_STRONG_INLINE SparseDiagonalProduct(const Lhs& lhs, const Rhs& rhs) | 
|  | : m_lhs(lhs), m_rhs(rhs) | 
|  | { | 
|  | ei_assert(lhs.cols() == rhs.rows() && "invalid sparse matrix * diagonal matrix product"); | 
|  | } | 
|  |  | 
|  | EIGEN_STRONG_INLINE Index rows() const { return m_lhs.rows(); } | 
|  | EIGEN_STRONG_INLINE Index cols() const { return m_rhs.cols(); } | 
|  |  | 
|  | EIGEN_STRONG_INLINE const _LhsNested& lhs() const { return m_lhs; } | 
|  | EIGEN_STRONG_INLINE const _RhsNested& rhs() const { return m_rhs; } | 
|  |  | 
|  | protected: | 
|  | LhsNested m_lhs; | 
|  | RhsNested m_rhs; | 
|  | }; | 
|  |  | 
|  |  | 
|  | template<typename Lhs, typename Rhs, typename SparseDiagonalProductType> | 
|  | class ei_sparse_diagonal_product_inner_iterator_selector | 
|  | <Lhs,Rhs,SparseDiagonalProductType,SDP_IsDiagonal,SDP_IsSparseRowMajor> | 
|  | : public CwiseUnaryOp<ei_scalar_multiple_op<typename Lhs::Scalar>,Rhs>::InnerIterator | 
|  | { | 
|  | typedef typename CwiseUnaryOp<ei_scalar_multiple_op<typename Lhs::Scalar>,Rhs>::InnerIterator Base; | 
|  | typedef typename Lhs::Index Index; | 
|  | public: | 
|  | inline ei_sparse_diagonal_product_inner_iterator_selector( | 
|  | const SparseDiagonalProductType& expr, Index outer) | 
|  | : Base(expr.rhs()*(expr.lhs().diagonal().coeff(outer)), outer) | 
|  | {} | 
|  | }; | 
|  |  | 
|  | template<typename Lhs, typename Rhs, typename SparseDiagonalProductType> | 
|  | class ei_sparse_diagonal_product_inner_iterator_selector | 
|  | <Lhs,Rhs,SparseDiagonalProductType,SDP_IsDiagonal,SDP_IsSparseColMajor> | 
|  | : public CwiseBinaryOp< | 
|  | ei_scalar_product_op<typename Lhs::Scalar>, | 
|  | SparseInnerVectorSet<Rhs,1>, | 
|  | typename Lhs::DiagonalVectorType>::InnerIterator | 
|  | { | 
|  | typedef typename CwiseBinaryOp< | 
|  | ei_scalar_product_op<typename Lhs::Scalar>, | 
|  | SparseInnerVectorSet<Rhs,1>, | 
|  | typename Lhs::DiagonalVectorType>::InnerIterator Base; | 
|  | typedef typename Lhs::Index Index; | 
|  | public: | 
|  | inline ei_sparse_diagonal_product_inner_iterator_selector( | 
|  | const SparseDiagonalProductType& expr, Index outer) | 
|  | : Base(expr.rhs().innerVector(outer) .cwiseProduct(expr.lhs().diagonal()), 0) | 
|  | {} | 
|  | }; | 
|  |  | 
|  | template<typename Lhs, typename Rhs, typename SparseDiagonalProductType> | 
|  | class ei_sparse_diagonal_product_inner_iterator_selector | 
|  | <Lhs,Rhs,SparseDiagonalProductType,SDP_IsSparseColMajor,SDP_IsDiagonal> | 
|  | : public CwiseUnaryOp<ei_scalar_multiple_op<typename Rhs::Scalar>,Lhs>::InnerIterator | 
|  | { | 
|  | typedef typename CwiseUnaryOp<ei_scalar_multiple_op<typename Rhs::Scalar>,Lhs>::InnerIterator Base; | 
|  | typedef typename Lhs::Index Index; | 
|  | public: | 
|  | inline ei_sparse_diagonal_product_inner_iterator_selector( | 
|  | const SparseDiagonalProductType& expr, Index outer) | 
|  | : Base(expr.lhs()*expr.rhs().diagonal().coeff(outer), outer) | 
|  | {} | 
|  | }; | 
|  |  | 
|  | template<typename Lhs, typename Rhs, typename SparseDiagonalProductType> | 
|  | class ei_sparse_diagonal_product_inner_iterator_selector | 
|  | <Lhs,Rhs,SparseDiagonalProductType,SDP_IsSparseRowMajor,SDP_IsDiagonal> | 
|  | : public CwiseBinaryOp< | 
|  | ei_scalar_product_op<typename Rhs::Scalar>, | 
|  | SparseInnerVectorSet<Lhs,1>, | 
|  | Transpose<typename Rhs::DiagonalVectorType> >::InnerIterator | 
|  | { | 
|  | typedef typename CwiseBinaryOp< | 
|  | ei_scalar_product_op<typename Rhs::Scalar>, | 
|  | SparseInnerVectorSet<Lhs,1>, | 
|  | Transpose<typename Rhs::DiagonalVectorType> >::InnerIterator Base; | 
|  | typedef typename Lhs::Index Index; | 
|  | public: | 
|  | inline ei_sparse_diagonal_product_inner_iterator_selector( | 
|  | const SparseDiagonalProductType& expr, Index outer) | 
|  | : Base(expr.lhs().innerVector(outer) .cwiseProduct(expr.rhs().diagonal().transpose()), 0) | 
|  | {} | 
|  | }; | 
|  |  | 
|  | // SparseMatrixBase functions | 
|  |  | 
|  | template<typename Derived> | 
|  | template<typename OtherDerived> | 
|  | const SparseDiagonalProduct<Derived,OtherDerived> | 
|  | SparseMatrixBase<Derived>::operator*(const DiagonalBase<OtherDerived> &other) const | 
|  | { | 
|  | return SparseDiagonalProduct<Derived,OtherDerived>(this->derived(), other.derived()); | 
|  | } | 
|  |  | 
|  | #endif // EIGEN_SPARSE_DIAGONAL_PRODUCT_H |