futex: Implement mechanism to wait on any of several futexes
This is a new futex operation, called FUTEX_WAIT_MULTIPLE, which allows a thread to wait on several futexes at the same time, and be awoken by any of them. In a sense, it implements one of the features that was supported by pooling on the old FUTEX_FD interface. The use case lies in the Wine implementation of the Windows NT interface WaitMultipleObjects. This Windows API function allows a thread to sleep waiting on the first of a set of event sources (mutexes, timers, signal, console input, etc) to signal. Considering this is a primitive synchronization operation for Windows applications, being able to quickly signal events on the producer side, and quickly go to sleep on the consumer side is essential for good performance of those running over Wine. Wine developers have an implementation that uses eventfd, but it suffers from FD exhaustion (there is applications that go to the order of multi-milion FDs), and higher CPU utilization than this new operation. The futex list is passed as an array of `struct futex_wait_block` (pointer, value, bitset) to the kernel, which will enqueue all of them and sleep if none was already triggered. It returns a hint of which futex caused the wake up event to userspace, but the hint doesn't guarantee that is the only futex triggered. Before calling the syscall again, userspace should traverse the list, trying to re-acquire any of the other futexes, to prevent an immediate -EWOULDBLOCK return code from the kernel. This was tested using three mechanisms: 1) By reimplementing FUTEX_WAIT in terms of FUTEX_WAIT_MULTIPLE and running the unmodified tools/testing/selftests/futex and a full linux distro on top of this kernel. 2) By an example code that exercises the FUTEX_WAIT_MULTIPLE path on a multi-threaded, event-handling setup. 3) By running the Wine fsync implementation and executing multi-threaded applications, in particular modern games, on top of this implementation. Changes were tested for the following ABIs: x86_64, i386 and x32. Support for x32 applications is not implemented since it would take a major rework adding a new entry point and splitting the current futex 64 entry point in two and we can't change the current x32 syscall number without breaking user space compatibility. CC: Steven Rostedt <rostedt@goodmis.org> Cc: Richard Yao <ryao@gentoo.org> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: Peter Zijlstra <peterz@infradead.org> Co-developed-by:Zebediah Figura <z.figura12@gmail.com> Signed-off-by:
Zebediah Figura <z.figura12@gmail.com> Co-developed-by:
Steven Noonan <steven@valvesoftware.com> Signed-off-by:
Steven Noonan <steven@valvesoftware.com> Co-developed-by:
Pierre-Loup A. Griffais <pgriffais@valvesoftware.com> Signed-off-by:
Pierre-Loup A. Griffais <pgriffais@valvesoftware.com> Signed-off-by:
Gabriel Krisman Bertazi <krisman@collabora.com> [Added compatibility code] Co-developed-by:
André Almeida <andrealmeid@collabora.com> Signed-off-by:
André Almeida <andrealmeid@collabora.com> --- Changes since v2: - Loop counters are now unsigned - Add ifdef around `in_x32_syscall()`, so this function is only compiled in architectures that declare it Changes since RFC: - Limit waitlist to 128 futexes - Simplify wait loop - Document functions - Reduce allocated space - Return hint if a futex was awoken during setup - Check if any futex was awoken prior to sleep - Drop relative timer logic - Add compatibility struct and entry points - Add selftests
Loading
Please register or sign in to comment