#ifndef vm_GlobalObject_h
#define vm_GlobalObject_h
#include "jsexn.h"
#include "jsnum.h"
#include "builtin/Array.h"
#include "builtin/Boolean.h"
#include "js/Vector.h"
#include "vm/ArrayBufferObject.h"
#include "vm/ErrorObject.h"
#include "vm/JSFunction.h"
#include "vm/Realm.h"
#include "vm/Runtime.h"
namespace js {
class Debugger;
class TypedObjectModuleObject;
class LexicalEnvironmentObject;
class RegExpStatics;
enum class ReferenceType;
class GlobalObject : public NativeObject {
static const unsigned APPLICATION_SLOTS = JSCLASS_GLOBAL_APPLICATION_SLOTS;
static const unsigned STANDARD_CLASS_SLOTS = JSProto_LIMIT * 2;
enum : unsigned {
EVAL = APPLICATION_SLOTS + STANDARD_CLASS_SLOTS,
THROWTYPEERROR,
LEXICAL_ENVIRONMENT,
EMPTY_GLOBAL_SCOPE,
ITERATOR_PROTO,
ARRAY_ITERATOR_PROTO,
STRING_ITERATOR_PROTO,
REGEXP_STRING_ITERATOR_PROTO,
GENERATOR_OBJECT_PROTO,
GENERATOR_FUNCTION_PROTO,
GENERATOR_FUNCTION,
ASYNC_FUNCTION_PROTO,
ASYNC_FUNCTION,
ASYNC_ITERATOR_PROTO,
ASYNC_FROM_SYNC_ITERATOR_PROTO,
ASYNC_GENERATOR,
ASYNC_GENERATOR_FUNCTION,
ASYNC_GENERATOR_PROTO,
MAP_ITERATOR_PROTO,
SET_ITERATOR_PROTO,
COLLATOR_PROTO,
NUMBER_FORMAT,
NUMBER_FORMAT_PROTO,
DATE_TIME_FORMAT,
DATE_TIME_FORMAT_PROTO,
PLURAL_RULES_PROTO,
RELATIVE_TIME_FORMAT_PROTO,
MODULE_PROTO,
IMPORT_ENTRY_PROTO,
EXPORT_ENTRY_PROTO,
REQUESTED_MODULE_PROTO,
REGEXP_STATICS,
RUNTIME_CODEGEN_ENABLED,
DEBUGGERS,
INTRINSICS,
FOR_OF_PIC_CHAIN,
WINDOW_PROXY,
GLOBAL_THIS_RESOLVED,
RESERVED_SLOTS
};
static_assert(JSCLASS_GLOBAL_SLOT_COUNT == RESERVED_SLOTS,
"global object slot counts are inconsistent");
static unsigned constructorSlot(JSProtoKey key) {
MOZ_ASSERT(key < JSProto_LIMIT);
return APPLICATION_SLOTS + key;
}
static unsigned prototypeSlot(JSProtoKey key) {
MOZ_ASSERT(key < JSProto_LIMIT);
return APPLICATION_SLOTS + JSProto_LIMIT + key;
}
public:
LexicalEnvironmentObject& lexicalEnvironment() const;
GlobalScope& emptyGlobalScope() const;
static constexpr size_t offsetOfLexicalEnvironmentSlot() {
static_assert(LEXICAL_ENVIRONMENT >= MAX_FIXED_SLOTS,
"Code assumes lexical environment is stored in dynamic slot");
return (LEXICAL_ENVIRONMENT - MAX_FIXED_SLOTS) * sizeof(Value);
}
void setOriginalEval(JSObject* evalobj) {
MOZ_ASSERT(getSlotRef(EVAL).isUndefined());
setSlot(EVAL, ObjectValue(*evalobj));
}
Value getConstructor(JSProtoKey key) const {
return getSlot(constructorSlot(key));
}
static bool skipDeselectedConstructor(JSContext* cx, JSProtoKey key);
static bool initBuiltinConstructor(JSContext* cx,
Handle<GlobalObject*> global,
JSProtoKey key, HandleObject ctor,
HandleObject proto);
private:
enum class IfClassIsDisabled { DoNothing, Throw };
static bool resolveConstructor(JSContext* cx, Handle<GlobalObject*> global,
JSProtoKey key, IfClassIsDisabled mode);
public:
static bool ensureConstructor(JSContext* cx, Handle<GlobalObject*> global,
JSProtoKey key) {
if (global->isStandardClassResolved(key)) {
return true;
}
return resolveConstructor(cx, global, key, IfClassIsDisabled::Throw);
}
static JSObject* getOrCreateConstructor(JSContext* cx, JSProtoKey key) {
MOZ_ASSERT(key != JSProto_Null);
Handle<GlobalObject*> global = cx->global();
if (!GlobalObject::ensureConstructor(cx, global, key)) {
return nullptr;
}
return &global->getConstructor(key).toObject();
}
static JSObject* getOrCreatePrototype(JSContext* cx, JSProtoKey key) {
MOZ_ASSERT(key != JSProto_Null);
Handle<GlobalObject*> global = cx->global();
if (!GlobalObject::ensureConstructor(cx, global, key)) {
return nullptr;
}
return &global->getPrototype(key).toObject();
}
JSObject* maybeGetPrototype(JSProtoKey protoKey) const {
MOZ_ASSERT(JSProto_Null < protoKey);
MOZ_ASSERT(protoKey < JSProto_LIMIT);
const Value& v = getPrototype(protoKey);
return v.isObject() ? &v.toObject() : nullptr;
}
static bool maybeResolveGlobalThis(JSContext* cx,
Handle<GlobalObject*> global,
bool* resolved);
void setConstructor(JSProtoKey key, const Value& v) {
setSlot(constructorSlot(key), v);
}
Value getPrototype(JSProtoKey key) const {
return getSlot(prototypeSlot(key));
}
void setPrototype(JSProtoKey key, const Value& value) {
setSlot(prototypeSlot(key), value);
}
bool isStandardClassResolved(JSProtoKey key) const {
Value value = getConstructor(key);
MOZ_ASSERT(value.isUndefined() || value.isObject() ||
value.isMagic(JS_OFF_THREAD_CONSTRUCTOR));
return !value.isUndefined();
}
private:
bool classIsInitialized(JSProtoKey key) const {
bool inited = !getConstructor(key).isUndefined();
MOZ_ASSERT(inited == !getPrototype(key).isUndefined());
return inited;
}
bool functionObjectClassesInitialized() const {
bool inited = classIsInitialized(JSProto_Function);
MOZ_ASSERT(inited == classIsInitialized(JSProto_Object));
return inited;
}
static GlobalObject* create(...) = delete;
friend struct ::JSRuntime;
static GlobalObject* createInternal(JSContext* cx, const Class* clasp);
public:
static GlobalObject* new_(JSContext* cx, const Class* clasp,
JSPrincipals* principals,
JS::OnNewGlobalHookOption hookOption,
const JS::RealmOptions& options);
bool shouldSplicePrototype();
static JSFunction* createConstructor(
JSContext* cx, JSNative ctor, JSAtom* name, unsigned length,
gc::AllocKind kind = gc::AllocKind::FUNCTION,
const JSJitInfo* jitInfo = nullptr);
static NativeObject* createBlankPrototype(JSContext* cx,
Handle<GlobalObject*> global,
const js::Class* clasp);
static NativeObject* createBlankPrototypeInheriting(JSContext* cx,
const js::Class* clasp,
HandleObject proto);
template <typename T>
static T* createBlankPrototype(JSContext* cx, Handle<GlobalObject*> global) {
NativeObject* res = createBlankPrototype(cx, global, &T::class_);
return res ? &res->template as<T>() : nullptr;
}
static NativeObject* getOrCreateObjectPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (global->functionObjectClassesInitialized()) {
return &global->getPrototype(JSProto_Object)
.toObject()
.as<NativeObject>();
}
if (!ensureConstructor(cx, global, JSProto_Object)) {
return nullptr;
}
return &global->getPrototype(JSProto_Object).toObject().as<NativeObject>();
}
static NativeObject* getOrCreateFunctionPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (global->functionObjectClassesInitialized()) {
return &global->getPrototype(JSProto_Function)
.toObject()
.as<NativeObject>();
}
if (!ensureConstructor(cx, global, JSProto_Object)) {
return nullptr;
}
return &global->getPrototype(JSProto_Function)
.toObject()
.as<NativeObject>();
}
static NativeObject* getOrCreateArrayPrototype(JSContext* cx,
Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_Array)) {
return nullptr;
}
return &global->getPrototype(JSProto_Array).toObject().as<NativeObject>();
}
NativeObject* maybeGetArrayPrototype() {
if (classIsInitialized(JSProto_Array)) {
return &getPrototype(JSProto_Array).toObject().as<NativeObject>();
}
return nullptr;
}
static NativeObject* getOrCreateBooleanPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_Boolean)) {
return nullptr;
}
return &global->getPrototype(JSProto_Boolean).toObject().as<NativeObject>();
}
static NativeObject* getOrCreateNumberPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_Number)) {
return nullptr;
}
return &global->getPrototype(JSProto_Number).toObject().as<NativeObject>();
}
static NativeObject* getOrCreateStringPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_String)) {
return nullptr;
}
return &global->getPrototype(JSProto_String).toObject().as<NativeObject>();
}
static NativeObject* getOrCreateSymbolPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_Symbol)) {
return nullptr;
}
return &global->getPrototype(JSProto_Symbol).toObject().as<NativeObject>();
}
static NativeObject* getOrCreateBigIntPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_BigInt)) {
return nullptr;
}
return &global->getPrototype(JSProto_BigInt).toObject().as<NativeObject>();
}
static NativeObject* getOrCreatePromisePrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_Promise)) {
return nullptr;
}
return &global->getPrototype(JSProto_Promise).toObject().as<NativeObject>();
}
static NativeObject* getOrCreateRegExpPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_RegExp)) {
return nullptr;
}
return &global->getPrototype(JSProto_RegExp).toObject().as<NativeObject>();
}
JSObject* maybeGetRegExpPrototype() {
if (classIsInitialized(JSProto_RegExp)) {
return &getPrototype(JSProto_RegExp).toObject();
}
return nullptr;
}
static NativeObject* getOrCreateSavedFramePrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_SavedFrame)) {
return nullptr;
}
return &global->getPrototype(JSProto_SavedFrame)
.toObject()
.as<NativeObject>();
}
static JSFunction* getOrCreateArrayBufferConstructor(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_ArrayBuffer)) {
return nullptr;
}
return &global->getConstructor(JSProto_ArrayBuffer)
.toObject()
.as<JSFunction>();
}
static JSObject* getOrCreateArrayBufferPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_ArrayBuffer)) {
return nullptr;
}
return &global->getPrototype(JSProto_ArrayBuffer).toObject();
}
static JSObject* getOrCreateSharedArrayBufferPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_SharedArrayBuffer)) {
return nullptr;
}
return &global->getPrototype(JSProto_SharedArrayBuffer).toObject();
}
static JSObject* getOrCreateCustomErrorPrototype(JSContext* cx,
Handle<GlobalObject*> global,
JSExnType exnType) {
JSProtoKey key = GetExceptionProtoKey(exnType);
if (!ensureConstructor(cx, global, key)) {
return nullptr;
}
return &global->getPrototype(key).toObject();
}
static JSFunction* getOrCreateErrorConstructor(JSContext* cx,
Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_Error)) {
return nullptr;
}
return &global->getConstructor(JSProto_Error).toObject().as<JSFunction>();
}
static JSObject* getOrCreateErrorPrototype(JSContext* cx,
Handle<GlobalObject*> global) {
return getOrCreateCustomErrorPrototype(cx, global, JSEXN_ERR);
}
static NativeObject* getOrCreateSetPrototype(JSContext* cx,
Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_Set)) {
return nullptr;
}
return &global->getPrototype(JSProto_Set).toObject().as<NativeObject>();
}
static NativeObject* getOrCreateWeakSetPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_WeakSet)) {
return nullptr;
}
return &global->getPrototype(JSProto_WeakSet).toObject().as<NativeObject>();
}
static JSObject* getOrCreateIntlObject(JSContext* cx,
Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, APPLICATION_SLOTS + JSProto_Intl,
initIntlObject);
}
static JSObject* getOrCreateTypedObjectModule(JSContext* cx,
Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global,
APPLICATION_SLOTS + JSProto_TypedObject,
initTypedObjectModule);
}
static TypeDescr* getOrCreateScalarTypeDescr(JSContext* cx,
Handle<GlobalObject*> global,
Scalar::Type scalarType);
static TypeDescr* getOrCreateReferenceTypeDescr(JSContext* cx,
Handle<GlobalObject*> global,
ReferenceType type);
TypedObjectModuleObject& getTypedObjectModule() const;
static JSObject* getOrCreateCollatorPrototype(JSContext* cx,
Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, COLLATOR_PROTO, initIntlObject);
}
static JSFunction* getOrCreateNumberFormatConstructor(
JSContext* cx, Handle<GlobalObject*> global) {
JSObject* obj =
getOrCreateObject(cx, global, NUMBER_FORMAT, initIntlObject);
return obj ? &obj->as<JSFunction>() : nullptr;
}
static JSObject* getOrCreateNumberFormatPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, NUMBER_FORMAT_PROTO, initIntlObject);
}
static JSFunction* getOrCreateDateTimeFormatConstructor(
JSContext* cx, Handle<GlobalObject*> global) {
JSObject* obj =
getOrCreateObject(cx, global, DATE_TIME_FORMAT, initIntlObject);
return obj ? &obj->as<JSFunction>() : nullptr;
}
static JSObject* getOrCreateDateTimeFormatPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, DATE_TIME_FORMAT_PROTO,
initIntlObject);
}
static JSObject* getOrCreatePluralRulesPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, PLURAL_RULES_PROTO, initIntlObject);
}
static JSObject* getOrCreateRelativeTimeFormatPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, RELATIVE_TIME_FORMAT_PROTO,
initIntlObject);
}
static bool ensureModulePrototypesCreated(JSContext* cx,
Handle<GlobalObject*> global);
static JSObject* getOrCreateModulePrototype(JSContext* cx,
Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, MODULE_PROTO, initModuleProto);
}
static JSObject* getOrCreateImportEntryPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, IMPORT_ENTRY_PROTO,
initImportEntryProto);
}
static JSObject* getOrCreateExportEntryPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, EXPORT_ENTRY_PROTO,
initExportEntryProto);
}
static JSObject* getOrCreateRequestedModulePrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, REQUESTED_MODULE_PROTO,
initRequestedModuleProto);
}
static JSFunction* getOrCreateTypedArrayConstructor(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_TypedArray)) {
return nullptr;
}
return &global->getConstructor(JSProto_TypedArray)
.toObject()
.as<JSFunction>();
}
static JSObject* getOrCreateTypedArrayPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_TypedArray)) {
return nullptr;
}
return &global->getPrototype(JSProto_TypedArray).toObject();
}
private:
typedef bool (*ObjectInitOp)(JSContext* cx, Handle<GlobalObject*> global);
static JSObject* getOrCreateObject(JSContext* cx,
Handle<GlobalObject*> global,
unsigned slot, ObjectInitOp init) {
Value v = global->getSlotRef(slot);
if (v.isObject()) {
return &v.toObject();
}
return createObject(cx, global, slot, init);
}
static JSObject* createObject(JSContext* cx, Handle<GlobalObject*> global,
unsigned slot, ObjectInitOp init);
public:
static NativeObject* getOrCreateIteratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(
getOrCreateObject(cx, global, ITERATOR_PROTO, initIteratorProto));
}
static NativeObject* getOrCreateArrayIteratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(getOrCreateObject(cx, global, ARRAY_ITERATOR_PROTO,
initArrayIteratorProto));
}
NativeObject* maybeGetArrayIteratorPrototype() {
Value v = getSlotRef(ARRAY_ITERATOR_PROTO);
if (v.isObject()) {
return &v.toObject().as<NativeObject>();
}
return nullptr;
}
static NativeObject* getOrCreateStringIteratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(getOrCreateObject(
cx, global, STRING_ITERATOR_PROTO, initStringIteratorProto));
}
static NativeObject* getOrCreateRegExpStringIteratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(getOrCreateObject(cx, global,
REGEXP_STRING_ITERATOR_PROTO,
initRegExpStringIteratorProto));
}
static NativeObject* getOrCreateGeneratorObjectPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(
getOrCreateObject(cx, global, GENERATOR_OBJECT_PROTO, initGenerators));
}
static JSObject* getOrCreateGeneratorFunctionPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, GENERATOR_FUNCTION_PROTO,
initGenerators);
}
static JSObject* getOrCreateGeneratorFunction(JSContext* cx,
Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, GENERATOR_FUNCTION, initGenerators);
}
static NativeObject* getOrCreateAsyncFunctionPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(
getOrCreateObject(cx, global, ASYNC_FUNCTION_PROTO, initAsyncFunction));
}
static JSObject* getOrCreateAsyncFunction(JSContext* cx,
Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, ASYNC_FUNCTION, initAsyncFunction);
}
static NativeObject* getOrCreateAsyncIteratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(getOrCreateObject(cx, global, ASYNC_ITERATOR_PROTO,
initAsyncGenerators));
}
static NativeObject* getOrCreateAsyncFromSyncIteratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(getOrCreateObject(
cx, global, ASYNC_FROM_SYNC_ITERATOR_PROTO, initAsyncGenerators));
}
static NativeObject* getOrCreateAsyncGenerator(JSContext* cx,
Handle<GlobalObject*> global) {
return MaybeNativeObject(
getOrCreateObject(cx, global, ASYNC_GENERATOR, initAsyncGenerators));
}
static JSObject* getOrCreateAsyncGeneratorFunction(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, ASYNC_GENERATOR_FUNCTION,
initAsyncGenerators);
}
static NativeObject* getOrCreateAsyncGeneratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return MaybeNativeObject(getOrCreateObject(
cx, global, ASYNC_GENERATOR_PROTO, initAsyncGenerators));
}
static JSObject* getOrCreateMapIteratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, MAP_ITERATOR_PROTO,
initMapIteratorProto);
}
static JSObject* getOrCreateSetIteratorPrototype(
JSContext* cx, Handle<GlobalObject*> global) {
return getOrCreateObject(cx, global, SET_ITERATOR_PROTO,
initSetIteratorProto);
}
static JSObject* getOrCreateDataViewPrototype(JSContext* cx,
Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_DataView)) {
return nullptr;
}
return &global->getPrototype(JSProto_DataView).toObject();
}
static JSFunction* getOrCreatePromiseConstructor(
JSContext* cx, Handle<GlobalObject*> global) {
if (!ensureConstructor(cx, global, JSProto_Promise)) {
return nullptr;
}
return &global->getConstructor(JSProto_Promise).toObject().as<JSFunction>();
}
static NativeObject* getIntrinsicsHolder(JSContext* cx,
Handle<GlobalObject*> global);
bool maybeExistingIntrinsicValue(PropertyName* name, Value* vp) {
Value slot = getReservedSlot(INTRINSICS);
if (slot.isUndefined()) {
*vp = UndefinedValue();
return false;
}
NativeObject* holder = &slot.toObject().as<NativeObject>();
Shape* shape = holder->lookupPure(name);
if (!shape) {
*vp = UndefinedValue();
return false;
}
*vp = holder->getSlot(shape->slot());
return true;
}
Value existingIntrinsicValue(PropertyName* name) {
Value val;
mozilla::DebugOnly<bool> exists = maybeExistingIntrinsicValue(name, &val);
MOZ_ASSERT(exists, "intrinsic must already have been added to holder");
return val;
}
static bool maybeGetIntrinsicValue(JSContext* cx,
Handle<GlobalObject*> global,
Handle<PropertyName*> name,
MutableHandleValue vp, bool* exists) {
NativeObject* holder = getIntrinsicsHolder(cx, global);
if (!holder) {
return false;
}
if (Shape* shape = holder->lookup(cx, name)) {
vp.set(holder->getSlot(shape->slot()));
*exists = true;
} else {
*exists = false;
}
return true;
}
static bool getIntrinsicValue(JSContext* cx, Handle<GlobalObject*> global,
HandlePropertyName name,
MutableHandleValue value) {
bool exists = false;
if (!GlobalObject::maybeGetIntrinsicValue(cx, global, name, value,
&exists)) {
return false;
}
if (exists) {
return true;
}
if (!cx->runtime()->cloneSelfHostedValue(cx, name, value)) {
return false;
}
return GlobalObject::addIntrinsicValue(cx, global, name, value);
}
static bool addIntrinsicValue(JSContext* cx, Handle<GlobalObject*> global,
HandlePropertyName name, HandleValue value);
static inline bool setIntrinsicValue(JSContext* cx,
Handle<GlobalObject*> global,
HandlePropertyName name,
HandleValue value);
static bool getSelfHostedFunction(JSContext* cx, Handle<GlobalObject*> global,
HandlePropertyName selfHostedName,
HandleAtom name, unsigned nargs,
MutableHandleValue funVal);
bool hasRegExpStatics() const;
static RegExpStatics* getRegExpStatics(JSContext* cx,
Handle<GlobalObject*> global);
RegExpStatics* getAlreadyCreatedRegExpStatics() const;
static JSObject* getOrCreateThrowTypeError(JSContext* cx,
Handle<GlobalObject*> global);
static bool isRuntimeCodeGenEnabled(JSContext* cx, HandleValue code,
Handle<GlobalObject*> global);
static bool getOrCreateEval(JSContext* cx, Handle<GlobalObject*> global,
MutableHandleObject eval);
bool valueIsEval(const Value& val);
static bool initIteratorProto(JSContext* cx, Handle<GlobalObject*> global);
static bool initArrayIteratorProto(JSContext* cx,
Handle<GlobalObject*> global);
static bool initStringIteratorProto(JSContext* cx,
Handle<GlobalObject*> global);
static bool initRegExpStringIteratorProto(JSContext* cx,
Handle<GlobalObject*> global);
static bool initGenerators(JSContext* cx, Handle<GlobalObject*> global);
static bool initAsyncFunction(JSContext* cx, Handle<GlobalObject*> global);
static bool initAsyncGenerators(JSContext* cx, Handle<GlobalObject*> global);
static bool initMapIteratorProto(JSContext* cx, Handle<GlobalObject*> global);
static bool initSetIteratorProto(JSContext* cx, Handle<GlobalObject*> global);
static bool initIntlObject(JSContext* cx, Handle<GlobalObject*> global);
static bool initModuleProto(JSContext* cx, Handle<GlobalObject*> global);
static bool initImportEntryProto(JSContext* cx, Handle<GlobalObject*> global);
static bool initExportEntryProto(JSContext* cx, Handle<GlobalObject*> global);
static bool initRequestedModuleProto(JSContext* cx,
Handle<GlobalObject*> global);
static bool initTypedObjectModule(JSContext* cx,
Handle<GlobalObject*> global);
static bool initStandardClasses(JSContext* cx, Handle<GlobalObject*> global);
static bool initSelfHostingBuiltins(JSContext* cx,
Handle<GlobalObject*> global,
const JSFunctionSpec* builtins);
typedef js::Vector<js::ReadBarriered<js::Debugger*>, 0, js::SystemAllocPolicy>
DebuggerVector;
DebuggerVector* getDebuggers() const;
static DebuggerVector* getOrCreateDebuggers(JSContext* cx,
Handle<GlobalObject*> global);
inline NativeObject* getForOfPICObject() {
Value forOfPIC = getReservedSlot(FOR_OF_PIC_CHAIN);
if (forOfPIC.isUndefined()) {
return nullptr;
}
return &forOfPIC.toObject().as<NativeObject>();
}
static NativeObject* getOrCreateForOfPICObject(JSContext* cx,
Handle<GlobalObject*> global);
JSObject* windowProxy() const {
return &getReservedSlot(WINDOW_PROXY).toObject();
}
JSObject* maybeWindowProxy() const {
Value v = getReservedSlot(WINDOW_PROXY);
MOZ_ASSERT(v.isObject() || v.isUndefined());
return v.isObject() ? &v.toObject() : nullptr;
}
void setWindowProxy(JSObject* windowProxy) {
setReservedSlot(WINDOW_PROXY, ObjectValue(*windowProxy));
}
struct OffThreadPlaceholderObject : public NativeObject {
static const int32_t SlotIndexSlot = 0;
static const Class class_;
static OffThreadPlaceholderObject* New(JSContext* cx, unsigned slot);
inline int32_t getSlotIndex() const;
};
static bool isOffThreadPrototypePlaceholder(JSObject* obj) {
return obj->is<OffThreadPlaceholderObject>();
}
JSObject* getPrototypeForOffThreadPlaceholder(JSObject* placeholder);
private:
static bool resolveOffThreadConstructor(JSContext* cx,
Handle<GlobalObject*> global,
JSProtoKey key);
static JSObject* createOffThreadObject(JSContext* cx,
Handle<GlobalObject*> global,
unsigned slot);
};
extern bool LinkConstructorAndPrototype(
JSContext* cx, JSObject* ctor, JSObject* proto,
unsigned prototypeAttrs = JSPROP_PERMANENT | JSPROP_READONLY,
unsigned constructorAttrs = 0);
extern bool DefinePropertiesAndFunctions(JSContext* cx, HandleObject obj,
const JSPropertySpec* ps,
const JSFunctionSpec* fs);
typedef HashSet<GlobalObject*, DefaultHasher<GlobalObject*>, SystemAllocPolicy>
GlobalObjectSet;
extern bool DefineToStringTag(JSContext* cx, HandleObject obj, JSAtom* tag);
template <JSNative ctor, unsigned length, gc::AllocKind kind,
const JSJitInfo* jitInfo = nullptr>
JSObject* GenericCreateConstructor(JSContext* cx, JSProtoKey key) {
PropertyName* name = (&cx->names().Null)[key];
return GlobalObject::createConstructor(cx, ctor, name, length, kind, jitInfo);
}
template <typename T>
JSObject* GenericCreatePrototype(JSContext* cx, JSProtoKey key) {
static_assert(
!std::is_same<T, PlainObject>::value,
"creating Object.prototype is very special and isn't handled here");
MOZ_ASSERT(&T::class_ == ProtoKeyToClass(key),
"type mismatch--probably too much copy/paste in your ClassSpec");
MOZ_ASSERT(
InheritanceProtoKeyForStandardClass(key) == JSProto_Object,
"subclasses (of anything but Object) can't use GenericCreatePrototype");
return GlobalObject::createBlankPrototype(cx, cx->global(), &T::protoClass_);
}
inline JSProtoKey StandardProtoKeyOrNull(const JSObject* obj) {
return JSCLASS_CACHED_PROTO_KEY(obj->getClass());
}
JSObject* NewSingletonObjectWithFunctionPrototype(JSContext* cx,
Handle<GlobalObject*> global);
}
template <>
inline bool JSObject::is<js::GlobalObject>() const {
return !!(getClass()->flags & JSCLASS_IS_GLOBAL);
}
#endif