Line | Branch | Exec | Source |
---|---|---|---|
1 | // This file is part of Bembel, the higher order C++ boundary element library. | ||
2 | // | ||
3 | // Copyright (C) 2022 see <http://www.bembel.eu> | ||
4 | // | ||
5 | // It was written as part of a cooperation of J. Doelz, H. Harbrecht, S. Kurz, | ||
6 | // M. Multerer, S. Schoeps, and F. Wolf at Technische Universitaet Darmstadt, | ||
7 | // Universitaet Basel, and Universita della Svizzera italiana, Lugano. This | ||
8 | // source code is subject to the GNU General Public License version 3 and | ||
9 | // provided WITHOUT ANY WARRANTY, see <http://www.bembel.eu> for further | ||
10 | // information. | ||
11 | #ifndef BEMBEL_SRC_H2MATRIX_EIGENHELPER_H2DENSEPRODUCT_HPP_ | ||
12 | #define BEMBEL_SRC_H2MATRIX_EIGENHELPER_H2DENSEPRODUCT_HPP_ | ||
13 | |||
14 | // The contents of this file are a modification of the file | ||
15 | // SparseDenseProduct.h from the Eigen library | ||
16 | namespace Eigen { | ||
17 | |||
18 | namespace internal { | ||
19 | |||
20 | template <typename H2LhsType, typename DenseRhsType, typename DenseResType, | ||
21 | typename AlphaType, | ||
22 | int LhsStorageOrder = | ||
23 | ((H2LhsType::Flags & RowMajorBit) == RowMajorBit) ? RowMajor | ||
24 | : ColMajor, | ||
25 | bool ColPerCol = ((DenseRhsType::Flags & RowMajorBit) == 0) || | ||
26 | DenseRhsType::ColsAtCompileTime == 1> | ||
27 | struct H2_time_dense_product_impl; | ||
28 | |||
29 | /** | ||
30 | * \brief H2-matrix-vector multiplication, works also for matrices by iterating | ||
31 | * over the columns. | ||
32 | */ | ||
33 | template <typename ScalarH2, typename DenseRhsType, typename DenseResType, | ||
34 | typename AlphaType> | ||
35 | struct H2_time_dense_product_impl<H2Matrix<ScalarH2>, DenseRhsType, | ||
36 | DenseResType, AlphaType, ColMajor, true> { | ||
37 | typedef typename traits<DenseRhsType>::Scalar ScalarRhs; | ||
38 | typedef typename traits<DenseResType>::Scalar ScalarRes; | ||
39 | 303 | static void run(const H2Matrix<ScalarH2>& lhs, const DenseRhsType& rhs, | |
40 | DenseResType& res, const AlphaType& alpha) { | ||
41 | // get H2-data | ||
42 | 303 | int max_level = | |
43 | 303 | lhs.get_block_cluster_tree()(0, 0).get_parameters().max_level_; | |
44 | 303 | int min_cluster_level = | |
45 | 303 | lhs.get_block_cluster_tree()(0, 0).get_parameters().min_cluster_level_; | |
46 |
1/2✓ Branch 1 taken 303 times.
✗ Branch 2 not taken.
|
303 | auto moment_matrix = lhs.get_fmm_moment_matrix(); |
47 |
1/2✓ Branch 1 taken 303 times.
✗ Branch 2 not taken.
|
303 | auto transfer_matrices = lhs.get_fmm_transfer_matrices(); |
48 | 303 | int vector_dimension = moment_matrix.size(); | |
49 | |||
50 |
2/2✓ Branch 2 taken 303 times.
✓ Branch 3 taken 303 times.
|
606 | for (Index c = 0; c < rhs.cols(); ++c) { |
51 | // go discontinuous in rhs | ||
52 |
1/2✓ Branch 1 taken 303 times.
✗ Branch 2 not taken.
|
303 | Matrix<ScalarH2, Dynamic, 1> long_rhs_all = |
53 |
3/6✓ Branch 1 taken 303 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 303 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 303 times.
✗ Branch 8 not taken.
|
606 | (lhs.get_transformation_matrix() * rhs.col(c)).eval(); |
54 | 303 | int vector_component_size = long_rhs_all.rows() / vector_dimension; | |
55 | |||
56 | // initialize destination | ||
57 |
1/2✓ Branch 2 taken 303 times.
✗ Branch 3 not taken.
|
303 | Matrix<ScalarRes, Dynamic, 1> long_dst_all(long_rhs_all.rows()); |
58 |
1/2✓ Branch 1 taken 303 times.
✗ Branch 2 not taken.
|
303 | long_dst_all.setZero(); |
59 | |||
60 |
2/2✓ Branch 0 taken 460 times.
✓ Branch 1 taken 303 times.
|
763 | for (int col_component = 0; col_component < vector_dimension; |
61 | ++col_component) { | ||
62 |
2/2✓ Branch 3 taken 774 times.
✓ Branch 4 taken 460 times.
|
1234 | for (int row_component = 0; row_component < vector_dimension; |
63 | ++row_component) { | ||
64 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
774 | Matrix<ScalarRhs, Dynamic, 1> long_rhs = long_rhs_all.segment( |
65 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
774 | col_component * vector_component_size, vector_component_size); |
66 |
1/2✓ Branch 2 taken 774 times.
✗ Branch 3 not taken.
|
774 | Matrix<ScalarRhs, Dynamic, 1> long_dst(long_rhs.rows()); |
67 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
774 | long_dst.setZero(); |
68 | |||
69 | // split long rhs into pieces by reshaping | ||
70 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
774 | Matrix<ScalarRhs, Dynamic, Dynamic> long_rhs_matrix = |
71 |
4/7✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 698 times.
✓ Branch 6 taken 76 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 698 times.
✗ Branch 9 not taken.
|
1548 | Map<Matrix<ScalarRhs, Dynamic, Dynamic>>( |
72 | 774 | long_rhs.data(), moment_matrix[col_component].cols(), | |
73 | 774 | long_rhs.rows() / moment_matrix[col_component].cols()); | |
74 | |||
75 | // do forward-transformation | ||
76 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
774 | std::vector<Matrix<ScalarRhs, Dynamic, Dynamic>> long_rhs_forward = |
77 | Bembel::H2Multipole::forwardTransformation( | ||
78 | 774 | moment_matrix[col_component], transfer_matrices, | |
79 | max_level - min_cluster_level, long_rhs_matrix); | ||
80 | |||
81 | #pragma omp parallel | ||
82 | { | ||
83 | // initialize target for each process | ||
84 |
1/2✓ Branch 2 taken 774 times.
✗ Branch 3 not taken.
|
774 | Matrix<ScalarRes, Dynamic, 1> my_long_dst(long_dst.rows()); |
85 | |||
86 | // initialize target of backward-transformation | ||
87 | std::vector<Matrix<ScalarRes, Dynamic, Dynamic>> | ||
88 | 774 | my_long_dst_backward; | |
89 |
2/2✓ Branch 1 taken 1548 times.
✓ Branch 2 taken 774 times.
|
2322 | for (int i = 0; i < long_rhs_forward.size(); ++i) |
90 |
3/6✓ Branch 3 taken 1548 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 1548 times.
✗ Branch 7 not taken.
✓ Branch 9 taken 1548 times.
✗ Branch 10 not taken.
|
3096 | my_long_dst_backward.push_back( |
91 | Matrix<ScalarRes, Dynamic, Dynamic>::Zero( | ||
92 | 3096 | long_rhs_forward[i].rows(), long_rhs_forward[i].cols())); | |
93 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
774 | my_long_dst.setZero(); |
94 | |||
95 | // matrix-vector | ||
96 | 774 | for (auto leaf = | |
97 | 774 | lhs.get_block_cluster_tree()(row_component, col_component) | |
98 | 774 | .clbegin(); | |
99 | 446598 | leaf != | |
100 | 446598 | lhs.get_block_cluster_tree()(row_component, col_component) | |
101 |
2/2✓ Branch 1 taken 445824 times.
✓ Branch 2 taken 774 times.
|
893196 | .clend(); |
102 | 445824 | ++leaf) { | |
103 | #pragma omp single nowait | ||
104 | { | ||
105 |
2/3✓ Branch 2 taken 295776 times.
✓ Branch 3 taken 150048 times.
✗ Branch 4 not taken.
|
445824 | switch ((*leaf)->get_cc()) { |
106 | // deal with matrix blocks | ||
107 | 295776 | case Bembel::BlockClusterAdmissibility::Dense: { | |
108 |
2/4✓ Branch 1 taken 295776 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 295776 times.
✗ Branch 5 not taken.
|
295776 | my_long_dst.segment((*leaf)->get_row_start_index(), |
109 |
1/2✓ Branch 2 taken 295776 times.
✗ Branch 3 not taken.
|
295776 | (*leaf)->get_row_end_index() - |
110 |
2/4✓ Branch 2 taken 295776 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 295776 times.
✗ Branch 7 not taken.
|
591552 | (*leaf)->get_row_start_index()) += |
111 |
1/2✓ Branch 4 taken 295776 times.
✗ Branch 5 not taken.
|
295776 | (*leaf)->get_leaf().get_F() * |
112 |
2/4✓ Branch 1 taken 295776 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 295776 times.
✗ Branch 5 not taken.
|
295776 | long_rhs.segment((*leaf)->get_col_start_index(), |
113 |
1/2✓ Branch 2 taken 295776 times.
✗ Branch 3 not taken.
|
295776 | (*leaf)->get_col_end_index() - |
114 |
1/2✓ Branch 2 taken 295776 times.
✗ Branch 3 not taken.
|
295776 | (*leaf)->get_col_start_index()); |
115 | 295776 | } break; | |
116 | // deal with low-rank blocks | ||
117 | 150048 | case Bembel::BlockClusterAdmissibility::LowRank: { | |
118 | const Bembel::ElementTreeNode* cluster1 = | ||
119 | 150048 | (*leaf)->get_cluster1(); | |
120 | const Bembel::ElementTreeNode* cluster2 = | ||
121 | 150048 | (*leaf)->get_cluster2(); | |
122 | 150048 | int cluster_level = cluster1->get_level(); | |
123 | 150048 | int fmm_level = lhs.get_block_cluster_tree()(0, 0) | |
124 | 150048 | .get_parameters() | |
125 | 300096 | .max_level_ - | |
126 | 150048 | lhs.get_block_cluster_tree()(0, 0) | |
127 | 150048 | .get_parameters() | |
128 | 150048 | .min_cluster_level_ - | |
129 | 150048 | (*leaf)->get_cluster1()->get_level(); | |
130 | 150048 | int cluster1_col = cluster1->id_; | |
131 | 150048 | int cluster2_col = cluster2->id_; | |
132 |
2/4✓ Branch 2 taken 150048 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 150048 times.
✗ Branch 6 not taken.
|
150048 | my_long_dst_backward[fmm_level].col(cluster1_col) += |
133 |
1/2✓ Branch 4 taken 150048 times.
✗ Branch 5 not taken.
|
150048 | (*leaf)->get_leaf().get_F() * |
134 |
1/2✓ Branch 2 taken 150048 times.
✗ Branch 3 not taken.
|
150048 | long_rhs_forward[fmm_level].col(cluster2_col); |
135 | 150048 | } break; | |
136 | // this leaf is not a low-rank block and not a dense block, | ||
137 | // thus it is not a leaf -> error | ||
138 | ✗ | default: | |
139 | ✗ | assert(0 && "This should never happen"); | |
140 | break; | ||
141 | } | ||
142 | } | ||
143 | } | ||
144 | |||
145 | // do backward transformation | ||
146 |
2/4✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 774 times.
✗ Branch 5 not taken.
|
774 | my_long_dst += Bembel::H2Multipole::backwardTransformation( |
147 | 774 | moment_matrix[row_component], transfer_matrices, | |
148 | max_level - min_cluster_level, my_long_dst_backward); | ||
149 | |||
150 | #pragma omp critical | ||
151 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
774 | long_dst += my_long_dst; |
152 | 774 | } | |
153 | |||
154 | // finish off vector component | ||
155 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
1548 | long_dst_all.segment(row_component * vector_component_size, |
156 |
1/2✓ Branch 1 taken 774 times.
✗ Branch 2 not taken.
|
774 | vector_component_size) += long_dst; |
157 | } | ||
158 | } | ||
159 | |||
160 | // go continuous and write output | ||
161 |
4/8✓ Branch 1 taken 303 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 303 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 303 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 303 times.
✗ Branch 11 not taken.
|
303 | res.col(c) += |
162 |
2/4✓ Branch 1 taken 303 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 303 times.
✗ Branch 5 not taken.
|
606 | alpha * lhs.get_transformation_matrix().transpose() * long_dst_all; |
163 | } | ||
164 | 303 | } | |
165 | }; | ||
166 | |||
167 | template <typename H2LhsType, typename DenseRhsType, typename DenseResType, | ||
168 | typename AlphaType> | ||
169 | 303 | inline void H2_time_dense_product(const H2LhsType& lhs, const DenseRhsType& rhs, | |
170 | DenseResType& res, const AlphaType& alpha) { | ||
171 | H2_time_dense_product_impl<H2LhsType, DenseRhsType, DenseResType, | ||
172 | 303 | AlphaType>::run(lhs, rhs, res, alpha); | |
173 | 303 | } | |
174 | |||
175 | /** | ||
176 | * \brief overwrite eigen implementation by using distributive law, i.e., | ||
177 | *compute A*c+B*c instead of (A+B)*c | ||
178 | */ | ||
179 | template <typename BinaryOp, typename BinaryLhs, typename BinaryRhs, | ||
180 | typename Rhs, int ProductType> | ||
181 | struct generic_product_impl<CwiseBinaryOp<BinaryOp, BinaryLhs, BinaryRhs>, Rhs, | ||
182 | H2, DenseShape, ProductType> | ||
183 | : generic_product_impl_base< | ||
184 | CwiseBinaryOp<BinaryOp, BinaryLhs, BinaryRhs>, Rhs, | ||
185 | generic_product_impl<CwiseBinaryOp<BinaryOp, BinaryLhs, BinaryRhs>, | ||
186 | Rhs, H2, DenseShape, ProductType>> { | ||
187 | typedef CwiseBinaryOp<BinaryOp, BinaryLhs, BinaryRhs> Lhs; | ||
188 | typedef typename Product<Lhs, Rhs>::Scalar Scalar; | ||
189 | |||
190 | // we only need to specify scaleAndAddTo, everything else is build on this in | ||
191 | // the background | ||
192 | template <typename Dest> | ||
193 | 134 | static void scaleAndAddTo(Dest& dst, const Lhs& lhs, const Rhs& rhs, | |
194 | const Scalar& alpha) { | ||
195 | typedef Product<BinaryLhs, Rhs, ProductType> LeftProduct; | ||
196 | typedef Product<BinaryRhs, Rhs, ProductType> RightProduct; | ||
197 | typedef CwiseBinaryOp<BinaryOp, LeftProduct, RightProduct> NewCwiseBinaryOp; | ||
198 | typedef Matrix<Scalar, Dynamic, Dynamic> DenseMatrix; | ||
199 | typedef CwiseNullaryOp<scalar_constant_op<Scalar>, DenseMatrix> | ||
200 | ScalarCwiseNullaryOp; | ||
201 | typedef scalar_constant_op<Scalar> ScalarConstantOp; | ||
202 | typedef scalar_product_op<Scalar, Scalar> ScalarTimesOp; | ||
203 | typedef CwiseBinaryOp<ScalarTimesOp, ScalarCwiseNullaryOp, NewCwiseBinaryOp> | ||
204 | ScalarTimesNewCwiseBinaryOp; | ||
205 | typedef typename traits<Lhs>::Scalar LhsScalar; | ||
206 | typedef typename traits<Rhs>::Scalar RhsScalar; | ||
207 | typedef add_assign_op<LhsScalar, RhsScalar> addAssignOp; | ||
208 | |||
209 | static_assert( | ||
210 | is_same<BinaryOp, scalar_sum_op<LhsScalar, RhsScalar>>::value || | ||
211 | is_same<BinaryOp, | ||
212 | scalar_difference_op<LhsScalar, RhsScalar>>::value, | ||
213 | "Product of CwiseBinaryOp not defined for this BinaryOp"); | ||
214 | |||
215 |
1/2✓ Branch 2 taken 67 times.
✗ Branch 3 not taken.
|
134 | LeftProduct lprod(lhs.lhs(), rhs); |
216 |
1/2✓ Branch 2 taken 67 times.
✗ Branch 3 not taken.
|
134 | RightProduct rprod(lhs.rhs(), rhs); |
217 |
1/2✓ Branch 2 taken 67 times.
✗ Branch 3 not taken.
|
134 | NewCwiseBinaryOp xpr_prod(lprod, rprod, lhs.functor()); |
218 |
1/2✓ Branch 3 taken 67 times.
✗ Branch 4 not taken.
|
134 | ScalarCwiseNullaryOp xpr_scalar(lprod.rows(), lprod.cols(), |
219 | 134 | ScalarConstantOp(alpha)); | |
220 |
1/2✓ Branch 2 taken 67 times.
✗ Branch 3 not taken.
|
134 | ScalarTimesNewCwiseBinaryOp xpr(xpr_scalar, xpr_prod, ScalarTimesOp()); |
221 | Assignment<Dest, ScalarTimesNewCwiseBinaryOp, addAssignOp, | ||
222 |
1/2✓ Branch 2 taken 67 times.
✗ Branch 3 not taken.
|
134 | Dense2Dense>::run(dst, xpr, addAssignOp()); |
223 | 134 | } | |
224 | }; | ||
225 | /** | ||
226 | * \brief overwrite eigen implementation by using associative law to compute | ||
227 | *a*(H*x) instead of (a*H)*x | ||
228 | */ | ||
229 | template <typename ProdLhsScalar, typename ProdRhsScalar, typename BinaryLhs, | ||
230 | typename BinaryRhs, typename Rhs, int ProductType> | ||
231 | struct generic_product_impl< | ||
232 | CwiseBinaryOp<scalar_product_op<ProdLhsScalar, ProdRhsScalar>, BinaryLhs, | ||
233 | BinaryRhs>, | ||
234 | Rhs, H2, DenseShape, ProductType> | ||
235 | : generic_product_impl_base< | ||
236 | CwiseBinaryOp<scalar_product_op<ProdLhsScalar, ProdRhsScalar>, | ||
237 | BinaryLhs, BinaryRhs>, | ||
238 | Rhs, | ||
239 | generic_product_impl< | ||
240 | CwiseBinaryOp<scalar_product_op<ProdLhsScalar, ProdRhsScalar>, | ||
241 | BinaryLhs, BinaryRhs>, | ||
242 | Rhs, H2, DenseShape, ProductType>> { | ||
243 | typedef CwiseBinaryOp<scalar_product_op<ProdLhsScalar, ProdRhsScalar>, | ||
244 | BinaryLhs, BinaryRhs> | ||
245 | Lhs; | ||
246 | typedef typename Product<Lhs, Rhs>::Scalar Scalar; | ||
247 | |||
248 | // we only need to specify scaleAndAddTo, everything else is build on this in | ||
249 | // the background | ||
250 | template <typename Dest> | ||
251 | 2 | static void scaleAndAddTo(Dest& dst, const Lhs& lhs, const Rhs& rhs, | |
252 | const Scalar& alpha) { | ||
253 | 2 | Scalar beta = lhs.lhs().functor()(); | |
254 | generic_product_impl<BinaryRhs, Rhs, H2, DenseShape, | ||
255 |
1/2✓ Branch 2 taken 2 times.
✗ Branch 3 not taken.
|
2 | ProductType>::scaleAndAddTo(dst, lhs.rhs(), rhs, |
256 |
1/2✓ Branch 1 taken 2 times.
✗ Branch 2 not taken.
|
2 | alpha * beta); |
257 | 2 | } | |
258 | }; | ||
259 | // same overwrite, but for const CwiseBinaryOp by inheritance from non-const | ||
260 | // version | ||
261 | template <typename BinaryOp, typename BinaryLhs, typename BinaryRhs, | ||
262 | typename Rhs, int ProductType> | ||
263 | struct generic_product_impl<const CwiseBinaryOp<BinaryOp, BinaryLhs, BinaryRhs>, | ||
264 | Rhs, H2, DenseShape, ProductType> | ||
265 | : generic_product_impl<CwiseBinaryOp<BinaryOp, BinaryLhs, BinaryRhs>, Rhs, | ||
266 | H2, DenseShape, ProductType> {}; | ||
267 | |||
268 | template <typename Lhs, typename Rhs, int ProductType> | ||
269 | struct generic_product_impl<Lhs, Rhs, H2, DenseShape, ProductType> | ||
270 | : generic_product_impl_base< | ||
271 | Lhs, Rhs, | ||
272 | generic_product_impl<Lhs, Rhs, H2, DenseShape, ProductType>> { | ||
273 | typedef typename Product<Lhs, Rhs>::Scalar Scalar; | ||
274 | |||
275 | // we only need to specify scaleAndAddTo, everything else is build on this in | ||
276 | // the background | ||
277 | template <typename Dest> | ||
278 | 379 | static void scaleAndAddTo(Dest& dst, const Lhs& lhs, const Rhs& rhs, | |
279 | const Scalar& alpha) { | ||
280 | typedef | ||
281 | typename nested_eval<Lhs, ((Rhs::Flags & RowMajorBit) == 0) | ||
282 | ? 1 | ||
283 | : Rhs::ColsAtCompileTime>::type LhsNested; | ||
284 | typedef typename nested_eval< | ||
285 | Rhs, ((Lhs::Flags & RowMajorBit) == 0) ? 1 : Dynamic>::type RhsNested; | ||
286 | 379 | LhsNested lhsNested(lhs); | |
287 | 379 | RhsNested rhsNested(rhs); | |
288 | 379 | internal::H2_time_dense_product(lhsNested, rhsNested, dst, alpha); | |
289 | 379 | } | |
290 | }; | ||
291 | |||
292 | template <typename Lhs, typename Rhs, int Options, int ProductTag> | ||
293 | struct product_evaluator<Product<Lhs, Rhs, Options>, ProductTag, H2, DenseShape> | ||
294 | : public evaluator<typename Product<Lhs, Rhs, Options>::PlainObject> { | ||
295 | typedef Product<Lhs, Rhs, Options> XprType; | ||
296 | typedef typename XprType::PlainObject PlainObject; | ||
297 | typedef evaluator<PlainObject> Base; | ||
298 | |||
299 | enum { Flags = Base::Flags }; | ||
300 | |||
301 | 208 | explicit product_evaluator(const XprType& xpr) | |
302 |
1/2✓ Branch 4 taken 106 times.
✗ Branch 5 not taken.
|
208 | : m_result(xpr.rows(), xpr.cols()) { |
303 |
1/2✓ Branch 2 taken 106 times.
✗ Branch 3 not taken.
|
208 | ::new (static_cast<Base*>(this)) Base(m_result); |
304 | 208 | generic_product_impl<Lhs, Rhs, H2, DenseShape, ProductTag>::evalTo( | |
305 |
1/2✓ Branch 3 taken 106 times.
✗ Branch 4 not taken.
|
208 | m_result, xpr.lhs(), xpr.rhs()); |
306 | 208 | } | |
307 | |||
308 | protected: | ||
309 | PlainObject m_result; | ||
310 | }; | ||
311 | |||
312 | } // end namespace internal | ||
313 | |||
314 | } // end namespace Eigen | ||
315 | |||
316 | #endif // BEMBEL_SRC_H2MATRIX_EIGENHELPER_H2DENSEPRODUCT_HPP_ | ||
317 |