util: emulate futex on FreeBSD using umtx
authorGreg V <greg@unrelenting.technology>
Sun, 24 Dec 2017 16:55:46 +0000 (19:55 +0300)
committerEric Engestrom <eric@engestrom.ch>
Sun, 3 Mar 2019 19:48:49 +0000 (19:48 +0000)
Obtained from: FreeBSD ports
Acked-by: Emil Velikov <emil.velikov@collabora.com>
Acked-by: Eric Engestrom <eric.engestrom@intel.com>
src/util/futex.h

index 4402893069d5722ad7b227957b42632f2cbf23de..a8b2c0f3527df7626958a9e50ea672cf1f016a15 100644 (file)
@@ -51,6 +51,40 @@ static inline int futex_wait(uint32_t *addr, int32_t value, const struct timespe
                     FUTEX_BITSET_MATCH_ANY);
 }
 
+#elif defined(__FreeBSD__)
+
+#include <assert.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <sys/types.h>
+#include <sys/umtx.h>
+#include <sys/time.h>
+
+static inline int futex_wake(uint32_t *addr, int count)
+{
+   assert(count == (int)(uint32_t)count); /* Check that bits weren't discarded */
+   return _umtx_op(addr, UMTX_OP_WAKE, (uint32_t)count, NULL, NULL) == -1 ? errno : 0;
+}
+
+static inline int futex_wait(uint32_t *addr, int32_t value, struct timespec *timeout)
+{
+   void *uaddr = NULL, *uaddr2 = NULL;
+
+   assert(value == (int)(uint32_t)value); /* Check that bits weren't discarded */
+
+   if (timeout != NULL) {
+      const struct _umtx_time tmo = {
+         ._timeout = *timeout,
+         ._flags = UMTX_ABSTIME,
+         ._clockid = CLOCK_MONOTONIC
+      };
+      uaddr = (void *)(uintptr_t)sizeof(tmo);
+      uaddr2 = (void *)&tmo;
+   }
+
+   return _umtx_op(addr, UMTX_OP_WAIT_UINT, (uint32_t)value, uaddr, uaddr2) == -1 ? errno : 0;
+}
+
 #endif
 
 #endif /* UTIL_FUTEX_H */