cpp-pthread  (v1.7.3)
Simple C++ wrapper to pthread functions.
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Modules Pages
condition_variable.hpp
1 //
2 // Created by herbert koelman on 13/03/2016.
3 // Copyright © 2016 urbix-software. All rights reserved.
4 //
5 
6 #ifndef pthread_condition_variable_hpp
7 #define pthread_condition_variable_hpp
8 
9 // must be include as first hearder file of each source code file (see IBM's
10 // recommandation for more info p.285 �8.3.1).
11 #include <pthread.h>
12 #include <string>
13 #include <ctime>
14 #include <sys/time.h>
15 
16 #include "pthread/config.h"
17 
18 #include "pthread/exceptions.hpp"
19 #include "pthread/mutex.hpp"
20 #include "pthread/lock_guard.hpp"
21 
22 namespace pthread {
23 
33  enum cv_status {
36  };
37 
57  public:
58 
70  void wait ( mutex &mtx );
71 
74  void wait ( lock_guard<pthread::mutex> lck);
75 
92  template<class Lambda> bool wait( mutex &mtx, Lambda lambda);
93 
110  template<class Lambda> bool wait( lock_guard<pthread::mutex> &lck, Lambda lambda);
111 
129  cv_status wait_for (mutex &mtx, int millis );
130 
133  cv_status wait_for (lock_guard<pthread::mutex> &lck, int millis );
134 
152  template<class Lambda> bool wait_for( mutex &mtx, int millis, Lambda lambda);
153 
171  template<class Lambda> bool wait_for( lock_guard<pthread::mutex> &lck, int millis, Lambda lambda);
172 
177 #if __cplusplus < 201103L
178  void notify_one () throw() ;
179 #else
180  void notify_one () noexcept;
181 #endif
182 
187 #if __cplusplus < 201103L
188  void notify_all () throw() ;
189 #else
190  void notify_all () noexcept ;
191 #endif
192 
193  // constructor/destructor ------------------------------------------------
194 
196  virtual ~condition_variable();
197 
198  private:
199  void milliseconds( int milliseconds);
200 
201  timespec timeout;
202  pthread_cond_t _condition;
203  };
204 
207  // template implementation ----------------------
208 
209  template<class Lambda> bool condition_variable::wait( mutex &mtx, Lambda lambda){
210 
211  bool stop_waiting = lambda();
212 
213  while(!stop_waiting){
214  wait(mtx);
215  stop_waiting = lambda(); // handling spurious wakeups
216  }
217 
218  return stop_waiting;
219  };
220 
221  template<class Lambda> bool condition_variable::wait( lock_guard<pthread::mutex> &lck, Lambda lambda){
222 
223  //return wait( *(lck.mutex()), lambda);
224  return wait( *(lck._mutex), lambda);
225  };
226 
227  template<class Lambda> bool condition_variable::wait_for( mutex &mtx, int millis, Lambda lambda){
228  int rc = 0;
229  cv_status status = no_timeout;
230 
231  milliseconds(millis); // update timeou
232  bool stop_waiting = lambda(); // returns ​false if the waiting should be continued.
233 
234  while((! stop_waiting) && (status == no_timeout)){
235 
236  rc = pthread_cond_timedwait ( &_condition, &mtx._mutex, &timeout );
237 
238  switch (rc){
239 
240  case ETIMEDOUT:
241  status = timedout;
242  break;
243 
244  case EINVAL:
245  throw condition_variable_exception("The value specified by abstime is invalid.", rc); // NOSONAR we use throw instead of return.
246  break;
247 
248  case EPERM:
249  throw condition_variable_exception("The mutex was not owned by the current thread at the time of the call.", rc); // NOSONAR we use throw instead of return.
250  break;
251  default:
252  status = no_timeout ;
253  break;
254  }
255 
256  stop_waiting = lambda();
257  }
258 
259 
260  return stop_waiting ; //status == cv_status::no_timeout;
261  };
262 
263  template<class Lambda> bool condition_variable::wait_for( lock_guard<pthread::mutex> &lck, int millis, Lambda lambda){
264 
265  // return wait_for(*(lck.mutex()),millis, lambda);
266  return wait_for(*(lck._mutex),millis, lambda);
267  };
268 
269 
270 } // namespace pthread
271 
272 
273 #endif
cv_status wait_for(mutex &mtx, int millis)
pthread_mutex_t _mutex
Definition: mutex.hpp:74