mirror of
https://gitlab.com/libeigen/eigen.git
synced 2026-04-10 11:34:33 +08:00
186 lines
6.9 KiB
C++
186 lines
6.9 KiB
C++
// 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;
|
||
|
||
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);
|
||
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);
|
||
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>());
|
||
}
|