/* Copyright (c) 2005-2021 Intel Corporation Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ #include "common/config.h" #include "test_join_node.h" #include "common/concepts_common.h" //! \file test_join_node_key_matching.cpp //! \brief Test for [flow_graph.join_node] specification #if __TBB_CPP17_DEDUCTION_GUIDES_PRESENT void test_deduction_guides() { using namespace tbb::flow; using tuple_type = std::tuple; graph g; auto body_int = [](const int&)->int { return 1; }; auto body_double = [](const double&)->int { return 1; }; join_node j1(g, body_int, body_int, body_double); static_assert(std::is_same_v>>); #if __TBB_PREVIEW_FLOW_GRAPH_NODE_SET broadcast_node b1(g), b2(g); broadcast_node b3(g); broadcast_node b4(g); join_node j2(follows(b1, b2, b3), body_int, body_int, body_double); static_assert(std::is_same_v>>); join_node j3(precedes(b4), body_int, body_int, body_double); static_assert(std::is_same_v>>); #endif join_node j4(j1); static_assert(std::is_same_v>>); } #endif template using make_tuple = decltype(std::tuple_cat(T1(), std::tuple())); using T1 = std::tuple>; using T2 = make_tuple>; using T3 = make_tuple>; using T4 = make_tuple>; using T5 = make_tuple>; using T6 = make_tuple>; using T7 = make_tuple>; using T8 = make_tuple>; using T9 = make_tuple>; using T10 = make_tuple>; #if TBB_TEST_LOW_WORKLOAD && TBB_USE_DEBUG // the compiler might generate huge object file in debug (>64M) #define TEST_CASE_TEMPLATE_N_ARGS(dec) TEST_CASE_TEMPLATE(dec, T, T2, T5, T8, T10) #else #define TEST_CASE_TEMPLATE_N_ARGS(dec) TEST_CASE_TEMPLATE(dec, T, T2, T3, T4, T5, T6, T7, T8, T9, T10) #endif //! Test serial key matching on special input types //! \brief \ref error_guessing TEST_CASE("Serial test on tuples") { INFO("key_matching\n"); generate_test, MyKeySecond >, tbb::flow::key_matching >::do_test(); generate_test, MyKeySecond >, tbb::flow::key_matching >::do_test(); generate_test, MyKeySecond, MyKeyWithBrokenMessageKey >, tbb::flow::key_matching >::do_test(); } //! Serial test with different tuple sizes //! \brief \ref error_guessing TEST_CASE_TEMPLATE_N_ARGS("Serial N tests on tuples") { generate_test>::do_test(); } #if __TBB_CPP17_DEDUCTION_GUIDES_PRESENT //! Test deduction guides //! \brief \ref requirement TEST_CASE("Test deduction guides"){ test_deduction_guides(); } #endif //! Test parallel key matching on special input types //! \brief \ref error_guessing TEST_CASE("Parallel test on tuples"){ generate_test, MyKeySecond >, tbb::flow::key_matching >::do_test(); generate_test, MyKeySecond >, tbb::flow::key_matching >::do_test(); generate_test, MyKeySecond >, tbb::flow::key_matching >::do_test(); } //! Parallel test with different tuple sizes //! \brief \ref error_guessing TEST_CASE_TEMPLATE_N_ARGS("Parallel N tests on tuples") { generate_test>::do_test(); } #if __TBB_CPP20_CONCEPTS_PRESENT template struct tuple_helper { using type = decltype(std::tuple_cat(std::declval>(), std::declval::type>())); }; template <> struct tuple_helper<1> { using type = std::tuple; }; template concept can_initialize_join_node = requires(tbb::flow::graph& g, Args... args) { tbb::flow::join_node::type, tbb::flow::key_matching>(g, args...); }; // Helper for the concepts which checks if key_matching join_node cannot be instantiated if // one of its constructor arguments do not satisfy join_node_function_object concept // This structure substitutes IncorrectT to the sequence of arguments on IncorrectArgIndex position // The remaining arguments in the sequence are CorrectT template struct multiple_arguments_initialization_helper { // Current index is not equal to IncorrectArgIndex - substitute CorrectT at the end of the arguments sequence and continue static constexpr bool value = multiple_arguments_initialization_helper::value; }; template struct multiple_arguments_initialization_helper { // Current index is equal to IncorrectArgIndex - substitute IncorrectT at the end of the sequence and continue // No more incorrect indices would be added - continue with MAX_TUPLE_TEST_SIZE variable as current incorrect index static constexpr bool value = multiple_arguments_initialization_helper::value; }; template struct multiple_arguments_initialization_helper<0, IncorrectArgIndex, CorrectT, IncorrectT, Args...> { // ArgCount is equal to 0 - no more arguments should be added // Check if join_node can be initialized with Args static constexpr bool value = can_initialize_join_node; }; // Helper which iterates over incorrect indices. value is true if initialization is successful for at least for one IncorrectArgIndex template struct incorrect_arg_index_iteration_helper { // CurrentIncorrectIndex is not equal to max - check with current and continue static constexpr bool value = multiple_arguments_initialization_helper::value || incorrect_arg_index_iteration_helper::value; }; template requires (ArgCount == CurrentIncorrectIndex + 1) struct incorrect_arg_index_iteration_helper { // CurrentIncorrectIndex is equal to max - check and stop static constexpr bool value = multiple_arguments_initialization_helper::value; }; // Helper which iterates over argument count. value is true if initialization (with all possible incorrect indices) is successful for at least one ArgCount template struct arg_count_iteration_helper { // CurrentArgCount is not equal to max - check and continue static constexpr bool value = incorrect_arg_index_iteration_helper::value || arg_count_iteration_helper::value; }; template struct arg_count_iteration_helper { // CurrentArgCount is equal to max - check and stop static constexpr bool value = incorrect_arg_index_iteration_helper::value; }; template concept can_initialize_join_node_with_incorrect_argument = arg_count_iteration_helper::value; template struct join_node_correct_initialization_helper { static constexpr bool value = join_node_correct_initialization_helper::value; }; template struct join_node_correct_initialization_helper<0, CorrectT, Args...> { static constexpr bool value = can_initialize_join_node; }; template struct arg_count_correct_initialization_helper { static constexpr bool value = join_node_correct_initialization_helper::value && arg_count_correct_initialization_helper::value; }; template struct arg_count_correct_initialization_helper { static constexpr bool value = join_node_correct_initialization_helper::value; }; template concept can_initialize_join_node_with_correct_argument = arg_count_correct_initialization_helper::value; //! \brief \ref error_guessing TEST_CASE("join_node constraints") { using namespace test_concepts::join_node_function_object; static_assert(can_initialize_join_node_with_correct_argument>); static_assert(!can_initialize_join_node_with_incorrect_argument, NonCopyable>); static_assert(!can_initialize_join_node_with_incorrect_argument, NonDestructible>); static_assert(!can_initialize_join_node_with_incorrect_argument, NoOperatorRoundBrackets>); static_assert(!can_initialize_join_node_with_incorrect_argument, WrongInputOperatorRoundBrackets>); static_assert(!can_initialize_join_node_with_incorrect_argument, WrongReturnOperatorRoundBrackets>); } #endif // __TBB_CPP20_CONCEPTS_PRESENT