renamed inverseProduct => solveTriangular

This commit is contained in:
Gael Guennebaud 2008-08-09 20:06:25 +00:00
parent d6e88f8155
commit b13148c358
16 changed files with 90 additions and 72 deletions

View File

@ -44,7 +44,7 @@ namespace Eigen {
#include "src/Core/Dot.h" #include "src/Core/Dot.h"
#include "src/Core/Product.h" #include "src/Core/Product.h"
#include "src/Core/DiagonalProduct.h" #include "src/Core/DiagonalProduct.h"
#include "src/Core/InverseProduct.h" #include "src/Core/SolveTriangular.h"
#include "src/Core/MapBase.h" #include "src/Core/MapBase.h"
#include "src/Core/Map.h" #include "src/Core/Map.h"
#include "src/Core/Block.h" #include "src/Core/Block.h"

View File

@ -141,7 +141,7 @@ typename Derived::Eval Cholesky<MatrixType>::solve(const MatrixBase<Derived> &b)
const int size = m_matrix.rows(); const int size = m_matrix.rows();
ei_assert(size==b.rows()); ei_assert(size==b.rows());
return m_matrix.adjoint().template part<Upper>().inverseProduct(matrixL().inverseProduct(b)); return m_matrix.adjoint().template part<Upper>().solveTriangular(matrixL().solveTriangular(b));
} }
/** \cholesky_module /** \cholesky_module

View File

@ -148,9 +148,9 @@ typename Derived::Eval CholeskyWithoutSquareRoot<MatrixType>::solve(const Matrix
ei_assert(size==b.rows()); ei_assert(size==b.rows());
return m_matrix.adjoint().template part<UnitUpper>() return m_matrix.adjoint().template part<UnitUpper>()
.inverseProduct( .solveTriangular(
(matrixL() (matrixL()
.inverseProduct(b)) .solveTriangular(b))
.cwise()/m_matrix.diagonal() .cwise()/m_matrix.diagonal()
); );
} }

View File

@ -320,10 +320,10 @@ template<typename Derived> class MatrixBase
Derived& operator*=(const MatrixBase<OtherDerived>& other); Derived& operator*=(const MatrixBase<OtherDerived>& other);
template<typename OtherDerived> template<typename OtherDerived>
typename OtherDerived::Eval inverseProduct(const MatrixBase<OtherDerived>& other) const; typename OtherDerived::Eval solveTriangular(const MatrixBase<OtherDerived>& other) const;
template<typename OtherDerived> template<typename OtherDerived>
void inverseProductInPlace(MatrixBase<OtherDerived>& other) const; void solveTriangularInPlace(MatrixBase<OtherDerived>& other) const;
template<typename OtherDerived> template<typename OtherDerived>

View File

@ -212,13 +212,13 @@ struct ei_trisolve_selector<Lhs,Rhs,Upper,ColMajor>
} }
}; };
/** "in-place" version of MatrixBase::inverseProduct() where the result is written in \a other /** "in-place" version of MatrixBase::solveTriangular() where the result is written in \a other
* *
* \sa inverseProduct() * \sa solveTriangular()
*/ */
template<typename Derived> template<typename Derived>
template<typename OtherDerived> template<typename OtherDerived>
void MatrixBase<Derived>::inverseProductInPlace(MatrixBase<OtherDerived>& other) const void MatrixBase<Derived>::solveTriangularInPlace(MatrixBase<OtherDerived>& other) const
{ {
ei_assert(derived().cols() == derived().rows()); ei_assert(derived().cols() == derived().rows());
ei_assert(derived().cols() == other.rows()); ei_assert(derived().cols() == other.rows());
@ -244,10 +244,10 @@ void MatrixBase<Derived>::inverseProductInPlace(MatrixBase<OtherDerived>& other)
*/ */
template<typename Derived> template<typename Derived>
template<typename OtherDerived> template<typename OtherDerived>
typename OtherDerived::Eval MatrixBase<Derived>::inverseProduct(const MatrixBase<OtherDerived>& other) const typename OtherDerived::Eval MatrixBase<Derived>::solveTriangular(const MatrixBase<OtherDerived>& other) const
{ {
typename OtherDerived::Eval res(other); typename OtherDerived::Eval res(other);
inverseProductInPlace(res); solveTriangularInPlace(res);
return res; return res;
} }

View File

@ -257,7 +257,7 @@ void LU<MatrixType>::computeKernel(Matrix<typename MatrixType::Scalar,
m_lu.corner(TopLeft, m_rank, m_rank) m_lu.corner(TopLeft, m_rank, m_rank)
.template marked<Upper>() .template marked<Upper>()
.inverseProductInPlace(y); .solveTriangularInPlace(y);
for(int i = 0; i < m_rank; i++) for(int i = 0; i < m_rank; i++)
result->row(m_q.coeff(i)) = y.row(i); result->row(m_q.coeff(i)) = y.row(i);
@ -309,7 +309,7 @@ bool LU<MatrixType>::solve(
l.setZero(); l.setZero();
l.corner(Eigen::TopLeft,rows,smalldim) l.corner(Eigen::TopLeft,rows,smalldim)
= m_lu.corner(Eigen::TopLeft,rows,smalldim); = m_lu.corner(Eigen::TopLeft,rows,smalldim);
l.template marked<UnitLower>().inverseProductInPlace(c); l.template marked<UnitLower>().solveTriangularInPlace(c);
// Step 3 // Step 3
if(!isSurjective()) if(!isSurjective())
@ -326,7 +326,7 @@ bool LU<MatrixType>::solve(
d(c.corner(TopLeft, m_rank, c.cols())); d(c.corner(TopLeft, m_rank, c.cols()));
m_lu.corner(TopLeft, m_rank, m_rank) m_lu.corner(TopLeft, m_rank, m_rank)
.template marked<Upper>() .template marked<Upper>()
.inverseProductInPlace(d); .solveTriangularInPlace(d);
// Step 4 // Step 4
result->resize(m_lu.cols(), b.cols()); result->resize(m_lu.cols(), b.cols());

View File

@ -230,17 +230,17 @@ compute(const MatrixType& matA, const MatrixType& matB, bool computeEigenvectors
Cholesky<MatrixType> cholB(matB); Cholesky<MatrixType> cholB(matB);
// compute C = inv(U') A inv(U) // compute C = inv(U') A inv(U)
MatrixType matC = cholB.matrixL().inverseProduct(matA); MatrixType matC = cholB.matrixL().solveTriangular(matA);
// FIXME since we currently do not support A * inv(U), // FIXME since we currently do not support A * inv(U),
// let's do (inv(U') A')' : // let's do (inv(U') A')' :
matC = (cholB.matrixL().inverseProduct(matC.adjoint())).adjoint(); matC = (cholB.matrixL().solveTriangular(matC.adjoint())).adjoint();
compute(matC, computeEigenvectors); compute(matC, computeEigenvectors);
if (computeEigenvectors) if (computeEigenvectors)
{ {
// transform back the eigen vectors: evecs = inv(U) * evecs // transform back the eigen vectors: evecs = inv(U) * evecs
m_eivec = cholB.matrixL().adjoint().template marked<Upper>().inverseProduct(m_eivec); m_eivec = cholB.matrixL().adjoint().template marked<Upper>().solveTriangular(m_eivec);
} }
} }

View File

@ -142,7 +142,7 @@ class SparseMatrixBase : public MatrixBase<Derived>
} }
template<typename OtherDerived> template<typename OtherDerived>
OtherDerived inverseProduct(const MatrixBase<OtherDerived>& other) const; OtherDerived solveTriangular(const MatrixBase<OtherDerived>& other) const;
protected: protected:

View File

@ -155,7 +155,7 @@ struct ei_sparse_trisolve_selector<Lhs,Rhs,Upper,ColMajor>
template<typename Derived> template<typename Derived>
template<typename OtherDerived> template<typename OtherDerived>
OtherDerived SparseMatrixBase<Derived>::inverseProduct(const MatrixBase<OtherDerived>& other) const OtherDerived SparseMatrixBase<Derived>::solveTriangular(const MatrixBase<OtherDerived>& other) const
{ {
ei_assert(derived().cols() == other.rows()); ei_assert(derived().cols() == other.rows());
ei_assert(!(Flags & ZeroDiagBit)); ei_assert(!(Flags & ZeroDiagBit));

View File

@ -18,7 +18,7 @@ using namespace Eigen;
#endif #endif
#ifndef TRIES #ifndef TRIES
#define TRIES 4 #define TRIES 10
#endif #endif
typedef float Scalar; typedef float Scalar;
@ -29,6 +29,13 @@ __attribute__ ((noinline)) void benchCholesky(const MatrixType& m)
int rows = m.rows(); int rows = m.rows();
int cols = m.cols(); int cols = m.cols();
int cost = 0;
for (int j=0; j<rows; ++j)
{
int r = std::max(rows - j -1,0);
cost += 2*(r*j+r+j);
}
int repeats = (REPEAT*1000)/(rows*rows); int repeats = (REPEAT*1000)/(rows*rows);
typedef typename MatrixType::Scalar Scalar; typedef typename MatrixType::Scalar Scalar;
@ -70,7 +77,8 @@ __attribute__ ((noinline)) void benchCholesky(const MatrixType& m)
std::cout << "fixed "; std::cout << "fixed ";
std::cout << covMat.rows() << " \t" std::cout << covMat.rows() << " \t"
<< (timerNoSqrt.value() * REPEAT) / repeats << "s \t" << (timerNoSqrt.value() * REPEAT) / repeats << "s \t"
<< (timerSqrt.value() * REPEAT) / repeats << "s"; << (timerSqrt.value() * REPEAT) / repeats << "s "
<< "(" << 1e-6 * cost*repeats/timerSqrt.value() << " MFLOPS)\n";
#ifdef BENCH_GSL #ifdef BENCH_GSL
@ -108,7 +116,7 @@ __attribute__ ((noinline)) void benchCholesky(const MatrixType& m)
int main(int argc, char* argv[]) int main(int argc, char* argv[])
{ {
const int dynsizes[] = {/*4,6,8,12,16,24,32,49,64,67,128,129,130,131,132,*/256,257,258,259,260,512,0}; const int dynsizes[] = {/*4,6,8,12,16,24,32,49,64,67,128,129,130,131,132,*/256,257,258,259,260,512,900,0};
std::cout << "size no sqrt standard"; std::cout << "size no sqrt standard";
#ifdef BENCH_GSL #ifdef BENCH_GSL
std::cout << " GSL (standard + double + ATLAS) "; std::cout << " GSL (standard + double + ATLAS) ";

View File

@ -22,7 +22,7 @@ int main(int argc, char* argv[])
int size = SIZE * 8; int size = SIZE * 8;
int size2 = size * size; int size2 = size * size;
Scalar* a = ei_aligned_malloc<Scalar>(size2); Scalar* a = ei_aligned_malloc<Scalar>(size2);
Scalar* b = ei_aligned_malloc<Scalar>(size2); Scalar* b = ei_aligned_malloc<Scalar>(size2+4)+1;
Scalar* c = ei_aligned_malloc<Scalar>(size2); Scalar* c = ei_aligned_malloc<Scalar>(size2);
for (int i=0; i<size; ++i) for (int i=0; i<size; ++i)
@ -33,22 +33,22 @@ int main(int argc, char* argv[])
BenchTimer timer; BenchTimer timer;
timer.reset(); timer.reset();
for (int k=0; k<3; ++k) for (int k=0; k<10; ++k)
{ {
timer.start(); timer.start();
benchVec(a, b, c, size2); benchVec(a, b, c, size2);
timer.stop(); timer.stop();
} }
std::cout << timer.value() << "s " << (double(size2*REPEAT)/timer.value())/(1024.*1024.*1024.) << " GFlops\n"; std::cout << timer.value() << "s " << (double(size2*REPEAT)/timer.value())/(1024.*1024.*1024.) << " GFlops\n";
return 0;
for (int innersize = size; innersize>2 ; --innersize) for (int innersize = size; innersize>2 ; --innersize)
{ {
if (size2%innersize==0) if (size2%innersize==0)
{ {
int outersize = size2/innersize; int outersize = size2/innersize;
MatrixXf ma = MatrixXf::map(a, innersize, outersize ); MatrixXf ma = Map<MatrixXf>(a, innersize, outersize );
MatrixXf mb = MatrixXf::map(b, innersize, outersize ); MatrixXf mb = Map<MatrixXf>(b, innersize, outersize );
MatrixXf mc = MatrixXf::map(c, innersize, outersize ); MatrixXf mc = Map<MatrixXf>(c, innersize, outersize );
timer.reset(); timer.reset();
for (int k=0; k<3; ++k) for (int k=0; k<3; ++k)
{ {
@ -60,9 +60,9 @@ int main(int argc, char* argv[])
} }
} }
VectorXf va = VectorXf::map(a, size2); VectorXf va = Map<VectorXf>(a, size2);
VectorXf vb = VectorXf::map(b, size2); VectorXf vb = Map<VectorXf>(b, size2);
VectorXf vc = VectorXf::map(c, size2); VectorXf vc = Map<VectorXf>(c, size2);
timer.reset(); timer.reset();
for (int k=0; k<3; ++k) for (int k=0; k<3; ++k)
{ {
@ -95,40 +95,40 @@ void benchVec(Scalar* a, Scalar* b, Scalar* c, int size)
for (int k=0; k<REPEAT; ++k) for (int k=0; k<REPEAT; ++k)
for (int i=0; i<size; i+=PacketSize*8) for (int i=0; i<size; i+=PacketSize*8)
{ {
a0 = ei_pload(&a[i]); // a0 = ei_pload(&a[i]);
b0 = ei_pload(&b[i]); // b0 = ei_pload(&b[i]);
a1 = ei_pload(&a[i+1*PacketSize]); // a1 = ei_pload(&a[i+1*PacketSize]);
b1 = ei_pload(&b[i+1*PacketSize]); // b1 = ei_pload(&b[i+1*PacketSize]);
a2 = ei_pload(&a[i+2*PacketSize]); // a2 = ei_pload(&a[i+2*PacketSize]);
b2 = ei_pload(&b[i+2*PacketSize]); // b2 = ei_pload(&b[i+2*PacketSize]);
a3 = ei_pload(&a[i+3*PacketSize]); // a3 = ei_pload(&a[i+3*PacketSize]);
b3 = ei_pload(&b[i+3*PacketSize]); // b3 = ei_pload(&b[i+3*PacketSize]);
ei_pstore(&a[i], ei_padd(a0, b0)); // ei_pstore(&a[i], ei_padd(a0, b0));
a0 = ei_pload(&a[i+4*PacketSize]); // a0 = ei_pload(&a[i+4*PacketSize]);
b0 = ei_pload(&b[i+4*PacketSize]); // b0 = ei_pload(&b[i+4*PacketSize]);
//
// ei_pstore(&a[i+1*PacketSize], ei_padd(a1, b1));
// a1 = ei_pload(&a[i+5*PacketSize]);
// b1 = ei_pload(&b[i+5*PacketSize]);
//
// ei_pstore(&a[i+2*PacketSize], ei_padd(a2, b2));
// a2 = ei_pload(&a[i+6*PacketSize]);
// b2 = ei_pload(&b[i+6*PacketSize]);
//
// ei_pstore(&a[i+3*PacketSize], ei_padd(a3, b3));
// a3 = ei_pload(&a[i+7*PacketSize]);
// b3 = ei_pload(&b[i+7*PacketSize]);
//
// ei_pstore(&a[i+4*PacketSize], ei_padd(a0, b0));
// ei_pstore(&a[i+5*PacketSize], ei_padd(a1, b1));
// ei_pstore(&a[i+6*PacketSize], ei_padd(a2, b2));
// ei_pstore(&a[i+7*PacketSize], ei_padd(a3, b3));
ei_pstore(&a[i+1*PacketSize], ei_padd(a1, b1)); ei_pstore(&a[i+2*PacketSize], ei_padd(ei_ploadu(&a[i+2*PacketSize]), ei_ploadu(&b[i+2*PacketSize])));
a1 = ei_pload(&a[i+5*PacketSize]); ei_pstore(&a[i+3*PacketSize], ei_padd(ei_ploadu(&a[i+3*PacketSize]), ei_ploadu(&b[i+3*PacketSize])));
b1 = ei_pload(&b[i+5*PacketSize]); ei_pstore(&a[i+4*PacketSize], ei_padd(ei_ploadu(&a[i+4*PacketSize]), ei_ploadu(&b[i+4*PacketSize])));
ei_pstore(&a[i+5*PacketSize], ei_padd(ei_ploadu(&a[i+5*PacketSize]), ei_ploadu(&b[i+5*PacketSize])));
ei_pstore(&a[i+2*PacketSize], ei_padd(a2, b2)); ei_pstore(&a[i+6*PacketSize], ei_padd(ei_ploadu(&a[i+6*PacketSize]), ei_ploadu(&b[i+6*PacketSize])));
a2 = ei_pload(&a[i+6*PacketSize]); ei_pstore(&a[i+7*PacketSize], ei_padd(ei_ploadu(&a[i+7*PacketSize]), ei_ploadu(&b[i+7*PacketSize])));
b2 = ei_pload(&b[i+6*PacketSize]);
ei_pstore(&a[i+3*PacketSize], ei_padd(a3, b3));
a3 = ei_pload(&a[i+7*PacketSize]);
b3 = ei_pload(&b[i+7*PacketSize]);
ei_pstore(&a[i+4*PacketSize], ei_padd(a0, b0));
ei_pstore(&a[i+5*PacketSize], ei_padd(a1, b1));
ei_pstore(&a[i+6*PacketSize], ei_padd(a2, b2));
ei_pstore(&a[i+7*PacketSize], ei_padd(a3, b3));
// ei_pstore(&a[i+2*PacketSize], ei_padd(ei_pload(&a[i+2*PacketSize]), ei_pload(&b[i+2*PacketSize])));
// ei_pstore(&a[i+3*PacketSize], ei_padd(ei_pload(&a[i+3*PacketSize]), ei_pload(&b[i+3*PacketSize])));
// ei_pstore(&a[i+4*PacketSize], ei_padd(ei_pload(&a[i+4*PacketSize]), ei_pload(&b[i+4*PacketSize])));
// ei_pstore(&a[i+5*PacketSize], ei_padd(ei_pload(&a[i+5*PacketSize]), ei_pload(&b[i+5*PacketSize])));
// ei_pstore(&a[i+6*PacketSize], ei_padd(ei_pload(&a[i+6*PacketSize]), ei_pload(&b[i+6*PacketSize])));
// ei_pstore(&a[i+7*PacketSize], ei_padd(ei_pload(&a[i+7*PacketSize]), ei_pload(&b[i+7*PacketSize])));
} }
} }

View File

@ -23,7 +23,6 @@
#include "action_matrix_vector_product.hh" #include "action_matrix_vector_product.hh"
#include "action_matrix_matrix_product.hh" #include "action_matrix_matrix_product.hh"
#include "action_axpy.hh" #include "action_axpy.hh"
#include "timers/x86_perf_analyzer.hh"
BTL_MAIN; BTL_MAIN;

View File

@ -134,7 +134,7 @@ public :
} }
static inline void trisolve_lower(const gene_matrix & L, const gene_vector& B, gene_vector& X, int N){ static inline void trisolve_lower(const gene_matrix & L, const gene_vector& B, gene_vector& X, int N){
X = L.template marked<Lower>().inverseProduct(B); X = L.template marked<Lower>().solveTriangular(B);
} }
static inline void cholesky(const gene_matrix & X, gene_matrix & C, int N){ static inline void cholesky(const gene_matrix & X, gene_matrix & C, int N){
@ -146,6 +146,7 @@ public :
static inline void lu_decomp(const gene_matrix & X, gene_matrix & C, int N){ static inline void lu_decomp(const gene_matrix & X, gene_matrix & C, int N){
C = X.lu().matrixLU(); C = X.lu().matrixLU();
// C = X.inverse();
} }
static inline void tridiagonalization(const gene_matrix & X, gene_matrix & C, int N){ static inline void tridiagonalization(const gene_matrix & X, gene_matrix & C, int N){

View File

@ -6,4 +6,4 @@ n.part<Eigen::Lower>() *= 2;
cout << "Here is the matrix n:" << endl << n << endl; cout << "Here is the matrix n:" << endl << n << endl;
cout << "And now here is m.inverse()*n, taking advantage of the fact that" cout << "And now here is m.inverse()*n, taking advantage of the fact that"
" m is upper-triangular:" << endl " m is upper-triangular:" << endl
<< m.marked<Eigen::Upper>().inverseProduct(n); << m.marked<Eigen::Upper>().solveTriangular(n);

View File

@ -66,6 +66,7 @@ template<typename MatrixType> void submatrices(const MatrixType& m)
m2 = MatrixType::Random(rows, cols), m2 = MatrixType::Random(rows, cols),
m3(rows, cols), m3(rows, cols),
mzero = MatrixType::Zero(rows, cols), mzero = MatrixType::Zero(rows, cols),
ones = MatrixType::Ones(rows, cols),
identity = Matrix<Scalar, MatrixType::RowsAtCompileTime, MatrixType::RowsAtCompileTime> identity = Matrix<Scalar, MatrixType::RowsAtCompileTime, MatrixType::RowsAtCompileTime>
::Identity(rows, rows), ::Identity(rows, rows),
square = Matrix<Scalar, MatrixType::RowsAtCompileTime, MatrixType::RowsAtCompileTime> square = Matrix<Scalar, MatrixType::RowsAtCompileTime, MatrixType::RowsAtCompileTime>
@ -121,6 +122,14 @@ template<typename MatrixType> void submatrices(const MatrixType& m)
Matrix<Scalar,Dynamic,Dynamic> b = m1.template block<BlockRows,BlockCols>(3,3); Matrix<Scalar,Dynamic,Dynamic> b = m1.template block<BlockRows,BlockCols>(3,3);
VERIFY_IS_APPROX(b, m1.block(3,3,BlockRows,BlockCols)); VERIFY_IS_APPROX(b, m1.block(3,3,BlockRows,BlockCols));
} }
// stress some basic stuffs with block matrices
VERIFY(ones.col(c1).sum() == Scalar(rows));
VERIFY(ones.row(r1).sum() == Scalar(cols));
VERIFY(ones.col(c1).dot(ones.col(c2)) == Scalar(rows));
std::cerr << ones.row(r1).dot(ones.row(r2)) << " == " << cols << "\n";
VERIFY(ones.row(r1).dot(ones.row(r2)) == Scalar(cols));
} }
void test_submatrices() void test_submatrices()
@ -131,5 +140,6 @@ void test_submatrices()
CALL_SUBTEST( submatrices(MatrixXcf(3, 3)) ); CALL_SUBTEST( submatrices(MatrixXcf(3, 3)) );
CALL_SUBTEST( submatrices(MatrixXi(8, 12)) ); CALL_SUBTEST( submatrices(MatrixXi(8, 12)) );
CALL_SUBTEST( submatrices(MatrixXcd(20, 20)) ); CALL_SUBTEST( submatrices(MatrixXcd(20, 20)) );
CALL_SUBTEST( submatrices(MatrixXf(20, 20)) );
} }
} }

View File

@ -80,15 +80,15 @@ template<typename MatrixType> void triangular(const MatrixType& m)
// test back and forward subsitution // test back and forward subsitution
m3 = m1.template part<Eigen::Lower>(); m3 = m1.template part<Eigen::Lower>();
VERIFY(m3.template marked<Eigen::Lower>().inverseProduct(m3).cwise().abs().isIdentity(test_precision<RealScalar>())); VERIFY(m3.template marked<Eigen::Lower>().solveTriangular(m3).cwise().abs().isIdentity(test_precision<RealScalar>()));
m3 = m1.template part<Eigen::Upper>(); m3 = m1.template part<Eigen::Upper>();
VERIFY(m3.template marked<Eigen::Upper>().inverseProduct(m3).cwise().abs().isIdentity(test_precision<RealScalar>())); VERIFY(m3.template marked<Eigen::Upper>().solveTriangular(m3).cwise().abs().isIdentity(test_precision<RealScalar>()));
// FIXME these tests failed due to numerical issues // FIXME these tests failed due to numerical issues
// m1 = MatrixType::Random(rows, cols); // m1 = MatrixType::Random(rows, cols);
// VERIFY_IS_APPROX(m1.template part<Eigen::Upper>().eval() * (m1.template part<Eigen::Upper>().inverseProduct(m2)), m2); // VERIFY_IS_APPROX(m1.template part<Eigen::Upper>().eval() * (m1.template part<Eigen::Upper>().solveTriangular(m2)), m2);
// VERIFY_IS_APPROX(m1.template part<Eigen::Lower>().eval() * (m1.template part<Eigen::Lower>().inverseProduct(m2)), m2); // VERIFY_IS_APPROX(m1.template part<Eigen::Lower>().eval() * (m1.template part<Eigen::Lower>().solveTriangular(m2)), m2);
VERIFY((m1.template part<Eigen::Upper>() * m2.template part<Eigen::Upper>()).isUpper()); VERIFY((m1.template part<Eigen::Upper>() * m2.template part<Eigen::Upper>()).isUpper());