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.
|
|
|
|
|
|
|
|
|
|
#include "src/futex-emulation.h"
|
|
|
|
|
|
|
|
|
|
#include <limits>
|
|
|
|
|
|
|
|
|
|
#include "src/base/macros.h"
|
|
|
|
|
#include "src/base/platform/time.h"
|
|
|
|
|
#include "src/conversions.h"
|
|
|
|
|
#include "src/handles-inl.h"
|
|
|
|
|
#include "src/isolate.h"
|
2017-01-09 13:43:28 +00:00
|
|
|
|
#include "src/objects-inl.h"
|
2018-08-16 16:01:36 +00:00
|
|
|
|
#include "src/objects/js-array-buffer-inl.h"
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
|
namespace internal {
|
|
|
|
|
|
2018-06-01 08:47:41 +00:00
|
|
|
|
using AtomicsWaitEvent = v8::Isolate::AtomicsWaitEvent;
|
|
|
|
|
|
2015-07-17 17:11:32 +00:00
|
|
|
|
base::LazyMutex FutexEmulation::mutex_ = LAZY_MUTEX_INITIALIZER;
|
|
|
|
|
base::LazyInstance<FutexWaitList>::type FutexEmulation::wait_list_ =
|
|
|
|
|
LAZY_INSTANCE_INITIALIZER;
|
|
|
|
|
|
|
|
|
|
|
2015-08-21 16:41:43 +00:00
|
|
|
|
void FutexWaitListNode::NotifyWake() {
|
|
|
|
|
// Lock the FutexEmulation mutex before notifying. We know that the mutex
|
|
|
|
|
// will have been unlocked if we are currently waiting on the condition
|
2018-11-27 23:05:52 +00:00
|
|
|
|
// variable. The mutex will not be locked if FutexEmulation::Wait hasn't
|
|
|
|
|
// locked it yet. In that case, we set the interrupted_
|
|
|
|
|
// flag to true, which will be tested after the mutex locked by a future wait.
|
2018-10-12 13:52:49 +00:00
|
|
|
|
base::MutexGuard lock_guard(FutexEmulation::mutex_.Pointer());
|
2018-11-27 23:05:52 +00:00
|
|
|
|
// if not waiting, this will not have any effect.
|
|
|
|
|
cond_.NotifyOne();
|
|
|
|
|
interrupted_ = true;
|
2015-08-21 16:41:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-07-17 17:11:32 +00:00
|
|
|
|
FutexWaitList::FutexWaitList() : head_(nullptr), tail_(nullptr) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void FutexWaitList::AddNode(FutexWaitListNode* node) {
|
|
|
|
|
DCHECK(node->prev_ == nullptr && node->next_ == nullptr);
|
|
|
|
|
if (tail_) {
|
|
|
|
|
tail_->next_ = node;
|
|
|
|
|
} else {
|
|
|
|
|
head_ = node;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node->prev_ = tail_;
|
|
|
|
|
node->next_ = nullptr;
|
|
|
|
|
tail_ = node;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void FutexWaitList::RemoveNode(FutexWaitListNode* node) {
|
|
|
|
|
if (node->prev_) {
|
|
|
|
|
node->prev_->next_ = node->next_;
|
|
|
|
|
} else {
|
|
|
|
|
head_ = node->next_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (node->next_) {
|
|
|
|
|
node->next_->prev_ = node->prev_;
|
|
|
|
|
} else {
|
|
|
|
|
tail_ = node->prev_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node->prev_ = node->next_ = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-01 08:47:41 +00:00
|
|
|
|
void AtomicsWaitWakeHandle::Wake() {
|
2018-06-18 22:23:30 +00:00
|
|
|
|
// Adding a separate `NotifyWake()` variant that doesn't acquire the lock
|
|
|
|
|
// itself would likely just add unnecessary complexity..
|
|
|
|
|
// The split lock by itself isn’t an issue, as long as the caller properly
|
|
|
|
|
// synchronizes this with the closing `AtomicsWaitCallback`.
|
|
|
|
|
{
|
2018-10-12 13:52:49 +00:00
|
|
|
|
base::MutexGuard lock_guard(FutexEmulation::mutex_.Pointer());
|
2018-06-18 22:23:30 +00:00
|
|
|
|
stopped_ = true;
|
|
|
|
|
}
|
2018-06-01 08:47:41 +00:00
|
|
|
|
isolate_->futex_wait_list_node()->NotifyWake();
|
|
|
|
|
}
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
2018-11-20 01:08:30 +00:00
|
|
|
|
enum WaitReturnValue : int { kOk = 0, kNotEqual = 1, kTimedOut = 2 };
|
|
|
|
|
|
2018-12-25 00:19:47 +00:00
|
|
|
|
Object FutexEmulation::WaitJs(Isolate* isolate,
|
|
|
|
|
Handle<JSArrayBuffer> array_buffer, size_t addr,
|
|
|
|
|
int32_t value, double rel_timeout_ms) {
|
|
|
|
|
Object res = Wait32(isolate, array_buffer, addr, value, rel_timeout_ms);
|
2018-11-20 01:08:30 +00:00
|
|
|
|
if (res->IsSmi()) {
|
|
|
|
|
int val = Smi::ToInt(res);
|
|
|
|
|
switch (val) {
|
|
|
|
|
case WaitReturnValue::kOk:
|
|
|
|
|
return ReadOnlyRoots(isolate).ok();
|
|
|
|
|
case WaitReturnValue::kNotEqual:
|
|
|
|
|
return ReadOnlyRoots(isolate).not_equal();
|
|
|
|
|
case WaitReturnValue::kTimedOut:
|
|
|
|
|
return ReadOnlyRoots(isolate).timed_out();
|
|
|
|
|
default:
|
|
|
|
|
UNREACHABLE();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-25 00:19:47 +00:00
|
|
|
|
Object FutexEmulation::Wait32(Isolate* isolate,
|
|
|
|
|
Handle<JSArrayBuffer> array_buffer, size_t addr,
|
|
|
|
|
int32_t value, double rel_timeout_ms) {
|
2018-11-30 01:03:35 +00:00
|
|
|
|
return Wait<int32_t>(isolate, array_buffer, addr, value, rel_timeout_ms);
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-25 00:19:47 +00:00
|
|
|
|
Object FutexEmulation::Wait64(Isolate* isolate,
|
|
|
|
|
Handle<JSArrayBuffer> array_buffer, size_t addr,
|
|
|
|
|
int64_t value, double rel_timeout_ms) {
|
2018-11-30 01:03:35 +00:00
|
|
|
|
return Wait<int64_t>(isolate, array_buffer, addr, value, rel_timeout_ms);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
2018-12-25 00:19:47 +00:00
|
|
|
|
Object FutexEmulation::Wait(Isolate* isolate,
|
|
|
|
|
Handle<JSArrayBuffer> array_buffer, size_t addr,
|
|
|
|
|
T value, double rel_timeout_ms) {
|
2018-09-13 17:18:18 +00:00
|
|
|
|
DCHECK_LT(addr, array_buffer->byte_length());
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
|
|
bool use_timeout = rel_timeout_ms != V8_INFINITY;
|
|
|
|
|
|
|
|
|
|
base::TimeDelta rel_timeout;
|
|
|
|
|
if (use_timeout) {
|
|
|
|
|
// Convert to nanoseconds.
|
|
|
|
|
double rel_timeout_ns = rel_timeout_ms *
|
|
|
|
|
base::Time::kNanosecondsPerMicrosecond *
|
|
|
|
|
base::Time::kMicrosecondsPerMillisecond;
|
|
|
|
|
if (rel_timeout_ns >
|
|
|
|
|
static_cast<double>(std::numeric_limits<int64_t>::max())) {
|
|
|
|
|
// 2**63 nanoseconds is 292 years. Let's just treat anything greater as
|
|
|
|
|
// infinite.
|
|
|
|
|
use_timeout = false;
|
|
|
|
|
} else {
|
|
|
|
|
rel_timeout = base::TimeDelta::FromNanoseconds(
|
|
|
|
|
static_cast<int64_t>(rel_timeout_ns));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-01 08:47:41 +00:00
|
|
|
|
AtomicsWaitWakeHandle stop_handle(isolate);
|
|
|
|
|
|
|
|
|
|
isolate->RunAtomicsWaitCallback(AtomicsWaitEvent::kStartWait, array_buffer,
|
|
|
|
|
addr, value, rel_timeout_ms, &stop_handle);
|
|
|
|
|
|
|
|
|
|
if (isolate->has_scheduled_exception()) {
|
|
|
|
|
return isolate->PromoteScheduledException();
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-25 00:19:47 +00:00
|
|
|
|
Object result;
|
2018-06-01 08:47:41 +00:00
|
|
|
|
AtomicsWaitEvent callback_result = AtomicsWaitEvent::kWokenUp;
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
2018-06-11 22:10:25 +00:00
|
|
|
|
do { // Not really a loop, just makes it easier to break out early.
|
2018-10-12 13:52:49 +00:00
|
|
|
|
base::MutexGuard lock_guard(mutex_.Pointer());
|
2018-11-27 23:05:52 +00:00
|
|
|
|
void* backing_store = array_buffer->backing_store();
|
|
|
|
|
|
|
|
|
|
FutexWaitListNode* node = isolate->futex_wait_list_node();
|
|
|
|
|
node->backing_store_ = backing_store;
|
|
|
|
|
node->wait_addr_ = addr;
|
|
|
|
|
node->waiting_ = true;
|
|
|
|
|
|
2018-06-18 22:23:30 +00:00
|
|
|
|
// Reset node->waiting_ = false when leaving this scope (but while
|
|
|
|
|
// still holding the lock).
|
|
|
|
|
ResetWaitingOnScopeExit reset_waiting(node);
|
2018-06-11 22:10:25 +00:00
|
|
|
|
|
2018-11-30 01:03:35 +00:00
|
|
|
|
T* p = reinterpret_cast<T*>(static_cast<int8_t*>(backing_store) + addr);
|
2018-06-11 22:10:25 +00:00
|
|
|
|
if (*p != value) {
|
2018-11-20 01:08:30 +00:00
|
|
|
|
result = Smi::FromInt(WaitReturnValue::kNotEqual);
|
2018-06-11 22:10:25 +00:00
|
|
|
|
callback_result = AtomicsWaitEvent::kNotEqual;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
base::TimeTicks timeout_time;
|
|
|
|
|
base::TimeTicks current_time;
|
|
|
|
|
|
|
|
|
|
if (use_timeout) {
|
|
|
|
|
current_time = base::TimeTicks::Now();
|
|
|
|
|
timeout_time = current_time + rel_timeout;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-29 21:33:04 +00:00
|
|
|
|
wait_list_.Pointer()->AddNode(node);
|
|
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
|
bool interrupted = node->interrupted_;
|
|
|
|
|
node->interrupted_ = false;
|
|
|
|
|
|
|
|
|
|
// Unlock the mutex here to prevent deadlock from lock ordering between
|
|
|
|
|
// mutex_ and mutexes locked by HandleInterrupts.
|
|
|
|
|
mutex_.Pointer()->Unlock();
|
|
|
|
|
|
|
|
|
|
// Because the mutex is unlocked, we have to be careful about not dropping
|
|
|
|
|
// an interrupt. The notification can happen in three different places:
|
|
|
|
|
// 1) Before Wait is called: the notification will be dropped, but
|
|
|
|
|
// interrupted_ will be set to 1. This will be checked below.
|
|
|
|
|
// 2) After interrupted has been checked here, but before mutex_ is
|
|
|
|
|
// acquired: interrupted is checked again below, with mutex_ locked.
|
|
|
|
|
// Because the wakeup signal also acquires mutex_, we know it will not
|
|
|
|
|
// be able to notify until mutex_ is released below, when waiting on
|
|
|
|
|
// the condition variable.
|
|
|
|
|
// 3) After the mutex is released in the call to WaitFor(): this
|
|
|
|
|
// notification will wake up the condition variable. node->waiting() will
|
|
|
|
|
// be false, so we'll loop and then check interrupts.
|
|
|
|
|
if (interrupted) {
|
2018-12-25 00:19:47 +00:00
|
|
|
|
Object interrupt_object = isolate->stack_guard()->HandleInterrupts();
|
2018-05-29 21:33:04 +00:00
|
|
|
|
if (interrupt_object->IsException(isolate)) {
|
|
|
|
|
result = interrupt_object;
|
2018-06-01 08:47:41 +00:00
|
|
|
|
callback_result = AtomicsWaitEvent::kTerminatedExecution;
|
2018-05-29 21:33:04 +00:00
|
|
|
|
mutex_.Pointer()->Lock();
|
|
|
|
|
break;
|
|
|
|
|
}
|
2015-08-21 16:41:43 +00:00
|
|
|
|
}
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
2018-05-29 21:33:04 +00:00
|
|
|
|
mutex_.Pointer()->Lock();
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
2018-05-29 21:33:04 +00:00
|
|
|
|
if (node->interrupted_) {
|
|
|
|
|
// An interrupt occurred while the mutex_ was unlocked. Don't wait yet.
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
2018-06-01 08:47:41 +00:00
|
|
|
|
if (stop_handle.has_stopped()) {
|
|
|
|
|
node->waiting_ = false;
|
|
|
|
|
callback_result = AtomicsWaitEvent::kAPIStopped;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-29 21:33:04 +00:00
|
|
|
|
if (!node->waiting_) {
|
2018-11-20 01:08:30 +00:00
|
|
|
|
result = Smi::FromInt(WaitReturnValue::kOk);
|
2015-08-21 16:41:43 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-29 21:33:04 +00:00
|
|
|
|
// No interrupts, now wait.
|
|
|
|
|
if (use_timeout) {
|
|
|
|
|
current_time = base::TimeTicks::Now();
|
|
|
|
|
if (current_time >= timeout_time) {
|
2018-11-20 01:08:30 +00:00
|
|
|
|
result = Smi::FromInt(WaitReturnValue::kTimedOut);
|
2018-06-01 08:47:41 +00:00
|
|
|
|
callback_result = AtomicsWaitEvent::kTimedOut;
|
2018-05-29 21:33:04 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
base::TimeDelta time_until_timeout = timeout_time - current_time;
|
|
|
|
|
DCHECK_GE(time_until_timeout.InMicroseconds(), 0);
|
|
|
|
|
bool wait_for_result =
|
|
|
|
|
node->cond_.WaitFor(mutex_.Pointer(), time_until_timeout);
|
|
|
|
|
USE(wait_for_result);
|
|
|
|
|
} else {
|
|
|
|
|
node->cond_.Wait(mutex_.Pointer());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Spurious wakeup, interrupt or timeout.
|
2015-07-17 17:11:32 +00:00
|
|
|
|
}
|
2015-08-21 16:41:43 +00:00
|
|
|
|
|
2018-05-29 21:33:04 +00:00
|
|
|
|
wait_list_.Pointer()->RemoveNode(node);
|
2018-09-14 13:13:54 +00:00
|
|
|
|
} while (false);
|
2018-06-11 22:10:25 +00:00
|
|
|
|
|
2018-06-01 08:47:41 +00:00
|
|
|
|
isolate->RunAtomicsWaitCallback(callback_result, array_buffer, addr, value,
|
|
|
|
|
rel_timeout_ms, nullptr);
|
|
|
|
|
|
|
|
|
|
if (isolate->has_scheduled_exception()) {
|
|
|
|
|
CHECK_NE(callback_result, AtomicsWaitEvent::kTerminatedExecution);
|
|
|
|
|
result = isolate->PromoteScheduledException();
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-17 17:11:32 +00:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-25 00:19:47 +00:00
|
|
|
|
Object FutexEmulation::Wake(Handle<JSArrayBuffer> array_buffer, size_t addr,
|
|
|
|
|
uint32_t num_waiters_to_wake) {
|
2018-09-13 17:18:18 +00:00
|
|
|
|
DCHECK_LT(addr, array_buffer->byte_length());
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
|
|
int waiters_woken = 0;
|
|
|
|
|
void* backing_store = array_buffer->backing_store();
|
|
|
|
|
|
2018-10-12 13:52:49 +00:00
|
|
|
|
base::MutexGuard lock_guard(mutex_.Pointer());
|
2015-07-17 17:11:32 +00:00
|
|
|
|
FutexWaitListNode* node = wait_list_.Pointer()->head_;
|
|
|
|
|
while (node && num_waiters_to_wake > 0) {
|
2018-11-30 01:03:35 +00:00
|
|
|
|
if (backing_store == node->backing_store_ && addr == node->wait_addr_ &&
|
|
|
|
|
node->waiting_) {
|
2015-07-17 17:11:32 +00:00
|
|
|
|
node->waiting_ = false;
|
|
|
|
|
node->cond_.NotifyOne();
|
2017-02-01 21:47:22 +00:00
|
|
|
|
if (num_waiters_to_wake != kWakeAll) {
|
|
|
|
|
--num_waiters_to_wake;
|
|
|
|
|
}
|
2015-07-17 17:11:32 +00:00
|
|
|
|
waiters_woken++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node = node->next_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Smi::FromInt(waiters_woken);
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-25 00:19:47 +00:00
|
|
|
|
Object FutexEmulation::NumWaitersForTesting(Handle<JSArrayBuffer> array_buffer,
|
|
|
|
|
size_t addr) {
|
2018-09-13 17:18:18 +00:00
|
|
|
|
DCHECK_LT(addr, array_buffer->byte_length());
|
2015-07-17 17:11:32 +00:00
|
|
|
|
void* backing_store = array_buffer->backing_store();
|
|
|
|
|
|
2018-10-12 13:52:49 +00:00
|
|
|
|
base::MutexGuard lock_guard(mutex_.Pointer());
|
2015-07-17 17:11:32 +00:00
|
|
|
|
|
|
|
|
|
int waiters = 0;
|
|
|
|
|
FutexWaitListNode* node = wait_list_.Pointer()->head_;
|
|
|
|
|
while (node) {
|
2015-11-20 11:39:25 +00:00
|
|
|
|
if (backing_store == node->backing_store_ && addr == node->wait_addr_ &&
|
|
|
|
|
node->waiting_) {
|
2015-07-17 17:11:32 +00:00
|
|
|
|
waiters++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node = node->next_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Smi::FromInt(waiters);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
|
} // namespace v8
|