cpp/logger/lfrb/Futex.h (89 lines of code) (raw):

/* * Copyright 2017 Facebook, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #pragma once #include <atomic> #include <cassert> #include <chrono> #include <limits> namespace facebook { namespace profilo { namespace logger { namespace lfrb { enum class FutexResult { VALUE_CHANGED, /* Futex value didn't match expected */ AWOKEN, /* futex wait matched with a futex wake */ INTERRUPTED, /* Spurious wake-up or signal caused futex wait failure */ TIMEDOUT }; /** * Futex is an atomic 32 bit unsigned integer that provides access to the * futex() syscall on that value. It is templated in such a way that it * can interact properly with DeterministicSchedule testing. * * If you don't know how to use futex(), you probably shouldn't be using * this class. Even if you do know how, you should have a good reason * (and benchmarks to back you up). */ template <template <typename> class Atom = std::atomic> struct Futex : Atom<uint32_t> { explicit constexpr Futex(uint32_t init = 0) : Atom<uint32_t>(init) {} Futex(Futex const&) = delete; Futex& operator=(Futex const&) = delete; /** Puts the thread to sleep if this->load() == expected. Returns true when * it is returning because it has consumed a wake() event, false for any * other return (signal, this->load() != expected, or spurious wakeup). */ bool futexWait(uint32_t expected, uint32_t waitMask = -1) { auto rv = futexWaitImpl(expected, nullptr, nullptr, waitMask); assert(rv != FutexResult::TIMEDOUT); return rv == FutexResult::AWOKEN; } /** Similar to futexWait but also accepts a timeout that gives the time until * when the call can block (time is the absolute time i.e time since epoch). * Allowed clock types: std::chrono::system_clock, std::chrono::steady_clock. * Returns one of FutexResult values. * * NOTE: On some systems steady_clock is just an alias for system_clock, * and is not actually steady.*/ template <class Clock, class Duration = typename Clock::duration> FutexResult futexWaitUntil( uint32_t expected, const std::chrono::time_point<Clock, Duration>& absTime, uint32_t waitMask = -1) { using std::chrono::duration_cast; using std::chrono::nanoseconds; using std::chrono::seconds; using std::chrono::steady_clock; using std::chrono::system_clock; using std::chrono::time_point; static_assert( (std::is_same<Clock, system_clock>::value || std::is_same<Clock, steady_clock>::value), "futexWaitUntil only knows std::chrono::{system_clock,steady_clock}"); assert((std::is_same<Clock, system_clock>::value) || Clock::is_steady); // We launder the clock type via a std::chrono::duration so that we // can compile both the true and false branch. Tricky case is when // steady_clock has a higher precision than system_clock (Xcode 6, // for example), for which time_point<system_clock> construction // refuses to do an implicit duration conversion. (duration is // happy to implicitly convert its denominator causing overflow, but // refuses conversion that might cause truncation.) We use explicit // duration_cast to work around this. Truncation does not actually // occur (unless Duration != Clock::duration) because the missing // implicit conversion is in the untaken branch. Duration absTimeDuration = absTime.time_since_epoch(); if (std::is_same<Clock, system_clock>::value) { time_point<system_clock> absSystemTime( duration_cast<system_clock::duration>(absTimeDuration)); return futexWaitImpl(expected, &absSystemTime, nullptr, waitMask); } else { time_point<steady_clock> absSteadyTime( duration_cast<steady_clock::duration>(absTimeDuration)); return futexWaitImpl(expected, nullptr, &absSteadyTime, waitMask); } } /** Wakens up to count waiters where (waitMask & wakeMask) != * 0, returning the number of awoken threads, or -1 if an error * occurred. Note that when constructing a concurrency primitive * that can guard its own destruction, it is likely that you will * want to ignore EINVAL here (as well as making sure that you * never touch the object after performing the memory store that * is the linearization point for unlock or control handoff). * See https://sourceware.org/bugzilla/show_bug.cgi?id=13690 */ int futexWake( int count = std::numeric_limits<int>::max(), uint32_t wakeMask = -1); private: /** Underlying implementation of futexWait and futexWaitUntil. * At most one of absSystemTime and absSteadyTime should be non-null. * Timeouts are separated into separate parameters to allow the * implementations to be elsewhere without templating on the clock * type, which is otherwise complicated by the fact that steady_clock * is the same as system_clock on some platforms. */ FutexResult futexWaitImpl( uint32_t expected, std::chrono::time_point<std::chrono::system_clock>* absSystemTime, std::chrono::time_point<std::chrono::steady_clock>* absSteadyTime, uint32_t waitMask); }; /** A std::atomic subclass that can be used to force Futex to emulate * the underlying futex() syscall. This is primarily useful to test or * benchmark the emulated implementation on systems that don't need it. */ template <typename T> struct EmulatedFutexAtomic : public std::atomic<T> { EmulatedFutexAtomic() noexcept = default; constexpr /* implicit */ EmulatedFutexAtomic(T init) noexcept : std::atomic<T>(init) {} // It doesn't copy or move EmulatedFutexAtomic(EmulatedFutexAtomic&& rhs) = delete; }; /* Available specializations, with definitions elsewhere */ template <> int Futex<std::atomic>::futexWake(int count, uint32_t wakeMask); template <> FutexResult Futex<std::atomic>::futexWaitImpl( uint32_t expected, std::chrono::time_point<std::chrono::system_clock>* absSystemTime, std::chrono::time_point<std::chrono::steady_clock>* absSteadyTime, uint32_t waitMask); template <> int Futex<EmulatedFutexAtomic>::futexWake(int count, uint32_t wakeMask); template <> FutexResult Futex<EmulatedFutexAtomic>::futexWaitImpl( uint32_t expected, std::chrono::time_point<std::chrono::system_clock>* absSystemTime, std::chrono::time_point<std::chrono::steady_clock>* absSteadyTime, uint32_t waitMask); } // namespace lfrb } // namespace logger } // namespace profilo } // namespace facebook