Qt
Internal/Contributor docs for the Qt SDK. <b>Note:</b> These are NOT official API docs; those are found <a href='https://doc.qt.io/'>here</a>.
Loading...
Searching...
No Matches
qsystemsemaphore_posix.cpp
Go to the documentation of this file.
1// Copyright (C) 2015 Konstantin Ritt <ritt.ks@gmail.com>
2// Copyright (C) 2016 The Qt Company Ltd.
3// Copyright (C) 2015 Klarälvdalens Datakonsult AB, a KDAB Group company, info@kdab.com, author Tobias Koenig <tobias.koenig@kdab.com>
4// Copyright (C) 2022 Intel Corporation.
5// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
6
7#include "qsystemsemaphore.h"
9
10#include <qdebug.h>
11#include <qfile.h>
12#include <qcoreapplication.h>
13
14#if QT_CONFIG(posix_sem)
15
16#include <sys/types.h>
17#include <fcntl.h>
18#include <errno.h>
19
20#ifdef Q_OS_UNIX
21# include "private/qcore_unix_p.h"
22#else
23# define QT_EINTR_LOOP_VAL(var, val, cmd) \
24 (void)var; var = cmd
25# define QT_EINTR_LOOP(var, cmd) QT_EINTR_LOOP_VAL(var, -1, cmd)
26#endif
27
28// OpenBSD 4.2 doesn't define EIDRM, see BUGS section:
29// http://www.openbsd.org/cgi-bin/man.cgi?query=semop&manpath=OpenBSD+4.2
30#if defined(Q_OS_OPENBSD) && !defined(EIDRM)
31#define EIDRM EINVAL
32#endif
33
35
36using namespace Qt::StringLiterals;
37
38bool QSystemSemaphorePosix::runtimeSupportCheck()
39{
40 static const bool result = []() {
41 sem_open("/", 0, 0, 0); // this WILL fail
42 return errno != ENOSYS;
43 }();
44 return result;
45}
46
47bool QSystemSemaphorePosix::handle(QSystemSemaphorePrivate *self, QSystemSemaphore::AccessMode mode)
48{
49 if (semaphore != SEM_FAILED)
50 return true; // we already have a semaphore
51
52 const QByteArray semName = QFile::encodeName(self->nativeKey.nativeKey());
53 if (semName.isEmpty()) {
54 self->setError(QSystemSemaphore::KeyError,
55 QSystemSemaphore::tr("%1: key is empty")
56 .arg("QSystemSemaphore::handle"_L1));
57 return false;
58 }
59
60 // Always try with O_EXCL so we know whether we created the semaphore.
61 int oflag = O_CREAT | O_EXCL;
62 for (int tryNum = 0, maxTries = 1; tryNum < maxTries; ++tryNum) {
63 do {
64 semaphore = ::sem_open(semName.constData(), oflag, 0600, self->initialValue);
65 } while (semaphore == SEM_FAILED && errno == EINTR);
66 if (semaphore == SEM_FAILED && errno == EEXIST) {
67 if (mode == QSystemSemaphore::Create) {
68 if (::sem_unlink(semName.constData()) == -1 && errno != ENOENT) {
69 self->setUnixErrorString("QSystemSemaphore::handle (sem_unlink)"_L1);
70 return false;
71 }
72 // Race condition: the semaphore might be recreated before
73 // we call sem_open again, so we'll retry several times.
74 maxTries = 3;
75 } else {
76 // Race condition: if it no longer exists at the next sem_open
77 // call, we won't realize we created it, so we'll leak it later.
78 oflag &= ~O_EXCL;
79 maxTries = 2;
80 }
81 } else {
82 break;
83 }
84 }
85
86 if (semaphore == SEM_FAILED) {
87 self->setUnixErrorString("QSystemSemaphore::handle"_L1);
88 return false;
89 }
90
91 createdSemaphore = (oflag & O_EXCL) != 0;
92
93 return true;
94}
95
96void QSystemSemaphorePosix::cleanHandle(QSystemSemaphorePrivate *self)
97{
98 if (semaphore != SEM_FAILED) {
99 if (::sem_close(semaphore) == -1) {
100 self->setUnixErrorString("QSystemSemaphore::cleanHandle (sem_close)"_L1);
101#if defined QSYSTEMSEMAPHORE_DEBUG
102 qDebug("QSystemSemaphore::cleanHandle sem_close failed.");
103#endif
104 }
105 semaphore = SEM_FAILED;
106 }
107
108 if (createdSemaphore) {
109 const QByteArray semName = QFile::encodeName(self->nativeKey.nativeKey());
110 if (::sem_unlink(semName) == -1 && errno != ENOENT) {
111 self->setUnixErrorString("QSystemSemaphore::cleanHandle (sem_unlink)"_L1);
112#if defined QSYSTEMSEMAPHORE_DEBUG
113 qDebug("QSystemSemaphorePosix::cleanHandle sem_unlink failed.");
114#endif
115 }
116 createdSemaphore = false;
117 }
118}
119
120bool QSystemSemaphorePosix::modifySemaphore(QSystemSemaphorePrivate *self, int count)
121{
122 if (!handle(self, QSystemSemaphore::Open))
123 return false;
124
125 if (count > 0) {
126 int cnt = count;
127 do {
128 if (::sem_post(semaphore) == -1) {
129#if defined(Q_OS_VXWORKS)
130 if (errno == EINVAL) {
131 semaphore = SEM_FAILED;
132 return modifySemaphore(self, cnt);
133 }
134#endif
135 self->setUnixErrorString("QSystemSemaphore::modifySemaphore (sem_post)"_L1);
136#if defined QSYSTEMSEMAPHORE_DEBUG
137 qDebug("QSystemSemaphorePosix::modify sem_post failed %d %d", count, errno);
138#endif
139 // rollback changes to preserve the SysV semaphore behavior
140 for ( ; cnt < count; ++cnt) {
141 int res;
142 QT_EINTR_LOOP(res, ::sem_wait(semaphore));
143 }
144 return false;
145 }
146 --cnt;
147 } while (cnt > 0);
148 } else {
149 int res;
150 QT_EINTR_LOOP(res, ::sem_wait(semaphore));
151 if (res == -1) {
152 // If the semaphore was removed be nice and create it and then modifySemaphore again
153 if (errno == EINVAL || errno == EIDRM) {
154 semaphore = SEM_FAILED;
155 return modifySemaphore(self, count);
156 }
157 self->setUnixErrorString("QSystemSemaphore::modifySemaphore (sem_wait)"_L1);
158#if defined QSYSTEMSEMAPHORE_DEBUG
159 qDebug("QSystemSemaphorePosix::modify sem_wait failed %d %d", count, errno);
160#endif
161 return false;
162 }
163 }
164
165 self->clearError();
166 return true;
167}
168
170
171#endif // QT_CONFIG(posix_sem)
\inmodule QtCore
Definition qbytearray.h:57
static QByteArray encodeName(const QString &fileName)
Converts fileName to an 8-bit encoding that you can use in native APIs.
Definition qfile.h:158
Combined button and popup list for selecting options.
QString self
Definition language.cpp:58
#define QT_EINTR_LOOP(var, cmd)
#define qDebug
[1]
Definition qlogging.h:164
GLuint64 GLenum void * handle
GLenum mode
GLenum GLenum GLsizei count
GLuint res
GLuint64EXT * result
[6]
SSL_CTX int void * arg