pandorafms/extras/anytermd/libpbe/include/Condition.hh

268 lines
7.5 KiB
C++

// include/Condition.hh
// This file is part of libpbe; see http://svn.chezphil.org/libpbe/
// (C) 2008 Philip Endecott
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 2 of the License, or
// any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#ifndef libpbe_Condition_hh
#define libpbe_Condition_hh
// This file provides a class Condition, which implements a condition variable
// compatible with N2447's condition_variable[_any?].
// If we have Futex - i.e. if this is Linux - then we implement Condition directly
// on top of Futex. Otherwise we implement it using pthread_condition.
#include "Futex.hh"
#include "atomic.hh"
#include "HPtime.hh"
#include <boost/noncopyable.hpp>
#include <climits>
#include <time.h>
#if defined(PBE_HAVE_FUTEX) && defined(PBE_HAVE_ATOMICS)
#include "Unlock.hh"
namespace pbe {
class Condition: boost::noncopyable {
uint32_t waiters;
uint32_t eventcount;
Futex<uint32_t> fut;
public:
Condition():
waiters(0), // Keep track of the number of waiting threads so that
// the notify operations can be no-ops in the case where
// there are no waiters.
eventcount(0), // Incremented on each notify_all. This is needed to
// detect any notify_all that occurs between unlocking
// the mutex and waiting on the futex.
fut(eventcount) {}
~Condition() {}
// Note that ~Condition does not wait for any waiters to stop waiting;
// undefined behaviour results if a Condition is destructed while a thread
// is still waiting on it.
void notify_one() {
if (atomic_read(waiters)==0) {
return;
}
unsigned int n_woken = fut.wake(1);
if (n_woken==0) {
notify_all(); // WHY???
return;
}
atomic_dec(waiters,n_woken);
}
void notify_all() {
if (atomic_read(waiters)==0) {
return;
}
atomic_inc(eventcount);
unsigned int n_woken = fut.wake(INT_MAX);
atomic_dec(waiters,n_woken);
}
template <typename LOCK_T>
void wait(LOCK_T& lock) {
atomic_inc(waiters);
uint32_t initial_eventcount = atomic_read(eventcount);
Unlock<typename LOCK_T::mutex_t> ul(*lock.mutex()); // Must not throw.
// If this could throw, we would need to --waiters.
do {
// fut.wait can return spuriously e.g. if a signal is handled.
// We retry in this case, so Condition.wait does not return spuriously.
fut.wait(initial_eventcount);
} while (atomic_read(eventcount)==initial_eventcount); // NO broken for notify_one
}
template <typename LOCK_T, typename Predicate>
void wait(LOCK_T& lock, Predicate pred) {
// Note that the following "trivial" implementation is specified by N2447;
// if notify_one() is called, the thread that is woken may test its condition
// and sleep again. There's no guarantee that the notification is delivered
// to a thread whose condition will evaluate true. So notify_one() is not
// useful in combination with predicated wait, unless perhaps all waiters
// have the same predicate.
while (!pred()) {
wait(lock);
}
}
template <typename LOCK_T>
bool timed_wait(LOCK_T& lock, HPtime abs_time) {
atomic_inc(waiters);
uint32_t initial_eventcount = atomic_read(eventcount);
Unlock<typename LOCK_T::mutex_t> ul(*lock.mutex()); // Must not throw.
// If this could throw, we would need to --waiters.
do {
// fut.wait can return spuriously e.g. if a signal is handled.
// We retry in this case, so Condition.wait does not return spuriously.
HPtime now = HPtime::now();
if (now>abs_time || !fut.timed_wait(initial_eventcount, abs_time-now)) {
return false;
}
} while (atomic_read(eventcount)==initial_eventcount); // NO broken for notify_one
return true;
}
template <typename LOCK_T>
bool timed_wait(LOCK_T& lock, float rel_time) {
HPtime abs_time = HPtime::now() + rel_time;
return timed_wait(lock,abs_time);
}
template <typename Lock, typename Predicate>
bool timed_wait(Lock& lock, ::time_t abs_time, Predicate pred) {
// See above. If the predicate becomes true and the timeout expires at
// the same time, we're supposed to return true indicating that the predicate
// was true.
while (!pred()) {
if (!timed_wait(lock,abs_time)) {
return pred();
}
}
return true;
}
// native_handle not provided
};
};
#else
#include <pthread.h>
#include "Exception.hh"
namespace pbe {
class Condition: boost::noncopyable {
pthread_cond_t cond;
public:
Condition() {
int r = pthread_cond_init(&cond, NULL);
if (r!=0) {
pbe::throw_ErrnoException("pthread_cond_init()",r);
}
}
~Condition() {
pthread_cond_destroy(&cond);
}
void notify_one() {
int r = pthread_cond_signal(&cond);
if (r!=0) {
pbe::throw_ErrnoException("pthread_cond_signal()",r);
}
}
void notify_all() {
int r = pthread_cond_broadcast(&cond);
if (r!=0) {
pbe::throw_ErrnoException("pthread_cond_broadcast()",r);
}
}
template <typename LOCK_T>
void wait(LOCK_T& lock) {
int r = pthread_cond_wait(&cond, lock.mutex()->native_handle());
if (r!=0) {
pbe::throw_ErrnoException("pthread_cond_wait()",r);
}
}
template <typename LOCK_T, typename Predicate>
void wait(LOCK_T& lock, Predicate pred) {
// Note that the following "trivial" implementation is specified by N2447;
// if notify_one() is called, the thread that is woken may test its condition
// and sleep again. There's no guarantee that the notification is delivered
// to a thread whose condition will evaluate true. So notify_one() is not
// useful in combination with predicated wait, unless perhaps all waiters
// have the same predicate.
while (!pred()) {
wait(lock);
}
}
template <typename LOCK_T>
bool timed_wait(LOCK_T& lock, ::time_t abs_time) {
timespec ts;
ts.tv_sec = abs_time;
ts.tv_nsec = 0;
int r = pthread_cond_timedwait(&cond, lock.mutex()->native_handle(), &ts);
switch (r) {
case 0: return true;
case ETIMEDOUT: return false;
default: pbe::throw_ErrnoException("pthread_cond_timedwait()",r);
}
// NOT REACHED
return true;
}
template <typename LOCK_T>
bool timed_wait(LOCK_T& lock, float rel_time) {
// This limits resolution to whole seconds.
return timed_wait(lock, static_cast<time_t>(time(NULL)+rel_time));
}
template <typename Lock, typename Predicate>
bool timed_wait(Lock& lock, ::time_t abs_time, Predicate pred) {
// See above. If the predicate becomes true and the timeout expires at
// the same time, we're supposed to return true indicating that the predicate
// was true.
while (!pred()) {
if (!timed_wait(lock,abs_time)) {
return pred();
}
}
return true;
}
pthread_cond_t* native_handle() {
return &cond;
}
};
};
#endif
#endif