From 5e28c41549c5e71011bb9312104d1d30e919e53f Mon Sep 17 00:00:00 2001 From: Christian Seiler Date: Thu, 14 Nov 2013 22:27:06 +0100 Subject: C++11: add template metaprogramming helpers Create a new directory CXX11 under unsupported/Eigen that contains code that requires C++11. In that directory, add a few generic templates useful for any module relying on C++11. These templates may be included with #include <[unsupported/]Eigen/CXX11/Core>. At the moment, this will only provide templates in the Eigen::internal namespace. --- unsupported/test/cxx11_meta.cpp | 343 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 343 insertions(+) create mode 100644 unsupported/test/cxx11_meta.cpp (limited to 'unsupported/test/cxx11_meta.cpp') diff --git a/unsupported/test/cxx11_meta.cpp b/unsupported/test/cxx11_meta.cpp new file mode 100644 index 000000000..03a9efe54 --- /dev/null +++ b/unsupported/test/cxx11_meta.cpp @@ -0,0 +1,343 @@ +// This file is part of Eigen, a lightweight C++ template library +// for linear algebra. +// +// Copyright (C) 2008-2009 Gael Guennebaud +// Copyright (C) 2006-2008 Benoit Jacob +// +// 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" + +#include + +using Eigen::internal::is_same; +using Eigen::internal::type_list; +using Eigen::internal::numeric_list; +using Eigen::internal::gen_numeric_list; +using Eigen::internal::gen_numeric_list_reversed; +using Eigen::internal::gen_numeric_list_swapped_pair; +using Eigen::internal::gen_numeric_list_repeated; +using Eigen::internal::concat; +using Eigen::internal::mconcat; +using Eigen::internal::take; +using Eigen::internal::skip; +using Eigen::internal::slice; +using Eigen::internal::get; +using Eigen::internal::id_numeric; +using Eigen::internal::id_type; +using Eigen::internal::is_same_gf; +using Eigen::internal::apply_op_from_left; +using Eigen::internal::apply_op_from_right; +using Eigen::internal::contained_in_list; +using Eigen::internal::contained_in_list_gf; +using Eigen::internal::arg_prod; +using Eigen::internal::arg_sum; +using Eigen::internal::sum_op; +using Eigen::internal::product_op; +using Eigen::internal::array_reverse; +using Eigen::internal::array_sum; +using Eigen::internal::array_prod; +using Eigen::internal::array_reduce; +using Eigen::internal::array_zip; +using Eigen::internal::array_zip_and_reduce; +using Eigen::internal::array_apply; +using Eigen::internal::array_apply_and_reduce; +using Eigen::internal::repeat; +using Eigen::internal::instantiate_by_c_array; + +struct dummy_a {}; +struct dummy_b {}; +struct dummy_c {}; +struct dummy_d {}; +struct dummy_e {}; + +// dummy operation for testing apply +template struct dummy_op; +template<> struct dummy_op { typedef dummy_c type; }; +template<> struct dummy_op { typedef dummy_d type; }; +template<> struct dummy_op { typedef dummy_a type; }; +template<> struct dummy_op { typedef dummy_d type; }; +template<> struct dummy_op { typedef dummy_b type; }; +template<> struct dummy_op { typedef dummy_d type; }; +template<> struct dummy_op { typedef dummy_e type; }; +template<> struct dummy_op { typedef dummy_e type; }; +template<> struct dummy_op { typedef dummy_e type; }; + +template struct dummy_test { constexpr static bool value = false; constexpr static int global_flags = 0; }; +template<> struct dummy_test { constexpr static bool value = true; constexpr static int global_flags = 1; }; +template<> struct dummy_test { constexpr static bool value = true; constexpr static int global_flags = 2; }; +template<> struct dummy_test { constexpr static bool value = true; constexpr static int global_flags = 4; }; + +struct times2_op { template static A run(A v) { return v * 2; } }; + +struct dummy_inst +{ + int c; + + dummy_inst() : c(0) {} + explicit dummy_inst(int) : c(1) {} + dummy_inst(int, int) : c(2) {} + dummy_inst(int, int, int) : c(3) {} + dummy_inst(int, int, int, int) : c(4) {} + dummy_inst(int, int, int, int, int) : c(5) {} +}; + +static void test_gen_numeric_list() +{ + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); +} + +static void test_concat() +{ + VERIFY((is_same, type_list<>>::type, type_list>::value)); + VERIFY((is_same, type_list>::type, type_list>::value)); + VERIFY((is_same, type_list>::type, type_list>::value)); + VERIFY((is_same, type_list>::type, type_list>::value)); + VERIFY((is_same, type_list>::type, type_list>::value)); + + VERIFY((is_same, numeric_list>::type, numeric_list>::value)); + VERIFY((is_same, numeric_list>::type, numeric_list>::value)); + VERIFY((is_same, numeric_list>::type, numeric_list>::value)); + VERIFY((is_same, numeric_list>::type, numeric_list>::value)); + VERIFY((is_same, numeric_list>::type, numeric_list>::value)); + + VERIFY((is_same>::type, type_list>::value)); + VERIFY((is_same, type_list>::type, type_list>::value)); + VERIFY((is_same, type_list, type_list>::type, type_list>::value)); + VERIFY((is_same, type_list>::type, type_list>::value)); + VERIFY((is_same, type_list>::type, type_list>::value)); + + VERIFY((is_same>::type, numeric_list>::value)); + VERIFY((is_same, numeric_list>::type, numeric_list>::value)); + VERIFY((is_same, numeric_list, numeric_list>::type, numeric_list>::value)); + VERIFY((is_same, numeric_list>::type, numeric_list>::value)); + VERIFY((is_same, numeric_list>::type, numeric_list>::value)); +} + +static void test_slice() +{ + typedef type_list tl; + typedef numeric_list il; + + VERIFY((is_same::type, type_list<>>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, type_list<>>::value)); + + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); + + VERIFY((is_same::type, typename take<3, tl>::type>::value)); + VERIFY((is_same::type, typename take<3, il>::type>::value)); + VERIFY((is_same::type, type_list>::value)); + VERIFY((is_same::type, numeric_list>::value)); +} + +static void test_get() +{ + typedef type_list tl; + typedef numeric_list il; + + VERIFY((is_same::type, dummy_a>::value)); + VERIFY((is_same::type, dummy_a>::value)); + VERIFY((is_same::type, dummy_b>::value)); + VERIFY((is_same::type, dummy_b>::value)); + VERIFY((is_same::type, dummy_c>::value)); + VERIFY((is_same::type, dummy_c>::value)); + + VERIFY_IS_EQUAL(((int)get<0, il>::value), 4); + VERIFY_IS_EQUAL(((int)get<1, il>::value), 8); + VERIFY_IS_EQUAL(((int)get<2, il>::value), 15); + VERIFY_IS_EQUAL(((int)get<3, il>::value), 16); + VERIFY_IS_EQUAL(((int)get<4, il>::value), 23); + VERIFY_IS_EQUAL(((int)get<5, il>::value), 42); +} + +static void test_id_helper(dummy_a a, dummy_a b, dummy_a c) +{ + (void)a; + (void)b; + (void)c; +} + +template +static void test_id_numeric() +{ + test_id_helper(typename id_numeric::type()...); +} + +template +static void test_id_type() +{ + test_id_helper(typename id_type::type()...); +} + +static void test_id() +{ + // don't call VERIFY here, just assume it works if it compiles + // (otherwise it will complain that it can't find the function) + test_id_numeric<1, 4, 6>(); + test_id_type(); +} + +static void test_is_same_gf() +{ + VERIFY((!is_same_gf::value)); + VERIFY((!!is_same_gf::value)); + VERIFY_IS_EQUAL((!!is_same_gf::global_flags), 0); + VERIFY_IS_EQUAL((!!is_same_gf::global_flags), 0); +} + +static void test_apply_op() +{ + typedef type_list tl; + VERIFY((!!is_same::type, type_list>::value)); + VERIFY((!!is_same::type, type_list>::value)); +} + +static void test_contained_in_list() +{ + typedef type_list tl; + + VERIFY((!!contained_in_list::value)); + VERIFY((!!contained_in_list::value)); + VERIFY((!!contained_in_list::value)); + VERIFY((!contained_in_list::value)); + VERIFY((!contained_in_list::value)); + + VERIFY((!!contained_in_list_gf::value)); + VERIFY((!!contained_in_list_gf::value)); + VERIFY((!!contained_in_list_gf::value)); + VERIFY((!contained_in_list_gf::value)); + VERIFY((!contained_in_list_gf::value)); + + VERIFY_IS_EQUAL(((int)contained_in_list_gf::global_flags), 1); + VERIFY_IS_EQUAL(((int)contained_in_list_gf::global_flags), 2); + VERIFY_IS_EQUAL(((int)contained_in_list_gf::global_flags), 4); + VERIFY_IS_EQUAL(((int)contained_in_list_gf::global_flags), 0); + VERIFY_IS_EQUAL(((int)contained_in_list_gf::global_flags), 0); +} + +static void test_arg_reductions() +{ + VERIFY_IS_EQUAL(arg_sum(1,2,3,4), 10); + VERIFY_IS_EQUAL(arg_prod(1,2,3,4), 24); + VERIFY_IS_APPROX(arg_sum(0.5, 2, 5), 7.5); + VERIFY_IS_APPROX(arg_prod(0.5, 2, 5), 5.0); +} + +static void test_array_reverse_and_reduce() +{ + std::array a{{4, 8, 15, 16, 23, 42}}; + std::array b{{42, 23, 16, 15, 8, 4}}; + + // there is no operator<< for std::array, so VERIFY_IS_EQUAL will + // not compile + VERIFY((array_reverse(a) == b)); + VERIFY((array_reverse(b) == a)); + VERIFY_IS_EQUAL((array_sum(a)), 108); + VERIFY_IS_EQUAL((array_sum(b)), 108); + VERIFY_IS_EQUAL((array_prod(a)), 7418880); + VERIFY_IS_EQUAL((array_prod(b)), 7418880); +} + +static void test_array_zip_and_apply() +{ + std::array a{{4, 8, 15, 16, 23, 42}}; + std::array b{{0, 1, 2, 3, 4, 5}}; + std::array c{{4, 9, 17, 19, 27, 47}}; + std::array d{{0, 8, 30, 48, 92, 210}}; + std::array e{{0, 2, 4, 6, 8, 10}}; + + VERIFY((array_zip(a, b) == c)); + VERIFY((array_zip(a, b) == d)); + VERIFY((array_apply(b) == e)); + VERIFY_IS_EQUAL((array_apply_and_reduce(a)), 216); + VERIFY_IS_EQUAL((array_apply_and_reduce(b)), 30); + VERIFY_IS_EQUAL((array_zip_and_reduce(a, b)), 14755932); + VERIFY_IS_EQUAL((array_zip_and_reduce(a, b)), 388); +} + +static void test_array_misc() +{ + std::array a3{{1, 1, 1}}; + std::array a6{{2, 2, 2, 2, 2, 2}}; + VERIFY((repeat<3, int>(1) == a3)); + VERIFY((repeat<6, int>(2) == a6)); + + int data[5] = { 0, 1, 2, 3, 4 }; + VERIFY_IS_EQUAL((instantiate_by_c_array(data).c), 0); + VERIFY_IS_EQUAL((instantiate_by_c_array(data).c), 1); + VERIFY_IS_EQUAL((instantiate_by_c_array(data).c), 2); + VERIFY_IS_EQUAL((instantiate_by_c_array(data).c), 3); + VERIFY_IS_EQUAL((instantiate_by_c_array(data).c), 4); + VERIFY_IS_EQUAL((instantiate_by_c_array(data).c), 5); +} + +void test_cxx11_meta() +{ + CALL_SUBTEST(test_gen_numeric_list()); + CALL_SUBTEST(test_concat()); + CALL_SUBTEST(test_slice()); + CALL_SUBTEST(test_get()); + CALL_SUBTEST(test_id()); + CALL_SUBTEST(test_is_same_gf()); + CALL_SUBTEST(test_apply_op()); + CALL_SUBTEST(test_contained_in_list()); + CALL_SUBTEST(test_arg_reductions()); + CALL_SUBTEST(test_array_reverse_and_reduce()); + CALL_SUBTEST(test_array_zip_and_apply()); + CALL_SUBTEST(test_array_misc()); +} + +/* + * kate: space-indent on; indent-width 2; mixedindent off; indent-mode cstyle; + */ -- cgit v1.2.3