[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}
This commit is contained in:
Adam Klein 2017-06-13 11:00:52 -07:00 committed by Commit Bot
parent 041d7339e3
commit 8196e10265
13 changed files with 168 additions and 225 deletions

View File

@ -1201,6 +1201,7 @@ v8_source_set("v8_base") {
"src/builtins/builtins-symbol.cc",
"src/builtins/builtins-typedarray.cc",
"src/builtins/builtins-utils.h",
"src/builtins/builtins-weak-collection.cc",
"src/builtins/builtins.cc",
"src/builtins/builtins.h",
"src/cached-powers.cc",

View File

@ -3091,8 +3091,7 @@ void NativeWeakMap::Set(Local<Value> v8_key, Local<Value> v8_value) {
DCHECK(false);
return;
}
int32_t hash = i::Object::GetOrCreateHash(isolate, key)->value();
i::JSWeakCollection::Set(weak_collection, key, value, hash);
i::JSWeakCollection::Set(weak_collection, key, value);
}
Local<Value> NativeWeakMap::Get(Local<Value> v8_key) const {
@ -3154,8 +3153,7 @@ bool NativeWeakMap::Delete(Local<Value> v8_key) {
DCHECK(false);
return false;
}
int32_t hash = i::Object::GetOrCreateHash(isolate, key)->value();
return i::JSWeakCollection::Delete(weak_collection, key, hash);
return i::JSWeakCollection::Delete(weak_collection, key);
}

View File

@ -3047,19 +3047,42 @@ void Genesis::InitializeGlobal(Handle<JSGlobalObject> global_object,
}
{ // -- W e a k M a p
Handle<JSFunction> js_weak_map_fun = InstallFunction(
global, "WeakMap", JS_WEAK_MAP_TYPE, JSWeakMap::kSize,
isolate->initial_object_prototype(), Builtins::kIllegal);
Handle<JSObject> prototype =
factory->NewJSObject(isolate->object_function(), TENURED);
Handle<JSFunction> js_weak_map_fun =
InstallFunction(global, "WeakMap", JS_WEAK_MAP_TYPE, JSWeakMap::kSize,
prototype, Builtins::kIllegal);
InstallWithIntrinsicDefaultProto(isolate, js_weak_map_fun,
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
Handle<JSFunction> js_weak_set_fun = InstallFunction(
global, "WeakSet", JS_WEAK_SET_TYPE, JSWeakSet::kSize,
isolate->initial_object_prototype(), Builtins::kIllegal);
Handle<JSObject> prototype =
factory->NewJSObject(isolate->object_function(), TENURED);
Handle<JSFunction> js_weak_set_fun =
InstallFunction(global, "WeakSet", JS_WEAK_SET_TYPE, JSWeakSet::kSize,
prototype, Builtins::kIllegal);
InstallWithIntrinsicDefaultProto(isolate, js_weak_set_fun,
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

View File

@ -583,6 +583,17 @@ namespace internal {
CPP(MapClear) \
CPP(MapForEach) \
\
/* WeakMap */ \
CPP(WeakMapGet) \
CPP(WeakMapSet) \
CPP(WeakMapDelete) \
CPP(WeakMapHas) \
\
/* WeakSet */ \
CPP(WeakSetAdd) \
CPP(WeakSetDelete) \
CPP(WeakSetHas) \
\
/* Math */ \
/* ES6 #sec-math.abs */ \
TFJ(MathAbs, 1, kX) \

View File

@ -0,0 +1,107 @@
// 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

View File

@ -11,20 +11,11 @@
// -------------------------------------------------------------------
// Imports
var GetExistingHash;
var GetHash;
var GlobalObject = global.Object;
var GlobalWeakMap = global.WeakMap;
var GlobalWeakSet = global.WeakSet;
var toStringTagSymbol = utils.ImportNow("to_string_tag_symbol");
utils.Import(function(from) {
GetExistingHash = from.GetExistingHash;
GetHash = from.GetHash;
});
// -------------------------------------------------------------------
// Harmony WeakMap
// WeakMap
function WeakMapConstructor(iterable) {
if (IS_UNDEFINED(new.target)) {
@ -47,73 +38,11 @@ function WeakMapConstructor(iterable) {
}
}
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);
}
// -------------------------------------------------------------------
%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
// WeakSet
function WeakSetConstructor(iterable) {
if (IS_UNDEFINED(new.target)) {
@ -133,56 +62,9 @@ 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);
%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
]);
})

View File

@ -18986,16 +18986,13 @@ void JSWeakCollection::Initialize(Handle<JSWeakCollection> weak_collection,
weak_collection->set_table(*table);
}
void JSWeakCollection::Set(Handle<JSWeakCollection> weak_collection,
Handle<Object> key, Handle<Object> value,
int32_t hash) {
Handle<Object> key, Handle<Object> value) {
DCHECK(key->IsJSReceiver() || key->IsSymbol());
Handle<ObjectHashTable> table(
ObjectHashTable::cast(weak_collection->table()));
DCHECK(table->IsKey(table->GetIsolate(), *key));
Handle<ObjectHashTable> new_table =
ObjectHashTable::Put(table, key, value, hash);
Handle<ObjectHashTable> new_table = ObjectHashTable::Put(table, key, value);
weak_collection->set_table(*new_table);
if (*table != *new_table) {
// Zap the old table since we didn't record slots for its elements.
@ -19003,16 +19000,15 @@ void JSWeakCollection::Set(Handle<JSWeakCollection> weak_collection,
}
}
bool JSWeakCollection::Delete(Handle<JSWeakCollection> weak_collection,
Handle<Object> key, int32_t hash) {
Handle<Object> key) {
DCHECK(key->IsJSReceiver() || key->IsSymbol());
Handle<ObjectHashTable> table(
ObjectHashTable::cast(weak_collection->table()));
DCHECK(table->IsKey(table->GetIsolate(), *key));
bool was_present = false;
Handle<ObjectHashTable> new_table =
ObjectHashTable::Remove(table, key, &was_present, hash);
ObjectHashTable::Remove(table, key, &was_present);
weak_collection->set_table(*new_table);
if (*table != *new_table) {
// Zap the old table since we didn't record slots for its elements.

View File

@ -6369,9 +6369,8 @@ class JSWeakCollection: public JSObject {
static void Initialize(Handle<JSWeakCollection> collection, Isolate* isolate);
static void Set(Handle<JSWeakCollection> collection, Handle<Object> key,
Handle<Object> value, int32_t hash);
static bool Delete(Handle<JSWeakCollection> collection, Handle<Object> key,
int32_t hash);
Handle<Object> value);
static bool Delete(Handle<JSWeakCollection> collection, Handle<Object> key);
static Handle<JSArray> GetEntries(Handle<JSWeakCollection> holder,
int max_entries);

View File

@ -255,68 +255,6 @@ 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) {
HandleScope scope(isolate);
DCHECK_EQ(2, args.length());

View File

@ -114,10 +114,6 @@ namespace internal {
F(GetWeakMapEntries, 2, 1) \
F(MapIteratorNext, 2, 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(IsJSMap, 1, 1) \
F(IsJSSet, 1, 1) \

View File

@ -658,6 +658,7 @@
'builtins/builtins-symbol.cc',
'builtins/builtins-typedarray.cc',
'builtins/builtins-utils.h',
'builtins/builtins-weak-collection.cc',
'builtins/builtins.cc',
'builtins/builtins.h',
'cached-powers.cc',

View File

@ -96,10 +96,8 @@ TEST(Weakness) {
Handle<Map> map = factory->NewMap(JS_OBJECT_TYPE, JSObject::kHeaderSize);
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
Handle<Smi> smi(Smi::FromInt(23), isolate);
int32_t hash = Object::GetOrCreateHash(isolate, key)->value();
JSWeakCollection::Set(weakmap, key, object, hash);
int32_t object_hash = Object::GetOrCreateHash(isolate, object)->value();
JSWeakCollection::Set(weakmap, object, smi, object_hash);
JSWeakCollection::Set(weakmap, key, object);
JSWeakCollection::Set(weakmap, object, smi);
}
CHECK_EQ(2, ObjectHashTable::cast(weakmap->table())->NumberOfElements());
@ -142,8 +140,7 @@ TEST(Shrinking) {
for (int i = 0; i < 32; i++) {
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
Handle<Smi> smi(Smi::FromInt(i), isolate);
int32_t object_hash = Object::GetOrCreateHash(isolate, object)->value();
JSWeakCollection::Set(weakmap, object, smi, object_hash);
JSWeakCollection::Set(weakmap, object, smi);
}
}
@ -190,8 +187,7 @@ TEST(Regress2060a) {
Handle<JSObject> object = factory->NewJSObject(function, TENURED);
CHECK(!heap->InNewSpace(*object));
CHECK(!first_page->Contains(object->address()));
int32_t hash = Object::GetOrCreateHash(isolate, key)->value();
JSWeakCollection::Set(weakmap, key, object, hash);
JSWeakCollection::Set(weakmap, key, object);
}
}
@ -232,8 +228,7 @@ TEST(Regress2060b) {
Handle<JSWeakMap> weakmap = AllocateJSWeakMap(isolate);
for (int i = 0; i < 32; i++) {
Handle<Smi> smi(Smi::FromInt(i), isolate);
int32_t hash = Object::GetOrCreateHash(isolate, keys[i])->value();
JSWeakCollection::Set(weakmap, keys[i], smi, hash);
JSWeakCollection::Set(weakmap, keys[i], smi);
}
// Force compacting garbage collection. The subsequent collections are used

View File

@ -97,8 +97,7 @@ TEST(WeakSet_Weakness) {
{
HandleScope scope(isolate);
Handle<Smi> smi(Smi::FromInt(23), isolate);
int32_t hash = Object::GetOrCreateHash(isolate, key)->value();
JSWeakCollection::Set(weakset, key, smi, hash);
JSWeakCollection::Set(weakset, key, smi);
}
CHECK_EQ(1, ObjectHashTable::cast(weakset->table())->NumberOfElements());
@ -141,8 +140,7 @@ TEST(WeakSet_Shrinking) {
for (int i = 0; i < 32; i++) {
Handle<JSObject> object = factory->NewJSObjectFromMap(map);
Handle<Smi> smi(Smi::FromInt(i), isolate);
int32_t hash = Object::GetOrCreateHash(isolate, object)->value();
JSWeakCollection::Set(weakset, object, smi, hash);
JSWeakCollection::Set(weakset, object, smi);
}
}
@ -189,8 +187,7 @@ TEST(WeakSet_Regress2060a) {
Handle<JSObject> object = factory->NewJSObject(function, TENURED);
CHECK(!heap->InNewSpace(*object));
CHECK(!first_page->Contains(object->address()));
int32_t hash = Object::GetOrCreateHash(isolate, key)->value();
JSWeakCollection::Set(weakset, key, object, hash);
JSWeakCollection::Set(weakset, key, object);
}
}
@ -231,8 +228,7 @@ TEST(WeakSet_Regress2060b) {
Handle<JSWeakSet> weakset = AllocateJSWeakSet(isolate);
for (int i = 0; i < 32; i++) {
Handle<Smi> smi(Smi::FromInt(i), isolate);
int32_t hash = Object::GetOrCreateHash(isolate, keys[i])->value();
JSWeakCollection::Set(weakset, keys[i], smi, hash);
JSWeakCollection::Set(weakset, keys[i], smi);
}
// Force compacting garbage collection. The subsequent collections are used