File: Condition.cpp

package info (click to toggle)
subcommander 2.0.0~b5p1-2
  • links: PTS
  • area: main
  • in suites: squeeze
  • size: 14,880 kB
  • ctags: 9,003
  • sloc: cpp: 63,591; sh: 3,904; xml: 1,984; makefile: 1,166; ansic: 786; ruby: 251; lisp: 24
file content (110 lines) | stat: -rw-r--r-- 2,091 bytes parent folder | download | duplicates (5)
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