Files
eigen/test/diagonal.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

186 lines
6.9 KiB
C++
Raw Normal View History

// This file is part of Eigen, a lightweight C++ template library
// for linear algebra.
//
// Copyright (C) 2006-2010 Benoit Jacob <jacob.benoit.1@gmail.com>
//
// This Source Code Form is subject to the terms of the Mozilla
// Public License v. 2.0. If a copy of the MPL was not distributed
// with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
#include "main.h"
template <typename MatrixType>
void diagonal(const MatrixType& m) {
typedef typename MatrixType::Scalar Scalar;
2010-06-20 17:37:56 +02:00
Index rows = m.rows();
Index cols = m.cols();
MatrixType m1 = MatrixType::Random(rows, cols), m2 = MatrixType::Random(rows, cols);
Scalar s1 = internal::random<Scalar>();
// check diagonal()
VERIFY_IS_APPROX(m1.diagonal(), m1.transpose().diagonal());
m2.diagonal() = 2 * m1.diagonal();
m2.diagonal()[0] *= 3;
if (rows > 2) {
enum { N1 = MatrixType::RowsAtCompileTime > 2 ? 2 : 0, N2 = MatrixType::RowsAtCompileTime > 1 ? -1 : 0 };
// check sub/super diagonal
if (MatrixType::SizeAtCompileTime != Dynamic) {
VERIFY(m1.template diagonal<N1>().RowsAtCompileTime == m1.diagonal(N1).size());
VERIFY(m1.template diagonal<N2>().RowsAtCompileTime == m1.diagonal(N2).size());
}
m2.template diagonal<N1>() = 2 * m1.template diagonal<N1>();
VERIFY_IS_APPROX(m2.template diagonal<N1>(), static_cast<Scalar>(2) * m1.diagonal(N1));
m2.template diagonal<N1>()[0] *= 3;
VERIFY_IS_APPROX(m2.template diagonal<N1>()[0], static_cast<Scalar>(6) * m1.template diagonal<N1>()[0]);
m2.template diagonal<N2>() = 2 * m1.template diagonal<N2>();
m2.template diagonal<N2>()[0] *= 3;
VERIFY_IS_APPROX(m2.template diagonal<N2>()[0], static_cast<Scalar>(6) * m1.template diagonal<N2>()[0]);
m2.diagonal(N1) = 2 * m1.diagonal(N1);
2013-03-01 00:03:28 +01:00
VERIFY_IS_APPROX(m2.template diagonal<N1>(), static_cast<Scalar>(2) * m1.diagonal(N1));
m2.diagonal(N1)[0] *= 3;
VERIFY_IS_APPROX(m2.diagonal(N1)[0], static_cast<Scalar>(6) * m1.diagonal(N1)[0]);
m2.diagonal(N2) = 2 * m1.diagonal(N2);
2013-03-01 00:03:28 +01:00
VERIFY_IS_APPROX(m2.template diagonal<N2>(), static_cast<Scalar>(2) * m1.diagonal(N2));
m2.diagonal(N2)[0] *= 3;
VERIFY_IS_APPROX(m2.diagonal(N2)[0], static_cast<Scalar>(6) * m1.diagonal(N2)[0]);
m2.diagonal(N2).x() = s1;
VERIFY_IS_APPROX(m2.diagonal(N2).x(), s1);
m2.diagonal(N2).coeffRef(0) = Scalar(2) * s1;
VERIFY_IS_APPROX(m2.diagonal(N2).coeff(0), Scalar(2) * s1);
}
VERIFY(m1.diagonal(cols).size() == 0);
VERIFY(m1.diagonal(-rows).size() == 0);
}
template <typename MatrixType>
void diagonal_assert(const MatrixType& m) {
Index rows = m.rows();
Index cols = m.cols();
MatrixType m1 = MatrixType::Random(rows, cols);
if (rows >= 2 && cols >= 2) {
VERIFY_RAISES_ASSERT(m1 += m1.diagonal());
VERIFY_RAISES_ASSERT(m1 -= m1.diagonal());
VERIFY_RAISES_ASSERT(m1.array() *= m1.diagonal().array());
VERIFY_RAISES_ASSERT(m1.array() /= m1.diagonal().array());
}
VERIFY_RAISES_ASSERT(m1.diagonal(cols + 1));
VERIFY_RAISES_ASSERT(m1.diagonal(-(rows + 1)));
}
// Test that (A * B).diagonal() gives the same result as (A * B).eval().diagonal().
// The diagonal-of-product path uses LazyProduct evaluation (see ProductEvaluators.h),
// which avoids computing the full product. Verify this optimization is correct.
template <typename Scalar>
void diagonal_of_product() {
const Index PS = internal::packet_traits<Scalar>::size;
const Index sizes[] = {1, 2, 3, PS - 1, PS, PS + 1, 2 * PS - 1, 2 * PS, 2 * PS + 1, 4 * PS, 4 * PS + 1};
typedef Matrix<Scalar, Dynamic, Dynamic> Mat;
typedef Matrix<Scalar, Dynamic, 1> Vec;
for (int si = 0; si < 11; ++si) {
Index n = sizes[si];
if (n <= 0) continue;
Mat A = Mat::Random(n, n);
Mat B = Mat::Random(n, n);
// Lazy diagonal vs explicit product diagonal
Vec diag_lazy = (A * B).diagonal();
Vec diag_explicit = (A * B).eval().diagonal();
VERIFY_IS_APPROX(diag_lazy, diag_explicit);
// Also test non-square: A is m×k, B is k×n
for (int k : {1, 3, (int)n}) {
if (k <= 0) continue;
Mat C = Mat::Random(n, k);
Mat D = Mat::Random(k, n);
Vec diag_lazy2 = (C * D).diagonal();
Vec diag_explicit2 = (C * D).eval().diagonal();
VERIFY_IS_APPROX(diag_lazy2, diag_explicit2);
}
}
}
// Test .select() at vectorization boundary sizes.
// select() uses CwiseTernaryOp which has packet-level evaluation with remainder handling.
template <typename Scalar>
void select_boundary() {
const Index PS = internal::packet_traits<Scalar>::size;
const Index sizes[] = {1, 2, 3, PS - 1, PS, PS + 1, 2 * PS - 1, 2 * PS, 2 * PS + 1, 4 * PS, 4 * PS + 1};
typedef Array<Scalar, Dynamic, 1> Arr;
for (int si = 0; si < 11; ++si) {
Index n = sizes[si];
if (n <= 0) continue;
Arr a = Arr::Random(n);
Arr b = Arr::Random(n);
auto cond = (a > Scalar(0));
// select with two arrays
Arr result = cond.select(a, b);
for (Index k = 0; k < n; ++k) {
Scalar expected = (a(k) > Scalar(0)) ? a(k) : b(k);
VERIFY_IS_APPROX(result(k), expected);
}
// select with scalar else
Arr result2 = cond.select(a, Scalar(0));
for (Index k = 0; k < n; ++k) {
Scalar expected = (a(k) > Scalar(0)) ? a(k) : Scalar(0);
VERIFY_IS_APPROX(result2(k), expected);
}
// select with scalar then
Arr result3 = cond.select(Scalar(42), b);
for (Index k = 0; k < n; ++k) {
Scalar expected = (a(k) > Scalar(0)) ? Scalar(42) : b(k);
VERIFY_IS_APPROX(result3(k), expected);
}
}
}
EIGEN_DECLARE_TEST(diagonal) {
for (int i = 0; i < g_repeat; i++) {
CALL_SUBTEST_1(diagonal(Matrix<float, 1, 1>()));
CALL_SUBTEST_1(diagonal(Matrix<float, 4, 9>()));
CALL_SUBTEST_1(diagonal(Matrix<float, 7, 3>()));
CALL_SUBTEST_2(diagonal(Matrix4d()));
CALL_SUBTEST_2(diagonal(
MatrixXcf(internal::random<int>(1, EIGEN_TEST_MAX_SIZE), internal::random<int>(1, EIGEN_TEST_MAX_SIZE))));
CALL_SUBTEST_2(diagonal(
MatrixXi(internal::random<int>(1, EIGEN_TEST_MAX_SIZE), internal::random<int>(1, EIGEN_TEST_MAX_SIZE))));
CALL_SUBTEST_2(diagonal(
MatrixXcd(internal::random<int>(1, EIGEN_TEST_MAX_SIZE), internal::random<int>(1, EIGEN_TEST_MAX_SIZE))));
CALL_SUBTEST_1(diagonal(
MatrixXf(internal::random<int>(1, EIGEN_TEST_MAX_SIZE), internal::random<int>(1, EIGEN_TEST_MAX_SIZE))));
CALL_SUBTEST_1(diagonal(Matrix<float, Dynamic, 4>(3, 4)));
CALL_SUBTEST_1(diagonal_assert(
MatrixXf(internal::random<int>(1, EIGEN_TEST_MAX_SIZE), internal::random<int>(1, EIGEN_TEST_MAX_SIZE))));
}
// Diagonal-of-product optimization (deterministic, outside g_repeat).
CALL_SUBTEST_3(diagonal_of_product<float>());
CALL_SUBTEST_3(diagonal_of_product<double>());
CALL_SUBTEST_3(diagonal_of_product<std::complex<float>>());
// Select at vectorization boundaries (deterministic, outside g_repeat).
CALL_SUBTEST_4(select_boundary<float>());
CALL_SUBTEST_4(select_boundary<double>());
CALL_SUBTEST_4(select_boundary<int>());
}