| // Copyright 2014 the V8 project authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "src/compiler/js-operator.h" |
| |
| #include <limits> |
| |
| #include "src/base/lazy-instance.h" |
| #include "src/compiler/js-graph.h" |
| #include "src/compiler/node-matchers.h" |
| #include "src/compiler/operator.h" |
| #include "src/handles/handles-inl.h" |
| #include "src/objects/objects-inl.h" |
| #include "src/objects/template-objects.h" |
| |
| namespace v8 { |
| namespace internal { |
| namespace compiler { |
| |
| namespace { |
| |
| // Returns properties for the given binary op. |
| constexpr Operator::Properties BinopProperties(Operator::Opcode opcode) { |
| CONSTEXPR_DCHECK(JSOperator::IsBinaryWithFeedback(opcode)); |
| return opcode == IrOpcode::kJSStrictEqual ? Operator::kPure |
| : Operator::kNoProperties; |
| } |
| |
| } // namespace |
| |
| namespace js_node_wrapper_utils { |
| |
| TNode<Oddball> UndefinedConstant(JSGraph* jsgraph) { |
| return TNode<Oddball>::UncheckedCast(jsgraph->UndefinedConstant()); |
| } |
| |
| } // namespace js_node_wrapper_utils |
| |
| FeedbackCellRef JSCreateClosureNode::GetFeedbackCellRefChecked( |
| JSHeapBroker* broker) const { |
| HeapObjectMatcher m(feedback_cell()); |
| CHECK(m.HasResolvedValue()); |
| return FeedbackCellRef(broker, m.ResolvedValue()); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, CallFrequency const& f) { |
| if (f.IsUnknown()) return os << "unknown"; |
| return os << f.value(); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, |
| ConstructForwardVarargsParameters const& p) { |
| return os << p.arity() << ", " << p.start_index(); |
| } |
| |
| ConstructForwardVarargsParameters const& ConstructForwardVarargsParametersOf( |
| Operator const* op) { |
| DCHECK_EQ(IrOpcode::kJSConstructForwardVarargs, op->opcode()); |
| return OpParameter<ConstructForwardVarargsParameters>(op); |
| } |
| |
| bool operator==(ConstructParameters const& lhs, |
| ConstructParameters const& rhs) { |
| return lhs.arity() == rhs.arity() && lhs.frequency() == rhs.frequency() && |
| lhs.feedback() == rhs.feedback(); |
| } |
| |
| bool operator!=(ConstructParameters const& lhs, |
| ConstructParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(ConstructParameters const& p) { |
| return base::hash_combine(p.arity(), p.frequency(), |
| FeedbackSource::Hash()(p.feedback())); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, ConstructParameters const& p) { |
| return os << p.arity() << ", " << p.frequency(); |
| } |
| |
| ConstructParameters const& ConstructParametersOf(Operator const* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSConstruct || |
| op->opcode() == IrOpcode::kJSConstructWithArrayLike || |
| op->opcode() == IrOpcode::kJSConstructWithSpread); |
| return OpParameter<ConstructParameters>(op); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, CallParameters const& p) { |
| return os << p.arity() << ", " << p.frequency() << ", " << p.convert_mode() |
| << ", " << p.speculation_mode() << ", " << p.feedback_relation(); |
| } |
| |
| const CallParameters& CallParametersOf(const Operator* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSCall || |
| op->opcode() == IrOpcode::kJSCallWithArrayLike || |
| op->opcode() == IrOpcode::kJSCallWithSpread); |
| return OpParameter<CallParameters>(op); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, |
| CallForwardVarargsParameters const& p) { |
| return os << p.arity() << ", " << p.start_index(); |
| } |
| |
| CallForwardVarargsParameters const& CallForwardVarargsParametersOf( |
| Operator const* op) { |
| DCHECK_EQ(IrOpcode::kJSCallForwardVarargs, op->opcode()); |
| return OpParameter<CallForwardVarargsParameters>(op); |
| } |
| |
| |
| bool operator==(CallRuntimeParameters const& lhs, |
| CallRuntimeParameters const& rhs) { |
| return lhs.id() == rhs.id() && lhs.arity() == rhs.arity(); |
| } |
| |
| |
| bool operator!=(CallRuntimeParameters const& lhs, |
| CallRuntimeParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| size_t hash_value(CallRuntimeParameters const& p) { |
| return base::hash_combine(p.id(), p.arity()); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, CallRuntimeParameters const& p) { |
| return os << p.id() << ", " << p.arity(); |
| } |
| |
| |
| const CallRuntimeParameters& CallRuntimeParametersOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSCallRuntime, op->opcode()); |
| return OpParameter<CallRuntimeParameters>(op); |
| } |
| |
| |
| ContextAccess::ContextAccess(size_t depth, size_t index, bool immutable) |
| : immutable_(immutable), |
| depth_(static_cast<uint16_t>(depth)), |
| index_(static_cast<uint32_t>(index)) { |
| DCHECK(depth <= std::numeric_limits<uint16_t>::max()); |
| DCHECK(index <= std::numeric_limits<uint32_t>::max()); |
| } |
| |
| |
| bool operator==(ContextAccess const& lhs, ContextAccess const& rhs) { |
| return lhs.depth() == rhs.depth() && lhs.index() == rhs.index() && |
| lhs.immutable() == rhs.immutable(); |
| } |
| |
| |
| bool operator!=(ContextAccess const& lhs, ContextAccess const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| size_t hash_value(ContextAccess const& access) { |
| return base::hash_combine(access.depth(), access.index(), access.immutable()); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, ContextAccess const& access) { |
| return os << access.depth() << ", " << access.index() << ", " |
| << access.immutable(); |
| } |
| |
| |
| ContextAccess const& ContextAccessOf(Operator const* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSLoadContext || |
| op->opcode() == IrOpcode::kJSStoreContext); |
| return OpParameter<ContextAccess>(op); |
| } |
| |
| CreateFunctionContextParameters::CreateFunctionContextParameters( |
| Handle<ScopeInfo> scope_info, int slot_count, ScopeType scope_type) |
| : scope_info_(scope_info), |
| slot_count_(slot_count), |
| scope_type_(scope_type) {} |
| |
| bool operator==(CreateFunctionContextParameters const& lhs, |
| CreateFunctionContextParameters const& rhs) { |
| return lhs.scope_info().location() == rhs.scope_info().location() && |
| lhs.slot_count() == rhs.slot_count() && |
| lhs.scope_type() == rhs.scope_type(); |
| } |
| |
| bool operator!=(CreateFunctionContextParameters const& lhs, |
| CreateFunctionContextParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(CreateFunctionContextParameters const& parameters) { |
| return base::hash_combine(parameters.scope_info().location(), |
| parameters.slot_count(), |
| static_cast<int>(parameters.scope_type())); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, |
| CreateFunctionContextParameters const& parameters) { |
| return os << parameters.slot_count() << ", " << parameters.scope_type(); |
| } |
| |
| CreateFunctionContextParameters const& CreateFunctionContextParametersOf( |
| Operator const* op) { |
| DCHECK_EQ(IrOpcode::kJSCreateFunctionContext, op->opcode()); |
| return OpParameter<CreateFunctionContextParameters>(op); |
| } |
| |
| bool operator==(StoreNamedOwnParameters const& lhs, |
| StoreNamedOwnParameters const& rhs) { |
| return lhs.name().location() == rhs.name().location() && |
| lhs.feedback() == rhs.feedback(); |
| } |
| |
| bool operator!=(StoreNamedOwnParameters const& lhs, |
| StoreNamedOwnParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(StoreNamedOwnParameters const& p) { |
| return base::hash_combine(p.name().location(), |
| FeedbackSource::Hash()(p.feedback())); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, StoreNamedOwnParameters const& p) { |
| return os << Brief(*p.name()); |
| } |
| |
| StoreNamedOwnParameters const& StoreNamedOwnParametersOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSStoreNamedOwn, op->opcode()); |
| return OpParameter<StoreNamedOwnParameters>(op); |
| } |
| |
| bool operator==(FeedbackParameter const& lhs, FeedbackParameter const& rhs) { |
| return lhs.feedback() == rhs.feedback(); |
| } |
| |
| bool operator!=(FeedbackParameter const& lhs, FeedbackParameter const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(FeedbackParameter const& p) { |
| return FeedbackSource::Hash()(p.feedback()); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, FeedbackParameter const& p) { |
| return os << p.feedback(); |
| } |
| |
| FeedbackParameter const& FeedbackParameterOf(const Operator* op) { |
| DCHECK(JSOperator::IsUnaryWithFeedback(op->opcode()) || |
| JSOperator::IsBinaryWithFeedback(op->opcode()) || |
| op->opcode() == IrOpcode::kJSCreateEmptyLiteralArray || |
| op->opcode() == IrOpcode::kJSInstanceOf || |
| op->opcode() == IrOpcode::kJSStoreDataPropertyInLiteral || |
| op->opcode() == IrOpcode::kJSStoreInArrayLiteral); |
| return OpParameter<FeedbackParameter>(op); |
| } |
| |
| bool operator==(NamedAccess const& lhs, NamedAccess const& rhs) { |
| return lhs.name().location() == rhs.name().location() && |
| lhs.language_mode() == rhs.language_mode() && |
| lhs.feedback() == rhs.feedback(); |
| } |
| |
| |
| bool operator!=(NamedAccess const& lhs, NamedAccess const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| size_t hash_value(NamedAccess const& p) { |
| return base::hash_combine(p.name().location(), p.language_mode(), |
| FeedbackSource::Hash()(p.feedback())); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, NamedAccess const& p) { |
| return os << Brief(*p.name()) << ", " << p.language_mode(); |
| } |
| |
| |
| NamedAccess const& NamedAccessOf(const Operator* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSLoadNamed || |
| op->opcode() == IrOpcode::kJSLoadNamedFromSuper || |
| op->opcode() == IrOpcode::kJSStoreNamed); |
| return OpParameter<NamedAccess>(op); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, PropertyAccess const& p) { |
| return os << p.language_mode() << ", " << p.feedback(); |
| } |
| |
| |
| bool operator==(PropertyAccess const& lhs, PropertyAccess const& rhs) { |
| return lhs.language_mode() == rhs.language_mode() && |
| lhs.feedback() == rhs.feedback(); |
| } |
| |
| |
| bool operator!=(PropertyAccess const& lhs, PropertyAccess const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| PropertyAccess const& PropertyAccessOf(const Operator* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSHasProperty || |
| op->opcode() == IrOpcode::kJSLoadProperty || |
| op->opcode() == IrOpcode::kJSStoreProperty); |
| return OpParameter<PropertyAccess>(op); |
| } |
| |
| |
| size_t hash_value(PropertyAccess const& p) { |
| return base::hash_combine(p.language_mode(), |
| FeedbackSource::Hash()(p.feedback())); |
| } |
| |
| |
| bool operator==(LoadGlobalParameters const& lhs, |
| LoadGlobalParameters const& rhs) { |
| return lhs.name().location() == rhs.name().location() && |
| lhs.feedback() == rhs.feedback() && |
| lhs.typeof_mode() == rhs.typeof_mode(); |
| } |
| |
| |
| bool operator!=(LoadGlobalParameters const& lhs, |
| LoadGlobalParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| size_t hash_value(LoadGlobalParameters const& p) { |
| return base::hash_combine(p.name().location(), p.typeof_mode()); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, LoadGlobalParameters const& p) { |
| return os << Brief(*p.name()) << ", " << p.typeof_mode(); |
| } |
| |
| |
| const LoadGlobalParameters& LoadGlobalParametersOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSLoadGlobal, op->opcode()); |
| return OpParameter<LoadGlobalParameters>(op); |
| } |
| |
| |
| bool operator==(StoreGlobalParameters const& lhs, |
| StoreGlobalParameters const& rhs) { |
| return lhs.language_mode() == rhs.language_mode() && |
| lhs.name().location() == rhs.name().location() && |
| lhs.feedback() == rhs.feedback(); |
| } |
| |
| |
| bool operator!=(StoreGlobalParameters const& lhs, |
| StoreGlobalParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| size_t hash_value(StoreGlobalParameters const& p) { |
| return base::hash_combine(p.language_mode(), p.name().location(), |
| FeedbackSource::Hash()(p.feedback())); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, StoreGlobalParameters const& p) { |
| return os << p.language_mode() << ", " << Brief(*p.name()); |
| } |
| |
| |
| const StoreGlobalParameters& StoreGlobalParametersOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSStoreGlobal, op->opcode()); |
| return OpParameter<StoreGlobalParameters>(op); |
| } |
| |
| |
| CreateArgumentsType const& CreateArgumentsTypeOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSCreateArguments, op->opcode()); |
| return OpParameter<CreateArgumentsType>(op); |
| } |
| |
| |
| bool operator==(CreateArrayParameters const& lhs, |
| CreateArrayParameters const& rhs) { |
| return lhs.arity() == rhs.arity() && |
| lhs.site().address() == rhs.site().address(); |
| } |
| |
| |
| bool operator!=(CreateArrayParameters const& lhs, |
| CreateArrayParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| size_t hash_value(CreateArrayParameters const& p) { |
| return base::hash_combine(p.arity(), p.site().address()); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, CreateArrayParameters const& p) { |
| os << p.arity(); |
| Handle<AllocationSite> site; |
| if (p.site().ToHandle(&site)) os << ", " << Brief(*site); |
| return os; |
| } |
| |
| const CreateArrayParameters& CreateArrayParametersOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSCreateArray, op->opcode()); |
| return OpParameter<CreateArrayParameters>(op); |
| } |
| |
| bool operator==(CreateArrayIteratorParameters const& lhs, |
| CreateArrayIteratorParameters const& rhs) { |
| return lhs.kind() == rhs.kind(); |
| } |
| |
| bool operator!=(CreateArrayIteratorParameters const& lhs, |
| CreateArrayIteratorParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(CreateArrayIteratorParameters const& p) { |
| return static_cast<size_t>(p.kind()); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, |
| CreateArrayIteratorParameters const& p) { |
| return os << p.kind(); |
| } |
| |
| const CreateArrayIteratorParameters& CreateArrayIteratorParametersOf( |
| const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSCreateArrayIterator, op->opcode()); |
| return OpParameter<CreateArrayIteratorParameters>(op); |
| } |
| |
| bool operator==(CreateCollectionIteratorParameters const& lhs, |
| CreateCollectionIteratorParameters const& rhs) { |
| return lhs.collection_kind() == rhs.collection_kind() && |
| lhs.iteration_kind() == rhs.iteration_kind(); |
| } |
| |
| bool operator!=(CreateCollectionIteratorParameters const& lhs, |
| CreateCollectionIteratorParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(CreateCollectionIteratorParameters const& p) { |
| return base::hash_combine(static_cast<size_t>(p.collection_kind()), |
| static_cast<size_t>(p.iteration_kind())); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, |
| CreateCollectionIteratorParameters const& p) { |
| return os << p.collection_kind() << ", " << p.iteration_kind(); |
| } |
| |
| const CreateCollectionIteratorParameters& CreateCollectionIteratorParametersOf( |
| const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSCreateCollectionIterator, op->opcode()); |
| return OpParameter<CreateCollectionIteratorParameters>(op); |
| } |
| |
| bool operator==(CreateBoundFunctionParameters const& lhs, |
| CreateBoundFunctionParameters const& rhs) { |
| return lhs.arity() == rhs.arity() && |
| lhs.map().location() == rhs.map().location(); |
| } |
| |
| bool operator!=(CreateBoundFunctionParameters const& lhs, |
| CreateBoundFunctionParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(CreateBoundFunctionParameters const& p) { |
| return base::hash_combine(p.arity(), p.map().location()); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, |
| CreateBoundFunctionParameters const& p) { |
| os << p.arity(); |
| if (!p.map().is_null()) os << ", " << Brief(*p.map()); |
| return os; |
| } |
| |
| const CreateBoundFunctionParameters& CreateBoundFunctionParametersOf( |
| const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSCreateBoundFunction, op->opcode()); |
| return OpParameter<CreateBoundFunctionParameters>(op); |
| } |
| |
| bool operator==(GetTemplateObjectParameters const& lhs, |
| GetTemplateObjectParameters const& rhs) { |
| return lhs.description().location() == rhs.description().location() && |
| lhs.shared().location() == rhs.shared().location() && |
| lhs.feedback() == rhs.feedback(); |
| } |
| |
| bool operator!=(GetTemplateObjectParameters const& lhs, |
| GetTemplateObjectParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(GetTemplateObjectParameters const& p) { |
| return base::hash_combine(p.description().location(), p.shared().location(), |
| FeedbackSource::Hash()(p.feedback())); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, |
| GetTemplateObjectParameters const& p) { |
| return os << Brief(*p.description()) << ", " << Brief(*p.shared()); |
| } |
| |
| const GetTemplateObjectParameters& GetTemplateObjectParametersOf( |
| const Operator* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSGetTemplateObject); |
| return OpParameter<GetTemplateObjectParameters>(op); |
| } |
| |
| bool operator==(CreateClosureParameters const& lhs, |
| CreateClosureParameters const& rhs) { |
| return lhs.allocation() == rhs.allocation() && |
| lhs.code().location() == rhs.code().location() && |
| lhs.shared_info().location() == rhs.shared_info().location(); |
| } |
| |
| |
| bool operator!=(CreateClosureParameters const& lhs, |
| CreateClosureParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| size_t hash_value(CreateClosureParameters const& p) { |
| return base::hash_combine(p.allocation(), p.shared_info().location()); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, CreateClosureParameters const& p) { |
| return os << p.allocation() << ", " << Brief(*p.shared_info()) << ", " |
| << Brief(*p.code()); |
| } |
| |
| |
| const CreateClosureParameters& CreateClosureParametersOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSCreateClosure, op->opcode()); |
| return OpParameter<CreateClosureParameters>(op); |
| } |
| |
| |
| bool operator==(CreateLiteralParameters const& lhs, |
| CreateLiteralParameters const& rhs) { |
| return lhs.constant().location() == rhs.constant().location() && |
| lhs.feedback() == rhs.feedback() && lhs.length() == rhs.length() && |
| lhs.flags() == rhs.flags(); |
| } |
| |
| |
| bool operator!=(CreateLiteralParameters const& lhs, |
| CreateLiteralParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| |
| size_t hash_value(CreateLiteralParameters const& p) { |
| return base::hash_combine(p.constant().location(), |
| FeedbackSource::Hash()(p.feedback()), p.length(), |
| p.flags()); |
| } |
| |
| |
| std::ostream& operator<<(std::ostream& os, CreateLiteralParameters const& p) { |
| return os << Brief(*p.constant()) << ", " << p.length() << ", " << p.flags(); |
| } |
| |
| |
| const CreateLiteralParameters& CreateLiteralParametersOf(const Operator* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSCreateLiteralArray || |
| op->opcode() == IrOpcode::kJSCreateLiteralObject || |
| op->opcode() == IrOpcode::kJSCreateLiteralRegExp); |
| return OpParameter<CreateLiteralParameters>(op); |
| } |
| |
| bool operator==(CloneObjectParameters const& lhs, |
| CloneObjectParameters const& rhs) { |
| return lhs.feedback() == rhs.feedback() && lhs.flags() == rhs.flags(); |
| } |
| |
| bool operator!=(CloneObjectParameters const& lhs, |
| CloneObjectParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(CloneObjectParameters const& p) { |
| return base::hash_combine(FeedbackSource::Hash()(p.feedback()), p.flags()); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, CloneObjectParameters const& p) { |
| return os << p.flags(); |
| } |
| |
| const CloneObjectParameters& CloneObjectParametersOf(const Operator* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSCloneObject); |
| return OpParameter<CloneObjectParameters>(op); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, GetIteratorParameters const& p) { |
| return os << p.loadFeedback() << ", " << p.callFeedback(); |
| } |
| |
| bool operator==(GetIteratorParameters const& lhs, |
| GetIteratorParameters const& rhs) { |
| return lhs.loadFeedback() == rhs.loadFeedback() && |
| lhs.callFeedback() == rhs.callFeedback(); |
| } |
| |
| bool operator!=(GetIteratorParameters const& lhs, |
| GetIteratorParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| GetIteratorParameters const& GetIteratorParametersOf(const Operator* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSGetIterator); |
| return OpParameter<GetIteratorParameters>(op); |
| } |
| |
| size_t hash_value(GetIteratorParameters const& p) { |
| return base::hash_combine(FeedbackSource::Hash()(p.loadFeedback()), |
| FeedbackSource::Hash()(p.callFeedback())); |
| } |
| |
| size_t hash_value(ForInMode const& mode) { return static_cast<uint8_t>(mode); } |
| |
| std::ostream& operator<<(std::ostream& os, ForInMode const& mode) { |
| switch (mode) { |
| case ForInMode::kUseEnumCacheKeysAndIndices: |
| return os << "UseEnumCacheKeysAndIndices"; |
| case ForInMode::kUseEnumCacheKeys: |
| return os << "UseEnumCacheKeys"; |
| case ForInMode::kGeneric: |
| return os << "Generic"; |
| } |
| UNREACHABLE(); |
| } |
| |
| bool operator==(ForInParameters const& lhs, ForInParameters const& rhs) { |
| return lhs.feedback() == rhs.feedback() && lhs.mode() == rhs.mode(); |
| } |
| |
| bool operator!=(ForInParameters const& lhs, ForInParameters const& rhs) { |
| return !(lhs == rhs); |
| } |
| |
| size_t hash_value(ForInParameters const& p) { |
| return base::hash_combine(FeedbackSource::Hash()(p.feedback()), p.mode()); |
| } |
| |
| std::ostream& operator<<(std::ostream& os, ForInParameters const& p) { |
| return os << p.feedback() << ", " << p.mode(); |
| } |
| |
| ForInParameters const& ForInParametersOf(const Operator* op) { |
| DCHECK(op->opcode() == IrOpcode::kJSForInNext || |
| op->opcode() == IrOpcode::kJSForInPrepare); |
| return OpParameter<ForInParameters>(op); |
| } |
| |
| #define CACHED_OP_LIST(V) \ |
| V(ToLength, Operator::kNoProperties, 1, 1) \ |
| V(ToName, Operator::kNoProperties, 1, 1) \ |
| V(ToNumber, Operator::kNoProperties, 1, 1) \ |
| V(ToNumberConvertBigInt, Operator::kNoProperties, 1, 1) \ |
| V(ToNumeric, Operator::kNoProperties, 1, 1) \ |
| V(ToObject, Operator::kFoldable, 1, 1) \ |
| V(ToString, Operator::kNoProperties, 1, 1) \ |
| V(Create, Operator::kNoProperties, 2, 1) \ |
| V(CreateIterResultObject, Operator::kEliminatable, 2, 1) \ |
| V(CreateStringIterator, Operator::kEliminatable, 1, 1) \ |
| V(CreateKeyValueArray, Operator::kEliminatable, 2, 1) \ |
| V(CreatePromise, Operator::kEliminatable, 0, 1) \ |
| V(CreateTypedArray, Operator::kNoProperties, 5, 1) \ |
| V(CreateObject, Operator::kNoProperties, 1, 1) \ |
| V(ObjectIsArray, Operator::kNoProperties, 1, 1) \ |
| V(HasInPrototypeChain, Operator::kNoProperties, 2, 1) \ |
| V(OrdinaryHasInstance, Operator::kNoProperties, 2, 1) \ |
| V(ForInEnumerate, Operator::kNoProperties, 1, 1) \ |
| V(AsyncFunctionEnter, Operator::kNoProperties, 2, 1) \ |
| V(AsyncFunctionReject, Operator::kNoDeopt | Operator::kNoThrow, 3, 1) \ |
| V(AsyncFunctionResolve, Operator::kNoDeopt | Operator::kNoThrow, 3, 1) \ |
| V(LoadMessage, Operator::kNoThrow | Operator::kNoWrite, 0, 1) \ |
| V(StoreMessage, Operator::kNoRead | Operator::kNoThrow, 1, 0) \ |
| V(GeneratorRestoreContinuation, Operator::kNoThrow, 1, 1) \ |
| V(GeneratorRestoreContext, Operator::kNoThrow, 1, 1) \ |
| V(GeneratorRestoreInputOrDebugPos, Operator::kNoThrow, 1, 1) \ |
| V(Debugger, Operator::kNoProperties, 0, 0) \ |
| V(FulfillPromise, Operator::kNoDeopt | Operator::kNoThrow, 2, 1) \ |
| V(PerformPromiseThen, Operator::kNoDeopt | Operator::kNoThrow, 4, 1) \ |
| V(PromiseResolve, Operator::kNoProperties, 2, 1) \ |
| V(RejectPromise, Operator::kNoDeopt | Operator::kNoThrow, 3, 1) \ |
| V(ResolvePromise, Operator::kNoDeopt | Operator::kNoThrow, 2, 1) \ |
| V(GetSuperConstructor, Operator::kNoWrite | Operator::kNoThrow, 1, 1) \ |
| V(ParseInt, Operator::kNoProperties, 2, 1) \ |
| V(RegExpTest, Operator::kNoProperties, 2, 1) |
| |
| struct JSOperatorGlobalCache final { |
| #define CACHED_OP(Name, properties, value_input_count, value_output_count) \ |
| struct Name##Operator final : public Operator { \ |
| Name##Operator() \ |
| : Operator(IrOpcode::kJS##Name, properties, "JS" #Name, \ |
| value_input_count, Operator::ZeroIfPure(properties), \ |
| Operator::ZeroIfEliminatable(properties), \ |
| value_output_count, Operator::ZeroIfPure(properties), \ |
| Operator::ZeroIfNoThrow(properties)) {} \ |
| }; \ |
| Name##Operator k##Name##Operator; |
| CACHED_OP_LIST(CACHED_OP) |
| #undef CACHED_OP |
| }; |
| |
| namespace { |
| DEFINE_LAZY_LEAKY_OBJECT_GETTER(JSOperatorGlobalCache, GetJSOperatorGlobalCache) |
| } // namespace |
| |
| JSOperatorBuilder::JSOperatorBuilder(Zone* zone) |
| : cache_(*GetJSOperatorGlobalCache()), zone_(zone) {} |
| |
| #define CACHED_OP(Name, properties, value_input_count, value_output_count) \ |
| const Operator* JSOperatorBuilder::Name() { \ |
| return &cache_.k##Name##Operator; \ |
| } |
| CACHED_OP_LIST(CACHED_OP) |
| #undef CACHED_OP |
| |
| #define UNARY_OP(JSName, Name) \ |
| const Operator* JSOperatorBuilder::Name(FeedbackSource const& feedback) { \ |
| FeedbackParameter parameters(feedback); \ |
| return zone()->New<Operator1<FeedbackParameter>>( \ |
| IrOpcode::k##JSName, Operator::kNoProperties, #JSName, 2, 1, 1, 1, 1, \ |
| 2, parameters); \ |
| } |
| JS_UNOP_WITH_FEEDBACK(UNARY_OP) |
| #undef UNARY_OP |
| |
| #define BINARY_OP(JSName, Name) \ |
| const Operator* JSOperatorBuilder::Name(FeedbackSource const& feedback) { \ |
| static constexpr auto kProperties = BinopProperties(IrOpcode::k##JSName); \ |
| FeedbackParameter parameters(feedback); \ |
| return zone()->New<Operator1<FeedbackParameter>>( \ |
| IrOpcode::k##JSName, kProperties, #JSName, 3, 1, 1, 1, 1, \ |
| Operator::ZeroIfNoThrow(kProperties), parameters); \ |
| } |
| JS_BINOP_WITH_FEEDBACK(BINARY_OP) |
| #undef BINARY_OP |
| |
| const Operator* JSOperatorBuilder::StoreDataPropertyInLiteral( |
| const FeedbackSource& feedback) { |
| static constexpr int kObject = 1; |
| static constexpr int kName = 1; |
| static constexpr int kValue = 1; |
| static constexpr int kFlags = 1; |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = |
| kObject + kName + kValue + kFlags + kFeedbackVector; |
| FeedbackParameter parameters(feedback); |
| return zone()->New<Operator1<FeedbackParameter>>( // -- |
| IrOpcode::kJSStoreDataPropertyInLiteral, |
| Operator::kNoThrow, // opcode |
| "JSStoreDataPropertyInLiteral", // name |
| kArity, 1, 1, 0, 1, 1, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::StoreInArrayLiteral( |
| const FeedbackSource& feedback) { |
| static constexpr int kArray = 1; |
| static constexpr int kIndex = 1; |
| static constexpr int kValue = 1; |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = kArray + kIndex + kValue + kFeedbackVector; |
| FeedbackParameter parameters(feedback); |
| return zone()->New<Operator1<FeedbackParameter>>( // -- |
| IrOpcode::kJSStoreInArrayLiteral, |
| Operator::kNoThrow, // opcode |
| "JSStoreInArrayLiteral", // name |
| kArity, 1, 1, 0, 1, 1, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CallForwardVarargs(size_t arity, |
| uint32_t start_index) { |
| CallForwardVarargsParameters parameters(arity, start_index); |
| return zone()->New<Operator1<CallForwardVarargsParameters>>( // -- |
| IrOpcode::kJSCallForwardVarargs, Operator::kNoProperties, // opcode |
| "JSCallForwardVarargs", // name |
| parameters.arity(), 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::Call( |
| size_t arity, CallFrequency const& frequency, |
| FeedbackSource const& feedback, ConvertReceiverMode convert_mode, |
| SpeculationMode speculation_mode, CallFeedbackRelation feedback_relation) { |
| CallParameters parameters(arity, frequency, feedback, convert_mode, |
| speculation_mode, feedback_relation); |
| return zone()->New<Operator1<CallParameters>>( // -- |
| IrOpcode::kJSCall, Operator::kNoProperties, // opcode |
| "JSCall", // name |
| parameters.arity(), 1, 1, 1, 1, 2, // inputs/outputs |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CallWithArrayLike( |
| const CallFrequency& frequency, const FeedbackSource& feedback, |
| SpeculationMode speculation_mode, CallFeedbackRelation feedback_relation) { |
| static constexpr int kTheArrayLikeObject = 1; |
| CallParameters parameters( |
| JSCallWithArrayLikeNode::ArityForArgc(kTheArrayLikeObject), frequency, |
| feedback, ConvertReceiverMode::kAny, speculation_mode, feedback_relation); |
| return zone()->New<Operator1<CallParameters>>( // -- |
| IrOpcode::kJSCallWithArrayLike, Operator::kNoProperties, // opcode |
| "JSCallWithArrayLike", // name |
| parameters.arity(), 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CallWithSpread( |
| uint32_t arity, CallFrequency const& frequency, |
| FeedbackSource const& feedback, SpeculationMode speculation_mode, |
| CallFeedbackRelation feedback_relation) { |
| DCHECK_IMPLIES(speculation_mode == SpeculationMode::kAllowSpeculation, |
| feedback.IsValid()); |
| CallParameters parameters(arity, frequency, feedback, |
| ConvertReceiverMode::kAny, speculation_mode, |
| feedback_relation); |
| return zone()->New<Operator1<CallParameters>>( // -- |
| IrOpcode::kJSCallWithSpread, Operator::kNoProperties, // opcode |
| "JSCallWithSpread", // name |
| parameters.arity(), 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CallRuntime(Runtime::FunctionId id) { |
| const Runtime::Function* f = Runtime::FunctionForId(id); |
| return CallRuntime(f, f->nargs); |
| } |
| |
| |
| const Operator* JSOperatorBuilder::CallRuntime(Runtime::FunctionId id, |
| size_t arity) { |
| const Runtime::Function* f = Runtime::FunctionForId(id); |
| return CallRuntime(f, arity); |
| } |
| |
| |
| const Operator* JSOperatorBuilder::CallRuntime(const Runtime::Function* f, |
| size_t arity) { |
| CallRuntimeParameters parameters(f->function_id, arity); |
| DCHECK(f->nargs == -1 || f->nargs == static_cast<int>(parameters.arity())); |
| return zone()->New<Operator1<CallRuntimeParameters>>( // -- |
| IrOpcode::kJSCallRuntime, Operator::kNoProperties, // opcode |
| "JSCallRuntime", // name |
| parameters.arity(), 1, 1, f->result_size, 1, 2, // inputs/outputs |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::ConstructForwardVarargs( |
| size_t arity, uint32_t start_index) { |
| ConstructForwardVarargsParameters parameters(arity, start_index); |
| return zone()->New<Operator1<ConstructForwardVarargsParameters>>( // -- |
| IrOpcode::kJSConstructForwardVarargs, Operator::kNoProperties, // opcode |
| "JSConstructForwardVarargs", // name |
| parameters.arity(), 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| // Note: frequency is taken by reference to work around a GCC bug |
| // on AIX (v8:8193). |
| const Operator* JSOperatorBuilder::Construct(uint32_t arity, |
| CallFrequency const& frequency, |
| FeedbackSource const& feedback) { |
| ConstructParameters parameters(arity, frequency, feedback); |
| return zone()->New<Operator1<ConstructParameters>>( // -- |
| IrOpcode::kJSConstruct, Operator::kNoProperties, // opcode |
| "JSConstruct", // name |
| parameters.arity(), 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::ConstructWithArrayLike( |
| CallFrequency const& frequency, FeedbackSource const& feedback) { |
| static constexpr int kTheArrayLikeObject = 1; |
| ConstructParameters parameters( |
| JSConstructWithArrayLikeNode::ArityForArgc(kTheArrayLikeObject), |
| frequency, feedback); |
| return zone()->New<Operator1<ConstructParameters>>( // -- |
| IrOpcode::kJSConstructWithArrayLike, // opcode |
| Operator::kNoProperties, // properties |
| "JSConstructWithArrayLike", // name |
| parameters.arity(), 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::ConstructWithSpread( |
| uint32_t arity, CallFrequency const& frequency, |
| FeedbackSource const& feedback) { |
| ConstructParameters parameters(arity, frequency, feedback); |
| return zone()->New<Operator1<ConstructParameters>>( // -- |
| IrOpcode::kJSConstructWithSpread, Operator::kNoProperties, // opcode |
| "JSConstructWithSpread", // name |
| parameters.arity(), 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::LoadNamed(Handle<Name> name, |
| const FeedbackSource& feedback) { |
| static constexpr int kObject = 1; |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = kObject + kFeedbackVector; |
| NamedAccess access(LanguageMode::kSloppy, name, feedback); |
| return zone()->New<Operator1<NamedAccess>>( // -- |
| IrOpcode::kJSLoadNamed, Operator::kNoProperties, // opcode |
| "JSLoadNamed", // name |
| kArity, 1, 1, 1, 1, 2, // counts |
| access); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::LoadNamedFromSuper( |
| Handle<Name> name, const FeedbackSource& feedback) { |
| static constexpr int kReceiver = 1; |
| static constexpr int kHomeObject = 1; |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = kReceiver + kHomeObject + kFeedbackVector; |
| NamedAccess access(LanguageMode::kSloppy, name, feedback); |
| return zone()->New<Operator1<NamedAccess>>( // -- |
| IrOpcode::kJSLoadNamedFromSuper, Operator::kNoProperties, // opcode |
| "JSLoadNamedFromSuper", // name |
| kArity, 1, 1, 1, 1, 2, // counts |
| access); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::LoadProperty( |
| FeedbackSource const& feedback) { |
| PropertyAccess access(LanguageMode::kSloppy, feedback); |
| return zone()->New<Operator1<PropertyAccess>>( // -- |
| IrOpcode::kJSLoadProperty, Operator::kNoProperties, // opcode |
| "JSLoadProperty", // name |
| 3, 1, 1, 1, 1, 2, // counts |
| access); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::GetIterator( |
| FeedbackSource const& load_feedback, FeedbackSource const& call_feedback) { |
| GetIteratorParameters access(load_feedback, call_feedback); |
| return zone()->New<Operator1<GetIteratorParameters>>( // -- |
| IrOpcode::kJSGetIterator, Operator::kNoProperties, // opcode |
| "JSGetIterator", // name |
| 2, 1, 1, 1, 1, 2, // counts |
| access); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::HasProperty(FeedbackSource const& feedback) { |
| PropertyAccess access(LanguageMode::kSloppy, feedback); |
| return zone()->New<Operator1<PropertyAccess>>( // -- |
| IrOpcode::kJSHasProperty, Operator::kNoProperties, // opcode |
| "JSHasProperty", // name |
| 3, 1, 1, 1, 1, 2, // counts |
| access); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::ForInNext(ForInMode mode, |
| const FeedbackSource& feedback) { |
| return zone()->New<Operator1<ForInParameters>>( // -- |
| IrOpcode::kJSForInNext, Operator::kNoProperties, // opcode |
| "JSForInNext", // name |
| 5, 1, 1, 1, 1, 2, // counts |
| ForInParameters{feedback, mode}); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::ForInPrepare( |
| ForInMode mode, const FeedbackSource& feedback) { |
| return zone()->New<Operator1<ForInParameters>>( // -- |
| IrOpcode::kJSForInPrepare, // opcode |
| Operator::kNoWrite | Operator::kNoThrow, // flags |
| "JSForInPrepare", // name |
| 2, 1, 1, 3, 1, 1, // counts |
| ForInParameters{feedback, mode}); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::GeneratorStore(int register_count) { |
| return zone()->New<Operator1<int>>( // -- |
| IrOpcode::kJSGeneratorStore, Operator::kNoThrow, // opcode |
| "JSGeneratorStore", // name |
| 3 + register_count, 1, 1, 0, 1, 0, // counts |
| register_count); // parameter |
| } |
| |
| int RegisterCountOf(Operator const* op) { |
| DCHECK_EQ(IrOpcode::kJSCreateAsyncFunctionObject, op->opcode()); |
| return OpParameter<int>(op); |
| } |
| |
| int GeneratorStoreValueCountOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSGeneratorStore, op->opcode()); |
| return OpParameter<int>(op); |
| } |
| |
| const Operator* JSOperatorBuilder::GeneratorRestoreRegister(int index) { |
| return zone()->New<Operator1<int>>( // -- |
| IrOpcode::kJSGeneratorRestoreRegister, Operator::kNoThrow, // opcode |
| "JSGeneratorRestoreRegister", // name |
| 1, 1, 1, 1, 1, 0, // counts |
| index); // parameter |
| } |
| |
| int RestoreRegisterIndexOf(const Operator* op) { |
| DCHECK_EQ(IrOpcode::kJSGeneratorRestoreRegister, op->opcode()); |
| return OpParameter<int>(op); |
| } |
| |
| const Operator* JSOperatorBuilder::StoreNamed(LanguageMode language_mode, |
| Handle<Name> name, |
| FeedbackSource const& feedback) { |
| static constexpr int kObject = 1; |
| static constexpr int kValue = 1; |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = kObject + kValue + kFeedbackVector; |
| NamedAccess access(language_mode, name, feedback); |
| return zone()->New<Operator1<NamedAccess>>( // -- |
| IrOpcode::kJSStoreNamed, Operator::kNoProperties, // opcode |
| "JSStoreNamed", // name |
| kArity, 1, 1, 0, 1, 2, // counts |
| access); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::StoreProperty( |
| LanguageMode language_mode, FeedbackSource const& feedback) { |
| PropertyAccess access(language_mode, feedback); |
| return zone()->New<Operator1<PropertyAccess>>( // -- |
| IrOpcode::kJSStoreProperty, Operator::kNoProperties, // opcode |
| "JSStoreProperty", // name |
| 4, 1, 1, 0, 1, 2, // counts |
| access); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::StoreNamedOwn( |
| Handle<Name> name, FeedbackSource const& feedback) { |
| static constexpr int kObject = 1; |
| static constexpr int kValue = 1; |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = kObject + kValue + kFeedbackVector; |
| StoreNamedOwnParameters parameters(name, feedback); |
| return zone()->New<Operator1<StoreNamedOwnParameters>>( // -- |
| IrOpcode::kJSStoreNamedOwn, Operator::kNoProperties, // opcode |
| "JSStoreNamedOwn", // name |
| kArity, 1, 1, 0, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::DeleteProperty() { |
| return zone()->New<Operator>( // -- |
| IrOpcode::kJSDeleteProperty, Operator::kNoProperties, // opcode |
| "JSDeleteProperty", // name |
| 3, 1, 1, 1, 1, 2); // counts |
| } |
| |
| const Operator* JSOperatorBuilder::CreateGeneratorObject() { |
| return zone()->New<Operator>( // -- |
| IrOpcode::kJSCreateGeneratorObject, Operator::kEliminatable, // opcode |
| "JSCreateGeneratorObject", // name |
| 2, 1, 1, 1, 1, 0); // counts |
| } |
| |
| const Operator* JSOperatorBuilder::LoadGlobal(const Handle<Name>& name, |
| const FeedbackSource& feedback, |
| TypeofMode typeof_mode) { |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = kFeedbackVector; |
| LoadGlobalParameters parameters(name, feedback, typeof_mode); |
| return zone()->New<Operator1<LoadGlobalParameters>>( // -- |
| IrOpcode::kJSLoadGlobal, Operator::kNoProperties, // opcode |
| "JSLoadGlobal", // name |
| kArity, 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::StoreGlobal(LanguageMode language_mode, |
| const Handle<Name>& name, |
| const FeedbackSource& feedback) { |
| static constexpr int kValue = 1; |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = kValue + kFeedbackVector; |
| StoreGlobalParameters parameters(language_mode, feedback, name); |
| return zone()->New<Operator1<StoreGlobalParameters>>( // -- |
| IrOpcode::kJSStoreGlobal, Operator::kNoProperties, // opcode |
| "JSStoreGlobal", // name |
| kArity, 1, 1, 0, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::HasContextExtension(size_t depth) { |
| return zone()->New<Operator1<size_t>>( // -- |
| IrOpcode::kJSHasContextExtension, // opcode |
| Operator::kNoWrite | Operator::kNoThrow, // flags |
| "JSHasContextExtension", // name |
| 0, 1, 0, 1, 1, 0, // counts |
| depth); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::LoadContext(size_t depth, size_t index, |
| bool immutable) { |
| ContextAccess access(depth, index, immutable); |
| return zone()->New<Operator1<ContextAccess>>( // -- |
| IrOpcode::kJSLoadContext, // opcode |
| Operator::kNoWrite | Operator::kNoThrow, // flags |
| "JSLoadContext", // name |
| 0, 1, 0, 1, 1, 0, // counts |
| access); // parameter |
| } |
| |
| |
| const Operator* JSOperatorBuilder::StoreContext(size_t depth, size_t index) { |
| ContextAccess access(depth, index, false); |
| return zone()->New<Operator1<ContextAccess>>( // -- |
| IrOpcode::kJSStoreContext, // opcode |
| Operator::kNoRead | Operator::kNoThrow, // flags |
| "JSStoreContext", // name |
| 1, 1, 1, 0, 1, 0, // counts |
| access); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::LoadModule(int32_t cell_index) { |
| return zone()->New<Operator1<int32_t>>( // -- |
| IrOpcode::kJSLoadModule, // opcode |
| Operator::kNoWrite | Operator::kNoThrow, // flags |
| "JSLoadModule", // name |
| 1, 1, 1, 1, 1, 0, // counts |
| cell_index); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::GetImportMeta() { |
| return zone()->New<Operator>( // -- |
| IrOpcode::kJSGetImportMeta, // opcode |
| Operator::kNoProperties, // flags |
| "JSGetImportMeta", // name |
| 0, 1, 1, 1, 1, 2); // counts |
| } |
| |
| const Operator* JSOperatorBuilder::StoreModule(int32_t cell_index) { |
| return zone()->New<Operator1<int32_t>>( // -- |
| IrOpcode::kJSStoreModule, // opcode |
| Operator::kNoRead | Operator::kNoThrow, // flags |
| "JSStoreModule", // name |
| 2, 1, 1, 0, 1, 0, // counts |
| cell_index); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateArguments(CreateArgumentsType type) { |
| return zone()->New<Operator1<CreateArgumentsType>>( // -- |
| IrOpcode::kJSCreateArguments, Operator::kEliminatable, // opcode |
| "JSCreateArguments", // name |
| 1, 1, 0, 1, 1, 0, // counts |
| type); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateArray( |
| size_t arity, MaybeHandle<AllocationSite> site) { |
| // constructor, new_target, arg1, ..., argN |
| int const value_input_count = static_cast<int>(arity) + 2; |
| CreateArrayParameters parameters(arity, site); |
| return zone()->New<Operator1<CreateArrayParameters>>( // -- |
| IrOpcode::kJSCreateArray, Operator::kNoProperties, // opcode |
| "JSCreateArray", // name |
| value_input_count, 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateArrayIterator(IterationKind kind) { |
| CreateArrayIteratorParameters parameters(kind); |
| return zone()->New<Operator1<CreateArrayIteratorParameters>>( // -- |
| IrOpcode::kJSCreateArrayIterator, Operator::kEliminatable, // opcode |
| "JSCreateArrayIterator", // name |
| 1, 1, 1, 1, 1, 0, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateAsyncFunctionObject( |
| int register_count) { |
| return zone()->New<Operator1<int>>( // -- |
| IrOpcode::kJSCreateAsyncFunctionObject, // opcode |
| Operator::kEliminatable, // flags |
| "JSCreateAsyncFunctionObject", // name |
| 3, 1, 1, 1, 1, 0, // counts |
| register_count); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateCollectionIterator( |
| CollectionKind collection_kind, IterationKind iteration_kind) { |
| CreateCollectionIteratorParameters parameters(collection_kind, |
| iteration_kind); |
| return zone()->New<Operator1<CreateCollectionIteratorParameters>>( |
| IrOpcode::kJSCreateCollectionIterator, Operator::kEliminatable, |
| "JSCreateCollectionIterator", 1, 1, 1, 1, 1, 0, parameters); |
| } |
| |
| const Operator* JSOperatorBuilder::CreateBoundFunction(size_t arity, |
| Handle<Map> map) { |
| // bound_target_function, bound_this, arg1, ..., argN |
| int const value_input_count = static_cast<int>(arity) + 2; |
| CreateBoundFunctionParameters parameters(arity, map); |
| return zone()->New<Operator1<CreateBoundFunctionParameters>>( // -- |
| IrOpcode::kJSCreateBoundFunction, Operator::kEliminatable, // opcode |
| "JSCreateBoundFunction", // name |
| value_input_count, 1, 1, 1, 1, 0, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateClosure( |
| Handle<SharedFunctionInfo> shared_info, Handle<Code> code, |
| AllocationType allocation) { |
| static constexpr int kFeedbackCell = 1; |
| static constexpr int kArity = kFeedbackCell; |
| CreateClosureParameters parameters(shared_info, code, allocation); |
| return zone()->New<Operator1<CreateClosureParameters>>( // -- |
| IrOpcode::kJSCreateClosure, Operator::kEliminatable, // opcode |
| "JSCreateClosure", // name |
| kArity, 1, 1, 1, 1, 0, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateLiteralArray( |
| Handle<ArrayBoilerplateDescription> description, |
| FeedbackSource const& feedback, int literal_flags, int number_of_elements) { |
| CreateLiteralParameters parameters(description, feedback, number_of_elements, |
| literal_flags); |
| return zone()->New<Operator1<CreateLiteralParameters>>( // -- |
| IrOpcode::kJSCreateLiteralArray, // opcode |
| Operator::kNoProperties, // properties |
| "JSCreateLiteralArray", // name |
| 1, 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateEmptyLiteralArray( |
| FeedbackSource const& feedback) { |
| static constexpr int kFeedbackVector = 1; |
| static constexpr int kArity = kFeedbackVector; |
| FeedbackParameter parameters(feedback); |
| return zone()->New<Operator1<FeedbackParameter>>( // -- |
| IrOpcode::kJSCreateEmptyLiteralArray, // opcode |
| Operator::kEliminatable, // properties |
| "JSCreateEmptyLiteralArray", // name |
| kArity, 1, 1, 1, 1, 0, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateArrayFromIterable() { |
| return zone()->New<Operator>( // -- |
| IrOpcode::kJSCreateArrayFromIterable, // opcode |
| Operator::kNoProperties, // properties |
| "JSCreateArrayFromIterable", // name |
| 1, 1, 1, 1, 1, 2); // counts |
| } |
| |
| const Operator* JSOperatorBuilder::CreateLiteralObject( |
| Handle<ObjectBoilerplateDescription> constant_properties, |
| FeedbackSource const& feedback, int literal_flags, |
| int number_of_properties) { |
| CreateLiteralParameters parameters(constant_properties, feedback, |
| number_of_properties, literal_flags); |
| return zone()->New<Operator1<CreateLiteralParameters>>( // -- |
| IrOpcode::kJSCreateLiteralObject, // opcode |
| Operator::kNoProperties, // properties |
| "JSCreateLiteralObject", // name |
| 1, 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::GetTemplateObject( |
| Handle<TemplateObjectDescription> description, |
| Handle<SharedFunctionInfo> shared, FeedbackSource const& feedback) { |
| GetTemplateObjectParameters parameters(description, shared, feedback); |
| return zone()->New<Operator1<GetTemplateObjectParameters>>( // -- |
| IrOpcode::kJSGetTemplateObject, // opcode |
| Operator::kEliminatable, // properties |
| "JSGetTemplateObject", // name |
| 1, 1, 1, 1, 1, 0, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CloneObject(FeedbackSource const& feedback, |
| int literal_flags) { |
| CloneObjectParameters parameters(feedback, literal_flags); |
| return zone()->New<Operator1<CloneObjectParameters>>( // -- |
| IrOpcode::kJSCloneObject, // opcode |
| Operator::kNoProperties, // properties |
| "JSCloneObject", // name |
| 2, 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::StackCheck(StackCheckKind kind) { |
| return zone()->New<Operator1<StackCheckKind>>( // -- |
| IrOpcode::kJSStackCheck, // opcode |
| Operator::kNoWrite, // properties |
| "JSStackCheck", // name |
| 0, 1, 1, 0, 1, 2, // counts |
| kind); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateEmptyLiteralObject() { |
| return zone()->New<Operator>( // -- |
| IrOpcode::kJSCreateEmptyLiteralObject, // opcode |
| Operator::kNoProperties, // properties |
| "JSCreateEmptyLiteralObject", // name |
| 0, 1, 1, 1, 1, 2); // counts |
| } |
| |
| const Operator* JSOperatorBuilder::CreateLiteralRegExp( |
| Handle<String> constant_pattern, FeedbackSource const& feedback, |
| int literal_flags) { |
| CreateLiteralParameters parameters(constant_pattern, feedback, -1, |
| literal_flags); |
| return zone()->New<Operator1<CreateLiteralParameters>>( // -- |
| IrOpcode::kJSCreateLiteralRegExp, // opcode |
| Operator::kNoProperties, // properties |
| "JSCreateLiteralRegExp", // name |
| 1, 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateFunctionContext( |
| Handle<ScopeInfo> scope_info, int slot_count, ScopeType scope_type) { |
| CreateFunctionContextParameters parameters(scope_info, slot_count, |
| scope_type); |
| return zone()->New<Operator1<CreateFunctionContextParameters>>( // -- |
| IrOpcode::kJSCreateFunctionContext, Operator::kNoProperties, // opcode |
| "JSCreateFunctionContext", // name |
| 0, 1, 1, 1, 1, 2, // counts |
| parameters); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateCatchContext( |
| const Handle<ScopeInfo>& scope_info) { |
| return zone()->New<Operator1<Handle<ScopeInfo>>>( |
| IrOpcode::kJSCreateCatchContext, Operator::kNoProperties, // opcode |
| "JSCreateCatchContext", // name |
| 1, 1, 1, 1, 1, 2, // counts |
| scope_info); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateWithContext( |
| const Handle<ScopeInfo>& scope_info) { |
| return zone()->New<Operator1<Handle<ScopeInfo>>>( |
| IrOpcode::kJSCreateWithContext, Operator::kNoProperties, // opcode |
| "JSCreateWithContext", // name |
| 1, 1, 1, 1, 1, 2, // counts |
| scope_info); // parameter |
| } |
| |
| const Operator* JSOperatorBuilder::CreateBlockContext( |
| const Handle<ScopeInfo>& scope_info) { |
| return zone()->New<Operator1<Handle<ScopeInfo>>>( // -- |
| IrOpcode::kJSCreateBlockContext, Operator::kNoProperties, // opcode |
| "JSCreateBlockContext", // name |
| 0, 1, 1, 1, 1, 2, // counts |
| scope_info); // parameter |
| } |
| |
| Handle<ScopeInfo> ScopeInfoOf(const Operator* op) { |
| DCHECK(IrOpcode::kJSCreateBlockContext == op->opcode() || |
| IrOpcode::kJSCreateWithContext == op->opcode() || |
| IrOpcode::kJSCreateCatchContext == op->opcode()); |
| return OpParameter<Handle<ScopeInfo>>(op); |
| } |
| |
| #undef CACHED_OP_LIST |
| |
| } // namespace compiler |
| } // namespace internal |
| } // namespace v8 |