mirror of
https://github.com/boostorg/thread.git
synced 2026-01-24 06:22:12 +00:00
* [@http://svn.boost.org/trac/boost/ticket/2309 #2309] Lack of g++ symbol visibility support in Boost.Thread. * [@http://svn.boost.org/trac/boost/ticket/2639 #2639] documentation should be extended(defer_lock, try_to_lock, ...). * [@http://svn.boost.org/trac/boost/ticket/3639 #3639] Boost.Thread doesn't build with Sun-5.9 on Linux. * [@http://svn.boost.org/trac/boost/ticket/3762 #3762] Thread can't be compiled with winscw (Codewarrior by Nokia). * [@http://svn.boost.org/trac/boost/ticket/3885 #3885] document about mix usage of boost.thread and native thread api. * [@http://svn.boost.org/trac/boost/ticket/3975 #3975] Incorrect precondition for promise::set_wait_callback(). * [@http://svn.boost.org/trac/boost/ticket/4048 #4048] thread::id formatting involves locale * [@http://svn.boost.org/trac/boost/ticket/4315 #4315] gcc 4.4 Warning: inline ... declared as dllimport: attribute ignored. * [@http://svn.boost.org/trac/boost/ticket/4480 #4480] OpenVMS patches for compiler issues workarounds. * [@http://svn.boost.org/trac/boost/ticket/4819 #4819] boost.thread's documentation misprints. * [@http://svn.boost.org/trac/boost/ticket/5423 #5423] thread issues with C++0x. * [@http://svn.boost.org/trac/boost/ticket/5617 #5617] boost::thread::id copy ctor. * [@http://svn.boost.org/trac/boost/ticket/5739 #5739] set-but-not-used warnings with gcc-4.6. * [@http://svn.boost.org/trac/boost/ticket/5826 #5826] threads.cpp: resource leak on threads creation failure. * [@http://svn.boost.org/trac/boost/ticket/5839 #5839] thread.cpp: ThreadProxy leaks on exceptions. * [@http://svn.boost.org/trac/boost/ticket/5859 #5859] win32 shared_mutex constructor leaks on exceptions. * [@http://svn.boost.org/trac/boost/ticket/6100 #6100] Compute hardware_concurrency() using get_nprocs() on GLIBC systems. * [@http://svn.boost.org/trac/boost/ticket/6168 #6168] recursive_mutex is using wrong config symbol (possible typo). * [@http://svn.boost.org/trac/boost/ticket/6175 #6175] Compile error with SunStudio. * [@http://svn.boost.org/trac/boost/ticket/6200 #6200] patch to have condition_variable and mutex error better handle EINTR. * [@http://svn.boost.org/trac/boost/ticket/6207 #6207] shared_lock swap compiler error on clang 3.0 c++11. * [@http://svn.boost.org/trac/boost/ticket/6208 #6208] try_lock_wrapper swap compiler error on clang 3.0 c++11. [SVN r76291]
235 lines
5.4 KiB
C++
235 lines
5.4 KiB
C++
// Copyright (C) 2001-2003
|
|
// William E. Kempf
|
|
// Copyright (C) 2008 Anthony Williams
|
|
//
|
|
// 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/detail/config.hpp>
|
|
|
|
#include <boost/thread/thread.hpp>
|
|
#include <boost/thread/xtime.hpp>
|
|
#include <boost/bind.hpp>
|
|
#include <boost/ref.hpp>
|
|
#include <boost/utility.hpp>
|
|
|
|
#include <boost/test/unit_test.hpp>
|
|
|
|
#define DEFAULT_EXECUTION_MONITOR_TYPE execution_monitor::use_sleep_only
|
|
#include <libs/thread/test/util.inl>
|
|
|
|
int test_value;
|
|
|
|
void simple_thread()
|
|
{
|
|
test_value = 999;
|
|
}
|
|
|
|
void comparison_thread(boost::thread::id parent)
|
|
{
|
|
boost::thread::id const my_id=boost::this_thread::get_id();
|
|
|
|
BOOST_CHECK(my_id != parent);
|
|
boost::thread::id const my_id2=boost::this_thread::get_id();
|
|
BOOST_CHECK(my_id == my_id2);
|
|
|
|
boost::thread::id const no_thread_id=boost::thread::id();
|
|
BOOST_CHECK(my_id != no_thread_id);
|
|
}
|
|
|
|
void test_sleep()
|
|
{
|
|
boost::xtime xt = delay(3);
|
|
boost::thread::sleep(xt);
|
|
|
|
// Ensure it's in a range instead of checking actual equality due to time
|
|
// lapse
|
|
BOOST_CHECK(in_range(xt, 2));
|
|
}
|
|
|
|
void do_test_creation()
|
|
{
|
|
test_value = 0;
|
|
boost::thread thrd(&simple_thread);
|
|
thrd.join();
|
|
BOOST_CHECK_EQUAL(test_value, 999);
|
|
}
|
|
|
|
void test_creation()
|
|
{
|
|
timed_test(&do_test_creation, 1);
|
|
}
|
|
|
|
void do_test_id_comparison()
|
|
{
|
|
boost::thread::id const self=boost::this_thread::get_id();
|
|
boost::thread thrd(boost::bind(&comparison_thread, self));
|
|
thrd.join();
|
|
}
|
|
|
|
void test_id_comparison()
|
|
{
|
|
timed_test(&do_test_id_comparison, 1);
|
|
}
|
|
|
|
void interruption_point_thread(boost::mutex* m,bool* failed)
|
|
{
|
|
boost::mutex::scoped_lock lk(*m);
|
|
boost::this_thread::interruption_point();
|
|
*failed=true;
|
|
}
|
|
|
|
void do_test_thread_interrupts_at_interruption_point()
|
|
{
|
|
boost::mutex m;
|
|
bool failed=false;
|
|
boost::mutex::scoped_lock lk(m);
|
|
boost::thread thrd(boost::bind(&interruption_point_thread,&m,&failed));
|
|
thrd.interrupt();
|
|
lk.unlock();
|
|
thrd.join();
|
|
BOOST_CHECK(!failed);
|
|
}
|
|
|
|
void test_thread_interrupts_at_interruption_point()
|
|
{
|
|
timed_test(&do_test_thread_interrupts_at_interruption_point, 1);
|
|
}
|
|
|
|
void disabled_interruption_point_thread(boost::mutex* m,bool* failed)
|
|
{
|
|
boost::mutex::scoped_lock lk(*m);
|
|
boost::this_thread::disable_interruption dc;
|
|
boost::this_thread::interruption_point();
|
|
*failed=false;
|
|
}
|
|
|
|
void do_test_thread_no_interrupt_if_interrupts_disabled_at_interruption_point()
|
|
{
|
|
boost::mutex m;
|
|
bool failed=true;
|
|
boost::mutex::scoped_lock lk(m);
|
|
boost::thread thrd(boost::bind(&disabled_interruption_point_thread,&m,&failed));
|
|
thrd.interrupt();
|
|
lk.unlock();
|
|
thrd.join();
|
|
BOOST_CHECK(!failed);
|
|
}
|
|
|
|
void test_thread_no_interrupt_if_interrupts_disabled_at_interruption_point()
|
|
{
|
|
timed_test(&do_test_thread_no_interrupt_if_interrupts_disabled_at_interruption_point, 1);
|
|
}
|
|
|
|
struct non_copyable_functor:
|
|
boost::noncopyable
|
|
{
|
|
unsigned value;
|
|
|
|
non_copyable_functor():
|
|
value(0)
|
|
{}
|
|
|
|
void operator()()
|
|
{
|
|
value=999;
|
|
}
|
|
};
|
|
|
|
void do_test_creation_through_reference_wrapper()
|
|
{
|
|
non_copyable_functor f;
|
|
|
|
boost::thread thrd(boost::ref(f));
|
|
thrd.join();
|
|
BOOST_CHECK_EQUAL(f.value, 999u);
|
|
}
|
|
|
|
void test_creation_through_reference_wrapper()
|
|
{
|
|
timed_test(&do_test_creation_through_reference_wrapper, 1);
|
|
}
|
|
|
|
struct long_running_thread
|
|
{
|
|
boost::condition_variable cond;
|
|
boost::mutex mut;
|
|
bool done;
|
|
|
|
long_running_thread():
|
|
done(false)
|
|
{}
|
|
|
|
void operator()()
|
|
{
|
|
boost::mutex::scoped_lock lk(mut);
|
|
while(!done)
|
|
{
|
|
cond.wait(lk);
|
|
}
|
|
}
|
|
};
|
|
|
|
void do_test_timed_join()
|
|
{
|
|
long_running_thread f;
|
|
boost::thread thrd(boost::ref(f));
|
|
BOOST_CHECK(thrd.joinable());
|
|
boost::system_time xt=delay(3);
|
|
bool const joined=thrd.timed_join(xt);
|
|
BOOST_CHECK(in_range(boost::get_xtime(xt), 2));
|
|
BOOST_CHECK(!joined);
|
|
BOOST_CHECK(thrd.joinable());
|
|
{
|
|
boost::mutex::scoped_lock lk(f.mut);
|
|
f.done=true;
|
|
f.cond.notify_one();
|
|
}
|
|
|
|
xt=delay(3);
|
|
bool const joined2=thrd.timed_join(xt);
|
|
boost::system_time const now=boost::get_system_time();
|
|
BOOST_CHECK(xt>now);
|
|
BOOST_CHECK(joined2);
|
|
BOOST_CHECK(!thrd.joinable());
|
|
}
|
|
|
|
void test_timed_join()
|
|
{
|
|
timed_test(&do_test_timed_join, 10);
|
|
}
|
|
|
|
void test_swap()
|
|
{
|
|
boost::thread t(simple_thread);
|
|
boost::thread t2(simple_thread);
|
|
boost::thread::id id1=t.get_id();
|
|
boost::thread::id id2=t2.get_id();
|
|
|
|
t.swap(t2);
|
|
BOOST_CHECK(t.get_id()==id2);
|
|
BOOST_CHECK(t2.get_id()==id1);
|
|
|
|
swap(t,t2);
|
|
BOOST_CHECK(t.get_id()==id1);
|
|
BOOST_CHECK(t2.get_id()==id2);
|
|
}
|
|
|
|
|
|
boost::unit_test::test_suite* init_unit_test_suite(int, char*[])
|
|
{
|
|
boost::unit_test::test_suite* test =
|
|
BOOST_TEST_SUITE("Boost.Threads: thread test suite");
|
|
|
|
test->add(BOOST_TEST_CASE(test_sleep));
|
|
test->add(BOOST_TEST_CASE(test_creation));
|
|
test->add(BOOST_TEST_CASE(test_id_comparison));
|
|
test->add(BOOST_TEST_CASE(test_thread_interrupts_at_interruption_point));
|
|
test->add(BOOST_TEST_CASE(test_thread_no_interrupt_if_interrupts_disabled_at_interruption_point));
|
|
test->add(BOOST_TEST_CASE(test_creation_through_reference_wrapper));
|
|
test->add(BOOST_TEST_CASE(test_timed_join));
|
|
test->add(BOOST_TEST_CASE(test_swap));
|
|
|
|
return test;
|
|
}
|