2015-07-17 17:11:32 +00:00
|
|
|
// Copyright 2015 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#ifndef V8_FUTEX_EMULATION_H_
|
|
|
|
#define V8_FUTEX_EMULATION_H_
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "src/allocation.h"
|
2015-08-21 16:41:43 +00:00
|
|
|
#include "src/base/atomicops.h"
|
2015-07-17 17:11:32 +00:00
|
|
|
#include "src/base/lazy-instance.h"
|
|
|
|
#include "src/base/macros.h"
|
|
|
|
#include "src/base/platform/condition-variable.h"
|
|
|
|
#include "src/base/platform/mutex.h"
|
|
|
|
|
|
|
|
// Support for emulating futexes, a low-level synchronization primitive. They
|
|
|
|
// are natively supported by Linux, but must be emulated for other platforms.
|
|
|
|
// This library emulates them on all platforms using mutexes and condition
|
|
|
|
// variables for consistency.
|
|
|
|
//
|
|
|
|
// This is used by the Futex API defined in the SharedArrayBuffer draft spec,
|
2016-07-13 18:29:48 +00:00
|
|
|
// found here: https://github.com/tc39/ecmascript_sharedmem
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
|
|
|
|
namespace base {
|
|
|
|
class TimeDelta;
|
|
|
|
} // base
|
|
|
|
|
|
|
|
namespace internal {
|
|
|
|
|
2017-01-27 13:53:13 +00:00
|
|
|
template <typename T>
|
|
|
|
class Handle;
|
2015-07-17 17:11:32 +00:00
|
|
|
class Isolate;
|
2015-08-11 12:00:01 +00:00
|
|
|
class JSArrayBuffer;
|
2015-07-17 17:11:32 +00:00
|
|
|
|
2018-06-01 08:47:41 +00:00
|
|
|
class AtomicsWaitWakeHandle {
|
|
|
|
public:
|
|
|
|
explicit AtomicsWaitWakeHandle(Isolate* isolate) : isolate_(isolate) {}
|
|
|
|
|
|
|
|
void Wake();
|
|
|
|
inline bool has_stopped() const { return stopped_; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
Isolate* isolate_;
|
|
|
|
bool stopped_ = false;
|
|
|
|
};
|
|
|
|
|
2015-07-17 17:11:32 +00:00
|
|
|
class FutexWaitListNode {
|
|
|
|
public:
|
|
|
|
FutexWaitListNode()
|
|
|
|
: prev_(nullptr),
|
|
|
|
next_(nullptr),
|
|
|
|
backing_store_(nullptr),
|
|
|
|
wait_addr_(0),
|
2015-08-21 16:41:43 +00:00
|
|
|
waiting_(false),
|
|
|
|
interrupted_(false) {}
|
|
|
|
|
|
|
|
void NotifyWake();
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
friend class FutexEmulation;
|
|
|
|
friend class FutexWaitList;
|
2018-06-18 22:23:30 +00:00
|
|
|
friend class ResetWaitingOnScopeExit;
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
base::ConditionVariable cond_;
|
2018-05-29 21:33:04 +00:00
|
|
|
// prev_ and next_ are protected by FutexEmulation::mutex_.
|
2015-07-17 17:11:32 +00:00
|
|
|
FutexWaitListNode* prev_;
|
|
|
|
FutexWaitListNode* next_;
|
|
|
|
void* backing_store_;
|
|
|
|
size_t wait_addr_;
|
2018-05-29 21:33:04 +00:00
|
|
|
// waiting_ and interrupted_ are protected by FutexEmulation::mutex_
|
2018-06-18 22:23:30 +00:00
|
|
|
// if this node is currently contained in FutexEmulation::wait_list_
|
|
|
|
// or an AtomicsWaitWakeHandle has access to it.
|
2015-07-17 17:11:32 +00:00
|
|
|
bool waiting_;
|
2015-08-21 16:41:43 +00:00
|
|
|
bool interrupted_;
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(FutexWaitListNode);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class FutexWaitList {
|
|
|
|
public:
|
|
|
|
FutexWaitList();
|
|
|
|
|
|
|
|
void AddNode(FutexWaitListNode* node);
|
|
|
|
void RemoveNode(FutexWaitListNode* node);
|
|
|
|
|
|
|
|
private:
|
|
|
|
friend class FutexEmulation;
|
|
|
|
|
|
|
|
FutexWaitListNode* head_;
|
|
|
|
FutexWaitListNode* tail_;
|
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(FutexWaitList);
|
|
|
|
};
|
|
|
|
|
2018-06-18 22:23:30 +00:00
|
|
|
class ResetWaitingOnScopeExit {
|
|
|
|
public:
|
|
|
|
explicit ResetWaitingOnScopeExit(FutexWaitListNode* node) : node_(node) {}
|
|
|
|
~ResetWaitingOnScopeExit() { node_->waiting_ = false; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
FutexWaitListNode* node_;
|
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(ResetWaitingOnScopeExit);
|
|
|
|
};
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
class FutexEmulation : public AllStatic {
|
|
|
|
public:
|
2017-02-01 21:47:22 +00:00
|
|
|
// Pass to Wake() to wake all waiters.
|
|
|
|
static const uint32_t kWakeAll = UINT32_MAX;
|
|
|
|
|
2016-07-13 18:29:48 +00:00
|
|
|
// Check that array_buffer[addr] == value, and return "not-equal" if not. If
|
2015-07-17 17:11:32 +00:00
|
|
|
// they are equal, block execution on |isolate|'s thread until woken via
|
|
|
|
// |Wake|, or when the time given in |rel_timeout_ms| elapses. Note that
|
|
|
|
// |rel_timeout_ms| can be Infinity.
|
2016-07-13 18:29:48 +00:00
|
|
|
// If woken, return "ok", otherwise return "timed-out". The initial check and
|
2015-07-17 17:11:32 +00:00
|
|
|
// the decision to wait happen atomically.
|
2018-11-20 01:08:30 +00:00
|
|
|
static Object* WaitJs(Isolate* isolate, Handle<JSArrayBuffer> array_buffer,
|
|
|
|
size_t addr, int32_t value, double rel_timeout_ms);
|
|
|
|
|
|
|
|
// Same as WaitJs above except it returns 0 (ok), 1 (not equal) and 2 (timed
|
|
|
|
// out) as expected by Wasm.
|
2018-11-30 01:03:35 +00:00
|
|
|
static Object* Wait32(Isolate* isolate, Handle<JSArrayBuffer> array_buffer,
|
|
|
|
size_t addr, int32_t value, double rel_timeout_ms);
|
|
|
|
|
|
|
|
// Same as Wait32 above except it checks for an int64_t value in the
|
|
|
|
// array_buffer.
|
|
|
|
static Object* Wait64(Isolate* isolate, Handle<JSArrayBuffer> array_buffer,
|
|
|
|
size_t addr, int64_t value, double rel_timeout_ms);
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
// Wake |num_waiters_to_wake| threads that are waiting on the given |addr|.
|
2017-02-01 21:47:22 +00:00
|
|
|
// |num_waiters_to_wake| can be kWakeAll, in which case all waiters are
|
|
|
|
// woken. The rest of the waiters will continue to wait. The return value is
|
|
|
|
// the number of woken waiters.
|
2018-07-18 16:05:36 +00:00
|
|
|
static Object* Wake(Handle<JSArrayBuffer> array_buffer, size_t addr,
|
|
|
|
uint32_t num_waiters_to_wake);
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
// Return the number of threads waiting on |addr|. Should only be used for
|
|
|
|
// testing.
|
2018-07-18 16:05:36 +00:00
|
|
|
static Object* NumWaitersForTesting(Handle<JSArrayBuffer> array_buffer,
|
2015-07-17 17:11:32 +00:00
|
|
|
size_t addr);
|
|
|
|
|
|
|
|
private:
|
2015-08-21 16:41:43 +00:00
|
|
|
friend class FutexWaitListNode;
|
2018-06-18 22:23:30 +00:00
|
|
|
friend class AtomicsWaitWakeHandle;
|
2015-08-21 16:41:43 +00:00
|
|
|
|
2018-11-30 01:03:35 +00:00
|
|
|
template <typename T>
|
|
|
|
static Object* Wait(Isolate* isolate, Handle<JSArrayBuffer> array_buffer,
|
|
|
|
size_t addr, T value, double rel_timeout_ms);
|
|
|
|
|
2018-05-29 21:33:04 +00:00
|
|
|
// `mutex_` protects the composition of `wait_list_` (i.e. no elements may be
|
|
|
|
// added or removed without holding this mutex), as well as the `waiting_`
|
|
|
|
// and `interrupted_` fields for each individual list node that is currently
|
|
|
|
// part of the list. It must be the mutex used together with the `cond_`
|
|
|
|
// condition variable of such nodes.
|
2015-07-17 17:11:32 +00:00
|
|
|
static base::LazyMutex mutex_;
|
|
|
|
static base::LazyInstance<FutexWaitList>::type wait_list_;
|
|
|
|
};
|
2015-09-30 13:46:56 +00:00
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
#endif // V8_FUTEX_EMULATION_H_
|