/* 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/parallel_for_each_common.h" #include "common/concepts_common.h" #include #include //! \file test_parallel_for_each.cpp //! \brief Test for [algorithms.parallel_for_each] //! Test forward access iterator support //! \brief \ref error_guessing \ref interface TEST_CASE("Forward iterator support") { for ( auto concurrency_level : utils::concurrency_range() ) { tbb::global_control control(tbb::global_control::max_allowed_parallelism, concurrency_level); for(size_t depth = 0; depth <= depths_nubmer; ++depth) { g_tasks_expected = 0; for (size_t i=0; i < depth; ++i) g_tasks_expected += FindNumOfTasks(g_depths[i].value()); TestIterator_Modifiable>(depth); } } } //! Test random access iterator support //! \brief \ref error_guessing \ref interface TEST_CASE("Random access iterator support") { for ( auto concurrency_level : utils::concurrency_range() ) { tbb::global_control control(tbb::global_control::max_allowed_parallelism, concurrency_level); for(size_t depth = 0; depth <= depths_nubmer; ++depth) { g_tasks_expected = 0; for (size_t i=0; i < depth; ++i) g_tasks_expected += FindNumOfTasks(g_depths[i].value()); TestIterator_Modifiable(depth); } } } //! Test const random access iterator support //! \brief \ref error_guessing \ref interface TEST_CASE("Const random access iterator support") { for ( auto concurrency_level : utils::concurrency_range() ) { tbb::global_control control(tbb::global_control::max_allowed_parallelism, concurrency_level); for(size_t depth = 0; depth <= depths_nubmer; ++depth) { g_tasks_expected = 0; for (size_t i=0; i < depth; ++i) g_tasks_expected += FindNumOfTasks(g_depths[i].value()); TestIterator_Const>(depth); } } } //! Test container based overload //! \brief \ref error_guessing \ref interface TEST_CASE("Container based overload - forward iterator based container") { container_based_overload_test_case(/*expected_value*/1); } //! Test container based overload //! \brief \ref error_guessing \ref interface TEST_CASE("Container based overload - random access iterator based container") { container_based_overload_test_case(/*expected_value*/1); } // Test for iterators over values convertible to work item type //! \brief \ref error_guessing \ref interface TEST_CASE("Using with values convertible to work item type") { for ( auto concurrency_level : utils::concurrency_range() ) { tbb::global_control control(tbb::global_control::max_allowed_parallelism, concurrency_level); using Iterator = size_t*; for(size_t depth = 0; depth <= depths_nubmer; ++depth) { g_tasks_expected = 0; for (size_t i=0; i < depth; ++i) g_tasks_expected += FindNumOfTasks(g_depths[i].value()); // Test for iterators over values convertible to work item type TestIterator_Common(depth); TestBody(depth); TestBody(depth); } } } //! Testing workers going to sleep //! \brief \ref resource_usage \ref stress TEST_CASE("That all workers sleep when no work") { const std::size_t N = 100000; std::vector vec(N, 0); tbb::parallel_for_each(vec.begin(), vec.end(), [&](std::size_t& in) { for (int i = 0; i < 1000; ++i) { ++in; } }); TestCPUUserTime(utils::get_platform_max_threads()); } #if __TBB_CPP20_CONCEPTS_PRESENT template concept can_call_parallel_for_each_with_iterator = requires( Iterator it, const Body& body, tbb::task_group_context ctx ) { tbb::parallel_for_each(it, it, body); tbb::parallel_for_each(it, it, body, ctx); }; template concept can_call_parallel_for_each_with_cbs = requires( ContainerBasedSequence cbs, const ContainerBasedSequence const_cbs, const Body& body, tbb::task_group_context ctx ) { tbb::parallel_for_each(cbs, body); tbb::parallel_for_each(cbs, body, ctx); tbb::parallel_for_each(const_cbs, body); tbb::parallel_for_each(const_cbs, body, ctx); }; template concept can_call_parallel_for_each = can_call_parallel_for_each_with_iterator && can_call_parallel_for_each_with_cbs; template using CorrectBody = test_concepts::parallel_for_each_body::Correct())>; void test_pfor_each_iterator_constraints() { using CorrectIterator = typename std::vector::iterator; // random_access_iterator using IncorrectIterator = std::ostream_iterator; // output_iterator static_assert(can_call_parallel_for_each_with_iterator>); static_assert(!can_call_parallel_for_each_with_iterator>); } void test_pfor_each_container_based_sequence_constraints() { using namespace test_concepts::container_based_sequence; using iterator = test_concepts::container_based_sequence::iterator; static_assert(can_call_parallel_for_each_with_cbs>); static_assert(!can_call_parallel_for_each_with_cbs>); static_assert(!can_call_parallel_for_each_with_cbs>); } void test_pfor_each_body_constraints() { using namespace test_concepts::parallel_for_each_body; static_assert(can_call_parallel_for_each>); static_assert(can_call_parallel_for_each>); static_assert(!can_call_parallel_for_each>); static_assert(!can_call_parallel_for_each>); static_assert(!can_call_parallel_for_each>); static_assert(!can_call_parallel_for_each>); static_assert(!can_call_parallel_for_each>); static_assert(!can_call_parallel_for_each>); static_assert(!can_call_parallel_for_each>); } //! \brief \ref error_guessing TEST_CASE("parallel_for_each constraints") { test_pfor_each_iterator_constraints(); test_pfor_each_container_based_sequence_constraints(); test_pfor_each_body_constraints(); } #endif // __TBB_CPP20_CONCEPTS_PRESENT