Skip to content

Commit

Permalink
uapi: futex: Add a futex syscall
Browse files Browse the repository at this point in the history
This commit adds two futex syscall wrappers that are exposed to
userspace.

Neither the kernel or glibc currently expose a futex wrapper, so
userspace is left performing raw syscalls. This has mostly been because
the overloading of one of the arguments makes it impossible to provide a
single type safe function.

Until recently the single syscall has worked fine. With the introduction
of a 64-bit time_t futex call on 32-bit architectures, this has become
more complex. The logic of handling the two possible futex syscalls is
complex and often implemented incorrectly.

This patch adds two futux syscall functions that correctly handle the
time_t complexity for userspace.

This idea is based on previous discussions:
https://lore.kernel.org/lkml/CAK8P3a3x_EyCiPDpMK54y=Rtm-Wb08ym2TNiuAZgXhYrThcWTw@mail.gmail.com/

Signed-off-by: Alistair Francis <alistair.francis@wdc.com>
  • Loading branch information
alistair23 authored and intel-lab-lkp committed Nov 26, 2021
1 parent f5b41b6 commit e6bfd30
Showing 1 changed file with 89 additions and 0 deletions.
89 changes: 89 additions & 0 deletions include/uapi/linux/futex_syscall.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
/* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */
/*
* Futex syscall helper functions
*
* Copyright (C) 2021 Western Digital. All Rights Reserved.
*
* Author: Alistair Francis <alistair.francis@wdc.com>
*/
#ifndef _UAPI_LINUX_FUTEX_SYSCALL_H
#define _UAPI_LINUX_FUTEX_SYSCALL_H

#include <asm/unistd.h>
#include <errno.h>
#include <linux/types.h>
#include <linux/time_types.h>
#include <stdint.h>
#include <sys/syscall.h>

/**
* futex_syscall_timeout() - __NR_futex/__NR_futex_time64 syscall wrapper
* @uaddr: address of first futex
* @op: futex op code
* @val: typically expected value of uaddr, but varies by op
* @timeout: an absolute struct timespec
* @uaddr2: address of second futex for some ops
* @val3: varies by op
*/
static inline int
__kernel_futex_syscall_timeout(volatile uint32_t *uaddr, int op, uint32_t val,
struct timespec *timeout, volatile uint32_t *uaddr2, int val3)
{
#if defined(__NR_futex_time64)
if (sizeof(*timeout) != sizeof(struct __kernel_old_timespec)) {
int ret = syscall(__NR_futex_time64, uaddr, op, val, timeout, uaddr2, val3);

if (ret == 0 || errno != ENOSYS)
return ret;
}
#endif

#if defined(__NR_futex)
if (sizeof(*timeout) == sizeof(struct __kernel_old_timespec))
return syscall(__NR_futex, uaddr, op, val, timeout, uaddr2, val3);

if (timeout && timeout->tv_sec == (long)timeout->tv_sec) {
struct __kernel_old_timespec ts_old;

ts_old.tv_sec = (__kernel_long_t) timeout->tv_sec;
ts_old.tv_nsec = (__kernel_long_t) timeout->tv_nsec;

return syscall(__NR_futex, uaddr, op, val, &ts_old, uaddr2, val3);
} else if (!timeout) {
return syscall(__NR_futex, uaddr, op, val, NULL, uaddr2, val3);
}
#endif

errno = ENOSYS;
return -1;
}

/**
* futex_syscall_nr_requeue() - __NR_futex/__NR_futex_time64 syscall wrapper
* @uaddr: address of first futex
* @op: futex op code
* @val: typically expected value of uaddr, but varies by op
* @nr_requeue: an op specific meaning
* @uaddr2: address of second futex for some ops
* @val3: varies by op
*/
static inline int
__kernel_futex_syscall_nr_requeue(volatile uint32_t *uaddr, int op, uint32_t val,
uint32_t nr_requeue, volatile uint32_t *uaddr2, int val3)
{
#if defined(__NR_futex_time64)
int ret = syscall(__NR_futex_time64, uaddr, op, val, nr_requeue, uaddr2, val3);

if (ret == 0 || errno != ENOSYS)
return ret;
#endif

#if defined(__NR_futex)
return syscall(__NR_futex, uaddr, op, val, nr_requeue, uaddr2, val3);
#endif

errno = ENOSYS;
return -1;
}

#endif /* _UAPI_LINUX_FUTEX_SYSCALL_H */

0 comments on commit e6bfd30

Please sign in to comment.