summaryrefslogtreecommitdiffstats
path: root/js/src/vm/ArgumentsObject.cpp
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-19 00:47:55 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-19 00:47:55 +0000
commit26a029d407be480d791972afb5975cf62c9360a6 (patch)
treef435a8308119effd964b339f76abb83a57c29483 /js/src/vm/ArgumentsObject.cpp
parentInitial commit. (diff)
downloadfirefox-26a029d407be480d791972afb5975cf62c9360a6.tar.xz
firefox-26a029d407be480d791972afb5975cf62c9360a6.zip
Adding upstream version 124.0.1.upstream/124.0.1
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'js/src/vm/ArgumentsObject.cpp')
-rw-r--r--js/src/vm/ArgumentsObject.cpp1184
1 files changed, 1184 insertions, 0 deletions
diff --git a/js/src/vm/ArgumentsObject.cpp b/js/src/vm/ArgumentsObject.cpp
new file mode 100644
index 0000000000..1bd2bb144c
--- /dev/null
+++ b/js/src/vm/ArgumentsObject.cpp
@@ -0,0 +1,1184 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * vim: set ts=8 sts=2 et sw=2 tw=80:
+ * 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 "vm/ArgumentsObject-inl.h"
+
+#include "mozilla/Maybe.h"
+#include "mozilla/PodOperations.h"
+
+#include <algorithm>
+
+#include "gc/GCContext.h"
+#include "jit/CalleeToken.h"
+#include "jit/JitFrames.h"
+#include "util/BitArray.h"
+#include "vm/GlobalObject.h"
+#include "vm/Stack.h"
+
+#include "gc/Nursery-inl.h"
+#include "vm/FrameIter-inl.h" // js::FrameIter::unaliasedForEachActual
+#include "vm/NativeObject-inl.h"
+#include "vm/Stack-inl.h"
+
+using namespace js;
+
+/* static */
+size_t RareArgumentsData::bytesRequired(size_t numActuals) {
+ size_t extraBytes = NumWordsForBitArrayOfLength(numActuals) * sizeof(size_t);
+ return offsetof(RareArgumentsData, deletedBits_) + extraBytes;
+}
+
+/* static */
+RareArgumentsData* RareArgumentsData::create(JSContext* cx,
+ ArgumentsObject* obj) {
+ size_t bytes = RareArgumentsData::bytesRequired(obj->initialLength());
+
+ uint8_t* data = AllocateCellBuffer<uint8_t>(cx, obj, bytes);
+ if (!data) {
+ return nullptr;
+ }
+
+ mozilla::PodZero(data, bytes);
+
+ AddCellMemory(obj, bytes, MemoryUse::RareArgumentsData);
+
+ return new (data) RareArgumentsData();
+}
+
+ArgumentsData::ArgumentsData(uint32_t numArgs) : args(numArgs) {
+ // |args| must be the last field.
+ static_assert(offsetof(ArgumentsData, args) + sizeof(args) ==
+ sizeof(ArgumentsData));
+}
+
+bool ArgumentsObject::createRareData(JSContext* cx) {
+ MOZ_ASSERT(!data()->rareData);
+
+ RareArgumentsData* rareData = RareArgumentsData::create(cx, this);
+ if (!rareData) {
+ return false;
+ }
+
+ data()->rareData = rareData;
+ markElementOverridden();
+ return true;
+}
+
+bool ArgumentsObject::markElementDeleted(JSContext* cx, uint32_t i) {
+ RareArgumentsData* data = getOrCreateRareData(cx);
+ if (!data) {
+ return false;
+ }
+
+ data->markElementDeleted(initialLength(), i);
+ return true;
+}
+
+/* static */
+void ArgumentsObject::MaybeForwardToCallObject(AbstractFramePtr frame,
+ ArgumentsObject* obj,
+ ArgumentsData* data) {
+ JSScript* script = frame.script();
+ if (frame.callee()->needsCallObject() && script->argsObjAliasesFormals()) {
+ obj->initFixedSlot(MAYBE_CALL_SLOT, ObjectValue(frame.callObj()));
+ for (PositionalFormalParameterIter fi(script); fi; fi++) {
+ if (fi.closedOver()) {
+ data->args.setElement(obj, fi.argumentSlot(),
+ MagicEnvSlotValue(fi.location().slot()));
+ obj->markArgumentForwarded();
+ }
+ }
+ }
+}
+
+/* static */
+void ArgumentsObject::MaybeForwardToCallObject(JSFunction* callee,
+ JSObject* callObj,
+ ArgumentsObject* obj,
+ ArgumentsData* data) {
+ JSScript* script = callee->nonLazyScript();
+ if (callee->needsCallObject() && script->argsObjAliasesFormals()) {
+ MOZ_ASSERT(callObj && callObj->is<CallObject>());
+ obj->initFixedSlot(MAYBE_CALL_SLOT, ObjectValue(*callObj));
+ for (PositionalFormalParameterIter fi(script); fi; fi++) {
+ if (fi.closedOver()) {
+ data->args.setElement(obj, fi.argumentSlot(),
+ MagicEnvSlotValue(fi.location().slot()));
+ obj->markArgumentForwarded();
+ }
+ }
+ }
+}
+
+struct CopyFrameArgs {
+ AbstractFramePtr frame_;
+
+ explicit CopyFrameArgs(AbstractFramePtr frame) : frame_(frame) {}
+
+ void copyActualArgs(ArgumentsObject* owner, GCOwnedArray<Value>& args,
+ unsigned numActuals) const {
+ MOZ_ASSERT_IF(frame_.isInterpreterFrame(),
+ !frame_.asInterpreterFrame()->runningInJit());
+
+ // Copy arguments.
+ Value* src = frame_.argv();
+ Value* end = src + numActuals;
+ args.withOwner(owner, [&](auto& args) {
+ auto* dst = args.begin();
+ while (src != end) {
+ (dst++)->init(*src++);
+ }
+ });
+ }
+
+ /*
+ * If a call object exists and the arguments object aliases formals, the
+ * call object is the canonical location for formals.
+ */
+ void maybeForwardToCallObject(ArgumentsObject* obj, ArgumentsData* data) {
+ ArgumentsObject::MaybeForwardToCallObject(frame_, obj, data);
+ }
+};
+
+struct CopyJitFrameArgs {
+ jit::JitFrameLayout* frame_;
+ HandleObject callObj_;
+
+ CopyJitFrameArgs(jit::JitFrameLayout* frame, HandleObject callObj)
+ : frame_(frame), callObj_(callObj) {}
+
+ void copyActualArgs(ArgumentsObject* owner, GCOwnedArray<Value>& args,
+ unsigned numActuals) const {
+ MOZ_ASSERT(frame_->numActualArgs() == numActuals);
+
+ Value* src = frame_->actualArgs();
+ Value* end = src + numActuals;
+ args.withOwner(owner, [&](auto& args) {
+ auto* dst = args.begin();
+ while (src != end) {
+ (dst++)->init(*src++);
+ }
+ });
+ }
+
+ /*
+ * If a call object exists and the arguments object aliases formals, the
+ * call object is the canonical location for formals.
+ */
+ void maybeForwardToCallObject(ArgumentsObject* obj, ArgumentsData* data) {
+ JSFunction* callee = jit::CalleeTokenToFunction(frame_->calleeToken());
+ ArgumentsObject::MaybeForwardToCallObject(callee, callObj_, obj, data);
+ }
+};
+
+struct CopyScriptFrameIterArgs {
+ ScriptFrameIter& iter_;
+ RootedValueVector actualArgs_;
+
+ explicit CopyScriptFrameIterArgs(JSContext* cx, ScriptFrameIter& iter)
+ : iter_(iter), actualArgs_(cx) {}
+
+ // Used to copy arguments to actualArgs_ to simplify copyArgs and
+ // ArgumentsObject allocation.
+ [[nodiscard]] bool init(JSContext* cx) {
+ unsigned numActuals = iter_.numActualArgs();
+ if (!actualArgs_.reserve(numActuals)) {
+ return false;
+ }
+
+ // Append actual arguments.
+ iter_.unaliasedForEachActual(
+ cx, [this](const Value& v) { actualArgs_.infallibleAppend(v); });
+ MOZ_RELEASE_ASSERT(actualArgs_.length() == numActuals);
+ return true;
+ }
+
+ void copyActualArgs(ArgumentsObject* owner, GCOwnedArray<Value>& args,
+ unsigned numActuals) const {
+ MOZ_ASSERT(actualArgs_.length() == numActuals);
+
+ args.withOwner(owner, [&](auto& args) {
+ auto* dst = args.begin();
+ for (Value v : actualArgs_) {
+ (dst++)->init(v);
+ }
+ });
+ }
+
+ /*
+ * Ion frames are copying every argument onto the stack, other locations are
+ * invalid.
+ */
+ void maybeForwardToCallObject(ArgumentsObject* obj, ArgumentsData* data) {
+ if (!iter_.isIon()) {
+ ArgumentsObject::MaybeForwardToCallObject(iter_.abstractFramePtr(), obj,
+ data);
+ }
+ }
+};
+
+struct CopyInlinedArgs {
+ HandleValueArray args_;
+ HandleObject callObj_;
+ HandleFunction callee_;
+
+ CopyInlinedArgs(HandleValueArray args, HandleObject callObj,
+ HandleFunction callee)
+ : args_(args), callObj_(callObj), callee_(callee) {}
+
+ void copyActualArgs(ArgumentsObject* owner, GCOwnedArray<Value>& args,
+ unsigned numActuals) const {
+ MOZ_ASSERT(numActuals <= args_.length());
+
+ args.withOwner(owner, [&](auto& args) {
+ auto* dst = args.begin();
+ for (uint32_t i = 0; i < numActuals; i++) {
+ (dst++)->init(args_[i]);
+ }
+ });
+ }
+
+ /*
+ * If a call object exists and the arguments object aliases formals, the
+ * call object is the canonical location for formals.
+ */
+ void maybeForwardToCallObject(ArgumentsObject* obj, ArgumentsData* data) {
+ ArgumentsObject::MaybeForwardToCallObject(callee_, callObj_, obj, data);
+ }
+};
+
+ArgumentsObject* ArgumentsObject::createTemplateObject(JSContext* cx,
+ bool mapped) {
+ const JSClass* clasp = mapped ? &MappedArgumentsObject::class_
+ : &UnmappedArgumentsObject::class_;
+
+ RootedObject proto(cx, &cx->global()->getObjectPrototype());
+
+ constexpr ObjectFlags objectFlags = {ObjectFlag::Indexed};
+ Rooted<SharedShape*> shape(cx, SharedShape::getInitialShape(
+ cx, clasp, cx->realm(), TaggedProto(proto),
+ FINALIZE_KIND, objectFlags));
+ if (!shape) {
+ return nullptr;
+ }
+
+ AutoSetNewObjectMetadata metadata(cx);
+ auto* obj = NativeObject::create<ArgumentsObject>(cx, FINALIZE_KIND,
+ gc::Heap::Tenured, shape);
+ if (!obj) {
+ return nullptr;
+ }
+
+ obj->initFixedSlot(ArgumentsObject::DATA_SLOT, PrivateValue(nullptr));
+ return obj;
+}
+
+ArgumentsObject* GlobalObject::maybeArgumentsTemplateObject(bool mapped) const {
+ return mapped ? data().mappedArgumentsTemplate
+ : data().unmappedArgumentsTemplate;
+}
+
+/* static */
+ArgumentsObject* GlobalObject::getOrCreateArgumentsTemplateObject(JSContext* cx,
+ bool mapped) {
+ GlobalObjectData& data = cx->global()->data();
+ HeapPtr<ArgumentsObject*>& obj =
+ mapped ? data.mappedArgumentsTemplate : data.unmappedArgumentsTemplate;
+
+ ArgumentsObject* templateObj = obj;
+ if (templateObj) {
+ return templateObj;
+ }
+
+ templateObj = ArgumentsObject::createTemplateObject(cx, mapped);
+ if (!templateObj) {
+ return nullptr;
+ }
+
+ obj.init(templateObj);
+ return templateObj;
+}
+
+template <typename CopyArgs>
+/* static */
+ArgumentsObject* ArgumentsObject::create(JSContext* cx, HandleFunction callee,
+ unsigned numActuals, CopyArgs& copy) {
+ // Self-hosted code should use the more efficient ArgumentsLength and
+ // GetArgument intrinsics instead of `arguments`.
+ MOZ_ASSERT(!callee->isSelfHostedBuiltin());
+
+ bool mapped = callee->baseScript()->hasMappedArgsObj();
+ ArgumentsObject* templateObj =
+ GlobalObject::getOrCreateArgumentsTemplateObject(cx, mapped);
+ if (!templateObj) {
+ return nullptr;
+ }
+
+ Rooted<SharedShape*> shape(cx, templateObj->sharedShape());
+
+ unsigned numFormals = callee->nargs();
+ unsigned numArgs = std::max(numActuals, numFormals);
+ unsigned numBytes = ArgumentsData::bytesRequired(numArgs);
+
+ AutoSetNewObjectMetadata metadata(cx);
+ auto* obj = NativeObject::create<ArgumentsObject>(cx, FINALIZE_KIND,
+ gc::Heap::Default, shape);
+ if (!obj) {
+ return nullptr;
+ }
+
+ ArgumentsData* data = reinterpret_cast<ArgumentsData*>(
+ AllocateCellBuffer<uint8_t>(cx, obj, numBytes));
+ if (!data) {
+ // Make the object safe for GC.
+ obj->initFixedSlot(DATA_SLOT, PrivateValue(nullptr));
+ return nullptr;
+ }
+
+ new (data) ArgumentsData(numArgs);
+
+ InitReservedSlot(obj, DATA_SLOT, data, numBytes, MemoryUse::ArgumentsData);
+ obj->initFixedSlot(CALLEE_SLOT, ObjectValue(*callee));
+ obj->initFixedSlot(INITIAL_LENGTH_SLOT,
+ Int32Value(numActuals << PACKED_BITS_COUNT));
+
+ // Copy [0, numActuals) into data->args.
+ copy.copyActualArgs(obj, data->args, numActuals);
+
+ // Fill in missing arguments with |undefined|.
+ data->args.withOwner(obj, [&](auto& args) {
+ for (size_t i = numActuals; i < numArgs; i++) {
+ args[i].init(UndefinedValue());
+ }
+ });
+
+ copy.maybeForwardToCallObject(obj, data);
+
+ MOZ_ASSERT(obj->initialLength() == numActuals);
+ MOZ_ASSERT(!obj->hasOverriddenLength());
+ return obj;
+}
+
+ArgumentsObject* ArgumentsObject::createExpected(JSContext* cx,
+ AbstractFramePtr frame) {
+ MOZ_ASSERT(frame.script()->needsArgsObj());
+ RootedFunction callee(cx, frame.callee());
+ CopyFrameArgs copy(frame);
+ ArgumentsObject* argsobj = create(cx, callee, frame.numActualArgs(), copy);
+ if (!argsobj) {
+ return nullptr;
+ }
+
+ frame.initArgsObj(*argsobj);
+ return argsobj;
+}
+
+ArgumentsObject* ArgumentsObject::createUnexpected(JSContext* cx,
+ ScriptFrameIter& iter) {
+ RootedFunction callee(cx, iter.callee(cx));
+ CopyScriptFrameIterArgs copy(cx, iter);
+ if (!copy.init(cx)) {
+ return nullptr;
+ }
+ return create(cx, callee, iter.numActualArgs(), copy);
+}
+
+ArgumentsObject* ArgumentsObject::createUnexpected(JSContext* cx,
+ AbstractFramePtr frame) {
+ RootedFunction callee(cx, frame.callee());
+ CopyFrameArgs copy(frame);
+ return create(cx, callee, frame.numActualArgs(), copy);
+}
+
+ArgumentsObject* ArgumentsObject::createForIon(JSContext* cx,
+ jit::JitFrameLayout* frame,
+ HandleObject scopeChain) {
+ jit::CalleeToken token = frame->calleeToken();
+ MOZ_ASSERT(jit::CalleeTokenIsFunction(token));
+ RootedFunction callee(cx, jit::CalleeTokenToFunction(token));
+ RootedObject callObj(
+ cx, scopeChain->is<CallObject>() ? scopeChain.get() : nullptr);
+ CopyJitFrameArgs copy(frame, callObj);
+ return create(cx, callee, frame->numActualArgs(), copy);
+}
+
+/* static */
+ArgumentsObject* ArgumentsObject::createFromValueArray(
+ JSContext* cx, HandleValueArray argsArray, HandleFunction callee,
+ HandleObject scopeChain, uint32_t numActuals) {
+ MOZ_ASSERT(numActuals <= MaxInlinedArgs);
+ RootedObject callObj(
+ cx, scopeChain->is<CallObject>() ? scopeChain.get() : nullptr);
+ CopyInlinedArgs copy(argsArray, callObj, callee);
+ return create(cx, callee, numActuals, copy);
+}
+
+/* static */
+ArgumentsObject* ArgumentsObject::createForInlinedIon(JSContext* cx,
+ Value* args,
+ HandleFunction callee,
+ HandleObject scopeChain,
+ uint32_t numActuals) {
+ RootedExternalValueArray rootedArgs(cx, numActuals, args);
+ HandleValueArray argsArray =
+ HandleValueArray::fromMarkedLocation(numActuals, args);
+
+ return createFromValueArray(cx, argsArray, callee, scopeChain, numActuals);
+}
+
+template <typename CopyArgs>
+/* static */
+ArgumentsObject* ArgumentsObject::finishPure(
+ JSContext* cx, ArgumentsObject* obj, JSFunction* callee, JSObject* callObj,
+ unsigned numActuals, CopyArgs& copy) {
+ unsigned numFormals = callee->nargs();
+ unsigned numArgs = std::max(numActuals, numFormals);
+ unsigned numBytes = ArgumentsData::bytesRequired(numArgs);
+
+ ArgumentsData* data = reinterpret_cast<ArgumentsData*>(
+ AllocateCellBuffer<uint8_t>(cx, obj, numBytes));
+ if (!data) {
+ // Make the object safe for GC. Don't report OOM, the slow path will
+ // retry the allocation.
+ cx->recoverFromOutOfMemory();
+ obj->initFixedSlot(DATA_SLOT, PrivateValue(nullptr));
+ return nullptr;
+ }
+
+ new (data) ArgumentsData(numArgs);
+
+ obj->initFixedSlot(INITIAL_LENGTH_SLOT,
+ Int32Value(numActuals << PACKED_BITS_COUNT));
+ obj->initFixedSlot(DATA_SLOT, PrivateValue(data));
+ AddCellMemory(obj, numBytes, MemoryUse::ArgumentsData);
+ obj->initFixedSlot(MAYBE_CALL_SLOT, UndefinedValue());
+ obj->initFixedSlot(CALLEE_SLOT, ObjectValue(*callee));
+
+ copy.copyActualArgs(obj, data->args, numActuals);
+
+ // Fill in missing arguments with |undefined|.
+ data->args.withOwner(obj, [&](auto& args) {
+ for (size_t i = numActuals; i < numArgs; i++) {
+ args[i].init(UndefinedValue());
+ }
+ });
+
+ if (callObj && callee->needsCallObject()) {
+ copy.maybeForwardToCallObject(obj, data);
+ }
+
+ MOZ_ASSERT(obj->initialLength() == numActuals);
+ MOZ_ASSERT(!obj->hasOverriddenLength());
+ return obj;
+}
+
+/* static */
+ArgumentsObject* ArgumentsObject::finishForIonPure(JSContext* cx,
+ jit::JitFrameLayout* frame,
+ JSObject* scopeChain,
+ ArgumentsObject* obj) {
+ // JIT code calls this directly (no callVM), because it's faster, so we're
+ // not allowed to GC in here.
+ AutoUnsafeCallWithABI unsafe;
+
+ JSFunction* callee = jit::CalleeTokenToFunction(frame->calleeToken());
+ RootedObject callObj(cx, scopeChain->is<CallObject>() ? scopeChain : nullptr);
+ CopyJitFrameArgs copy(frame, callObj);
+
+ unsigned numActuals = frame->numActualArgs();
+
+ return finishPure(cx, obj, callee, callObj, numActuals, copy);
+}
+
+/* static */
+ArgumentsObject* ArgumentsObject::finishInlineForIonPure(
+ JSContext* cx, JSObject* rawCallObj, JSFunction* rawCallee, Value* args,
+ uint32_t numActuals, ArgumentsObject* obj) {
+ // JIT code calls this directly (no callVM), because it's faster, so we're
+ // not allowed to GC in here.
+ AutoUnsafeCallWithABI unsafe;
+
+ MOZ_ASSERT(numActuals <= MaxInlinedArgs);
+
+ RootedObject callObj(cx, rawCallObj);
+ RootedFunction callee(cx, rawCallee);
+ RootedExternalValueArray rootedArgs(cx, numActuals, args);
+ HandleValueArray argsArray =
+ HandleValueArray::fromMarkedLocation(numActuals, args);
+
+ CopyInlinedArgs copy(argsArray, callObj, callee);
+
+ return finishPure(cx, obj, callee, callObj, numActuals, copy);
+}
+
+/* static */
+bool ArgumentsObject::obj_delProperty(JSContext* cx, HandleObject obj,
+ HandleId id, ObjectOpResult& result) {
+ ArgumentsObject& argsobj = obj->as<ArgumentsObject>();
+ if (id.isInt()) {
+ unsigned arg = unsigned(id.toInt());
+ if (argsobj.isElement(arg)) {
+ if (!argsobj.markElementDeleted(cx, arg)) {
+ return false;
+ }
+ }
+ } else if (id.isAtom(cx->names().length)) {
+ argsobj.markLengthOverridden();
+ } else if (id.isAtom(cx->names().callee)) {
+ argsobj.as<MappedArgumentsObject>().markCalleeOverridden();
+ } else if (id.isWellKnownSymbol(JS::SymbolCode::iterator)) {
+ argsobj.markIteratorOverridden();
+ }
+ return result.succeed();
+}
+
+/* static */
+bool ArgumentsObject::obj_mayResolve(const JSAtomState& names, jsid id,
+ JSObject*) {
+ // Arguments might resolve indexes, Symbol.iterator, or length/callee.
+ if (id.isAtom()) {
+ JSAtom* atom = id.toAtom();
+ return atom->isIndex() || atom == names.length || atom == names.callee;
+ }
+
+ return id.isInt() || id.isWellKnownSymbol(JS::SymbolCode::iterator);
+}
+
+bool js::MappedArgGetter(JSContext* cx, HandleObject obj, HandleId id,
+ MutableHandleValue vp) {
+ MappedArgumentsObject& argsobj = obj->as<MappedArgumentsObject>();
+ if (id.isInt()) {
+ /*
+ * arg can exceed the number of arguments if a script changed the
+ * prototype to point to another Arguments object with a bigger argc.
+ */
+ unsigned arg = unsigned(id.toInt());
+ if (argsobj.isElement(arg)) {
+ vp.set(argsobj.element(arg));
+ }
+ } else if (id.isAtom(cx->names().length)) {
+ if (!argsobj.hasOverriddenLength()) {
+ vp.setInt32(argsobj.initialLength());
+ }
+ } else {
+ MOZ_ASSERT(id.isAtom(cx->names().callee));
+ if (!argsobj.hasOverriddenCallee()) {
+ vp.setObject(argsobj.callee());
+ }
+ }
+ return true;
+}
+
+bool js::MappedArgSetter(JSContext* cx, HandleObject obj, HandleId id,
+ HandleValue v, ObjectOpResult& result) {
+ Handle<MappedArgumentsObject*> argsobj = obj.as<MappedArgumentsObject>();
+
+ Rooted<mozilla::Maybe<PropertyDescriptor>> desc(cx);
+ if (!GetOwnPropertyDescriptor(cx, argsobj, id, &desc)) {
+ return false;
+ }
+ MOZ_ASSERT(desc.isSome());
+ MOZ_ASSERT(desc->isDataDescriptor());
+ MOZ_ASSERT(desc->writable());
+ MOZ_ASSERT(!desc->resolving());
+
+ if (id.isInt()) {
+ unsigned arg = unsigned(id.toInt());
+ if (argsobj->isElement(arg)) {
+ argsobj->setElement(arg, v);
+ return result.succeed();
+ }
+ } else {
+ MOZ_ASSERT(id.isAtom(cx->names().length) || id.isAtom(cx->names().callee));
+ }
+
+ /*
+ * For simplicity we use delete/define to replace the property with a
+ * simple data property. Note that we rely on ArgumentsObject::obj_delProperty
+ * to set the corresponding override-bit.
+ * Note also that we must define the property instead of setting it in case
+ * the user has changed the prototype to an object that has a setter for
+ * this id.
+ */
+ Rooted<PropertyDescriptor> desc_(cx, *desc);
+ desc_.setValue(v);
+ ObjectOpResult ignored;
+ return NativeDeleteProperty(cx, argsobj, id, ignored) &&
+ NativeDefineProperty(cx, argsobj, id, desc_, result);
+}
+
+/* static */
+bool ArgumentsObject::getArgumentsIterator(JSContext* cx,
+ MutableHandleValue val) {
+ Handle<PropertyName*> shName = cx->names().dollar_ArrayValues_;
+ Rooted<JSAtom*> name(cx, cx->names().values);
+ return GlobalObject::getSelfHostedFunction(cx, cx->global(), shName, name, 0,
+ val);
+}
+
+/* static */
+bool ArgumentsObject::reifyLength(JSContext* cx, Handle<ArgumentsObject*> obj) {
+ if (obj->hasOverriddenLength()) {
+ return true;
+ }
+
+ RootedId id(cx, NameToId(cx->names().length));
+ RootedValue val(cx, Int32Value(obj->initialLength()));
+ if (!NativeDefineDataProperty(cx, obj, id, val, JSPROP_RESOLVING)) {
+ return false;
+ }
+
+ obj->markLengthOverridden();
+ return true;
+}
+
+/* static */
+bool ArgumentsObject::reifyIterator(JSContext* cx,
+ Handle<ArgumentsObject*> obj) {
+ if (obj->hasOverriddenIterator()) {
+ return true;
+ }
+
+ RootedId iteratorId(cx, PropertyKey::Symbol(cx->wellKnownSymbols().iterator));
+ RootedValue val(cx);
+ if (!ArgumentsObject::getArgumentsIterator(cx, &val)) {
+ return false;
+ }
+ if (!NativeDefineDataProperty(cx, obj, iteratorId, val, JSPROP_RESOLVING)) {
+ return false;
+ }
+
+ obj->markIteratorOverridden();
+ return true;
+}
+
+/* static */
+bool MappedArgumentsObject::reifyCallee(JSContext* cx,
+ Handle<MappedArgumentsObject*> obj) {
+ if (obj->hasOverriddenCallee()) {
+ return true;
+ }
+
+ Rooted<PropertyKey> key(cx, NameToId(cx->names().callee));
+ Rooted<Value> val(cx, ObjectValue(obj->callee()));
+ if (!NativeDefineDataProperty(cx, obj, key, val, JSPROP_RESOLVING)) {
+ return false;
+ }
+
+ obj->markCalleeOverridden();
+ return true;
+}
+
+static bool ResolveArgumentsProperty(JSContext* cx,
+ Handle<ArgumentsObject*> obj, HandleId id,
+ PropertyFlags flags, bool* resolvedp) {
+ MOZ_ASSERT(id.isInt() || id.isAtom(cx->names().length) ||
+ id.isAtom(cx->names().callee));
+ MOZ_ASSERT(flags.isCustomDataProperty());
+
+ if (!NativeObject::addCustomDataProperty(cx, obj, id, flags)) {
+ return false;
+ }
+
+ *resolvedp = true;
+ return true;
+}
+
+/* static */
+bool MappedArgumentsObject::obj_resolve(JSContext* cx, HandleObject obj,
+ HandleId id, bool* resolvedp) {
+ Rooted<MappedArgumentsObject*> argsobj(cx, &obj->as<MappedArgumentsObject>());
+
+ if (id.isWellKnownSymbol(JS::SymbolCode::iterator)) {
+ if (argsobj->hasOverriddenIterator()) {
+ return true;
+ }
+
+ if (!reifyIterator(cx, argsobj)) {
+ return false;
+ }
+ *resolvedp = true;
+ return true;
+ }
+
+ PropertyFlags flags = {PropertyFlag::CustomDataProperty,
+ PropertyFlag::Configurable, PropertyFlag::Writable};
+ if (id.isInt()) {
+ uint32_t arg = uint32_t(id.toInt());
+ if (!argsobj->isElement(arg)) {
+ return true;
+ }
+
+ flags.setFlag(PropertyFlag::Enumerable);
+ } else if (id.isAtom(cx->names().length)) {
+ if (argsobj->hasOverriddenLength()) {
+ return true;
+ }
+ } else {
+ if (!id.isAtom(cx->names().callee)) {
+ return true;
+ }
+
+ if (argsobj->hasOverriddenCallee()) {
+ return true;
+ }
+ }
+
+ return ResolveArgumentsProperty(cx, argsobj, id, flags, resolvedp);
+}
+
+/* static */
+bool MappedArgumentsObject::obj_enumerate(JSContext* cx, HandleObject obj) {
+ Rooted<MappedArgumentsObject*> argsobj(cx, &obj->as<MappedArgumentsObject>());
+
+ RootedId id(cx);
+ bool found;
+
+ // Trigger reflection.
+ id = NameToId(cx->names().length);
+ if (!HasOwnProperty(cx, argsobj, id, &found)) {
+ return false;
+ }
+
+ id = NameToId(cx->names().callee);
+ if (!HasOwnProperty(cx, argsobj, id, &found)) {
+ return false;
+ }
+
+ id = PropertyKey::Symbol(cx->wellKnownSymbols().iterator);
+ if (!HasOwnProperty(cx, argsobj, id, &found)) {
+ return false;
+ }
+
+ for (unsigned i = 0; i < argsobj->initialLength(); i++) {
+ id = PropertyKey::Int(i);
+ if (!HasOwnProperty(cx, argsobj, id, &found)) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+static bool DefineMappedIndex(JSContext* cx, Handle<MappedArgumentsObject*> obj,
+ HandleId id,
+ MutableHandle<PropertyDescriptor> desc,
+ ObjectOpResult& result) {
+ // The custom data properties (see MappedArgGetter, MappedArgSetter) have to
+ // be (re)defined manually because PropertyDescriptor and NativeDefineProperty
+ // don't support these special properties.
+ //
+ // This exists in order to let JS code change the configurable/enumerable
+ // attributes for these properties.
+ //
+ // Note: because this preserves the default mapped-arguments behavior, we
+ // don't need to mark elements as overridden or deleted.
+
+ MOZ_ASSERT(id.isInt());
+ MOZ_ASSERT(obj->isElement(id.toInt()));
+ MOZ_ASSERT(!obj->containsDenseElement(id.toInt()));
+
+ MOZ_ASSERT(!desc.isAccessorDescriptor());
+
+ // Mapped properties aren't used when defining a non-writable property.
+ MOZ_ASSERT(!desc.hasWritable() || desc.writable());
+
+ // First, resolve the property to simplify the code below.
+ PropertyResult prop;
+ if (!NativeLookupOwnProperty<CanGC>(cx, obj, id, &prop)) {
+ return false;
+ }
+
+ MOZ_ASSERT(prop.isNativeProperty());
+
+ PropertyInfo propInfo = prop.propertyInfo();
+ MOZ_ASSERT(propInfo.writable());
+ MOZ_ASSERT(propInfo.isCustomDataProperty());
+
+ // Change the property's attributes by implementing the relevant parts of
+ // ValidateAndApplyPropertyDescriptor (ES2021 draft, 10.1.6.3), in particular
+ // steps 4 and 9.
+
+ // Determine whether the property should be configurable and/or enumerable.
+ bool configurable = propInfo.configurable();
+ bool enumerable = propInfo.enumerable();
+ if (configurable) {
+ if (desc.hasConfigurable()) {
+ configurable = desc.configurable();
+ }
+ if (desc.hasEnumerable()) {
+ enumerable = desc.enumerable();
+ }
+ } else {
+ // Property is not configurable so disallow any attribute changes.
+ if ((desc.hasConfigurable() && desc.configurable()) ||
+ (desc.hasEnumerable() && enumerable != desc.enumerable())) {
+ return result.fail(JSMSG_CANT_REDEFINE_PROP);
+ }
+ }
+
+ PropertyFlags flags = propInfo.flags();
+ flags.setFlag(PropertyFlag::Configurable, configurable);
+ flags.setFlag(PropertyFlag::Enumerable, enumerable);
+ if (!NativeObject::changeCustomDataPropAttributes(cx, obj, id, flags)) {
+ return false;
+ }
+
+ return result.succeed();
+}
+
+// ES 2017 draft 9.4.4.2
+/* static */
+bool MappedArgumentsObject::obj_defineProperty(JSContext* cx, HandleObject obj,
+ HandleId id,
+ Handle<PropertyDescriptor> desc,
+ ObjectOpResult& result) {
+ // Step 1.
+ Rooted<MappedArgumentsObject*> argsobj(cx, &obj->as<MappedArgumentsObject>());
+
+ // Steps 2-3.
+ bool isMapped = false;
+ if (id.isInt()) {
+ unsigned arg = unsigned(id.toInt());
+ isMapped = argsobj->isElement(arg);
+ }
+
+ // Step 4.
+ Rooted<PropertyDescriptor> newArgDesc(cx, desc);
+
+ // Step 5.
+ bool defineMapped = false;
+ if (!desc.isAccessorDescriptor() && isMapped) {
+ // Step 5.a.
+ if (desc.hasWritable() && !desc.writable()) {
+ if (!desc.hasValue()) {
+ RootedValue v(cx, argsobj->element(id.toInt()));
+ newArgDesc.setValue(v);
+ }
+ } else {
+ // In this case the live mapping is supposed to keep working.
+ defineMapped = true;
+ }
+ }
+
+ // Step 6. NativeDefineProperty will lookup [[Value]] for us.
+ if (defineMapped) {
+ if (!DefineMappedIndex(cx, argsobj, id, &newArgDesc, result)) {
+ return false;
+ }
+ } else {
+ if (!NativeDefineProperty(cx, obj.as<NativeObject>(), id, newArgDesc,
+ result)) {
+ return false;
+ }
+ }
+ // Step 7.
+ if (!result.ok()) {
+ return true;
+ }
+
+ // Step 8.
+ if (isMapped) {
+ unsigned arg = unsigned(id.toInt());
+ if (desc.isAccessorDescriptor()) {
+ if (!argsobj->markElementDeleted(cx, arg)) {
+ return false;
+ }
+ } else {
+ if (desc.hasValue()) {
+ argsobj->setElement(arg, desc.value());
+ }
+ if (desc.hasWritable() && !desc.writable()) {
+ if (!argsobj->markElementDeleted(cx, arg)) {
+ return false;
+ }
+ }
+ }
+ }
+
+ // Step 9.
+ return result.succeed();
+}
+
+bool js::UnmappedArgGetter(JSContext* cx, HandleObject obj, HandleId id,
+ MutableHandleValue vp) {
+ UnmappedArgumentsObject& argsobj = obj->as<UnmappedArgumentsObject>();
+
+ if (id.isInt()) {
+ /*
+ * arg can exceed the number of arguments if a script changed the
+ * prototype to point to another Arguments object with a bigger argc.
+ */
+ unsigned arg = unsigned(id.toInt());
+ if (argsobj.isElement(arg)) {
+ vp.set(argsobj.element(arg));
+ }
+ } else {
+ MOZ_ASSERT(id.isAtom(cx->names().length));
+ if (!argsobj.hasOverriddenLength()) {
+ vp.setInt32(argsobj.initialLength());
+ }
+ }
+ return true;
+}
+
+bool js::UnmappedArgSetter(JSContext* cx, HandleObject obj, HandleId id,
+ HandleValue v, ObjectOpResult& result) {
+ Handle<UnmappedArgumentsObject*> argsobj = obj.as<UnmappedArgumentsObject>();
+
+ Rooted<mozilla::Maybe<PropertyDescriptor>> desc(cx);
+ if (!GetOwnPropertyDescriptor(cx, argsobj, id, &desc)) {
+ return false;
+ }
+ MOZ_ASSERT(desc.isSome());
+ MOZ_ASSERT(desc->isDataDescriptor());
+ MOZ_ASSERT(desc->writable());
+ MOZ_ASSERT(!desc->resolving());
+
+ if (id.isInt()) {
+ unsigned arg = unsigned(id.toInt());
+ if (arg < argsobj->initialLength()) {
+ argsobj->setElement(arg, v);
+ return result.succeed();
+ }
+ } else {
+ MOZ_ASSERT(id.isAtom(cx->names().length));
+ }
+
+ /*
+ * For simplicity we use delete/define to replace the property with a
+ * simple data property. Note that we rely on ArgumentsObject::obj_delProperty
+ * to set the corresponding override-bit.
+ */
+ Rooted<PropertyDescriptor> desc_(cx, *desc);
+ desc_.setValue(v);
+ ObjectOpResult ignored;
+ return NativeDeleteProperty(cx, argsobj, id, ignored) &&
+ NativeDefineProperty(cx, argsobj, id, desc_, result);
+}
+
+/* static */
+bool UnmappedArgumentsObject::obj_resolve(JSContext* cx, HandleObject obj,
+ HandleId id, bool* resolvedp) {
+ Rooted<UnmappedArgumentsObject*> argsobj(cx,
+ &obj->as<UnmappedArgumentsObject>());
+
+ if (id.isWellKnownSymbol(JS::SymbolCode::iterator)) {
+ if (argsobj->hasOverriddenIterator()) {
+ return true;
+ }
+
+ if (!reifyIterator(cx, argsobj)) {
+ return false;
+ }
+ *resolvedp = true;
+ return true;
+ }
+
+ if (id.isAtom(cx->names().callee)) {
+ RootedObject throwTypeError(
+ cx, GlobalObject::getOrCreateThrowTypeError(cx, cx->global()));
+ if (!throwTypeError) {
+ return false;
+ }
+
+ unsigned attrs = JSPROP_RESOLVING | JSPROP_PERMANENT;
+ if (!NativeDefineAccessorProperty(cx, argsobj, id, throwTypeError,
+ throwTypeError, attrs)) {
+ return false;
+ }
+
+ *resolvedp = true;
+ return true;
+ }
+
+ PropertyFlags flags = {PropertyFlag::CustomDataProperty,
+ PropertyFlag::Configurable, PropertyFlag::Writable};
+ if (id.isInt()) {
+ uint32_t arg = uint32_t(id.toInt());
+ if (!argsobj->isElement(arg)) {
+ return true;
+ }
+
+ flags.setFlag(PropertyFlag::Enumerable);
+ } else if (id.isAtom(cx->names().length)) {
+ if (argsobj->hasOverriddenLength()) {
+ return true;
+ }
+ } else {
+ return true;
+ }
+
+ return ResolveArgumentsProperty(cx, argsobj, id, flags, resolvedp);
+}
+
+/* static */
+bool UnmappedArgumentsObject::obj_enumerate(JSContext* cx, HandleObject obj) {
+ Rooted<UnmappedArgumentsObject*> argsobj(cx,
+ &obj->as<UnmappedArgumentsObject>());
+
+ RootedId id(cx);
+ bool found;
+
+ // Trigger reflection.
+ id = NameToId(cx->names().length);
+ if (!HasOwnProperty(cx, argsobj, id, &found)) {
+ return false;
+ }
+
+ id = NameToId(cx->names().callee);
+ if (!HasOwnProperty(cx, argsobj, id, &found)) {
+ return false;
+ }
+
+ id = PropertyKey::Symbol(cx->wellKnownSymbols().iterator);
+ if (!HasOwnProperty(cx, argsobj, id, &found)) {
+ return false;
+ }
+
+ for (unsigned i = 0; i < argsobj->initialLength(); i++) {
+ id = PropertyKey::Int(i);
+ if (!HasOwnProperty(cx, argsobj, id, &found)) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+void ArgumentsObject::finalize(JS::GCContext* gcx, JSObject* obj) {
+ MOZ_ASSERT(!IsInsideNursery(obj));
+ ArgumentsObject& argsobj = obj->as<ArgumentsObject>();
+ if (argsobj.data()) {
+ gcx->free_(&argsobj, argsobj.maybeRareData(),
+ RareArgumentsData::bytesRequired(argsobj.initialLength()),
+ MemoryUse::RareArgumentsData);
+ gcx->free_(&argsobj, argsobj.data(),
+ ArgumentsData::bytesRequired(argsobj.data()->numArgs()),
+ MemoryUse::ArgumentsData);
+ }
+}
+
+void ArgumentsObject::trace(JSTracer* trc, JSObject* obj) {
+ ArgumentsObject& argsobj = obj->as<ArgumentsObject>();
+ // Template objects have no ArgumentsData.
+ if (ArgumentsData* data = argsobj.data()) {
+ data->args.trace(trc);
+ }
+}
+
+/* static */
+size_t ArgumentsObject::objectMoved(JSObject* dst, JSObject* src) {
+ ArgumentsObject* ndst = &dst->as<ArgumentsObject>();
+ const ArgumentsObject* nsrc = &src->as<ArgumentsObject>();
+ MOZ_ASSERT(ndst->data() == nsrc->data());
+
+ if (!IsInsideNursery(src)) {
+ return 0;
+ }
+
+ Nursery& nursery = dst->runtimeFromMainThread()->gc.nursery();
+
+ size_t nbytesTotal = 0;
+
+ ArgumentsData* data = nsrc->data();
+ uint32_t nDataBytes = ArgumentsData::bytesRequired(nsrc->data()->numArgs());
+ Nursery::WasBufferMoved result = nursery.maybeMoveBufferOnPromotion(
+ &data, dst, nDataBytes, MemoryUse::ArgumentsData);
+ if (result == Nursery::BufferMoved) {
+ ndst->initFixedSlot(DATA_SLOT, PrivateValue(data));
+ nbytesTotal += nDataBytes;
+ }
+
+ if (RareArgumentsData* rareData = nsrc->maybeRareData()) {
+ uint32_t nRareBytes =
+ RareArgumentsData::bytesRequired(nsrc->initialLength());
+ Nursery::WasBufferMoved result = nursery.maybeMoveBufferOnPromotion(
+ &rareData, dst, nRareBytes, MemoryUse::RareArgumentsData);
+ if (result == Nursery::BufferMoved) {
+ ndst->data()->rareData = rareData;
+ nbytesTotal += nRareBytes;
+ }
+ }
+
+ return nbytesTotal;
+}
+
+/*
+ * The classes below collaborate to lazily reflect and synchronize actual
+ * argument values, argument count, and callee function object stored in a
+ * stack frame with their corresponding property values in the frame's
+ * arguments object.
+ */
+const JSClassOps MappedArgumentsObject::classOps_ = {
+ nullptr, // addProperty
+ ArgumentsObject::obj_delProperty, // delProperty
+ MappedArgumentsObject::obj_enumerate, // enumerate
+ nullptr, // newEnumerate
+ MappedArgumentsObject::obj_resolve, // resolve
+ ArgumentsObject::obj_mayResolve, // mayResolve
+ ArgumentsObject::finalize, // finalize
+ nullptr, // call
+ nullptr, // construct
+ ArgumentsObject::trace, // trace
+};
+
+const js::ClassExtension MappedArgumentsObject::classExt_ = {
+ ArgumentsObject::objectMoved, // objectMovedOp
+};
+
+const ObjectOps MappedArgumentsObject::objectOps_ = {
+ nullptr, // lookupProperty
+ MappedArgumentsObject::obj_defineProperty, // defineProperty
+ nullptr, // hasProperty
+ nullptr, // getProperty
+ nullptr, // setProperty
+ nullptr, // getOwnPropertyDescriptor
+ nullptr, // deleteProperty
+ nullptr, // getElements
+ nullptr, // funToString
+};
+
+const JSClass MappedArgumentsObject::class_ = {
+ "Arguments",
+ JSCLASS_DELAY_METADATA_BUILDER |
+ JSCLASS_HAS_RESERVED_SLOTS(MappedArgumentsObject::RESERVED_SLOTS) |
+ JSCLASS_HAS_CACHED_PROTO(JSProto_Object) |
+ JSCLASS_SKIP_NURSERY_FINALIZE | JSCLASS_BACKGROUND_FINALIZE,
+ &MappedArgumentsObject::classOps_,
+ nullptr,
+ &MappedArgumentsObject::classExt_,
+ &MappedArgumentsObject::objectOps_};
+
+/*
+ * Unmapped arguments is significantly less magical than mapped arguments, so
+ * it is represented by a different class while sharing some functionality.
+ */
+const JSClassOps UnmappedArgumentsObject::classOps_ = {
+ nullptr, // addProperty
+ ArgumentsObject::obj_delProperty, // delProperty
+ UnmappedArgumentsObject::obj_enumerate, // enumerate
+ nullptr, // newEnumerate
+ UnmappedArgumentsObject::obj_resolve, // resolve
+ ArgumentsObject::obj_mayResolve, // mayResolve
+ ArgumentsObject::finalize, // finalize
+ nullptr, // call
+ nullptr, // construct
+ ArgumentsObject::trace, // trace
+};
+
+const js::ClassExtension UnmappedArgumentsObject::classExt_ = {
+ ArgumentsObject::objectMoved, // objectMovedOp
+};
+
+const JSClass UnmappedArgumentsObject::class_ = {
+ "Arguments",
+ JSCLASS_DELAY_METADATA_BUILDER |
+ JSCLASS_HAS_RESERVED_SLOTS(UnmappedArgumentsObject::RESERVED_SLOTS) |
+ JSCLASS_HAS_CACHED_PROTO(JSProto_Object) |
+ JSCLASS_SKIP_NURSERY_FINALIZE | JSCLASS_BACKGROUND_FINALIZE,
+ &UnmappedArgumentsObject::classOps_, nullptr,
+ &UnmappedArgumentsObject::classExt_};