qwaitcondition_unix.cpp
Go to the documentation of this file.
1 /****************************************************************************
2 **
3 ** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
4 ** Contact: Nokia Corporation (qt-info@nokia.com)
5 **
6 ** This file is part of the QtCore module of the Qt Toolkit.
7 **
8 ** $QT_BEGIN_LICENSE:LGPL$
9 ** No Commercial Usage
10 ** This file contains pre-release code and may not be distributed.
11 ** You may use this file in accordance with the terms and conditions
12 ** contained in the either Technology Preview License Agreement or the
13 ** Beta Release License Agreement.
14 **
15 ** GNU Lesser General Public License Usage
16 ** Alternatively, this file may be used under the terms of the GNU Lesser
17 ** General Public License version 2.1 as published by the Free Software
18 ** Foundation and appearing in the file LICENSE.LGPL included in the
19 ** packaging of this file. Please review the following information to
20 ** ensure the GNU Lesser General Public License version 2.1 requirements
21 ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
22 **
23 ** In addition, as a special exception, Nokia gives you certain
24 ** additional rights. These rights are described in the Nokia Qt LGPL
25 ** Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this
26 ** package.
27 **
28 ** GNU General Public License Usage
29 ** Alternatively, this file may be used under the terms of the GNU
30 ** General Public License version 3.0 as published by the Free Software
31 ** Foundation and appearing in the file LICENSE.GPL included in the
32 ** packaging of this file. Please review the following information to
33 ** ensure the GNU General Public License version 3.0 requirements will be
34 ** met: http://www.gnu.org/copyleft/gpl.html.
35 **
36 ** If you are unsure which license is appropriate for your use, please
37 ** contact the sales department at http://www.qtsoftware.com/contact.
38 ** $QT_END_LICENSE$
39 **
40 ****************************************************************************/
41 
42 #include "qwaitcondition.h"
43 #include "qmutex.h"
44 #include <pthread.h>
45 
46 #define MIN(a,b) ((a)<(b)?(a):(b))
47 
48 static void report_error(int code, const char *where, const char *what)
49 {
50  if (code != 0)
51  qWarning("%s: %s failure: %d", where, what, code);
52 }
53 
55 {
56  public:
57  pthread_mutex_t mutex;
58  pthread_cond_t cond;
59  int waiters;
60  int wakeups;
61 
62  void wait()
63  {
64  int code;
65  for (;;)
66  {
67  code = pthread_cond_wait(&cond, &mutex);
68  if (code == 0 && wakeups == 0)
69  {
70  // many vendors warn of spurios wakeups from
71  // pthread_cond_wait(), especially after signal delivery,
72  // even though POSIX doesn't allow for it... sigh
73  continue;
74  }
75  break;
76  }
77 
78  --waiters;
79  if (code == 0)
80  {
81  --wakeups;
82  }
83  else
84  {
85  report_error(code, "QWaitCondition::wait()", "cv wait");
86  }
87  report_error(pthread_mutex_unlock(&mutex), "QWaitCondition::wait()", "mutex unlock");
88  }
89 };
90 
91 
93 {
95  report_error(pthread_mutex_init(&d->mutex, NULL), "QWaitCondition", "mutex init");
96  report_error(pthread_cond_init(&d->cond, NULL), "QWaitCondition", "cv init");
97  d->waiters = d->wakeups = 0;
98 }
99 
100 
102 {
103  report_error(pthread_cond_destroy(&d->cond), "QWaitCondition", "cv destroy");
104  report_error(pthread_mutex_destroy(&d->mutex), "QWaitCondition", "mutex destroy");
105  delete d;
106 }
107 
109 {
110  report_error(pthread_mutex_lock(&d->mutex), "QWaitCondition::wakeOne()", "mutex lock");
111  d->wakeups = MIN(d->wakeups + 1, d->waiters);
112  report_error(pthread_cond_signal(&d->cond), "QWaitCondition::wakeOne()", "cv signal");
113  report_error(pthread_mutex_unlock(&d->mutex), "QWaitCondition::wakeOne()", "mutex unlock");
114 }
115 
117 {
118  report_error(pthread_mutex_lock(&d->mutex), "QWaitCondition::wakeAll()", "mutex lock");
119  d->wakeups = d->waiters;
120  report_error(pthread_cond_broadcast(&d->cond), "QWaitCondition::wakeAll()", "cv broadcast");
121  report_error(pthread_mutex_unlock(&d->mutex), "QWaitCondition::wakeAll()", "mutex unlock");
122 }
123 
125 {
126  if (!mutex) return;
127 
128  report_error(pthread_mutex_lock(&d->mutex), "QWaitCondition::wait()", "mutex lock");
129  ++d->waiters;
130  mutex->unlock();
131  d->wait();
132  mutex->lock();
133 }
134 
void lock()
Definition: qmutex.cpp:58
Definition: qmutex.h:49
void qWarning(const char *msg,...)
Definition: qglobal.cpp:409
#define MIN(a, b)
void unlock()
Definition: qmutex.cpp:85
CodeOutputInterface * code
static void report_error(int code, const char *where, const char *what)
void wait(QMutex *mutex)