/* ====================================================================
 * 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


syntax highlighted by Code2HTML, v. 0.9.1