Skip to content

Commit 8f5de1d

Browse files
committed
make condition_variable_any steady when BOOST_THREAD_HAS_CONDATTR_SET_CLOCK_MONOTONIC.
1 parent 0fdd4fc commit 8f5de1d

File tree

2 files changed

+112
-64
lines changed

2 files changed

+112
-64
lines changed

include/boost/thread/pthread/condition_variable.hpp

+86-35
Original file line numberDiff line numberDiff line change
@@ -156,11 +156,11 @@ namespace boost
156156
{
157157
boost::throw_exception(thread_resource_error(res, "boost::condition_variable_any::condition_variable_any() failed in pthread_mutex_init"));
158158
}
159-
int const res2=pthread_cond_init(&cond,NULL);
159+
int const res2 = detail::monotonic_pthread_cond_init(cond);
160160
if(res2)
161161
{
162162
BOOST_VERIFY(!pthread_mutex_destroy(&internal_mutex));
163-
boost::throw_exception(thread_resource_error(res2, "boost::condition_variable_any::condition_variable_any() failed in pthread_cond_init"));
163+
boost::throw_exception(thread_resource_error(res2, "boost::condition_variable_any::condition_variable_any() failed in detail::monotonic_pthread_cond_init"));
164164
}
165165
}
166166
~condition_variable_any()
@@ -240,6 +240,8 @@ namespace boost
240240
return timed_wait(m,get_system_time()+wait_duration,pred);
241241
}
242242
#endif
243+
#ifndef BOOST_THREAD_HAS_CONDATTR_SET_CLOCK_MONOTONIC
244+
243245
#ifdef BOOST_THREAD_USES_CHRONO
244246
template <class lock_type,class Duration>
245247
cv_status
@@ -268,22 +270,6 @@ namespace boost
268270
return Clock::now() < t ? cv_status::no_timeout : cv_status::timeout;
269271
}
270272

271-
template <class lock_type, class Clock, class Duration, class Predicate>
272-
bool
273-
wait_until(
274-
lock_type& lock,
275-
const chrono::time_point<Clock, Duration>& t,
276-
Predicate pred)
277-
{
278-
while (!pred())
279-
{
280-
if (wait_until(lock, t) == cv_status::timeout)
281-
return pred();
282-
}
283-
return true;
284-
}
285-
286-
287273
template <class lock_type, class Rep, class Period>
288274
cv_status
289275
wait_for(
@@ -299,35 +285,100 @@ namespace boost
299285

300286
}
301287

288+
template <class lock_type>
289+
cv_status wait_until(
290+
lock_type& lk,
291+
chrono::time_point<chrono::system_clock, chrono::nanoseconds> tp)
292+
{
293+
using namespace chrono;
294+
nanoseconds d = tp.time_since_epoch();
295+
timespec ts = boost::detail::to_timespec(d);
296+
if (do_wait_until(lk, ts)) return cv_status::no_timeout;
297+
else return cv_status::timeout;
298+
}
299+
#endif
300+
#else // defined BOOST_THREAD_HAS_CONDATTR_SET_CLOCK_MONOTONIC
301+
#ifdef BOOST_THREAD_USES_CHRONO
302302

303-
template <class lock_type, class Rep, class Period, class Predicate>
304-
bool
305-
wait_for(
306-
lock_type& lock,
307-
const chrono::duration<Rep, Period>& d,
308-
Predicate pred)
303+
template <class lock_type, class Duration>
304+
cv_status
305+
wait_until(
306+
lock_type& lock,
307+
const chrono::time_point<chrono::steady_clock, Duration>& t)
309308
{
310-
return wait_until(lock, chrono::steady_clock::now() + d, boost::move(pred));
309+
using namespace chrono;
310+
typedef time_point<steady_clock, nanoseconds> nano_sys_tmpt;
311+
wait_until(lock,
312+
nano_sys_tmpt(ceil<nanoseconds>(t.time_since_epoch())));
313+
return steady_clock::now() < t ? cv_status::no_timeout :
314+
cv_status::timeout;
315+
}
316+
317+
template <class lock_type, class Clock, class Duration>
318+
cv_status
319+
wait_until(
320+
lock_type& lock,
321+
const chrono::time_point<Clock, Duration>& t)
322+
{
323+
using namespace chrono;
324+
steady_clock::time_point s_now = steady_clock::now();
325+
typename Clock::time_point c_now = Clock::now();
326+
wait_until(lock, s_now + ceil<nanoseconds>(t - c_now));
327+
return Clock::now() < t ? cv_status::no_timeout : cv_status::timeout;
328+
}
311329

312-
// while (!pred())
313-
// {
314-
// if (wait_for(lock, d) == cv_status::timeout)
315-
// return pred();
316-
// }
317-
// return true;
330+
template <class lock_type, class Rep, class Period>
331+
cv_status
332+
wait_for(
333+
lock_type& lock,
334+
const chrono::duration<Rep, Period>& d)
335+
{
336+
using namespace chrono;
337+
steady_clock::time_point c_now = steady_clock::now();
338+
wait_until(lock, c_now + ceil<nanoseconds>(d));
339+
return steady_clock::now() - c_now < d ? cv_status::no_timeout :
340+
cv_status::timeout;
318341
}
319342

320-
template <class lock_type>
321-
cv_status wait_until(
322-
lock_type& lk,
323-
chrono::time_point<chrono::system_clock, chrono::nanoseconds> tp)
343+
inline cv_status wait_until(
344+
unique_lock<mutex>& lk,
345+
chrono::time_point<chrono::steady_clock, chrono::nanoseconds> tp)
324346
{
325347
using namespace chrono;
326348
nanoseconds d = tp.time_since_epoch();
327349
timespec ts = boost::detail::to_timespec(d);
328350
if (do_wait_until(lk, ts)) return cv_status::no_timeout;
329351
else return cv_status::timeout;
330352
}
353+
354+
#endif
355+
#endif // defined BOOST_THREAD_HAS_CONDATTR_SET_CLOCK_MONOTONIC
356+
357+
#ifdef BOOST_THREAD_USES_CHRONO
358+
template <class lock_type, class Clock, class Duration, class Predicate>
359+
bool
360+
wait_until(
361+
lock_type& lock,
362+
const chrono::time_point<Clock, Duration>& t,
363+
Predicate pred)
364+
{
365+
while (!pred())
366+
{
367+
if (wait_until(lock, t) == cv_status::timeout)
368+
return pred();
369+
}
370+
return true;
371+
}
372+
373+
template <class lock_type, class Rep, class Period, class Predicate>
374+
bool
375+
wait_for(
376+
lock_type& lock,
377+
const chrono::duration<Rep, Period>& d,
378+
Predicate pred)
379+
{
380+
return wait_until(lock, chrono::steady_clock::now() + d, boost::move(pred));
381+
}
331382
#endif
332383

333384
void notify_one() BOOST_NOEXCEPT

include/boost/thread/pthread/condition_variable_fwd.hpp

+26-29
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,26 @@
2828

2929
namespace boost
3030
{
31+
namespace detail {
32+
inline int monotonic_pthread_cond_init(pthread_cond_t& cond) {
33+
34+
#ifdef BOOST_THREAD_HAS_CONDATTR_SET_CLOCK_MONOTONIC
35+
pthread_condattr_t attr;
36+
int res = pthread_condattr_init(&attr);
37+
if (res)
38+
{
39+
return res;
40+
}
41+
pthread_condattr_setclock(&attr, CLOCK_MONOTONIC);
42+
res=pthread_cond_init(&cond,&attr);
43+
pthread_condattr_destroy(&attr);
44+
return res;
45+
#else
46+
return pthread_cond_init(&cond,NULL);
47+
#endif
48+
49+
}
50+
}
3151

3252
class condition_variable
3353
{
@@ -56,35 +76,19 @@ namespace boost
5676
condition_variable()
5777
{
5878
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
59-
int const res=pthread_mutex_init(&internal_mutex,NULL);
79+
int res=pthread_mutex_init(&internal_mutex,NULL);
6080
if(res)
6181
{
6282
boost::throw_exception(thread_resource_error(res, "boost::condition_variable::condition_variable() constructor failed in pthread_mutex_init"));
6383
}
6484
#endif
65-
66-
#ifdef BOOST_THREAD_HAS_CONDATTR_SET_CLOCK_MONOTONIC
67-
pthread_condattr_t attr;
68-
int res2 = pthread_condattr_init(&attr);
69-
if(res2)
70-
{
71-
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
72-
BOOST_VERIFY(!pthread_mutex_destroy(&internal_mutex));
73-
#endif
74-
boost::throw_exception(thread_resource_error(res2, "boost::condition_variable_steady::condition_variable_steady() constructor failed in pthread_condattr_init"));
75-
}
76-
pthread_condattr_setclock(&attr, CLOCK_MONOTONIC);
77-
res2=pthread_cond_init(&cond,&attr);
78-
pthread_condattr_destroy(&attr);
79-
#else
80-
int const res2=pthread_cond_init(&cond,NULL);
81-
#endif
82-
if(res2)
85+
res = detail::monotonic_pthread_cond_init(cond);
86+
if (res)
8387
{
8488
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
8589
BOOST_VERIFY(!pthread_mutex_destroy(&internal_mutex));
8690
#endif
87-
boost::throw_exception(thread_resource_error(res2, "boost::condition_variable::condition_variable() constructor failed in pthread_cond_init"));
91+
boost::throw_exception(thread_resource_error(res, "boost::condition_variable::condition_variable() constructor failed in detail::monotonic_pthread_cond_init"));
8892
}
8993
}
9094
~condition_variable()
@@ -249,7 +253,7 @@ namespace boost
249253
}
250254
#endif
251255

252-
#else
256+
#else // defined BOOST_THREAD_HAS_CONDATTR_SET_CLOCK_MONOTONIC
253257
#ifdef BOOST_THREAD_USES_CHRONO
254258

255259
template <class Duration>
@@ -304,7 +308,7 @@ namespace boost
304308
}
305309
#endif
306310

307-
#endif
311+
#endif // defined BOOST_THREAD_HAS_CONDATTR_SET_CLOCK_MONOTONIC
308312

309313
#ifdef BOOST_THREAD_USES_CHRONO
310314
template <class Clock, class Duration, class Predicate>
@@ -330,13 +334,6 @@ namespace boost
330334
Predicate pred)
331335
{
332336
return wait_until(lock, chrono::steady_clock::now() + d, boost::move(pred));
333-
334-
// while (!pred())
335-
// {
336-
// if (wait_for(lock, d) == cv_status::timeout)
337-
// return pred();
338-
// }
339-
// return true;
340337
}
341338
#endif
342339

0 commit comments

Comments
 (0)