Fix for UBSan build
[platform/upstream/doxygen.git] / qtools / qwaitcondition_unix.cpp
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
54 class QWaitConditionPrivate
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
92 QWaitCondition::QWaitCondition()
93 {
94     d = new QWaitConditionPrivate;
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
101 QWaitCondition::~QWaitCondition()
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
108 void QWaitCondition::wakeOne()
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
116 void QWaitCondition::wakeAll()
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
124 void QWaitCondition::wait(QMutex *mutex)
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