#include "builtin/WeakSetObject.h"
#include "jsapi.h"
#include "builtin/MapObject.h"
#include "js/PropertySpec.h"
#include "vm/GlobalObject.h"
#include "vm/Iteration.h"
#include "vm/JSContext.h"
#include "vm/SelfHosting.h"
#include "builtin/WeakMapObject-inl.h"
#include "vm/Interpreter-inl.h"
#include "vm/JSObject-inl.h"
#include "vm/NativeObject-inl.h"
using namespace js;
MOZ_ALWAYS_INLINE bool WeakSetObject::is(HandleValue v) {
return v.isObject() && v.toObject().is<WeakSetObject>();
}
MOZ_ALWAYS_INLINE bool WeakSetObject::add_impl(
JSContext* cx, const CallArgs& args) {
MOZ_ASSERT(is(args.thisv()));
if (!args.get(0).isObject()) {
ReportNotObjectWithName(cx, "WeakSet value", args.get(0));
return false;
}
RootedObject value(cx, &args[0].toObject());
Rooted<WeakSetObject*> map(cx, &args.thisv().toObject().as<WeakSetObject>());
if (!WeakCollectionPutEntryInternal(cx, map, value, TrueHandleValue)) {
return false;
}
args.rval().set(args.thisv());
return true;
}
bool WeakSetObject::add(JSContext* cx, unsigned argc, Value* vp) {
CallArgs args = CallArgsFromVp(argc, vp);
return CallNonGenericMethod<WeakSetObject::is, WeakSetObject::add_impl>(cx,
args);
}
MOZ_ALWAYS_INLINE bool WeakSetObject::delete_impl(
JSContext* cx, const CallArgs& args) {
MOZ_ASSERT(is(args.thisv()));
if (!args.get(0).isObject()) {
args.rval().setBoolean(false);
return true;
}
if (ObjectValueMap* map =
args.thisv().toObject().as<WeakSetObject>().getMap()) {
JSObject* value = &args[0].toObject();
if (ObjectValueMap::Ptr ptr = map->lookup(value)) {
map->remove(ptr);
args.rval().setBoolean(true);
return true;
}
}
args.rval().setBoolean(false);
return true;
}
bool WeakSetObject::delete_(JSContext* cx, unsigned argc, Value* vp) {
CallArgs args = CallArgsFromVp(argc, vp);
return CallNonGenericMethod<WeakSetObject::is, WeakSetObject::delete_impl>(
cx, args);
}
MOZ_ALWAYS_INLINE bool WeakSetObject::has_impl(
JSContext* cx, const CallArgs& args) {
MOZ_ASSERT(is(args.thisv()));
if (!args.get(0).isObject()) {
args.rval().setBoolean(false);
return true;
}
if (ObjectValueMap* map =
args.thisv().toObject().as<WeakSetObject>().getMap()) {
JSObject* value = &args[0].toObject();
if (map->has(value)) {
args.rval().setBoolean(true);
return true;
}
}
args.rval().setBoolean(false);
return true;
}
bool WeakSetObject::has(JSContext* cx, unsigned argc, Value* vp) {
CallArgs args = CallArgsFromVp(argc, vp);
return CallNonGenericMethod<WeakSetObject::is, WeakSetObject::has_impl>(cx,
args);
}
const ClassSpec WeakSetObject::classSpec_ = {
GenericCreateConstructor<WeakSetObject::construct, 0,
gc::AllocKind::FUNCTION>,
GenericCreatePrototype<WeakSetObject>,
nullptr,
nullptr,
WeakSetObject::methods,
WeakSetObject::properties,
};
const Class WeakSetObject::class_ = {
"WeakSet",
JSCLASS_HAS_PRIVATE | JSCLASS_HAS_CACHED_PROTO(JSProto_WeakSet) |
JSCLASS_BACKGROUND_FINALIZE,
&WeakCollectionObject::classOps_, &WeakSetObject::classSpec_};
const Class WeakSetObject::protoClass_ = {
js_Object_str, JSCLASS_HAS_CACHED_PROTO(JSProto_WeakSet), JS_NULL_CLASS_OPS,
&WeakSetObject::classSpec_};
const JSPropertySpec WeakSetObject::properties[] = {
JS_STRING_SYM_PS(toStringTag, "WeakSet", JSPROP_READONLY), JS_PS_END};
const JSFunctionSpec WeakSetObject::methods[] = {
JS_FN("add", add, 1, 0), JS_FN("delete", delete_, 1, 0),
JS_FN("has", has, 1, 0), JS_FS_END};
WeakSetObject* WeakSetObject::create(JSContext* cx,
HandleObject proto ) {
return NewObjectWithClassProto<WeakSetObject>(cx, proto);
}
bool WeakSetObject::isBuiltinAdd(HandleValue add) {
return IsNativeFunction(add, WeakSetObject::add);
}
bool WeakSetObject::construct(JSContext* cx, unsigned argc, Value* vp) {
CallArgs args = CallArgsFromVp(argc, vp);
if (!ThrowIfNotConstructing(cx, args, "WeakSet")) {
return false;
}
RootedObject proto(cx);
if (!GetPrototypeFromBuiltinConstructor(cx, args, JSProto_WeakSet, &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);
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()) {
ReportNotObjectWithName(cx, "WeakSet value", keyVal);
return false;
}
keyObject = &keyVal.toObject();
if (!WeakCollectionPutEntryInternal(cx, obj, keyObject,
TrueHandleValue)) {
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;
}
JS_FRIEND_API bool JS_NondeterministicGetWeakSetKeys(JSContext* cx,
HandleObject objArg,
MutableHandleObject ret) {
RootedObject obj(cx, UncheckedUnwrap(objArg));
if (!obj || !obj->is<WeakSetObject>()) {
ret.set(nullptr);
return true;
}
return WeakCollectionObject::nondeterministicGetKeys(
cx, obj.as<WeakCollectionObject>(), ret);
}