/* 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. */ //! \file test_global_control.cpp //! \brief Test for [sched.global_control] specification #define TBB_PREVIEW_WAITING_FOR_WORKERS 1 #include "common/test.h" #include "common/utils.h" #include "common/spin_barrier.h" #include "common/utils_concurrency_limit.h" #include "tbb/global_control.h" #include "tbb/parallel_for.h" #include "tbb/task_group.h" #include "tbb/task_arena.h" #include struct task_scheduler_handle_guard { tbb::task_scheduler_handle m_handle{}; task_scheduler_handle_guard() { m_handle = tbb::task_scheduler_handle::get(); } ~task_scheduler_handle_guard() { tbb::task_scheduler_handle::release(m_handle); } tbb::task_scheduler_handle& get() { return m_handle; } }; namespace TestBlockingTerminateNS { struct EmptyBody { void operator()() const {} void operator()( int ) const {} }; struct TestAutoInitBody { void operator()( int ) const { tbb::parallel_for( 0, 100, EmptyBody() ); } }; static std::atomic gSeed; static std::atomic gNumSuccesses; class TestMultpleWaitBody { bool myAutoInit; public: TestMultpleWaitBody( bool autoInit = false ) : myAutoInit( autoInit ) {} void operator()( int ) const { task_scheduler_handle_guard init; if ( !myAutoInit ) { tbb::parallel_for(0, 10, EmptyBody()); } utils::FastRandom<> rnd( ++gSeed ); // In case of auto init sub-tests we skip // - case #4 to avoid recursion // - case #5 because it is explicit initialization const int numCases = myAutoInit ? 4 : 6; switch ( rnd.get() % numCases ) { case 0: { tbb::task_arena a; a.enqueue( EmptyBody() ); break; } case 1: { tbb::task_group tg; EmptyBody eb; tg.run( eb ); tg.wait(); break; } case 2: tbb::parallel_for( 0, 100, EmptyBody() ); break; case 3: /* do nothing */ break; case 4: // Create and join several threads with auto initialized scheduler. utils::NativeParallelFor( rnd.get() % 5 + 1, TestMultpleWaitBody( true ) ); break; case 5: { task_scheduler_handle_guard init2; bool res = tbb::finalize( init2.get(), std::nothrow ); REQUIRE( !res ); } break; } if ( !myAutoInit && tbb::finalize( init.get(), std::nothrow ) ) ++gNumSuccesses; } }; void TestMultpleWait() { const int minThreads = 1; const int maxThreads = 16; const int numRepeats = 5; // Initialize seed with different values on different machines. gSeed = tbb::this_task_arena::max_concurrency(); for ( int repeats = 0; repeats 0, "At least one blocking terminate must return 'true'" ); } } } #if TBB_USE_EXCEPTIONS template void TestException( F &f ) { utils::suppress_unused_warning( f ); bool caught = false; try { f(); REQUIRE( false ); } catch ( const tbb::unsafe_wait& e) { const char* msg = e.what(); REQUIRE((msg && std::strlen(msg) != 0)); caught = true; } catch ( ... ) { REQUIRE( false ); } REQUIRE( caught ); } class ExceptionTest1 { task_scheduler_handle_guard tsi1; int myIndex; public: ExceptionTest1( int index ) : myIndex( index ) {} void operator()() { task_scheduler_handle_guard tsi2; tbb::parallel_for(0, 2, EmptyBody()); // auto-init tbb::finalize((myIndex == 0 ? tsi1.get() : tsi2.get())); REQUIRE_MESSAGE( false, "Blocking terminate did not throw the exception" ); } }; struct ExceptionTest2 { class Body { utils::SpinBarrier& myBarrier; public: Body( utils::SpinBarrier& barrier ) : myBarrier( barrier ) {} void operator()( int ) const { myBarrier.wait(); task_scheduler_handle_guard init; tbb::finalize( init.get() ); REQUIRE_MESSAGE( false, "Blocking terminate did not throw the exception inside the parallel region" ); } }; void operator()() { const int numThreads = 4; tbb::global_control init(tbb::global_control::max_allowed_parallelism, numThreads); tbb::task_arena a(numThreads); a.execute([&] { utils::SpinBarrier barrier(numThreads); tbb::parallel_for(0, numThreads, Body(barrier)); REQUIRE_MESSAGE(false, "Parallel loop did not throw the exception"); }); } }; void TestExceptions() { ExceptionTest1 Test1(0); TestException( Test1 ); ExceptionTest1 Test2(1); TestException( Test2 ); if (utils::get_platform_max_threads() > 1) { // TODO: Fix the arena leak issue on single threaded machine // (see https://github.com/oneapi-src/oneTBB/issues/396) ExceptionTest2 Test3; TestException(Test3); } } #endif /* TBB_USE_EXCEPTIONS */ } // namespace TestBlockingTerminateNS void TestTerminationAndAutoinit(bool autoinit) { task_scheduler_handle_guard ctl1; task_scheduler_handle_guard ctl2; if (autoinit) { tbb::parallel_for(0, 10, TestBlockingTerminateNS::EmptyBody()); } bool res1 = tbb::finalize(ctl1.get(), std::nothrow); if (autoinit) { REQUIRE(!res1); } else { REQUIRE(res1); } bool res2 = tbb::finalize(ctl2.get(), std::nothrow); REQUIRE(res2); } //! Check no reference leak for an external thread //! \brief \ref regression \ref error_guessing TEST_CASE("test decrease reference") { tbb::task_scheduler_handle handle = tbb::task_scheduler_handle::get(); std::thread thr([] { tbb::parallel_for(0, 1, [](int) {}); } ); thr.join(); REQUIRE(tbb::finalize(handle, std::nothrow)); } //! Testing lifetime control conformance //! \brief \ref interface \ref requirement TEST_CASE("prolong lifetime simple") { tbb::task_scheduler_handle hdl1 = tbb::task_scheduler_handle::get(); { tbb::parallel_for(0, 10, TestBlockingTerminateNS::EmptyBody()); tbb::task_scheduler_handle hdl2 = tbb::task_scheduler_handle::get(); tbb::task_scheduler_handle::release(hdl2); } bool ok = tbb::finalize(hdl1, std::nothrow); REQUIRE(ok); } //! Testing lifetime control conformance //! \brief \ref interface \ref requirement TEST_CASE("prolong lifetime simple 2") { TestTerminationAndAutoinit(false); TestTerminationAndAutoinit(true); } //! Testing handle check for emptiness //! \brief \ref interface \ref requirement TEST_CASE("null handle check") { tbb::task_scheduler_handle hndl; REQUIRE_FALSE(hndl); } //! Testing handle check for emptiness //! \brief \ref interface \ref requirement TEST_CASE("null handle check 2") { tbb::task_scheduler_handle hndl = tbb::task_scheduler_handle::get(); bool not_empty = (bool)hndl; tbb::finalize(hndl, std::nothrow); REQUIRE(not_empty); REQUIRE_FALSE(hndl); } //! Testing handle check for emptiness //! \brief \ref interface \ref requirement TEST_CASE("null handle check 3") { tbb::task_scheduler_handle handle1 = tbb::task_scheduler_handle::get(); tbb::task_scheduler_handle handle2(std::move(handle1)); bool handle1_empty = !handle1; bool handle2_not_empty = (bool)handle2; tbb::finalize(handle2, std::nothrow); REQUIRE(handle1_empty); REQUIRE(handle2_not_empty); } #if TBB_USE_EXCEPTIONS //! Testing lifetime control advanced //! \brief \ref interface \ref requirement TEST_CASE("prolong lifetime advanced") { // Exceptions test leaves auto-initialized sheduler after, // because all blocking terminate calls are inside the parallel region, // thus resulting in false termination result. utils::NativeParallelFor(1, [&](int) { TestBlockingTerminateNS::TestExceptions(); }); } #endif //! Testing multiple wait //! \brief \ref interface \ref requirement TEST_CASE("prolong lifetime multiple wait") { TestBlockingTerminateNS::TestMultpleWait(); } //! Testing global_control is created on one thread and destroyed on another. //! \brief \ref interface \ref requirement TEST_CASE("cross thread 1") { // created GC, parallel_for on another thread - finalize tbb::task_scheduler_handle ctl = tbb::task_scheduler_handle::get(); utils::NativeParallelFor(1, [&](int) { tbb::parallel_for(0, 10, TestBlockingTerminateNS::EmptyBody()); bool res = tbb::finalize(ctl, std::nothrow); REQUIRE(res); }); } //! Testing global_control is created on one thread and destroyed on another. //! \brief \ref interface \ref requirement TEST_CASE("cross thread 2") { // created GC, called parallel_for on this thread, killed the thread - and finalize on another thread tbb::task_scheduler_handle ctl; utils::NativeParallelFor(1, [&](int) { ctl = tbb::task_scheduler_handle::get(); tbb::parallel_for(0, 10, TestBlockingTerminateNS::EmptyBody()); }); bool res = tbb::finalize(ctl, std::nothrow); REQUIRE(res); } //! Testing multiple wait //! \brief \ref interface \ref requirement TEST_CASE("simple prolong lifetime 3") { // Parallel region tbb::parallel_for(0, 10, TestBlockingTerminateNS::EmptyBody()); // Termination tbb::task_scheduler_handle ctl = tbb::task_scheduler_handle::get(); bool res = tbb::finalize(ctl, std::nothrow); REQUIRE(res); // New parallel region tbb::parallel_for(0, 10, TestBlockingTerminateNS::EmptyBody()); }