12 #ifndef KRONECKER_TENSOR_PRODUCT_H
13 #define KRONECKER_TENSOR_PRODUCT_H
17 template<
typename Scalar,
int Options,
typename Index>
class SparseMatrix;
29 template<
typename Lhs,
typename Rhs>
33 typedef ReturnByValue<KroneckerProduct> Base;
34 typedef typename Base::Scalar Scalar;
35 typedef typename Base::Index Index;
44 template<
typename Dest>
void evalTo(Dest& dst)
const;
46 inline Index rows()
const {
return m_A.rows() * m_B.rows(); }
47 inline Index cols()
const {
return m_A.cols() * m_B.cols(); }
49 Scalar coeff(Index row, Index col)
const
51 return m_A.coeff(row / m_B.rows(), col / m_B.cols()) *
52 m_B.coeff(row % m_B.rows(), col % m_B.cols());
55 Scalar coeff(Index i)
const
58 return m_A.coeff(i / m_A.size()) * m_B.coeff(i % m_A.size());
62 typename Lhs::Nested m_A;
63 typename Rhs::Nested m_B;
79 template<
typename Lhs,
typename Rhs>
83 typedef typename internal::traits<KroneckerProductSparse>::Index Index;
92 template<
typename Dest>
void evalTo(Dest& dst)
const;
94 inline Index rows()
const {
return m_A.rows() * m_B.rows(); }
95 inline Index cols()
const {
return m_A.cols() * m_B.cols(); }
97 template<
typename Scalar,
int Options,
typename Index>
98 operator SparseMatrix<Scalar, Options, Index>()
100 SparseMatrix<Scalar, Options, Index> result;
106 typename Lhs::Nested m_A;
107 typename Rhs::Nested m_B;
110 template<
typename Lhs,
typename Rhs>
111 template<
typename Dest>
114 const int BlockRows = Rhs::RowsAtCompileTime,
115 BlockCols = Rhs::ColsAtCompileTime;
116 const Index Br = m_B.rows(),
118 for (Index i=0; i < m_A.rows(); ++i)
119 for (Index j=0; j < m_A.cols(); ++j)
120 Block<Dest,BlockRows,BlockCols>(dst,i*Br,j*Bc,Br,Bc) = m_A.coeff(i,j) * m_B;
123 template<
typename Lhs,
typename Rhs>
124 template<
typename Dest>
127 const Index Br = m_B.rows(),
129 dst.resize(rows(),cols());
130 dst.resizeNonZeros(0);
131 dst.reserve(m_A.nonZeros() * m_B.nonZeros());
133 for (Index kA=0; kA < m_A.outerSize(); ++kA)
135 for (Index kB=0; kB < m_B.outerSize(); ++kB)
137 for (
typename Lhs::InnerIterator itA(m_A,kA); itA; ++itA)
139 for (
typename Rhs::InnerIterator itB(m_B,kB); itB; ++itB)
141 const Index i = itA.row() * Br + itB.row(),
142 j = itA.col() * Bc + itB.col();
143 dst.insert(i,j) = itA.value() * itB.value();
152 template<
typename _Lhs,
typename _Rhs>
155 typedef typename remove_all<_Lhs>::type Lhs;
156 typedef typename remove_all<_Rhs>::type Rhs;
157 typedef typename scalar_product_traits<typename Lhs::Scalar, typename Rhs::Scalar>::ReturnType Scalar;
160 Rows = size_at_compile_time<traits<Lhs>::RowsAtCompileTime, traits<Rhs>::RowsAtCompileTime>::ret,
161 Cols = size_at_compile_time<traits<Lhs>::ColsAtCompileTime, traits<Rhs>::ColsAtCompileTime>::ret,
162 MaxRows = size_at_compile_time<traits<Lhs>::MaxRowsAtCompileTime, traits<Rhs>::MaxRowsAtCompileTime>::ret,
163 MaxCols = size_at_compile_time<traits<Lhs>::MaxColsAtCompileTime, traits<Rhs>::MaxColsAtCompileTime>::ret,
164 CoeffReadCost = Lhs::CoeffReadCost + Rhs::CoeffReadCost + NumTraits<Scalar>::MulCost
167 typedef Matrix<Scalar,Rows,Cols> ReturnType;
170 template<
typename _Lhs,
typename _Rhs>
171 struct traits<KroneckerProductSparse<_Lhs,_Rhs> >
173 typedef MatrixXpr XprKind;
174 typedef typename remove_all<_Lhs>::type Lhs;
175 typedef typename remove_all<_Rhs>::type Rhs;
176 typedef typename scalar_product_traits<typename Lhs::Scalar, typename Rhs::Scalar>::ReturnType Scalar;
177 typedef typename promote_storage_type<typename traits<Lhs>::StorageKind,
typename traits<Rhs>::StorageKind>::ret StorageKind;
178 typedef typename promote_index_type<typename Lhs::Index, typename Rhs::Index>::type Index;
181 LhsFlags = Lhs::Flags,
182 RhsFlags = Rhs::Flags,
184 RowsAtCompileTime = size_at_compile_time<traits<Lhs>::RowsAtCompileTime, traits<Rhs>::RowsAtCompileTime>::ret,
185 ColsAtCompileTime = size_at_compile_time<traits<Lhs>::ColsAtCompileTime, traits<Rhs>::ColsAtCompileTime>::ret,
186 MaxRowsAtCompileTime = size_at_compile_time<traits<Lhs>::MaxRowsAtCompileTime, traits<Rhs>::MaxRowsAtCompileTime>::ret,
187 MaxColsAtCompileTime = size_at_compile_time<traits<Lhs>::MaxColsAtCompileTime, traits<Rhs>::MaxColsAtCompileTime>::ret,
189 EvalToRowMajor = (LhsFlags & RhsFlags & RowMajorBit),
190 RemovedBits = ~(EvalToRowMajor ? 0 : RowMajorBit),
192 Flags = ((LhsFlags | RhsFlags) & HereditaryBits & RemovedBits)
193 | EvalBeforeNestingBit | EvalBeforeAssigningBit,
194 CoeffReadCost = Dynamic
219 template<
typename A,
typename B>
236 template<
typename A,
typename B>
244 #endif // KRONECKER_TENSOR_PRODUCT_H
void evalTo(Dest &dst) const
Evaluate the Kronecker tensor product.
Definition: KroneckerTensorProduct.h:125
KroneckerProduct< A, B > kroneckerProduct(const MatrixBase< A > &a, const MatrixBase< B > &b)
Definition: KroneckerTensorProduct.h:220
void evalTo(Dest &dst) const
Evaluate the Kronecker tensor product.
Definition: KroneckerTensorProduct.h:112
KroneckerProductSparse(const Lhs &A, const Rhs &B)
Constructor.
Definition: KroneckerTensorProduct.h:87
Kronecker tensor product helper class for dense matrices.
Definition: KroneckerTensorProduct.h:30
Kronecker tensor product helper class for sparse matrices.
Definition: KroneckerTensorProduct.h:80
KroneckerProduct(const Lhs &A, const Rhs &B)
Constructor.
Definition: KroneckerTensorProduct.h:39