summaryrefslogtreecommitdiffstats
path: root/js/src/builtin/WeakSetObject.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'js/src/builtin/WeakSetObject.cpp')
-rw-r--r--js/src/builtin/WeakSetObject.cpp170
1 files changed, 170 insertions, 0 deletions
diff --git a/js/src/builtin/WeakSetObject.cpp b/js/src/builtin/WeakSetObject.cpp
new file mode 100644
index 000000000..7ea3f2fef
--- /dev/null
+++ b/js/src/builtin/WeakSetObject.cpp
@@ -0,0 +1,170 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
+ * vim: set ts=8 sts=4 et sw=4 tw=99:
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#include "builtin/WeakSetObject.h"
+
+#include "jsapi.h"
+#include "jscntxt.h"
+#include "jsiter.h"
+
+#include "builtin/MapObject.h"
+#include "builtin/SelfHostingDefines.h"
+#include "builtin/WeakMapObject.h"
+#include "vm/GlobalObject.h"
+#include "vm/SelfHosting.h"
+
+#include "jsobjinlines.h"
+
+#include "vm/Interpreter-inl.h"
+#include "vm/NativeObject-inl.h"
+
+using namespace js;
+
+const Class WeakSetObject::class_ = {
+ "WeakSet",
+ JSCLASS_HAS_CACHED_PROTO(JSProto_WeakSet) |
+ JSCLASS_HAS_RESERVED_SLOTS(WeakSetObject::RESERVED_SLOTS)
+};
+
+const JSPropertySpec WeakSetObject::properties[] = {
+ JS_PS_END
+};
+
+const JSFunctionSpec WeakSetObject::methods[] = {
+ JS_SELF_HOSTED_FN("add", "WeakSet_add", 1, 0),
+ JS_SELF_HOSTED_FN("delete", "WeakSet_delete", 1, 0),
+ JS_SELF_HOSTED_FN("has", "WeakSet_has", 1, 0),
+ JS_FS_END
+};
+
+JSObject*
+WeakSetObject::initClass(JSContext* cx, JSObject* obj)
+{
+ Rooted<GlobalObject*> global(cx, &obj->as<GlobalObject>());
+ RootedPlainObject proto(cx, NewBuiltinClassInstance<PlainObject>(cx));
+ if (!proto)
+ return nullptr;
+
+ Rooted<JSFunction*> ctor(cx, global->createConstructor(cx, construct, ClassName(JSProto_WeakSet, cx), 0));
+ if (!ctor ||
+ !LinkConstructorAndPrototype(cx, ctor, proto) ||
+ !DefinePropertiesAndFunctions(cx, proto, properties, methods) ||
+ !DefineToStringTag(cx, proto, cx->names().WeakSet) ||
+ !GlobalObject::initBuiltinConstructor(cx, global, JSProto_WeakSet, ctor, proto))
+ {
+ return nullptr;
+ }
+ return proto;
+}
+
+WeakSetObject*
+WeakSetObject::create(JSContext* cx, HandleObject proto /* = nullptr */)
+{
+ RootedObject map(cx, NewBuiltinClassInstance<WeakMapObject>(cx));
+ if (!map)
+ return nullptr;
+
+ WeakSetObject* obj = NewObjectWithClassProto<WeakSetObject>(cx, proto);
+ if (!obj)
+ return nullptr;
+
+ obj->setReservedSlot(WEAKSET_MAP_SLOT, ObjectValue(*map));
+ return obj;
+}
+
+bool
+WeakSetObject::isBuiltinAdd(HandleValue add, JSContext* cx)
+{
+ JSFunction* addFn;
+ return IsFunctionObject(add, &addFn) && IsSelfHostedFunctionWithName(addFn, cx->names().WeakSet_add);
+}
+
+bool
+WeakSetObject::construct(JSContext* cx, unsigned argc, Value* vp)
+{
+ // Based on our "Set" implementation instead of the more general ES6 steps.
+ CallArgs args = CallArgsFromVp(argc, vp);
+
+ if (!ThrowIfNotConstructing(cx, args, "WeakSet"))
+ return false;
+
+ RootedObject proto(cx);
+ RootedObject newTarget(cx, &args.newTarget().toObject());
+ if (!GetPrototypeFromConstructor(cx, newTarget, &proto))
+ return false;
+
+ Rooted<WeakSetObject*> obj(cx, WeakSetObject::create(cx, proto));
+ if (!obj)
+ return false;
+
+ if (!args.get(0).isNullOrUndefined()) {
+ RootedValue iterable(cx, args[0]);
+ bool optimized = false;
+ if (!IsOptimizableInitForSet<GlobalObject::getOrCreateWeakSetPrototype, isBuiltinAdd>(cx, obj, iterable, &optimized))
+ return false;
+
+ if (optimized) {
+ RootedValue keyVal(cx);
+ RootedObject keyObject(cx);
+ RootedValue placeholder(cx, BooleanValue(true));
+ RootedObject map(cx, &obj->getReservedSlot(WEAKSET_MAP_SLOT).toObject());
+ RootedArrayObject array(cx, &iterable.toObject().as<ArrayObject>());
+ for (uint32_t index = 0; index < array->getDenseInitializedLength(); ++index) {
+ keyVal.set(array->getDenseElement(index));
+ MOZ_ASSERT(!keyVal.isMagic(JS_ELEMENTS_HOLE));
+
+ if (keyVal.isPrimitive()) {
+ UniqueChars bytes =
+ DecompileValueGenerator(cx, JSDVG_SEARCH_STACK, keyVal, nullptr);
+ if (!bytes)
+ return false;
+ JS_ReportErrorNumberLatin1(cx, GetErrorMessage, nullptr,
+ JSMSG_NOT_NONNULL_OBJECT, bytes.get());
+ return false;
+ }
+
+ keyObject = &keyVal.toObject();
+ if (!SetWeakMapEntry(cx, map, keyObject, placeholder))
+ return false;
+ }
+ } else {
+ FixedInvokeArgs<1> args2(cx);
+ args2[0].set(args[0]);
+
+ RootedValue thisv(cx, ObjectValue(*obj));
+ if (!CallSelfHostedFunction(cx, cx->names().WeakSetConstructorInit, thisv, args2, args2.rval()))
+ return false;
+ }
+ }
+
+ args.rval().setObject(*obj);
+ return true;
+}
+
+
+JSObject*
+js::InitWeakSetClass(JSContext* cx, HandleObject obj)
+{
+ return WeakSetObject::initClass(cx, obj);
+}
+
+JS_FRIEND_API(bool)
+JS_NondeterministicGetWeakSetKeys(JSContext* cx, HandleObject objArg, MutableHandleObject ret)
+{
+ RootedObject obj(cx, objArg);
+ obj = UncheckedUnwrap(obj);
+ if (!obj || !obj->is<WeakSetObject>()) {
+ ret.set(nullptr);
+ return true;
+ }
+
+ Rooted<WeakSetObject*> weakset(cx, &obj->as<WeakSetObject>());
+ if (!weakset)
+ return false;
+
+ RootedObject map(cx, &weakset->getReservedSlot(WEAKSET_MAP_SLOT).toObject());
+ return JS_NondeterministicGetWeakMapKeys(cx, map, ret);
+}