interprocess/test/condition_test_template.hpp
2013-09-15 19:50:54 +00:00

402 lines
12 KiB
C++

// Copyright (C) 2001-2003
// William E. Kempf
//
// Permission to use, copy, modify, distribute and sell this software
// and its documentation for any purpose is hereby granted without fee,
// provided that the above copyright notice appear in all copies and
// that both that copyright notice and this permission notice appear
// in supporting documentation. William E. Kempf makes no representations
// about the suitability of this software for any purpose.
// It is provided "as is" without express or implied warranty.
//////////////////////////////////////////////////////////////////////////////
//
// (C) Copyright Ion Gaztanaga 2005-2012. Distributed under the Boost
// Software License, Version 1.0. (See accompanying file
// LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
//
// See http://www.boost.org/libs/interprocess for documentation.
//
//////////////////////////////////////////////////////////////////////////////
#ifndef BOOST_INTERPROCESS_CONDITION_TEST_TEMPLATE_HPP
#define BOOST_INTERPROCESS_CONDITION_TEST_TEMPLATE_HPP
#include <boost/interprocess/detail/config_begin.hpp>
#include <boost/interprocess/detail/workaround.hpp>
#include <boost/interprocess/detail/os_thread_functions.hpp>
#include "boost_interprocess_check.hpp"
#include <boost/interprocess/sync/scoped_lock.hpp>
#include <boost/date_time/posix_time/posix_time_types.hpp>
#include <iostream>
namespace boost{
namespace interprocess{
namespace test {
boost::posix_time::ptime ptime_delay(int secs)
{
return microsec_clock::universal_time() +
boost::posix_time::time_duration(0, 0, secs);
}
template <typename F, typename T>
class binder
{
public:
binder(const F& f, const T& p)
: func(f), param(p) { }
void operator()() const { func(param); }
private:
F func;
T param;
};
template <typename F, typename T>
binder<F, T> bind_function(F func, T param)
{
return binder<F, T>(func, param);
}
template <class Condition, class Mutex>
struct condition_test_data
{
condition_test_data() : notified(0), awoken(0) { }
~condition_test_data()
{}
Mutex mutex;
Condition condition;
int notified;
int awoken;
};
template <class Condition, class Mutex>
void condition_test_thread(condition_test_data<Condition, Mutex>* data)
{
boost::interprocess::scoped_lock<Mutex>
lock(data->mutex);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
while (!(data->notified > 0))
data->condition.wait(lock);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
data->awoken++;
}
struct cond_predicate
{
cond_predicate(int& var, int val) : _var(var), _val(val) { }
bool operator()() { return _var == _val; }
int& _var;
int _val;
};
template <class Condition, class Mutex>
void condition_test_waits(condition_test_data<Condition, Mutex>* data)
{
boost::interprocess::scoped_lock<Mutex>
lock(data->mutex);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
// Test wait.
while (data->notified != 1)
data->condition.wait(lock);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
BOOST_INTERPROCESS_CHECK(data->notified == 1);
data->awoken++;
data->condition.notify_one();
// Test predicate wait.
data->condition.wait(lock, cond_predicate(data->notified, 2));
BOOST_INTERPROCESS_CHECK(lock ? true : false);
BOOST_INTERPROCESS_CHECK(data->notified == 2);
data->awoken++;
data->condition.notify_one();
// Test timed_wait.
while (data->notified != 3)
data->condition.timed_wait(lock, ptime_delay(5));
BOOST_INTERPROCESS_CHECK(lock ? true : false);
BOOST_INTERPROCESS_CHECK(data->notified == 3);
data->awoken++;
data->condition.notify_one();
// Test predicate timed_wait.
cond_predicate pred(data->notified, 4);
bool ret = data->condition.timed_wait(lock, ptime_delay(5), pred);
BOOST_INTERPROCESS_CHECK(ret);(void)ret;
BOOST_INTERPROCESS_CHECK(lock ? true : false);
BOOST_INTERPROCESS_CHECK(pred());
BOOST_INTERPROCESS_CHECK(data->notified == 4);
data->awoken++;
data->condition.notify_one();
}
template <class Condition, class Mutex>
void do_test_condition_notify_one()
{
condition_test_data<Condition, Mutex> data;
boost::interprocess::ipcdetail::OS_thread_t thread;
boost::interprocess::ipcdetail::thread_launch(thread, bind_function(&condition_test_thread<Condition, Mutex>, &data));
//Make sure thread is blocked
boost::interprocess::ipcdetail::thread_sleep(1000);
{
boost::interprocess::scoped_lock<Mutex>
lock(data.mutex);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
data.notified++;
data.condition.notify_one();
}
boost::interprocess::ipcdetail::thread_join(thread);
BOOST_INTERPROCESS_CHECK(data.awoken == 1);
}
template <class Condition, class Mutex>
void do_test_condition_notify_all()
{
const int NUMTHREADS = 3;
boost::interprocess::ipcdetail::OS_thread_t thgroup[NUMTHREADS];
condition_test_data<Condition, Mutex> data;
for(int i = 0; i< NUMTHREADS; ++i){
boost::interprocess::ipcdetail::thread_launch(thgroup[i], bind_function(&condition_test_thread<Condition, Mutex>, &data));
}
//Make sure all threads are blocked
boost::interprocess::ipcdetail::thread_sleep(1000);
{
boost::interprocess::scoped_lock<Mutex>
lock(data.mutex);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
data.notified++;
}
data.condition.notify_all();
for(int i = 0; i< NUMTHREADS; ++i){
boost::interprocess::ipcdetail::thread_join(thgroup[i]);
}
BOOST_INTERPROCESS_CHECK(data.awoken == NUMTHREADS);
}
template <class Condition, class Mutex>
void do_test_condition_waits()
{
condition_test_data<Condition, Mutex> data;
boost::interprocess::ipcdetail::OS_thread_t thread;
boost::interprocess::ipcdetail::thread_launch(thread, bind_function(&condition_test_waits<Condition, Mutex>, &data));
{
boost::interprocess::scoped_lock<Mutex>
lock(data.mutex);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
boost::interprocess::ipcdetail::thread_sleep(1000);
data.notified++;
data.condition.notify_one();
while (data.awoken != 1)
data.condition.wait(lock);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
BOOST_INTERPROCESS_CHECK(data.awoken == 1);
boost::interprocess::ipcdetail::thread_sleep(1000);
data.notified++;
data.condition.notify_one();
while (data.awoken != 2)
data.condition.wait(lock);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
BOOST_INTERPROCESS_CHECK(data.awoken == 2);
boost::interprocess::ipcdetail::thread_sleep(1000);
data.notified++;
data.condition.notify_one();
while (data.awoken != 3)
data.condition.wait(lock);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
BOOST_INTERPROCESS_CHECK(data.awoken == 3);
boost::interprocess::ipcdetail::thread_sleep(1000);
data.notified++;
data.condition.notify_one();
while (data.awoken != 4)
data.condition.wait(lock);
BOOST_INTERPROCESS_CHECK(lock ? true : false);
BOOST_INTERPROCESS_CHECK(data.awoken == 4);
}
boost::interprocess::ipcdetail::thread_join(thread);
BOOST_INTERPROCESS_CHECK(data.awoken == 4);
}
/*
//Message queue simulation test
template <class Condition>
inline Condition &cond_empty()
{
static Condition cond_empty;
return cond_empty;
}
template <class Condition>
inline Condition &cond_full()
{
static Condition cond_full;
return cond_full;
}
template <class Mutex>
inline Mutex &mutex()
{
static Mutex mut;
return mut;
}
*/
static volatile int count = 0;
static volatile int waiting_readers = 0;
static volatile int waiting_writer = 0;
const int queue_size = 3;
const int thread_factor = 10;
const int NumThreads = thread_factor*queue_size;
//Function that removes items from queue
template <class Condition, class Mutex>
struct condition_func
{
condition_func(Condition &cond_full, Condition &cond_empty, Mutex &mutex)
: cond_full_(cond_full), cond_empty_(cond_empty), mutex_(mutex)
{}
void operator()()
{
boost::interprocess::scoped_lock<Mutex>lock(mutex_);
while(count == 0){
++waiting_readers;
cond_empty_.wait(lock);
--waiting_readers;
}
--count;
if(waiting_writer)
cond_full_.notify_one();
}
Condition &cond_full_;
Condition &cond_empty_;
Mutex &mutex_;
};
//Queue functions
template <class Condition, class Mutex>
void do_test_condition_queue_notify_one(void)
{
//Force mutex and condition creation
Condition cond_empty;
Condition cond_full;
Mutex mutex;
//Create threads that will decrease count
{
//Initialize counters
count = 0;
waiting_readers = 0;
waiting_writer = 0;
boost::interprocess::ipcdetail::OS_thread_t thgroup[NumThreads];
for(int i = 0; i< NumThreads; ++i){
condition_func<Condition, Mutex> func(cond_full, cond_empty, mutex);
boost::interprocess::ipcdetail::thread_launch(thgroup[i], func);
}
//Add 20 elements one by one in the queue simulation
//The sender will block if it fills the queue
for(int i = 0; i < NumThreads; ++i){
boost::interprocess::scoped_lock<Mutex> lock(mutex);
while(count == queue_size){
++waiting_writer;
cond_full.wait(lock);
--waiting_writer;
}
count++;
if(waiting_readers)
cond_empty.notify_one();
}
for(int i = 0; i< NumThreads; ++i){
boost::interprocess::ipcdetail::thread_join(thgroup[i]);
}
BOOST_INTERPROCESS_CHECK(count == 0);
BOOST_INTERPROCESS_CHECK(waiting_readers == 0);
BOOST_INTERPROCESS_CHECK(waiting_writer == 0);
}
}
//Queue functions
template <class Condition, class Mutex>
void do_test_condition_queue_notify_all(void)
{
//Force mutex and condition creation
Condition cond_empty;
Condition cond_full;
Mutex mutex;
//Create threads that will decrease count
{
//Initialize counters
count = 0;
waiting_readers = 0;
waiting_writer = 0;
boost::interprocess::ipcdetail::OS_thread_t thgroup[NumThreads];
for(int i = 0; i< NumThreads; ++i){
condition_func<Condition, Mutex> func(cond_full, cond_empty, mutex);
boost::interprocess::ipcdetail::thread_launch(thgroup[i], func);
}
//Fill queue to the max size and notify all several times
for(int i = 0; i < NumThreads; ++i){
boost::interprocess::scoped_lock<Mutex>lock(mutex);
while(count == queue_size){
++waiting_writer;
cond_full.wait(lock);
--waiting_writer;
}
count++;
if(waiting_readers)
cond_empty.notify_all();
}
for(int i = 0; i< NumThreads; ++i){
boost::interprocess::ipcdetail::thread_join(thgroup[i]);
}
BOOST_INTERPROCESS_CHECK(count == 0);
BOOST_INTERPROCESS_CHECK(waiting_readers == 0);
BOOST_INTERPROCESS_CHECK(waiting_writer == 0);
}
}
template <class Condition, class Mutex>
bool do_test_condition()
{
std::cout << "do_test_condition_notify_one<" << typeid(Condition).name() << "," << typeid(Mutex).name() << '\n' << std::endl;
do_test_condition_notify_one<Condition, Mutex>();
std::cout << "do_test_condition_notify_all<" << typeid(Condition).name() << "," << typeid(Mutex).name() << '\n' << std::endl;
do_test_condition_notify_all<Condition, Mutex>();
std::cout << "do_test_condition_waits<" << typeid(Condition).name() << "," << typeid(Mutex).name() << '\n' << std::endl;
do_test_condition_waits<Condition, Mutex>();
std::cout << "do_test_condition_queue_notify_one<" << typeid(Condition).name() << "," << typeid(Mutex).name() << '\n' << std::endl;
do_test_condition_queue_notify_one<Condition, Mutex>();
std::cout << "do_test_condition_queue_notify_all<" << typeid(Condition).name() << "," << typeid(Mutex).name() << '\n' << std::endl;
do_test_condition_queue_notify_all<Condition, Mutex>();
return true;
}
} //namespace test
} //namespace interprocess{
} //namespace boost{
#include <boost/interprocess/detail/config_end.hpp>
#endif //#ifndef BOOST_INTERPROCESS_CONDITION_TEST_TEMPLATE_HPP