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
qfutex_linux_p.h
Go to the documentation of this file.
1// Copyright (C) 2023 Intel Corporation.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
4#ifndef QFUTEX_LINUX_P_H
5#define QFUTEX_LINUX_P_H
6
7//
8// W A R N I N G
9// -------------
10//
11// This file is not part of the Qt API. It exists purely as an
12// implementation detail. This header file may change from version to
13// version without notice, or even be removed.
14//
15// We mean it.
16//
17
18#include <private/qcore_unix_p.h>
19#include <qdeadlinetimer.h>
20#include <qtsan_impl.h>
21
22#include <asm/unistd.h>
23#include <errno.h>
24#include <limits.h>
25#include <linux/futex.h>
26#include <sys/syscall.h>
27#include <unistd.h>
28
29// RISC-V does not supply __NR_futex
30#ifndef __NR_futex
31# define __NR_futex __NR_futex_time64
32#endif
33
34#define QT_ALWAYS_USE_FUTEX
35
37
38namespace QtLinuxFutex {
39constexpr inline bool futexAvailable() { return true; }
40
41inline int _q_futex(int *addr, int op, int val, quintptr val2 = 0,
42 int *addr2 = nullptr, int val3 = 0) noexcept
43{
45
46 // we use __NR_futex because some libcs (like Android's bionic) don't
47 // provide SYS_futex etc.
48 int result = syscall(__NR_futex, addr, op | FUTEX_PRIVATE_FLAG, val, val2, addr2, val3);
49
51
52 return result;
53}
54template <typename T> int *addr(T *ptr)
55{
56 int *int_addr = reinterpret_cast<int *>(ptr);
57#if Q_BYTE_ORDER == Q_BIG_ENDIAN
58 if (sizeof(T) > sizeof(int))
59 int_addr++; //We want a pointer to the least significant half
60#endif
61 return int_addr;
62}
63
64template <typename Atomic>
65inline void futexWait(Atomic &futex, typename Atomic::Type expectedValue)
66{
67 _q_futex(addr(&futex), FUTEX_WAIT, qintptr(expectedValue));
68}
69template <typename Atomic>
70inline bool futexWait(Atomic &futex, typename Atomic::Type expectedValue, QDeadlineTimer deadline)
71{
72 auto timeout = deadline.deadline<std::chrono::steady_clock>().time_since_epoch();
73 struct timespec ts = durationToTimespec(timeout);
74 int r = _q_futex(addr(&futex), FUTEX_WAIT_BITSET, qintptr(expectedValue), quintptr(&ts),
75 nullptr, FUTEX_BITSET_MATCH_ANY);
76 return r == 0 || errno != ETIMEDOUT;
77}
78template <typename Atomic> inline void futexWakeOne(Atomic &futex)
79{
80 _q_futex(addr(&futex), FUTEX_WAKE, 1);
81}
82template <typename Atomic> inline void futexWakeAll(Atomic &futex)
83{
84 _q_futex(addr(&futex), FUTEX_WAKE, INT_MAX);
85}
86template <typename Atomic> inline
87void futexWakeOp(Atomic &futex1, int wake1, int wake2, Atomic &futex2, quint32 op)
88{
89 _q_futex(addr(&futex1), FUTEX_WAKE_OP, wake1, wake2, addr(&futex2), op);
90}} // namespace QtLinuxFutex
91namespace QtFutex = QtLinuxFutex;
92
94
95#endif // QFUTEX_LINUX_P_H
\inmodule QtCore
qint64 deadline() const noexcept Q_DECL_PURE_FUNCTION
Returns the absolute time point for the deadline stored in QDeadlineTimer object, calculated in milli...
Combined button and popup list for selecting options.
void futexWakeOp(Atomic &futex1, int wake1, int wake2, Atomic &futex2, quint32 op)
void futexWait(Atomic &futex, typename Atomic::Type expectedValue)
void futexWakeAll(Atomic &futex)
constexpr bool futexAvailable()
int _q_futex(int *addr, int op, int val, quintptr val2=0, int *addr2=nullptr, int val3=0) noexcept
void futexWakeOne(Atomic &futex)
void futexRelease(void *, void *=nullptr)
Definition qtsan_impl.h:63
void futexAcquire(void *, void *=nullptr)
Definition qtsan_impl.h:62
timespec durationToTimespec(std::chrono::nanoseconds timeout) noexcept
#define __NR_futex
static ControlElement< T > * ptr(QWidget *widget)
GLboolean r
[2]
GLbitfield GLuint64 timeout
[4]
GLuint GLfloat * val
GLenum const void * addr
GLuint64EXT * result
[6]
unsigned int quint32
Definition qtypes.h:50
size_t quintptr
Definition qtypes.h:167
ptrdiff_t qintptr
Definition qtypes.h:166
QDeadlineTimer deadline(30s)