diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-28 14:29:10 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-28 14:29:10 +0000 |
commit | 2aa4a82499d4becd2284cdb482213d541b8804dd (patch) | |
tree | b80bf8bf13c3766139fbacc530efd0dd9d54394c /js/src/jit/WarpCacheIRTranspiler.cpp | |
parent | Initial commit. (diff) | |
download | firefox-2aa4a82499d4becd2284cdb482213d541b8804dd.tar.xz firefox-2aa4a82499d4becd2284cdb482213d541b8804dd.zip |
Adding upstream version 86.0.1.upstream/86.0.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'js/src/jit/WarpCacheIRTranspiler.cpp')
-rw-r--r-- | js/src/jit/WarpCacheIRTranspiler.cpp | 4617 |
1 files changed, 4617 insertions, 0 deletions
diff --git a/js/src/jit/WarpCacheIRTranspiler.cpp b/js/src/jit/WarpCacheIRTranspiler.cpp new file mode 100644 index 0000000000..d0a592b613 --- /dev/null +++ b/js/src/jit/WarpCacheIRTranspiler.cpp @@ -0,0 +1,4617 @@ +/* -*- 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 "jit/WarpCacheIRTranspiler.h" + +#include "mozilla/Maybe.h" + +#include "jsmath.h" + +#include "builtin/DataViewObject.h" +#include "jit/AtomicOp.h" +#include "jit/CacheIR.h" +#include "jit/CacheIRCompiler.h" +#include "jit/CacheIROpsGenerated.h" +#include "jit/CompileInfo.h" +#include "jit/LIR.h" +#include "jit/MIR.h" +#include "jit/MIRGenerator.h" +#include "jit/MIRGraph.h" +#include "jit/WarpBuilder.h" +#include "jit/WarpBuilderShared.h" +#include "jit/WarpSnapshot.h" +#include "js/ScalarType.h" // js::Scalar::Type +#include "vm/ArgumentsObject.h" +#include "vm/BytecodeLocation.h" +#include "wasm/WasmInstance.h" + +#include "gc/ObjectKind-inl.h" + +using namespace js; +using namespace js::jit; + +// The CacheIR transpiler generates MIR from Baseline CacheIR. +class MOZ_RAII WarpCacheIRTranspiler : public WarpBuilderShared { + WarpBuilder* builder_; + BytecodeLocation loc_; + const CacheIRStubInfo* stubInfo_; + const uint8_t* stubData_; + + // Vector mapping OperandId to corresponding MDefinition. + using MDefinitionStackVector = Vector<MDefinition*, 8, SystemAllocPolicy>; + MDefinitionStackVector operands_; + + CallInfo* callInfo_; + + // Array mapping call arguments to OperandId. + using ArgumentKindArray = + mozilla::EnumeratedArray<ArgumentKind, ArgumentKind::NumKinds, OperandId>; + ArgumentKindArray argumentOperandIds_; + + void setArgumentId(ArgumentKind kind, OperandId id) { + MOZ_ASSERT(kind != ArgumentKind::Callee); + MOZ_ASSERT(!argumentOperandIds_[kind].valid()); + argumentOperandIds_[kind] = id; + } + + void updateArgumentsFromOperands(); + +#ifdef DEBUG + // Used to assert that there is only one effectful instruction + // per stub. And that this instruction has a resume point. + MInstruction* effectful_ = nullptr; + bool pushedResult_ = false; +#endif + + inline void addUnchecked(MInstruction* ins) { + current->add(ins); + + // If we have not set a more specific bailout kind, mark this instruction + // as transpiled CacheIR. If one of these instructions bails out, we + // expect to hit the baseline fallback stub and invalidate the Warp script + // in tryAttach. + if (ins->bailoutKind() == BailoutKind::Unknown) { + ins->setBailoutKind(BailoutKind::TranspiledCacheIR); + } + } + + inline void add(MInstruction* ins) { + MOZ_ASSERT(!ins->isEffectful()); + addUnchecked(ins); + } + + inline void addEffectful(MInstruction* ins) { + MOZ_ASSERT(ins->isEffectful()); + MOZ_ASSERT(!effectful_, "Can only have one effectful instruction"); + addUnchecked(ins); +#ifdef DEBUG + effectful_ = ins; +#endif + } + + // Bypasses all checks in addEffectful. Only used for testing functions. + inline void addEffectfulUnsafe(MInstruction* ins) { + MOZ_ASSERT(ins->isEffectful()); + addUnchecked(ins); + } + + [[nodiscard]] bool resumeAfterUnchecked(MInstruction* ins) { + return WarpBuilderShared::resumeAfter(ins, loc_); + } + [[nodiscard]] bool resumeAfter(MInstruction* ins) { + MOZ_ASSERT(effectful_ == ins); + return resumeAfterUnchecked(ins); + } + + // CacheIR instructions writing to the IC's result register (the *Result + // instructions) must call this to push the result onto the virtual stack. + void pushResult(MDefinition* result) { + MOZ_ASSERT(!pushedResult_, "Can't have more than one result"); + current->push(result); +#ifdef DEBUG + pushedResult_ = true; +#endif + } + + MDefinition* getOperand(OperandId id) const { return operands_[id.id()]; } + + void setOperand(OperandId id, MDefinition* def) { operands_[id.id()] = def; } + + [[nodiscard]] bool defineOperand(OperandId id, MDefinition* def) { + MOZ_ASSERT(id.id() == operands_.length()); + return operands_.append(def); + } + + uintptr_t readStubWord(uint32_t offset) { + return stubInfo_->getStubRawWord(stubData_, offset); + } + + Shape* shapeStubField(uint32_t offset) { + return reinterpret_cast<Shape*>(readStubWord(offset)); + } + const JSClass* classStubField(uint32_t offset) { + return reinterpret_cast<const JSClass*>(readStubWord(offset)); + } + JSString* stringStubField(uint32_t offset) { + return reinterpret_cast<JSString*>(readStubWord(offset)); + } + JS::Symbol* symbolStubField(uint32_t offset) { + return reinterpret_cast<JS::Symbol*>(readStubWord(offset)); + } + ObjectGroup* groupStubField(uint32_t offset) { + return reinterpret_cast<ObjectGroup*>(readStubWord(offset)); + } + BaseScript* baseScriptStubField(uint32_t offset) { + return reinterpret_cast<BaseScript*>(readStubWord(offset)); + } + const JSJitInfo* jitInfoStubField(uint32_t offset) { + return reinterpret_cast<const JSJitInfo*>(readStubWord(offset)); + } + JS::ExpandoAndGeneration* expandoAndGenerationField(uint32_t offset) { + return reinterpret_cast<JS::ExpandoAndGeneration*>(readStubWord(offset)); + } + const wasm::FuncExport* wasmFuncExportField(uint32_t offset) { + return reinterpret_cast<const wasm::FuncExport*>(readStubWord(offset)); + } + const void* rawPointerField(uint32_t offset) { + return reinterpret_cast<const void*>(readStubWord(offset)); + } + jsid idStubField(uint32_t offset) { + return jsid::fromRawBits(readStubWord(offset)); + } + int32_t int32StubField(uint32_t offset) { + return static_cast<int32_t>(readStubWord(offset)); + } + uint32_t uint32StubField(uint32_t offset) { + return static_cast<uint32_t>(readStubWord(offset)); + } + uint64_t uint64StubField(uint32_t offset) { + return static_cast<uint64_t>(stubInfo_->getStubRawInt64(stubData_, offset)); + } + + // This must only be called when the caller knows the object is tenured and + // not a nursery index. + JSObject* tenuredObjectStubField(uint32_t offset) { + WarpObjectField field = WarpObjectField::fromData(readStubWord(offset)); + return field.toObject(); + } + + // Returns either MConstant or MNurseryIndex. See WarpObjectField. + MInstruction* objectStubField(uint32_t offset); + + [[nodiscard]] bool emitGuardTo(ValOperandId inputId, MIRType type); + + [[nodiscard]] bool emitToString(OperandId inputId, StringOperandId resultId); + + template <typename T> + [[nodiscard]] bool emitDoubleBinaryArithResult(NumberOperandId lhsId, + NumberOperandId rhsId); + + template <typename T> + [[nodiscard]] bool emitInt32BinaryArithResult(Int32OperandId lhsId, + Int32OperandId rhsId); + + template <typename T> + [[nodiscard]] bool emitBigIntBinaryArithResult(BigIntOperandId lhsId, + BigIntOperandId rhsId); + + template <typename T> + [[nodiscard]] bool emitBigIntBinaryArithEffectfulResult( + BigIntOperandId lhsId, BigIntOperandId rhsId); + + template <typename T> + [[nodiscard]] bool emitBigIntUnaryArithResult(BigIntOperandId inputId); + + [[nodiscard]] bool emitCompareResult(JSOp op, OperandId lhsId, + OperandId rhsId, + MCompare::CompareType compareType); + + [[nodiscard]] bool emitNewIteratorResult(MNewIterator::Type type, + uint32_t templateObjectOffset); + + MInstruction* addBoundsCheck(MDefinition* index, MDefinition* length); + + bool emitAddAndStoreSlotShared(MAddAndStoreSlot::Kind kind, + ObjOperandId objId, uint32_t offsetOffset, + ValOperandId rhsId, uint32_t newShapeOffset); + + void addDataViewData(MDefinition* obj, Scalar::Type type, + MDefinition** offset, MInstruction** elements); + + [[nodiscard]] bool emitAtomicsBinaryOp(ObjOperandId objId, + Int32OperandId indexId, + Int32OperandId valueId, + Scalar::Type elementType, AtomicOp op); + + [[nodiscard]] bool emitLoadArgumentSlot(ValOperandId resultId, + uint32_t slotIndex); + + // Calls are either Native (native function without a JitEntry), + // a DOM Native (native function with a JitInfo OpType::Method), + // or Scripted (scripted function or native function with a JitEntry). + enum class CallKind { Native, DOM, Scripted }; + + [[nodiscard]] bool updateCallInfo(MDefinition* callee, CallFlags flags); + + [[nodiscard]] bool emitCallFunction(ObjOperandId calleeId, + Int32OperandId argcId, + mozilla::Maybe<ObjOperandId> thisObjId, + CallFlags flags, CallKind kind); + [[nodiscard]] bool emitFunApplyArgs(WrappedFunction* wrappedTarget, + CallFlags flags); + + MDefinition* convertWasmArg(MDefinition* arg, wasm::ValType::Kind kind); + + WrappedFunction* maybeWrappedFunction(MDefinition* callee, CallKind kind, + uint16_t nargs, FunctionFlags flags); + WrappedFunction* maybeCallTarget(MDefinition* callee, CallKind kind); + + bool maybeCreateThis(MDefinition* callee, CallFlags flags, CallKind kind); + + [[nodiscard]] bool emitCallGetterResult(CallKind kind, + ValOperandId receiverId, + uint32_t getterOffset, bool sameRealm, + uint32_t nargsAndFlagsOffset); + [[nodiscard]] bool emitCallSetter(CallKind kind, ObjOperandId receiverId, + uint32_t setterOffset, ValOperandId rhsId, + bool sameRealm, + uint32_t nargsAndFlagsOffset); + + CACHE_IR_TRANSPILER_GENERATED + + public: + WarpCacheIRTranspiler(WarpBuilder* builder, BytecodeLocation loc, + CallInfo* callInfo, const WarpCacheIR* cacheIRSnapshot) + : WarpBuilderShared(builder->snapshot(), builder->mirGen(), + builder->currentBlock()), + builder_(builder), + loc_(loc), + stubInfo_(cacheIRSnapshot->stubInfo()), + stubData_(cacheIRSnapshot->stubData()), + callInfo_(callInfo) {} + + [[nodiscard]] bool transpile(std::initializer_list<MDefinition*> inputs); +}; + +bool WarpCacheIRTranspiler::transpile( + std::initializer_list<MDefinition*> inputs) { + if (!operands_.append(inputs.begin(), inputs.end())) { + return false; + } + + CacheIRReader reader(stubInfo_); + do { + CacheOp op = reader.readOp(); + switch (op) { +#define DEFINE_OP(op, ...) \ + case CacheOp::op: \ + if (!emit##op(reader)) { \ + return false; \ + } \ + break; + CACHE_IR_TRANSPILER_OPS(DEFINE_OP) +#undef DEFINE_OP + + default: + fprintf(stderr, "Unsupported op: %s\n", CacheIROpNames[size_t(op)]); + MOZ_CRASH("Unsupported op"); + } + } while (reader.more()); + + // Effectful instructions should have a resume point. MIonToWasmCall is an + // exception: we can attach the resume point to the MInt64ToBigInt instruction + // instead. + MOZ_ASSERT_IF(effectful_, + effectful_->resumePoint() || effectful_->isIonToWasmCall()); + return true; +} + +MInstruction* WarpCacheIRTranspiler::objectStubField(uint32_t offset) { + WarpObjectField field = WarpObjectField::fromData(readStubWord(offset)); + + if (field.isNurseryIndex()) { + auto* ins = MNurseryObject::New(alloc(), field.toNurseryIndex()); + add(ins); + return ins; + } + + auto* ins = MConstant::NewObject(alloc(), field.toObject()); + add(ins); + return ins; +} + +bool WarpCacheIRTranspiler::emitGuardClass(ObjOperandId objId, + GuardClassKind kind) { + MDefinition* def = getOperand(objId); + + const JSClass* classp = nullptr; + switch (kind) { + case GuardClassKind::Array: + classp = &ArrayObject::class_; + break; + case GuardClassKind::ArrayBuffer: + classp = &ArrayBufferObject::class_; + break; + case GuardClassKind::SharedArrayBuffer: + classp = &SharedArrayBufferObject::class_; + break; + case GuardClassKind::DataView: + classp = &DataViewObject::class_; + break; + case GuardClassKind::MappedArguments: + classp = &MappedArgumentsObject::class_; + break; + case GuardClassKind::UnmappedArguments: + classp = &UnmappedArgumentsObject::class_; + break; + case GuardClassKind::WindowProxy: + classp = mirGen().runtime->maybeWindowProxyClass(); + break; + case GuardClassKind::JSFunction: + classp = &JSFunction::class_; + break; + default: + MOZ_CRASH("not yet supported"); + } + MOZ_ASSERT(classp); + + auto* ins = MGuardToClass::New(alloc(), def, classp); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardAnyClass(ObjOperandId objId, + uint32_t claspOffset) { + MDefinition* def = getOperand(objId); + const JSClass* classp = classStubField(claspOffset); + + auto* ins = MGuardToClass::New(alloc(), def, classp); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardShape(ObjOperandId objId, + uint32_t shapeOffset) { + MDefinition* def = getOperand(objId); + Shape* shape = shapeStubField(shapeOffset); + + auto* ins = MGuardShape::New(alloc(), def, shape); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardGroup(ObjOperandId objId, + uint32_t groupOffset) { + MDefinition* def = getOperand(objId); + ObjectGroup* group = groupStubField(groupOffset); + + auto* ins = MGuardObjectGroup::New(alloc(), def, group, + /* bailOnEquality = */ false); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardNullProto(ObjOperandId objId) { + MDefinition* def = getOperand(objId); + + auto* ins = MGuardNullProto::New(alloc(), def); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIsNativeObject(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardIsNativeObject::New(alloc(), obj); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIsProxy(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardIsProxy::New(alloc(), obj); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIsNotProxy(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardIsNotProxy::New(alloc(), obj); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIsNotDOMProxy(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardIsNotDOMProxy::New(alloc(), obj); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardHasGetterSetter(ObjOperandId objId, + uint32_t shapeOffset) { + MDefinition* obj = getOperand(objId); + Shape* shape = shapeStubField(shapeOffset); + + auto* ins = MGuardHasGetterSetter::New(alloc(), obj, shape); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitProxyGetResult(ObjOperandId objId, + uint32_t idOffset) { + MDefinition* obj = getOperand(objId); + jsid id = idStubField(idOffset); + + auto* ins = MProxyGet::New(alloc(), obj, id); + addEffectful(ins); + + pushResult(ins); + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitProxyGetByValueResult(ObjOperandId objId, + ValOperandId idId) { + MDefinition* obj = getOperand(objId); + MDefinition* id = getOperand(idId); + + auto* ins = MProxyGetByValue::New(alloc(), obj, id); + addEffectful(ins); + + pushResult(ins); + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitProxyHasPropResult(ObjOperandId objId, + ValOperandId idId, + bool hasOwn) { + MDefinition* obj = getOperand(objId); + MDefinition* id = getOperand(idId); + + auto* ins = MProxyHasProp::New(alloc(), obj, id, hasOwn); + addEffectful(ins); + + pushResult(ins); + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitProxySet(ObjOperandId objId, uint32_t idOffset, + ValOperandId rhsId, bool strict) { + MDefinition* obj = getOperand(objId); + jsid id = idStubField(idOffset); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MProxySet::New(alloc(), obj, id, rhs, strict); + addEffectful(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitProxySetByValue(ObjOperandId objId, + ValOperandId idId, + ValOperandId rhsId, + bool strict) { + MDefinition* obj = getOperand(objId); + MDefinition* id = getOperand(idId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MProxySetByValue::New(alloc(), obj, id, rhs, strict); + addEffectful(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitCallSetArrayLength(ObjOperandId objId, + bool strict, + ValOperandId rhsId) { + MDefinition* obj = getOperand(objId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MCallSetArrayLength::New(alloc(), obj, rhs, strict); + addEffectful(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitCallDOMGetterResult(ObjOperandId objId, + uint32_t jitInfoOffset) { + MDefinition* obj = getOperand(objId); + const JSJitInfo* jitInfo = jitInfoStubField(jitInfoOffset); + + MInstruction* ins; + if (jitInfo->isAlwaysInSlot) { + ins = MGetDOMMember::New(alloc(), jitInfo, obj, nullptr, nullptr); + } else { + // TODO(post-Warp): realms, guard operands (movable?). + ins = MGetDOMProperty::New(alloc(), jitInfo, DOMObjectKind::Native, + (JS::Realm*)mirGen().realm->realmPtr(), obj, + nullptr, nullptr); + } + + if (!ins) { + return false; + } + + if (ins->isEffectful()) { + addEffectful(ins); + pushResult(ins); + return resumeAfter(ins); + } + + add(ins); + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitCallDOMSetter(ObjOperandId objId, + uint32_t jitInfoOffset, + ValOperandId rhsId) { + MDefinition* obj = getOperand(objId); + const JSJitInfo* jitInfo = jitInfoStubField(jitInfoOffset); + MDefinition* value = getOperand(rhsId); + + MOZ_ASSERT(jitInfo->type() == JSJitInfo::Setter); + auto* set = + MSetDOMProperty::New(alloc(), jitInfo->setter, DOMObjectKind::Native, + (JS::Realm*)mirGen().realm->realmPtr(), obj, value); + addEffectful(set); + return resumeAfter(set); +} + +bool WarpCacheIRTranspiler::emitLoadDOMExpandoValue(ObjOperandId objId, + ValOperandId resultId) { + MDefinition* proxy = getOperand(objId); + + auto* ins = MLoadDOMExpandoValue::New(alloc(), proxy); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitLoadDOMExpandoValueGuardGeneration( + ObjOperandId objId, uint32_t expandoAndGenerationOffset, + uint32_t generationOffset, ValOperandId resultId) { + MDefinition* proxy = getOperand(objId); + JS::ExpandoAndGeneration* expandoAndGeneration = + expandoAndGenerationField(expandoAndGenerationOffset); + uint64_t generation = uint64StubField(generationOffset); + + auto* ins = MLoadDOMExpandoValueGuardGeneration::New( + alloc(), proxy, expandoAndGeneration, generation); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitLoadDOMExpandoValueIgnoreGeneration( + ObjOperandId objId, ValOperandId resultId) { + MDefinition* proxy = getOperand(objId); + + auto* ins = MLoadDOMExpandoValueIgnoreGeneration::New(alloc(), proxy); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitGuardDOMExpandoMissingOrGuardShape( + ValOperandId expandoId, uint32_t shapeOffset) { + MDefinition* expando = getOperand(expandoId); + Shape* shape = shapeStubField(shapeOffset); + + auto* ins = MGuardDOMExpandoMissingOrGuardShape::New(alloc(), expando, shape); + add(ins); + + setOperand(expandoId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMegamorphicLoadSlotResult(ObjOperandId objId, + uint32_t nameOffset) { + MDefinition* obj = getOperand(objId); + PropertyName* name = stringStubField(nameOffset)->asAtom().asPropertyName(); + + auto* ins = MMegamorphicLoadSlot::New(alloc(), obj, name); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMegamorphicLoadSlotByValueResult( + ObjOperandId objId, ValOperandId idId) { + MDefinition* obj = getOperand(objId); + MDefinition* id = getOperand(idId); + + auto* ins = MMegamorphicLoadSlotByValue::New(alloc(), obj, id); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMegamorphicStoreSlot(ObjOperandId objId, + uint32_t nameOffset, + ValOperandId rhsId) { + MDefinition* obj = getOperand(objId); + PropertyName* name = stringStubField(nameOffset)->asAtom().asPropertyName(); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MMegamorphicStoreSlot::New(alloc(), obj, name, rhs); + addEffectful(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitMegamorphicHasPropResult(ObjOperandId objId, + ValOperandId idId, + bool hasOwn) { + MDefinition* obj = getOperand(objId); + MDefinition* id = getOperand(idId); + + auto* ins = MMegamorphicHasProp::New(alloc(), obj, id, hasOwn); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMegamorphicSetElement(ObjOperandId objId, + ValOperandId idId, + ValOperandId rhsId, + bool strict) { + MDefinition* obj = getOperand(objId); + MDefinition* id = getOperand(idId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MCallSetElement::New(alloc(), obj, id, rhs, strict); + addEffectful(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitGuardIsNotArrayBufferMaybeShared( + ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardIsNotArrayBufferMaybeShared::New(alloc(), obj); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIsTypedArray(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardIsTypedArray::New(alloc(), obj); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardProto(ObjOperandId objId, + uint32_t protoOffset) { + MDefinition* def = getOperand(objId); + MDefinition* proto = objectStubField(protoOffset); + + auto* ins = MGuardProto::New(alloc(), def, proto); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardDynamicSlotIsSpecificObject( + ObjOperandId objId, ObjOperandId expectedId, uint32_t slotOffset) { + size_t slotIndex = int32StubField(slotOffset); + MDefinition* obj = getOperand(objId); + MDefinition* expected = getOperand(expectedId); + + auto* slots = MSlots::New(alloc(), obj); + add(slots); + + auto* load = MLoadDynamicSlot::New(alloc(), slots, slotIndex); + add(load); + + auto* unbox = MUnbox::New(alloc(), load, MIRType::Object, MUnbox::Fallible); + add(unbox); + + auto* guard = MGuardObjectIdentity::New(alloc(), unbox, expected, + /* bailOnEquality = */ false); + add(guard); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardSpecificAtom(StringOperandId strId, + uint32_t expectedOffset) { + MDefinition* str = getOperand(strId); + JSString* expected = stringStubField(expectedOffset); + + auto* ins = MGuardSpecificAtom::New(alloc(), str, &expected->asAtom()); + add(ins); + + setOperand(strId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardSpecificSymbol(SymbolOperandId symId, + uint32_t expectedOffset) { + MDefinition* symbol = getOperand(symId); + JS::Symbol* expected = symbolStubField(expectedOffset); + + auto* ins = MGuardSpecificSymbol::New(alloc(), symbol, expected); + add(ins); + + setOperand(symId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardSpecificObject(ObjOperandId objId, + uint32_t expectedOffset) { + MDefinition* obj = getOperand(objId); + MDefinition* expected = objectStubField(expectedOffset); + + auto* ins = MGuardObjectIdentity::New(alloc(), obj, expected, + /* bailOnEquality = */ false); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardSpecificFunction( + ObjOperandId objId, uint32_t expectedOffset, uint32_t nargsAndFlagsOffset) { + MDefinition* obj = getOperand(objId); + MDefinition* expected = objectStubField(expectedOffset); + uint32_t nargsAndFlags = uint32StubField(nargsAndFlagsOffset); + + uint16_t nargs = nargsAndFlags >> 16; + FunctionFlags flags = FunctionFlags(uint16_t(nargsAndFlags)); + + auto* ins = MGuardSpecificFunction::New(alloc(), obj, expected, nargs, flags); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardFunctionScript( + ObjOperandId funId, uint32_t expectedOffset, uint32_t nargsAndFlagsOffset) { + MDefinition* fun = getOperand(funId); + BaseScript* expected = baseScriptStubField(expectedOffset); + uint32_t nargsAndFlags = uint32StubField(nargsAndFlagsOffset); + + uint16_t nargs = nargsAndFlags >> 16; + FunctionFlags flags = FunctionFlags(uint16_t(nargsAndFlags)); + + auto* ins = MGuardFunctionScript::New(alloc(), fun, expected, nargs, flags); + add(ins); + + setOperand(funId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardStringToIndex(StringOperandId strId, + Int32OperandId resultId) { + MDefinition* str = getOperand(strId); + + auto* ins = MGuardStringToIndex::New(alloc(), str); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitGuardStringToInt32(StringOperandId strId, + Int32OperandId resultId) { + MDefinition* str = getOperand(strId); + + auto* ins = MGuardStringToInt32::New(alloc(), str); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitGuardStringToNumber(StringOperandId strId, + NumberOperandId resultId) { + MDefinition* str = getOperand(strId); + + auto* ins = MGuardStringToDouble::New(alloc(), str); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitGuardNoDenseElements(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardNoDenseElements::New(alloc(), obj); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardMagicValue(ValOperandId valId, + JSWhyMagic magic) { + MDefinition* val = getOperand(valId); + + auto* ins = MGuardValue::New(alloc(), val, MagicValue(magic)); + add(ins); + + setOperand(valId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardFrameHasNoArgumentsObject() { + // WarpOracle ensures this op isn't transpiled in functions that need an + // arguments object. + MOZ_ASSERT(!currentBlock()->info().needsArgsObj()); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardFunctionHasJitEntry(ObjOperandId funId, + bool constructing) { + MDefinition* fun = getOperand(funId); + uint16_t expectedFlags = FunctionFlags::HasJitEntryFlags(constructing); + uint16_t unexpectedFlags = 0; + + auto* ins = + MGuardFunctionFlags::New(alloc(), fun, expectedFlags, unexpectedFlags); + add(ins); + + setOperand(funId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardFunctionHasNoJitEntry(ObjOperandId funId) { + MDefinition* fun = getOperand(funId); + uint16_t expectedFlags = 0; + uint16_t unexpectedFlags = + FunctionFlags::HasJitEntryFlags(/*isConstructing=*/false); + + auto* ins = + MGuardFunctionFlags::New(alloc(), fun, expectedFlags, unexpectedFlags); + add(ins); + + setOperand(funId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardFunctionIsNonBuiltinCtor( + ObjOperandId funId) { + MDefinition* fun = getOperand(funId); + + auto* ins = MGuardFunctionIsNonBuiltinCtor::New(alloc(), fun); + add(ins); + + setOperand(funId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardFunctionIsConstructor(ObjOperandId funId) { + MDefinition* fun = getOperand(funId); + uint16_t expectedFlags = FunctionFlags::CONSTRUCTOR; + uint16_t unexpectedFlags = 0; + + auto* ins = + MGuardFunctionFlags::New(alloc(), fun, expectedFlags, unexpectedFlags); + add(ins); + + setOperand(funId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardNotClassConstructor(ObjOperandId funId) { + MDefinition* fun = getOperand(funId); + + auto* ins = + MGuardFunctionKind::New(alloc(), fun, FunctionFlags::ClassConstructor, + /*bailOnEquality=*/true); + add(ins); + + setOperand(funId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardArrayIsPacked(ObjOperandId arrayId) { + MDefinition* array = getOperand(arrayId); + + auto* ins = MGuardArrayIsPacked::New(alloc(), array); + add(ins); + + setOperand(arrayId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardArgumentsObjectNotOverriddenIterator( + ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardArgumentsObjectNotOverriddenIterator::New(alloc(), obj); + add(ins); + + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadFrameCalleeResult() { + if (const CallInfo* callInfo = builder_->inlineCallInfo()) { + pushResult(callInfo->callee()); + return true; + } + + auto* ins = MCallee::New(alloc()); + add(ins); + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadFrameNumActualArgsResult() { + if (const CallInfo* callInfo = builder_->inlineCallInfo()) { + auto* ins = constant(Int32Value(callInfo->argc())); + pushResult(ins); + return true; + } + + auto* ins = MArgumentsLength::New(alloc()); + add(ins); + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadFrameArgumentResult( + Int32OperandId indexId) { + // We don't support arguments[i] in inlined functions. Scripts using + // arguments[i] are marked as uninlineable in arguments analysis. + MOZ_ASSERT(!builder_->inlineCallInfo()); + + MDefinition* index = getOperand(indexId); + + auto* length = MArgumentsLength::New(alloc()); + add(length); + + index = addBoundsCheck(index, length); + + auto* load = MGetFrameArgument::New(alloc(), index); + add(load); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardNonDoubleType(ValOperandId inputId, + ValueType type) { + switch (type) { + case ValueType::String: + case ValueType::Symbol: + case ValueType::BigInt: + case ValueType::Int32: + case ValueType::Boolean: + return emitGuardTo(inputId, MIRTypeFromValueType(JSValueType(type))); + case ValueType::Undefined: + return emitGuardIsUndefined(inputId); + case ValueType::Null: + return emitGuardIsNull(inputId); + case ValueType::Double: + case ValueType::Magic: + case ValueType::PrivateGCThing: + case ValueType::Object: + break; + } + + MOZ_CRASH("unexpected type"); +} + +bool WarpCacheIRTranspiler::emitGuardTo(ValOperandId inputId, MIRType type) { + MDefinition* def = getOperand(inputId); + if (def->type() == type) { + return true; + } + + auto* ins = MUnbox::New(alloc(), def, type, MUnbox::Fallible); + add(ins); + + setOperand(inputId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardToObject(ValOperandId inputId) { + return emitGuardTo(inputId, MIRType::Object); +} + +bool WarpCacheIRTranspiler::emitGuardToString(ValOperandId inputId) { + return emitGuardTo(inputId, MIRType::String); +} + +bool WarpCacheIRTranspiler::emitGuardToSymbol(ValOperandId inputId) { + return emitGuardTo(inputId, MIRType::Symbol); +} + +bool WarpCacheIRTranspiler::emitGuardToBigInt(ValOperandId inputId) { + return emitGuardTo(inputId, MIRType::BigInt); +} + +bool WarpCacheIRTranspiler::emitGuardToBoolean(ValOperandId inputId) { + return emitGuardTo(inputId, MIRType::Boolean); +} + +bool WarpCacheIRTranspiler::emitGuardToInt32(ValOperandId inputId) { + return emitGuardTo(inputId, MIRType::Int32); +} + +bool WarpCacheIRTranspiler::emitGuardBooleanToInt32(ValOperandId inputId, + Int32OperandId resultId) { + MDefinition* input = getOperand(inputId); + + MDefinition* boolean; + if (input->type() == MIRType::Boolean) { + boolean = input; + } else { + auto* unbox = + MUnbox::New(alloc(), input, MIRType::Boolean, MUnbox::Fallible); + add(unbox); + boolean = unbox; + } + + auto* ins = MToIntegerInt32::New(alloc(), boolean); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitGuardIsNumber(ValOperandId inputId) { + // Prefer MToDouble because it gets further optimizations downstream. + MDefinition* def = getOperand(inputId); + if (def->type() == MIRType::Int32) { + auto* ins = MToDouble::New(alloc(), def); + add(ins); + + setOperand(inputId, ins); + return true; + } + + // MIRType::Double also implies int32 in Ion. + return emitGuardTo(inputId, MIRType::Double); +} + +bool WarpCacheIRTranspiler::emitGuardIsNullOrUndefined(ValOperandId inputId) { + MDefinition* input = getOperand(inputId); + if (input->type() == MIRType::Null || input->type() == MIRType::Undefined) { + return true; + } + + auto* ins = MGuardNullOrUndefined::New(alloc(), input); + add(ins); + + setOperand(inputId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIsNull(ValOperandId inputId) { + MDefinition* input = getOperand(inputId); + if (input->type() == MIRType::Null) { + return true; + } + + auto* ins = MGuardValue::New(alloc(), input, NullValue()); + add(ins); + setOperand(inputId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIsUndefined(ValOperandId inputId) { + MDefinition* input = getOperand(inputId); + if (input->type() == MIRType::Undefined) { + return true; + } + + auto* ins = MGuardValue::New(alloc(), input, UndefinedValue()); + add(ins); + setOperand(inputId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIsExtensible(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGuardIsExtensible::New(alloc(), obj); + add(ins); + setOperand(objId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardInt32IsNonNegative( + Int32OperandId indexId) { + MDefinition* index = getOperand(indexId); + + auto* ins = MGuardInt32IsNonNegative::New(alloc(), index); + add(ins); + setOperand(indexId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIndexGreaterThanDenseInitLength( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* ins = MGuardIndexGreaterThanDenseInitLength::New(alloc(), obj, index); + add(ins); + setOperand(indexId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGuardIndexIsValidUpdateOrAdd( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* ins = MGuardIndexIsValidUpdateOrAdd::New(alloc(), obj, index); + add(ins); + setOperand(indexId, ins); + return true; +} + +bool WarpCacheIRTranspiler::emitCallAddOrUpdateSparseElementHelper( + ObjOperandId objId, Int32OperandId idId, ValOperandId rhsId, bool strict) { + MDefinition* obj = getOperand(objId); + MDefinition* id = getOperand(idId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MCallAddOrUpdateSparseElement::New(alloc(), obj, id, rhs, strict); + addEffectful(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitGuardTagNotEqual(ValueTagOperandId lhsId, + ValueTagOperandId rhsId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MGuardTagNotEqual::New(alloc(), lhs, rhs); + add(ins); + + return true; +} + +bool WarpCacheIRTranspiler::emitGuardToInt32Index(ValOperandId inputId, + Int32OperandId resultId) { + MDefinition* input = getOperand(inputId); + auto* ins = + MToNumberInt32::New(alloc(), input, IntConversionInputKind::NumbersOnly); + + // ToPropertyKey(-0) is "0", so we can silently convert -0 to 0 here. + ins->setNeedsNegativeZeroCheck(false); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitGuardToTypedArrayIndex( + ValOperandId inputId, Int32OperandId resultId) { + MDefinition* input = getOperand(inputId); + + MDefinition* number; + if (input->type() == MIRType::Int32 || input->type() == MIRType::Double) { + number = input; + } else { + auto* unbox = + MUnbox::New(alloc(), input, MIRType::Double, MUnbox::Fallible); + add(unbox); + number = unbox; + } + + auto* ins = MTypedArrayIndexToInt32::New(alloc(), number); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitTruncateDoubleToUInt32( + NumberOperandId inputId, Int32OperandId resultId) { + MDefinition* input = getOperand(inputId); + auto* ins = MTruncateToInt32::New(alloc(), input); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitGuardToInt32ModUint32(ValOperandId valId, + Int32OperandId resultId) { + MDefinition* input = getOperand(valId); + auto* ins = MTruncateToInt32::New(alloc(), input); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitGuardToUint8Clamped(ValOperandId valId, + Int32OperandId resultId) { + MDefinition* input = getOperand(valId); + auto* ins = MClampToUint8::New(alloc(), input); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitToString(OperandId inputId, + StringOperandId resultId) { + MDefinition* input = getOperand(inputId); + auto* ins = + MToString::New(alloc(), input, MToString::SideEffectHandling::Bailout); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitCallInt32ToString(Int32OperandId inputId, + StringOperandId resultId) { + return emitToString(inputId, resultId); +} + +bool WarpCacheIRTranspiler::emitCallNumberToString(NumberOperandId inputId, + StringOperandId resultId) { + return emitToString(inputId, resultId); +} + +bool WarpCacheIRTranspiler::emitBooleanToString(BooleanOperandId inputId, + StringOperandId resultId) { + return emitToString(inputId, resultId); +} + +bool WarpCacheIRTranspiler::emitBooleanToNumber(BooleanOperandId inputId, + NumberOperandId resultId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MToDouble::New(alloc(), input); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitLoadInt32Result(Int32OperandId valId) { + MDefinition* val = getOperand(valId); + MOZ_ASSERT(val->type() == MIRType::Int32); + pushResult(val); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadDoubleResult(NumberOperandId valId) { + MDefinition* val = getOperand(valId); + MOZ_ASSERT(val->type() == MIRType::Double); + pushResult(val); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadBigIntResult(BigIntOperandId valId) { + MDefinition* val = getOperand(valId); + MOZ_ASSERT(val->type() == MIRType::BigInt); + pushResult(val); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadObjectResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + MOZ_ASSERT(obj->type() == MIRType::Object); + pushResult(obj); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadStringResult(StringOperandId strId) { + MDefinition* str = getOperand(strId); + MOZ_ASSERT(str->type() == MIRType::String); + pushResult(str); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadSymbolResult(SymbolOperandId symId) { + MDefinition* sym = getOperand(symId); + MOZ_ASSERT(sym->type() == MIRType::Symbol); + pushResult(sym); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadUndefinedResult() { + pushResult(constant(UndefinedValue())); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadBooleanResult(bool val) { + pushResult(constant(BooleanValue(val))); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadInt32Constant(uint32_t valOffset, + Int32OperandId resultId) { + int32_t val = int32StubField(valOffset); + auto* valConst = constant(Int32Value(val)); + return defineOperand(resultId, valConst); +} + +bool WarpCacheIRTranspiler::emitLoadBooleanConstant(bool val, + BooleanOperandId resultId) { + auto* valConst = constant(BooleanValue(val)); + return defineOperand(resultId, valConst); +} + +bool WarpCacheIRTranspiler::emitLoadUndefined(ValOperandId resultId) { + auto* valConst = constant(UndefinedValue()); + return defineOperand(resultId, valConst); +} + +bool WarpCacheIRTranspiler::emitLoadConstantString(uint32_t strOffset, + StringOperandId resultId) { + JSString* val = stringStubField(strOffset); + auto* valConst = constant(StringValue(val)); + return defineOperand(resultId, valConst); +} + +bool WarpCacheIRTranspiler::emitLoadConstantStringResult(uint32_t strOffset) { + JSString* val = stringStubField(strOffset); + auto* valConst = constant(StringValue(val)); + pushResult(valConst); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadTypeOfObjectResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + auto* ins = MTypeOf::New(alloc(), obj); + add(ins); + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadEnclosingEnvironment( + ObjOperandId objId, ObjOperandId resultId) { + MDefinition* env = getOperand(objId); + auto* ins = MEnclosingEnvironment::New(alloc(), env); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitLoadObject(ObjOperandId resultId, + uint32_t objOffset) { + MInstruction* ins = objectStubField(objOffset); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitLoadProto(ObjOperandId objId, + ObjOperandId resultId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MObjectStaticProto::New(alloc(), obj); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitLoadInstanceOfObjectResult( + ValOperandId lhsId, ObjOperandId protoId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* proto = getOperand(protoId); + + auto* instanceOf = MInstanceOf::New(alloc(), lhs, proto); + addEffectful(instanceOf); + + pushResult(instanceOf); + return resumeAfter(instanceOf); +} + +bool WarpCacheIRTranspiler::emitLoadValueTag(ValOperandId valId, + ValueTagOperandId resultId) { + MDefinition* val = getOperand(valId); + + auto* ins = MLoadValueTag::New(alloc(), val); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitLoadDynamicSlotResult(ObjOperandId objId, + uint32_t offsetOffset) { + int32_t offset = int32StubField(offsetOffset); + + MDefinition* obj = getOperand(objId); + size_t slotIndex = NativeObject::getDynamicSlotIndexFromOffset(offset); + + auto* slots = MSlots::New(alloc(), obj); + add(slots); + + auto* load = MLoadDynamicSlot::New(alloc(), slots, slotIndex); + add(load); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadFixedSlotResult(ObjOperandId objId, + uint32_t offsetOffset) { + int32_t offset = int32StubField(offsetOffset); + + MDefinition* obj = getOperand(objId); + uint32_t slotIndex = NativeObject::getFixedSlotIndexFromOffset(offset); + + auto* load = MLoadFixedSlot::New(alloc(), obj, slotIndex); + add(load); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadFixedSlotTypedResult(ObjOperandId objId, + uint32_t offsetOffset, + ValueType type) { + int32_t offset = int32StubField(offsetOffset); + + MDefinition* obj = getOperand(objId); + uint32_t slotIndex = NativeObject::getFixedSlotIndexFromOffset(offset); + + auto* load = MLoadFixedSlot::New(alloc(), obj, slotIndex); + load->setResultType(MIRTypeFromValueType(JSValueType(type))); + add(load); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadEnvironmentFixedSlotResult( + ObjOperandId objId, uint32_t offsetOffset) { + int32_t offset = int32StubField(offsetOffset); + + MDefinition* obj = getOperand(objId); + uint32_t slotIndex = NativeObject::getFixedSlotIndexFromOffset(offset); + + auto* load = MLoadFixedSlot::New(alloc(), obj, slotIndex); + add(load); + + auto* lexicalCheck = MLexicalCheck::New(alloc(), load); + add(lexicalCheck); + + if (snapshot().bailoutInfo().failedLexicalCheck()) { + lexicalCheck->setNotMovable(); + } + + pushResult(lexicalCheck); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadEnvironmentDynamicSlotResult( + ObjOperandId objId, uint32_t offsetOffset) { + int32_t offset = int32StubField(offsetOffset); + + MDefinition* obj = getOperand(objId); + size_t slotIndex = NativeObject::getDynamicSlotIndexFromOffset(offset); + + auto* slots = MSlots::New(alloc(), obj); + add(slots); + + auto* load = MLoadDynamicSlot::New(alloc(), slots, slotIndex); + add(load); + + auto* lexicalCheck = MLexicalCheck::New(alloc(), load); + add(lexicalCheck); + + if (snapshot().bailoutInfo().failedLexicalCheck()) { + lexicalCheck->setNotMovable(); + } + + pushResult(lexicalCheck); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadInt32ArrayLengthResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* length = MArrayLength::New(alloc(), elements); + add(length); + + pushResult(length); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadInt32ArrayLength(ObjOperandId objId, + Int32OperandId resultId) { + MDefinition* obj = getOperand(objId); + + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* length = MArrayLength::New(alloc(), elements); + add(length); + + return defineOperand(resultId, length); +} + +bool WarpCacheIRTranspiler::emitLoadArgumentsObjectArgResult( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* load = MLoadArgumentsObjectArg::New(alloc(), obj, index); + add(load); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadArgumentsObjectLengthResult( + ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* length = MArgumentsObjectLength::New(alloc(), obj); + add(length); + + pushResult(length); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadFunctionLengthResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* length = MFunctionLength::New(alloc(), obj); + add(length); + + pushResult(length); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadFunctionNameResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* name = MFunctionName::New(alloc(), obj); + add(name); + + pushResult(name); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadArrayBufferByteLengthInt32Result( + ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* length = MArrayBufferByteLengthInt32::New(alloc(), obj); + add(length); + + pushResult(length); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadTypedArrayLengthResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + pushResult(length); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadStringLengthResult(StringOperandId strId) { + MDefinition* str = getOperand(strId); + + auto* length = MStringLength::New(alloc(), str); + add(length); + + pushResult(length); + return true; +} + +MInstruction* WarpCacheIRTranspiler::addBoundsCheck(MDefinition* index, + MDefinition* length) { + MInstruction* check = MBoundsCheck::New(alloc(), index, length); + add(check); + + if (snapshot().bailoutInfo().failedBoundsCheck()) { + check->setNotMovable(); + } + + if (JitOptions.spectreIndexMasking) { + // Use a separate MIR instruction for the index masking. Doing this as + // part of MBoundsCheck would be unsound because bounds checks can be + // optimized or eliminated completely. Consider this: + // + // for (var i = 0; i < x; i++) + // res = arr[i]; + // + // If we can prove |x < arr.length|, we are able to eliminate the bounds + // check, but we should not get rid of the index masking because the + // |i < x| branch could still be mispredicted. + // + // Using a separate instruction lets us eliminate the bounds check + // without affecting the index masking. + check = MSpectreMaskIndex::New(alloc(), check, length); + add(check); + } + + return check; +} + +bool WarpCacheIRTranspiler::emitLoadDenseElementResult(ObjOperandId objId, + Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* length = MInitializedLength::New(alloc(), elements); + add(length); + + index = addBoundsCheck(index, length); + + bool needsHoleCheck = true; + auto* load = MLoadElement::New(alloc(), elements, index, needsHoleCheck); + add(load); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadDenseElementHoleResult( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* length = MInitializedLength::New(alloc(), elements); + add(length); + + bool needsHoleCheck = true; + auto* load = + MLoadElementHole::New(alloc(), elements, index, length, needsHoleCheck); + add(load); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitCallGetSparseElementResult( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* call = MCallGetSparseElement::New(alloc(), obj, index); + addEffectful(call); + + pushResult(call); + return resumeAfter(call); +} + +bool WarpCacheIRTranspiler::emitCallNativeGetElementResult( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* call = MCallNativeGetElement::New(alloc(), obj, index); + addEffectful(call); + + pushResult(call); + return resumeAfter(call); +} + +bool WarpCacheIRTranspiler::emitLoadDenseElementExistsResult( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + // Get the elements vector. + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* length = MInitializedLength::New(alloc(), elements); + add(length); + + // Check if id < initLength. + index = addBoundsCheck(index, length); + + // And check elem[id] is not a hole. + auto* guard = MGuardElementNotHole::New(alloc(), elements, index); + add(guard); + + pushResult(constant(BooleanValue(true))); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadDenseElementHoleExistsResult( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + // Get the elements vector. + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* length = MInitializedLength::New(alloc(), elements); + add(length); + + // Check if id < initLength and elem[id] not a hole. + auto* ins = MInArray::New(alloc(), elements, index, length, obj); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitCallObjectHasSparseElementResult( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* ins = MCallObjectHasSparseElement::New(alloc(), obj, index); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadTypedArrayElementExistsResult( + ObjOperandId objId, Int32OperandId indexId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + // Unsigned comparison to catch negative indices. + auto* ins = + MCompare::New(alloc(), index, length, JSOp::Lt, MCompare::Compare_UInt32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadTypedArrayElementResult( + ObjOperandId objId, Int32OperandId indexId, Scalar::Type elementType, + bool handleOOB, bool allowDoubleForUint32) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + if (handleOOB) { + auto* load = MLoadTypedArrayElementHole::New( + alloc(), obj, index, elementType, allowDoubleForUint32); + add(load); + + pushResult(load); + return true; + } + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + index = addBoundsCheck(index, length); + + auto* elements = MArrayBufferViewElements::New(alloc(), obj); + add(elements); + + auto* load = MLoadUnboxedScalar::New(alloc(), elements, index, elementType); + load->setResultType( + MIRTypeForArrayBufferViewRead(elementType, allowDoubleForUint32)); + add(load); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadStringCharResult(StringOperandId strId, + Int32OperandId indexId) { + MDefinition* str = getOperand(strId); + MDefinition* index = getOperand(indexId); + + auto* length = MStringLength::New(alloc(), str); + add(length); + + index = addBoundsCheck(index, length); + + auto* charCode = MCharCodeAt::New(alloc(), str, index); + add(charCode); + + auto* fromCharCode = MFromCharCode::New(alloc(), charCode); + add(fromCharCode); + + pushResult(fromCharCode); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadStringCharCodeResult( + StringOperandId strId, Int32OperandId indexId) { + MDefinition* str = getOperand(strId); + MDefinition* index = getOperand(indexId); + + auto* length = MStringLength::New(alloc(), str); + add(length); + + index = addBoundsCheck(index, length); + + auto* charCode = MCharCodeAt::New(alloc(), str, index); + add(charCode); + + pushResult(charCode); + return true; +} + +bool WarpCacheIRTranspiler::emitNewStringObjectResult( + uint32_t templateObjectOffset, StringOperandId strId) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + MDefinition* string = getOperand(strId); + + auto* obj = MNewStringObject::New(alloc(), string, templateObj); + addEffectful(obj); + + pushResult(obj); + return resumeAfter(obj); +} + +bool WarpCacheIRTranspiler::emitStringFromCharCodeResult( + Int32OperandId codeId) { + MDefinition* code = getOperand(codeId); + + auto* fromCharCode = MFromCharCode::New(alloc(), code); + add(fromCharCode); + + pushResult(fromCharCode); + return true; +} + +bool WarpCacheIRTranspiler::emitStringFromCodePointResult( + Int32OperandId codeId) { + MDefinition* code = getOperand(codeId); + + auto* fromCodePoint = MFromCodePoint::New(alloc(), code); + add(fromCodePoint); + + pushResult(fromCodePoint); + return true; +} + +bool WarpCacheIRTranspiler::emitStringToLowerCaseResult(StringOperandId strId) { + MDefinition* str = getOperand(strId); + + auto* convert = + MStringConvertCase::New(alloc(), str, MStringConvertCase::LowerCase); + add(convert); + + pushResult(convert); + return true; +} + +bool WarpCacheIRTranspiler::emitStringToUpperCaseResult(StringOperandId strId) { + MDefinition* str = getOperand(strId); + + auto* convert = + MStringConvertCase::New(alloc(), str, MStringConvertCase::UpperCase); + add(convert); + + pushResult(convert); + return true; +} + +bool WarpCacheIRTranspiler::emitStoreDynamicSlot(ObjOperandId objId, + uint32_t offsetOffset, + ValOperandId rhsId) { + int32_t offset = int32StubField(offsetOffset); + + MDefinition* obj = getOperand(objId); + size_t slotIndex = NativeObject::getDynamicSlotIndexFromOffset(offset); + MDefinition* rhs = getOperand(rhsId); + + auto* barrier = MPostWriteBarrier::New(alloc(), obj, rhs); + add(barrier); + + auto* slots = MSlots::New(alloc(), obj); + add(slots); + + auto* store = MStoreDynamicSlot::NewBarriered(alloc(), slots, slotIndex, rhs); + addEffectful(store); + return resumeAfter(store); +} + +bool WarpCacheIRTranspiler::emitStoreFixedSlot(ObjOperandId objId, + uint32_t offsetOffset, + ValOperandId rhsId) { + int32_t offset = int32StubField(offsetOffset); + + MDefinition* obj = getOperand(objId); + size_t slotIndex = NativeObject::getFixedSlotIndexFromOffset(offset); + MDefinition* rhs = getOperand(rhsId); + + auto* barrier = MPostWriteBarrier::New(alloc(), obj, rhs); + add(barrier); + + auto* store = MStoreFixedSlot::NewBarriered(alloc(), obj, slotIndex, rhs); + addEffectful(store); + return resumeAfter(store); +} + +bool WarpCacheIRTranspiler::emitStoreFixedSlotUndefinedResult( + ObjOperandId objId, uint32_t offsetOffset, ValOperandId rhsId) { + int32_t offset = int32StubField(offsetOffset); + + MDefinition* obj = getOperand(objId); + size_t slotIndex = NativeObject::getFixedSlotIndexFromOffset(offset); + MDefinition* rhs = getOperand(rhsId); + + auto* barrier = MPostWriteBarrier::New(alloc(), obj, rhs); + add(barrier); + + auto* store = MStoreFixedSlot::NewBarriered(alloc(), obj, slotIndex, rhs); + addEffectful(store); + + auto* undef = constant(UndefinedValue()); + pushResult(undef); + + return resumeAfter(store); +} + +bool WarpCacheIRTranspiler::emitAddAndStoreSlotShared( + MAddAndStoreSlot::Kind kind, ObjOperandId objId, uint32_t offsetOffset, + ValOperandId rhsId, uint32_t newShapeOffset) { + int32_t offset = int32StubField(offsetOffset); + Shape* shape = shapeStubField(newShapeOffset); + + MDefinition* obj = getOperand(objId); + MDefinition* rhs = getOperand(rhsId); + + auto* barrier = MPostWriteBarrier::New(alloc(), obj, rhs); + add(barrier); + + auto* addAndStore = + MAddAndStoreSlot::New(alloc(), obj, rhs, kind, offset, shape); + addEffectful(addAndStore); + + return resumeAfter(addAndStore); +} + +bool WarpCacheIRTranspiler::emitAddAndStoreFixedSlot(ObjOperandId objId, + uint32_t offsetOffset, + ValOperandId rhsId, + uint32_t newShapeOffset) { + return emitAddAndStoreSlotShared(MAddAndStoreSlot::Kind::FixedSlot, objId, + offsetOffset, rhsId, newShapeOffset); +} + +bool WarpCacheIRTranspiler::emitAddAndStoreDynamicSlot( + ObjOperandId objId, uint32_t offsetOffset, ValOperandId rhsId, + uint32_t newShapeOffset) { + return emitAddAndStoreSlotShared(MAddAndStoreSlot::Kind::DynamicSlot, objId, + offsetOffset, rhsId, newShapeOffset); +} + +bool WarpCacheIRTranspiler::emitAllocateAndStoreDynamicSlot( + ObjOperandId objId, uint32_t offsetOffset, ValOperandId rhsId, + uint32_t newShapeOffset, uint32_t numNewSlotsOffset) { + int32_t offset = int32StubField(offsetOffset); + Shape* shape = shapeStubField(newShapeOffset); + uint32_t numNewSlots = uint32StubField(numNewSlotsOffset); + + MDefinition* obj = getOperand(objId); + MDefinition* rhs = getOperand(rhsId); + + auto* barrier = MPostWriteBarrier::New(alloc(), obj, rhs); + add(barrier); + + auto* allocateAndStore = + MAllocateAndStoreSlot::New(alloc(), obj, rhs, offset, shape, numNewSlots); + addEffectful(allocateAndStore); + + return resumeAfter(allocateAndStore); +} + +bool WarpCacheIRTranspiler::emitStoreDenseElement(ObjOperandId objId, + Int32OperandId indexId, + ValOperandId rhsId) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + MDefinition* rhs = getOperand(rhsId); + + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* length = MInitializedLength::New(alloc(), elements); + add(length); + + index = addBoundsCheck(index, length); + + auto* barrier = MPostWriteElementBarrier::New(alloc(), obj, rhs, index); + add(barrier); + + bool needsHoleCheck = true; + auto* store = + MStoreElement::New(alloc(), elements, index, rhs, needsHoleCheck); + store->setNeedsBarrier(); + addEffectful(store); + return resumeAfter(store); +} + +bool WarpCacheIRTranspiler::emitStoreDenseElementHole(ObjOperandId objId, + Int32OperandId indexId, + ValOperandId rhsId, + bool handleAdd) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + MDefinition* rhs = getOperand(rhsId); + + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* barrier = MPostWriteElementBarrier::New(alloc(), obj, rhs, index); + add(barrier); + + MInstruction* store; + MStoreElementCommon* common; + if (handleAdd) { + // TODO(post-Warp): Consider changing MStoreElementHole to match IC code. + auto* ins = MStoreElementHole::New(alloc(), obj, elements, index, rhs); + store = ins; + common = ins; + } else { + auto* length = MInitializedLength::New(alloc(), elements); + add(length); + + index = addBoundsCheck(index, length); + + bool needsHoleCheck = false; + auto* ins = + MStoreElement::New(alloc(), elements, index, rhs, needsHoleCheck); + store = ins; + common = ins; + } + common->setNeedsBarrier(); + addEffectful(store); + + return resumeAfter(store); +} + +bool WarpCacheIRTranspiler::emitStoreTypedArrayElement(ObjOperandId objId, + Scalar::Type elementType, + Int32OperandId indexId, + uint32_t rhsId, + bool handleOOB) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + MDefinition* rhs = getOperand(ValOperandId(rhsId)); + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + if (!handleOOB) { + // MStoreTypedArrayElementHole does the bounds checking. + index = addBoundsCheck(index, length); + } + + auto* elements = MArrayBufferViewElements::New(alloc(), obj); + add(elements); + + MInstruction* store; + if (handleOOB) { + store = MStoreTypedArrayElementHole::New(alloc(), elements, length, index, + rhs, elementType); + } else { + store = + MStoreUnboxedScalar::New(alloc(), elements, index, rhs, elementType); + } + addEffectful(store); + return resumeAfter(store); +} + +void WarpCacheIRTranspiler::addDataViewData(MDefinition* obj, Scalar::Type type, + MDefinition** offset, + MInstruction** elements) { + MInstruction* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + // Adjust the length to account for accesses near the end of the dataview. + if (size_t byteSize = Scalar::byteSize(type); byteSize > 1) { + // To ensure |0 <= offset && offset + byteSize <= length|, we can either + // emit |BoundsCheck(offset, length)| followed by + // |BoundsCheck(offset + (byteSize - 1), length)|, or alternatively emit + // |BoundsCheck(offset, Max(length - (byteSize - 1), 0))|. The latter should + // result in faster code when LICM moves the length adjustment and also + // ensures Spectre index masking occurs after all bounds checks. + + auto* byteSizeMinusOne = MConstant::New(alloc(), Int32Value(byteSize - 1)); + add(byteSizeMinusOne); + + length = MSub::New(alloc(), length, byteSizeMinusOne, MIRType::Int32); + length->toSub()->setTruncateKind(TruncateKind::Truncate); + add(length); + + // |length| mustn't be negative for MBoundsCheck. + auto* zero = MConstant::New(alloc(), Int32Value(0)); + add(zero); + + length = MMinMax::New(alloc(), length, zero, MIRType::Int32, true); + add(length); + } + + *offset = addBoundsCheck(*offset, length); + + *elements = MArrayBufferViewElements::New(alloc(), obj); + add(*elements); +} + +bool WarpCacheIRTranspiler::emitLoadDataViewValueResult( + ObjOperandId objId, Int32OperandId offsetId, + BooleanOperandId littleEndianId, Scalar::Type elementType, + bool allowDoubleForUint32) { + MDefinition* obj = getOperand(objId); + MDefinition* offset = getOperand(offsetId); + MDefinition* littleEndian = getOperand(littleEndianId); + + // Add bounds check and get the DataViewObject's elements. + MInstruction* elements; + addDataViewData(obj, elementType, &offset, &elements); + + // Load the element. + MInstruction* load; + if (Scalar::byteSize(elementType) == 1) { + load = MLoadUnboxedScalar::New(alloc(), elements, offset, elementType); + } else { + load = MLoadDataViewElement::New(alloc(), elements, offset, littleEndian, + elementType); + } + add(load); + + MIRType knownType = + MIRTypeForArrayBufferViewRead(elementType, allowDoubleForUint32); + load->setResultType(knownType); + + pushResult(load); + return true; +} + +bool WarpCacheIRTranspiler::emitStoreDataViewValueResult( + ObjOperandId objId, Int32OperandId offsetId, uint32_t valueId, + BooleanOperandId littleEndianId, Scalar::Type elementType) { + MDefinition* obj = getOperand(objId); + MDefinition* offset = getOperand(offsetId); + MDefinition* value = getOperand(ValOperandId(valueId)); + MDefinition* littleEndian = getOperand(littleEndianId); + + // Add bounds check and get the DataViewObject's elements. + MInstruction* elements; + addDataViewData(obj, elementType, &offset, &elements); + + // Store the element. + MInstruction* store; + if (Scalar::byteSize(elementType) == 1) { + store = + MStoreUnboxedScalar::New(alloc(), elements, offset, value, elementType); + } else { + store = MStoreDataViewElement::New(alloc(), elements, offset, value, + littleEndian, elementType); + } + addEffectful(store); + + pushResult(constant(UndefinedValue())); + + return resumeAfter(store); +} + +bool WarpCacheIRTranspiler::emitInt32IncResult(Int32OperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* constOne = MConstant::New(alloc(), Int32Value(1)); + add(constOne); + + auto* ins = MAdd::New(alloc(), input, constOne, MIRType::Int32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitDoubleIncResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* constOne = MConstant::New(alloc(), DoubleValue(1.0)); + add(constOne); + + auto* ins = MAdd::New(alloc(), input, constOne, MIRType::Double); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitInt32DecResult(Int32OperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* constOne = MConstant::New(alloc(), Int32Value(1)); + add(constOne); + + auto* ins = MSub::New(alloc(), input, constOne, MIRType::Int32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitDoubleDecResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* constOne = MConstant::New(alloc(), DoubleValue(1.0)); + add(constOne); + + auto* ins = MSub::New(alloc(), input, constOne, MIRType::Double); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitInt32NegationResult(Int32OperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* constNegOne = MConstant::New(alloc(), Int32Value(-1)); + add(constNegOne); + + auto* ins = MMul::New(alloc(), input, constNegOne, MIRType::Int32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitDoubleNegationResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* constNegOne = MConstant::New(alloc(), DoubleValue(-1.0)); + add(constNegOne); + + auto* ins = MMul::New(alloc(), input, constNegOne, MIRType::Double); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitInt32NotResult(Int32OperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MBitNot::New(alloc(), input); + add(ins); + + pushResult(ins); + return true; +} + +template <typename T> +bool WarpCacheIRTranspiler::emitDoubleBinaryArithResult(NumberOperandId lhsId, + NumberOperandId rhsId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = T::New(alloc(), lhs, rhs, MIRType::Double); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitDoubleAddResult(NumberOperandId lhsId, + NumberOperandId rhsId) { + return emitDoubleBinaryArithResult<MAdd>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitDoubleSubResult(NumberOperandId lhsId, + NumberOperandId rhsId) { + return emitDoubleBinaryArithResult<MSub>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitDoubleMulResult(NumberOperandId lhsId, + NumberOperandId rhsId) { + return emitDoubleBinaryArithResult<MMul>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitDoubleDivResult(NumberOperandId lhsId, + NumberOperandId rhsId) { + return emitDoubleBinaryArithResult<MDiv>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitDoubleModResult(NumberOperandId lhsId, + NumberOperandId rhsId) { + return emitDoubleBinaryArithResult<MMod>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitDoublePowResult(NumberOperandId lhsId, + NumberOperandId rhsId) { + return emitDoubleBinaryArithResult<MPow>(lhsId, rhsId); +} + +template <typename T> +bool WarpCacheIRTranspiler::emitInt32BinaryArithResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = T::New(alloc(), lhs, rhs, MIRType::Int32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitInt32AddResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MAdd>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32SubResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MSub>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32MulResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MMul>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32DivResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MDiv>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32ModResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MMod>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32PowResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MPow>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32BitOrResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MBitOr>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32BitXorResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MBitXor>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32BitAndResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MBitAnd>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32LeftShiftResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MLsh>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32RightShiftResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitInt32BinaryArithResult<MRsh>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitInt32URightShiftResult(Int32OperandId lhsId, + Int32OperandId rhsId, + bool allowDouble) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + MIRType specialization = allowDouble ? MIRType::Double : MIRType::Int32; + auto* ins = MUrsh::New(alloc(), lhs, rhs, specialization); + add(ins); + + pushResult(ins); + return true; +} + +template <typename T> +bool WarpCacheIRTranspiler::emitBigIntBinaryArithResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = T::New(alloc(), lhs, rhs); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitBigIntAddResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithResult<MBigIntAdd>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntSubResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithResult<MBigIntSub>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntMulResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithResult<MBigIntMul>(lhsId, rhsId); +} + +template <typename T> +bool WarpCacheIRTranspiler::emitBigIntBinaryArithEffectfulResult( + BigIntOperandId lhsId, BigIntOperandId rhsId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = T::New(alloc(), lhs, rhs); + + if (ins->isEffectful()) { + addEffectful(ins); + + pushResult(ins); + return resumeAfter(ins); + } + + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitBigIntDivResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithEffectfulResult<MBigIntDiv>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntModResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithEffectfulResult<MBigIntMod>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntPowResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithEffectfulResult<MBigIntPow>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntBitAndResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithResult<MBigIntBitAnd>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntBitOrResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithResult<MBigIntBitOr>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntBitXorResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithResult<MBigIntBitXor>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntLeftShiftResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithResult<MBigIntLsh>(lhsId, rhsId); +} + +bool WarpCacheIRTranspiler::emitBigIntRightShiftResult(BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitBigIntBinaryArithResult<MBigIntRsh>(lhsId, rhsId); +} + +template <typename T> +bool WarpCacheIRTranspiler::emitBigIntUnaryArithResult( + BigIntOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = T::New(alloc(), input); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitBigIntIncResult(BigIntOperandId inputId) { + return emitBigIntUnaryArithResult<MBigIntIncrement>(inputId); +} + +bool WarpCacheIRTranspiler::emitBigIntDecResult(BigIntOperandId inputId) { + return emitBigIntUnaryArithResult<MBigIntDecrement>(inputId); +} + +bool WarpCacheIRTranspiler::emitBigIntNegationResult(BigIntOperandId inputId) { + return emitBigIntUnaryArithResult<MBigIntNegate>(inputId); +} + +bool WarpCacheIRTranspiler::emitBigIntNotResult(BigIntOperandId inputId) { + return emitBigIntUnaryArithResult<MBigIntBitNot>(inputId); +} + +bool WarpCacheIRTranspiler::emitCallStringConcatResult(StringOperandId lhsId, + StringOperandId rhsId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MConcat::New(alloc(), lhs, rhs); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitCompareResult( + JSOp op, OperandId lhsId, OperandId rhsId, + MCompare::CompareType compareType) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MCompare::New(alloc(), lhs, rhs, op, compareType); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitCompareInt32Result(JSOp op, + Int32OperandId lhsId, + Int32OperandId rhsId) { + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_Int32); +} + +bool WarpCacheIRTranspiler::emitCompareDoubleResult(JSOp op, + NumberOperandId lhsId, + NumberOperandId rhsId) { + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_Double); +} + +bool WarpCacheIRTranspiler::emitCompareObjectResult(JSOp op, ObjOperandId lhsId, + ObjOperandId rhsId) { + MOZ_ASSERT(IsEqualityOp(op)); + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_Object); +} + +bool WarpCacheIRTranspiler::emitCompareStringResult(JSOp op, + StringOperandId lhsId, + StringOperandId rhsId) { + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_String); +} + +bool WarpCacheIRTranspiler::emitCompareSymbolResult(JSOp op, + SymbolOperandId lhsId, + SymbolOperandId rhsId) { + MOZ_ASSERT(IsEqualityOp(op)); + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_Symbol); +} + +bool WarpCacheIRTranspiler::emitCompareBigIntResult(JSOp op, + BigIntOperandId lhsId, + BigIntOperandId rhsId) { + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_BigInt); +} + +bool WarpCacheIRTranspiler::emitCompareBigIntInt32Result(JSOp op, + BigIntOperandId lhsId, + Int32OperandId rhsId) { + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_BigInt_Int32); +} + +bool WarpCacheIRTranspiler::emitCompareBigIntNumberResult( + JSOp op, BigIntOperandId lhsId, NumberOperandId rhsId) { + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_BigInt_Double); +} + +bool WarpCacheIRTranspiler::emitCompareBigIntStringResult( + JSOp op, BigIntOperandId lhsId, StringOperandId rhsId) { + return emitCompareResult(op, lhsId, rhsId, MCompare::Compare_BigInt_String); +} + +bool WarpCacheIRTranspiler::emitCompareNullUndefinedResult( + JSOp op, bool isUndefined, ValOperandId inputId) { + MDefinition* input = getOperand(inputId); + + MOZ_ASSERT(IsEqualityOp(op)); + + // A previously emitted guard ensures that one side of the comparison + // is null or undefined. + MDefinition* cst = + isUndefined ? constant(UndefinedValue()) : constant(NullValue()); + auto compareType = + isUndefined ? MCompare::Compare_Undefined : MCompare::Compare_Null; + auto* ins = MCompare::New(alloc(), input, cst, op, compareType); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitCompareDoubleSameValueResult( + NumberOperandId lhsId, NumberOperandId rhsId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* sameValue = MSameValue::New(alloc(), lhs, rhs); + add(sameValue); + + pushResult(sameValue); + return true; +} + +bool WarpCacheIRTranspiler::emitIndirectTruncateInt32Result( + Int32OperandId valId) { + MDefinition* val = getOperand(valId); + MOZ_ASSERT(val->type() == MIRType::Int32); + + auto* truncate = + MLimitedTruncate::New(alloc(), val, TruncateKind::IndirectTruncate); + add(truncate); + + pushResult(truncate); + return true; +} + +bool WarpCacheIRTranspiler::emitMathHypot2NumberResult( + NumberOperandId firstId, NumberOperandId secondId) { + MDefinitionVector vector(alloc()); + if (!vector.reserve(2)) { + return false; + } + + vector.infallibleAppend(getOperand(firstId)); + vector.infallibleAppend(getOperand(secondId)); + + auto* ins = MHypot::New(alloc(), vector); + if (!ins) { + return false; + } + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathHypot3NumberResult( + NumberOperandId firstId, NumberOperandId secondId, + NumberOperandId thirdId) { + MDefinitionVector vector(alloc()); + if (!vector.reserve(3)) { + return false; + } + + vector.infallibleAppend(getOperand(firstId)); + vector.infallibleAppend(getOperand(secondId)); + vector.infallibleAppend(getOperand(thirdId)); + + auto* ins = MHypot::New(alloc(), vector); + if (!ins) { + return false; + } + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathHypot4NumberResult( + NumberOperandId firstId, NumberOperandId secondId, NumberOperandId thirdId, + NumberOperandId fourthId) { + MDefinitionVector vector(alloc()); + if (!vector.reserve(4)) { + return false; + } + + vector.infallibleAppend(getOperand(firstId)); + vector.infallibleAppend(getOperand(secondId)); + vector.infallibleAppend(getOperand(thirdId)); + vector.infallibleAppend(getOperand(fourthId)); + + auto* ins = MHypot::New(alloc(), vector); + if (!ins) { + return false; + } + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathRandomResult(uint32_t rngOffset) { +#ifdef DEBUG + // CodeGenerator uses CompileRealm::addressOfRandomNumberGenerator. Assert it + // matches the RNG pointer stored in the stub field. + const void* rng = rawPointerField(rngOffset); + MOZ_ASSERT(rng == mirGen().realm->addressOfRandomNumberGenerator()); +#endif + + auto* ins = MRandom::New(alloc()); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitInt32MinMax(bool isMax, Int32OperandId firstId, + Int32OperandId secondId, + Int32OperandId resultId) { + MDefinition* first = getOperand(firstId); + MDefinition* second = getOperand(secondId); + + auto* ins = MMinMax::New(alloc(), first, second, MIRType::Int32, isMax); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitNumberMinMax(bool isMax, + NumberOperandId firstId, + NumberOperandId secondId, + NumberOperandId resultId) { + MDefinition* first = getOperand(firstId); + MDefinition* second = getOperand(secondId); + + auto* ins = MMinMax::New(alloc(), first, second, MIRType::Double, isMax); + add(ins); + + return defineOperand(resultId, ins); +} + +bool WarpCacheIRTranspiler::emitMathAbsInt32Result(Int32OperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MAbs::New(alloc(), input, MIRType::Int32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathAbsNumberResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MAbs::New(alloc(), input, MIRType::Double); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathClz32Result(Int32OperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MClz::New(alloc(), input, MIRType::Int32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathSignInt32Result(Int32OperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MSign::New(alloc(), input, MIRType::Int32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathSignNumberResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MSign::New(alloc(), input, MIRType::Double); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathSignNumberToInt32Result( + NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MSign::New(alloc(), input, MIRType::Int32); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathImulResult(Int32OperandId lhsId, + Int32OperandId rhsId) { + MDefinition* lhs = getOperand(lhsId); + MDefinition* rhs = getOperand(rhsId); + + auto* ins = MMul::New(alloc(), lhs, rhs, MIRType::Int32, MMul::Integer); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathFloorToInt32Result( + NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MFloor::New(alloc(), input); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathCeilToInt32Result(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MCeil::New(alloc(), input); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathTruncToInt32Result( + NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MTrunc::New(alloc(), input); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathRoundToInt32Result( + NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MRound::New(alloc(), input); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathSqrtNumberResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MSqrt::New(alloc(), input, MIRType::Double); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathFRoundNumberResult( + NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + auto* ins = MToFloat32::New(alloc(), input); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathAtan2NumberResult(NumberOperandId yId, + NumberOperandId xId) { + MDefinition* y = getOperand(yId); + MDefinition* x = getOperand(xId); + + auto* ins = MAtan2::New(alloc(), y, x); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathFunctionNumberResult( + NumberOperandId inputId, UnaryMathFunction fun) { + MDefinition* input = getOperand(inputId); + + auto* ins = MMathFunction::New(alloc(), input, fun); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathFloorNumberResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + MInstruction* ins; + if (MNearbyInt::HasAssemblerSupport(RoundingMode::Down)) { + ins = MNearbyInt::New(alloc(), input, MIRType::Double, RoundingMode::Down); + } else { + ins = MMathFunction::New(alloc(), input, UnaryMathFunction::Floor); + } + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathCeilNumberResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + MInstruction* ins; + if (MNearbyInt::HasAssemblerSupport(RoundingMode::Up)) { + ins = MNearbyInt::New(alloc(), input, MIRType::Double, RoundingMode::Up); + } else { + ins = MMathFunction::New(alloc(), input, UnaryMathFunction::Ceil); + } + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitMathTruncNumberResult(NumberOperandId inputId) { + MDefinition* input = getOperand(inputId); + + MInstruction* ins; + if (MNearbyInt::HasAssemblerSupport(RoundingMode::TowardsZero)) { + ins = MNearbyInt::New(alloc(), input, MIRType::Double, + RoundingMode::TowardsZero); + } else { + ins = MMathFunction::New(alloc(), input, UnaryMathFunction::Trunc); + } + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitObjectToStringResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MObjectClassToString::New(alloc(), obj); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitReflectGetPrototypeOfResult( + ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MGetPrototypeOf::New(alloc(), obj); + addEffectful(ins); + pushResult(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitArrayPush(ObjOperandId objId, + ValOperandId rhsId) { + MDefinition* obj = getOperand(objId); + MDefinition* value = getOperand(rhsId); + + auto* elements = MElements::New(alloc(), obj); + add(elements); + + auto* initLength = MInitializedLength::New(alloc(), elements); + add(initLength); + + auto* barrier = + MPostWriteElementBarrier::New(alloc(), obj, value, initLength); + add(barrier); + + auto* ins = MArrayPush::New(alloc(), obj, value); + addEffectful(ins); + pushResult(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitArrayJoinResult(ObjOperandId objId, + StringOperandId sepId) { + MDefinition* obj = getOperand(objId); + MDefinition* sep = getOperand(sepId); + + auto* join = MArrayJoin::New(alloc(), obj, sep); + addEffectful(join); + + pushResult(join); + return resumeAfter(join); +} + +bool WarpCacheIRTranspiler::emitPackedArrayPopResult(ObjOperandId arrayId) { + MDefinition* array = getOperand(arrayId); + + auto* ins = MArrayPopShift::New(alloc(), array, MArrayPopShift::Pop); + addEffectful(ins); + + pushResult(ins); + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitPackedArrayShiftResult(ObjOperandId arrayId) { + MDefinition* array = getOperand(arrayId); + + auto* ins = MArrayPopShift::New(alloc(), array, MArrayPopShift::Shift); + addEffectful(ins); + + pushResult(ins); + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitPackedArraySliceResult( + uint32_t templateObjectOffset, ObjOperandId arrayId, Int32OperandId beginId, + Int32OperandId endId) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + + MDefinition* array = getOperand(arrayId); + MDefinition* begin = getOperand(beginId); + MDefinition* end = getOperand(endId); + + // TODO: support pre-tenuring. + gc::InitialHeap heap = gc::DefaultHeap; + + auto* ins = MArraySlice::New(alloc(), array, begin, end, templateObj, heap); + addEffectful(ins); + + pushResult(ins); + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitHasClassResult(ObjOperandId objId, + uint32_t claspOffset) { + MDefinition* obj = getOperand(objId); + const JSClass* clasp = classStubField(claspOffset); + + auto* hasClass = MHasClass::New(alloc(), obj, clasp); + add(hasClass); + + pushResult(hasClass); + return true; +} + +bool WarpCacheIRTranspiler::emitCallRegExpMatcherResult( + ObjOperandId regexpId, StringOperandId inputId, + Int32OperandId lastIndexId) { + MDefinition* regexp = getOperand(regexpId); + MDefinition* input = getOperand(inputId); + MDefinition* lastIndex = getOperand(lastIndexId); + + auto* matcher = MRegExpMatcher::New(alloc(), regexp, input, lastIndex); + addEffectful(matcher); + pushResult(matcher); + + return resumeAfter(matcher); +} + +bool WarpCacheIRTranspiler::emitCallRegExpSearcherResult( + ObjOperandId regexpId, StringOperandId inputId, + Int32OperandId lastIndexId) { + MDefinition* regexp = getOperand(regexpId); + MDefinition* input = getOperand(inputId); + MDefinition* lastIndex = getOperand(lastIndexId); + + auto* searcher = MRegExpSearcher::New(alloc(), regexp, input, lastIndex); + addEffectful(searcher); + pushResult(searcher); + + return resumeAfter(searcher); +} + +bool WarpCacheIRTranspiler::emitCallRegExpTesterResult( + ObjOperandId regexpId, StringOperandId inputId, + Int32OperandId lastIndexId) { + MDefinition* regexp = getOperand(regexpId); + MDefinition* input = getOperand(inputId); + MDefinition* lastIndex = getOperand(lastIndexId); + + auto* tester = MRegExpTester::New(alloc(), regexp, input, lastIndex); + addEffectful(tester); + pushResult(tester); + + return resumeAfter(tester); +} + +bool WarpCacheIRTranspiler::emitCallSubstringKernelResult( + StringOperandId strId, Int32OperandId beginId, Int32OperandId lengthId) { + MDefinition* str = getOperand(strId); + MDefinition* begin = getOperand(beginId); + MDefinition* length = getOperand(lengthId); + + auto* substr = MSubstr::New(alloc(), str, begin, length); + add(substr); + + pushResult(substr); + return true; +} + +bool WarpCacheIRTranspiler::emitStringReplaceStringResult( + StringOperandId strId, StringOperandId patternId, + StringOperandId replacementId) { + MDefinition* str = getOperand(strId); + MDefinition* pattern = getOperand(patternId); + MDefinition* replacement = getOperand(replacementId); + + auto* replace = MStringReplace::New(alloc(), str, pattern, replacement); + add(replace); + + pushResult(replace); + return true; +} + +bool WarpCacheIRTranspiler::emitStringSplitStringResult( + StringOperandId strId, StringOperandId separatorId) { + MDefinition* str = getOperand(strId); + MDefinition* separator = getOperand(separatorId); + + auto* split = MStringSplit::New(alloc(), str, separator); + add(split); + + pushResult(split); + return true; +} + +bool WarpCacheIRTranspiler::emitRegExpPrototypeOptimizableResult( + ObjOperandId protoId) { + MDefinition* proto = getOperand(protoId); + + auto* optimizable = MRegExpPrototypeOptimizable::New(alloc(), proto); + add(optimizable); + + pushResult(optimizable); + return true; +} + +bool WarpCacheIRTranspiler::emitRegExpInstanceOptimizableResult( + ObjOperandId regexpId, ObjOperandId protoId) { + MDefinition* regexp = getOperand(regexpId); + MDefinition* proto = getOperand(protoId); + + auto* optimizable = MRegExpInstanceOptimizable::New(alloc(), regexp, proto); + add(optimizable); + + pushResult(optimizable); + return true; +} + +bool WarpCacheIRTranspiler::emitGetFirstDollarIndexResult( + StringOperandId strId) { + MDefinition* str = getOperand(strId); + + auto* firstDollarIndex = MGetFirstDollarIndex::New(alloc(), str); + add(firstDollarIndex); + + pushResult(firstDollarIndex); + return true; +} + +bool WarpCacheIRTranspiler::emitIsArrayResult(ValOperandId inputId) { + MDefinition* value = getOperand(inputId); + + auto* isArray = MIsArray::New(alloc(), value); + addEffectful(isArray); + pushResult(isArray); + + return resumeAfter(isArray); +} + +bool WarpCacheIRTranspiler::emitIsObjectResult(ValOperandId inputId) { + MDefinition* value = getOperand(inputId); + + if (value->type() == MIRType::Object) { + pushResult(constant(BooleanValue(true))); + } else { + auto* isObject = MIsObject::New(alloc(), value); + add(isObject); + pushResult(isObject); + } + + return true; +} + +bool WarpCacheIRTranspiler::emitIsPackedArrayResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* isPackedArray = MIsPackedArray::New(alloc(), obj); + add(isPackedArray); + + pushResult(isPackedArray); + return true; +} + +bool WarpCacheIRTranspiler::emitIsCallableResult(ValOperandId inputId) { + MDefinition* value = getOperand(inputId); + + auto* isCallable = MIsCallable::New(alloc(), value); + add(isCallable); + + pushResult(isCallable); + return true; +} + +bool WarpCacheIRTranspiler::emitIsConstructorResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* isConstructor = MIsConstructor::New(alloc(), obj); + add(isConstructor); + + pushResult(isConstructor); + return true; +} + +bool WarpCacheIRTranspiler::emitIsCrossRealmArrayConstructorResult( + ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MIsCrossRealmArrayConstructor::New(alloc(), obj); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitIsTypedArrayResult(ObjOperandId objId, + bool isPossiblyWrapped) { + MDefinition* obj = getOperand(objId); + + auto* ins = MIsTypedArray::New(alloc(), obj, isPossiblyWrapped); + if (isPossiblyWrapped) { + addEffectful(ins); + } else { + add(ins); + } + + pushResult(ins); + + if (isPossiblyWrapped) { + if (!resumeAfter(ins)) { + return false; + } + } + + return true; +} + +bool WarpCacheIRTranspiler::emitTypedArrayByteOffsetResult(ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MArrayBufferViewByteOffset::New(alloc(), obj); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitTypedArrayElementShiftResult( + ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MTypedArrayElementShift::New(alloc(), obj); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitIsTypedArrayConstructorResult( + ObjOperandId objId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MIsTypedArrayConstructor::New(alloc(), obj); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitGetNextMapSetEntryForIteratorResult( + ObjOperandId iterId, ObjOperandId resultArrId, bool isMap) { + MDefinition* iter = getOperand(iterId); + MDefinition* resultArr = getOperand(resultArrId); + + MGetNextEntryForIterator::Mode mode = + isMap ? MGetNextEntryForIterator::Map : MGetNextEntryForIterator::Set; + auto* ins = MGetNextEntryForIterator::New(alloc(), iter, resultArr, mode); + addEffectful(ins); + pushResult(ins); + + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitFrameIsConstructingResult() { + if (const CallInfo* callInfo = builder_->inlineCallInfo()) { + auto* ins = constant(BooleanValue(callInfo->constructing())); + pushResult(ins); + return true; + } + + auto* ins = MIsConstructing::New(alloc()); + add(ins); + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitFinishBoundFunctionInitResult( + ObjOperandId boundId, ObjOperandId targetId, Int32OperandId argCountId) { + MDefinition* bound = getOperand(boundId); + MDefinition* target = getOperand(targetId); + MDefinition* argCount = getOperand(argCountId); + + auto* ins = MFinishBoundFunctionInit::New(alloc(), bound, target, argCount); + addEffectful(ins); + + pushResult(constant(UndefinedValue())); + return resumeAfter(ins); +} + +bool WarpCacheIRTranspiler::emitNewIteratorResult( + MNewIterator::Type type, uint32_t templateObjectOffset) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + + auto* templateConst = constant(ObjectValue(*templateObj)); + auto* iter = MNewIterator::New(alloc(), templateConst, type); + add(iter); + + pushResult(iter); + return true; +} + +bool WarpCacheIRTranspiler::emitNewArrayIteratorResult( + uint32_t templateObjectOffset) { + return emitNewIteratorResult(MNewIterator::ArrayIterator, + templateObjectOffset); +} + +bool WarpCacheIRTranspiler::emitNewStringIteratorResult( + uint32_t templateObjectOffset) { + return emitNewIteratorResult(MNewIterator::StringIterator, + templateObjectOffset); +} + +bool WarpCacheIRTranspiler::emitNewRegExpStringIteratorResult( + uint32_t templateObjectOffset) { + return emitNewIteratorResult(MNewIterator::RegExpStringIterator, + templateObjectOffset); +} + +bool WarpCacheIRTranspiler::emitObjectCreateResult( + uint32_t templateObjectOffset) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + + auto* templateConst = constant(ObjectValue(*templateObj)); + + // TODO: support pre-tenuring. + gc::InitialHeap heap = gc::DefaultHeap; + auto* obj = + MNewObject::New(alloc(), templateConst, heap, MNewObject::ObjectCreate); + addEffectful(obj); + + pushResult(obj); + return resumeAfter(obj); +} + +bool WarpCacheIRTranspiler::emitNewArrayFromLengthResult( + uint32_t templateObjectOffset, Int32OperandId lengthId) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + MDefinition* length = getOperand(lengthId); + + // TODO: support pre-tenuring. + gc::InitialHeap heap = gc::DefaultHeap; + + if (length->isConstant()) { + int32_t lenInt32 = length->toConstant()->toInt32(); + if (lenInt32 >= 0 && + uint32_t(lenInt32) == templateObj->as<ArrayObject>().length()) { + uint32_t len = uint32_t(lenInt32); + auto* templateConst = constant(ObjectValue(*templateObj)); + + size_t inlineLength = + gc::GetGCKindSlots(templateObj->asTenured().getAllocKind()) - + ObjectElements::VALUES_PER_HEADER; + + MNewArray* obj; + if (len > inlineLength) { + obj = MNewArray::NewVM(alloc(), len, templateConst, heap); + } else { + obj = MNewArray::New(alloc(), len, templateConst, heap); + } + add(obj); + pushResult(obj); + return true; + } + } + + auto* obj = MNewArrayDynamicLength::New(alloc(), templateObj, heap, length); + addEffectful(obj); + pushResult(obj); + return resumeAfter(obj); +} + +bool WarpCacheIRTranspiler::emitNewTypedArrayFromLengthResult( + uint32_t templateObjectOffset, Int32OperandId lengthId) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + MDefinition* length = getOperand(lengthId); + + // TODO: support pre-tenuring. + gc::InitialHeap heap = gc::DefaultHeap; + + if (length->isConstant()) { + int32_t len = length->toConstant()->toInt32(); + if (len > 0 && + uint32_t(len) == templateObj->as<TypedArrayObject>().length().get()) { + auto* templateConst = constant(ObjectValue(*templateObj)); + auto* obj = MNewTypedArray::New(alloc(), templateConst, heap); + add(obj); + pushResult(obj); + return true; + } + } + + auto* obj = + MNewTypedArrayDynamicLength::New(alloc(), templateObj, heap, length); + addEffectful(obj); + pushResult(obj); + return resumeAfter(obj); +} + +bool WarpCacheIRTranspiler::emitNewTypedArrayFromArrayBufferResult( + uint32_t templateObjectOffset, ObjOperandId bufferId, + ValOperandId byteOffsetId, ValOperandId lengthId) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + MDefinition* buffer = getOperand(bufferId); + MDefinition* byteOffset = getOperand(byteOffsetId); + MDefinition* length = getOperand(lengthId); + + // TODO: support pre-tenuring. + gc::InitialHeap heap = gc::DefaultHeap; + + auto* obj = MNewTypedArrayFromArrayBuffer::New(alloc(), templateObj, heap, + buffer, byteOffset, length); + addEffectful(obj); + + pushResult(obj); + return resumeAfter(obj); +} + +bool WarpCacheIRTranspiler::emitNewTypedArrayFromArrayResult( + uint32_t templateObjectOffset, ObjOperandId arrayId) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + MDefinition* array = getOperand(arrayId); + + // TODO: support pre-tenuring. + gc::InitialHeap heap = gc::DefaultHeap; + + auto* obj = MNewTypedArrayFromArray::New(alloc(), templateObj, heap, array); + addEffectful(obj); + + pushResult(obj); + return resumeAfter(obj); +} + +bool WarpCacheIRTranspiler::emitAtomicsCompareExchangeResult( + ObjOperandId objId, Int32OperandId indexId, Int32OperandId expectedId, + Int32OperandId replacementId, Scalar::Type elementType) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + MDefinition* expected = getOperand(expectedId); + MDefinition* replacement = getOperand(replacementId); + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + index = addBoundsCheck(index, length); + + auto* elements = MArrayBufferViewElements::New(alloc(), obj); + add(elements); + + bool allowDoubleForUint32 = true; + MIRType knownType = + MIRTypeForArrayBufferViewRead(elementType, allowDoubleForUint32); + + auto* cas = MCompareExchangeTypedArrayElement::New( + alloc(), elements, index, elementType, expected, replacement); + cas->setResultType(knownType); + addEffectful(cas); + + pushResult(cas); + return resumeAfter(cas); +} + +bool WarpCacheIRTranspiler::emitAtomicsExchangeResult( + ObjOperandId objId, Int32OperandId indexId, Int32OperandId valueId, + Scalar::Type elementType) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + MDefinition* value = getOperand(valueId); + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + index = addBoundsCheck(index, length); + + auto* elements = MArrayBufferViewElements::New(alloc(), obj); + add(elements); + + bool allowDoubleForUint32 = true; + MIRType knownType = + MIRTypeForArrayBufferViewRead(elementType, allowDoubleForUint32); + + auto* exchange = MAtomicExchangeTypedArrayElement::New( + alloc(), elements, index, value, elementType); + exchange->setResultType(knownType); + addEffectful(exchange); + + pushResult(exchange); + return resumeAfter(exchange); +} + +bool WarpCacheIRTranspiler::emitAtomicsBinaryOp(ObjOperandId objId, + Int32OperandId indexId, + Int32OperandId valueId, + Scalar::Type elementType, + AtomicOp op) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + MDefinition* value = getOperand(valueId); + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + index = addBoundsCheck(index, length); + + auto* elements = MArrayBufferViewElements::New(alloc(), obj); + add(elements); + + bool allowDoubleForUint32 = true; + MIRType knownType = + MIRTypeForArrayBufferViewRead(elementType, allowDoubleForUint32); + + auto* binop = MAtomicTypedArrayElementBinop::New(alloc(), op, elements, index, + elementType, value); + binop->setResultType(knownType); + addEffectful(binop); + + pushResult(binop); + return resumeAfter(binop); +} + +bool WarpCacheIRTranspiler::emitAtomicsAddResult(ObjOperandId objId, + Int32OperandId indexId, + Int32OperandId valueId, + Scalar::Type elementType) { + return emitAtomicsBinaryOp(objId, indexId, valueId, elementType, + AtomicFetchAddOp); +} + +bool WarpCacheIRTranspiler::emitAtomicsSubResult(ObjOperandId objId, + Int32OperandId indexId, + Int32OperandId valueId, + Scalar::Type elementType) { + return emitAtomicsBinaryOp(objId, indexId, valueId, elementType, + AtomicFetchSubOp); +} + +bool WarpCacheIRTranspiler::emitAtomicsAndResult(ObjOperandId objId, + Int32OperandId indexId, + Int32OperandId valueId, + Scalar::Type elementType) { + return emitAtomicsBinaryOp(objId, indexId, valueId, elementType, + AtomicFetchAndOp); +} + +bool WarpCacheIRTranspiler::emitAtomicsOrResult(ObjOperandId objId, + Int32OperandId indexId, + Int32OperandId valueId, + Scalar::Type elementType) { + return emitAtomicsBinaryOp(objId, indexId, valueId, elementType, + AtomicFetchOrOp); +} + +bool WarpCacheIRTranspiler::emitAtomicsXorResult(ObjOperandId objId, + Int32OperandId indexId, + Int32OperandId valueId, + Scalar::Type elementType) { + return emitAtomicsBinaryOp(objId, indexId, valueId, elementType, + AtomicFetchXorOp); +} + +bool WarpCacheIRTranspiler::emitAtomicsLoadResult(ObjOperandId objId, + Int32OperandId indexId, + Scalar::Type elementType) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + index = addBoundsCheck(index, length); + + auto* elements = MArrayBufferViewElements::New(alloc(), obj); + add(elements); + + bool allowDoubleForUint32 = true; + MIRType knownType = + MIRTypeForArrayBufferViewRead(elementType, allowDoubleForUint32); + + auto* load = MLoadUnboxedScalar::New(alloc(), elements, index, elementType, + DoesRequireMemoryBarrier); + load->setResultType(knownType); + addEffectful(load); + + pushResult(load); + return resumeAfter(load); +} + +bool WarpCacheIRTranspiler::emitAtomicsStoreResult(ObjOperandId objId, + Int32OperandId indexId, + Int32OperandId valueId, + Scalar::Type elementType) { + MDefinition* obj = getOperand(objId); + MDefinition* index = getOperand(indexId); + MDefinition* value = getOperand(valueId); + + auto* length = MArrayBufferViewLength::New(alloc(), obj); + add(length); + + index = addBoundsCheck(index, length); + + auto* elements = MArrayBufferViewElements::New(alloc(), obj); + add(elements); + + auto* store = MStoreUnboxedScalar::New(alloc(), elements, index, value, + elementType, DoesRequireMemoryBarrier); + addEffectful(store); + + pushResult(value); + return resumeAfter(store); +} + +bool WarpCacheIRTranspiler::emitAtomicsIsLockFreeResult( + Int32OperandId valueId) { + MDefinition* value = getOperand(valueId); + + auto* ilf = MAtomicIsLockFree::New(alloc(), value); + add(ilf); + + pushResult(ilf); + return true; +} + +bool WarpCacheIRTranspiler::emitBigIntAsIntNResult(Int32OperandId bitsId, + BigIntOperandId bigIntId) { + MDefinition* bits = getOperand(bitsId); + MDefinition* bigInt = getOperand(bigIntId); + + auto* ins = MBigIntAsIntN::New(alloc(), bits, bigInt); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitBigIntAsUintNResult(Int32OperandId bitsId, + BigIntOperandId bigIntId) { + MDefinition* bits = getOperand(bitsId); + MDefinition* bigInt = getOperand(bigIntId); + + auto* ins = MBigIntAsUintN::New(alloc(), bits, bigInt); + add(ins); + + pushResult(ins); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadValueTruthyResult(ValOperandId inputId) { + MDefinition* input = getOperand(inputId); + + // Convert to bool with the '!!' idiom. + auto* resultInverted = MNot::New(alloc(), input); + add(resultInverted); + auto* result = MNot::New(alloc(), resultInverted); + add(result); + + pushResult(result); + return true; +} + +bool WarpCacheIRTranspiler::emitLoadWrapperTarget(ObjOperandId objId, + ObjOperandId resultId) { + MDefinition* obj = getOperand(objId); + + auto* ins = MLoadWrapperTarget::New(alloc(), obj); + add(ins); + + return defineOperand(resultId, ins); +} + +// When we transpile a call, we may generate guards for some +// arguments. To make sure the call instruction depends on those +// guards, when the transpiler creates an operand for an argument, we +// register the OperandId of that argument in argumentIds_. (See +// emitLoadArgumentSlot.) Before generating the call, we update the +// CallInfo to use the appropriate value from operands_. +// Note: The callee is an explicit argument to the call op, and is +// tracked separately. +void WarpCacheIRTranspiler::updateArgumentsFromOperands() { + for (uint32_t i = 0; i < uint32_t(ArgumentKind::NumKinds); i++) { + ArgumentKind kind = ArgumentKind(i); + OperandId id = argumentOperandIds_[kind]; + if (id.valid()) { + switch (kind) { + case ArgumentKind::This: + callInfo_->setThis(getOperand(id)); + break; + case ArgumentKind::NewTarget: + callInfo_->setNewTarget(getOperand(id)); + break; + case ArgumentKind::Arg0: + callInfo_->setArg(0, getOperand(id)); + break; + case ArgumentKind::Arg1: + callInfo_->setArg(1, getOperand(id)); + break; + case ArgumentKind::Arg2: + callInfo_->setArg(2, getOperand(id)); + break; + case ArgumentKind::Arg3: + callInfo_->setArg(3, getOperand(id)); + break; + case ArgumentKind::Arg4: + callInfo_->setArg(4, getOperand(id)); + break; + case ArgumentKind::Arg5: + callInfo_->setArg(5, getOperand(id)); + break; + case ArgumentKind::Arg6: + callInfo_->setArg(6, getOperand(id)); + break; + case ArgumentKind::Arg7: + callInfo_->setArg(7, getOperand(id)); + break; + case ArgumentKind::Callee: + case ArgumentKind::NumKinds: + MOZ_CRASH("Unexpected argument kind"); + } + } + } +} + +bool WarpCacheIRTranspiler::emitLoadArgumentSlot(ValOperandId resultId, + uint32_t slotIndex) { + // Reverse of GetIndexOfArgument. + + // Layout: + // NewTarget | Args.. (reversed) | ThisValue | Callee + // 0 | ArgC .. Arg1 Arg0 (+1) | argc (+1) | argc + 1 (+ 1) + // ^ (if constructing) + + // NewTarget (optional) + if (callInfo_->constructing()) { + if (slotIndex == 0) { + setArgumentId(ArgumentKind::NewTarget, resultId); + return defineOperand(resultId, callInfo_->getNewTarget()); + } + + slotIndex -= 1; // Adjust slot index to match non-constructing calls. + } + + // Args.. + if (slotIndex < callInfo_->argc()) { + uint32_t arg = callInfo_->argc() - 1 - slotIndex; + ArgumentKind kind = ArgumentKindForArgIndex(arg); + MOZ_ASSERT(kind < ArgumentKind::NumKinds); + setArgumentId(kind, resultId); + return defineOperand(resultId, callInfo_->getArg(arg)); + } + + // ThisValue + if (slotIndex == callInfo_->argc()) { + setArgumentId(ArgumentKind::This, resultId); + return defineOperand(resultId, callInfo_->thisArg()); + } + + // Callee + MOZ_ASSERT(slotIndex == callInfo_->argc() + 1); + return defineOperand(resultId, callInfo_->callee()); +} + +bool WarpCacheIRTranspiler::emitLoadArgumentFixedSlot(ValOperandId resultId, + uint8_t slotIndex) { + return emitLoadArgumentSlot(resultId, slotIndex); +} + +bool WarpCacheIRTranspiler::emitLoadArgumentDynamicSlot(ValOperandId resultId, + Int32OperandId argcId, + uint8_t slotIndex) { +#ifdef DEBUG + MDefinition* argc = getOperand(argcId); + MOZ_ASSERT(argc->toConstant()->toInt32() == + static_cast<int32_t>(callInfo_->argc())); +#endif + + return emitLoadArgumentSlot(resultId, callInfo_->argc() + slotIndex); +} + +WrappedFunction* WarpCacheIRTranspiler::maybeWrappedFunction( + MDefinition* callee, CallKind kind, uint16_t nargs, FunctionFlags flags) { + MOZ_ASSERT(callee->isConstant() || callee->isNurseryObject()); + + // If this is a native without a JitEntry, WrappedFunction needs to know the + // target JSFunction. + // TODO: support nursery-allocated natives with WrappedFunction, maybe by + // storing the JSNative in the Baseline stub like flags/nargs. + bool isNative = flags.isNativeWithoutJitEntry(); + if (isNative && !callee->isConstant()) { + return nullptr; + } + + JSFunction* nativeTarget = nullptr; + if (isNative) { + nativeTarget = &callee->toConstant()->toObject().as<JSFunction>(); + } + + WrappedFunction* wrappedTarget = + new (alloc()) WrappedFunction(nativeTarget, nargs, flags); + MOZ_ASSERT_IF(kind == CallKind::Native || kind == CallKind::DOM, + wrappedTarget->isNativeWithoutJitEntry()); + MOZ_ASSERT_IF(kind == CallKind::Scripted, wrappedTarget->hasJitEntry()); + return wrappedTarget; +} + +WrappedFunction* WarpCacheIRTranspiler::maybeCallTarget(MDefinition* callee, + CallKind kind) { + // CacheIR emits the following for specialized calls: + // GuardSpecificFunction <callee> <func> .. + // Call(Native|Scripted)Function <callee> .. + // or: + // GuardClass <callee> .. + // GuardFunctionScript <callee> <script> .. + // CallScriptedFunction <callee> .. + // + // We can use the <func> JSFunction or <script> BaseScript to specialize this + // call. + if (callee->isGuardSpecificFunction()) { + auto* guard = callee->toGuardSpecificFunction(); + return maybeWrappedFunction(guard->expected(), kind, guard->nargs(), + guard->flags()); + } + if (callee->isGuardFunctionScript()) { + MOZ_ASSERT(kind == CallKind::Scripted); + auto* guard = callee->toGuardFunctionScript(); + WrappedFunction* wrappedTarget = new (alloc()) WrappedFunction( + /* nativeFun = */ nullptr, guard->nargs(), guard->flags()); + MOZ_ASSERT(wrappedTarget->hasJitEntry()); + return wrappedTarget; + } + return nullptr; +} + +// If it is possible to use MCall for this call, update callInfo_ to use +// the correct arguments. Otherwise, update the ArgFormat of callInfo_. +bool WarpCacheIRTranspiler::updateCallInfo(MDefinition* callee, + CallFlags flags) { + // The transpilation will add various guards to the callee. + // We replace the callee referenced by the CallInfo, so that + // the resulting call instruction depends on these guards. + callInfo_->setCallee(callee); + + // The transpilation may also add guards to other arguments. + // We replace those arguments in the CallInfo here. + updateArgumentsFromOperands(); + + switch (flags.getArgFormat()) { + case CallFlags::Standard: + MOZ_ASSERT(callInfo_->argFormat() == CallInfo::ArgFormat::Standard); + break; + case CallFlags::Spread: + MOZ_ASSERT(callInfo_->argFormat() == CallInfo::ArgFormat::Array); + break; + case CallFlags::FunCall: + // Note: We already changed the callee to the target + // function instead of the |call| function. + MOZ_ASSERT(!callInfo_->constructing()); + MOZ_ASSERT(callInfo_->argFormat() == CallInfo::ArgFormat::Standard); + + if (callInfo_->argc() == 0) { + // Special case for fun.call() with no arguments. + auto* undef = constant(UndefinedValue()); + callInfo_->setThis(undef); + } else { + // The first argument for |call| is the new this value. + callInfo_->setThis(callInfo_->getArg(0)); + + // Shift down all other arguments by removing the first. + callInfo_->removeArg(0); + } + break; + case CallFlags::FunApplyArgs: + MOZ_ASSERT(!callInfo_->constructing()); + MOZ_ASSERT(callInfo_->argFormat() == CallInfo::ArgFormat::Standard); + + // If we are building an inlined function, we know the arguments + // being used. + if (const CallInfo* outerCallInfo = builder_->inlineCallInfo()) { + MDefinition* argFunc = callInfo_->thisArg(); + MDefinition* argThis = callInfo_->getArg(0); + + if (!callInfo_->replaceArgs(outerCallInfo->argv())) { + return false; + } + callInfo_->setCallee(argFunc); + callInfo_->setThis(argThis); + } else { + callInfo_->setArgFormat(CallInfo::ArgFormat::FunApplyArgs); + } + break; + case CallFlags::FunApplyArray: { + MDefinition* argFunc = callInfo_->thisArg(); + MDefinition* argThis = callInfo_->getArg(0); + callInfo_->setCallee(argFunc); + callInfo_->setThis(argThis); + callInfo_->setArgFormat(CallInfo::ArgFormat::Array); + break; + } + default: + MOZ_CRASH("Unsupported arg format"); + } + return true; +} + +// Returns true if we are generating a call to CreateThisFromIon and +// must check its return value. +bool WarpCacheIRTranspiler::maybeCreateThis(MDefinition* callee, + CallFlags flags, CallKind kind) { + MOZ_ASSERT(kind != CallKind::DOM, "DOM functions are not constructors"); + MDefinition* thisArg = callInfo_->thisArg(); + + if (kind == CallKind::Native) { + // Native functions keep the is-constructing MagicValue as |this|. + // If one of the arguments uses spread syntax this can be a loop phi with + // MIRType::Value. + MOZ_ASSERT(thisArg->type() == MIRType::MagicIsConstructing || + thisArg->isPhi()); + return false; + } + MOZ_ASSERT(kind == CallKind::Scripted); + + if (thisArg->isCreateThisWithTemplate()) { + // We have already updated |this| based on MetaTwoByte. We do + // not need to generate a check. + return false; + } + if (flags.needsUninitializedThis()) { + MConstant* uninit = constant(MagicValue(JS_UNINITIALIZED_LEXICAL)); + thisArg->setImplicitlyUsedUnchecked(); + callInfo_->setThis(uninit); + return false; + } + // See the Native case above. + MOZ_ASSERT(thisArg->type() == MIRType::MagicIsConstructing || + thisArg->isPhi()); + + MDefinition* newTarget = callInfo_->getNewTarget(); + auto* createThis = MCreateThis::New(alloc(), callee, newTarget); + add(createThis); + + thisArg->setImplicitlyUsedUnchecked(); + callInfo_->setThis(createThis); + + return true; +} + +bool WarpCacheIRTranspiler::emitCallFunction( + ObjOperandId calleeId, Int32OperandId argcId, + mozilla::Maybe<ObjOperandId> thisObjId, CallFlags flags, CallKind kind) { + MDefinition* callee = getOperand(calleeId); +#ifdef DEBUG + MDefinition* argc = getOperand(argcId); + MOZ_ASSERT(argc->toConstant()->toInt32() == + static_cast<int32_t>(callInfo_->argc())); +#endif + + if (!updateCallInfo(callee, flags)) { + return false; + } + + if (kind == CallKind::DOM) { + MOZ_ASSERT(flags.getArgFormat() == CallFlags::Standard); + // For DOM calls |this| has a class guard. + MDefinition* thisObj = getOperand(*thisObjId); + callInfo_->setThis(thisObj); + } + + WrappedFunction* wrappedTarget = maybeCallTarget(callee, kind); + + bool needsThisCheck = false; + if (callInfo_->constructing()) { + MOZ_ASSERT(flags.isConstructing()); + needsThisCheck = maybeCreateThis(callee, flags, kind); + if (needsThisCheck) { + wrappedTarget = nullptr; + } + } + + switch (callInfo_->argFormat()) { + case CallInfo::ArgFormat::Standard: { + MCall* call = makeCall(*callInfo_, needsThisCheck, wrappedTarget, + kind == CallKind::DOM); + if (!call) { + return false; + } + + if (flags.isSameRealm()) { + call->setNotCrossRealm(); + } + + if (call->isEffectful()) { + addEffectful(call); + pushResult(call); + return resumeAfter(call); + } + + MOZ_ASSERT(kind == CallKind::DOM); + add(call); + pushResult(call); + return true; + } + case CallInfo::ArgFormat::Array: { + MInstruction* call = + makeSpreadCall(*callInfo_, flags.isSameRealm(), wrappedTarget); + if (!call) { + return false; + } + addEffectful(call); + pushResult(call); + + return resumeAfter(call); + } + case CallInfo::ArgFormat::FunApplyArgs: { + return emitFunApplyArgs(wrappedTarget, flags); + } + } + MOZ_CRASH("unreachable"); +} + +bool WarpCacheIRTranspiler::emitFunApplyArgs(WrappedFunction* wrappedTarget, + CallFlags flags) { + MOZ_ASSERT(!callInfo_->constructing()); + MOZ_ASSERT(!builder_->inlineCallInfo()); + + MDefinition* argFunc = callInfo_->thisArg(); + MDefinition* argThis = callInfo_->getArg(0); + + MArgumentsLength* numArgs = MArgumentsLength::New(alloc()); + add(numArgs); + + MApplyArgs* apply = + MApplyArgs::New(alloc(), wrappedTarget, argFunc, numArgs, argThis); + + if (flags.isSameRealm()) { + apply->setNotCrossRealm(); + } + if (callInfo_->ignoresReturnValue()) { + apply->setIgnoresReturnValue(); + } + + addEffectful(apply); + pushResult(apply); + + return resumeAfter(apply); +} + +#ifndef JS_SIMULATOR +bool WarpCacheIRTranspiler::emitCallNativeFunction(ObjOperandId calleeId, + Int32OperandId argcId, + CallFlags flags, + bool ignoresReturnValue) { + // Instead of ignoresReturnValue we use CallInfo::ignoresReturnValue. + return emitCallFunction(calleeId, argcId, mozilla::Nothing(), flags, + CallKind::Native); +} + +bool WarpCacheIRTranspiler::emitCallDOMFunction(ObjOperandId calleeId, + Int32OperandId argcId, + ObjOperandId thisObjId, + CallFlags flags) { + return emitCallFunction(calleeId, argcId, mozilla::Some(thisObjId), flags, + CallKind::DOM); +} +#else +bool WarpCacheIRTranspiler::emitCallNativeFunction(ObjOperandId calleeId, + Int32OperandId argcId, + CallFlags flags, + uint32_t targetOffset) { + return emitCallFunction(calleeId, argcId, mozilla::Nothing(), flags, + CallKind::Native); +} + +bool WarpCacheIRTranspiler::emitCallDOMFunction(ObjOperandId calleeId, + Int32OperandId argcId, + ObjOperandId thisObjId, + CallFlags flags, + uint32_t targetOffset) { + return emitCallFunction(calleeId, argcId, mozilla::Some(thisObjId), flags, + CallKind::DOM); +} +#endif + +bool WarpCacheIRTranspiler::emitCallScriptedFunction(ObjOperandId calleeId, + Int32OperandId argcId, + CallFlags flags) { + return emitCallFunction(calleeId, argcId, mozilla::Nothing(), flags, + CallKind::Scripted); +} + +bool WarpCacheIRTranspiler::emitCallInlinedFunction(ObjOperandId calleeId, + Int32OperandId argcId, + uint32_t icScriptOffset, + CallFlags flags) { + if (callInfo_->isInlined()) { + // We are transpiling to generate the correct guards. We also + // update the CallInfo to use the correct arguments. Code for the + // inlined function itself will be generated in + // WarpBuilder::buildInlinedCall. + MDefinition* callee = getOperand(calleeId); + if (!updateCallInfo(callee, flags)) { + return false; + } + if (callInfo_->constructing()) { + MOZ_ASSERT(flags.isConstructing()); + + // We call maybeCreateThis to update |this|, but inlined constructors + // never need a VM call. CallIRGenerator::getThisForScripted ensures that + // we don't attach a specialized stub unless we have a template object or + // know that the constructor needs uninitialized this. + MOZ_ALWAYS_FALSE(maybeCreateThis(callee, flags, CallKind::Scripted)); + mozilla::DebugOnly<MDefinition*> thisArg = callInfo_->thisArg(); + MOZ_ASSERT(thisArg->isCreateThisWithTemplate() || + thisArg->type() == MIRType::MagicUninitializedLexical); + } + + switch (callInfo_->argFormat()) { + case CallInfo::ArgFormat::Standard: + break; + default: + MOZ_CRASH("Unsupported arg format"); + } + return true; + } + return emitCallFunction(calleeId, argcId, mozilla::Nothing(), flags, + CallKind::Scripted); +} + +bool WarpCacheIRTranspiler::emitCallWasmFunction(ObjOperandId calleeId, + Int32OperandId argcId, + CallFlags flags, + uint32_t funcExportOffset, + uint32_t instanceOffset) { + MDefinition* callee = getOperand(calleeId); +#ifdef DEBUG + MDefinition* argc = getOperand(argcId); + MOZ_ASSERT(argc->toConstant()->toInt32() == + static_cast<int32_t>(callInfo_->argc())); +#endif + JSObject* instanceObject = tenuredObjectStubField(instanceOffset); + const wasm::FuncExport* funcExport = wasmFuncExportField(funcExportOffset); + const wasm::FuncType& sig = funcExport->funcType(); + + if (!updateCallInfo(callee, flags)) { + return false; + } + + static_assert(wasm::MaxArgsForJitInlineCall <= MaxNumLInstructionOperands, + "arguments must fit in LIR operands"); + MOZ_ASSERT(sig.args().length() <= wasm::MaxArgsForJitInlineCall); + + MOZ_ASSERT(callInfo_->argFormat() == CallInfo::ArgFormat::Standard); + + auto* wasmInstanceObj = &instanceObject->as<WasmInstanceObject>(); + auto* call = MIonToWasmCall::New(alloc(), wasmInstanceObj, *funcExport); + if (!call) { + return false; + } + + mozilla::Maybe<MDefinition*> undefined; + for (size_t i = 0; i < sig.args().length(); i++) { + if (!alloc().ensureBallast()) { + return false; + } + + MDefinition* arg; + if (i < callInfo_->argc()) { + arg = callInfo_->getArg(i); + } else { + if (!undefined) { + undefined.emplace(constant(UndefinedValue())); + } + arg = convertWasmArg(*undefined, sig.args()[i].kind()); + } + call->initArg(i, arg); + } + + addEffectful(call); + + // Add any post-function call conversions that are necessary. + MInstruction* postConversion = call; + const wasm::ValTypeVector& results = sig.results(); + MOZ_ASSERT(results.length() <= 1, "Multi-value returns not supported."); + if (results.length() == 0) { + // No results to convert. + } else { + switch (results[0].kind()) { + case wasm::ValType::I64: + // JS expects a BigInt from I64 types. + postConversion = MInt64ToBigInt::New(alloc(), call); + + // Make non-movable so we can attach a resume point. + postConversion->setNotMovable(); + + add(postConversion); + break; + default: + // No spectre.index_masking of i32 results required, as the generated + // stub takes care of that. + break; + } + } + + // The resume point has to be attached to the post-conversion instruction + // (if present) instead of to the call. This way, if the call triggers an + // invalidation bailout, we will have the BigInt value on the Baseline stack. + // Potential alternative solution: attach the resume point to the call and + // have bailouts turn the Int64 value into a BigInt, maybe with a recover + // instruction. + pushResult(postConversion); + return resumeAfterUnchecked(postConversion); +} + +MDefinition* WarpCacheIRTranspiler::convertWasmArg(MDefinition* arg, + wasm::ValType::Kind kind) { + // An invariant in this code is that any type conversion operation that has + // externally visible effects, such as invoking valueOf on an object argument, + // must bailout so that we don't have to worry about replaying effects during + // argument conversion. + MInstruction* conversion = nullptr; + switch (kind) { + case wasm::ValType::I32: + conversion = MTruncateToInt32::New(alloc(), arg); + break; + case wasm::ValType::I64: + conversion = MToInt64::New(alloc(), arg); + break; + case wasm::ValType::F32: + conversion = MToFloat32::New(alloc(), arg); + break; + case wasm::ValType::F64: + conversion = MToDouble::New(alloc(), arg); + break; + case wasm::ValType::V128: + MOZ_CRASH("Unexpected type for Wasm JitEntry"); + case wasm::ValType::Ref: + // Transform the JS representation into an AnyRef representation. + // The resulting type is MIRType::RefOrNull. These cases are all + // effect-free. + switch (arg->type()) { + case MIRType::Object: + conversion = MWasmAnyRefFromJSObject::New(alloc(), arg); + break; + case MIRType::Null: + arg->setImplicitlyUsedUnchecked(); + conversion = MWasmNullConstant::New(alloc()); + break; + default: + conversion = MWasmBoxValue::New(alloc(), arg); + break; + } + break; + } + + add(conversion); + return conversion; +} + +bool WarpCacheIRTranspiler::emitGuardWasmArg(ValOperandId argId, + wasm::ValType::Kind kind) { + MDefinition* arg = getOperand(argId); + MDefinition* conversion = convertWasmArg(arg, kind); + + setOperand(argId, conversion); + return true; +} + +bool WarpCacheIRTranspiler::emitCallGetterResult(CallKind kind, + ValOperandId receiverId, + uint32_t getterOffset, + bool sameRealm, + uint32_t nargsAndFlagsOffset) { + MDefinition* receiver = getOperand(receiverId); + MDefinition* getter = objectStubField(getterOffset); + uint32_t nargsAndFlags = uint32StubField(nargsAndFlagsOffset); + + uint16_t nargs = nargsAndFlags >> 16; + FunctionFlags flags = FunctionFlags(uint16_t(nargsAndFlags)); + WrappedFunction* wrappedTarget = + maybeWrappedFunction(getter, kind, nargs, flags); + + jsbytecode* pc = loc_.toRawBytecode(); + bool ignoresRval = BytecodeIsPopped(pc); + CallInfo callInfo(alloc(), pc, /* constructing = */ false, ignoresRval); + callInfo.initForGetterCall(getter, receiver); + + MCall* call = makeCall(callInfo, /* needsThisCheck = */ false, wrappedTarget); + if (!call) { + return false; + } + + if (sameRealm) { + call->setNotCrossRealm(); + } + + addEffectful(call); + pushResult(call); + + return resumeAfter(call); +} + +bool WarpCacheIRTranspiler::emitCallScriptedGetterResult( + ValOperandId receiverId, uint32_t getterOffset, bool sameRealm, + uint32_t nargsAndFlagsOffset) { + return emitCallGetterResult(CallKind::Scripted, receiverId, getterOffset, + sameRealm, nargsAndFlagsOffset); +} + +bool WarpCacheIRTranspiler::emitCallInlinedGetterResult( + ValOperandId receiverId, uint32_t getterOffset, uint32_t icScriptOffset, + bool sameRealm, uint32_t nargsAndFlagsOffset) { + if (callInfo_) { + MOZ_ASSERT(callInfo_->isInlined()); + // We are transpiling to generate the correct guards. We also update the + // CallInfo to use the correct arguments. Code for the inlined getter + // itself will be generated in WarpBuilder::buildInlinedCall. + MDefinition* receiver = getOperand(receiverId); + MDefinition* getter = objectStubField(getterOffset); + callInfo_->initForGetterCall(getter, receiver); + + // Make sure there's enough room to push the arguments on the stack. + if (!current->ensureHasSlots(2)) { + return false; + } + + return true; + } + + return emitCallGetterResult(CallKind::Scripted, receiverId, getterOffset, + sameRealm, nargsAndFlagsOffset); +} + +bool WarpCacheIRTranspiler::emitCallNativeGetterResult( + ValOperandId receiverId, uint32_t getterOffset, bool sameRealm, + uint32_t nargsAndFlagsOffset) { + return emitCallGetterResult(CallKind::Native, receiverId, getterOffset, + sameRealm, nargsAndFlagsOffset); +} + +bool WarpCacheIRTranspiler::emitCallSetter(CallKind kind, + ObjOperandId receiverId, + uint32_t setterOffset, + ValOperandId rhsId, bool sameRealm, + uint32_t nargsAndFlagsOffset) { + MDefinition* receiver = getOperand(receiverId); + MDefinition* setter = objectStubField(setterOffset); + MDefinition* rhs = getOperand(rhsId); + uint32_t nargsAndFlags = uint32StubField(nargsAndFlagsOffset); + + uint16_t nargs = nargsAndFlags >> 16; + FunctionFlags flags = FunctionFlags(uint16_t(nargsAndFlags)); + WrappedFunction* wrappedTarget = + maybeWrappedFunction(setter, kind, nargs, flags); + + jsbytecode* pc = loc_.toRawBytecode(); + CallInfo callInfo(alloc(), pc, /* constructing = */ false, + /* ignoresReturnValue = */ true); + callInfo.initForSetterCall(setter, receiver, rhs); + + MCall* call = makeCall(callInfo, /* needsThisCheck = */ false, wrappedTarget); + if (!call) { + return false; + } + + if (sameRealm) { + call->setNotCrossRealm(); + } + + addEffectful(call); + return resumeAfter(call); +} + +bool WarpCacheIRTranspiler::emitCallScriptedSetter( + ObjOperandId receiverId, uint32_t setterOffset, ValOperandId rhsId, + bool sameRealm, uint32_t nargsAndFlagsOffset) { + return emitCallSetter(CallKind::Scripted, receiverId, setterOffset, rhsId, + sameRealm, nargsAndFlagsOffset); +} + +bool WarpCacheIRTranspiler::emitCallInlinedSetter( + ObjOperandId receiverId, uint32_t setterOffset, ValOperandId rhsId, + uint32_t icScriptOffset, bool sameRealm, uint32_t nargsAndFlagsOffset) { + if (callInfo_) { + MOZ_ASSERT(callInfo_->isInlined()); + // We are transpiling to generate the correct guards. We also update the + // CallInfo to use the correct arguments. Code for the inlined setter + // itself will be generated in WarpBuilder::buildInlinedCall. + MDefinition* receiver = getOperand(receiverId); + MDefinition* setter = objectStubField(setterOffset); + MDefinition* rhs = getOperand(rhsId); + callInfo_->initForSetterCall(setter, receiver, rhs); + + // Make sure there's enough room to push the arguments on the stack. + if (!current->ensureHasSlots(3)) { + return false; + } + + return true; + } + + return emitCallSetter(CallKind::Scripted, receiverId, setterOffset, rhsId, + sameRealm, nargsAndFlagsOffset); +} + +bool WarpCacheIRTranspiler::emitCallNativeSetter(ObjOperandId receiverId, + uint32_t setterOffset, + ValOperandId rhsId, + bool sameRealm, + uint32_t nargsAndFlagsOffset) { + return emitCallSetter(CallKind::Native, receiverId, setterOffset, rhsId, + sameRealm, nargsAndFlagsOffset); +} + +// TODO(post-Warp): rename the MetaTwoByte op when IonBuilder is gone. +bool WarpCacheIRTranspiler::emitMetaTwoByte(uint32_t functionObjectOffset, + uint32_t templateObjectOffset) { + JSObject* templateObj = tenuredObjectStubField(templateObjectOffset); + MConstant* templateConst = constant(ObjectValue(*templateObj)); + + // TODO: support pre-tenuring. + gc::InitialHeap heap = gc::DefaultHeap; + + auto* createThis = MCreateThisWithTemplate::New(alloc(), templateConst, heap); + add(createThis); + + callInfo_->thisArg()->setImplicitlyUsedUnchecked(); + callInfo_->setThis(createThis); + return true; +} + +bool WarpCacheIRTranspiler::emitReturnFromIC() { return true; } + +bool WarpCacheIRTranspiler::emitBailout() { + auto* bail = MBail::New(alloc()); + add(bail); + + return true; +} + +bool WarpCacheIRTranspiler::emitAssertRecoveredOnBailoutResult( + ValOperandId valId, bool mustBeRecovered) { + MDefinition* val = getOperand(valId); + + // Don't assert for recovered instructions when recovering is disabled. + if (JitOptions.disableRecoverIns) { + pushResult(constant(UndefinedValue())); + return true; + } + + if (JitOptions.checkRangeAnalysis) { + // If we are checking the range of all instructions, then the guards + // inserted by Range Analysis prevent the use of recover instruction. Thus, + // we just disable these checks. + pushResult(constant(UndefinedValue())); + return true; + } + + auto* assert = MAssertRecoveredOnBailout::New(alloc(), val, mustBeRecovered); + addEffectfulUnsafe(assert); + current->push(assert); + + // Create an instruction sequence which implies that the argument of the + // assertRecoveredOnBailout function would be encoded at least in one + // Snapshot. + auto* nop = MNop::New(alloc()); + add(nop); + + auto* resumePoint = MResumePoint::New( + alloc(), nop->block(), loc_.toRawBytecode(), MResumePoint::ResumeAfter); + if (!resumePoint) { + return false; + } + nop->setResumePoint(resumePoint); + + auto* encode = MEncodeSnapshot::New(alloc()); + addEffectfulUnsafe(encode); + + current->pop(); + + pushResult(constant(UndefinedValue())); + return true; +} + +static void MaybeSetImplicitlyUsed(uint32_t numInstructionIdsBefore, + MDefinition* input) { + // When building MIR from bytecode, for each MDefinition that's an operand to + // a bytecode instruction, we must either add an SSA use or set the + // ImplicitlyUsed flag on that definition. The ImplicitlyUsed flag prevents + // the backend from optimizing-out values that will be used by Baseline after + // a bailout. + // + // WarpBuilder uses WarpPoppedValueUseChecker to assert this invariant in + // debug builds. + // + // This function is responsible for setting the ImplicitlyUsed flag for an + // input when using the transpiler. It looks at the input's most recent use + // and if that's an instruction that was added while transpiling this JSOp + // (based on the MIR instruction id) we don't set the ImplicitlyUsed flag. + + if (input->isImplicitlyUsed()) { + // Nothing to do. + return; + } + + // If the most recent use of 'input' is an instruction we just added, there is + // nothing to do. + MDefinition* inputUse = input->maybeMostRecentlyAddedDefUse(); + if (inputUse && inputUse->id() >= numInstructionIdsBefore) { + return; + } + + // The transpiler didn't add a use for 'input'. + input->setImplicitlyUsed(); +} + +bool jit::TranspileCacheIRToMIR(WarpBuilder* builder, BytecodeLocation loc, + const WarpCacheIR* cacheIRSnapshot, + std::initializer_list<MDefinition*> inputs, + CallInfo* maybeCallInfo) { + uint32_t numInstructionIdsBefore = + builder->mirGen().graph().getNumInstructionIds(); + + WarpCacheIRTranspiler transpiler(builder, loc, maybeCallInfo, + cacheIRSnapshot); + if (!transpiler.transpile(inputs)) { + return false; + } + + for (MDefinition* input : inputs) { + MaybeSetImplicitlyUsed(numInstructionIdsBefore, input); + } + + if (maybeCallInfo) { + auto maybeSetFlag = [numInstructionIdsBefore](MDefinition* def) { + MaybeSetImplicitlyUsed(numInstructionIdsBefore, def); + }; + maybeCallInfo->forEachCallOperand(maybeSetFlag); + } + + return true; +} |