mirror of
https://gitlab.com/libeigen/eigen.git
synced 2025-03-01 18:26:24 +08:00
285 lines
13 KiB
C++
285 lines
13 KiB
C++
// This file is part of Eigen, a lightweight C++ template library
|
|
// for linear algebra.
|
|
//
|
|
// Copyright (C) 2013 Hauke Heibel <hauke.heibel@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/.
|
|
|
|
#define EIGEN_TESTING_PLAINOBJECT_CTOR
|
|
|
|
#include "main.h"
|
|
#include "AnnoyingScalar.h"
|
|
#include "MovableScalar.h"
|
|
#include "SafeScalar.h"
|
|
|
|
#include <Eigen/Core>
|
|
|
|
using DenseStorageD3x3 = Eigen::DenseStorage<double, 9, 3, 3, 0>;
|
|
#if !defined(EIGEN_DENSE_STORAGE_CTOR_PLUGIN)
|
|
static_assert(std::is_trivially_copy_constructible<DenseStorageD3x3>::value,
|
|
"DenseStorage not trivially_copy_constructible");
|
|
static_assert(std::is_trivially_move_constructible<DenseStorageD3x3>::value,
|
|
"DenseStorage not trivially_move_constructible");
|
|
static_assert(std::is_trivially_copy_assignable<DenseStorageD3x3>::value, "DenseStorage not trivially_copy_assignable");
|
|
static_assert(std::is_trivially_move_assignable<DenseStorageD3x3>::value, "DenseStorage not trivially_move_assignable");
|
|
#endif
|
|
// all plain object types conform to standard layout
|
|
static_assert(std::is_standard_layout<Matrix4f>::value, "Matrix4f not standard_layout");
|
|
static_assert(std::is_standard_layout<Array4f>::value, "Array4f not standard_layout");
|
|
static_assert(std::is_standard_layout<VectorXf>::value, "VectorXf not standard_layout");
|
|
static_assert(std::is_standard_layout<ArrayXf>::value, "ArrayXf not standard_layout");
|
|
static_assert(std::is_standard_layout<MatrixXf>::value, "MatrixXf not standard_layout");
|
|
static_assert(std::is_standard_layout<ArrayXXf>::value, "ArrayXXf not standard_layout");
|
|
// all fixed-size, fixed-dimension plain object types are trivially default constructible
|
|
static_assert(std::is_trivially_default_constructible<Matrix4f>::value, "Matrix4f not trivially_default_constructible");
|
|
static_assert(std::is_trivially_default_constructible<Array4f>::value, "Array4f not trivially_default_constructible");
|
|
// all fixed-size, fixed-dimension plain object types are trivially move constructible
|
|
static_assert(std::is_trivially_move_constructible<Matrix4f>::value, "Matrix4f not trivially_move_constructible");
|
|
static_assert(std::is_trivially_move_constructible<Array4f>::value, "Array4f not trivially_move_constructible");
|
|
// all statically-allocated plain object types are trivially destructible
|
|
static_assert(std::is_trivially_destructible<Matrix4f>::value, "Matrix4f not trivially_destructible");
|
|
static_assert(std::is_trivially_destructible<Array4f>::value, "Array4f not trivially_destructible");
|
|
static_assert(std::is_trivially_destructible<Matrix<float, 4, Dynamic, 0, 4, 4>>::value,
|
|
"Matrix4X44 not trivially_destructible");
|
|
static_assert(std::is_trivially_destructible<Array<float, 4, Dynamic, 0, 4, 4>>::value,
|
|
"Array4X44 not trivially_destructible");
|
|
#if !defined(EIGEN_DENSE_STORAGE_CTOR_PLUGIN)
|
|
// all fixed-size, fixed-dimension plain object types are trivially copy constructible
|
|
static_assert(std::is_trivially_copy_constructible<Matrix4f>::value, "Matrix4f not trivially_copy_constructible");
|
|
static_assert(std::is_trivially_copy_constructible<Array4f>::value, "Array4f not trivially_copy_constructible");
|
|
#endif
|
|
|
|
template <typename T, int Size, int Rows, int Cols>
|
|
void dense_storage_copy(int rows, int cols) {
|
|
typedef DenseStorage<T, Size, Rows, Cols, 0> DenseStorageType;
|
|
|
|
const int size = rows * cols;
|
|
DenseStorageType reference(size, rows, cols);
|
|
T* raw_reference = reference.data();
|
|
for (int i = 0; i < size; ++i) raw_reference[i] = internal::random<T>();
|
|
|
|
DenseStorageType copied_reference(reference);
|
|
const T* raw_copied_reference = copied_reference.data();
|
|
for (int i = 0; i < size; ++i) VERIFY_IS_EQUAL(raw_reference[i], raw_copied_reference[i]);
|
|
}
|
|
|
|
template <typename T, int Size, int Rows, int Cols>
|
|
void dense_storage_assignment(int rows, int cols) {
|
|
typedef DenseStorage<T, Size, Rows, Cols, 0> DenseStorageType;
|
|
|
|
const int size = rows * cols;
|
|
DenseStorageType reference(size, rows, cols);
|
|
T* raw_reference = reference.data();
|
|
for (int i = 0; i < size; ++i) raw_reference[i] = internal::random<T>();
|
|
|
|
DenseStorageType copied_reference;
|
|
copied_reference = reference;
|
|
const T* raw_copied_reference = copied_reference.data();
|
|
for (int i = 0; i < size; ++i) VERIFY_IS_EQUAL(raw_reference[i], raw_copied_reference[i]);
|
|
}
|
|
|
|
template <typename T, int Size, int Rows, int Cols>
|
|
void dense_storage_swap(int rowsa, int colsa, int rowsb, int colsb) {
|
|
typedef DenseStorage<T, Size, Rows, Cols, 0> DenseStorageType;
|
|
|
|
const int sizea = rowsa * colsa;
|
|
ArrayX<T> referencea(sizea);
|
|
referencea.setRandom();
|
|
DenseStorageType a(sizea, rowsa, colsa);
|
|
for (int i = 0; i < sizea; ++i) a.data()[i] = referencea(i);
|
|
|
|
const int sizeb = rowsb * colsb;
|
|
ArrayX<T> referenceb(sizeb);
|
|
referenceb.setRandom();
|
|
DenseStorageType b(sizeb, rowsb, colsb);
|
|
for (int i = 0; i < sizeb; ++i) b.data()[i] = referenceb(i);
|
|
|
|
a.swap(b);
|
|
|
|
for (int i = 0; i < sizea; i++) VERIFY_IS_EQUAL(b.data()[i], referencea(i));
|
|
for (int i = 0; i < sizeb; i++) VERIFY_IS_EQUAL(a.data()[i], referenceb(i));
|
|
}
|
|
|
|
template <typename T, int Size, std::size_t Alignment>
|
|
void dense_storage_alignment() {
|
|
struct alignas(Alignment) Empty1 {};
|
|
VERIFY_IS_EQUAL(std::alignment_of<Empty1>::value, Alignment);
|
|
|
|
struct EIGEN_ALIGN_TO_BOUNDARY(Alignment) Empty2 {};
|
|
VERIFY_IS_EQUAL(std::alignment_of<Empty2>::value, Alignment);
|
|
|
|
struct Nested1 {
|
|
EIGEN_ALIGN_TO_BOUNDARY(Alignment) T data[Size];
|
|
};
|
|
VERIFY_IS_EQUAL(std::alignment_of<Nested1>::value, Alignment);
|
|
|
|
VERIFY_IS_EQUAL((std::alignment_of<internal::plain_array<T, Size, AutoAlign, Alignment>>::value), Alignment);
|
|
|
|
const std::size_t default_alignment = internal::compute_default_alignment<T, Size>::value;
|
|
if (default_alignment > 0) {
|
|
VERIFY_IS_EQUAL((std::alignment_of<DenseStorage<T, Size, 1, 1, AutoAlign>>::value), default_alignment);
|
|
VERIFY_IS_EQUAL((std::alignment_of<Matrix<T, Size, 1, AutoAlign>>::value), default_alignment);
|
|
struct Nested2 {
|
|
Matrix<T, Size, 1, AutoAlign> mat;
|
|
};
|
|
VERIFY_IS_EQUAL(std::alignment_of<Nested2>::value, default_alignment);
|
|
}
|
|
}
|
|
|
|
template <typename T>
|
|
void dense_storage_tests() {
|
|
// Dynamic Storage.
|
|
dense_storage_copy<T, Dynamic, Dynamic, Dynamic>(4, 3);
|
|
dense_storage_copy<T, Dynamic, Dynamic, 3>(4, 3);
|
|
dense_storage_copy<T, Dynamic, 4, Dynamic>(4, 3);
|
|
// Fixed Storage.
|
|
dense_storage_copy<T, 12, 4, 3>(4, 3);
|
|
dense_storage_copy<T, 12, Dynamic, Dynamic>(4, 3);
|
|
dense_storage_copy<T, 12, 4, Dynamic>(4, 3);
|
|
dense_storage_copy<T, 12, Dynamic, 3>(4, 3);
|
|
// Fixed Storage with Uninitialized Elements.
|
|
dense_storage_copy<T, 18, Dynamic, Dynamic>(4, 3);
|
|
dense_storage_copy<T, 18, 4, Dynamic>(4, 3);
|
|
dense_storage_copy<T, 18, Dynamic, 3>(4, 3);
|
|
|
|
// Dynamic Storage.
|
|
dense_storage_assignment<T, Dynamic, Dynamic, Dynamic>(4, 3);
|
|
dense_storage_assignment<T, Dynamic, Dynamic, 3>(4, 3);
|
|
dense_storage_assignment<T, Dynamic, 4, Dynamic>(4, 3);
|
|
// Fixed Storage.
|
|
dense_storage_assignment<T, 12, 4, 3>(4, 3);
|
|
dense_storage_assignment<T, 12, Dynamic, Dynamic>(4, 3);
|
|
dense_storage_assignment<T, 12, 4, Dynamic>(4, 3);
|
|
dense_storage_assignment<T, 12, Dynamic, 3>(4, 3);
|
|
// Fixed Storage with Uninitialized Elements.
|
|
dense_storage_assignment<T, 18, Dynamic, Dynamic>(4, 3);
|
|
dense_storage_assignment<T, 18, 4, Dynamic>(4, 3);
|
|
dense_storage_assignment<T, 18, Dynamic, 3>(4, 3);
|
|
|
|
// Dynamic Storage.
|
|
dense_storage_swap<T, Dynamic, Dynamic, Dynamic>(4, 3, 4, 3);
|
|
dense_storage_swap<T, Dynamic, Dynamic, Dynamic>(4, 3, 2, 1);
|
|
dense_storage_swap<T, Dynamic, Dynamic, Dynamic>(2, 1, 4, 3);
|
|
dense_storage_swap<T, Dynamic, Dynamic, 3>(4, 3, 4, 3);
|
|
dense_storage_swap<T, Dynamic, Dynamic, 3>(4, 3, 2, 3);
|
|
dense_storage_swap<T, Dynamic, Dynamic, 3>(2, 3, 4, 3);
|
|
dense_storage_swap<T, Dynamic, 4, Dynamic>(4, 3, 4, 3);
|
|
dense_storage_swap<T, Dynamic, 4, Dynamic>(4, 3, 4, 1);
|
|
dense_storage_swap<T, Dynamic, 4, Dynamic>(4, 1, 4, 3);
|
|
// Fixed Storage.
|
|
dense_storage_swap<T, 12, 4, 3>(4, 3, 4, 3);
|
|
dense_storage_swap<T, 12, Dynamic, Dynamic>(4, 3, 4, 3);
|
|
dense_storage_swap<T, 12, Dynamic, Dynamic>(4, 3, 2, 1);
|
|
dense_storage_swap<T, 12, Dynamic, Dynamic>(2, 1, 4, 3);
|
|
dense_storage_swap<T, 12, 4, Dynamic>(4, 3, 4, 3);
|
|
dense_storage_swap<T, 12, 4, Dynamic>(4, 3, 4, 1);
|
|
dense_storage_swap<T, 12, 4, Dynamic>(4, 1, 4, 3);
|
|
dense_storage_swap<T, 12, Dynamic, 3>(4, 3, 4, 3);
|
|
dense_storage_swap<T, 12, Dynamic, 3>(4, 3, 2, 3);
|
|
dense_storage_swap<T, 12, Dynamic, 3>(2, 3, 4, 3);
|
|
// Fixed Storage with Uninitialized Elements.
|
|
dense_storage_swap<T, 18, Dynamic, Dynamic>(4, 3, 4, 3);
|
|
dense_storage_swap<T, 18, Dynamic, Dynamic>(4, 3, 2, 1);
|
|
dense_storage_swap<T, 18, Dynamic, Dynamic>(2, 1, 4, 3);
|
|
dense_storage_swap<T, 18, 4, Dynamic>(4, 3, 4, 3);
|
|
dense_storage_swap<T, 18, 4, Dynamic>(4, 3, 4, 1);
|
|
dense_storage_swap<T, 18, 4, Dynamic>(4, 1, 4, 3);
|
|
dense_storage_swap<T, 18, Dynamic, 3>(4, 3, 4, 3);
|
|
dense_storage_swap<T, 18, Dynamic, 3>(4, 3, 2, 3);
|
|
dense_storage_swap<T, 18, Dynamic, 3>(2, 3, 4, 3);
|
|
|
|
dense_storage_alignment<T, 16, 8>();
|
|
dense_storage_alignment<T, 16, 16>();
|
|
dense_storage_alignment<T, 16, 32>();
|
|
dense_storage_alignment<T, 16, 64>();
|
|
}
|
|
|
|
template <typename PlainType>
|
|
void plaintype_tests() {
|
|
constexpr int RowsAtCompileTime = PlainType::RowsAtCompileTime;
|
|
constexpr int ColsAtCompileTime = PlainType::ColsAtCompileTime;
|
|
constexpr int MaxRowsAtCompileTime = PlainType::MaxRowsAtCompileTime;
|
|
constexpr int MaxColsAtCompileTime = PlainType::MaxColsAtCompileTime;
|
|
const Index expectedDefaultRows = RowsAtCompileTime == Dynamic ? 0 : RowsAtCompileTime;
|
|
const Index expectedDefaultCols = ColsAtCompileTime == Dynamic ? 0 : ColsAtCompileTime;
|
|
const Index minRows = RowsAtCompileTime == Dynamic ? 0 : RowsAtCompileTime;
|
|
const Index minCols = ColsAtCompileTime == Dynamic ? 0 : ColsAtCompileTime;
|
|
const Index maxRows = MaxRowsAtCompileTime == Dynamic ? 100 : MaxRowsAtCompileTime;
|
|
const Index maxCols = MaxColsAtCompileTime == Dynamic ? 100 : MaxColsAtCompileTime;
|
|
const Index rows = internal::random<Index>(minRows, maxRows);
|
|
const Index cols = internal::random<Index>(minCols, maxCols);
|
|
// default construction
|
|
PlainType m0;
|
|
VERIFY_IS_EQUAL(m0.rows(), expectedDefaultRows);
|
|
VERIFY_IS_EQUAL(m0.cols(), expectedDefaultCols);
|
|
m0.resize(rows, cols);
|
|
m0.setRandom();
|
|
// copy construction
|
|
PlainType m1(m0);
|
|
VERIFY_IS_EQUAL(m1.rows(), m0.rows());
|
|
VERIFY_IS_EQUAL(m1.cols(), m0.cols());
|
|
VERIFY_IS_CWISE_EQUAL(m1, m0);
|
|
// move construction
|
|
PlainType m2(std::move(m1));
|
|
VERIFY_IS_EQUAL(m2.rows(), m0.rows());
|
|
VERIFY_IS_EQUAL(m2.cols(), m0.cols());
|
|
VERIFY_IS_CWISE_EQUAL(m2, m0);
|
|
// check that object is usable after move construction
|
|
m1.resize(minRows, minCols);
|
|
m1.setRandom();
|
|
// copy assignment
|
|
m1 = m0;
|
|
VERIFY_IS_EQUAL(m1.rows(), m0.rows());
|
|
VERIFY_IS_EQUAL(m1.cols(), m0.cols());
|
|
VERIFY_IS_CWISE_EQUAL(m1, m0);
|
|
// move assignment
|
|
m2.resize(minRows, minCols);
|
|
m2.setRandom();
|
|
m2 = std::move(m1);
|
|
VERIFY_IS_EQUAL(m2.rows(), m0.rows());
|
|
VERIFY_IS_EQUAL(m2.cols(), m0.cols());
|
|
VERIFY_IS_CWISE_EQUAL(m2, m0);
|
|
// check that object is usable after move assignment
|
|
m1.resize(minRows, minCols);
|
|
m1.setRandom();
|
|
m1 = m2;
|
|
VERIFY_IS_EQUAL(m1.rows(), m0.rows());
|
|
VERIFY_IS_EQUAL(m1.cols(), m0.cols());
|
|
VERIFY_IS_CWISE_EQUAL(m1, m0);
|
|
}
|
|
|
|
EIGEN_DECLARE_TEST(dense_storage) {
|
|
dense_storage_tests<int>();
|
|
dense_storage_tests<float>();
|
|
dense_storage_tests<SafeScalar<float>>();
|
|
dense_storage_tests<MovableScalar<float>>();
|
|
dense_storage_tests<AnnoyingScalar>();
|
|
for (int i = 0; i < g_repeat; i++) {
|
|
plaintype_tests<Matrix<float, 0, 0, ColMajor>>();
|
|
plaintype_tests<Matrix<float, Dynamic, Dynamic, ColMajor, 0, 0>>();
|
|
|
|
plaintype_tests<Matrix<float, 16, 16, ColMajor>>();
|
|
plaintype_tests<Matrix<float, 16, Dynamic, ColMajor>>();
|
|
plaintype_tests<Matrix<float, Dynamic, Dynamic, ColMajor>>();
|
|
plaintype_tests<Matrix<float, Dynamic, Dynamic, ColMajor, 16, 16>>();
|
|
|
|
plaintype_tests<Matrix<SafeScalar<float>, 16, 16, ColMajor>>();
|
|
plaintype_tests<Matrix<SafeScalar<float>, 16, Dynamic, ColMajor>>();
|
|
plaintype_tests<Matrix<SafeScalar<float>, Dynamic, Dynamic, ColMajor>>();
|
|
plaintype_tests<Matrix<SafeScalar<float>, Dynamic, Dynamic, ColMajor, 16, 16>>();
|
|
|
|
plaintype_tests<Matrix<MovableScalar<float>, 16, 16, ColMajor>>();
|
|
plaintype_tests<Matrix<MovableScalar<float>, 16, Dynamic, ColMajor>>();
|
|
plaintype_tests<Matrix<MovableScalar<float>, Dynamic, Dynamic, ColMajor>>();
|
|
plaintype_tests<Matrix<MovableScalar<float>, Dynamic, Dynamic, ColMajor, 16, 16>>();
|
|
|
|
plaintype_tests<Matrix<AnnoyingScalar, 16, 16, ColMajor>>();
|
|
plaintype_tests<Matrix<AnnoyingScalar, 16, Dynamic, ColMajor>>();
|
|
plaintype_tests<Matrix<AnnoyingScalar, Dynamic, Dynamic, ColMajor>>();
|
|
plaintype_tests<Matrix<AnnoyingScalar, Dynamic, Dynamic, ColMajor, 16, 16>>();
|
|
}
|
|
}
|
|
|
|
#undef EIGEN_TESTING_PLAINOBJECT_CTOR |