mirror of
https://github.com/ecency/ecency-mobile.git
synced 2024-12-23 13:22:02 +03:00
167 lines
6.8 KiB
C++
167 lines
6.8 KiB
C++
/*
|
|
* Copyright 2016 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 <chrono>
|
|
#include <limits>
|
|
#include <assert.h>
|
|
#include <boost/noncopyable.hpp>
|
|
|
|
#include <folly/portability/Unistd.h>
|
|
|
|
namespace folly { namespace detail {
|
|
|
|
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>, boost::noncopyable {
|
|
|
|
explicit Futex(uint32_t init = 0) : Atom<uint32_t>(init) {}
|
|
|
|
/** 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);
|
|
|
|
}}
|