From e6577761affbd2d5eb0ba73e506c3ee7001eb209 Mon Sep 17 00:00:00 2001 From: Yuuichi Asahi Date: Tue, 16 Jul 2024 01:39:33 +0900 Subject: [PATCH] More traits for unary and binary operations --- common/src/KokkosFFT_traits.hpp | 110 +++++++++ common/unit_test/Test_Traits.cpp | 380 +++++++++++++++++++++++++++++++ common/unit_test/Test_Utils.hpp | 96 +++++++- 3 files changed, 585 insertions(+), 1 deletion(-) diff --git a/common/src/KokkosFFT_traits.hpp b/common/src/KokkosFFT_traits.hpp index ec89df49..c48cf8cc 100644 --- a/common/src/KokkosFFT_traits.hpp +++ b/common/src/KokkosFFT_traits.hpp @@ -9,6 +9,9 @@ namespace KokkosFFT { namespace Impl { + +// Traits for unary operation + template struct base_floating_point { using value_type = T; @@ -107,6 +110,113 @@ template inline constexpr bool is_admissible_view_v = is_admissible_view::value; +template +struct is_operatable_view : std::false_type {}; + +template +struct is_operatable_view< + ExecutionSpace, ViewType, + std::enable_if_t< + is_admissible_view_v && + Kokkos::SpaceAccessibility< + ExecutionSpace, typename ViewType::memory_space>::accessible>> + : std::true_type {}; + +/// \brief Helper to check if a View is an acceptable View for Kokkos-FFT and +/// memory space +/// is accessible from the ExecutionSpace +template +inline constexpr bool is_operatable_view_v = + is_operatable_view::value; + +// Traits for binary operations +template +struct have_same_precision : std::false_type {}; + +template +struct have_same_precision< + T1, T2, + std::enable_if_t::value && + !Kokkos::is_view::value && + std::is_same_v, + base_floating_point_type>>> + : std::true_type {}; + +template +struct have_same_precision< + InViewType, OutViewType, + std::enable_if_t< + Kokkos::is_view::value && + Kokkos::is_view::value && + std::is_same_v< + base_floating_point_type, + base_floating_point_type< + typename OutViewType::non_const_value_type>>>> + : std::true_type {}; + +/// \brief Helper to check if two value have the same base precision type. +/// When applied to Kokkos::View, then check if values of views have the +/// same base precision type. +template +inline constexpr bool have_same_precision_v = + have_same_precision::value; + +template +struct have_same_layout : std::false_type {}; + +template +struct have_same_layout< + InViewType, OutViewType, + std::enable_if_t::value && + Kokkos::is_view::value && + std::is_same_v>> + : std::true_type {}; + +/// \brief Helper to check if two views have the same layout type. +template +inline constexpr bool have_same_layout_v = + have_same_layout::value; + +template +struct have_same_rank : std::false_type {}; + +template +struct have_same_rank< + InViewType, OutViewType, + std::enable_if_t::value && + Kokkos::is_view::value && + InViewType::rank() == OutViewType::rank()>> + : std::true_type {}; + +/// \brief Helper to check if two views have the same rank. +template +inline constexpr bool have_same_rank_v = + have_same_rank::value; + +template +struct are_operatable_views : std::false_type {}; + +template +struct are_operatable_views< + ExecutionSpace, InViewType, OutViewType, + std::enable_if_t && + is_operatable_view_v && + have_same_precision_v && + have_same_layout_v && + have_same_rank_v>> + : std::true_type {}; + +/// \brief Helper to check if Views are acceptable View for Kokkos-FFT and +/// memory space are accessible from the ExecutionSpace. +/// In addition, precisions, layout and rank are checked to be identical. +template +inline constexpr bool are_operatable_views_v = + are_operatable_views::value; + +// Other traits + /// \brief Helper to define a managable View type from the original view type template struct managable_view_type { diff --git a/common/unit_test/Test_Traits.cpp b/common/unit_test/Test_Traits.cpp index 01eb2e02..780bb45f 100644 --- a/common/unit_test/Test_Traits.cpp +++ b/common/unit_test/Test_Traits.cpp @@ -6,6 +6,16 @@ #include "KokkosFFT_traits.hpp" #include "Test_Utils.hpp" +// All the tests in this file are compile time tests, so we skip all the tests +// by GTEST_SKIP(). + +// Define the types to combine +using base_real_types = std::tuple; + +// Define the layouts to combine +using base_layout_types = + std::tuple; + using real_types = ::testing::Types; using view_types = ::testing::Types, @@ -18,10 +28,19 @@ using view_types = std::pair, std::pair>; +// Define all the combinations +using paired_view_types = + tuple_to_types_t>; + template struct RealAndComplexTypes : public ::testing::Test { using real_type = T; using complex_type = Kokkos::complex; + + virtual void SetUp() { + GTEST_SKIP() << "Skipping all tests for this fixture"; + } }; template @@ -29,10 +48,26 @@ struct RealAndComplexViewTypes : public ::testing::Test { using real_type = typename T::first_type; using complex_type = Kokkos::complex; using layout_type = typename T::second_type; + virtual void SetUp() { + GTEST_SKIP() << "Skipping all tests for this fixture"; + } +}; + +template +struct PairedViewTypes : public ::testing::Test { + using real_type1 = typename std::tuple_element_t<0, T>; + using layout_type1 = typename std::tuple_element_t<1, T>; + using real_type2 = typename std::tuple_element_t<2, T>; + using layout_type2 = typename std::tuple_element_t<3, T>; + + virtual void SetUp() { + GTEST_SKIP() << "Skipping all tests for this fixture"; + } }; TYPED_TEST_SUITE(RealAndComplexTypes, real_types); TYPED_TEST_SUITE(RealAndComplexViewTypes, view_types); +TYPED_TEST_SUITE(PairedViewTypes, paired_view_types); // Tests for real type deduction template @@ -142,6 +177,45 @@ void test_admissible_view_type() { } } +// \brief Test if a View is operatable +// \tparam ExecutionSpace1 Execution space for the device +// \tparam ExecutionSpace2 Execution space for the View memory space +// \tparam T Value type of the View +// \tparam LayoutType Layout type of the View +template +void test_operatable_view_type() { + using ViewType = Kokkos::View; + using real_type = KokkosFFT::Impl::base_floating_point_type; + if constexpr (Kokkos::SpaceAccessibility< + ExecutionSpace1, + typename ViewType::memory_space>::accessible) { + if constexpr ((std::is_same_v || + std::is_same_v< + real_type, + double>)&&(std::is_same_v || + std::is_same_v)) { + static_assert( + KokkosFFT::Impl::is_operatable_view_v, + "View value type must be float, double, " + "Kokkos::Complex, Kokkos::Complex. Layout " + "must be either LayoutLeft or LayoutRight."); + } else { + static_assert( + !KokkosFFT::Impl::is_operatable_view_v, + "View value type must be float, double, " + "Kokkos::Complex, Kokkos::Complex. Layout " + "must be either LayoutLeft or LayoutRight."); + } + } else { + static_assert( + !KokkosFFT::Impl::is_operatable_view_v, + "execution_space cannot access data in ViewType"); + } +} + TYPED_TEST(RealAndComplexViewTypes, admissible_value_type) { using real_type = typename TestFixture::real_type; using complex_type = typename TestFixture::complex_type; @@ -168,3 +242,309 @@ TYPED_TEST(RealAndComplexViewTypes, admissible_view_type) { test_admissible_view_type(); test_admissible_view_type(); } + +TYPED_TEST(RealAndComplexViewTypes, operatable_view_type) { + using real_type = typename TestFixture::real_type; + using complex_type = typename TestFixture::complex_type; + using layout_type = typename TestFixture::layout_type; + using host_space = Kokkos::DefaultHostExecutionSpace; + using device_space = Kokkos::DefaultExecutionSpace; + + test_operatable_view_type(); + test_operatable_view_type(); + test_operatable_view_type(); + test_operatable_view_type(); + + test_operatable_view_type(); + test_operatable_view_type(); + test_operatable_view_type(); + test_operatable_view_type(); +} + +// Tests for multiple Views +template +void test_have_same_precision() { + using real_type1 = RealType1; + using real_type2 = RealType2; + using complex_type1 = Kokkos::complex; + using complex_type2 = Kokkos::complex; + + using RealViewType1 = Kokkos::View; + using ComplexViewType1 = Kokkos::View; + using RealViewType2 = Kokkos::View; + using ComplexViewType2 = Kokkos::View; + + if constexpr (std::is_same_v) { + // Tests for values + static_assert( + KokkosFFT::Impl::have_same_precision_v, + "Values have the same base precisions"); + static_assert( + KokkosFFT::Impl::have_same_precision_v, + "Values have the same base precisions"); + static_assert( + KokkosFFT::Impl::have_same_precision_v, + "Values have the same base precisions"); + + // Tests for Views + static_assert( + KokkosFFT::Impl::have_same_precision_v, + "ViewTypes have the same base precisions"); + static_assert( + KokkosFFT::Impl::have_same_precision_v, + "ViewTypes have the same base precisions"); + static_assert( + KokkosFFT::Impl::have_same_precision_v, + "ViewTypes have the same base precisions"); + static_assert(KokkosFFT::Impl::have_same_precision_v, + "ViewTypes have the same base precisions"); + } else { + // Tests for values + static_assert( + !KokkosFFT::Impl::have_same_precision_v, + "Values have the same base precisions"); + static_assert( + !KokkosFFT::Impl::have_same_precision_v, + "Values have the same base precisions"); + + // Tests for Views + static_assert( + !KokkosFFT::Impl::have_same_precision_v, + "ViewTypes have the same base precisions"); + static_assert(!KokkosFFT::Impl::have_same_precision_v, + "ViewTypes have the same base precisions"); + static_assert(!KokkosFFT::Impl::have_same_precision_v, + "ViewTypes have the same base precisions"); + static_assert(!KokkosFFT::Impl::have_same_precision_v, + "ViewTypes have the same base precisions"); + } +} + +template +void test_have_same_layout() { + using ViewType1 = Kokkos::View; + using ViewType2 = Kokkos::View; + + if constexpr (std::is_same_v) { + // Tests for Views + static_assert(KokkosFFT::Impl::have_same_layout_v, + "ViewTypes have the same layout"); + } else { + // Tests for Views + static_assert(!KokkosFFT::Impl::have_same_layout_v, + "ViewTypes have the same layout"); + } +} + +template +void test_have_same_rank() { + using DynamicRank1ViewType = Kokkos::View; + using DynamicRank2ViewType = Kokkos::View; + using StaticRank1ViewType = Kokkos::View; + using StaticRank2ViewType = Kokkos::View; + using DynamicStaticRank2ViewType = Kokkos::View; + static_assert(KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); + static_assert(KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); + static_assert(KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); + + static_assert(!KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); + static_assert(!KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); + static_assert(!KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); + static_assert(!KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); + static_assert(!KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); + static_assert(!KokkosFFT::Impl::have_same_rank_v, + "ViewTypes have the same rank"); +} + +// \brief Test if two Views are operatable +// \tparam ExecutionSpace1 Execution space for the device +// \tparam ExecutionSpace2 Execution space for the View memory space +// \tparam RealType1 Base Real Value type of the View1 +// \tparam LayoutType1 Layout type of the View1 +// \tparam RealType2 Base Real Value type of the View2 +// \tparam LayoutType2 Layout type of the View2 +template +void test_are_operatable_views() { + using real_type1 = RealType1; + using real_type2 = RealType2; + using complex_type1 = Kokkos::complex; + using complex_type2 = Kokkos::complex; + + using RealViewType1 = Kokkos::View; + using ComplexViewType1 = + Kokkos::View; + using RealViewType2 = Kokkos::View; + using ComplexViewType2 = + Kokkos::View; + using RealViewType3 = + Kokkos::View; + using ComplexViewType3 = + Kokkos::View; + + // Tests that the Views are accessible from the ExecutionSpace + if constexpr (Kokkos::SpaceAccessibility< + ExecutionSpace1, + typename RealViewType1::memory_space>::accessible) { + // Tests that the Views have the same precision in float or double + if constexpr (std::is_same_v && + (std::is_same_v || + std::is_same_v)) { + // Tests that the Views have the same layout in LayoutLeft or LayoutRight + if constexpr (std::is_same_v && + (std::is_same_v || + std::is_same_v)) { + // Tests that the Views are operatable if they have the same rank + static_assert(KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, RealViewType1, RealViewType2>, + "InViewType and OutViewType must have the same rank"); + static_assert(KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, RealViewType1, ComplexViewType2>, + "InViewType and OutViewType must have the same rank"); + static_assert(KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, RealViewType2>, + "InViewType and OutViewType must have the same rank"); + static_assert(KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, ComplexViewType2>, + "InViewType and OutViewType must have the same rank"); + + // Tests that the Views are not operatable if the ranks are not the same + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, RealViewType1, RealViewType3>, + "InViewType and OutViewType must have the same rank"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, RealViewType1, ComplexViewType3>, + "InViewType and OutViewType must have the same rank"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, RealViewType3>, + "InViewType and OutViewType must have the same rank"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, ComplexViewType3>, + "InViewType and OutViewType must have the same rank"); + } else { + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, RealViewType1, RealViewType2>, + "Layouts are not identical or one of them is not " + "LayoutLeft or LayoutRight"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, RealViewType1, ComplexViewType2>, + "Layouts are not identical or one of them is not " + "LayoutLeft or LayoutRight"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, RealViewType2>, + "Layouts are not identical or one of them is not " + "LayoutLeft or LayoutRight"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, ComplexViewType2>, + "Layouts are not identical or one of them is not " + "LayoutLeft or LayoutRight"); + } + } else { + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, RealViewType1, RealViewType2>, + "Base value types are not identical or one of them is not " + "float or double"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, RealViewType1, ComplexViewType2>, + "Base value types are not identical or one of them is not " + "float or double"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, RealViewType2>, + "Base value types are not identical or one of them is not " + "float or double"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, ComplexViewType2>, + "Base value types are not identical or one of them is not " + "float or double"); + } + } else { + // Views are not operatable because they are not accessible from + // ExecutionSpace + static_assert( + !KokkosFFT::Impl::are_operatable_views_v, + "Either InViewType or OutViewType is not accessible from " + "ExecutionSpace"); + static_assert( + !KokkosFFT::Impl::are_operatable_views_v, + "Either InViewType or OutViewType is not accessible from " + "ExecutionSpace"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, RealViewType2>, + "Either InViewType or OutViewType is not accessible from " + "ExecutionSpace"); + static_assert(!KokkosFFT::Impl::are_operatable_views_v< + ExecutionSpace1, ComplexViewType1, ComplexViewType2>, + "Either InViewType or OutViewType is not accessible from " + "ExecutionSpace"); + } +} + +TYPED_TEST(PairedViewTypes, have_same_precision) { + using real_type1 = typename TestFixture::real_type1; + using real_type2 = typename TestFixture::real_type2; + using layout_type = typename TestFixture::layout_type1; + + test_have_same_precision(); +} + +TYPED_TEST(PairedViewTypes, have_same_layout) { + using real_type = typename TestFixture::real_type1; + using layout_type1 = typename TestFixture::layout_type1; + using layout_type2 = typename TestFixture::layout_type2; + + test_have_same_layout(); +} + +TYPED_TEST(PairedViewTypes, have_same_rank) { + using real_type = typename TestFixture::real_type1; + using layout_type = typename TestFixture::layout_type1; + + test_have_same_rank(); +} + +TYPED_TEST(PairedViewTypes, are_operatable_views) { + using real_type1 = typename TestFixture::real_type1; + using layout_type1 = typename TestFixture::layout_type1; + using real_type2 = typename TestFixture::real_type2; + using layout_type2 = typename TestFixture::layout_type2; + using host_space = Kokkos::DefaultHostExecutionSpace; + using device_space = Kokkos::DefaultExecutionSpace; + + test_are_operatable_views(); + test_are_operatable_views(); + test_are_operatable_views(); + test_are_operatable_views(); +} diff --git a/common/unit_test/Test_Utils.hpp b/common/unit_test/Test_Utils.hpp index d30aa9b0..28fe30ca 100644 --- a/common/unit_test/Test_Utils.hpp +++ b/common/unit_test/Test_Utils.hpp @@ -5,6 +5,9 @@ #ifndef TEST_UTILS_HPP #define TEST_UTILS_HPP +#include +#include +#include #include "Test_Types.hpp" template @@ -54,4 +57,95 @@ void display(std::string name, std::vector& values) { } } -#endif \ No newline at end of file +/// Transform a sequence S to a tuple: +/// - a std::integer_sequence to a +/// std::tuple...> +/// - a std::pair to a std::tuple +/// - identity otherwise (std::tuple) +template +struct to_tuple { + using type = S; +}; + +template +struct to_tuple> { + using type = std::tuple...>; +}; + +template +struct to_tuple> { + using type = std::tuple; +}; + +template +using to_tuple_t = typename to_tuple::type; + +template +struct for_each_tuple_cat; + +template +struct for_each_tuple_cat, Tuple> { + using type = std::tuple(), + std::declval()))...>; +}; + +/// Construct a tuple of tuples that is the result of the concatenation of the +/// tuples in TupleOfTuples with Tuple. +template +using for_each_tuple_cat_t = + typename for_each_tuple_cat::type; + +static_assert( + std::is_same_v, + std::tuple>, + std::tuple>, + std::tuple, + std::tuple>>); + +static_assert( + std::is_same_v>, + std::tuple>, + std::tuple>>); + +template +struct cartesian_product_impl; + +template +struct cartesian_product_impl< + std::tuple, TailTuples...>, OutTupleOfTuples> + : cartesian_product_impl< + std::tuple, + decltype(std::tuple_cat( + std::declval>>()...))> { +}; + +template +struct cartesian_product_impl, OutTupleOfTuples> { + using type = OutTupleOfTuples; +}; + +/// Generate a std::tuple cartesian product from multiple tuple-like structures +/// (std::tuple, std::integer_sequence and std::pair) Do not rely on the +/// ordering result. +template +using cartesian_product_t = + typename cartesian_product_impl...>, + std::tuple>>::type; + +/// Transform a std::tuple to a testing::Types, identity +/// otherwise +template +struct tuple_to_types { + using type = T; +}; + +template +struct tuple_to_types> { + using type = testing::Types; +}; + +template +using tuple_to_types_t = typename tuple_to_types::type; + +#endif