// 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/typer.h" #include <iomanip> #include "src/base/flags.h" #include "src/codegen/tick-counter.h" #include "src/compiler/common-operator.h" #include "src/compiler/graph-reducer.h" #include "src/compiler/js-heap-broker.h" #include "src/compiler/js-operator.h" #include "src/compiler/linkage.h" #include "src/compiler/loop-variable-optimizer.h" #include "src/compiler/node-properties.h" #include "src/compiler/node.h" #include "src/compiler/operation-typer.h" #include "src/compiler/simplified-operator.h" #include "src/compiler/type-cache.h" #include "src/init/bootstrapper.h" #include "src/objects/objects-inl.h" namespace v8 { namespace internal { namespace compiler { class Typer::Decorator final : public GraphDecorator { public: explicit Decorator(Typer* typer) : typer_(typer) {} void Decorate(Node* node) final; private: Typer* const typer_; }; Typer::Typer(JSHeapBroker* broker, Flags flags, Graph* graph, TickCounter* tick_counter) : flags_(flags), graph_(graph), decorator_(nullptr), cache_(TypeCache::Get()), broker_(broker), operation_typer_(broker, zone()), tick_counter_(tick_counter) { singleton_false_ = operation_typer_.singleton_false(); singleton_true_ = operation_typer_.singleton_true(); decorator_ = new (zone()) Decorator(this); graph_->AddDecorator(decorator_); } Typer::~Typer() { graph_->RemoveDecorator(decorator_); } class Typer::Visitor : public Reducer { public: explicit Visitor(Typer* typer, LoopVariableOptimizer* induction_vars) : typer_(typer), induction_vars_(induction_vars), weakened_nodes_(typer->zone()) {} const char* reducer_name() const override { return "Typer"; } Reduction Reduce(Node* node) override { if (node->op()->ValueOutputCount() == 0) return NoChange(); return UpdateType(node, TypeNode(node)); } Type TypeNode(Node* node) { switch (node->opcode()) { #define DECLARE_UNARY_CASE(x, ...) \ case IrOpcode::k##x: \ return Type##x(Operand(node, 0)); JS_SIMPLE_UNOP_LIST(DECLARE_UNARY_CASE) SIMPLIFIED_NUMBER_UNOP_LIST(DECLARE_UNARY_CASE) SIMPLIFIED_BIGINT_UNOP_LIST(DECLARE_UNARY_CASE) SIMPLIFIED_SPECULATIVE_NUMBER_UNOP_LIST(DECLARE_UNARY_CASE) SIMPLIFIED_SPECULATIVE_BIGINT_UNOP_LIST(DECLARE_UNARY_CASE) #undef DECLARE_UNARY_CASE #define DECLARE_BINARY_CASE(x, ...) \ case IrOpcode::k##x: \ return Type##x(Operand(node, 0), Operand(node, 1)); JS_SIMPLE_BINOP_LIST(DECLARE_BINARY_CASE) SIMPLIFIED_NUMBER_BINOP_LIST(DECLARE_BINARY_CASE) SIMPLIFIED_BIGINT_BINOP_LIST(DECLARE_BINARY_CASE) SIMPLIFIED_SPECULATIVE_NUMBER_BINOP_LIST(DECLARE_BINARY_CASE) SIMPLIFIED_SPECULATIVE_BIGINT_BINOP_LIST(DECLARE_BINARY_CASE) #undef DECLARE_BINARY_CASE #define DECLARE_OTHER_CASE(x, ...) \ case IrOpcode::k##x: \ return Type##x(node); DECLARE_OTHER_CASE(Start) DECLARE_OTHER_CASE(IfException) COMMON_OP_LIST(DECLARE_OTHER_CASE) SIMPLIFIED_COMPARE_BINOP_LIST(DECLARE_OTHER_CASE) SIMPLIFIED_OTHER_OP_LIST(DECLARE_OTHER_CASE) JS_OBJECT_OP_LIST(DECLARE_OTHER_CASE) JS_CONTEXT_OP_LIST(DECLARE_OTHER_CASE) JS_OTHER_OP_LIST(DECLARE_OTHER_CASE) #undef DECLARE_OTHER_CASE #define DECLARE_IMPOSSIBLE_CASE(x, ...) case IrOpcode::k##x: DECLARE_IMPOSSIBLE_CASE(Loop) DECLARE_IMPOSSIBLE_CASE(Branch) DECLARE_IMPOSSIBLE_CASE(IfTrue) DECLARE_IMPOSSIBLE_CASE(IfFalse) DECLARE_IMPOSSIBLE_CASE(IfSuccess) DECLARE_IMPOSSIBLE_CASE(Switch) DECLARE_IMPOSSIBLE_CASE(IfValue) DECLARE_IMPOSSIBLE_CASE(IfDefault) DECLARE_IMPOSSIBLE_CASE(Merge) DECLARE_IMPOSSIBLE_CASE(Deoptimize) DECLARE_IMPOSSIBLE_CASE(DeoptimizeIf) DECLARE_IMPOSSIBLE_CASE(DeoptimizeUnless) DECLARE_IMPOSSIBLE_CASE(TrapIf) DECLARE_IMPOSSIBLE_CASE(TrapUnless) DECLARE_IMPOSSIBLE_CASE(Return) DECLARE_IMPOSSIBLE_CASE(TailCall) DECLARE_IMPOSSIBLE_CASE(Terminate) DECLARE_IMPOSSIBLE_CASE(Throw) DECLARE_IMPOSSIBLE_CASE(End) SIMPLIFIED_CHANGE_OP_LIST(DECLARE_IMPOSSIBLE_CASE) SIMPLIFIED_CHECKED_OP_LIST(DECLARE_IMPOSSIBLE_CASE) MACHINE_SIMD_OP_LIST(DECLARE_IMPOSSIBLE_CASE) MACHINE_OP_LIST(DECLARE_IMPOSSIBLE_CASE) #undef DECLARE_IMPOSSIBLE_CASE UNREACHABLE(); } } Type TypeConstant(Handle<Object> value); bool InductionVariablePhiTypeIsPrefixedPoint( InductionVariable* induction_var); private: Typer* typer_; LoopVariableOptimizer* induction_vars_; ZoneSet<NodeId> weakened_nodes_; #define DECLARE_METHOD(x, ...) inline Type Type##x(Node* node); DECLARE_METHOD(Start) DECLARE_METHOD(IfException) COMMON_OP_LIST(DECLARE_METHOD) SIMPLIFIED_COMPARE_BINOP_LIST(DECLARE_METHOD) SIMPLIFIED_OTHER_OP_LIST(DECLARE_METHOD) JS_OBJECT_OP_LIST(DECLARE_METHOD) JS_CONTEXT_OP_LIST(DECLARE_METHOD) JS_OTHER_OP_LIST(DECLARE_METHOD) #undef DECLARE_METHOD #define DECLARE_METHOD(x, ...) inline Type Type##x(Type input); JS_SIMPLE_UNOP_LIST(DECLARE_METHOD) #undef DECLARE_METHOD Type TypeOrNone(Node* node) { return NodeProperties::IsTyped(node) ? NodeProperties::GetType(node) : Type::None(); } Type Operand(Node* node, int i) { Node* operand_node = NodeProperties::GetValueInput(node, i); return TypeOrNone(operand_node); } Type Weaken(Node* node, Type current_type, Type previous_type); Zone* zone() { return typer_->zone(); } Graph* graph() { return typer_->graph(); } void SetWeakened(NodeId node_id) { weakened_nodes_.insert(node_id); } bool IsWeakened(NodeId node_id) { return weakened_nodes_.find(node_id) != weakened_nodes_.end(); } using UnaryTyperFun = Type (*)(Type, Typer* t); using BinaryTyperFun = Type (*)(Type, Type, Typer* t); inline Type TypeUnaryOp(Node* node, UnaryTyperFun); inline Type TypeBinaryOp(Node* node, BinaryTyperFun); inline Type TypeUnaryOp(Type input, UnaryTyperFun); inline Type TypeBinaryOp(Type left, Type right, BinaryTyperFun); static Type BinaryNumberOpTyper(Type lhs, Type rhs, Typer* t, BinaryTyperFun f); enum ComparisonOutcomeFlags { kComparisonTrue = 1, kComparisonFalse = 2, kComparisonUndefined = 4 }; using ComparisonOutcome = base::Flags<ComparisonOutcomeFlags>; static ComparisonOutcome Invert(ComparisonOutcome, Typer*); static Type FalsifyUndefined(ComparisonOutcome, Typer*); static Type BitwiseNot(Type, Typer*); static Type Decrement(Type, Typer*); static Type Increment(Type, Typer*); static Type Negate(Type, Typer*); static Type ToPrimitive(Type, Typer*); static Type ToBoolean(Type, Typer*); static Type ToInteger(Type, Typer*); static Type ToLength(Type, Typer*); static Type ToName(Type, Typer*); static Type ToNumber(Type, Typer*); static Type ToNumberConvertBigInt(Type, Typer*); static Type ToNumeric(Type, Typer*); static Type ToObject(Type, Typer*); static Type ToString(Type, Typer*); #define DECLARE_METHOD(Name) \ static Type Name(Type type, Typer* t) { \ return t->operation_typer_.Name(type); \ } SIMPLIFIED_NUMBER_UNOP_LIST(DECLARE_METHOD) SIMPLIFIED_BIGINT_UNOP_LIST(DECLARE_METHOD) SIMPLIFIED_SPECULATIVE_NUMBER_UNOP_LIST(DECLARE_METHOD) SIMPLIFIED_SPECULATIVE_BIGINT_UNOP_LIST(DECLARE_METHOD) #undef DECLARE_METHOD #define DECLARE_METHOD(Name) \ static Type Name(Type lhs, Type rhs, Typer* t) { \ return t->operation_typer_.Name(lhs, rhs); \ } SIMPLIFIED_NUMBER_BINOP_LIST(DECLARE_METHOD) SIMPLIFIED_BIGINT_BINOP_LIST(DECLARE_METHOD) SIMPLIFIED_SPECULATIVE_NUMBER_BINOP_LIST(DECLARE_METHOD) SIMPLIFIED_SPECULATIVE_BIGINT_BINOP_LIST(DECLARE_METHOD) #undef DECLARE_METHOD #define DECLARE_METHOD(Name, ...) \ inline Type Type##Name(Type left, Type right) { \ return TypeBinaryOp(left, right, Name##Typer); \ } JS_SIMPLE_BINOP_LIST(DECLARE_METHOD) #undef DECLARE_METHOD #define DECLARE_METHOD(Name, ...) \ inline Type Type##Name(Type left, Type right) { \ return TypeBinaryOp(left, right, Name); \ } SIMPLIFIED_NUMBER_BINOP_LIST(DECLARE_METHOD) SIMPLIFIED_BIGINT_BINOP_LIST(DECLARE_METHOD) SIMPLIFIED_SPECULATIVE_NUMBER_BINOP_LIST(DECLARE_METHOD) SIMPLIFIED_SPECULATIVE_BIGINT_BINOP_LIST(DECLARE_METHOD) #undef DECLARE_METHOD #define DECLARE_METHOD(Name, ...) \ inline Type Type##Name(Type input) { return TypeUnaryOp(input, Name); } SIMPLIFIED_NUMBER_UNOP_LIST(DECLARE_METHOD) SIMPLIFIED_BIGINT_UNOP_LIST(DECLARE_METHOD) SIMPLIFIED_SPECULATIVE_NUMBER_UNOP_LIST(DECLARE_METHOD) SIMPLIFIED_SPECULATIVE_BIGINT_UNOP_LIST(DECLARE_METHOD) #undef DECLARE_METHOD static Type ObjectIsArrayBufferView(Type, Typer*); static Type ObjectIsBigInt(Type, Typer*); static Type ObjectIsCallable(Type, Typer*); static Type ObjectIsConstructor(Type, Typer*); static Type ObjectIsDetectableCallable(Type, Typer*); static Type ObjectIsMinusZero(Type, Typer*); static Type NumberIsMinusZero(Type, Typer*); static Type ObjectIsNaN(Type, Typer*); static Type NumberIsNaN(Type, Typer*); static Type ObjectIsNonCallable(Type, Typer*); static Type ObjectIsNumber(Type, Typer*); static Type ObjectIsReceiver(Type, Typer*); static Type ObjectIsSmi(Type, Typer*); static Type ObjectIsString(Type, Typer*); static Type ObjectIsSymbol(Type, Typer*); static Type ObjectIsUndetectable(Type, Typer*); static ComparisonOutcome JSCompareTyper(Type, Type, Typer*); static ComparisonOutcome NumberCompareTyper(Type, Type, Typer*); #define DECLARE_METHOD(x, ...) static Type x##Typer(Type, Type, Typer*); JS_SIMPLE_BINOP_LIST(DECLARE_METHOD) #undef DECLARE_METHOD static Type JSCallTyper(Type, Typer*); static Type NumberEqualTyper(Type, Type, Typer*); static Type NumberLessThanTyper(Type, Type, Typer*); static Type NumberLessThanOrEqualTyper(Type, Type, Typer*); static Type ReferenceEqualTyper(Type, Type, Typer*); static Type SameValueTyper(Type, Type, Typer*); static Type SameValueNumbersOnlyTyper(Type, Type, Typer*); static Type StringFromSingleCharCodeTyper(Type, Typer*); static Type StringFromSingleCodePointTyper(Type, Typer*); Reduction UpdateType(Node* node, Type current) { if (NodeProperties::IsTyped(node)) { // Widen the type of a previously typed node. Type previous = NodeProperties::GetType(node); if (node->opcode() == IrOpcode::kPhi || node->opcode() == IrOpcode::kInductionVariablePhi) { // Speed up termination in the presence of range types: current = Weaken(node, current, previous); } if (V8_UNLIKELY(!previous.Is(current))) { AllowHandleDereference allow; std::ostringstream ostream; node->Print(ostream); FATAL("UpdateType error for node %s", ostream.str().c_str()); } NodeProperties::SetType(node, current); if (!current.Is(previous)) { // If something changed, revisit all uses. return Changed(node); } return NoChange(); } else { // No previous type, simply update the type. NodeProperties::SetType(node, current); return Changed(node); } } }; void Typer::Run() { Run(NodeVector(zone()), nullptr); } void Typer::Run(const NodeVector& roots, LoopVariableOptimizer* induction_vars) { if (induction_vars != nullptr) { induction_vars->ChangeToInductionVariablePhis(); } Visitor visitor(this, induction_vars); GraphReducer graph_reducer(zone(), graph(), tick_counter_); graph_reducer.AddReducer(&visitor); for (Node* const root : roots) graph_reducer.ReduceNode(root); graph_reducer.ReduceGraph(); if (induction_vars != nullptr) { // Validate the types computed by TypeInductionVariablePhi. for (auto entry : induction_vars->induction_variables()) { InductionVariable* induction_var = entry.second; if (induction_var->phi()->opcode() == IrOpcode::kInductionVariablePhi) { CHECK(visitor.InductionVariablePhiTypeIsPrefixedPoint(induction_var)); } } induction_vars->ChangeToPhisAndInsertGuards(); } } void Typer::Decorator::Decorate(Node* node) { if (node->op()->ValueOutputCount() > 0) { // Only eagerly type-decorate nodes with known input types. // Other cases will generally require a proper fixpoint iteration with Run. bool is_typed = NodeProperties::IsTyped(node); if (is_typed || NodeProperties::AllValueInputsAreTyped(node)) { Visitor typing(typer_, nullptr); Type type = typing.TypeNode(node); if (is_typed) { type = Type::Intersect(type, NodeProperties::GetType(node), typer_->zone()); } NodeProperties::SetType(node, type); } } } // ----------------------------------------------------------------------------- // Helper functions that lift a function f on types to a function on bounds, // and uses that to type the given node. Note that f is never called with None // as an argument. Type Typer::Visitor::TypeUnaryOp(Node* node, UnaryTyperFun f) { Type input = Operand(node, 0); return TypeUnaryOp(input, f); } Type Typer::Visitor::TypeUnaryOp(Type input, UnaryTyperFun f) { return input.IsNone() ? Type::None() : f(input, typer_); } Type Typer::Visitor::TypeBinaryOp(Node* node, BinaryTyperFun f) { Type left = Operand(node, 0); Type right = Operand(node, 1); return TypeBinaryOp(left, right, f); } Type Typer::Visitor::TypeBinaryOp(Type left, Type right, BinaryTyperFun f) { return left.IsNone() || right.IsNone() ? Type::None() : f(left, right, typer_); } Type Typer::Visitor::BinaryNumberOpTyper(Type lhs, Type rhs, Typer* t, BinaryTyperFun f) { lhs = ToNumeric(lhs, t); rhs = ToNumeric(rhs, t); bool lhs_is_number = lhs.Is(Type::Number()); bool rhs_is_number = rhs.Is(Type::Number()); if (lhs_is_number && rhs_is_number) { return f(lhs, rhs, t); } // In order to maintain monotonicity, the following two conditions are // intentionally asymmetric. if (lhs_is_number) { return Type::Number(); } if (lhs.Is(Type::BigInt())) { return Type::BigInt(); } return Type::Numeric(); } Typer::Visitor::ComparisonOutcome Typer::Visitor::Invert( ComparisonOutcome outcome, Typer* t) { ComparisonOutcome result(0); if ((outcome & kComparisonUndefined) != 0) result |= kComparisonUndefined; if ((outcome & kComparisonTrue) != 0) result |= kComparisonFalse; if ((outcome & kComparisonFalse) != 0) result |= kComparisonTrue; return result; } Type Typer::Visitor::FalsifyUndefined(ComparisonOutcome outcome, Typer* t) { if (outcome == 0) return Type::None(); if ((outcome & kComparisonFalse) != 0 || (outcome & kComparisonUndefined) != 0) { return (outcome & kComparisonTrue) != 0 ? Type::Boolean() : t->singleton_false_; } DCHECK_NE(0, outcome & kComparisonTrue); return t->singleton_true_; } Type Typer::Visitor::BitwiseNot(Type type, Typer* t) { type = ToNumeric(type, t); if (type.Is(Type::Number())) { return NumberBitwiseXor(type, t->cache_->kSingletonMinusOne, t); } return Type::Numeric(); } Type Typer::Visitor::Decrement(Type type, Typer* t) { type = ToNumeric(type, t); if (type.Is(Type::Number())) { return NumberSubtract(type, t->cache_->kSingletonOne, t); } return Type::Numeric(); } Type Typer::Visitor::Increment(Type type, Typer* t) { type = ToNumeric(type, t); if (type.Is(Type::Number())) { return NumberAdd(type, t->cache_->kSingletonOne, t); } return Type::Numeric(); } Type Typer::Visitor::Negate(Type type, Typer* t) { type = ToNumeric(type, t); if (type.Is(Type::Number())) { return NumberMultiply(type, t->cache_->kSingletonMinusOne, t); } return Type::Numeric(); } // Type conversion. Type Typer::Visitor::ToPrimitive(Type type, Typer* t) { if (type.Is(Type::Primitive()) && !type.Maybe(Type::Receiver())) { return type; } return Type::Primitive(); } Type Typer::Visitor::ToBoolean(Type type, Typer* t) { return t->operation_typer()->ToBoolean(type); } // static Type Typer::Visitor::ToInteger(Type type, Typer* t) { // ES6 section 7.1.4 ToInteger ( argument ) type = ToNumber(type, t); if (type.Is(t->cache_->kInteger)) return type; if (type.Is(t->cache_->kIntegerOrMinusZeroOrNaN)) { return Type::Union(Type::Intersect(type, t->cache_->kInteger, t->zone()), t->cache_->kSingletonZero, t->zone()); } return t->cache_->kInteger; } // static Type Typer::Visitor::ToLength(Type type, Typer* t) { // ES6 section 7.1.15 ToLength ( argument ) type = ToInteger(type, t); if (type.IsNone()) return type; double min = type.Min(); double max = type.Max(); if (max <= 0.0) { return Type::Constant(0, t->zone()); } if (min >= kMaxSafeInteger) { return Type::Constant(kMaxSafeInteger, t->zone()); } if (min <= 0.0) min = 0.0; if (max >= kMaxSafeInteger) max = kMaxSafeInteger; return Type::Range(min, max, t->zone()); } // static Type Typer::Visitor::ToName(Type type, Typer* t) { // ES6 section 7.1.14 ToPropertyKey ( argument ) type = ToPrimitive(type, t); if (type.Is(Type::Name())) return type; if (type.Maybe(Type::Symbol())) return Type::Name(); return ToString(type, t); } // static Type Typer::Visitor::ToNumber(Type type, Typer* t) { return t->operation_typer_.ToNumber(type); } // static Type Typer::Visitor::ToNumberConvertBigInt(Type type, Typer* t) { return t->operation_typer_.ToNumberConvertBigInt(type); } // static Type Typer::Visitor::ToNumeric(Type type, Typer* t) { return t->operation_typer_.ToNumeric(type); } // static Type Typer::Visitor::ToObject(Type type, Typer* t) { // ES6 section 7.1.13 ToObject ( argument ) if (type.Is(Type::Receiver())) return type; if (type.Is(Type::Primitive())) return Type::OtherObject(); if (!type.Maybe(Type::OtherUndetectable())) { return Type::DetectableReceiver(); } return Type::Receiver(); } // static Type Typer::Visitor::ToString(Type type, Typer* t) { // ES6 section 7.1.12 ToString ( argument ) type = ToPrimitive(type, t); if (type.Is(Type::String())) return type; return Type::String(); } // Type checks. Type Typer::Visitor::ObjectIsArrayBufferView(Type type, Typer* t) { // TODO(turbofan): Introduce a Type::ArrayBufferView? CHECK(!type.IsNone()); if (!type.Maybe(Type::OtherObject())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsBigInt(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::BigInt())) return t->singleton_true_; if (!type.Maybe(Type::BigInt())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsCallable(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::Callable())) return t->singleton_true_; if (!type.Maybe(Type::Callable())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsConstructor(Type type, Typer* t) { // TODO(turbofan): Introduce a Type::Constructor? CHECK(!type.IsNone()); if (!type.Maybe(Type::Callable())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsDetectableCallable(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::DetectableCallable())) return t->singleton_true_; if (!type.Maybe(Type::DetectableCallable())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsMinusZero(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::MinusZero())) return t->singleton_true_; if (!type.Maybe(Type::MinusZero())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::NumberIsMinusZero(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::MinusZero())) return t->singleton_true_; if (!type.Maybe(Type::MinusZero())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsNaN(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::NaN())) return t->singleton_true_; if (!type.Maybe(Type::NaN())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::NumberIsNaN(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::NaN())) return t->singleton_true_; if (!type.Maybe(Type::NaN())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsNonCallable(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::NonCallable())) return t->singleton_true_; if (!type.Maybe(Type::NonCallable())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsNumber(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::Number())) return t->singleton_true_; if (!type.Maybe(Type::Number())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsReceiver(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::Receiver())) return t->singleton_true_; if (!type.Maybe(Type::Receiver())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsSmi(Type type, Typer* t) { if (!type.Maybe(Type::SignedSmall())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsString(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::String())) return t->singleton_true_; if (!type.Maybe(Type::String())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsSymbol(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::Symbol())) return t->singleton_true_; if (!type.Maybe(Type::Symbol())) return t->singleton_false_; return Type::Boolean(); } Type Typer::Visitor::ObjectIsUndetectable(Type type, Typer* t) { CHECK(!type.IsNone()); if (type.Is(Type::Undetectable())) return t->singleton_true_; if (!type.Maybe(Type::Undetectable())) return t->singleton_false_; return Type::Boolean(); } // ----------------------------------------------------------------------------- // Control operators. Type Typer::Visitor::TypeStart(Node* node) { return Type::Internal(); } Type Typer::Visitor::TypeIfException(Node* node) { return Type::NonInternal(); } // Common operators. Type Typer::Visitor::TypeParameter(Node* node) { Node* const start = node->InputAt(0); DCHECK_EQ(IrOpcode::kStart, start->opcode()); int const parameter_count = start->op()->ValueOutputCount() - 4; DCHECK_LE(1, parameter_count); int const index = ParameterIndexOf(node->op()); if (index == Linkage::kJSCallClosureParamIndex) { return Type::Function(); } else if (index == 0) { if (typer_->flags() & Typer::kThisIsReceiver) { return Type::Receiver(); } else { // Parameter[this] can be the_hole for derived class constructors. return Type::Union(Type::Hole(), Type::NonInternal(), typer_->zone()); } } else if (index == Linkage::GetJSCallNewTargetParamIndex(parameter_count)) { if (typer_->flags() & Typer::kNewTargetIsReceiver) { return Type::Receiver(); } else { return Type::Union(Type::Receiver(), Type::Undefined(), typer_->zone()); } } else if (index == Linkage::GetJSCallArgCountParamIndex(parameter_count)) { return Type::Range(0.0, FixedArray::kMaxLength, typer_->zone()); } else if (index == Linkage::GetJSCallContextParamIndex(parameter_count)) { return Type::OtherInternal(); } return Type::NonInternal(); } Type Typer::Visitor::TypeOsrValue(Node* node) { if (OsrValueIndexOf(node->op()) == Linkage::kOsrContextSpillSlotIndex) { return Type::OtherInternal(); } else { return Type::Any(); } } Type Typer::Visitor::TypeRetain(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeInt32Constant(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeInt64Constant(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeTaggedIndexConstant(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeRelocatableInt32Constant(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeRelocatableInt64Constant(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeFloat32Constant(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeFloat64Constant(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeNumberConstant(Node* node) { double number = OpParameter<double>(node->op()); return Type::Constant(number, zone()); } Type Typer::Visitor::TypeHeapConstant(Node* node) { return TypeConstant(HeapConstantOf(node->op())); } Type Typer::Visitor::TypeCompressedHeapConstant(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeExternalConstant(Node* node) { return Type::ExternalPointer(); } Type Typer::Visitor::TypePointerConstant(Node* node) { return Type::ExternalPointer(); } Type Typer::Visitor::TypeSelect(Node* node) { return Type::Union(Operand(node, 1), Operand(node, 2), zone()); } Type Typer::Visitor::TypePhi(Node* node) { int arity = node->op()->ValueInputCount(); Type type = Operand(node, 0); for (int i = 1; i < arity; ++i) { type = Type::Union(type, Operand(node, i), zone()); } return type; } Type Typer::Visitor::TypeInductionVariablePhi(Node* node) { int arity = NodeProperties::GetControlInput(node)->op()->ControlInputCount(); DCHECK_EQ(IrOpcode::kLoop, NodeProperties::GetControlInput(node)->opcode()); DCHECK_EQ(2, NodeProperties::GetControlInput(node)->InputCount()); Type initial_type = Operand(node, 0); Type increment_type = Operand(node, 2); // Fallback to normal phi typing in a variety of cases: // - when the induction variable is not initially of type Integer, because we // want to work with ranges in the algorithm below. // - when the increment is zero, because in that case normal phi typing will // generally yield a more precise type. // - when the induction variable can become NaN (through addition/subtraction // of opposing infinities), because the code below can't handle that case. if (initial_type.IsNone() || increment_type.Is(typer_->cache_->kSingletonZero) || !initial_type.Is(typer_->cache_->kInteger) || !increment_type.Is(typer_->cache_->kInteger) || increment_type.Min() == -V8_INFINITY || increment_type.Max() == +V8_INFINITY) { // Unfortunately, without baking in the previous type, monotonicity might be // violated because we might not yet have retyped the incrementing operation // even though the increment's type might been already reflected in the // induction variable phi. Type type = NodeProperties::IsTyped(node) ? NodeProperties::GetType(node) : Type::None(); for (int i = 0; i < arity; ++i) { type = Type::Union(type, Operand(node, i), zone()); } return type; } auto res = induction_vars_->induction_variables().find(node->id()); DCHECK_NE(res, induction_vars_->induction_variables().end()); InductionVariable* induction_var = res->second; InductionVariable::ArithmeticType arithmetic_type = induction_var->Type(); double min = -V8_INFINITY; double max = V8_INFINITY; double increment_min; double increment_max; if (arithmetic_type == InductionVariable::ArithmeticType::kAddition) { increment_min = increment_type.Min(); increment_max = increment_type.Max(); } else { DCHECK_EQ(arithmetic_type, InductionVariable::ArithmeticType::kSubtraction); increment_min = -increment_type.Max(); increment_max = -increment_type.Min(); } if (increment_min >= 0) { // Increasing sequence. min = initial_type.Min(); for (auto bound : induction_var->upper_bounds()) { Type bound_type = TypeOrNone(bound.bound); // If the type is not an integer, just skip the bound. if (!bound_type.Is(typer_->cache_->kInteger)) continue; // If the type is not inhabited, then we can take the initial value. if (bound_type.IsNone()) { max = initial_type.Max(); break; } double bound_max = bound_type.Max(); if (bound.kind == InductionVariable::kStrict) { bound_max -= 1; } max = std::min(max, bound_max + increment_max); } // The upper bound must be at least the initial value's upper bound. max = std::max(max, initial_type.Max()); } else if (increment_max <= 0) { // Decreasing sequence. max = initial_type.Max(); for (auto bound : induction_var->lower_bounds()) { Type bound_type = TypeOrNone(bound.bound); // If the type is not an integer, just skip the bound. if (!bound_type.Is(typer_->cache_->kInteger)) continue; // If the type is not inhabited, then we can take the initial value. if (bound_type.IsNone()) { min = initial_type.Min(); break; } double bound_min = bound_type.Min(); if (bound.kind == InductionVariable::kStrict) { bound_min += 1; } min = std::max(min, bound_min + increment_min); } // The lower bound must be at most the initial value's lower bound. min = std::min(min, initial_type.Min()); } else { // If the increment can be both positive and negative, the variable can go // arbitrarily far. Use the maximal range in that case. Note that this may // be less precise than what ordinary typing would produce. min = -V8_INFINITY; max = +V8_INFINITY; } if (FLAG_trace_turbo_loop) { StdoutStream{} << std::setprecision(10) << "Loop (" << NodeProperties::GetControlInput(node)->id() << ") variable bounds in " << (arithmetic_type == InductionVariable::ArithmeticType::kAddition ? "addition" : "subtraction") << " for phi " << node->id() << ": (" << min << ", " << max << ")\n"; } return Type::Range(min, max, typer_->zone()); } bool Typer::Visitor::InductionVariablePhiTypeIsPrefixedPoint( InductionVariable* induction_var) { Node* node = induction_var->phi(); DCHECK_EQ(node->opcode(), IrOpcode::kInductionVariablePhi); Type type = NodeProperties::GetType(node); Type initial_type = Operand(node, 0); Node* arith = node->InputAt(1); Type increment_type = Operand(node, 2); // Intersect {type} with useful bounds. for (auto bound : induction_var->upper_bounds()) { Type bound_type = TypeOrNone(bound.bound); if (!bound_type.Is(typer_->cache_->kInteger)) continue; if (!bound_type.IsNone()) { bound_type = Type::Range( -V8_INFINITY, bound_type.Max() - (bound.kind == InductionVariable::kStrict), zone()); } type = Type::Intersect(type, bound_type, typer_->zone()); } for (auto bound : induction_var->lower_bounds()) { Type bound_type = TypeOrNone(bound.bound); if (!bound_type.Is(typer_->cache_->kInteger)) continue; if (!bound_type.IsNone()) { bound_type = Type::Range( bound_type.Min() + (bound.kind == InductionVariable::kStrict), +V8_INFINITY, typer_->zone()); } type = Type::Intersect(type, bound_type, typer_->zone()); } // Apply ordinary typing to the "increment" operation. // clang-format off switch (arith->opcode()) { #define CASE(x) \ case IrOpcode::k##x: \ type = Type##x(type, increment_type); \ break; CASE(JSAdd) CASE(JSSubtract) CASE(NumberAdd) CASE(NumberSubtract) CASE(SpeculativeNumberAdd) CASE(SpeculativeNumberSubtract) CASE(SpeculativeSafeIntegerAdd) CASE(SpeculativeSafeIntegerSubtract) #undef CASE default: UNREACHABLE(); } // clang-format on type = Type::Union(initial_type, type, typer_->zone()); return type.Is(NodeProperties::GetType(node)); } Type Typer::Visitor::TypeEffectPhi(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeLoopExit(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeLoopExitValue(Node* node) { return Operand(node, 0); } Type Typer::Visitor::TypeLoopExitEffect(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeEnsureWritableFastElements(Node* node) { return Operand(node, 1); } Type Typer::Visitor::TypeMaybeGrowFastElements(Node* node) { return Operand(node, 1); } Type Typer::Visitor::TypeTransitionElementsKind(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeCheckpoint(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeBeginRegion(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeFinishRegion(Node* node) { return Operand(node, 0); } Type Typer::Visitor::TypeFrameState(Node* node) { // TODO(rossberg): Ideally FrameState wouldn't have a value output. return Type::Internal(); } Type Typer::Visitor::TypeStateValues(Node* node) { return Type::Internal(); } Type Typer::Visitor::TypeTypedStateValues(Node* node) { return Type::Internal(); } Type Typer::Visitor::TypeObjectId(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeArgumentsElementsState(Node* node) { return Type::Internal(); } Type Typer::Visitor::TypeArgumentsLengthState(Node* node) { return Type::Internal(); } Type Typer::Visitor::TypeObjectState(Node* node) { return Type::Internal(); } Type Typer::Visitor::TypeTypedObjectState(Node* node) { return Type::Internal(); } Type Typer::Visitor::TypeCall(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeFastApiCall(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeProjection(Node* node) { Type const type = Operand(node, 0); if (type.Is(Type::None())) return Type::None(); int const index = static_cast<int>(ProjectionIndexOf(node->op())); if (type.IsTuple() && index < type.AsTuple()->Arity()) { return type.AsTuple()->Element(index); } return Type::Any(); } Type Typer::Visitor::TypeMapGuard(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeTypeGuard(Node* node) { Type const type = Operand(node, 0); return typer_->operation_typer()->TypeTypeGuard(node->op(), type); } Type Typer::Visitor::TypeFoldConstant(Node* node) { return Operand(node, 0); } Type Typer::Visitor::TypeDead(Node* node) { return Type::None(); } Type Typer::Visitor::TypeDeadValue(Node* node) { return Type::None(); } Type Typer::Visitor::TypeUnreachable(Node* node) { return Type::None(); } Type Typer::Visitor::TypeStaticAssert(Node* node) { UNREACHABLE(); } // JS comparison operators. Type Typer::Visitor::JSEqualTyper(Type lhs, Type rhs, Typer* t) { if (lhs.IsNone() || rhs.IsNone()) return Type::None(); if (lhs.Is(Type::NaN()) || rhs.Is(Type::NaN())) return t->singleton_false_; if (lhs.Is(Type::NullOrUndefined()) && rhs.Is(Type::NullOrUndefined())) { return t->singleton_true_; } if (lhs.Is(Type::Number()) && rhs.Is(Type::Number()) && (lhs.Max() < rhs.Min() || lhs.Min() > rhs.Max())) { return t->singleton_false_; } if (lhs.IsSingleton() && rhs.Is(lhs)) { // Types are equal and are inhabited only by a single semantic value, // which is not nan due to the earlier check. DCHECK(lhs.Is(rhs)); return t->singleton_true_; } return Type::Boolean(); } Type Typer::Visitor::JSStrictEqualTyper(Type lhs, Type rhs, Typer* t) { return t->operation_typer()->StrictEqual(lhs, rhs); } // The EcmaScript specification defines the four relational comparison operators // (<, <=, >=, >) with the help of a single abstract one. It behaves like < // but returns undefined when the inputs cannot be compared. // We implement the typing analogously. Typer::Visitor::ComparisonOutcome Typer::Visitor::JSCompareTyper(Type lhs, Type rhs, Typer* t) { lhs = ToPrimitive(lhs, t); rhs = ToPrimitive(rhs, t); if (lhs.Maybe(Type::String()) && rhs.Maybe(Type::String())) { return ComparisonOutcome(kComparisonTrue) | ComparisonOutcome(kComparisonFalse); } lhs = ToNumeric(lhs, t); rhs = ToNumeric(rhs, t); if (lhs.Is(Type::Number()) && rhs.Is(Type::Number())) { return NumberCompareTyper(lhs, rhs, t); } return ComparisonOutcome(kComparisonTrue) | ComparisonOutcome(kComparisonFalse) | ComparisonOutcome(kComparisonUndefined); } Typer::Visitor::ComparisonOutcome Typer::Visitor::NumberCompareTyper(Type lhs, Type rhs, Typer* t) { DCHECK(lhs.Is(Type::Number())); DCHECK(rhs.Is(Type::Number())); if (lhs.IsNone() || rhs.IsNone()) return {}; // Shortcut for NaNs. if (lhs.Is(Type::NaN()) || rhs.Is(Type::NaN())) return kComparisonUndefined; ComparisonOutcome result; if (lhs.IsHeapConstant() && rhs.Is(lhs)) { // Types are equal and are inhabited only by a single semantic value. result = kComparisonFalse; } else if (lhs.Min() >= rhs.Max()) { result = kComparisonFalse; } else if (lhs.Max() < rhs.Min()) { result = kComparisonTrue; } else { return ComparisonOutcome(kComparisonTrue) | ComparisonOutcome(kComparisonFalse) | ComparisonOutcome(kComparisonUndefined); } // Add the undefined if we could see NaN. if (lhs.Maybe(Type::NaN()) || rhs.Maybe(Type::NaN())) { result |= kComparisonUndefined; } return result; } Type Typer::Visitor::JSLessThanTyper(Type lhs, Type rhs, Typer* t) { return FalsifyUndefined(JSCompareTyper(lhs, rhs, t), t); } Type Typer::Visitor::JSGreaterThanTyper(Type lhs, Type rhs, Typer* t) { return FalsifyUndefined(JSCompareTyper(rhs, lhs, t), t); } Type Typer::Visitor::JSLessThanOrEqualTyper(Type lhs, Type rhs, Typer* t) { return FalsifyUndefined(Invert(JSCompareTyper(rhs, lhs, t), t), t); } Type Typer::Visitor::JSGreaterThanOrEqualTyper(Type lhs, Type rhs, Typer* t) { return FalsifyUndefined(Invert(JSCompareTyper(lhs, rhs, t), t), t); } // JS bitwise operators. Type Typer::Visitor::JSBitwiseOrTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberBitwiseOr); } Type Typer::Visitor::JSBitwiseAndTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberBitwiseAnd); } Type Typer::Visitor::JSBitwiseXorTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberBitwiseXor); } Type Typer::Visitor::JSShiftLeftTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberShiftLeft); } Type Typer::Visitor::JSShiftRightTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberShiftRight); } Type Typer::Visitor::JSShiftRightLogicalTyper(Type lhs, Type rhs, Typer* t) { return NumberShiftRightLogical(ToNumber(lhs, t), ToNumber(rhs, t), t); } // JS arithmetic operators. Type Typer::Visitor::JSAddTyper(Type lhs, Type rhs, Typer* t) { lhs = ToPrimitive(lhs, t); rhs = ToPrimitive(rhs, t); if (lhs.Maybe(Type::String()) || rhs.Maybe(Type::String())) { if (lhs.Is(Type::String()) || rhs.Is(Type::String())) { return Type::String(); } else { return Type::NumericOrString(); } } // The addition must be numeric. return BinaryNumberOpTyper(lhs, rhs, t, NumberAdd); } Type Typer::Visitor::JSSubtractTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberSubtract); } Type Typer::Visitor::JSMultiplyTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberMultiply); } Type Typer::Visitor::JSDivideTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberDivide); } Type Typer::Visitor::JSModulusTyper(Type lhs, Type rhs, Typer* t) { return BinaryNumberOpTyper(lhs, rhs, t, NumberModulus); } Type Typer::Visitor::JSExponentiateTyper(Type lhs, Type rhs, Typer* t) { // TODO(neis): Refine using BinaryNumberOpTyper? return Type::Numeric(); } // JS unary operators. #define DEFINE_METHOD(Name) \ Type Typer::Visitor::TypeJS##Name(Type input) { \ return TypeUnaryOp(input, Name); \ } DEFINE_METHOD(BitwiseNot) DEFINE_METHOD(Decrement) DEFINE_METHOD(Increment) DEFINE_METHOD(Negate) DEFINE_METHOD(ToLength) DEFINE_METHOD(ToName) DEFINE_METHOD(ToNumber) DEFINE_METHOD(ToNumberConvertBigInt) DEFINE_METHOD(ToNumeric) DEFINE_METHOD(ToObject) DEFINE_METHOD(ToString) #undef DEFINE_METHOD Type Typer::Visitor::TypeTypeOf(Node* node) { return Type::InternalizedString(); } // JS conversion operators. Type Typer::Visitor::TypeToBoolean(Node* node) { return TypeUnaryOp(node, ToBoolean); } // JS object operators. Type Typer::Visitor::TypeJSCreate(Node* node) { return Type::Object(); } Type Typer::Visitor::TypeJSCreateArguments(Node* node) { switch (CreateArgumentsTypeOf(node->op())) { case CreateArgumentsType::kRestParameter: return Type::Array(); case CreateArgumentsType::kMappedArguments: case CreateArgumentsType::kUnmappedArguments: return Type::OtherObject(); } UNREACHABLE(); } Type Typer::Visitor::TypeJSCreateArray(Node* node) { return Type::Array(); } Type Typer::Visitor::TypeJSCreateArrayIterator(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateAsyncFunctionObject(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateCollectionIterator(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateBoundFunction(Node* node) { return Type::BoundFunction(); } Type Typer::Visitor::TypeJSCreateGeneratorObject(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateClosure(Node* node) { return Type::Function(); } Type Typer::Visitor::TypeJSCreateIterResultObject(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateStringIterator(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateKeyValueArray(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateObject(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreatePromise(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateTypedArray(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateLiteralArray(Node* node) { return Type::Array(); } Type Typer::Visitor::TypeJSCreateEmptyLiteralArray(Node* node) { return Type::Array(); } Type Typer::Visitor::TypeJSCreateArrayFromIterable(Node* node) { return Type::Array(); } Type Typer::Visitor::TypeJSCreateLiteralObject(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateEmptyLiteralObject(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCloneObject(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSCreateLiteralRegExp(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSGetTemplateObject(Node* node) { return Type::Array(); } Type Typer::Visitor::TypeJSLoadProperty(Node* node) { return Type::NonInternal(); } Type Typer::Visitor::TypeJSLoadNamed(Node* node) { return Type::NonInternal(); } Type Typer::Visitor::TypeJSLoadGlobal(Node* node) { return Type::NonInternal(); } Type Typer::Visitor::TypeJSParseInt(Type input) { return Type::Number(); } Type Typer::Visitor::TypeJSRegExpTest(Node* node) { return Type::Boolean(); } // Returns a somewhat larger range if we previously assigned // a (smaller) range to this node. This is used to speed up // the fixpoint calculation in case there appears to be a loop // in the graph. In the current implementation, we are // increasing the limits to the closest power of two. Type Typer::Visitor::Weaken(Node* node, Type current_type, Type previous_type) { static const double kWeakenMinLimits[] = { 0.0, -1073741824.0, -2147483648.0, -4294967296.0, -8589934592.0, -17179869184.0, -34359738368.0, -68719476736.0, -137438953472.0, -274877906944.0, -549755813888.0, -1099511627776.0, -2199023255552.0, -4398046511104.0, -8796093022208.0, -17592186044416.0, -35184372088832.0, -70368744177664.0, -140737488355328.0, -281474976710656.0, -562949953421312.0}; static const double kWeakenMaxLimits[] = { 0.0, 1073741823.0, 2147483647.0, 4294967295.0, 8589934591.0, 17179869183.0, 34359738367.0, 68719476735.0, 137438953471.0, 274877906943.0, 549755813887.0, 1099511627775.0, 2199023255551.0, 4398046511103.0, 8796093022207.0, 17592186044415.0, 35184372088831.0, 70368744177663.0, 140737488355327.0, 281474976710655.0, 562949953421311.0}; STATIC_ASSERT(arraysize(kWeakenMinLimits) == arraysize(kWeakenMaxLimits)); // If the types have nothing to do with integers, return the types. Type const integer = typer_->cache_->kInteger; if (!previous_type.Maybe(integer)) { return current_type; } DCHECK(current_type.Maybe(integer)); Type current_integer = Type::Intersect(current_type, integer, zone()); Type previous_integer = Type::Intersect(previous_type, integer, zone()); // Once we start weakening a node, we should always weaken. if (!IsWeakened(node->id())) { // Only weaken if there is range involved; we should converge quickly // for all other types (the exception is a union of many constants, // but we currently do not increase the number of constants in unions). Type previous = previous_integer.GetRange(); Type current = current_integer.GetRange(); if (current.IsInvalid() || previous.IsInvalid()) { return current_type; } // Range is involved => we are weakening. SetWeakened(node->id()); } double current_min = current_integer.Min(); double new_min = current_min; // Find the closest lower entry in the list of allowed // minima (or negative infinity if there is no such entry). if (current_min != previous_integer.Min()) { new_min = -V8_INFINITY; for (double const min : kWeakenMinLimits) { if (min <= current_min) { new_min = min; break; } } } double current_max = current_integer.Max(); double new_max = current_max; // Find the closest greater entry in the list of allowed // maxima (or infinity if there is no such entry). if (current_max != previous_integer.Max()) { new_max = V8_INFINITY; for (double const max : kWeakenMaxLimits) { if (max >= current_max) { new_max = max; break; } } } return Type::Union(current_type, Type::Range(new_min, new_max, typer_->zone()), typer_->zone()); } Type Typer::Visitor::TypeJSStoreProperty(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSStoreNamed(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSStoreGlobal(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSStoreNamedOwn(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSStoreDataPropertyInLiteral(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSStoreInArrayLiteral(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSDeleteProperty(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeJSHasProperty(Node* node) { return Type::Boolean(); } // JS instanceof operator. Type Typer::Visitor::JSHasInPrototypeChainTyper(Type lhs, Type rhs, Typer* t) { return Type::Boolean(); } Type Typer::Visitor::JSInstanceOfTyper(Type lhs, Type rhs, Typer* t) { return Type::Boolean(); } Type Typer::Visitor::JSOrdinaryHasInstanceTyper(Type lhs, Type rhs, Typer* t) { return Type::Boolean(); } Type Typer::Visitor::TypeJSGetSuperConstructor(Node* node) { return Type::Callable(); } // JS context operators. Type Typer::Visitor::TypeJSHasContextExtension(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeJSLoadContext(Node* node) { ContextAccess const& access = ContextAccessOf(node->op()); switch (access.index()) { case Context::PREVIOUS_INDEX: case Context::SCOPE_INFO_INDEX: return Type::OtherInternal(); default: return Type::Any(); } } Type Typer::Visitor::TypeJSStoreContext(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSCreateFunctionContext(Node* node) { return Type::OtherInternal(); } Type Typer::Visitor::TypeJSCreateCatchContext(Node* node) { return Type::OtherInternal(); } Type Typer::Visitor::TypeJSCreateWithContext(Node* node) { return Type::OtherInternal(); } Type Typer::Visitor::TypeJSCreateBlockContext(Node* node) { return Type::OtherInternal(); } // JS other operators. Type Typer::Visitor::TypeJSConstructForwardVarargs(Node* node) { return Type::Receiver(); } Type Typer::Visitor::TypeJSConstruct(Node* node) { return Type::Receiver(); } Type Typer::Visitor::TypeJSConstructWithArrayLike(Node* node) { return Type::Receiver(); } Type Typer::Visitor::TypeJSConstructWithSpread(Node* node) { return Type::Receiver(); } Type Typer::Visitor::TypeJSObjectIsArray(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeDateNow(Node* node) { return Type::Number(); } Type Typer::Visitor::JSCallTyper(Type fun, Typer* t) { if (!fun.IsHeapConstant() || !fun.AsHeapConstant()->Ref().IsJSFunction()) { return Type::NonInternal(); } JSFunctionRef function = fun.AsHeapConstant()->Ref().AsJSFunction(); if (!function.serialized()) { TRACE_BROKER_MISSING(t->broker(), "data for function " << function); return Type::NonInternal(); } if (!function.shared().HasBuiltinId()) { return Type::NonInternal(); } switch (function.shared().builtin_id()) { case Builtins::kMathRandom: return Type::PlainNumber(); case Builtins::kMathFloor: case Builtins::kMathCeil: case Builtins::kMathRound: case Builtins::kMathTrunc: return t->cache_->kIntegerOrMinusZeroOrNaN; // Unary math functions. case Builtins::kMathAbs: case Builtins::kMathExp: return Type::Union(Type::PlainNumber(), Type::NaN(), t->zone()); case Builtins::kMathAcos: case Builtins::kMathAcosh: case Builtins::kMathAsin: case Builtins::kMathAsinh: case Builtins::kMathAtan: case Builtins::kMathAtanh: case Builtins::kMathCbrt: case Builtins::kMathCos: case Builtins::kMathExpm1: case Builtins::kMathFround: case Builtins::kMathLog: case Builtins::kMathLog1p: case Builtins::kMathLog10: case Builtins::kMathLog2: case Builtins::kMathSin: case Builtins::kMathSqrt: case Builtins::kMathTan: return Type::Number(); case Builtins::kMathSign: return t->cache_->kMinusOneToOneOrMinusZeroOrNaN; // Binary math functions. case Builtins::kMathAtan2: case Builtins::kMathPow: case Builtins::kMathMax: case Builtins::kMathMin: case Builtins::kMathHypot: return Type::Number(); case Builtins::kMathImul: return Type::Signed32(); case Builtins::kMathClz32: return t->cache_->kZeroToThirtyTwo; // Date functions. case Builtins::kDateNow: return t->cache_->kTimeValueType; case Builtins::kDatePrototypeGetDate: return t->cache_->kJSDateDayType; case Builtins::kDatePrototypeGetDay: return t->cache_->kJSDateWeekdayType; case Builtins::kDatePrototypeGetFullYear: return t->cache_->kJSDateYearType; case Builtins::kDatePrototypeGetHours: return t->cache_->kJSDateHourType; case Builtins::kDatePrototypeGetMilliseconds: return Type::Union(Type::Range(0.0, 999.0, t->zone()), Type::NaN(), t->zone()); case Builtins::kDatePrototypeGetMinutes: return t->cache_->kJSDateMinuteType; case Builtins::kDatePrototypeGetMonth: return t->cache_->kJSDateMonthType; case Builtins::kDatePrototypeGetSeconds: return t->cache_->kJSDateSecondType; case Builtins::kDatePrototypeGetTime: return t->cache_->kJSDateValueType; // Symbol functions. case Builtins::kSymbolConstructor: return Type::Symbol(); case Builtins::kSymbolPrototypeToString: return Type::String(); case Builtins::kSymbolPrototypeValueOf: return Type::Symbol(); // BigInt functions. case Builtins::kBigIntConstructor: return Type::BigInt(); // Number functions. case Builtins::kNumberConstructor: return Type::Number(); case Builtins::kNumberIsFinite: case Builtins::kNumberIsInteger: case Builtins::kNumberIsNaN: case Builtins::kNumberIsSafeInteger: return Type::Boolean(); case Builtins::kNumberParseFloat: return Type::Number(); case Builtins::kNumberParseInt: return t->cache_->kIntegerOrMinusZeroOrNaN; case Builtins::kNumberToString: return Type::String(); // String functions. case Builtins::kStringConstructor: return Type::String(); case Builtins::kStringPrototypeCharCodeAt: return Type::Union(Type::Range(0, kMaxUInt16, t->zone()), Type::NaN(), t->zone()); case Builtins::kStringCharAt: return Type::String(); case Builtins::kStringPrototypeCodePointAt: return Type::Union(Type::Range(0.0, String::kMaxCodePoint, t->zone()), Type::Undefined(), t->zone()); case Builtins::kStringPrototypeConcat: case Builtins::kStringFromCharCode: case Builtins::kStringFromCodePoint: return Type::String(); case Builtins::kStringPrototypeIndexOf: case Builtins::kStringPrototypeLastIndexOf: return Type::Range(-1.0, String::kMaxLength, t->zone()); case Builtins::kStringPrototypeEndsWith: case Builtins::kStringPrototypeIncludes: return Type::Boolean(); case Builtins::kStringRaw: case Builtins::kStringRepeat: case Builtins::kStringPrototypeSlice: return Type::String(); case Builtins::kStringPrototypeStartsWith: return Type::Boolean(); case Builtins::kStringPrototypeSubstr: case Builtins::kStringSubstring: case Builtins::kStringPrototypeToString: #ifdef V8_INTL_SUPPORT case Builtins::kStringPrototypeToLowerCaseIntl: case Builtins::kStringPrototypeToUpperCaseIntl: #else case Builtins::kStringPrototypeToLowerCase: case Builtins::kStringPrototypeToUpperCase: #endif case Builtins::kStringPrototypeTrim: case Builtins::kStringPrototypeTrimEnd: case Builtins::kStringPrototypeTrimStart: case Builtins::kStringPrototypeValueOf: return Type::String(); case Builtins::kStringPrototypeIterator: case Builtins::kStringIteratorPrototypeNext: return Type::OtherObject(); case Builtins::kArrayPrototypeEntries: case Builtins::kArrayPrototypeKeys: case Builtins::kArrayPrototypeValues: case Builtins::kTypedArrayPrototypeEntries: case Builtins::kTypedArrayPrototypeKeys: case Builtins::kTypedArrayPrototypeValues: case Builtins::kArrayIteratorPrototypeNext: case Builtins::kMapIteratorPrototypeNext: case Builtins::kSetIteratorPrototypeNext: return Type::OtherObject(); case Builtins::kTypedArrayPrototypeToStringTag: return Type::Union(Type::InternalizedString(), Type::Undefined(), t->zone()); // Array functions. case Builtins::kArrayIsArray: return Type::Boolean(); case Builtins::kArrayConcat: return Type::Receiver(); case Builtins::kArrayEvery: return Type::Boolean(); case Builtins::kArrayPrototypeFill: case Builtins::kArrayFilter: return Type::Receiver(); case Builtins::kArrayPrototypeFindIndex: return Type::Range(-1, kMaxSafeInteger, t->zone()); case Builtins::kArrayForEach: return Type::Undefined(); case Builtins::kArrayIncludes: return Type::Boolean(); case Builtins::kArrayIndexOf: return Type::Range(-1, kMaxSafeInteger, t->zone()); case Builtins::kArrayPrototypeJoin: return Type::String(); case Builtins::kArrayPrototypeLastIndexOf: return Type::Range(-1, kMaxSafeInteger, t->zone()); case Builtins::kArrayMap: return Type::Receiver(); case Builtins::kArrayPush: return t->cache_->kPositiveSafeInteger; case Builtins::kArrayPrototypeReverse: case Builtins::kArrayPrototypeSlice: return Type::Receiver(); case Builtins::kArraySome: return Type::Boolean(); case Builtins::kArrayPrototypeSplice: return Type::Receiver(); case Builtins::kArrayUnshift: return t->cache_->kPositiveSafeInteger; // ArrayBuffer functions. case Builtins::kArrayBufferIsView: return Type::Boolean(); // Object functions. case Builtins::kObjectAssign: return Type::Receiver(); case Builtins::kObjectCreate: return Type::OtherObject(); case Builtins::kObjectIs: case Builtins::kObjectPrototypeHasOwnProperty: case Builtins::kObjectPrototypeIsPrototypeOf: return Type::Boolean(); case Builtins::kObjectToString: return Type::String(); case Builtins::kPromiseAll: return Type::Receiver(); case Builtins::kPromisePrototypeThen: return Type::Receiver(); case Builtins::kPromiseRace: return Type::Receiver(); case Builtins::kPromiseReject: return Type::Receiver(); case Builtins::kPromiseResolveTrampoline: return Type::Receiver(); // RegExp functions. case Builtins::kRegExpPrototypeCompile: return Type::OtherObject(); case Builtins::kRegExpPrototypeExec: return Type::Union(Type::Array(), Type::Null(), t->zone()); case Builtins::kRegExpPrototypeTest: return Type::Boolean(); case Builtins::kRegExpPrototypeToString: return Type::String(); // Function functions. case Builtins::kFunctionPrototypeBind: return Type::BoundFunction(); case Builtins::kFunctionPrototypeHasInstance: return Type::Boolean(); // Global functions. case Builtins::kGlobalDecodeURI: case Builtins::kGlobalDecodeURIComponent: case Builtins::kGlobalEncodeURI: case Builtins::kGlobalEncodeURIComponent: case Builtins::kGlobalEscape: case Builtins::kGlobalUnescape: return Type::String(); case Builtins::kGlobalIsFinite: case Builtins::kGlobalIsNaN: return Type::Boolean(); // Map functions. case Builtins::kMapPrototypeClear: case Builtins::kMapPrototypeForEach: return Type::Undefined(); case Builtins::kMapPrototypeDelete: case Builtins::kMapPrototypeHas: return Type::Boolean(); case Builtins::kMapPrototypeEntries: case Builtins::kMapPrototypeKeys: case Builtins::kMapPrototypeSet: case Builtins::kMapPrototypeValues: return Type::OtherObject(); // Set functions. case Builtins::kSetPrototypeAdd: case Builtins::kSetPrototypeEntries: case Builtins::kSetPrototypeValues: return Type::OtherObject(); case Builtins::kSetPrototypeClear: case Builtins::kSetPrototypeForEach: return Type::Undefined(); case Builtins::kSetPrototypeDelete: case Builtins::kSetPrototypeHas: return Type::Boolean(); // WeakMap functions. case Builtins::kWeakMapPrototypeDelete: case Builtins::kWeakMapPrototypeHas: return Type::Boolean(); case Builtins::kWeakMapPrototypeSet: return Type::OtherObject(); // WeakSet functions. case Builtins::kWeakSetPrototypeAdd: return Type::OtherObject(); case Builtins::kWeakSetPrototypeDelete: case Builtins::kWeakSetPrototypeHas: return Type::Boolean(); default: return Type::NonInternal(); } } Type Typer::Visitor::TypeJSCallForwardVarargs(Node* node) { return TypeUnaryOp(node, JSCallTyper); } Type Typer::Visitor::TypeJSCall(Node* node) { // TODO(bmeurer): We could infer better types if we wouldn't ignore the // argument types for the JSCallTyper above. return TypeUnaryOp(node, JSCallTyper); } Type Typer::Visitor::TypeJSCallWithArrayLike(Node* node) { return TypeUnaryOp(node, JSCallTyper); } Type Typer::Visitor::TypeJSCallWithSpread(Node* node) { return TypeUnaryOp(node, JSCallTyper); } Type Typer::Visitor::TypeJSCallRuntime(Node* node) { switch (CallRuntimeParametersOf(node->op()).id()) { case Runtime::kInlineIsJSReceiver: return TypeUnaryOp(node, ObjectIsReceiver); case Runtime::kInlineIsSmi: return TypeUnaryOp(node, ObjectIsSmi); case Runtime::kInlineIsArray: case Runtime::kInlineIsRegExp: return Type::Boolean(); case Runtime::kInlineCreateIterResultObject: return Type::OtherObject(); case Runtime::kInlineToLength: return TypeUnaryOp(node, ToLength); case Runtime::kInlineToNumber: return TypeUnaryOp(node, ToNumber); case Runtime::kInlineToObject: return TypeUnaryOp(node, ToObject); case Runtime::kInlineToStringRT: return TypeUnaryOp(node, ToString); case Runtime::kHasInPrototypeChain: return Type::Boolean(); default: break; } // TODO(turbofan): This should be Type::NonInternal(), but unfortunately we // have a few weird runtime calls that return the hole or even FixedArrays; // change this once those weird runtime calls have been removed. return Type::Any(); } Type Typer::Visitor::TypeJSForInEnumerate(Node* node) { return Type::OtherInternal(); } Type Typer::Visitor::TypeJSForInNext(Node* node) { return Type::Union(Type::String(), Type::Undefined(), zone()); } Type Typer::Visitor::TypeJSForInPrepare(Node* node) { STATIC_ASSERT(Map::Bits3::EnumLengthBits::kMax <= FixedArray::kMaxLength); Type const cache_type = Type::Union(Type::SignedSmall(), Type::OtherInternal(), zone()); Type const cache_array = Type::OtherInternal(); Type const cache_length = typer_->cache_->kFixedArrayLengthType; return Type::Tuple(cache_type, cache_array, cache_length, zone()); } Type Typer::Visitor::TypeJSLoadMessage(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeJSStoreMessage(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSLoadModule(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeJSStoreModule(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSGeneratorStore(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeJSGeneratorRestoreContinuation(Node* node) { return Type::SignedSmall(); } Type Typer::Visitor::TypeJSGeneratorRestoreContext(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeJSGeneratorRestoreRegister(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeJSGeneratorRestoreInputOrDebugPos(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeJSStackCheck(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeJSDebugger(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeJSAsyncFunctionEnter(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSAsyncFunctionReject(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSAsyncFunctionResolve(Node* node) { return Type::OtherObject(); } Type Typer::Visitor::TypeJSFulfillPromise(Node* node) { return Type::Undefined(); } Type Typer::Visitor::TypeJSPerformPromiseThen(Node* node) { return Type::Receiver(); } Type Typer::Visitor::TypeJSPromiseResolve(Node* node) { return Type::Receiver(); } Type Typer::Visitor::TypeJSRejectPromise(Node* node) { return Type::Undefined(); } Type Typer::Visitor::TypeJSResolvePromise(Node* node) { return Type::Undefined(); } // Simplified operators. Type Typer::Visitor::TypeBooleanNot(Node* node) { return Type::Boolean(); } // static Type Typer::Visitor::NumberEqualTyper(Type lhs, Type rhs, Typer* t) { return JSEqualTyper(ToNumber(lhs, t), ToNumber(rhs, t), t); } // static Type Typer::Visitor::NumberLessThanTyper(Type lhs, Type rhs, Typer* t) { return FalsifyUndefined( NumberCompareTyper(ToNumber(lhs, t), ToNumber(rhs, t), t), t); } // static Type Typer::Visitor::NumberLessThanOrEqualTyper(Type lhs, Type rhs, Typer* t) { return FalsifyUndefined( Invert(JSCompareTyper(ToNumber(rhs, t), ToNumber(lhs, t), t), t), t); } Type Typer::Visitor::TypeNumberEqual(Node* node) { return TypeBinaryOp(node, NumberEqualTyper); } Type Typer::Visitor::TypeNumberLessThan(Node* node) { return TypeBinaryOp(node, NumberLessThanTyper); } Type Typer::Visitor::TypeNumberLessThanOrEqual(Node* node) { return TypeBinaryOp(node, NumberLessThanOrEqualTyper); } Type Typer::Visitor::TypeSpeculativeNumberEqual(Node* node) { return TypeBinaryOp(node, NumberEqualTyper); } Type Typer::Visitor::TypeSpeculativeNumberLessThan(Node* node) { return TypeBinaryOp(node, NumberLessThanTyper); } Type Typer::Visitor::TypeSpeculativeNumberLessThanOrEqual(Node* node) { return TypeBinaryOp(node, NumberLessThanOrEqualTyper); } Type Typer::Visitor::TypeStringConcat(Node* node) { return Type::String(); } Type Typer::Visitor::TypeStringToNumber(Node* node) { return TypeUnaryOp(node, ToNumber); } Type Typer::Visitor::TypePlainPrimitiveToNumber(Node* node) { return TypeUnaryOp(node, ToNumber); } Type Typer::Visitor::TypePlainPrimitiveToWord32(Node* node) { return Type::Integral32(); } Type Typer::Visitor::TypePlainPrimitiveToFloat64(Node* node) { return Type::Number(); } // static Type Typer::Visitor::ReferenceEqualTyper(Type lhs, Type rhs, Typer* t) { if (lhs.IsHeapConstant() && rhs.Is(lhs)) { return t->singleton_true_; } return Type::Boolean(); } Type Typer::Visitor::TypeReferenceEqual(Node* node) { return TypeBinaryOp(node, ReferenceEqualTyper); } // static Type Typer::Visitor::SameValueTyper(Type lhs, Type rhs, Typer* t) { return t->operation_typer()->SameValue(lhs, rhs); } // static Type Typer::Visitor::SameValueNumbersOnlyTyper(Type lhs, Type rhs, Typer* t) { return t->operation_typer()->SameValueNumbersOnly(lhs, rhs); } Type Typer::Visitor::TypeSameValue(Node* node) { return TypeBinaryOp(node, SameValueTyper); } Type Typer::Visitor::TypeSameValueNumbersOnly(Node* node) { return TypeBinaryOp(node, SameValueNumbersOnlyTyper); } Type Typer::Visitor::TypeNumberSameValue(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeStringEqual(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeStringLessThan(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeStringLessThanOrEqual(Node* node) { return Type::Boolean(); } Type Typer::Visitor::StringFromSingleCharCodeTyper(Type type, Typer* t) { return Type::String(); } Type Typer::Visitor::StringFromSingleCodePointTyper(Type type, Typer* t) { return Type::String(); } Type Typer::Visitor::TypeStringToLowerCaseIntl(Node* node) { return Type::String(); } Type Typer::Visitor::TypeStringToUpperCaseIntl(Node* node) { return Type::String(); } Type Typer::Visitor::TypeStringCharCodeAt(Node* node) { return typer_->cache_->kUint16; } Type Typer::Visitor::TypeStringCodePointAt(Node* node) { return Type::Range(0.0, String::kMaxCodePoint, zone()); } Type Typer::Visitor::TypeStringFromSingleCharCode(Node* node) { return TypeUnaryOp(node, StringFromSingleCharCodeTyper); } Type Typer::Visitor::TypeStringFromSingleCodePoint(Node* node) { return TypeUnaryOp(node, StringFromSingleCodePointTyper); } Type Typer::Visitor::TypeStringFromCodePointAt(Node* node) { return Type::String(); } Type Typer::Visitor::TypeStringIndexOf(Node* node) { return Type::Range(-1.0, String::kMaxLength, zone()); } Type Typer::Visitor::TypeStringLength(Node* node) { return typer_->cache_->kStringLengthType; } Type Typer::Visitor::TypeStringSubstring(Node* node) { return Type::String(); } Type Typer::Visitor::TypePoisonIndex(Node* node) { return Type::Union(Operand(node, 0), typer_->cache_->kSingletonZero, zone()); } Type Typer::Visitor::TypeCheckBounds(Node* node) { return typer_->operation_typer_.CheckBounds(Operand(node, 0), Operand(node, 1)); } Type Typer::Visitor::TypeCheckHeapObject(Node* node) { Type type = Operand(node, 0); return type; } Type Typer::Visitor::TypeCheckIf(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeCheckInternalizedString(Node* node) { Type arg = Operand(node, 0); return Type::Intersect(arg, Type::InternalizedString(), zone()); } Type Typer::Visitor::TypeCheckMaps(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeCompareMaps(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeCheckNumber(Node* node) { return typer_->operation_typer_.CheckNumber(Operand(node, 0)); } Type Typer::Visitor::TypeCheckReceiver(Node* node) { Type arg = Operand(node, 0); return Type::Intersect(arg, Type::Receiver(), zone()); } Type Typer::Visitor::TypeCheckReceiverOrNullOrUndefined(Node* node) { Type arg = Operand(node, 0); return Type::Intersect(arg, Type::ReceiverOrNullOrUndefined(), zone()); } Type Typer::Visitor::TypeCheckSmi(Node* node) { Type arg = Operand(node, 0); return Type::Intersect(arg, Type::SignedSmall(), zone()); } Type Typer::Visitor::TypeCheckString(Node* node) { Type arg = Operand(node, 0); return Type::Intersect(arg, Type::String(), zone()); } Type Typer::Visitor::TypeCheckSymbol(Node* node) { Type arg = Operand(node, 0); return Type::Intersect(arg, Type::Symbol(), zone()); } Type Typer::Visitor::TypeCheckFloat64Hole(Node* node) { return typer_->operation_typer_.CheckFloat64Hole(Operand(node, 0)); } Type Typer::Visitor::TypeCheckNotTaggedHole(Node* node) { Type type = Operand(node, 0); type = Type::Intersect(type, Type::NonInternal(), zone()); return type; } Type Typer::Visitor::TypeCheckClosure(Node* node) { return Type::Function(); } Type Typer::Visitor::TypeConvertReceiver(Node* node) { Type arg = Operand(node, 0); return typer_->operation_typer_.ConvertReceiver(arg); } Type Typer::Visitor::TypeConvertTaggedHoleToUndefined(Node* node) { Type type = Operand(node, 0); return typer_->operation_typer()->ConvertTaggedHoleToUndefined(type); } Type Typer::Visitor::TypeCheckEqualsInternalizedString(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeCheckEqualsSymbol(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeAllocate(Node* node) { return AllocateTypeOf(node->op()); } Type Typer::Visitor::TypeAllocateRaw(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeLoadFieldByIndex(Node* node) { return Type::NonInternal(); } Type Typer::Visitor::TypeLoadField(Node* node) { return FieldAccessOf(node->op()).type; } Type Typer::Visitor::TypeLoadMessage(Node* node) { return Type::Any(); } Type Typer::Visitor::TypeLoadElement(Node* node) { return ElementAccessOf(node->op()).type; } Type Typer::Visitor::TypeLoadStackArgument(Node* node) { return Type::NonInternal(); } Type Typer::Visitor::TypeLoadFromObject(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeLoadTypedElement(Node* node) { switch (ExternalArrayTypeOf(node->op())) { #define TYPED_ARRAY_CASE(ElemType, type, TYPE, ctype) \ case kExternal##ElemType##Array: \ return typer_->cache_->k##ElemType; TYPED_ARRAYS(TYPED_ARRAY_CASE) #undef TYPED_ARRAY_CASE } UNREACHABLE(); } Type Typer::Visitor::TypeLoadDataViewElement(Node* node) { switch (ExternalArrayTypeOf(node->op())) { #define TYPED_ARRAY_CASE(ElemType, type, TYPE, ctype) \ case kExternal##ElemType##Array: \ return typer_->cache_->k##ElemType; TYPED_ARRAYS(TYPED_ARRAY_CASE) #undef TYPED_ARRAY_CASE } UNREACHABLE(); } Type Typer::Visitor::TypeStoreField(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeStoreMessage(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeStoreElement(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeStoreToObject(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeTransitionAndStoreElement(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeTransitionAndStoreNumberElement(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeTransitionAndStoreNonNumberElement(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeStoreSignedSmallElement(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeStoreTypedElement(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeStoreDataViewElement(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeObjectIsArrayBufferView(Node* node) { return TypeUnaryOp(node, ObjectIsArrayBufferView); } Type Typer::Visitor::TypeObjectIsBigInt(Node* node) { return TypeUnaryOp(node, ObjectIsBigInt); } Type Typer::Visitor::TypeObjectIsCallable(Node* node) { return TypeUnaryOp(node, ObjectIsCallable); } Type Typer::Visitor::TypeObjectIsConstructor(Node* node) { return TypeUnaryOp(node, ObjectIsConstructor); } Type Typer::Visitor::TypeObjectIsDetectableCallable(Node* node) { return TypeUnaryOp(node, ObjectIsDetectableCallable); } Type Typer::Visitor::TypeObjectIsMinusZero(Node* node) { return TypeUnaryOp(node, ObjectIsMinusZero); } Type Typer::Visitor::TypeNumberIsMinusZero(Node* node) { return TypeUnaryOp(node, NumberIsMinusZero); } Type Typer::Visitor::TypeNumberIsFloat64Hole(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeNumberIsFinite(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeObjectIsFiniteNumber(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeNumberIsInteger(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeObjectIsSafeInteger(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeNumberIsSafeInteger(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeObjectIsInteger(Node* node) { return Type::Boolean(); } Type Typer::Visitor::TypeObjectIsNaN(Node* node) { return TypeUnaryOp(node, ObjectIsNaN); } Type Typer::Visitor::TypeNumberIsNaN(Node* node) { return TypeUnaryOp(node, NumberIsNaN); } Type Typer::Visitor::TypeObjectIsNonCallable(Node* node) { return TypeUnaryOp(node, ObjectIsNonCallable); } Type Typer::Visitor::TypeObjectIsNumber(Node* node) { return TypeUnaryOp(node, ObjectIsNumber); } Type Typer::Visitor::TypeObjectIsReceiver(Node* node) { return TypeUnaryOp(node, ObjectIsReceiver); } Type Typer::Visitor::TypeObjectIsSmi(Node* node) { return TypeUnaryOp(node, ObjectIsSmi); } Type Typer::Visitor::TypeObjectIsString(Node* node) { return TypeUnaryOp(node, ObjectIsString); } Type Typer::Visitor::TypeObjectIsSymbol(Node* node) { return TypeUnaryOp(node, ObjectIsSymbol); } Type Typer::Visitor::TypeObjectIsUndetectable(Node* node) { return TypeUnaryOp(node, ObjectIsUndetectable); } Type Typer::Visitor::TypeArgumentsLength(Node* node) { return TypeCache::Get()->kArgumentsLengthType; } Type Typer::Visitor::TypeArgumentsFrame(Node* node) { return Type::ExternalPointer(); } Type Typer::Visitor::TypeNewDoubleElements(Node* node) { return Type::OtherInternal(); } Type Typer::Visitor::TypeNewSmiOrObjectElements(Node* node) { return Type::OtherInternal(); } Type Typer::Visitor::TypeNewArgumentsElements(Node* node) { return Type::OtherInternal(); } Type Typer::Visitor::TypeNewConsString(Node* node) { return Type::String(); } Type Typer::Visitor::TypeDelayedStringConstant(Node* node) { return Type::String(); } Type Typer::Visitor::TypeFindOrderedHashMapEntry(Node* node) { return Type::Range(-1.0, FixedArray::kMaxLength, zone()); } Type Typer::Visitor::TypeFindOrderedHashMapEntryForInt32Key(Node* node) { return Type::Range(-1.0, FixedArray::kMaxLength, zone()); } Type Typer::Visitor::TypeRuntimeAbort(Node* node) { UNREACHABLE(); } Type Typer::Visitor::TypeAssertType(Node* node) { UNREACHABLE(); } // Heap constants. Type Typer::Visitor::TypeConstant(Handle<Object> value) { return Type::Constant(typer_->broker(), value, zone()); } Type Typer::Visitor::TypeJSGetIterator(Node* node) { return Type::Any(); } } // namespace compiler } // namespace internal } // namespace v8