blob: 86a4dd0afae7fc513994fd4a8c3de57a692eff2b [file] [log] [blame]
/*
Copyright 2005-2010 Intel Corporation. All Rights Reserved.
This file is part of Threading Building Blocks.
Threading Building Blocks is free software; you can redistribute it
and/or modify it under the terms of the GNU General Public License
version 2 as published by the Free Software Foundation.
Threading Building Blocks is distributed in the hope that it will be
useful, but WITHOUT ANY WARRANTY; without even the implied warranty
of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Threading Building Blocks; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
As a special exception, you may use this file as part of a free software
library without restriction. Specifically, if other files instantiate
templates or use macros or inline functions from this file, or you compile
this file and link it with other files to produce an executable, this
file does not by itself cause the resulting executable to be covered by
the GNU General Public License. This exception does not however
invalidate any other reasons why the executable file might be covered by
the GNU General Public License.
*/
// test reader_writer_lock
#include "tbb/reader_writer_lock.h"
#include "tbb/atomic.h"
#include "tbb/tbb_exception.h"
#include "harness_assert.h"
#include "harness.h"
tbb::reader_writer_lock the_mutex;
const int MAX_WORK = 10000;
tbb::atomic<size_t> active_readers, active_writers;
tbb::atomic<bool> sim_readers;
int BusyWork(int percentOfMaxWork) {
int iters = 0;
for (int i=0; i<MAX_WORK*((double)percentOfMaxWork/100.0); ++i) {
iters++;
}
return iters;
}
struct StressRWLBody : NoAssign {
const int nThread;
const int percentMax;
StressRWLBody(int nThread_, int percentMax_) : nThread(nThread_), percentMax(percentMax_) {}
void operator()(const int /* threadID */ ) const {
int nIters = 100;
int r_result=0, w_result=0;
for(int i=0; i<nIters; ++i) {
// test unscoped blocking write lock
the_mutex.lock();
w_result += BusyWork(percentMax);
#if TBB_USE_EXCEPTIONS && !__TBB_THROW_ACROSS_MODULE_BOUNDARY_BROKEN
// test exception for recursive write lock
bool was_caught = false;
try {
the_mutex.lock();
}
catch(tbb::improper_lock& ex) {
REMARK("improper_lock: %s\n", ex.what());
was_caught = true;
}
catch(...) {
REPORT("Wrong exception caught during recursive lock attempt.");
}
ASSERT(was_caught, "Recursive lock attempt exception not caught properly.");
// test exception for recursive read lock
was_caught = false;
try {
the_mutex.lock_read();
}
catch(tbb::improper_lock& ex) {
REMARK("improper_lock: %s\n", ex.what());
was_caught = true;
}
catch(...) {
REPORT("Wrong exception caught during recursive lock attempt.");
}
ASSERT(was_caught, "Recursive lock attempt exception not caught properly.");
#endif /* TBB_USE_EXCEPTIONS && !__TBB_THROW_ACROSS_MODULE_BOUNDARY_BROKEN */
the_mutex.unlock();
// test unscoped non-blocking write lock
if (the_mutex.try_lock()) {
w_result += BusyWork(percentMax);
the_mutex.unlock();
}
// test unscoped blocking read lock
the_mutex.lock_read();
r_result += BusyWork(percentMax);
the_mutex.unlock();
// test unscoped non-blocking read lock
if(the_mutex.try_lock_read()) {
r_result += BusyWork(percentMax);
the_mutex.unlock();
}
{ // test scoped blocking write lock
tbb::reader_writer_lock::scoped_lock my_lock(the_mutex);
w_result += BusyWork(percentMax);
}
{ // test scoped blocking read lock
tbb::reader_writer_lock::scoped_lock_read my_lock(the_mutex);
r_result += BusyWork(percentMax);
}
}
REMARK("%d reader %d writer iterations of busy work were completed.", r_result, w_result);
}
};
struct CorrectRWLScopedBody : NoAssign {
const int nThread;
CorrectRWLScopedBody(int nThread_) : nThread(nThread_) {}
void operator()(const int /* threadID */ ) const {
bool is_reader;
for (int i=0; i<50; i++) {
if (i%5==0) is_reader = false; // 1 writer for every 5 readers
else is_reader = true;
if (is_reader) {
tbb::reader_writer_lock::scoped_lock_read my_lock(the_mutex);
active_readers++;
if (active_readers > 1) sim_readers = true;
ASSERT(active_writers==0, "Active writers in read-locked region.");
Harness::Sleep(10);
active_readers--;
}
else { // is writer
tbb::reader_writer_lock::scoped_lock my_lock(the_mutex);
active_writers++;
ASSERT(active_readers==0, "Active readers in write-locked region.");
ASSERT(active_writers<=1, "More than one active writer in write-locked region.");
Harness::Sleep(10);
active_writers--;
}
}
}
};
struct CorrectRWLBody : NoAssign {
const int nThread;
CorrectRWLBody(int nThread_) : nThread(nThread_) {}
void operator()(const int /* threadID */ ) const {
bool is_reader;
for (int i=0; i<50; i++) {
if (i%5==0) is_reader = false; // 1 writer for every 5 readers
else is_reader = true;
if (is_reader) {
the_mutex.lock_read();
active_readers++;
if (active_readers > 1) sim_readers = true;
ASSERT(active_writers==0, "Active writers in read-locked region.");
}
else { // is writer
the_mutex.lock();
active_writers++;
ASSERT(active_readers==0, "Active readers in write-locked region.");
ASSERT(active_writers<=1, "More than one active writer in write-locked region.");
}
Harness::Sleep(10);
if (is_reader) {
active_readers--;
}
else { // is writer
active_writers--;
}
the_mutex.unlock();
}
}
};
void TestReaderWriterLockOnNThreads(int nThreads) {
// Stress-test all interfaces
for (int pc=0; pc<101; pc+=20) {
REMARK("\nTesting reader_writer_lock with %d threads, percent of MAX_WORK=%d", nThreads, pc);
StressRWLBody myStressBody(nThreads, pc);
NativeParallelFor(nThreads, myStressBody);
}
// Test mutual exclusion in direct locking mode
CorrectRWLBody myCorrectBody(nThreads);
active_writers = active_readers = 0;
sim_readers = false;
NativeParallelFor(nThreads, myCorrectBody);
ASSERT(sim_readers || nThreads<2, "There were no simultaneous readers.");
REMARK("Unscoped lock testing succeeded on %d threads.", nThreads);
// Test mutual exclusionin scoped locking mode
CorrectRWLScopedBody myCorrectScopedBody(nThreads);
active_writers = active_readers = 0;
sim_readers = false;
NativeParallelFor(nThreads, myCorrectScopedBody);
ASSERT(sim_readers || nThreads<2, "There were no simultaneous readers.");
REMARK("Scoped lock testing succeeded on %d threads.", nThreads);
}
void TestReaderWriterLock() {
for(int p = MinThread; p <= MaxThread; p++) {
TestReaderWriterLockOnNThreads(p);
}
}
int TestMain() {
if(MinThread <= 0) MinThread = 1;
if(MaxThread > 0) {
TestReaderWriterLock();
}
return Harness::Done;
}