2
0
mirror of https://github.com/boostorg/sync.git synced 2026-01-19 16:52:11 +00:00
Files
sync/test/run/condition_variable_notify_all.cpp
Andrey Semashev ffd2a86d5f Ported tests to lightweight_test.hpp.
This will allow to work around compilation warnings and errors on
MinGW coming from Boost.Test.
2019-01-03 21:22:44 +03:00

240 lines
6.0 KiB
C++

// Copyright (C) 2007 Anthony Williams
// Copyright (C) 2013 Andrey Semashev
//
// 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)
#include <boost/thread/thread.hpp>
#include <boost/core/lightweight_test.hpp>
#include "utils.hpp"
#include "condition_test_common.hpp"
unsigned const number_of_test_threads = 5;
void do_test_condition_notify_all_wakes_from_wait()
{
wait_for_flag data;
boost::thread_group group;
try
{
for(unsigned i=0;i<number_of_test_threads;++i)
{
group.create_thread(bind_thread_func(&wait_for_flag::wait_without_predicate, data));
}
{
boost::sync::unique_lock<boost::sync::mutex> lock(data.mutex);
data.flag=true;
data.cond_var.notify_all();
}
group.join_all();
BOOST_TEST_EQ(data.woken,number_of_test_threads);
}
catch(...)
{
group.join_all();
throw;
}
}
void test_condition_notify_all_wakes_from_wait()
{
timed_test(&do_test_condition_notify_all_wakes_from_wait, timeout_seconds);
}
void do_test_condition_notify_all_wakes_from_wait_with_predicate()
{
wait_for_flag data;
boost::thread_group group;
try
{
for(unsigned i=0;i<number_of_test_threads;++i)
{
group.create_thread(bind_thread_func(&wait_for_flag::wait_with_predicate, data));
}
{
boost::sync::unique_lock<boost::sync::mutex> lock(data.mutex);
data.flag=true;
data.cond_var.notify_all();
}
group.join_all();
BOOST_TEST_EQ(data.woken,number_of_test_threads);
}
catch(...)
{
group.join_all();
throw;
}
}
void test_condition_notify_all_wakes_from_wait_with_predicate()
{
timed_test(&do_test_condition_notify_all_wakes_from_wait_with_predicate, timeout_seconds);
}
void do_test_condition_notify_all_wakes_from_timed_wait()
{
wait_for_flag data;
boost::thread_group group;
try
{
for(unsigned i=0;i<number_of_test_threads;++i)
{
group.create_thread(bind_thread_func(&wait_for_flag::timed_wait_without_predicate, data));
}
{
boost::sync::unique_lock<boost::sync::mutex> lock(data.mutex);
data.flag=true;
data.cond_var.notify_all();
}
group.join_all();
BOOST_TEST_EQ(data.woken,number_of_test_threads);
}
catch(...)
{
group.join_all();
throw;
}
}
void test_condition_notify_all_wakes_from_timed_wait()
{
timed_test(&do_test_condition_notify_all_wakes_from_timed_wait, timeout_seconds);
}
void do_test_condition_notify_all_wakes_from_timed_wait_with_predicate()
{
wait_for_flag data;
boost::thread_group group;
try
{
for(unsigned i=0;i<number_of_test_threads;++i)
{
group.create_thread(bind_thread_func(&wait_for_flag::timed_wait_with_predicate, data));
}
{
boost::sync::unique_lock<boost::sync::mutex> lock(data.mutex);
data.flag=true;
data.cond_var.notify_all();
}
group.join_all();
BOOST_TEST_EQ(data.woken,number_of_test_threads);
}
catch(...)
{
group.join_all();
throw;
}
}
void test_condition_notify_all_wakes_from_timed_wait_with_predicate()
{
timed_test(&do_test_condition_notify_all_wakes_from_timed_wait_with_predicate, timeout_seconds);
}
void do_test_condition_notify_all_wakes_from_relative_timed_wait_with_predicate()
{
wait_for_flag data;
boost::thread_group group;
try
{
for(unsigned i=0;i<number_of_test_threads;++i)
{
group.create_thread(bind_thread_func(&wait_for_flag::relative_timed_wait_with_predicate, data));
}
{
boost::sync::unique_lock<boost::sync::mutex> lock(data.mutex);
data.flag=true;
data.cond_var.notify_all();
}
group.join_all();
BOOST_TEST_EQ(data.woken,number_of_test_threads);
}
catch(...)
{
group.join_all();
throw;
}
}
void test_condition_notify_all_wakes_from_relative_timed_wait_with_predicate()
{
timed_test(&do_test_condition_notify_all_wakes_from_relative_timed_wait_with_predicate, timeout_seconds);
}
namespace {
boost::sync::mutex multiple_wake_mutex;
boost::sync::condition_variable multiple_wake_cond;
unsigned multiple_wake_count = 0;
void wait_for_condvar_and_increase_count()
{
boost::sync::unique_lock<boost::sync::mutex> lk(multiple_wake_mutex);
multiple_wake_cond.wait(lk);
++multiple_wake_count;
}
} // namespace
void do_test_notify_all_following_notify_one_wakes_all_threads()
{
boost::thread thread1(wait_for_condvar_and_increase_count);
boost::thread thread2(wait_for_condvar_and_increase_count);
boost::this_thread::sleep(boost::posix_time::milliseconds(200));
multiple_wake_cond.notify_one();
boost::thread thread3(wait_for_condvar_and_increase_count);
boost::this_thread::sleep(boost::posix_time::milliseconds(200));
multiple_wake_cond.notify_one();
multiple_wake_cond.notify_all();
boost::this_thread::sleep(boost::posix_time::milliseconds(200));
{
boost::sync::unique_lock<boost::sync::mutex> lk(multiple_wake_mutex);
BOOST_TEST_EQ(multiple_wake_count, 3u);
}
thread1.join();
thread2.join();
thread3.join();
}
void test_notify_all_following_notify_one_wakes_all_threads()
{
timed_test(&do_test_notify_all_following_notify_one_wakes_all_threads, timeout_seconds);
}
int main()
{
test_condition_notify_all_wakes_from_wait();
test_condition_notify_all_wakes_from_wait_with_predicate();
test_condition_notify_all_wakes_from_timed_wait();
test_condition_notify_all_wakes_from_timed_wait_with_predicate();
test_condition_notify_all_wakes_from_relative_timed_wait_with_predicate();
test_notify_all_following_notify_one_wakes_all_threads();
return boost::report_errors();
}