1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
|
/* ====================================================================
* Copyright (c) 2003-2006, The Subcommander Crew
* http://subcommander.tigris.org
*
* Subcommander is licensed as described in the file doc/COPYING, which
* you should have received as part of this distribution.
* ====================================================================
*/
// sc
#include "Condition.h"
#include "Guard.h"
// apr
#include <apr_thread_cond.h>
namespace sc
{
/**
* Create a condition variable.
*/
Condition::Condition() : _c(false)
{
apr_thread_cond_create( &_condition, _pool );
}
/**
* Clean up.
*/
Condition::~Condition()
{
apr_thread_cond_destroy(_condition);
}
/**
* Wake up a single waiting thread.
* \sa wakeAll()
*/
void Condition::wakeOne()
{
Guard<Mutex> guard(_mutex);
_c = true;
apr_thread_cond_signal(_condition);
}
/**
* Wake up all waiting threads.
* \sa wait()
*/
void Condition::wakeAll()
{
Guard<Mutex> guard(_mutex);
_c = true;
apr_thread_cond_broadcast(_condition);
}
/**
* Wait for the condition. This call blocks until the condition is set.
*/
void Condition::wait()
{
Guard<Mutex> guard(_mutex);
// the loop handles 2 special cases:
// 1. don't block the calling thread if the condition is already set.
// 2. only return if the condition is set. The condition may be reset
// before the waking thread reaquires the mutex. (is that correct?)
while( ! _c )
{
apr_thread_cond_wait( _condition, _mutex._mutex );
}
}
/**
* Wait for the condition. This call blocks until the condition is set
* (returns true) or a time span of microsec has elapsed (returns false).
*/
bool Condition::wait( unsigned long millies )
{
Guard<Mutex> guard(_mutex);
while( ! _c )
{
apr_status_t status =
apr_thread_cond_timedwait( _condition, _mutex._mutex, millies*1000 );
if( status == APR_TIMEUP )
{
return false;
}
}
return true;
}
/**
* Reset the condition.
*/
void Condition::reset()
{
Guard<Mutex> guard(_mutex);
_c = false;
}
} // namespace
|