5#ifndef QFUTEX_LINUX_P_H
6#define QFUTEX_LINUX_P_H
19#include <private/qcore_unix_p.h>
20#include <qdeadlinetimer.h>
21#include <qtsan_impl.h>
23#include <asm/unistd.h>
26#include <linux/futex.h>
27#include <sys/syscall.h>
32# define __NR_futex __NR_futex_time64
35#define QT_ALWAYS_USE_FUTEX
42inline long _q_futex(
int *addr,
int op,
int val, quintptr val2 = 0,
43 int *addr2 =
nullptr,
int val3 = 0)
noexcept
49 long result = syscall(
__NR_futex, addr, op | FUTEX_PRIVATE_FLAG, val, val2, addr2, val3);
55template <
typename T>
int *
addr(T *ptr)
57 int *int_addr =
reinterpret_cast<
int *>(ptr);
58#if Q_BYTE_ORDER == Q_BIG_ENDIAN
59 if (
sizeof(T) >
sizeof(
int))
65template <
typename Atomic>
66inline void futexWait(Atomic &futex,
typename Atomic::Type expectedValue)
68 _q_futex(addr(&futex), FUTEX_WAIT, qintptr(expectedValue));
70template <
typename Atomic>
71inline bool futexWait(Atomic &futex,
typename Atomic::Type expectedValue, QDeadlineTimer deadline)
73 auto timeout = deadline.deadline<
std::chrono::steady_clock>().time_since_epoch();
74 struct timespec ts = durationToTimespec(timeout);
75 long r = _q_futex(addr(&futex), FUTEX_WAIT_BITSET, qintptr(expectedValue), quintptr(&ts),
76 nullptr, FUTEX_BITSET_MATCH_ANY);
77 return r == 0 || errno != ETIMEDOUT;
81 _q_futex(addr(&futex), FUTEX_WAKE, 1);
85 _q_futex(addr(&futex), FUTEX_WAKE, INT_MAX);
87template <
typename Atomic>
inline
88void futexWakeOp(Atomic &futex1,
int wake1,
int wake2, Atomic &futex2, quint32 op)
90 _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)