4#ifndef QFUTEX_LINUX_P_H
5#define QFUTEX_LINUX_P_H
18#include <private/qcore_unix_p.h>
19#include <qdeadlinetimer.h>
20#include <qtsan_impl.h>
22#include <asm/unistd.h>
25#include <linux/futex.h>
26#include <sys/syscall.h>
31# define __NR_futex __NR_futex_time64
34#define QT_ALWAYS_USE_FUTEX
41inline long _q_futex(
int *addr,
int op,
int val, quintptr val2 = 0,
42 int *addr2 =
nullptr,
int val3 = 0)
noexcept
48 long result = syscall(
__NR_futex, addr, op | FUTEX_PRIVATE_FLAG, val, val2, addr2, val3);
54template <
typename T>
int *
addr(T *ptr)
56 int *int_addr =
reinterpret_cast<
int *>(ptr);
57#if Q_BYTE_ORDER == Q_BIG_ENDIAN
58 if (
sizeof(T) >
sizeof(
int))
64template <
typename Atomic>
65inline void futexWait(Atomic &futex,
typename Atomic::Type expectedValue)
67 _q_futex(addr(&futex), FUTEX_WAIT, qintptr(expectedValue));
69template <
typename Atomic>
70inline bool futexWait(Atomic &futex,
typename Atomic::Type expectedValue, QDeadlineTimer deadline)
72 auto timeout = deadline.deadline<
std::chrono::steady_clock>().time_since_epoch();
73 struct timespec ts = durationToTimespec(timeout);
74 long r = _q_futex(addr(&futex), FUTEX_WAIT_BITSET, qintptr(expectedValue), quintptr(&ts),
75 nullptr, FUTEX_BITSET_MATCH_ANY);
76 return r == 0 || errno != ETIMEDOUT;
80 _q_futex(addr(&futex), FUTEX_WAKE, 1);
84 _q_futex(addr(&futex), FUTEX_WAKE, INT_MAX);
86template <
typename Atomic>
inline
87void futexWakeOp(Atomic &futex1,
int wake1,
int wake2, Atomic &futex2, quint32 op)
89 _q_futex(addr(&futex1), FUTEX_WAKE_OP, wake1, wake2, addr(&futex2), op);
void futexWakeOp(Atomic &futex1, int wake1, int wake2, Atomic &futex2, quint32 op)
bool futexWait(Atomic &futex, typename Atomic::Type expectedValue, QDeadlineTimer deadline)
void futexWait(Atomic &futex, typename Atomic::Type expectedValue)
void futexWakeAll(Atomic &futex)
constexpr bool futexAvailable()
long _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)
void futexAcquire(void *, void *=nullptr)