1237 lines
34 KiB
C++
1237 lines
34 KiB
C++
// (C) Copyright 2008-10 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)
|
|
|
|
#define BOOST_THREAD_VERSION 2
|
|
#define BOOST_TEST_MODULE Boost.Threads: futures test suite
|
|
|
|
#include <boost/thread/thread_only.hpp>
|
|
#include <boost/thread/mutex.hpp>
|
|
#include <boost/thread/condition.hpp>
|
|
#include <boost/thread/future.hpp>
|
|
#include <utility>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <iostream>
|
|
#include <boost/thread/detail/log.hpp>
|
|
|
|
#include <boost/test/unit_test.hpp>
|
|
|
|
#ifdef BOOST_MSVC
|
|
# pragma warning(disable: 4267) // conversion from ... to ..., possible loss of data
|
|
#endif
|
|
|
|
#ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
|
|
template<typename T>
|
|
typename boost::remove_reference<T>::type&& cast_to_rval(T&& t)
|
|
{
|
|
return static_cast<typename boost::remove_reference<T>::type&&>(t);
|
|
}
|
|
#else
|
|
#if defined BOOST_THREAD_USES_MOVE
|
|
template<typename T>
|
|
boost::rv<T>& cast_to_rval(T& t)
|
|
{
|
|
return boost::move(t);
|
|
}
|
|
#else
|
|
template<typename T>
|
|
boost::detail::thread_move_t<T> cast_to_rval(T& t)
|
|
{
|
|
return boost::move(t);
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
struct X
|
|
{
|
|
public:
|
|
int i;
|
|
|
|
BOOST_THREAD_MOVABLE_ONLY(X)
|
|
X():
|
|
i(42)
|
|
{}
|
|
X(BOOST_THREAD_RV_REF(X) other):
|
|
i(BOOST_THREAD_RV(other).i)
|
|
{
|
|
BOOST_THREAD_RV(other).i=0;
|
|
}
|
|
X& operator=(BOOST_THREAD_RV_REF(X) other)
|
|
{
|
|
i=BOOST_THREAD_RV(other).i;
|
|
BOOST_THREAD_RV(other).i=0;
|
|
return *this;
|
|
}
|
|
~X()
|
|
{}
|
|
};
|
|
namespace boost {
|
|
BOOST_THREAD_DCL_MOVABLE(X)
|
|
}
|
|
|
|
int make_int()
|
|
{
|
|
return 42;
|
|
}
|
|
|
|
int throw_runtime_error()
|
|
{
|
|
throw std::runtime_error("42");
|
|
}
|
|
|
|
void set_promise_thread(boost::promise<int>* p)
|
|
{
|
|
p->set_value(42);
|
|
}
|
|
|
|
struct my_exception
|
|
{};
|
|
|
|
void set_promise_exception_thread(boost::promise<int>* p)
|
|
{
|
|
p->set_exception(boost::copy_exception(my_exception()));
|
|
}
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(test_store_value_from_thread)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
try {
|
|
boost::promise<int> pi2;
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::unique_future<int> fi2(BOOST_THREAD_MAKE_RV_REF(pi2.get_future()));
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::thread(set_promise_thread,&pi2);
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
int j=fi2.get();
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
BOOST_CHECK(j==42);
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
BOOST_CHECK(fi2.is_ready());
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
BOOST_CHECK(fi2.has_value());
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
BOOST_CHECK(!fi2.has_exception());
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
BOOST_CHECK(fi2.get_state()==boost::future_state::ready);
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
}
|
|
catch (...)
|
|
{
|
|
BOOST_CHECK(false&&"Exception thrown");
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_store_exception)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<int> pi3;
|
|
boost::unique_future<int> fi3(BOOST_THREAD_MAKE_RV_REF(pi3.get_future()));
|
|
boost::thread(set_promise_exception_thread,&pi3);
|
|
try
|
|
{
|
|
fi3.get();
|
|
BOOST_CHECK(false);
|
|
}
|
|
catch(my_exception)
|
|
{
|
|
BOOST_CHECK(true);
|
|
}
|
|
|
|
BOOST_CHECK(fi3.is_ready());
|
|
BOOST_CHECK(!fi3.has_value());
|
|
BOOST_CHECK(fi3.has_exception());
|
|
BOOST_CHECK(fi3.get_state()==boost::future_state::ready);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_initial_state)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::unique_future<int> fi;
|
|
BOOST_CHECK(!fi.is_ready());
|
|
BOOST_CHECK(!fi.has_value());
|
|
BOOST_CHECK(!fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::uninitialized);
|
|
int i;
|
|
try
|
|
{
|
|
i=fi.get();
|
|
(void)i;
|
|
BOOST_CHECK(false);
|
|
}
|
|
catch(boost::future_uninitialized)
|
|
{
|
|
BOOST_CHECK(true);
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_waiting_future)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<int> pi;
|
|
boost::unique_future<int> fi;
|
|
fi=BOOST_THREAD_MAKE_RV_REF(pi.get_future());
|
|
|
|
int i=0;
|
|
BOOST_CHECK(!fi.is_ready());
|
|
BOOST_CHECK(!fi.has_value());
|
|
BOOST_CHECK(!fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::waiting);
|
|
BOOST_CHECK(i==0);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_cannot_get_future_twice)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<int> pi;
|
|
BOOST_THREAD_MAKE_RV_REF(pi.get_future());
|
|
|
|
try
|
|
{
|
|
pi.get_future();
|
|
BOOST_CHECK(false);
|
|
}
|
|
catch(boost::future_already_retrieved&)
|
|
{
|
|
BOOST_CHECK(true);
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_set_value_updates_future_state)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<int> pi;
|
|
boost::unique_future<int> fi;
|
|
fi=BOOST_THREAD_MAKE_RV_REF(pi.get_future());
|
|
|
|
pi.set_value(42);
|
|
|
|
BOOST_CHECK(fi.is_ready());
|
|
BOOST_CHECK(fi.has_value());
|
|
BOOST_CHECK(!fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::ready);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_set_value_can_be_retrieved)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<int> pi;
|
|
boost::unique_future<int> fi;
|
|
fi=BOOST_THREAD_MAKE_RV_REF(pi.get_future());
|
|
|
|
pi.set_value(42);
|
|
|
|
int i=0;
|
|
BOOST_CHECK(i=fi.get());
|
|
BOOST_CHECK(i==42);
|
|
BOOST_CHECK(fi.is_ready());
|
|
BOOST_CHECK(fi.has_value());
|
|
BOOST_CHECK(!fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::ready);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_set_value_can_be_moved)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
// boost::promise<int> pi;
|
|
// boost::unique_future<int> fi;
|
|
// fi=BOOST_THREAD_MAKE_RV_REF(pi.get_future());
|
|
|
|
// pi.set_value(42);
|
|
|
|
// int i=0;
|
|
// BOOST_CHECK(i=fi.get());
|
|
// BOOST_CHECK(i==42);
|
|
// BOOST_CHECK(fi.is_ready());
|
|
// BOOST_CHECK(fi.has_value());
|
|
// BOOST_CHECK(!fi.has_exception());
|
|
// BOOST_CHECK(fi.get_state()==boost::future_state::ready);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_future_from_packaged_task_is_waiting)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int);
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
int i=0;
|
|
BOOST_CHECK(!fi.is_ready());
|
|
BOOST_CHECK(!fi.has_value());
|
|
BOOST_CHECK(!fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::waiting);
|
|
BOOST_CHECK(i==0);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_invoking_a_packaged_task_populates_future)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int);
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
pt();
|
|
|
|
int i=0;
|
|
BOOST_CHECK(fi.is_ready());
|
|
BOOST_CHECK(fi.has_value());
|
|
BOOST_CHECK(!fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::ready);
|
|
BOOST_CHECK(i=fi.get());
|
|
BOOST_CHECK(i==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_invoking_a_packaged_task_twice_throws)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int);
|
|
|
|
pt();
|
|
try
|
|
{
|
|
pt();
|
|
BOOST_CHECK(false);
|
|
}
|
|
catch(boost::task_already_started)
|
|
{
|
|
BOOST_CHECK(true);
|
|
}
|
|
}
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(test_cannot_get_future_twice_from_task)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int);
|
|
pt.get_future();
|
|
try
|
|
{
|
|
pt.get_future();
|
|
BOOST_CHECK(false);
|
|
}
|
|
catch(boost::future_already_retrieved)
|
|
{
|
|
BOOST_CHECK(true);
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_task_stores_exception_if_function_throws)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(throw_runtime_error);
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
pt();
|
|
|
|
BOOST_CHECK(fi.is_ready());
|
|
BOOST_CHECK(!fi.has_value());
|
|
BOOST_CHECK(fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::ready);
|
|
try
|
|
{
|
|
fi.get();
|
|
BOOST_CHECK(false);
|
|
}
|
|
catch(std::exception&)
|
|
{
|
|
BOOST_CHECK(true);
|
|
}
|
|
catch(...)
|
|
{
|
|
BOOST_CHECK(!"Unknown exception thrown");
|
|
}
|
|
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_void_promise)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<void> p;
|
|
boost::unique_future<void> f(BOOST_THREAD_MAKE_RV_REF(p.get_future()));
|
|
p.set_value();
|
|
BOOST_CHECK(f.is_ready());
|
|
BOOST_CHECK(f.has_value());
|
|
BOOST_CHECK(!f.has_exception());
|
|
BOOST_CHECK(f.get_state()==boost::future_state::ready);
|
|
f.get();
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_reference_promise)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<int&> p;
|
|
boost::unique_future<int&> f(BOOST_THREAD_MAKE_RV_REF(p.get_future()));
|
|
int i=42;
|
|
p.set_value(i);
|
|
BOOST_CHECK(f.is_ready());
|
|
BOOST_CHECK(f.has_value());
|
|
BOOST_CHECK(!f.has_exception());
|
|
BOOST_CHECK(f.get_state()==boost::future_state::ready);
|
|
BOOST_CHECK(&f.get()==&i);
|
|
}
|
|
|
|
void do_nothing()
|
|
{}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_task_returning_void)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<void> pt(do_nothing);
|
|
boost::unique_future<void> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
pt();
|
|
|
|
BOOST_CHECK(fi.is_ready());
|
|
BOOST_CHECK(fi.has_value());
|
|
BOOST_CHECK(!fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::ready);
|
|
}
|
|
|
|
int global_ref_target=0;
|
|
|
|
int& return_ref()
|
|
{
|
|
return global_ref_target;
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_task_returning_reference)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int&> pt(return_ref);
|
|
boost::unique_future<int&> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
pt();
|
|
|
|
BOOST_CHECK(fi.is_ready());
|
|
BOOST_CHECK(fi.has_value());
|
|
BOOST_CHECK(!fi.has_exception());
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::ready);
|
|
int& i=fi.get();
|
|
BOOST_CHECK(&i==&global_ref_target);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_shared_future)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int);
|
|
boost::unique_future<int> fi=pt.get_future();
|
|
|
|
boost::shared_future<int> sf(::cast_to_rval(fi));
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::uninitialized);
|
|
|
|
pt();
|
|
|
|
int i=0;
|
|
BOOST_CHECK(sf.is_ready());
|
|
BOOST_CHECK(sf.has_value());
|
|
BOOST_CHECK(!sf.has_exception());
|
|
BOOST_CHECK(sf.get_state()==boost::future_state::ready);
|
|
BOOST_CHECK(i=sf.get());
|
|
BOOST_CHECK(i==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_copies_of_shared_future_become_ready_together)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int);
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
boost::shared_future<int> sf(::cast_to_rval(fi));
|
|
boost::shared_future<int> sf2(sf);
|
|
boost::shared_future<int> sf3;
|
|
sf3=sf;
|
|
BOOST_CHECK(sf.get_state()==boost::future_state::waiting);
|
|
BOOST_CHECK(sf2.get_state()==boost::future_state::waiting);
|
|
BOOST_CHECK(sf3.get_state()==boost::future_state::waiting);
|
|
|
|
pt();
|
|
|
|
int i=0;
|
|
BOOST_CHECK(sf.is_ready());
|
|
BOOST_CHECK(sf.has_value());
|
|
BOOST_CHECK(!sf.has_exception());
|
|
BOOST_CHECK(sf.get_state()==boost::future_state::ready);
|
|
BOOST_CHECK(i=sf.get());
|
|
BOOST_CHECK(i==42);
|
|
i=0;
|
|
BOOST_CHECK(sf2.is_ready());
|
|
BOOST_CHECK(sf2.has_value());
|
|
BOOST_CHECK(!sf2.has_exception());
|
|
BOOST_CHECK(sf2.get_state()==boost::future_state::ready);
|
|
BOOST_CHECK(i=sf2.get());
|
|
BOOST_CHECK(i==42);
|
|
i=0;
|
|
BOOST_CHECK(sf3.is_ready());
|
|
BOOST_CHECK(sf3.has_value());
|
|
BOOST_CHECK(!sf3.has_exception());
|
|
BOOST_CHECK(sf3.get_state()==boost::future_state::ready);
|
|
BOOST_CHECK(i=sf3.get());
|
|
BOOST_CHECK(i==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_shared_future_can_be_move_assigned_from_unique_future)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int);
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
boost::shared_future<int> sf;
|
|
sf=::cast_to_rval(fi);
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::uninitialized);
|
|
|
|
BOOST_CHECK(!sf.is_ready());
|
|
BOOST_CHECK(!sf.has_value());
|
|
BOOST_CHECK(!sf.has_exception());
|
|
BOOST_CHECK(sf.get_state()==boost::future_state::waiting);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_shared_future_void)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<void> pt(do_nothing);
|
|
boost::unique_future<void> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
boost::shared_future<void> sf(::cast_to_rval(fi));
|
|
BOOST_CHECK(fi.get_state()==boost::future_state::uninitialized);
|
|
|
|
pt();
|
|
|
|
BOOST_CHECK(sf.is_ready());
|
|
BOOST_CHECK(sf.has_value());
|
|
BOOST_CHECK(!sf.has_exception());
|
|
BOOST_CHECK(sf.get_state()==boost::future_state::ready);
|
|
sf.get();
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_shared_future_ref)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<int&> p;
|
|
boost::shared_future<int&> f(BOOST_THREAD_MAKE_RV_REF(p.get_future()));
|
|
int i=42;
|
|
p.set_value(i);
|
|
BOOST_CHECK(f.is_ready());
|
|
BOOST_CHECK(f.has_value());
|
|
BOOST_CHECK(!f.has_exception());
|
|
BOOST_CHECK(f.get_state()==boost::future_state::ready);
|
|
BOOST_CHECK(&f.get()==&i);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_can_get_a_second_future_from_a_moved_promise)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<int> pi;
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pi.get_future()));
|
|
|
|
boost::promise<int> pi2(::cast_to_rval(pi));
|
|
boost::unique_future<int> fi2(BOOST_THREAD_MAKE_RV_REF(pi.get_future()));
|
|
|
|
pi2.set_value(3);
|
|
BOOST_CHECK(fi.is_ready());
|
|
BOOST_CHECK(!fi2.is_ready());
|
|
BOOST_CHECK(fi.get()==3);
|
|
pi.set_value(42);
|
|
BOOST_CHECK(fi2.is_ready());
|
|
BOOST_CHECK(fi2.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_can_get_a_second_future_from_a_moved_void_promise)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<void> pi;
|
|
boost::unique_future<void> fi(BOOST_THREAD_MAKE_RV_REF(pi.get_future()));
|
|
|
|
boost::promise<void> pi2(::cast_to_rval(pi));
|
|
boost::unique_future<void> fi2(BOOST_THREAD_MAKE_RV_REF(pi.get_future()));
|
|
|
|
pi2.set_value();
|
|
BOOST_CHECK(fi.is_ready());
|
|
BOOST_CHECK(!fi2.is_ready());
|
|
pi.set_value();
|
|
BOOST_CHECK(fi2.is_ready());
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_unique_future_for_move_only_udt)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<X> pt;
|
|
boost::unique_future<X> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
pt.set_value(X());
|
|
X res(fi.get());
|
|
BOOST_CHECK(res.i==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_unique_future_for_string)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::promise<std::string> pt;
|
|
boost::unique_future<std::string> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
pt.set_value(std::string("hello"));
|
|
std::string res(fi.get());
|
|
BOOST_CHECK(res=="hello");
|
|
|
|
boost::promise<std::string> pt2;
|
|
fi=BOOST_THREAD_MAKE_RV_REF(pt2.get_future());
|
|
|
|
std::string const s="goodbye";
|
|
|
|
pt2.set_value(s);
|
|
res=fi.get();
|
|
BOOST_CHECK(res=="goodbye");
|
|
|
|
boost::promise<std::string> pt3;
|
|
fi=BOOST_THREAD_MAKE_RV_REF(pt3.get_future());
|
|
|
|
std::string s2="foo";
|
|
|
|
pt3.set_value(s2);
|
|
res=fi.get();
|
|
BOOST_CHECK(res=="foo");
|
|
}
|
|
|
|
boost::mutex callback_mutex;
|
|
unsigned callback_called=0;
|
|
|
|
void wait_callback(boost::promise<int>& pi)
|
|
{
|
|
boost::lock_guard<boost::mutex> lk(callback_mutex);
|
|
++callback_called;
|
|
try
|
|
{
|
|
pi.set_value(42);
|
|
}
|
|
catch(...)
|
|
{
|
|
}
|
|
}
|
|
|
|
void do_nothing_callback(boost::promise<int>& /*pi*/)
|
|
{
|
|
boost::lock_guard<boost::mutex> lk(callback_mutex);
|
|
++callback_called;
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_callback)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
callback_called=0;
|
|
boost::promise<int> pi;
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pi.get_future()));
|
|
pi.set_wait_callback(wait_callback);
|
|
fi.wait();
|
|
BOOST_CHECK(callback_called);
|
|
BOOST_CHECK(fi.get()==42);
|
|
fi.wait();
|
|
fi.wait();
|
|
BOOST_CHECK(callback_called==1);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_callback_with_timed_wait)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
callback_called=0;
|
|
boost::promise<int> pi;
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pi.get_future()));
|
|
pi.set_wait_callback(do_nothing_callback);
|
|
bool success=fi.timed_wait(boost::posix_time::milliseconds(10));
|
|
BOOST_CHECK(callback_called);
|
|
BOOST_CHECK(!success);
|
|
success=fi.timed_wait(boost::posix_time::milliseconds(10));
|
|
BOOST_CHECK(!success);
|
|
success=fi.timed_wait(boost::posix_time::milliseconds(10));
|
|
BOOST_CHECK(!success);
|
|
BOOST_CHECK(callback_called==3);
|
|
pi.set_value(42);
|
|
success=fi.timed_wait(boost::posix_time::milliseconds(10));
|
|
BOOST_CHECK(success);
|
|
BOOST_CHECK(callback_called==3);
|
|
BOOST_CHECK(fi.get()==42);
|
|
BOOST_CHECK(callback_called==3);
|
|
}
|
|
|
|
|
|
void wait_callback_for_task(boost::packaged_task<int>& pt)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::lock_guard<boost::mutex> lk(callback_mutex);
|
|
++callback_called;
|
|
try
|
|
{
|
|
pt();
|
|
}
|
|
catch(...)
|
|
{
|
|
}
|
|
}
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_callback_for_packaged_task)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
callback_called=0;
|
|
boost::packaged_task<int> pt(make_int);
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
pt.set_wait_callback(wait_callback_for_task);
|
|
fi.wait();
|
|
BOOST_CHECK(callback_called);
|
|
BOOST_CHECK(fi.get()==42);
|
|
fi.wait();
|
|
fi.wait();
|
|
BOOST_CHECK(callback_called==1);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_packaged_task_can_be_moved)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int);
|
|
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
|
|
BOOST_CHECK(!fi.is_ready());
|
|
|
|
boost::packaged_task<int> pt2(::cast_to_rval(pt));
|
|
|
|
BOOST_CHECK(!fi.is_ready());
|
|
try
|
|
{
|
|
pt();
|
|
BOOST_CHECK(!"Can invoke moved task!");
|
|
}
|
|
catch(boost::task_moved&)
|
|
{
|
|
}
|
|
|
|
BOOST_CHECK(!fi.is_ready());
|
|
|
|
pt2();
|
|
|
|
BOOST_CHECK(fi.is_ready());
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_destroying_a_promise_stores_broken_promise)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::unique_future<int> f;
|
|
|
|
{
|
|
boost::promise<int> p;
|
|
f=BOOST_THREAD_MAKE_RV_REF(p.get_future());
|
|
}
|
|
BOOST_CHECK(f.is_ready());
|
|
BOOST_CHECK(f.has_exception());
|
|
try
|
|
{
|
|
f.get();
|
|
}
|
|
catch(boost::broken_promise&)
|
|
{
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_destroying_a_packaged_task_stores_broken_promise)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::unique_future<int> f;
|
|
|
|
{
|
|
boost::packaged_task<int> p(make_int);
|
|
f=BOOST_THREAD_MAKE_RV_REF(p.get_future());
|
|
}
|
|
BOOST_CHECK(f.is_ready());
|
|
BOOST_CHECK(f.has_exception());
|
|
try
|
|
{
|
|
f.get();
|
|
}
|
|
catch(boost::broken_promise&)
|
|
{
|
|
}
|
|
}
|
|
|
|
int make_int_slowly()
|
|
{
|
|
boost::this_thread::sleep(boost::posix_time::seconds(1));
|
|
return 42;
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_two_futures_1)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2);
|
|
|
|
BOOST_CHECK(future==0);
|
|
BOOST_CHECK(f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(f1.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_two_futures_2)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt2));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2);
|
|
|
|
BOOST_CHECK(future==1);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(f2.is_ready());
|
|
BOOST_CHECK(f2.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_three_futures_1)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3);
|
|
|
|
BOOST_CHECK(future==0);
|
|
BOOST_CHECK(f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(f1.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_three_futures_2)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt2));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3);
|
|
|
|
BOOST_CHECK(future==1);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(f2.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_three_futures_3)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt3));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3);
|
|
|
|
BOOST_CHECK(future==2);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(f3.is_ready());
|
|
BOOST_CHECK(f3.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_four_futures_1)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4);
|
|
|
|
BOOST_CHECK(future==0);
|
|
BOOST_CHECK(f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(!f4.is_ready());
|
|
BOOST_CHECK(f1.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_four_futures_2)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt2));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4);
|
|
|
|
BOOST_CHECK(future==1);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(!f4.is_ready());
|
|
BOOST_CHECK(f2.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_four_futures_3)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt3));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4);
|
|
|
|
BOOST_CHECK(future==2);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(f3.is_ready());
|
|
BOOST_CHECK(!f4.is_ready());
|
|
BOOST_CHECK(f3.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_four_futures_4)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt4));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4);
|
|
|
|
BOOST_CHECK(future==3);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(f4.is_ready());
|
|
BOOST_CHECK(f4.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_five_futures_1)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
boost::packaged_task<int> pt5(make_int_slowly);
|
|
boost::unique_future<int> f5(BOOST_THREAD_MAKE_RV_REF(pt5.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4,f5);
|
|
|
|
BOOST_CHECK(future==0);
|
|
BOOST_CHECK(f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(!f4.is_ready());
|
|
BOOST_CHECK(!f5.is_ready());
|
|
BOOST_CHECK(f1.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_five_futures_2)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
boost::packaged_task<int> pt5(make_int_slowly);
|
|
boost::unique_future<int> f5(BOOST_THREAD_MAKE_RV_REF(pt5.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt2));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4,f5);
|
|
|
|
BOOST_CHECK(future==1);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(!f4.is_ready());
|
|
BOOST_CHECK(!f5.is_ready());
|
|
BOOST_CHECK(f2.get()==42);
|
|
}
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_five_futures_3)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
boost::packaged_task<int> pt5(make_int_slowly);
|
|
boost::unique_future<int> f5(BOOST_THREAD_MAKE_RV_REF(pt5.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt3));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4,f5);
|
|
|
|
BOOST_CHECK(future==2);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(f3.is_ready());
|
|
BOOST_CHECK(!f4.is_ready());
|
|
BOOST_CHECK(!f5.is_ready());
|
|
BOOST_CHECK(f3.get()==42);
|
|
}
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_five_futures_4)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
boost::packaged_task<int> pt5(make_int_slowly);
|
|
boost::unique_future<int> f5(BOOST_THREAD_MAKE_RV_REF(pt5.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt4));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4,f5);
|
|
|
|
BOOST_CHECK(future==3);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(f4.is_ready());
|
|
BOOST_CHECK(!f5.is_ready());
|
|
BOOST_CHECK(f4.get()==42);
|
|
}
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_of_five_futures_5)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> f1(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> f2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
boost::packaged_task<int> pt3(make_int_slowly);
|
|
boost::unique_future<int> f3(BOOST_THREAD_MAKE_RV_REF(pt3.get_future()));
|
|
boost::packaged_task<int> pt4(make_int_slowly);
|
|
boost::unique_future<int> f4(BOOST_THREAD_MAKE_RV_REF(pt4.get_future()));
|
|
boost::packaged_task<int> pt5(make_int_slowly);
|
|
boost::unique_future<int> f5(BOOST_THREAD_MAKE_RV_REF(pt5.get_future()));
|
|
|
|
boost::thread(::cast_to_rval(pt5));
|
|
|
|
unsigned const future=boost::wait_for_any(f1,f2,f3,f4,f5);
|
|
|
|
BOOST_CHECK(future==4);
|
|
BOOST_CHECK(!f1.is_ready());
|
|
BOOST_CHECK(!f2.is_ready());
|
|
BOOST_CHECK(!f3.is_ready());
|
|
BOOST_CHECK(!f4.is_ready());
|
|
BOOST_CHECK(f5.is_ready());
|
|
BOOST_CHECK(f5.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_either_invokes_callbacks)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
callback_called=0;
|
|
boost::packaged_task<int> pt(make_int_slowly);
|
|
boost::unique_future<int> fi(BOOST_THREAD_MAKE_RV_REF(pt.get_future()));
|
|
boost::packaged_task<int> pt2(make_int_slowly);
|
|
boost::unique_future<int> fi2(BOOST_THREAD_MAKE_RV_REF(pt2.get_future()));
|
|
pt.set_wait_callback(wait_callback_for_task);
|
|
|
|
boost::thread(::cast_to_rval(pt));
|
|
|
|
boost::wait_for_any(fi,fi2);
|
|
BOOST_CHECK(callback_called==1);
|
|
BOOST_CHECK(fi.get()==42);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_any_from_range)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
unsigned const count=10;
|
|
for(unsigned i=0;i<count;++i)
|
|
{
|
|
boost::packaged_task<int> tasks[count];
|
|
boost::unique_future<int> futures[count];
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
tasks[j]=boost::packaged_task<int>(make_int_slowly);
|
|
futures[j]=BOOST_THREAD_MAKE_RV_REF(tasks[j].get_future());
|
|
}
|
|
boost::thread(::cast_to_rval(tasks[i]));
|
|
|
|
BOOST_CHECK(boost::wait_for_any(futures,futures)==futures);
|
|
|
|
boost::unique_future<int>* const future=boost::wait_for_any(futures,futures+count);
|
|
|
|
BOOST_CHECK(future==(futures+i));
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
if(j!=i)
|
|
{
|
|
BOOST_CHECK(!futures[j].is_ready());
|
|
}
|
|
else
|
|
{
|
|
BOOST_CHECK(futures[j].is_ready());
|
|
}
|
|
}
|
|
BOOST_CHECK(futures[i].get()==42);
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_all_from_range)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
unsigned const count=10;
|
|
boost::unique_future<int> futures[count];
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
boost::packaged_task<int> task(make_int_slowly);
|
|
futures[j]=BOOST_THREAD_MAKE_RV_REF(task.get_future());
|
|
boost::thread(::cast_to_rval(task));
|
|
}
|
|
|
|
boost::wait_for_all(futures,futures+count);
|
|
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
BOOST_CHECK(futures[j].is_ready());
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_all_two_futures)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
unsigned const count=2;
|
|
boost::unique_future<int> futures[count];
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
boost::packaged_task<int> task(make_int_slowly);
|
|
futures[j]=BOOST_THREAD_MAKE_RV_REF(task.get_future());
|
|
boost::thread(::cast_to_rval(task));
|
|
}
|
|
|
|
boost::wait_for_all(futures[0],futures[1]);
|
|
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
BOOST_CHECK(futures[j].is_ready());
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_all_three_futures)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
unsigned const count=3;
|
|
boost::unique_future<int> futures[count];
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
boost::packaged_task<int> task(make_int_slowly);
|
|
futures[j]=BOOST_THREAD_MAKE_RV_REF(task.get_future());
|
|
boost::thread(::cast_to_rval(task));
|
|
}
|
|
|
|
boost::wait_for_all(futures[0],futures[1],futures[2]);
|
|
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
BOOST_CHECK(futures[j].is_ready());
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_all_four_futures)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
unsigned const count=4;
|
|
boost::unique_future<int> futures[count];
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
boost::packaged_task<int> task(make_int_slowly);
|
|
futures[j]=BOOST_THREAD_MAKE_RV_REF(task.get_future());
|
|
boost::thread(::cast_to_rval(task));
|
|
}
|
|
|
|
boost::wait_for_all(futures[0],futures[1],futures[2],futures[3]);
|
|
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
BOOST_CHECK(futures[j].is_ready());
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_wait_for_all_five_futures)
|
|
{
|
|
BOOST_DETAIL_THREAD_LOG;
|
|
unsigned const count=5;
|
|
boost::unique_future<int> futures[count];
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
boost::packaged_task<int> task(make_int_slowly);
|
|
futures[j]=BOOST_THREAD_MAKE_RV_REF(task.get_future());
|
|
boost::thread(::cast_to_rval(task));
|
|
}
|
|
|
|
boost::wait_for_all(futures[0],futures[1],futures[2],futures[3],futures[4]);
|
|
|
|
for(unsigned j=0;j<count;++j)
|
|
{
|
|
BOOST_CHECK(futures[j].is_ready());
|
|
}
|
|
}
|