Revert "[builtins] Move most WeakMap/WeakSet code from JS to C++ builtins"
This reverts commit 8196e10265
.
Reason for revert: Performance regression due to hashcode lookup.
Original change's description:
> [builtins] Move most WeakMap/WeakSet code from JS to C++ builtins
>
> They were already implemented mostly in C++ (only error/negative
> cases were handled in script), so this is mostly just a cleanup.
> Only the constructors remain in script after this CL.
>
> Bug: v8:6354
> Cq-Include-Trybots: master.tryserver.chromium.linux:linux_chromium_rel_ng
> Change-Id: I5b3579337a8e33dc30d49c2da5cfd42baec697bb
> Reviewed-on: https://chromium-review.googlesource.com/531670
> Reviewed-by: Camillo Bruni <cbruni@chromium.org>
> Reviewed-by: Sathya Gunasekaran <gsathya@chromium.org>
> Commit-Queue: Adam Klein <adamk@chromium.org>
> Cr-Commit-Position: refs/heads/master@{#45924}
TBR=adamk@chromium.org,cbruni@chromium.org,gsathya@chromium.org
Bug: v8:6354, chromium:733238
Cq-Include-Trybots: master.tryserver.chromium.linux:linux_chromium_rel_ng
Change-Id: Ia5a741b9587886298f3ca057f6a6adeba556b8e0
Reviewed-on: https://chromium-review.googlesource.com/537207
Reviewed-by: Adam Klein <adamk@chromium.org>
Commit-Queue: Adam Klein <adamk@chromium.org>
Cr-Commit-Position: refs/heads/master@{#45966}
This commit is contained in:
parent
bc2c785c20
commit
431abca0ca
1
BUILD.gn
1
BUILD.gn
@ -1201,7 +1201,6 @@ v8_source_set("v8_base") {
|
|||||||
"src/builtins/builtins-symbol.cc",
|
"src/builtins/builtins-symbol.cc",
|
||||||
"src/builtins/builtins-typedarray.cc",
|
"src/builtins/builtins-typedarray.cc",
|
||||||
"src/builtins/builtins-utils.h",
|
"src/builtins/builtins-utils.h",
|
||||||
"src/builtins/builtins-weak-collection.cc",
|
|
||||||
"src/builtins/builtins.cc",
|
"src/builtins/builtins.cc",
|
||||||
"src/builtins/builtins.h",
|
"src/builtins/builtins.h",
|
||||||
"src/cached-powers.cc",
|
"src/cached-powers.cc",
|
||||||
|
@ -3091,7 +3091,8 @@ void NativeWeakMap::Set(Local<Value> v8_key, Local<Value> v8_value) {
|
|||||||
DCHECK(false);
|
DCHECK(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
i::JSWeakCollection::Set(weak_collection, key, value);
|
int32_t hash = i::Object::GetOrCreateHash(isolate, key)->value();
|
||||||
|
i::JSWeakCollection::Set(weak_collection, key, value, hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
Local<Value> NativeWeakMap::Get(Local<Value> v8_key) const {
|
Local<Value> NativeWeakMap::Get(Local<Value> v8_key) const {
|
||||||
@ -3153,7 +3154,8 @@ bool NativeWeakMap::Delete(Local<Value> v8_key) {
|
|||||||
DCHECK(false);
|
DCHECK(false);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return i::JSWeakCollection::Delete(weak_collection, key);
|
int32_t hash = i::Object::GetOrCreateHash(isolate, key)->value();
|
||||||
|
return i::JSWeakCollection::Delete(weak_collection, key, hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -3098,42 +3098,19 @@ void Genesis::InitializeGlobal(Handle<JSGlobalObject> global_object,
|
|||||||
}
|
}
|
||||||
|
|
||||||
{ // -- W e a k M a p
|
{ // -- W e a k M a p
|
||||||
Handle<JSObject> prototype =
|
Handle<JSFunction> js_weak_map_fun = InstallFunction(
|
||||||
factory->NewJSObject(isolate->object_function(), TENURED);
|
global, "WeakMap", JS_WEAK_MAP_TYPE, JSWeakMap::kSize,
|
||||||
Handle<JSFunction> js_weak_map_fun =
|
isolate->initial_object_prototype(), Builtins::kIllegal);
|
||||||
InstallFunction(global, "WeakMap", JS_WEAK_MAP_TYPE, JSWeakMap::kSize,
|
|
||||||
prototype, Builtins::kIllegal);
|
|
||||||
InstallWithIntrinsicDefaultProto(isolate, js_weak_map_fun,
|
InstallWithIntrinsicDefaultProto(isolate, js_weak_map_fun,
|
||||||
Context::JS_WEAK_MAP_FUN_INDEX);
|
Context::JS_WEAK_MAP_FUN_INDEX);
|
||||||
JSObject::AddProperty(prototype, factory->constructor_string(),
|
|
||||||
js_weak_map_fun, DONT_ENUM);
|
|
||||||
JSObject::AddProperty(
|
|
||||||
prototype, factory->to_string_tag_symbol(), factory->WeakMap_string(),
|
|
||||||
static_cast<PropertyAttributes>(DONT_ENUM | READ_ONLY));
|
|
||||||
SimpleInstallFunction(prototype, "get", Builtins::kWeakMapGet, 1, true);
|
|
||||||
SimpleInstallFunction(prototype, "has", Builtins::kWeakMapHas, 1, true);
|
|
||||||
SimpleInstallFunction(prototype, "delete", Builtins::kWeakMapDelete, 1,
|
|
||||||
true);
|
|
||||||
SimpleInstallFunction(prototype, "set", Builtins::kWeakMapSet, 2, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{ // -- W e a k S e t
|
{ // -- W e a k S e t
|
||||||
Handle<JSObject> prototype =
|
Handle<JSFunction> js_weak_set_fun = InstallFunction(
|
||||||
factory->NewJSObject(isolate->object_function(), TENURED);
|
global, "WeakSet", JS_WEAK_SET_TYPE, JSWeakSet::kSize,
|
||||||
Handle<JSFunction> js_weak_set_fun =
|
isolate->initial_object_prototype(), Builtins::kIllegal);
|
||||||
InstallFunction(global, "WeakSet", JS_WEAK_SET_TYPE, JSWeakSet::kSize,
|
|
||||||
prototype, Builtins::kIllegal);
|
|
||||||
InstallWithIntrinsicDefaultProto(isolate, js_weak_set_fun,
|
InstallWithIntrinsicDefaultProto(isolate, js_weak_set_fun,
|
||||||
Context::JS_WEAK_SET_FUN_INDEX);
|
Context::JS_WEAK_SET_FUN_INDEX);
|
||||||
JSObject::AddProperty(prototype, factory->constructor_string(),
|
|
||||||
js_weak_set_fun, DONT_ENUM);
|
|
||||||
JSObject::AddProperty(
|
|
||||||
prototype, factory->to_string_tag_symbol(), factory->WeakSet_string(),
|
|
||||||
static_cast<PropertyAttributes>(DONT_ENUM | READ_ONLY));
|
|
||||||
SimpleInstallFunction(prototype, "add", Builtins::kWeakSetAdd, 1, true);
|
|
||||||
SimpleInstallFunction(prototype, "delete", Builtins::kWeakSetDelete, 1,
|
|
||||||
true);
|
|
||||||
SimpleInstallFunction(prototype, "has", Builtins::kWeakSetHas, 1, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{ // -- P r o x y
|
{ // -- P r o x y
|
||||||
|
@ -583,17 +583,6 @@ namespace internal {
|
|||||||
CPP(MapClear) \
|
CPP(MapClear) \
|
||||||
CPP(MapForEach) \
|
CPP(MapForEach) \
|
||||||
\
|
\
|
||||||
/* WeakMap */ \
|
|
||||||
CPP(WeakMapGet) \
|
|
||||||
CPP(WeakMapSet) \
|
|
||||||
CPP(WeakMapDelete) \
|
|
||||||
CPP(WeakMapHas) \
|
|
||||||
\
|
|
||||||
/* WeakSet */ \
|
|
||||||
CPP(WeakSetAdd) \
|
|
||||||
CPP(WeakSetDelete) \
|
|
||||||
CPP(WeakSetHas) \
|
|
||||||
\
|
|
||||||
/* Math */ \
|
/* Math */ \
|
||||||
/* ES6 #sec-math.abs */ \
|
/* ES6 #sec-math.abs */ \
|
||||||
TFJ(MathAbs, 1, kX) \
|
TFJ(MathAbs, 1, kX) \
|
||||||
|
@ -1,107 +0,0 @@
|
|||||||
// Copyright 2017 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/builtins/builtins-utils.h"
|
|
||||||
#include "src/builtins/builtins.h"
|
|
||||||
|
|
||||||
#include "src/objects-inl.h"
|
|
||||||
|
|
||||||
namespace v8 {
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
// Returns the value keyed by |key|, or the hole if |key| is not present.
|
|
||||||
Object* WeakCollectionLookup(Handle<JSWeakCollection> collection,
|
|
||||||
Handle<JSReceiver> key) {
|
|
||||||
return ObjectHashTable::cast(collection->table())->Lookup(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
Object* WeakCollectionHas(Isolate* isolate, Handle<JSWeakCollection> collection,
|
|
||||||
Handle<JSReceiver> key) {
|
|
||||||
return isolate->heap()->ToBoolean(
|
|
||||||
!WeakCollectionLookup(collection, key)->IsTheHole(isolate));
|
|
||||||
}
|
|
||||||
|
|
||||||
} // anonymous namespace
|
|
||||||
|
|
||||||
BUILTIN(WeakMapGet) {
|
|
||||||
HandleScope scope(isolate);
|
|
||||||
CHECK_RECEIVER(JSWeakMap, map, "WeakMap.prototype.get");
|
|
||||||
if (!args.atOrUndefined(isolate, 1)->IsJSReceiver()) {
|
|
||||||
return isolate->heap()->undefined_value();
|
|
||||||
}
|
|
||||||
Handle<JSReceiver> key = args.at<JSReceiver>(1);
|
|
||||||
Object* lookup = WeakCollectionLookup(map, key);
|
|
||||||
return lookup->IsTheHole(isolate) ? isolate->heap()->undefined_value()
|
|
||||||
: lookup;
|
|
||||||
}
|
|
||||||
|
|
||||||
BUILTIN(WeakMapHas) {
|
|
||||||
HandleScope scope(isolate);
|
|
||||||
CHECK_RECEIVER(JSWeakMap, map, "WeakMap.prototype.has");
|
|
||||||
if (!args.atOrUndefined(isolate, 1)->IsJSReceiver()) {
|
|
||||||
return isolate->heap()->false_value();
|
|
||||||
}
|
|
||||||
Handle<JSReceiver> key = args.at<JSReceiver>(1);
|
|
||||||
return WeakCollectionHas(isolate, map, key);
|
|
||||||
}
|
|
||||||
|
|
||||||
BUILTIN(WeakMapDelete) {
|
|
||||||
HandleScope scope(isolate);
|
|
||||||
CHECK_RECEIVER(JSWeakMap, map, "WeakMap.prototype.delete");
|
|
||||||
if (!args.atOrUndefined(isolate, 1)->IsJSReceiver()) {
|
|
||||||
return isolate->heap()->false_value();
|
|
||||||
}
|
|
||||||
Handle<JSReceiver> key = args.at<JSReceiver>(1);
|
|
||||||
return isolate->heap()->ToBoolean(JSWeakCollection::Delete(map, key));
|
|
||||||
}
|
|
||||||
|
|
||||||
BUILTIN(WeakMapSet) {
|
|
||||||
HandleScope scope(isolate);
|
|
||||||
CHECK_RECEIVER(JSWeakMap, map, "WeakMap.prototype.set");
|
|
||||||
if (!args.atOrUndefined(isolate, 1)->IsJSReceiver()) {
|
|
||||||
THROW_NEW_ERROR_RETURN_FAILURE(
|
|
||||||
isolate, NewTypeError(MessageTemplate::kInvalidWeakMapKey));
|
|
||||||
}
|
|
||||||
Handle<JSReceiver> key = args.at<JSReceiver>(1);
|
|
||||||
Handle<Object> value = args.atOrUndefined(isolate, 2);
|
|
||||||
JSWeakCollection::Set(map, key, value);
|
|
||||||
return *map;
|
|
||||||
}
|
|
||||||
|
|
||||||
BUILTIN(WeakSetAdd) {
|
|
||||||
HandleScope scope(isolate);
|
|
||||||
CHECK_RECEIVER(JSWeakSet, set, "WeakSet.prototype.set");
|
|
||||||
if (!args.atOrUndefined(isolate, 1)->IsJSReceiver()) {
|
|
||||||
THROW_NEW_ERROR_RETURN_FAILURE(
|
|
||||||
isolate, NewTypeError(MessageTemplate::kInvalidWeakSetValue));
|
|
||||||
}
|
|
||||||
Handle<JSReceiver> key = args.at<JSReceiver>(1);
|
|
||||||
JSWeakCollection::Set(set, key, isolate->factory()->true_value());
|
|
||||||
return *set;
|
|
||||||
}
|
|
||||||
|
|
||||||
BUILTIN(WeakSetDelete) {
|
|
||||||
HandleScope scope(isolate);
|
|
||||||
CHECK_RECEIVER(JSWeakSet, set, "WeakSet.prototype.delete");
|
|
||||||
if (!args.atOrUndefined(isolate, 1)->IsJSReceiver()) {
|
|
||||||
return isolate->heap()->false_value();
|
|
||||||
}
|
|
||||||
Handle<JSReceiver> key = args.at<JSReceiver>(1);
|
|
||||||
return isolate->heap()->ToBoolean(JSWeakCollection::Delete(set, key));
|
|
||||||
}
|
|
||||||
|
|
||||||
BUILTIN(WeakSetHas) {
|
|
||||||
HandleScope scope(isolate);
|
|
||||||
CHECK_RECEIVER(JSWeakSet, set, "WeakSet.prototype.has");
|
|
||||||
if (!args.atOrUndefined(isolate, 1)->IsJSReceiver()) {
|
|
||||||
return isolate->heap()->false_value();
|
|
||||||
}
|
|
||||||
Handle<JSReceiver> key = args.at<JSReceiver>(1);
|
|
||||||
return WeakCollectionHas(isolate, set, key);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace internal
|
|
||||||
} // namespace v8
|
|
@ -11,11 +11,20 @@
|
|||||||
// -------------------------------------------------------------------
|
// -------------------------------------------------------------------
|
||||||
// Imports
|
// Imports
|
||||||
|
|
||||||
|
var GetExistingHash;
|
||||||
|
var GetHash;
|
||||||
|
var GlobalObject = global.Object;
|
||||||
var GlobalWeakMap = global.WeakMap;
|
var GlobalWeakMap = global.WeakMap;
|
||||||
var GlobalWeakSet = global.WeakSet;
|
var GlobalWeakSet = global.WeakSet;
|
||||||
|
var toStringTagSymbol = utils.ImportNow("to_string_tag_symbol");
|
||||||
|
|
||||||
|
utils.Import(function(from) {
|
||||||
|
GetExistingHash = from.GetExistingHash;
|
||||||
|
GetHash = from.GetHash;
|
||||||
|
});
|
||||||
|
|
||||||
// -------------------------------------------------------------------
|
// -------------------------------------------------------------------
|
||||||
// WeakMap
|
// Harmony WeakMap
|
||||||
|
|
||||||
function WeakMapConstructor(iterable) {
|
function WeakMapConstructor(iterable) {
|
||||||
if (IS_UNDEFINED(new.target)) {
|
if (IS_UNDEFINED(new.target)) {
|
||||||
@ -38,11 +47,73 @@ function WeakMapConstructor(iterable) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
%SetCode(GlobalWeakMap, WeakMapConstructor);
|
|
||||||
%FunctionSetLength(GlobalWeakMap, 0);
|
function WeakMapGet(key) {
|
||||||
|
if (!IS_WEAKMAP(this)) {
|
||||||
|
throw %make_type_error(kIncompatibleMethodReceiver,
|
||||||
|
'WeakMap.prototype.get', this);
|
||||||
|
}
|
||||||
|
if (!IS_RECEIVER(key)) return UNDEFINED;
|
||||||
|
var hash = GetExistingHash(key);
|
||||||
|
if (IS_UNDEFINED(hash)) return UNDEFINED;
|
||||||
|
return %WeakCollectionGet(this, key, hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function WeakMapSet(key, value) {
|
||||||
|
if (!IS_WEAKMAP(this)) {
|
||||||
|
throw %make_type_error(kIncompatibleMethodReceiver,
|
||||||
|
'WeakMap.prototype.set', this);
|
||||||
|
}
|
||||||
|
if (!IS_RECEIVER(key)) throw %make_type_error(kInvalidWeakMapKey);
|
||||||
|
return %WeakCollectionSet(this, key, value, GetHash(key));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function WeakMapHas(key) {
|
||||||
|
if (!IS_WEAKMAP(this)) {
|
||||||
|
throw %make_type_error(kIncompatibleMethodReceiver,
|
||||||
|
'WeakMap.prototype.has', this);
|
||||||
|
}
|
||||||
|
if (!IS_RECEIVER(key)) return false;
|
||||||
|
var hash = GetExistingHash(key);
|
||||||
|
if (IS_UNDEFINED(hash)) return false;
|
||||||
|
return %WeakCollectionHas(this, key, hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function WeakMapDelete(key) {
|
||||||
|
if (!IS_WEAKMAP(this)) {
|
||||||
|
throw %make_type_error(kIncompatibleMethodReceiver,
|
||||||
|
'WeakMap.prototype.delete', this);
|
||||||
|
}
|
||||||
|
if (!IS_RECEIVER(key)) return false;
|
||||||
|
var hash = GetExistingHash(key);
|
||||||
|
if (IS_UNDEFINED(hash)) return false;
|
||||||
|
return %WeakCollectionDelete(this, key, hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// -------------------------------------------------------------------
|
// -------------------------------------------------------------------
|
||||||
// WeakSet
|
|
||||||
|
%SetCode(GlobalWeakMap, WeakMapConstructor);
|
||||||
|
%FunctionSetLength(GlobalWeakMap, 0);
|
||||||
|
%FunctionSetPrototype(GlobalWeakMap, new GlobalObject());
|
||||||
|
%AddNamedProperty(GlobalWeakMap.prototype, "constructor", GlobalWeakMap,
|
||||||
|
DONT_ENUM);
|
||||||
|
%AddNamedProperty(GlobalWeakMap.prototype, toStringTagSymbol, "WeakMap",
|
||||||
|
DONT_ENUM | READ_ONLY);
|
||||||
|
|
||||||
|
// Set up the non-enumerable functions on the WeakMap prototype object.
|
||||||
|
utils.InstallFunctions(GlobalWeakMap.prototype, DONT_ENUM, [
|
||||||
|
"get", WeakMapGet,
|
||||||
|
"set", WeakMapSet,
|
||||||
|
"has", WeakMapHas,
|
||||||
|
"delete", WeakMapDelete
|
||||||
|
]);
|
||||||
|
|
||||||
|
// -------------------------------------------------------------------
|
||||||
|
// Harmony WeakSet
|
||||||
|
|
||||||
function WeakSetConstructor(iterable) {
|
function WeakSetConstructor(iterable) {
|
||||||
if (IS_UNDEFINED(new.target)) {
|
if (IS_UNDEFINED(new.target)) {
|
||||||
@ -62,9 +133,56 @@ function WeakSetConstructor(iterable) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function WeakSetAdd(value) {
|
||||||
|
if (!IS_WEAKSET(this)) {
|
||||||
|
throw %make_type_error(kIncompatibleMethodReceiver,
|
||||||
|
'WeakSet.prototype.add', this);
|
||||||
|
}
|
||||||
|
if (!IS_RECEIVER(value)) throw %make_type_error(kInvalidWeakSetValue);
|
||||||
|
return %WeakCollectionSet(this, value, true, GetHash(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function WeakSetHas(value) {
|
||||||
|
if (!IS_WEAKSET(this)) {
|
||||||
|
throw %make_type_error(kIncompatibleMethodReceiver,
|
||||||
|
'WeakSet.prototype.has', this);
|
||||||
|
}
|
||||||
|
if (!IS_RECEIVER(value)) return false;
|
||||||
|
var hash = GetExistingHash(value);
|
||||||
|
if (IS_UNDEFINED(hash)) return false;
|
||||||
|
return %WeakCollectionHas(this, value, hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function WeakSetDelete(value) {
|
||||||
|
if (!IS_WEAKSET(this)) {
|
||||||
|
throw %make_type_error(kIncompatibleMethodReceiver,
|
||||||
|
'WeakSet.prototype.delete', this);
|
||||||
|
}
|
||||||
|
if (!IS_RECEIVER(value)) return false;
|
||||||
|
var hash = GetExistingHash(value);
|
||||||
|
if (IS_UNDEFINED(hash)) return false;
|
||||||
|
return %WeakCollectionDelete(this, value, hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// -------------------------------------------------------------------
|
// -------------------------------------------------------------------
|
||||||
|
|
||||||
%SetCode(GlobalWeakSet, WeakSetConstructor);
|
%SetCode(GlobalWeakSet, WeakSetConstructor);
|
||||||
%FunctionSetLength(GlobalWeakSet, 0);
|
%FunctionSetLength(GlobalWeakSet, 0);
|
||||||
|
%FunctionSetPrototype(GlobalWeakSet, new GlobalObject());
|
||||||
|
%AddNamedProperty(GlobalWeakSet.prototype, "constructor", GlobalWeakSet,
|
||||||
|
DONT_ENUM);
|
||||||
|
%AddNamedProperty(GlobalWeakSet.prototype, toStringTagSymbol, "WeakSet",
|
||||||
|
DONT_ENUM | READ_ONLY);
|
||||||
|
|
||||||
|
// Set up the non-enumerable functions on the WeakSet prototype object.
|
||||||
|
utils.InstallFunctions(GlobalWeakSet.prototype, DONT_ENUM, [
|
||||||
|
"add", WeakSetAdd,
|
||||||
|
"has", WeakSetHas,
|
||||||
|
"delete", WeakSetDelete
|
||||||
|
]);
|
||||||
|
|
||||||
})
|
})
|
||||||
|
@ -18897,13 +18897,16 @@ void JSWeakCollection::Initialize(Handle<JSWeakCollection> weak_collection,
|
|||||||
weak_collection->set_table(*table);
|
weak_collection->set_table(*table);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void JSWeakCollection::Set(Handle<JSWeakCollection> weak_collection,
|
void JSWeakCollection::Set(Handle<JSWeakCollection> weak_collection,
|
||||||
Handle<Object> key, Handle<Object> value) {
|
Handle<Object> key, Handle<Object> value,
|
||||||
|
int32_t hash) {
|
||||||
DCHECK(key->IsJSReceiver() || key->IsSymbol());
|
DCHECK(key->IsJSReceiver() || key->IsSymbol());
|
||||||
Handle<ObjectHashTable> table(
|
Handle<ObjectHashTable> table(
|
||||||
ObjectHashTable::cast(weak_collection->table()));
|
ObjectHashTable::cast(weak_collection->table()));
|
||||||
DCHECK(table->IsKey(table->GetIsolate(), *key));
|
DCHECK(table->IsKey(table->GetIsolate(), *key));
|
||||||
Handle<ObjectHashTable> new_table = ObjectHashTable::Put(table, key, value);
|
Handle<ObjectHashTable> new_table =
|
||||||
|
ObjectHashTable::Put(table, key, value, hash);
|
||||||
weak_collection->set_table(*new_table);
|
weak_collection->set_table(*new_table);
|
||||||
if (*table != *new_table) {
|
if (*table != *new_table) {
|
||||||
// Zap the old table since we didn't record slots for its elements.
|
// Zap the old table since we didn't record slots for its elements.
|
||||||
@ -18911,15 +18914,16 @@ void JSWeakCollection::Set(Handle<JSWeakCollection> weak_collection,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool JSWeakCollection::Delete(Handle<JSWeakCollection> weak_collection,
|
bool JSWeakCollection::Delete(Handle<JSWeakCollection> weak_collection,
|
||||||
Handle<Object> key) {
|
Handle<Object> key, int32_t hash) {
|
||||||
DCHECK(key->IsJSReceiver() || key->IsSymbol());
|
DCHECK(key->IsJSReceiver() || key->IsSymbol());
|
||||||
Handle<ObjectHashTable> table(
|
Handle<ObjectHashTable> table(
|
||||||
ObjectHashTable::cast(weak_collection->table()));
|
ObjectHashTable::cast(weak_collection->table()));
|
||||||
DCHECK(table->IsKey(table->GetIsolate(), *key));
|
DCHECK(table->IsKey(table->GetIsolate(), *key));
|
||||||
bool was_present = false;
|
bool was_present = false;
|
||||||
Handle<ObjectHashTable> new_table =
|
Handle<ObjectHashTable> new_table =
|
||||||
ObjectHashTable::Remove(table, key, &was_present);
|
ObjectHashTable::Remove(table, key, &was_present, hash);
|
||||||
weak_collection->set_table(*new_table);
|
weak_collection->set_table(*new_table);
|
||||||
if (*table != *new_table) {
|
if (*table != *new_table) {
|
||||||
// Zap the old table since we didn't record slots for its elements.
|
// Zap the old table since we didn't record slots for its elements.
|
||||||
|
@ -6351,8 +6351,9 @@ class JSWeakCollection: public JSObject {
|
|||||||
|
|
||||||
static void Initialize(Handle<JSWeakCollection> collection, Isolate* isolate);
|
static void Initialize(Handle<JSWeakCollection> collection, Isolate* isolate);
|
||||||
static void Set(Handle<JSWeakCollection> collection, Handle<Object> key,
|
static void Set(Handle<JSWeakCollection> collection, Handle<Object> key,
|
||||||
Handle<Object> value);
|
Handle<Object> value, int32_t hash);
|
||||||
static bool Delete(Handle<JSWeakCollection> collection, Handle<Object> key);
|
static bool Delete(Handle<JSWeakCollection> collection, Handle<Object> key,
|
||||||
|
int32_t hash);
|
||||||
static Handle<JSArray> GetEntries(Handle<JSWeakCollection> holder,
|
static Handle<JSArray> GetEntries(Handle<JSWeakCollection> holder,
|
||||||
int max_entries);
|
int max_entries);
|
||||||
|
|
||||||
|
@ -255,6 +255,68 @@ RUNTIME_FUNCTION(Runtime_WeakCollectionInitialize) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
RUNTIME_FUNCTION(Runtime_WeakCollectionGet) {
|
||||||
|
HandleScope scope(isolate);
|
||||||
|
DCHECK_EQ(3, args.length());
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(Object, key, 1);
|
||||||
|
CONVERT_SMI_ARG_CHECKED(hash, 2)
|
||||||
|
CHECK(key->IsJSReceiver() || key->IsSymbol());
|
||||||
|
Handle<ObjectHashTable> table(
|
||||||
|
ObjectHashTable::cast(weak_collection->table()));
|
||||||
|
CHECK(table->IsKey(isolate, *key));
|
||||||
|
Handle<Object> lookup(table->Lookup(key, hash), isolate);
|
||||||
|
return lookup->IsTheHole(isolate) ? isolate->heap()->undefined_value()
|
||||||
|
: *lookup;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
RUNTIME_FUNCTION(Runtime_WeakCollectionHas) {
|
||||||
|
HandleScope scope(isolate);
|
||||||
|
DCHECK_EQ(3, args.length());
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(Object, key, 1);
|
||||||
|
CONVERT_SMI_ARG_CHECKED(hash, 2)
|
||||||
|
CHECK(key->IsJSReceiver() || key->IsSymbol());
|
||||||
|
Handle<ObjectHashTable> table(
|
||||||
|
ObjectHashTable::cast(weak_collection->table()));
|
||||||
|
CHECK(table->IsKey(isolate, *key));
|
||||||
|
Handle<Object> lookup(table->Lookup(key, hash), isolate);
|
||||||
|
return isolate->heap()->ToBoolean(!lookup->IsTheHole(isolate));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
RUNTIME_FUNCTION(Runtime_WeakCollectionDelete) {
|
||||||
|
HandleScope scope(isolate);
|
||||||
|
DCHECK_EQ(3, args.length());
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(Object, key, 1);
|
||||||
|
CONVERT_SMI_ARG_CHECKED(hash, 2)
|
||||||
|
CHECK(key->IsJSReceiver() || key->IsSymbol());
|
||||||
|
Handle<ObjectHashTable> table(
|
||||||
|
ObjectHashTable::cast(weak_collection->table()));
|
||||||
|
CHECK(table->IsKey(isolate, *key));
|
||||||
|
bool was_present = JSWeakCollection::Delete(weak_collection, key, hash);
|
||||||
|
return isolate->heap()->ToBoolean(was_present);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
RUNTIME_FUNCTION(Runtime_WeakCollectionSet) {
|
||||||
|
HandleScope scope(isolate);
|
||||||
|
DCHECK_EQ(4, args.length());
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(Object, key, 1);
|
||||||
|
CHECK(key->IsJSReceiver() || key->IsSymbol());
|
||||||
|
CONVERT_ARG_HANDLE_CHECKED(Object, value, 2);
|
||||||
|
CONVERT_SMI_ARG_CHECKED(hash, 3)
|
||||||
|
Handle<ObjectHashTable> table(
|
||||||
|
ObjectHashTable::cast(weak_collection->table()));
|
||||||
|
CHECK(table->IsKey(isolate, *key));
|
||||||
|
JSWeakCollection::Set(weak_collection, key, value, hash);
|
||||||
|
return *weak_collection;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
RUNTIME_FUNCTION(Runtime_GetWeakSetValues) {
|
RUNTIME_FUNCTION(Runtime_GetWeakSetValues) {
|
||||||
HandleScope scope(isolate);
|
HandleScope scope(isolate);
|
||||||
DCHECK_EQ(2, args.length());
|
DCHECK_EQ(2, args.length());
|
||||||
|
@ -114,6 +114,10 @@ namespace internal {
|
|||||||
F(GetWeakMapEntries, 2, 1) \
|
F(GetWeakMapEntries, 2, 1) \
|
||||||
F(MapIteratorNext, 2, 1) \
|
F(MapIteratorNext, 2, 1) \
|
||||||
F(WeakCollectionInitialize, 1, 1) \
|
F(WeakCollectionInitialize, 1, 1) \
|
||||||
|
F(WeakCollectionGet, 3, 1) \
|
||||||
|
F(WeakCollectionHas, 3, 1) \
|
||||||
|
F(WeakCollectionDelete, 3, 1) \
|
||||||
|
F(WeakCollectionSet, 4, 1) \
|
||||||
F(GetWeakSetValues, 2, 1) \
|
F(GetWeakSetValues, 2, 1) \
|
||||||
F(IsJSMap, 1, 1) \
|
F(IsJSMap, 1, 1) \
|
||||||
F(IsJSSet, 1, 1) \
|
F(IsJSSet, 1, 1) \
|
||||||
|
@ -658,7 +658,6 @@
|
|||||||
'builtins/builtins-symbol.cc',
|
'builtins/builtins-symbol.cc',
|
||||||
'builtins/builtins-typedarray.cc',
|
'builtins/builtins-typedarray.cc',
|
||||||
'builtins/builtins-utils.h',
|
'builtins/builtins-utils.h',
|
||||||
'builtins/builtins-weak-collection.cc',
|
|
||||||
'builtins/builtins.cc',
|
'builtins/builtins.cc',
|
||||||
'builtins/builtins.h',
|
'builtins/builtins.h',
|
||||||
'cached-powers.cc',
|
'cached-powers.cc',
|
||||||
|
@ -96,8 +96,10 @@ TEST(Weakness) {
|
|||||||
Handle<Map> map = factory->NewMap(JS_OBJECT_TYPE, JSObject::kHeaderSize);
|
Handle<Map> map = factory->NewMap(JS_OBJECT_TYPE, JSObject::kHeaderSize);
|
||||||
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
|
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
|
||||||
Handle<Smi> smi(Smi::FromInt(23), isolate);
|
Handle<Smi> smi(Smi::FromInt(23), isolate);
|
||||||
JSWeakCollection::Set(weakmap, key, object);
|
int32_t hash = Object::GetOrCreateHash(isolate, key)->value();
|
||||||
JSWeakCollection::Set(weakmap, object, smi);
|
JSWeakCollection::Set(weakmap, key, object, hash);
|
||||||
|
int32_t object_hash = Object::GetOrCreateHash(isolate, object)->value();
|
||||||
|
JSWeakCollection::Set(weakmap, object, smi, object_hash);
|
||||||
}
|
}
|
||||||
CHECK_EQ(2, ObjectHashTable::cast(weakmap->table())->NumberOfElements());
|
CHECK_EQ(2, ObjectHashTable::cast(weakmap->table())->NumberOfElements());
|
||||||
|
|
||||||
@ -140,7 +142,8 @@ TEST(Shrinking) {
|
|||||||
for (int i = 0; i < 32; i++) {
|
for (int i = 0; i < 32; i++) {
|
||||||
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
|
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
|
||||||
Handle<Smi> smi(Smi::FromInt(i), isolate);
|
Handle<Smi> smi(Smi::FromInt(i), isolate);
|
||||||
JSWeakCollection::Set(weakmap, object, smi);
|
int32_t object_hash = Object::GetOrCreateHash(isolate, object)->value();
|
||||||
|
JSWeakCollection::Set(weakmap, object, smi, object_hash);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -187,7 +190,8 @@ TEST(Regress2060a) {
|
|||||||
Handle<JSObject> object = factory->NewJSObject(function, TENURED);
|
Handle<JSObject> object = factory->NewJSObject(function, TENURED);
|
||||||
CHECK(!heap->InNewSpace(*object));
|
CHECK(!heap->InNewSpace(*object));
|
||||||
CHECK(!first_page->Contains(object->address()));
|
CHECK(!first_page->Contains(object->address()));
|
||||||
JSWeakCollection::Set(weakmap, key, object);
|
int32_t hash = Object::GetOrCreateHash(isolate, key)->value();
|
||||||
|
JSWeakCollection::Set(weakmap, key, object, hash);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -228,7 +232,8 @@ TEST(Regress2060b) {
|
|||||||
Handle<JSWeakMap> weakmap = AllocateJSWeakMap(isolate);
|
Handle<JSWeakMap> weakmap = AllocateJSWeakMap(isolate);
|
||||||
for (int i = 0; i < 32; i++) {
|
for (int i = 0; i < 32; i++) {
|
||||||
Handle<Smi> smi(Smi::FromInt(i), isolate);
|
Handle<Smi> smi(Smi::FromInt(i), isolate);
|
||||||
JSWeakCollection::Set(weakmap, keys[i], smi);
|
int32_t hash = Object::GetOrCreateHash(isolate, keys[i])->value();
|
||||||
|
JSWeakCollection::Set(weakmap, keys[i], smi, hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Force compacting garbage collection. The subsequent collections are used
|
// Force compacting garbage collection. The subsequent collections are used
|
||||||
|
@ -97,7 +97,8 @@ TEST(WeakSet_Weakness) {
|
|||||||
{
|
{
|
||||||
HandleScope scope(isolate);
|
HandleScope scope(isolate);
|
||||||
Handle<Smi> smi(Smi::FromInt(23), isolate);
|
Handle<Smi> smi(Smi::FromInt(23), isolate);
|
||||||
JSWeakCollection::Set(weakset, key, smi);
|
int32_t hash = Object::GetOrCreateHash(isolate, key)->value();
|
||||||
|
JSWeakCollection::Set(weakset, key, smi, hash);
|
||||||
}
|
}
|
||||||
CHECK_EQ(1, ObjectHashTable::cast(weakset->table())->NumberOfElements());
|
CHECK_EQ(1, ObjectHashTable::cast(weakset->table())->NumberOfElements());
|
||||||
|
|
||||||
@ -140,7 +141,8 @@ TEST(WeakSet_Shrinking) {
|
|||||||
for (int i = 0; i < 32; i++) {
|
for (int i = 0; i < 32; i++) {
|
||||||
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
|
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
|
||||||
Handle<Smi> smi(Smi::FromInt(i), isolate);
|
Handle<Smi> smi(Smi::FromInt(i), isolate);
|
||||||
JSWeakCollection::Set(weakset, object, smi);
|
int32_t hash = Object::GetOrCreateHash(isolate, object)->value();
|
||||||
|
JSWeakCollection::Set(weakset, object, smi, hash);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -187,7 +189,8 @@ TEST(WeakSet_Regress2060a) {
|
|||||||
Handle<JSObject> object = factory->NewJSObject(function, TENURED);
|
Handle<JSObject> object = factory->NewJSObject(function, TENURED);
|
||||||
CHECK(!heap->InNewSpace(*object));
|
CHECK(!heap->InNewSpace(*object));
|
||||||
CHECK(!first_page->Contains(object->address()));
|
CHECK(!first_page->Contains(object->address()));
|
||||||
JSWeakCollection::Set(weakset, key, object);
|
int32_t hash = Object::GetOrCreateHash(isolate, key)->value();
|
||||||
|
JSWeakCollection::Set(weakset, key, object, hash);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -228,7 +231,8 @@ TEST(WeakSet_Regress2060b) {
|
|||||||
Handle<JSWeakSet> weakset = AllocateJSWeakSet(isolate);
|
Handle<JSWeakSet> weakset = AllocateJSWeakSet(isolate);
|
||||||
for (int i = 0; i < 32; i++) {
|
for (int i = 0; i < 32; i++) {
|
||||||
Handle<Smi> smi(Smi::FromInt(i), isolate);
|
Handle<Smi> smi(Smi::FromInt(i), isolate);
|
||||||
JSWeakCollection::Set(weakset, keys[i], smi);
|
int32_t hash = Object::GetOrCreateHash(isolate, keys[i])->value();
|
||||||
|
JSWeakCollection::Set(weakset, keys[i], smi, hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Force compacting garbage collection. The subsequent collections are used
|
// Force compacting garbage collection. The subsequent collections are used
|
||||||
|
Loading…
Reference in New Issue
Block a user