Commit 47207c26 authored by Adam Klein's avatar Adam Klein Committed by V8 LUCI CQ

Revert "[shared-struct] Add Atomics.Condition"

This reverts commit e2066ff6.

Reason for revert: fails tests on GC stress bot:
https://ci.chromium.org/ui/p/v8/builders/ci/V8%20Linux64%20GC%20Stress%20-%20custom%20snapshot/42868/overview

Original change's description:
> [shared-struct] Add Atomics.Condition
>
> Bug: v8:12547
> Change-Id: Id439aef9cab3348171a23378cdd47ede5f4d7288
> Cq-Include-Trybots: luci.v8.try:v8_linux_arm64_rel_ng,v8_linux64_tsan_rel_ng
> Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3630350
> Reviewed-by: Dominik Inführ <dinfuehr@chromium.org>
> Reviewed-by: Adam Klein <adamk@chromium.org>
> Commit-Queue: Shu-yu Guo <syg@chromium.org>
> Cr-Commit-Position: refs/heads/main@{#81734}

Bug: v8:12547
Change-Id: I237b744e5be8725cbe41ca73076d951018ca80a0
Cq-Include-Trybots: luci.v8.try:v8_linux_arm64_rel_ng,v8_linux64_tsan_rel_ng
No-Presubmit: true
No-Tree-Checks: true
No-Try: true
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3763784
Auto-Submit: Adam Klein <adamk@chromium.org>
Commit-Queue: Rubber Stamper <rubber-stamper@appspot.gserviceaccount.com>
Bot-Commit: Rubber Stamper <rubber-stamper@appspot.gserviceaccount.com>
Cr-Commit-Position: refs/heads/main@{#81735}
parent e2066ff6
...@@ -29,8 +29,8 @@ BUILTIN(AtomicsMutexLock) { ...@@ -29,8 +29,8 @@ BUILTIN(AtomicsMutexLock) {
Handle<JSAtomicsMutex> js_mutex = Handle<JSAtomicsMutex>::cast(js_mutex_obj); Handle<JSAtomicsMutex> js_mutex = Handle<JSAtomicsMutex>::cast(js_mutex_obj);
Handle<Object> run_under_lock = args.atOrUndefined(isolate, 2); Handle<Object> run_under_lock = args.atOrUndefined(isolate, 2);
if (!run_under_lock->IsCallable()) { if (!run_under_lock->IsCallable()) {
THROW_NEW_ERROR_RETURN_FAILURE( THROW_NEW_ERROR_RETURN_FAILURE(isolate,
isolate, NewTypeError(MessageTemplate::kNotCallable, run_under_lock)); NewTypeError(MessageTemplate::kNotCallable));
} }
// Like Atomics.wait, synchronous locking may block, and so is disallowed on // Like Atomics.wait, synchronous locking may block, and so is disallowed on
...@@ -39,9 +39,7 @@ BUILTIN(AtomicsMutexLock) { ...@@ -39,9 +39,7 @@ BUILTIN(AtomicsMutexLock) {
// This is not a recursive lock, so also throw if recursively locking. // This is not a recursive lock, so also throw if recursively locking.
if (!isolate->allow_atomics_wait() || js_mutex->IsCurrentThreadOwner()) { if (!isolate->allow_atomics_wait() || js_mutex->IsCurrentThreadOwner()) {
THROW_NEW_ERROR_RETURN_FAILURE( THROW_NEW_ERROR_RETURN_FAILURE(
isolate, NewTypeError(MessageTemplate::kAtomicsOperationNotAllowed, isolate, NewTypeError(MessageTemplate::kAtomicsMutexLockNotAllowed));
isolate->factory()->NewStringFromAsciiChecked(
method_name)));
} }
Handle<Object> result; Handle<Object> result;
...@@ -71,8 +69,8 @@ BUILTIN(AtomicsMutexTryLock) { ...@@ -71,8 +69,8 @@ BUILTIN(AtomicsMutexTryLock) {
Handle<JSAtomicsMutex> js_mutex = Handle<JSAtomicsMutex>::cast(js_mutex_obj); Handle<JSAtomicsMutex> js_mutex = Handle<JSAtomicsMutex>::cast(js_mutex_obj);
Handle<Object> run_under_lock = args.atOrUndefined(isolate, 2); Handle<Object> run_under_lock = args.atOrUndefined(isolate, 2);
if (!run_under_lock->IsCallable()) { if (!run_under_lock->IsCallable()) {
THROW_NEW_ERROR_RETURN_FAILURE( THROW_NEW_ERROR_RETURN_FAILURE(isolate,
isolate, NewTypeError(MessageTemplate::kNotCallable, run_under_lock)); NewTypeError(MessageTemplate::kNotCallable));
} }
JSAtomicsMutex::TryLockGuard try_lock_guard(isolate, js_mutex); JSAtomicsMutex::TryLockGuard try_lock_guard(isolate, js_mutex);
...@@ -88,96 +86,5 @@ BUILTIN(AtomicsMutexTryLock) { ...@@ -88,96 +86,5 @@ BUILTIN(AtomicsMutexTryLock) {
return ReadOnlyRoots(isolate).false_value(); return ReadOnlyRoots(isolate).false_value();
} }
BUILTIN(AtomicsConditionConstructor) {
DCHECK(FLAG_harmony_struct);
HandleScope scope(isolate);
return *JSAtomicsCondition::Create(isolate);
}
BUILTIN(AtomicsConditionWait) {
DCHECK(FLAG_harmony_struct);
constexpr char method_name[] = "Atomics.Condition.wait";
HandleScope scope(isolate);
Handle<Object> js_condition_obj = args.atOrUndefined(isolate, 1);
Handle<Object> js_mutex_obj = args.atOrUndefined(isolate, 2);
Handle<Object> timeout_obj = args.atOrUndefined(isolate, 3);
if (!js_condition_obj->IsJSAtomicsCondition() ||
!js_mutex_obj->IsJSAtomicsMutex()) {
THROW_NEW_ERROR_RETURN_FAILURE(
isolate, NewTypeError(MessageTemplate::kMethodInvokedOnWrongType,
isolate->factory()->NewStringFromAsciiChecked(
method_name)));
}
base::Optional<base::TimeDelta> timeout = base::nullopt;
if (!timeout_obj->IsUndefined(isolate)) {
ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, timeout_obj,
Object::ToNumber(isolate, timeout_obj));
double ms = timeout_obj->Number();
if (!std::isnan(ms)) {
if (ms < 0) ms = 0;
if (ms <= static_cast<double>(std::numeric_limits<int64_t>::max())) {
timeout = base::TimeDelta::FromMilliseconds(static_cast<int64_t>(ms));
}
}
}
if (!isolate->allow_atomics_wait()) {
THROW_NEW_ERROR_RETURN_FAILURE(
isolate, NewTypeError(MessageTemplate::kAtomicsOperationNotAllowed,
isolate->factory()->NewStringFromAsciiChecked(
method_name)));
}
Handle<JSAtomicsCondition> js_condition =
Handle<JSAtomicsCondition>::cast(js_condition_obj);
Handle<JSAtomicsMutex> js_mutex = Handle<JSAtomicsMutex>::cast(js_mutex_obj);
if (!js_mutex->IsCurrentThreadOwner()) {
THROW_NEW_ERROR_RETURN_FAILURE(
isolate,
NewTypeError(MessageTemplate::kAtomicsMutexNotOwnedByCurrentThread));
}
return isolate->heap()->ToBoolean(
JSAtomicsCondition::WaitFor(isolate, js_condition, js_mutex, timeout));
}
BUILTIN(AtomicsConditionNotify) {
DCHECK(FLAG_harmony_struct);
constexpr char method_name[] = "Atomics.Condition.notify";
HandleScope scope(isolate);
Handle<Object> js_condition_obj = args.atOrUndefined(isolate, 1);
Handle<Object> count_obj = args.atOrUndefined(isolate, 2);
if (!js_condition_obj->IsJSAtomicsCondition()) {
THROW_NEW_ERROR_RETURN_FAILURE(
isolate, NewTypeError(MessageTemplate::kMethodInvokedOnWrongType,
isolate->factory()->NewStringFromAsciiChecked(
method_name)));
}
uint32_t count;
if (count_obj->IsUndefined(isolate)) {
count = JSAtomicsCondition::kAllWaiters;
} else {
ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, count_obj,
Object::ToInteger(isolate, count_obj));
double count_double = count_obj->Number();
if (count_double < 0) {
count_double = 0;
} else if (count_double > JSAtomicsCondition::kAllWaiters) {
count_double = JSAtomicsCondition::kAllWaiters;
}
count = static_cast<uint32_t>(count_double);
}
Handle<JSAtomicsCondition> js_condition =
Handle<JSAtomicsCondition>::cast(js_condition_obj);
return *isolate->factory()->NewNumberFromUint(
js_condition->Notify(isolate, count));
}
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
...@@ -1004,9 +1004,6 @@ namespace internal { ...@@ -1004,9 +1004,6 @@ namespace internal {
CPP(AtomicsMutexConstructor) \ CPP(AtomicsMutexConstructor) \
CPP(AtomicsMutexLock) \ CPP(AtomicsMutexLock) \
CPP(AtomicsMutexTryLock) \ CPP(AtomicsMutexTryLock) \
CPP(AtomicsConditionConstructor) \
CPP(AtomicsConditionWait) \
CPP(AtomicsConditionNotify) \
\ \
/* AsyncGenerator */ \ /* AsyncGenerator */ \
\ \
......
...@@ -231,9 +231,7 @@ Object DoWait(Isolate* isolate, FutexEmulation::WaitMode mode, ...@@ -231,9 +231,7 @@ Object DoWait(Isolate* isolate, FutexEmulation::WaitMode mode,
if (mode == FutexEmulation::WaitMode::kSync && if (mode == FutexEmulation::WaitMode::kSync &&
!isolate->allow_atomics_wait()) { !isolate->allow_atomics_wait()) {
THROW_NEW_ERROR_RETURN_FAILURE( THROW_NEW_ERROR_RETURN_FAILURE(
isolate, NewTypeError(MessageTemplate::kAtomicsOperationNotAllowed, isolate, NewTypeError(MessageTemplate::kAtomicsWaitNotAllowed));
isolate->factory()->NewStringFromAsciiChecked(
"Atomics.wait")));
} }
Handle<JSArrayBuffer> array_buffer = sta->GetBuffer(); Handle<JSArrayBuffer> array_buffer = sta->GetBuffer();
......
...@@ -41,9 +41,9 @@ namespace internal { ...@@ -41,9 +41,9 @@ namespace internal {
T(AwaitNotInDebugEvaluate, \ T(AwaitNotInDebugEvaluate, \
"await can not be used when evaluating code " \ "await can not be used when evaluating code " \
"while paused in the debugger") \ "while paused in the debugger") \
T(AtomicsMutexNotOwnedByCurrentThread, \ T(AtomicsMutexLockNotAllowed, \
"Atomics.Mutex is not owned by the current agent") \ "Atomics.Mutex.lock cannot be called in this context") \
T(AtomicsOperationNotAllowed, "% cannot be called in this context") \ T(AtomicsWaitNotAllowed, "Atomics.wait cannot be called in this context") \
T(BadRoundingType, "RoundingType is not fractionDigits") \ T(BadRoundingType, "RoundingType is not fractionDigits") \
T(BadSortComparisonFunction, \ T(BadSortComparisonFunction, \
"The comparison function must be either a function or undefined") \ "The comparison function must be either a function or undefined") \
......
...@@ -264,7 +264,6 @@ Type::bitset BitsetType::Lub(const MapRefLike& map) { ...@@ -264,7 +264,6 @@ Type::bitset BitsetType::Lub(const MapRefLike& map) {
case JS_SHADOW_REALM_TYPE: case JS_SHADOW_REALM_TYPE:
case JS_SHARED_ARRAY_TYPE: case JS_SHARED_ARRAY_TYPE:
case JS_SHARED_STRUCT_TYPE: case JS_SHARED_STRUCT_TYPE:
case JS_ATOMICS_CONDITION_TYPE:
case JS_ATOMICS_MUTEX_TYPE: case JS_ATOMICS_MUTEX_TYPE:
case JS_TEMPORAL_CALENDAR_TYPE: case JS_TEMPORAL_CALENDAR_TYPE:
case JS_TEMPORAL_DURATION_TYPE: case JS_TEMPORAL_DURATION_TYPE:
......
...@@ -551,8 +551,7 @@ void Map::MapVerify(Isolate* isolate) { ...@@ -551,8 +551,7 @@ void Map::MapVerify(Isolate* isolate) {
JSObject::GetEmbedderFieldCount(*this) * kEmbedderDataSlotSize, JSObject::GetEmbedderFieldCount(*this) * kEmbedderDataSlotSize,
inobject_fields_start_offset); inobject_fields_start_offset);
if (IsJSSharedStructMap() || IsJSSharedArrayMap() || IsJSAtomicsMutex() || if (IsJSSharedStructMap() || IsJSSharedArrayMap()) {
IsJSAtomicsCondition()) {
CHECK(InSharedHeap()); CHECK(InSharedHeap());
CHECK(GetBackPointer().IsUndefined(isolate)); CHECK(GetBackPointer().IsUndefined(isolate));
Object maybe_cell = prototype_validity_cell(); Object maybe_cell = prototype_validity_cell();
...@@ -1266,12 +1265,10 @@ void JSAtomicsMutex::JSAtomicsMutexVerify(Isolate* isolate) { ...@@ -1266,12 +1265,10 @@ void JSAtomicsMutex::JSAtomicsMutexVerify(Isolate* isolate) {
CHECK(IsJSAtomicsMutex()); CHECK(IsJSAtomicsMutex());
CHECK(InSharedWritableHeap()); CHECK(InSharedWritableHeap());
JSObjectVerify(isolate); JSObjectVerify(isolate);
} Map mutex_map = map();
CHECK(mutex_map.GetBackPointer().IsUndefined(isolate));
void JSAtomicsCondition::JSAtomicsConditionVerify(Isolate* isolate) { CHECK(!mutex_map.is_extensible());
CHECK(IsJSAtomicsCondition()); CHECK(!mutex_map.is_prototype_map());
CHECK(InSharedHeap());
JSObjectVerify(isolate);
} }
void JSSharedArray::JSSharedArrayVerify(Isolate* isolate) { void JSSharedArray::JSSharedArrayVerify(Isolate* isolate) {
......
...@@ -1495,15 +1495,6 @@ void JSAtomicsMutex::JSAtomicsMutexPrint(std::ostream& os) { ...@@ -1495,15 +1495,6 @@ void JSAtomicsMutex::JSAtomicsMutexPrint(std::ostream& os) {
JSObjectPrintBody(os, *this); JSObjectPrintBody(os, *this);
} }
void JSAtomicsCondition::JSAtomicsConditionPrint(std::ostream& os) {
JSObjectPrintHeader(os, *this, "JSAtomicsCondition");
Isolate* isolate = GetIsolateFromWritableObject(*this);
os << "\n - isolate: " << isolate;
if (isolate->is_shared()) os << " (shared)";
os << "\n - state: " << this->state();
JSObjectPrintBody(os, *this);
}
void JSWeakMap::JSWeakMapPrint(std::ostream& os) { void JSWeakMap::JSWeakMapPrint(std::ostream& os) {
JSObjectPrintHeader(os, *this, "JSWeakMap"); JSObjectPrintHeader(os, *this, "JSWeakMap");
os << "\n - table: " << Brief(table()); os << "\n - table: " << Brief(table());
......
...@@ -36,7 +36,6 @@ namespace internal { ...@@ -36,7 +36,6 @@ namespace internal {
V(JSFinalizationRegistry) \ V(JSFinalizationRegistry) \
V(JSFunction) \ V(JSFunction) \
V(JSObject) \ V(JSObject) \
V(JSSynchronizationPrimitive) \
V(JSTypedArray) \ V(JSTypedArray) \
V(WeakCell) \ V(WeakCell) \
V(JSWeakCollection) \ V(JSWeakCollection) \
......
...@@ -4665,10 +4665,8 @@ void Genesis::InitializeGlobal_harmony_struct() { ...@@ -4665,10 +4665,8 @@ void Genesis::InitializeGlobal_harmony_struct() {
DONT_ENUM); DONT_ENUM);
} }
// TODO(v8:12547): Make a single canonical copy of the Mutex and Condition
// maps.
{ // Atomics.Mutex { // Atomics.Mutex
// TODO(syg): Make a single canonical copy of the map.
Handle<String> mutex_str = Handle<String> mutex_str =
isolate()->factory()->InternalizeUtf8String("Mutex"); isolate()->factory()->InternalizeUtf8String("Mutex");
Handle<JSFunction> mutex_fun = CreateSharedObjectConstructor( Handle<JSFunction> mutex_fun = CreateSharedObjectConstructor(
...@@ -4687,27 +4685,6 @@ void Genesis::InitializeGlobal_harmony_struct() { ...@@ -4687,27 +4685,6 @@ void Genesis::InitializeGlobal_harmony_struct() {
SimpleInstallFunction(isolate(), mutex_fun, "tryLock", SimpleInstallFunction(isolate(), mutex_fun, "tryLock",
Builtin::kAtomicsMutexTryLock, 2, true); Builtin::kAtomicsMutexTryLock, 2, true);
} }
{ // Atomics.Condition
Handle<String> condition_str =
isolate()->factory()->InternalizeUtf8String("Condition");
Handle<JSFunction> condition_fun = CreateSharedObjectConstructor(
isolate(), condition_str, JS_ATOMICS_CONDITION_TYPE,
JSAtomicsCondition::kHeaderSize, TERMINAL_FAST_ELEMENTS_KIND,
Builtin::kAtomicsConditionConstructor);
condition_fun->shared().set_internal_formal_parameter_count(
JSParameterCount(0));
condition_fun->shared().set_length(0);
native_context()->set_js_atomics_condition_map(
condition_fun->initial_map());
JSObject::AddProperty(isolate(), isolate()->atomics_object(), condition_str,
condition_fun, DONT_ENUM);
SimpleInstallFunction(isolate(), condition_fun, "wait",
Builtin::kAtomicsConditionWait, 2, false);
SimpleInstallFunction(isolate(), condition_fun, "notify",
Builtin::kAtomicsConditionNotify, 2, false);
}
} }
void Genesis::InitializeGlobal_harmony_array_find_last() { void Genesis::InitializeGlobal_harmony_array_find_last() {
......
...@@ -178,7 +178,6 @@ enum ContextLookupFlags { ...@@ -178,7 +178,6 @@ enum ContextLookupFlags {
js_array_packed_double_elements_map) \ js_array_packed_double_elements_map) \
V(JS_ARRAY_HOLEY_DOUBLE_ELEMENTS_MAP_INDEX, Map, \ V(JS_ARRAY_HOLEY_DOUBLE_ELEMENTS_MAP_INDEX, Map, \
js_array_holey_double_elements_map) \ js_array_holey_double_elements_map) \
V(JS_ATOMICS_CONDITION_MAP, Map, js_atomics_condition_map) \
V(JS_ATOMICS_MUTEX_MAP, Map, js_atomics_mutex_map) \ V(JS_ATOMICS_MUTEX_MAP, Map, js_atomics_mutex_map) \
V(JS_MAP_FUN_INDEX, JSFunction, js_map_fun) \ V(JS_MAP_FUN_INDEX, JSFunction, js_map_fun) \
V(JS_MAP_MAP_INDEX, Map, js_map_map) \ V(JS_MAP_MAP_INDEX, Map, js_map_map) \
......
...@@ -19,15 +19,6 @@ namespace internal { ...@@ -19,15 +19,6 @@ namespace internal {
#include "torque-generated/src/objects/js-atomics-synchronization-tq-inl.inc" #include "torque-generated/src/objects/js-atomics-synchronization-tq-inl.inc"
TQ_OBJECT_CONSTRUCTORS_IMPL(JSSynchronizationPrimitive)
std::atomic<JSSynchronizationPrimitive::StateT>*
JSSynchronizationPrimitive::AtomicStatePtr(int offset) {
StateT* state_ptr = reinterpret_cast<StateT*>(field_address(offset));
DCHECK(IsAligned(reinterpret_cast<uintptr_t>(state_ptr), sizeof(StateT)));
return base::AsAtomicPtr(state_ptr);
}
TQ_OBJECT_CONSTRUCTORS_IMPL(JSAtomicsMutex) TQ_OBJECT_CONSTRUCTORS_IMPL(JSAtomicsMutex)
CAST_ACCESSOR(JSAtomicsMutex) CAST_ACCESSOR(JSAtomicsMutex)
...@@ -121,7 +112,9 @@ void JSAtomicsMutex::ClearOwnerThread() { ...@@ -121,7 +112,9 @@ void JSAtomicsMutex::ClearOwnerThread() {
} }
std::atomic<JSAtomicsMutex::StateT>* JSAtomicsMutex::AtomicStatePtr() { std::atomic<JSAtomicsMutex::StateT>* JSAtomicsMutex::AtomicStatePtr() {
return JSSynchronizationPrimitive::AtomicStatePtr(kStateOffset); StateT* state_ptr = reinterpret_cast<StateT*>(field_address(kStateOffset));
DCHECK(IsAligned(reinterpret_cast<uintptr_t>(state_ptr), sizeof(StateT)));
return base::AsAtomicPtr(state_ptr);
} }
std::atomic<int32_t>* JSAtomicsMutex::AtomicOwnerThreadIdPtr() { std::atomic<int32_t>* JSAtomicsMutex::AtomicOwnerThreadIdPtr() {
...@@ -130,14 +123,6 @@ std::atomic<int32_t>* JSAtomicsMutex::AtomicOwnerThreadIdPtr() { ...@@ -130,14 +123,6 @@ std::atomic<int32_t>* JSAtomicsMutex::AtomicOwnerThreadIdPtr() {
return base::AsAtomicPtr(owner_thread_id_ptr); return base::AsAtomicPtr(owner_thread_id_ptr);
} }
TQ_OBJECT_CONSTRUCTORS_IMPL(JSAtomicsCondition)
CAST_ACCESSOR(JSAtomicsCondition)
std::atomic<JSAtomicsCondition::StateT>* JSAtomicsCondition::AtomicStatePtr() {
return JSSynchronizationPrimitive::AtomicStatePtr(kStateOffset);
}
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
......
This diff is collapsed.
...@@ -7,7 +7,6 @@ ...@@ -7,7 +7,6 @@
#include <atomic> #include <atomic>
#include "src/base/platform/time.h"
#include "src/execution/thread-id.h" #include "src/execution/thread-id.h"
#include "src/objects/js-objects.h" #include "src/objects/js-objects.h"
...@@ -23,33 +22,6 @@ namespace detail { ...@@ -23,33 +22,6 @@ namespace detail {
class WaiterQueueNode; class WaiterQueueNode;
} // namespace detail } // namespace detail
// Base class for JSAtomicsMutex and JSAtomicsCondition
class JSSynchronizationPrimitive
: public TorqueGeneratedJSSynchronizationPrimitive<
JSSynchronizationPrimitive, JSObject> {
public:
// Synchronization only store raw data as state.
static constexpr int kEndOfTaggedFieldsOffset = JSObject::kHeaderSize;
class BodyDescriptor;
TQ_OBJECT_CONSTRUCTORS(JSSynchronizationPrimitive)
protected:
#ifdef V8_SANDBOXED_EXTERNAL_POINTERS
using StateT = uint32_t;
static_assert(sizeof(StateT) == kExternalPointerSlotSize);
#else
using StateT = uintptr_t;
#endif
// The reason this takes the offset explicitly is because Mutex and Condition
// have different offsets. Ideally the state field would be a field in
// JSSynchronizationPrimitive itself, but due to natural alignment
// requirements for atomic access, it lives in the subclasses to avoid wasting
// extra bytes on padding.
inline std::atomic<StateT>* AtomicStatePtr(int offset);
};
// A non-recursive mutex that is exposed to JS. // A non-recursive mutex that is exposed to JS.
// //
// It has the following properties: // It has the following properties:
...@@ -69,8 +41,7 @@ class JSSynchronizationPrimitive ...@@ -69,8 +41,7 @@ class JSSynchronizationPrimitive
// it implements a futex in userland. The algorithm is inspired by WebKit's // it implements a futex in userland. The algorithm is inspired by WebKit's
// ParkingLot. // ParkingLot.
class JSAtomicsMutex class JSAtomicsMutex
: public TorqueGeneratedJSAtomicsMutex<JSAtomicsMutex, : public TorqueGeneratedJSAtomicsMutex<JSAtomicsMutex, JSObject> {
JSSynchronizationPrimitive> {
public: public:
// A non-copyable wrapper class that provides an RAII-style mechanism for // A non-copyable wrapper class that provides an RAII-style mechanism for
// owning the JSAtomicsMutex. // owning the JSAtomicsMutex.
...@@ -125,6 +96,9 @@ class JSAtomicsMutex ...@@ -125,6 +96,9 @@ class JSAtomicsMutex
inline bool IsHeld(); inline bool IsHeld();
inline bool IsCurrentThreadOwner(); inline bool IsCurrentThreadOwner();
static constexpr int kEndOfTaggedFieldsOffset = JSObject::kHeaderSize;
class BodyDescriptor;
TQ_OBJECT_CONSTRUCTORS(JSAtomicsMutex) TQ_OBJECT_CONSTRUCTORS(JSAtomicsMutex)
private: private:
...@@ -136,6 +110,13 @@ class JSAtomicsMutex ...@@ -136,6 +110,13 @@ class JSAtomicsMutex
static constexpr int kIsWaiterQueueLockedBit = 1 << 1; static constexpr int kIsWaiterQueueLockedBit = 1 << 1;
static constexpr int kLockBitsSize = 2; static constexpr int kLockBitsSize = 2;
#ifdef V8_SANDBOXED_EXTERNAL_POINTERS
using StateT = uint32_t;
static_assert(sizeof(StateT) == kExternalPointerSlotSize);
#else
using StateT = uintptr_t;
#endif
static constexpr StateT kUnlocked = 0; static constexpr StateT kUnlocked = 0;
static constexpr StateT kLockedUncontended = 1; static constexpr StateT kLockedUncontended = 1;
...@@ -157,70 +138,12 @@ class JSAtomicsMutex ...@@ -157,70 +138,12 @@ class JSAtomicsMutex
V8_EXPORT_PRIVATE void UnlockSlowPath(Isolate* requester, V8_EXPORT_PRIVATE void UnlockSlowPath(Isolate* requester,
std::atomic<StateT>* state); std::atomic<StateT>* state);
using TorqueGeneratedJSAtomicsMutex<JSAtomicsMutex, JSObject>::state;
using TorqueGeneratedJSAtomicsMutex<JSAtomicsMutex, JSObject>::set_state;
using TorqueGeneratedJSAtomicsMutex<JSAtomicsMutex, using TorqueGeneratedJSAtomicsMutex<JSAtomicsMutex,
JSSynchronizationPrimitive>::state; JSObject>::owner_thread_id;
using TorqueGeneratedJSAtomicsMutex<JSAtomicsMutex, using TorqueGeneratedJSAtomicsMutex<JSAtomicsMutex,
JSSynchronizationPrimitive>::set_state; JSObject>::set_owner_thread_id;
using TorqueGeneratedJSAtomicsMutex<
JSAtomicsMutex, JSSynchronizationPrimitive>::owner_thread_id;
using TorqueGeneratedJSAtomicsMutex<
JSAtomicsMutex, JSSynchronizationPrimitive>::set_owner_thread_id;
};
// A condition variable that is exposed to JS.
//
// It has the following properties:
// - Slim: 4-8 bytes. Lock state is 4 bytes when
// V8_SANDBOXED_EXTERNAL_POINTERS, and sizeof(void*) otherwise.
// - Moving GC safe. It uses an index into the shared Isolate's external
// pointer table to store a queue of sleeping threads.
// - Parks the main thread LocalHeap when waiting. Unparks the main thread
// LocalHeap after waking up.
//
// This condition variable manages its own queue of waiting threads, like
// JSAtomicsMutex. The algorithm is inspired by WebKit's ParkingLot.
class JSAtomicsCondition
: public TorqueGeneratedJSAtomicsCondition<JSAtomicsCondition,
JSSynchronizationPrimitive> {
public:
DECL_CAST(JSAtomicsCondition)
DECL_PRINTER(JSAtomicsCondition)
EXPORT_DECL_VERIFIER(JSAtomicsCondition)
V8_EXPORT_PRIVATE static Handle<JSAtomicsCondition> Create(Isolate* isolate);
V8_EXPORT_PRIVATE static bool WaitFor(
Isolate* requester, Handle<JSAtomicsCondition> cv,
Handle<JSAtomicsMutex> mutex, base::Optional<base::TimeDelta> timeout);
static constexpr uint32_t kAllWaiters = UINT32_MAX;
// Notify {count} waiters. Returns the number of waiters woken up.
V8_EXPORT_PRIVATE uint32_t Notify(Isolate* requester, uint32_t count);
Object NumWaitersForTesting(Isolate* isolate);
TQ_OBJECT_CONSTRUCTORS(JSAtomicsCondition)
private:
friend class detail::WaiterQueueNode;
// There is 1 lock bit: whether the waiter queue is locked.
static constexpr int kIsWaiterQueueLockedBit = 1 << 0;
static constexpr int kLockBitsSize = 1;
static constexpr StateT kEmptyState = 0;
static constexpr StateT kLockBitsMask = (1 << kLockBitsSize) - 1;
static constexpr StateT kWaiterQueueHeadMask = ~kLockBitsMask;
inline std::atomic<StateT>* AtomicStatePtr();
bool TryLockWaiterQueueExplicit(std::atomic<StateT>* state, StateT& expected);
using TorqueGeneratedJSAtomicsCondition<JSAtomicsCondition,
JSSynchronizationPrimitive>::state;
using TorqueGeneratedJSAtomicsCondition<
JSAtomicsCondition, JSSynchronizationPrimitive>::set_state;
}; };
} // namespace internal } // namespace internal
......
...@@ -2,13 +2,9 @@ ...@@ -2,13 +2,9 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
@abstract extern class JSAtomicsMutex extends JSObject {
extern class JSSynchronizationPrimitive extends JSObject {
}
extern class JSAtomicsMutex extends JSSynchronizationPrimitive {
// owner_thread_id must come first to ensure that the state field is // owner_thread_id must come first to ensure that the state field is
// naturally aligned. // uintptr-aligned.
@if(V8_NOT_SANDBOXED_EXTERNAL_POINTERS_AND_TAGGED_SIZE_8_BYTES) @if(V8_NOT_SANDBOXED_EXTERNAL_POINTERS_AND_TAGGED_SIZE_8_BYTES)
owner_thread_id: intptr; owner_thread_id: intptr;
@ifnot(V8_NOT_SANDBOXED_EXTERNAL_POINTERS_AND_TAGGED_SIZE_8_BYTES) @ifnot(V8_NOT_SANDBOXED_EXTERNAL_POINTERS_AND_TAGGED_SIZE_8_BYTES)
...@@ -17,13 +13,3 @@ extern class JSAtomicsMutex extends JSSynchronizationPrimitive { ...@@ -17,13 +13,3 @@ extern class JSAtomicsMutex extends JSSynchronizationPrimitive {
@if(V8_SANDBOXED_EXTERNAL_POINTERS) state: uint32; @if(V8_SANDBOXED_EXTERNAL_POINTERS) state: uint32;
@ifnot(V8_SANDBOXED_EXTERNAL_POINTERS) state: uintptr; @ifnot(V8_SANDBOXED_EXTERNAL_POINTERS) state: uintptr;
} }
extern class JSAtomicsCondition extends JSSynchronizationPrimitive {
// Padding to ensure that the state field is naturally aligned.
@if(V8_NOT_SANDBOXED_EXTERNAL_POINTERS_AND_TAGGED_SIZE_8_BYTES) padding: void;
@ifnot(V8_NOT_SANDBOXED_EXTERNAL_POINTERS_AND_TAGGED_SIZE_8_BYTES)
padding: uint32;
@if(V8_SANDBOXED_EXTERNAL_POINTERS) state: uint32;
@ifnot(V8_SANDBOXED_EXTERNAL_POINTERS) state: uintptr;
}
...@@ -2461,8 +2461,6 @@ int JSObject::GetHeaderSize(InstanceType type, ...@@ -2461,8 +2461,6 @@ int JSObject::GetHeaderSize(InstanceType type,
return JSSharedStruct::kHeaderSize; return JSSharedStruct::kHeaderSize;
case JS_ATOMICS_MUTEX_TYPE: case JS_ATOMICS_MUTEX_TYPE:
return JSAtomicsMutex::kHeaderSize; return JSAtomicsMutex::kHeaderSize;
case JS_ATOMICS_CONDITION_TYPE:
return JSAtomicsCondition::kHeaderSize;
case JS_TEMPORAL_CALENDAR_TYPE: case JS_TEMPORAL_CALENDAR_TYPE:
return JSTemporalCalendar::kHeaderSize; return JSTemporalCalendar::kHeaderSize;
case JS_TEMPORAL_DURATION_TYPE: case JS_TEMPORAL_DURATION_TYPE:
......
...@@ -286,6 +286,7 @@ VisitorId Map::GetVisitorId(Map map) { ...@@ -286,6 +286,7 @@ VisitorId Map::GetVisitorId(Map map) {
case JS_SHADOW_REALM_TYPE: case JS_SHADOW_REALM_TYPE:
case JS_SHARED_ARRAY_TYPE: case JS_SHARED_ARRAY_TYPE:
case JS_SHARED_STRUCT_TYPE: case JS_SHARED_STRUCT_TYPE:
case JS_ATOMICS_MUTEX_TYPE:
case JS_STRING_ITERATOR_PROTOTYPE_TYPE: case JS_STRING_ITERATOR_PROTOTYPE_TYPE:
case JS_STRING_ITERATOR_TYPE: case JS_STRING_ITERATOR_TYPE:
case JS_TEMPORAL_CALENDAR_TYPE: case JS_TEMPORAL_CALENDAR_TYPE:
...@@ -324,7 +325,6 @@ VisitorId Map::GetVisitorId(Map map) { ...@@ -324,7 +325,6 @@ VisitorId Map::GetVisitorId(Map map) {
#endif // V8_ENABLE_WEBASSEMBLY #endif // V8_ENABLE_WEBASSEMBLY
case JS_BOUND_FUNCTION_TYPE: case JS_BOUND_FUNCTION_TYPE:
case JS_WRAPPED_FUNCTION_TYPE: { case JS_WRAPPED_FUNCTION_TYPE: {
// Is GetEmbedderFieldCount(map) > 0 for Atomics.Mutex?
const bool has_raw_data_fields = const bool has_raw_data_fields =
COMPRESS_POINTERS_BOOL && JSObject::GetEmbedderFieldCount(map) > 0; COMPRESS_POINTERS_BOOL && JSObject::GetEmbedderFieldCount(map) > 0;
return has_raw_data_fields ? kVisitJSObject : kVisitJSObjectFast; return has_raw_data_fields ? kVisitJSObject : kVisitJSObjectFast;
...@@ -344,10 +344,6 @@ VisitorId Map::GetVisitorId(Map map) { ...@@ -344,10 +344,6 @@ VisitorId Map::GetVisitorId(Map map) {
case JS_FINALIZATION_REGISTRY_TYPE: case JS_FINALIZATION_REGISTRY_TYPE:
return kVisitJSFinalizationRegistry; return kVisitJSFinalizationRegistry;
case JS_ATOMICS_MUTEX_TYPE:
case JS_ATOMICS_CONDITION_TYPE:
return kVisitJSSynchronizationPrimitive;
case FILLER_TYPE: case FILLER_TYPE:
case FOREIGN_TYPE: case FOREIGN_TYPE:
case HEAP_NUMBER_TYPE: case HEAP_NUMBER_TYPE:
......
...@@ -53,7 +53,6 @@ enum InstanceType : uint16_t; ...@@ -53,7 +53,6 @@ enum InstanceType : uint16_t;
V(JSFunction) \ V(JSFunction) \
V(JSObject) \ V(JSObject) \
V(JSObjectFast) \ V(JSObjectFast) \
V(JSSynchronizationPrimitive) \
V(JSTypedArray) \ V(JSTypedArray) \
V(JSWeakRef) \ V(JSWeakRef) \
V(JSWeakCollection) \ V(JSWeakCollection) \
......
...@@ -132,7 +132,6 @@ class ZoneForwardList; ...@@ -132,7 +132,6 @@ class ZoneForwardList;
V(JSAsyncFromSyncIterator) \ V(JSAsyncFromSyncIterator) \
V(JSAsyncFunctionObject) \ V(JSAsyncFunctionObject) \
V(JSAsyncGeneratorObject) \ V(JSAsyncGeneratorObject) \
V(JSAtomicsCondition) \
V(JSAtomicsMutex) \ V(JSAtomicsMutex) \
V(JSBoundFunction) \ V(JSBoundFunction) \
V(JSCollection) \ V(JSCollection) \
...@@ -168,7 +167,6 @@ class ZoneForwardList; ...@@ -168,7 +167,6 @@ class ZoneForwardList;
V(JSSharedStruct) \ V(JSSharedStruct) \
V(JSSpecialObject) \ V(JSSpecialObject) \
V(JSStringIterator) \ V(JSStringIterator) \
V(JSSynchronizationPrimitive) \
V(JSTemporalCalendar) \ V(JSTemporalCalendar) \
V(JSTemporalDuration) \ V(JSTemporalDuration) \
V(JSTemporalInstant) \ V(JSTemporalInstant) \
......
...@@ -21,7 +21,7 @@ ...@@ -21,7 +21,7 @@
#include "src/objects/free-space-inl.h" #include "src/objects/free-space-inl.h"
#include "src/objects/hash-table.h" #include "src/objects/hash-table.h"
#include "src/objects/heap-number.h" #include "src/objects/heap-number.h"
#include "src/objects/js-atomics-synchronization-inl.h" #include "src/objects/js-atomics-synchronization.h"
#include "src/objects/js-collection.h" #include "src/objects/js-collection.h"
#include "src/objects/js-weak-refs.h" #include "src/objects/js-weak-refs.h"
#include "src/objects/literal-objects.h" #include "src/objects/literal-objects.h"
...@@ -668,8 +668,7 @@ class JSWeakCollection::BodyDescriptorImpl final : public BodyDescriptorBase { ...@@ -668,8 +668,7 @@ class JSWeakCollection::BodyDescriptorImpl final : public BodyDescriptorBase {
} }
}; };
class JSSynchronizationPrimitive::BodyDescriptor final class JSAtomicsMutex::BodyDescriptor final : public BodyDescriptorBase {
: public BodyDescriptorBase {
public: public:
static bool IsValidSlot(Map map, HeapObject obj, int offset) { static bool IsValidSlot(Map map, HeapObject obj, int offset) {
if (offset < kEndOfTaggedFieldsOffset) return true; if (offset < kEndOfTaggedFieldsOffset) return true;
...@@ -681,6 +680,7 @@ class JSSynchronizationPrimitive::BodyDescriptor final ...@@ -681,6 +680,7 @@ class JSSynchronizationPrimitive::BodyDescriptor final
static inline void IterateBody(Map map, HeapObject obj, int object_size, static inline void IterateBody(Map map, HeapObject obj, int object_size,
ObjectVisitor* v) { ObjectVisitor* v) {
IteratePointers(obj, kPropertiesOrHashOffset, kEndOfTaggedFieldsOffset, v); IteratePointers(obj, kPropertiesOrHashOffset, kEndOfTaggedFieldsOffset, v);
IterateJSObjectBodyImpl(map, obj, kHeaderSize, object_size, v);
} }
static inline int SizeOf(Map map, HeapObject object) { static inline int SizeOf(Map map, HeapObject object) {
...@@ -1289,8 +1289,7 @@ auto BodyDescriptorApply(InstanceType type, Args&&... args) { ...@@ -1289,8 +1289,7 @@ auto BodyDescriptorApply(InstanceType type, Args&&... args) {
case JS_PROXY_TYPE: case JS_PROXY_TYPE:
return CALL_APPLY(JSProxy); return CALL_APPLY(JSProxy);
case JS_ATOMICS_MUTEX_TYPE: case JS_ATOMICS_MUTEX_TYPE:
case JS_ATOMICS_CONDITION_TYPE: return CALL_APPLY(JSAtomicsMutex);
return CALL_APPLY(JSSynchronizationPrimitive);
case FOREIGN_TYPE: case FOREIGN_TYPE:
return CALL_APPLY(Foreign); return CALL_APPLY(Foreign);
case MAP_TYPE: case MAP_TYPE:
......
...@@ -1183,7 +1183,6 @@ bool Object::IsShared() const { ...@@ -1183,7 +1183,6 @@ bool Object::IsShared() const {
case JS_SHARED_ARRAY_TYPE: case JS_SHARED_ARRAY_TYPE:
case JS_SHARED_STRUCT_TYPE: case JS_SHARED_STRUCT_TYPE:
case JS_ATOMICS_MUTEX_TYPE: case JS_ATOMICS_MUTEX_TYPE:
case JS_ATOMICS_CONDITION_TYPE:
DCHECK(object.InSharedHeap()); DCHECK(object.InSharedHeap());
return true; return true;
case INTERNALIZED_STRING_TYPE: case INTERNALIZED_STRING_TYPE:
......
...@@ -615,7 +615,6 @@ Maybe<bool> ValueSerializer::WriteJSReceiver(Handle<JSReceiver> receiver) { ...@@ -615,7 +615,6 @@ Maybe<bool> ValueSerializer::WriteJSReceiver(Handle<JSReceiver> receiver) {
case JS_SHARED_STRUCT_TYPE: case JS_SHARED_STRUCT_TYPE:
return WriteJSSharedStruct(Handle<JSSharedStruct>::cast(receiver)); return WriteJSSharedStruct(Handle<JSSharedStruct>::cast(receiver));
case JS_ATOMICS_MUTEX_TYPE: case JS_ATOMICS_MUTEX_TYPE:
case JS_ATOMICS_CONDITION_TYPE:
return WriteSharedObject(receiver); return WriteSharedObject(receiver);
#if V8_ENABLE_WEBASSEMBLY #if V8_ENABLE_WEBASSEMBLY
case WASM_MODULE_OBJECT_TYPE: case WASM_MODULE_OBJECT_TYPE:
......
...@@ -23,7 +23,6 @@ ...@@ -23,7 +23,6 @@
#include "src/heap/heap-inl.h" // For ToBoolean. TODO(jkummerow): Drop. #include "src/heap/heap-inl.h" // For ToBoolean. TODO(jkummerow): Drop.
#include "src/heap/heap-write-barrier-inl.h" #include "src/heap/heap-write-barrier-inl.h"
#include "src/ic/stub-cache.h" #include "src/ic/stub-cache.h"
#include "src/objects/js-atomics-synchronization-inl.h"
#include "src/objects/js-function-inl.h" #include "src/objects/js-function-inl.h"
#include "src/objects/js-regexp-inl.h" #include "src/objects/js-regexp-inl.h"
#include "src/objects/smi.h" #include "src/objects/smi.h"
...@@ -1670,12 +1669,5 @@ RUNTIME_FUNCTION(Runtime_SharedGC) { ...@@ -1670,12 +1669,5 @@ RUNTIME_FUNCTION(Runtime_SharedGC) {
return ReadOnlyRoots(isolate).undefined_value(); return ReadOnlyRoots(isolate).undefined_value();
} }
RUNTIME_FUNCTION(Runtime_AtomicsConditionNumWaitersForTesting) {
HandleScope scope(isolate);
DCHECK_EQ(1, args.length());
Handle<JSAtomicsCondition> cv = args.at<JSAtomicsCondition>(0);
return cv->NumWaitersForTesting(isolate);
}
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
...@@ -106,10 +106,8 @@ class V8_NODISCARD ClearThreadInWasmScope { ...@@ -106,10 +106,8 @@ class V8_NODISCARD ClearThreadInWasmScope {
Isolate* isolate_; Isolate* isolate_;
}; };
Object ThrowWasmError(Isolate* isolate, MessageTemplate message, Object ThrowWasmError(Isolate* isolate, MessageTemplate message) {
Handle<Object> arg0 = Handle<Object>()) { Handle<JSObject> error_obj = isolate->factory()->NewWasmRuntimeError(message);
Handle<JSObject> error_obj =
isolate->factory()->NewWasmRuntimeError(message, arg0);
JSObject::AddProperty(isolate, error_obj, JSObject::AddProperty(isolate, error_obj,
isolate->factory()->wasm_uncatchable_symbol(), isolate->factory()->wasm_uncatchable_symbol(),
isolate->factory()->true_value(), NONE); isolate->factory()->true_value(), NONE);
...@@ -386,9 +384,7 @@ RUNTIME_FUNCTION(Runtime_WasmI32AtomicWait) { ...@@ -386,9 +384,7 @@ RUNTIME_FUNCTION(Runtime_WasmI32AtomicWait) {
// Trap if memory is not shared, or wait is not allowed on the isolate // Trap if memory is not shared, or wait is not allowed on the isolate
if (!array_buffer->is_shared() || !isolate->allow_atomics_wait()) { if (!array_buffer->is_shared() || !isolate->allow_atomics_wait()) {
return ThrowWasmError( return ThrowWasmError(isolate, MessageTemplate::kAtomicsWaitNotAllowed);
isolate, MessageTemplate::kAtomicsOperationNotAllowed,
isolate->factory()->NewStringFromAsciiChecked("Atomics.wait"));
} }
return FutexEmulation::WaitWasm32(isolate, array_buffer, offset, return FutexEmulation::WaitWasm32(isolate, array_buffer, offset,
expected_value, timeout_ns.AsInt64()); expected_value, timeout_ns.AsInt64());
...@@ -411,9 +407,7 @@ RUNTIME_FUNCTION(Runtime_WasmI64AtomicWait) { ...@@ -411,9 +407,7 @@ RUNTIME_FUNCTION(Runtime_WasmI64AtomicWait) {
// Trap if memory is not shared, or if wait is not allowed on the isolate // Trap if memory is not shared, or if wait is not allowed on the isolate
if (!array_buffer->is_shared() || !isolate->allow_atomics_wait()) { if (!array_buffer->is_shared() || !isolate->allow_atomics_wait()) {
return ThrowWasmError( return ThrowWasmError(isolate, MessageTemplate::kAtomicsWaitNotAllowed);
isolate, MessageTemplate::kAtomicsOperationNotAllowed,
isolate->factory()->NewStringFromAsciiChecked("Atomics.wait"));
} }
return FutexEmulation::WaitWasm64(isolate, array_buffer, offset, return FutexEmulation::WaitWasm64(isolate, array_buffer, offset,
expected_value.AsInt64(), expected_value.AsInt64(),
......
...@@ -65,8 +65,7 @@ namespace internal { ...@@ -65,8 +65,7 @@ namespace internal {
F(SetAllowAtomicsWait, 1, 1) \ F(SetAllowAtomicsWait, 1, 1) \
F(AtomicsLoadSharedStructOrArray, 2, 1) \ F(AtomicsLoadSharedStructOrArray, 2, 1) \
F(AtomicsStoreSharedStructOrArray, 3, 1) \ F(AtomicsStoreSharedStructOrArray, 3, 1) \
F(AtomicsExchangeSharedStructOrArray, 3, 1) \ F(AtomicsExchangeSharedStructOrArray, 3, 1)
F(AtomicsConditionNumWaitersForTesting, 1, 1)
#define FOR_EACH_INTRINSIC_BIGINT(F, I) \ #define FOR_EACH_INTRINSIC_BIGINT(F, I) \
F(BigIntBinaryOp, 3, 1) \ F(BigIntBinaryOp, 3, 1) \
......
// Copyright 2022 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.
//
// Flags: --harmony-struct --allow-natives-syntax
"use strict";
if (this.Worker) {
(function TestWait() {
let workerScript =
`onmessage = function(msg) {
let mutex = msg.mutex;
let cv = msg.cv;
let res = Atomics.Mutex.lock(mutex, function() {
return Atomics.Condition.wait(cv, mutex);
});
postMessage(res);
};`;
let mutex = new Atomics.Mutex;
let cv = new Atomics.Condition;
let msg = {mutex, cv};
let worker1 = new Worker(workerScript, { type: 'string' });
let worker2 = new Worker(workerScript, { type: 'string' });
worker1.postMessage(msg);
worker2.postMessage(msg);
// Spin until both workers are waiting.
while (%AtomicsConditionNumWaitersForTesting(cv) != 2) {}
assertEquals(2, Atomics.Condition.notify(cv, 2));
assertEquals(true, worker1.getMessage());
assertEquals(true, worker2.getMessage());
worker1.terminate();
worker2.terminate();
})();
}
// Copyright 2022 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.
//
// Flags: --harmony-struct --allow-natives-syntax
let mutex = new Atomics.Mutex;
let cv = new Atomics.Condition;
(function TestConditionWaitNotAllowed() {
assertThrows(() => {
Atomics.Mutex.lock(mutex, () => {
%SetAllowAtomicsWait(false);
Atomics.Condition.wait(cv, mutex);
});
});
%SetAllowAtomicsWait(true);
})();
(function TestConditionMutexNotHeld() {
// Cannot wait on a mutex not owned by the current thread.
assertThrows(() => {
Atomics.Condition.wait(cv, mutex);
});
})();
(function TestConditionNoWaiters() {
// Notify returns number of threads woken up.
assertEquals(0, Atomics.Condition.notify(cv));
})();
(function TestConditionWaitTimeout() {
Atomics.Mutex.lock(mutex, () => {
assertEquals(false, Atomics.Condition.wait(cv, mutex, 100));
});
})();
...@@ -413,7 +413,7 @@ v8_source_set("unittests_sources") { ...@@ -413,7 +413,7 @@ v8_source_set("unittests_sources") {
"interpreter/source-position-matcher.cc", "interpreter/source-position-matcher.cc",
"interpreter/source-position-matcher.h", "interpreter/source-position-matcher.h",
"interpreter/source-positions-unittest.cc", "interpreter/source-positions-unittest.cc",
"js-atomics/js-atomics-synchronization-primitive-unittest.cc", "js-atomics/js-atomics-mutex-unittest.cc",
"libplatform/default-job-unittest.cc", "libplatform/default-job-unittest.cc",
"libplatform/default-platform-unittest.cc", "libplatform/default-platform-unittest.cc",
"libplatform/default-worker-threads-task-runner-unittest.cc", "libplatform/default-worker-threads-task-runner-unittest.cc",
......
...@@ -13,7 +13,6 @@ namespace v8 { ...@@ -13,7 +13,6 @@ namespace v8 {
namespace internal { namespace internal {
using JSAtomicsMutexTest = TestWithSharedIsolate; using JSAtomicsMutexTest = TestWithSharedIsolate;
using JSAtomicsConditionTest = TestWithSharedIsolate;
namespace { namespace {
...@@ -37,26 +36,13 @@ class ClientIsolateWithContextWrapper final { ...@@ -37,26 +36,13 @@ class ClientIsolateWithContextWrapper final {
v8::Context::Scope context_scope_; v8::Context::Scope context_scope_;
}; };
class ParkingThread : public v8::base::Thread { class LockingThread final : public v8::base::Thread {
public:
explicit ParkingThread(const Options& options) : v8::base::Thread(options) {}
void ParkedJoin(const ParkedScope& scope) {
USE(scope);
Join();
}
private:
using base::Thread::Join;
};
class LockingThread final : public ParkingThread {
public: public:
LockingThread(v8::Isolate* shared_isolate, Handle<JSAtomicsMutex> mutex, LockingThread(v8::Isolate* shared_isolate, Handle<JSAtomicsMutex> mutex,
ParkingSemaphore* sema_ready, ParkingSemaphore* sema_ready,
ParkingSemaphore* sema_execute_start, ParkingSemaphore* sema_execute_start,
ParkingSemaphore* sema_execute_complete) ParkingSemaphore* sema_execute_complete)
: ParkingThread(Options("LockingThread")), : Thread(Options("ThreadWithAtomicsMutex")),
shared_isolate_(shared_isolate), shared_isolate_(shared_isolate),
mutex_(mutex), mutex_(mutex),
sema_ready_(sema_ready), sema_ready_(sema_ready),
...@@ -80,7 +66,14 @@ class LockingThread final : public ParkingThread { ...@@ -80,7 +66,14 @@ class LockingThread final : public ParkingThread {
sema_execute_complete_->Signal(); sema_execute_complete_->Signal();
} }
private: void ParkedJoin(const ParkedScope& scope) {
USE(scope);
Join();
}
protected:
using base::Thread::Join;
v8::Isolate* shared_isolate_; v8::Isolate* shared_isolate_;
Handle<JSAtomicsMutex> mutex_; Handle<JSAtomicsMutex> mutex_;
ParkingSemaphore* sema_ready_; ParkingSemaphore* sema_ready_;
...@@ -132,112 +125,5 @@ TEST_F(JSAtomicsMutexTest, Contention) { ...@@ -132,112 +125,5 @@ TEST_F(JSAtomicsMutexTest, Contention) {
EXPECT_FALSE(contended_mutex->IsHeld()); EXPECT_FALSE(contended_mutex->IsHeld());
} }
namespace {
class WaitOnConditionThread final : public ParkingThread {
public:
WaitOnConditionThread(v8::Isolate* shared_isolate,
Handle<JSAtomicsMutex> mutex,
Handle<JSAtomicsCondition> condition,
uint32_t* waiting_threads_count,
ParkingSemaphore* sema_ready,
ParkingSemaphore* sema_execute_complete)
: ParkingThread(Options("WaitOnConditionThread")),
shared_isolate_(shared_isolate),
mutex_(mutex),
condition_(condition),
waiting_threads_count_(waiting_threads_count),
sema_ready_(sema_ready),
sema_execute_complete_(sema_execute_complete) {}
void Run() override {
ClientIsolateWithContextWrapper client_isolate_wrapper(shared_isolate_);
Isolate* isolate = client_isolate_wrapper.isolate();
sema_ready_->Signal();
HandleScope scope(isolate);
JSAtomicsMutex::Lock(isolate, mutex_);
while (keep_waiting) {
(*waiting_threads_count_)++;
EXPECT_TRUE(JSAtomicsCondition::WaitFor(isolate, condition_, mutex_,
base::nullopt));
(*waiting_threads_count_)--;
}
mutex_->Unlock(isolate);
sema_execute_complete_->Signal();
}
bool keep_waiting = true;
private:
v8::Isolate* shared_isolate_;
Handle<JSAtomicsMutex> mutex_;
Handle<JSAtomicsCondition> condition_;
uint32_t* waiting_threads_count_;
ParkingSemaphore* sema_ready_;
ParkingSemaphore* sema_execute_complete_;
};
} // namespace
TEST_F(JSAtomicsConditionTest, NotifyAll) {
if (!IsJSSharedMemorySupported()) return;
FLAG_harmony_struct = true;
v8::Isolate* shared_isolate = v8_isolate();
ClientIsolateWithContextWrapper client_isolate_wrapper(shared_isolate);
Isolate* client_isolate = client_isolate_wrapper.isolate();
constexpr uint32_t kThreads = 32;
Handle<JSAtomicsMutex> mutex = JSAtomicsMutex::Create(client_isolate);
Handle<JSAtomicsCondition> condition =
JSAtomicsCondition::Create(client_isolate);
uint32_t waiting_threads_count = 0;
ParkingSemaphore sema_ready(0);
ParkingSemaphore sema_execute_complete(0);
std::vector<std::unique_ptr<WaitOnConditionThread>> threads;
for (uint32_t i = 0; i < kThreads; i++) {
auto thread = std::make_unique<WaitOnConditionThread>(
shared_isolate, mutex, condition, &waiting_threads_count, &sema_ready,
&sema_execute_complete);
CHECK(thread->Start());
threads.push_back(std::move(thread));
}
LocalIsolate* local_isolate = client_isolate->main_thread_local_isolate();
for (uint32_t i = 0; i < kThreads; i++) {
sema_ready.ParkedWait(local_isolate);
}
// Wait until all threads are waiting on the condition.
for (;;) {
JSAtomicsMutex::LockGuard lock_guard(client_isolate, mutex);
uint32_t count = waiting_threads_count;
if (count == kThreads) break;
}
// Wake all the threads up.
for (uint32_t i = 0; i < kThreads; i++) {
threads[i]->keep_waiting = false;
}
EXPECT_EQ(kThreads,
condition->Notify(client_isolate, JSAtomicsCondition::kAllWaiters));
for (uint32_t i = 0; i < kThreads; i++) {
sema_execute_complete.ParkedWait(local_isolate);
}
ParkedScope parked(local_isolate);
for (auto& thread : threads) {
thread->ParkedJoin(parked);
}
EXPECT_EQ(0U, waiting_threads_count);
EXPECT_FALSE(mutex->IsHeld());
}
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
...@@ -224,58 +224,57 @@ INSTANCE_TYPES = { ...@@ -224,58 +224,57 @@ INSTANCE_TYPES = {
2098: "JS_ASYNC_GENERATOR_OBJECT_TYPE", 2098: "JS_ASYNC_GENERATOR_OBJECT_TYPE",
2099: "JS_MAP_TYPE", 2099: "JS_MAP_TYPE",
2100: "JS_SET_TYPE", 2100: "JS_SET_TYPE",
2101: "JS_ATOMICS_CONDITION_TYPE", 2101: "JS_WEAK_MAP_TYPE",
2102: "JS_ATOMICS_MUTEX_TYPE", 2102: "JS_WEAK_SET_TYPE",
2103: "JS_WEAK_MAP_TYPE", 2103: "JS_ARGUMENTS_OBJECT_TYPE",
2104: "JS_WEAK_SET_TYPE", 2104: "JS_ARRAY_TYPE",
2105: "JS_ARGUMENTS_OBJECT_TYPE", 2105: "JS_ARRAY_ITERATOR_TYPE",
2106: "JS_ARRAY_TYPE", 2106: "JS_ASYNC_FROM_SYNC_ITERATOR_TYPE",
2107: "JS_ARRAY_ITERATOR_TYPE", 2107: "JS_ATOMICS_MUTEX_TYPE",
2108: "JS_ASYNC_FROM_SYNC_ITERATOR_TYPE", 2108: "JS_COLLATOR_TYPE",
2109: "JS_COLLATOR_TYPE", 2109: "JS_CONTEXT_EXTENSION_OBJECT_TYPE",
2110: "JS_CONTEXT_EXTENSION_OBJECT_TYPE", 2110: "JS_DATE_TYPE",
2111: "JS_DATE_TYPE", 2111: "JS_DATE_TIME_FORMAT_TYPE",
2112: "JS_DATE_TIME_FORMAT_TYPE", 2112: "JS_DISPLAY_NAMES_TYPE",
2113: "JS_DISPLAY_NAMES_TYPE", 2113: "JS_ERROR_TYPE",
2114: "JS_ERROR_TYPE", 2114: "JS_EXTERNAL_OBJECT_TYPE",
2115: "JS_EXTERNAL_OBJECT_TYPE", 2115: "JS_FINALIZATION_REGISTRY_TYPE",
2116: "JS_FINALIZATION_REGISTRY_TYPE", 2116: "JS_LIST_FORMAT_TYPE",
2117: "JS_LIST_FORMAT_TYPE", 2117: "JS_LOCALE_TYPE",
2118: "JS_LOCALE_TYPE", 2118: "JS_MESSAGE_OBJECT_TYPE",
2119: "JS_MESSAGE_OBJECT_TYPE", 2119: "JS_NUMBER_FORMAT_TYPE",
2120: "JS_NUMBER_FORMAT_TYPE", 2120: "JS_PLURAL_RULES_TYPE",
2121: "JS_PLURAL_RULES_TYPE", 2121: "JS_REG_EXP_TYPE",
2122: "JS_REG_EXP_TYPE", 2122: "JS_REG_EXP_STRING_ITERATOR_TYPE",
2123: "JS_REG_EXP_STRING_ITERATOR_TYPE", 2123: "JS_RELATIVE_TIME_FORMAT_TYPE",
2124: "JS_RELATIVE_TIME_FORMAT_TYPE", 2124: "JS_SEGMENT_ITERATOR_TYPE",
2125: "JS_SEGMENT_ITERATOR_TYPE", 2125: "JS_SEGMENTER_TYPE",
2126: "JS_SEGMENTER_TYPE", 2126: "JS_SEGMENTS_TYPE",
2127: "JS_SEGMENTS_TYPE", 2127: "JS_SHADOW_REALM_TYPE",
2128: "JS_SHADOW_REALM_TYPE", 2128: "JS_SHARED_ARRAY_TYPE",
2129: "JS_SHARED_ARRAY_TYPE", 2129: "JS_SHARED_STRUCT_TYPE",
2130: "JS_SHARED_STRUCT_TYPE", 2130: "JS_STRING_ITERATOR_TYPE",
2131: "JS_STRING_ITERATOR_TYPE", 2131: "JS_TEMPORAL_CALENDAR_TYPE",
2132: "JS_TEMPORAL_CALENDAR_TYPE", 2132: "JS_TEMPORAL_DURATION_TYPE",
2133: "JS_TEMPORAL_DURATION_TYPE", 2133: "JS_TEMPORAL_INSTANT_TYPE",
2134: "JS_TEMPORAL_INSTANT_TYPE", 2134: "JS_TEMPORAL_PLAIN_DATE_TYPE",
2135: "JS_TEMPORAL_PLAIN_DATE_TYPE", 2135: "JS_TEMPORAL_PLAIN_DATE_TIME_TYPE",
2136: "JS_TEMPORAL_PLAIN_DATE_TIME_TYPE", 2136: "JS_TEMPORAL_PLAIN_MONTH_DAY_TYPE",
2137: "JS_TEMPORAL_PLAIN_MONTH_DAY_TYPE", 2137: "JS_TEMPORAL_PLAIN_TIME_TYPE",
2138: "JS_TEMPORAL_PLAIN_TIME_TYPE", 2138: "JS_TEMPORAL_PLAIN_YEAR_MONTH_TYPE",
2139: "JS_TEMPORAL_PLAIN_YEAR_MONTH_TYPE", 2139: "JS_TEMPORAL_TIME_ZONE_TYPE",
2140: "JS_TEMPORAL_TIME_ZONE_TYPE", 2140: "JS_TEMPORAL_ZONED_DATE_TIME_TYPE",
2141: "JS_TEMPORAL_ZONED_DATE_TIME_TYPE", 2141: "JS_V8_BREAK_ITERATOR_TYPE",
2142: "JS_V8_BREAK_ITERATOR_TYPE", 2142: "JS_WEAK_REF_TYPE",
2143: "JS_WEAK_REF_TYPE", 2143: "WASM_EXCEPTION_PACKAGE_TYPE",
2144: "WASM_EXCEPTION_PACKAGE_TYPE", 2144: "WASM_GLOBAL_OBJECT_TYPE",
2145: "WASM_GLOBAL_OBJECT_TYPE", 2145: "WASM_INSTANCE_OBJECT_TYPE",
2146: "WASM_INSTANCE_OBJECT_TYPE", 2146: "WASM_MEMORY_OBJECT_TYPE",
2147: "WASM_MEMORY_OBJECT_TYPE", 2147: "WASM_MODULE_OBJECT_TYPE",
2148: "WASM_MODULE_OBJECT_TYPE", 2148: "WASM_SUSPENDER_OBJECT_TYPE",
2149: "WASM_SUSPENDER_OBJECT_TYPE", 2149: "WASM_TABLE_OBJECT_TYPE",
2150: "WASM_TABLE_OBJECT_TYPE", 2150: "WASM_TAG_OBJECT_TYPE",
2151: "WASM_TAG_OBJECT_TYPE", 2151: "WASM_VALUE_OBJECT_TYPE",
2152: "WASM_VALUE_OBJECT_TYPE",
} }
# List of known V8 maps. # List of known V8 maps.
...@@ -455,8 +454,8 @@ KNOWN_MAPS = { ...@@ -455,8 +454,8 @@ KNOWN_MAPS = {
("read_only_space", 0x06a9d): (138, "StoreHandler1Map"), ("read_only_space", 0x06a9d): (138, "StoreHandler1Map"),
("read_only_space", 0x06ac5): (138, "StoreHandler2Map"), ("read_only_space", 0x06ac5): (138, "StoreHandler2Map"),
("read_only_space", 0x06aed): (138, "StoreHandler3Map"), ("read_only_space", 0x06aed): (138, "StoreHandler3Map"),
("map_space", 0x02139): (2115, "ExternalMap"), ("map_space", 0x02139): (2114, "ExternalMap"),
("map_space", 0x02161): (2119, "JSMessageObjectMap"), ("map_space", 0x02161): (2118, "JSMessageObjectMap"),
} }
# List of known V8 objects. # List of known V8 objects.
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment