/* Copyright (c) 2005-2020 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. */ ////// Test configuration //////////////////////////////////////////////////// #define SECONDS_RATIO 1000000 // microseconds #ifndef REPEAT_K #define REPEAT_K 50 // repeat coefficient #endif int outer_work[] = {/*256,*/ 64, 16, 4, 0}; int inner_work[] = {32, 8, 0 }; // keep it to calibrate the time of work without synchronization #define BOX1 "baseline" #define BOX1TEST TimeTest< TBB_Mutex, SECONDS_RATIO > // enable/disable tests for: #define BOX2 "spin_mutex" #define BOX2TEST TimeTest< TBB_Mutex, SECONDS_RATIO > // enable/disable tests for: #define BOX3 "spin_rw_mutex" #define BOX3TEST TimeTest< TBB_Mutex, SECONDS_RATIO > // enable/disable tests for: #define BOX4 "queuing_mutex" #define BOX4TEST TimeTest< TBB_Mutex, SECONDS_RATIO > // enable/disable tests for: //#define BOX5 "queuing_rw_mutex" #define BOX5TEST TimeTest< TBB_Mutex, SECONDS_RATIO > ////////////////////////////////////////////////////////////////////////////// #include #include #include // Need std::swap #include // Need std::pair #include #include "tbb/tbb_stddef.h" #include "tbb/null_mutex.h" #include "tbb/spin_rw_mutex.h" #include "tbb/spin_mutex.h" #include "tbb/queuing_mutex.h" #include "tbb/queuing_rw_mutex.h" #include "tbb/mutex.h" #if INTEL_TRIAL==2 #include "tbb/parallel_for.h" // enable threading by TBB scheduler #include "tbb/task_scheduler_init.h" #include "tbb/blocked_range.h" #endif // for test #include "time_framework.h" using namespace tbb; using namespace tbb::internal; ///////////////////////////////////////////////////////////////////////////////////////// //! base class for tests family struct TestLocks : TesterBase { // Inherits "value", "threads_count", and other variables TestLocks() : TesterBase(/*number of modes*/sizeof(outer_work)/sizeof(int)) {} //! returns name of test part/mode /*override*/std::string get_name(int testn) { std::ostringstream buf; buf.width(4); buf.fill('0'); buf << outer_work[testn]; // mode number return buf.str(); } //! enables results types and returns theirs suffixes /*override*/const char *get_result_type(int, result_t type) const { switch(type) { case MIN: return " min"; case MAX: return " max"; default: return 0; } } //! repeats count int repeat_until(int /*test_n*/) const { return REPEAT_K*100;//TODO: suggest better? } //! fake work void do_work(int work) volatile { for(int i = 0; i < work; i++) { volatile int x = i; __TBB_Pause(0); // just to call inline assembler x *= work/threads_count; } } }; //! template test unit for any of TBB mutexes template struct TBB_Mutex : TestLocks { M mutex; double test(int testn, int /*threadn*/) { for(int r = 0; r < repeat_until(testn); ++r) { do_work(outer_work[testn]); { typename M::scoped_lock with(mutex); do_work(/*inner work*/value); } } return 0; } }; ///////////////////////////////////////////////////////////////////////////////////////// //Using BOX declarations #include "time_sandbox.h" // run tests for each of inner work value void RunLoops(test_sandbox &the_test, int thread) { for( unsigned i=0; i1) Verbose = true; int DefThread = task_scheduler_init::default_num_threads(); MinThread = 1; MaxThread = DefThread+1; ParseCommandLine( argc, argv ); ASSERT(MinThread <= MaxThread, 0); #if INTEL_TRIAL && defined(__TBB_parallel_for_H) task_scheduler_init me(MaxThread); #endif { test_sandbox the_test("time_locked_work", StatisticsCollector::ByThreads); //TODO: refactor this out as RunThreads(test&) for( int t = MinThread; t < DefThread && t <= MaxThread; t *= 2) RunLoops( the_test, t ); // execute undersubscribed threads if( DefThread > MinThread && DefThread <= MaxThread ) RunLoops( the_test, DefThread ); // execute on all hw threads if( DefThread < MaxThread) RunLoops( the_test, MaxThread ); // execute requested oversubscribed threads the_test.report.SetTitle("Time of lock/unlock for mutex Name with Outer and Inner work"); //the_test.report.SetStatisticFormula("1AVG per size", "=AVERAGE(ROUNDS)"); the_test.report.Print(StatisticsCollector::HTMLFile|StatisticsCollector::ExcelXML, /*ModeName*/ "Outer work"); } return 0; }