| Index: sdk/lib/_internal/compiler/implementation/inferrer/inferrer_visitor.dart
|
| diff --git a/sdk/lib/_internal/compiler/implementation/inferrer/inferrer_visitor.dart b/sdk/lib/_internal/compiler/implementation/inferrer/inferrer_visitor.dart
|
| deleted file mode 100644
|
| index 65ea6bc8cf39868dea103e4205248cd2572b5089..0000000000000000000000000000000000000000
|
| --- a/sdk/lib/_internal/compiler/implementation/inferrer/inferrer_visitor.dart
|
| +++ /dev/null
|
| @@ -1,1277 +0,0 @@
|
| -// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
|
| -// for details. All rights reserved. Use of this source code is governed by a
|
| -// BSD-style license that can be found in the LICENSE file.
|
| -
|
| -library inferrer_visitor;
|
| -
|
| -import '../dart2jslib.dart' hide Selector, TypedSelector;
|
| -import '../dart_types.dart';
|
| -import '../elements/elements.dart';
|
| -import '../tree/tree.dart';
|
| -import '../universe/universe.dart';
|
| -import '../util/util.dart';
|
| -import '../types/types.dart' show TypeMask;
|
| -import 'dart:collection' show IterableMixin;
|
| -
|
| -/**
|
| - * The interface [InferrerVisitor] will use when working on types.
|
| - */
|
| -abstract class TypeSystem<T> {
|
| - T get dynamicType;
|
| - T get nullType;
|
| - T get intType;
|
| - T get uint31Type;
|
| - T get uint32Type;
|
| - T get positiveIntType;
|
| - T get doubleType;
|
| - T get numType;
|
| - T get boolType;
|
| - T get functionType;
|
| - T get listType;
|
| - T get constListType;
|
| - T get fixedListType;
|
| - T get growableListType;
|
| - T get mapType;
|
| - T get constMapType;
|
| - T get stringType;
|
| - T get typeType;
|
| -
|
| - T stringLiteralType(DartString value);
|
| -
|
| - T nonNullSubtype(ClassElement type);
|
| - T nonNullSubclass(ClassElement type);
|
| - T nonNullExact(ClassElement type);
|
| - T nonNullEmpty();
|
| - bool isNull(T type);
|
| - Selector newTypedSelector(T receiver, Selector selector);
|
| -
|
| - T allocateList(T type,
|
| - Node node,
|
| - Element enclosing,
|
| - [T elementType, int length]);
|
| -
|
| - T allocateMap(T type, Node node, Element element, [List<T> keyType,
|
| - List<T> valueType]);
|
| -
|
| - T allocateClosure(Node node, Element element);
|
| -
|
| - /**
|
| - * Returns the least upper bound between [firstType] and
|
| - * [secondType].
|
| - */
|
| - T computeLUB(T firstType, T secondType);
|
| -
|
| - /**
|
| - * Returns the intersection between [T] and [annotation].
|
| - * [isNullable] indicates whether the annotation implies a null
|
| - * type.
|
| - */
|
| - T narrowType(T type, DartType annotation, {bool isNullable: true});
|
| -
|
| - /**
|
| - * Returns a new type that unions [firstInput] and [secondInput].
|
| - */
|
| - T allocateDiamondPhi(T firstInput, T secondInput);
|
| -
|
| - /**
|
| - * Returns a new type for holding the potential types of [element].
|
| - * [inputType] is the first incoming type of the phi.
|
| - */
|
| - T allocatePhi(Node node, Local variable, T inputType);
|
| -
|
| -
|
| - /**
|
| - * Returns a new type for holding the potential types of [element].
|
| - * [inputType] is the first incoming type of the phi. [allocateLoopPhi]
|
| - * only differs from [allocatePhi] in that it allows the underlying
|
| - * implementation of [TypeSystem] to differentiate Phi nodes due to loops
|
| - * from other merging uses.
|
| - */
|
| - T allocateLoopPhi(Node node, Local variable, T inputType);
|
| -
|
| - /**
|
| - * Simplies the phi representing [element] and of the type
|
| - * [phiType]. For example, if this phi has one incoming input, an
|
| - * implementation of this method could just return that incoming
|
| - * input type.
|
| - */
|
| - T simplifyPhi(Node node, Local variable, T phiType);
|
| -
|
| - /**
|
| - * Adds [newType] as an input of [phiType].
|
| - */
|
| - T addPhiInput(Local variable, T phiType, T newType);
|
| -
|
| - /**
|
| - * Returns `true` if `selector` should be updated to reflect the new
|
| - * `receiverType`.
|
| - */
|
| - bool selectorNeedsUpdate(T receiverType, Selector selector);
|
| -
|
| - /**
|
| - * Returns a new receiver type for this [selector] applied to
|
| - * [receiverType].
|
| - */
|
| - T refineReceiver(Selector selector, T receiverType);
|
| -
|
| - /**
|
| - * Returns the internal inferrer representation for [mask].
|
| - */
|
| - T getConcreteTypeFor(TypeMask mask);
|
| -}
|
| -
|
| -/**
|
| - * A variable scope holds types for variables. It has a link to a
|
| - * parent scope, but never changes the types in that parent. Instead,
|
| - * updates to locals of a parent scope are put in the current scope.
|
| - * The inferrer makes sure updates get merged into the parent scope,
|
| - * once the control flow block has been visited.
|
| - */
|
| -class VariableScope<T> {
|
| - Map<Local, T> variables;
|
| -
|
| - /// The parent of this scope. Null for the root scope.
|
| - final VariableScope<T> parent;
|
| -
|
| - /// The [Node] that created this scope.
|
| - final Node block;
|
| -
|
| - VariableScope(this.block, [parent])
|
| - : this.variables = null,
|
| - this.parent = parent;
|
| -
|
| - VariableScope.deepCopyOf(VariableScope<T> other)
|
| - : variables = other.variables == null
|
| - ? null
|
| - : new Map<Local, T>.from(other.variables),
|
| - block = other.block,
|
| - parent = other.parent == null
|
| - ? null
|
| - : new VariableScope<T>.deepCopyOf(other.parent);
|
| -
|
| - T operator [](Local variable) {
|
| - T result;
|
| - if (variables == null || (result = variables[variable]) == null) {
|
| - return parent == null ? null : parent[variable];
|
| - }
|
| - return result;
|
| - }
|
| -
|
| - void operator []=(Local variable, T mask) {
|
| - assert(mask != null);
|
| - if (variables == null) {
|
| - variables = new Map<Local, T>();
|
| - }
|
| - variables[variable] = mask;
|
| - }
|
| -
|
| - void forEachOwnLocal(void f(Local variable, T type)) {
|
| - if (variables == null) return;
|
| - variables.forEach(f);
|
| - }
|
| -
|
| - void forEachLocalUntilNode(Node node,
|
| - void f(Local variable, T type),
|
| - [Setlet<Local> seenLocals]) {
|
| - if (seenLocals == null) seenLocals = new Setlet<Local>();
|
| - if (variables != null) {
|
| - variables.forEach((variable, type) {
|
| - if (seenLocals.contains(variable)) return;
|
| - seenLocals.add(variable);
|
| - f(variable, type);
|
| - });
|
| - }
|
| - if (block == node) return;
|
| - if (parent != null) parent.forEachLocalUntilNode(node, f, seenLocals);
|
| - }
|
| -
|
| - void forEachLocal(void f(Local variable, T type)) {
|
| - forEachLocalUntilNode(null, f);
|
| - }
|
| -
|
| - bool updates(Local variable) {
|
| - if (variables == null) return false;
|
| - return variables.containsKey(variable);
|
| - }
|
| -
|
| - String toString() {
|
| - String rest = parent == null ? "null" : parent.toString();
|
| - return '$variables $rest';
|
| - }
|
| -}
|
| -
|
| -class FieldInitializationScope<T> {
|
| - final TypeSystem<T> types;
|
| - Map<Element, T> fields;
|
| - bool isThisExposed;
|
| -
|
| - FieldInitializationScope(this.types) : isThisExposed = false;
|
| -
|
| - FieldInitializationScope.internalFrom(FieldInitializationScope<T> other)
|
| - : types = other.types,
|
| - isThisExposed = other.isThisExposed;
|
| -
|
| - factory FieldInitializationScope.from(FieldInitializationScope<T> other) {
|
| - if (other == null) return null;
|
| - return new FieldInitializationScope<T>.internalFrom(other);
|
| - }
|
| -
|
| - void updateField(Element field, T type) {
|
| - if (isThisExposed) return;
|
| - if (fields == null) fields = new Map<Element, T>();
|
| - fields[field] = type;
|
| - }
|
| -
|
| - T readField(Element field) {
|
| - return fields == null ? null : fields[field];
|
| - }
|
| -
|
| - void forEach(void f(Element element, T type)) {
|
| - if (fields == null) return;
|
| - fields.forEach(f);
|
| - }
|
| -
|
| - void mergeDiamondFlow(FieldInitializationScope<T> thenScope,
|
| - FieldInitializationScope<T> elseScope) {
|
| - // Quick bailout check. If [isThisExposed] is true, we know the
|
| - // code following won't do anything.
|
| - if (isThisExposed) return;
|
| - if (elseScope == null || elseScope.fields == null) {
|
| - elseScope = this;
|
| - }
|
| -
|
| - thenScope.forEach((Element field, T type) {
|
| - T otherType = elseScope.readField(field);
|
| - if (otherType == null) return;
|
| - updateField(field, types.allocateDiamondPhi(type, otherType));
|
| - });
|
| - isThisExposed = thenScope.isThisExposed || elseScope.isThisExposed;
|
| - }
|
| -}
|
| -
|
| -/**
|
| - * Placeholder for inferred arguments types on sends.
|
| - */
|
| -class ArgumentsTypes<T> extends IterableMixin<T> {
|
| - final List<T> positional;
|
| - final Map<String, T> named;
|
| - ArgumentsTypes(this.positional, named)
|
| - : this.named = (named == null || named.isEmpty) ? const {} : named {
|
| - assert(this.positional.every((T type) => type != null));
|
| - assert(this.named.values.every((T type) => type != null));
|
| - }
|
| -
|
| - int get length => positional.length + named.length;
|
| -
|
| - Iterator<T> get iterator => new ArgumentsTypesIterator(this);
|
| -
|
| - String toString() => "{ positional = $positional, named = $named }";
|
| -
|
| - bool operator==(other) {
|
| - if (positional.length != other.positional.length) return false;
|
| - if (named.length != other.named.length) return false;
|
| - for (int i = 0; i < positional.length; i++) {
|
| - if (positional[i] != other.positional[i]) return false;
|
| - }
|
| - named.forEach((name, type) {
|
| - if (other.named[name] != type) return false;
|
| - });
|
| - return true;
|
| - }
|
| -
|
| - int get hashCode => throw new UnsupportedError('ArgumentsTypes.hashCode');
|
| -
|
| - bool hasNoArguments() => positional.isEmpty && named.isEmpty;
|
| -
|
| - bool hasOnePositionalArgumentThatMatches(bool f(T type)) {
|
| - return named.isEmpty && positional.length == 1 && f(positional[0]);
|
| - }
|
| -
|
| - void forEach(void f(T type)) {
|
| - positional.forEach(f);
|
| - named.values.forEach(f);
|
| - }
|
| -
|
| - bool every(bool f(T type)) {
|
| - return positional.every(f) && named.values.every(f);
|
| - }
|
| -
|
| - bool contains(T type) {
|
| - return positional.contains(type) || named.containsValue(type);
|
| - }
|
| -}
|
| -
|
| -class ArgumentsTypesIterator<T> implements Iterator<T> {
|
| - final Iterator<T> positional;
|
| - final Iterator<T> named;
|
| - bool _iteratePositional = true;
|
| -
|
| - ArgumentsTypesIterator(ArgumentsTypes<T> iteratee)
|
| - : positional = iteratee.positional.iterator,
|
| - named = iteratee.named.values.iterator;
|
| -
|
| - Iterator<T> get _currentIterator => _iteratePositional ? positional : named;
|
| -
|
| - T get current => _currentIterator.current;
|
| -
|
| - bool moveNext() {
|
| - if (_iteratePositional && positional.moveNext()) {
|
| - return true;
|
| - }
|
| - _iteratePositional = false;
|
| - return named.moveNext();
|
| - }
|
| -}
|
| -
|
| -
|
| -abstract class MinimalInferrerEngine<T> {
|
| - /**
|
| - * Returns the type of [element].
|
| - */
|
| - T typeOfElement(Element element);
|
| -
|
| - /**
|
| - * Records that [node] sets non-final field [element] to be of type
|
| - * [type].
|
| - */
|
| - void recordTypeOfNonFinalField(Node node, Element field, T type);
|
| -
|
| - /**
|
| - * Records that the captured variable [local] is read.
|
| - */
|
| - void recordCapturedLocalRead(Local local);
|
| -
|
| - /**
|
| - * Records that the variable [local] is being updated.
|
| - */
|
| - void recordLocalUpdate(Local local, T type);
|
| -}
|
| -
|
| -/**
|
| - * Placeholder for inferred types of local variables.
|
| - */
|
| -class LocalsHandler<T> {
|
| - final Compiler compiler;
|
| - final TypeSystem<T> types;
|
| - final MinimalInferrerEngine<T> inferrer;
|
| - final VariableScope<T> locals;
|
| - final Map<Local, Element> captured;
|
| - final Map<Local, Element> capturedAndBoxed;
|
| - final FieldInitializationScope<T> fieldScope;
|
| - LocalsHandler<T> tryBlock;
|
| - bool seenReturnOrThrow = false;
|
| - bool seenBreakOrContinue = false;
|
| -
|
| - bool get aborts {
|
| - return seenReturnOrThrow || seenBreakOrContinue;
|
| - }
|
| - bool get inTryBlock => tryBlock != null;
|
| -
|
| - LocalsHandler(this.inferrer,
|
| - this.types,
|
| - this.compiler,
|
| - Node block,
|
| - [this.fieldScope])
|
| - : locals = new VariableScope<T>(block),
|
| - captured = new Map<Local, Element>(),
|
| - capturedAndBoxed = new Map<Local, Element>(),
|
| - tryBlock = null;
|
| -
|
| - LocalsHandler.from(LocalsHandler<T> other,
|
| - Node block,
|
| - {bool useOtherTryBlock: true})
|
| - : locals = new VariableScope<T>(block, other.locals),
|
| - fieldScope = new FieldInitializationScope<T>.from(other.fieldScope),
|
| - captured = other.captured,
|
| - capturedAndBoxed = other.capturedAndBoxed,
|
| - types = other.types,
|
| - inferrer = other.inferrer,
|
| - compiler = other.compiler {
|
| - tryBlock = useOtherTryBlock ? other.tryBlock : this;
|
| - }
|
| -
|
| - LocalsHandler.deepCopyOf(LocalsHandler<T> other)
|
| - : locals = new VariableScope<T>.deepCopyOf(other.locals),
|
| - fieldScope = new FieldInitializationScope<T>.from(other.fieldScope),
|
| - captured = other.captured,
|
| - capturedAndBoxed = other.capturedAndBoxed,
|
| - tryBlock = other.tryBlock,
|
| - types = other.types,
|
| - inferrer = other.inferrer,
|
| - compiler = other.compiler;
|
| -
|
| - T use(Local local) {
|
| - if (capturedAndBoxed.containsKey(local)) {
|
| - return inferrer.typeOfElement(capturedAndBoxed[local]);
|
| - } else {
|
| - if (captured.containsKey(local)) {
|
| - inferrer.recordCapturedLocalRead(local);
|
| - }
|
| - return locals[local];
|
| - }
|
| - }
|
| -
|
| - void update(LocalElement local, T type, Node node) {
|
| - assert(type != null);
|
| - if (compiler.trustTypeAnnotations || compiler.enableTypeAssertions) {
|
| - type = types.narrowType(type, local.type);
|
| - }
|
| - updateLocal() {
|
| - T currentType = locals[local];
|
| - locals[local] = type;
|
| - if (currentType != type) {
|
| - inferrer.recordLocalUpdate(local, type);
|
| - }
|
| - }
|
| - if (capturedAndBoxed.containsKey(local)) {
|
| - inferrer.recordTypeOfNonFinalField(
|
| - node, capturedAndBoxed[local], type);
|
| - } else if (inTryBlock) {
|
| - // We don't know if an assignment in a try block
|
| - // will be executed, so all assigments in that block are
|
| - // potential types after we have left it. We update the parent
|
| - // of the try block so that, at exit of the try block, we get
|
| - // the right phi for it.
|
| - T existing = tryBlock.locals.parent[local];
|
| - if (existing != null) {
|
| - T phiType = types.allocatePhi(tryBlock.locals.block, local, existing);
|
| - T inputType = types.addPhiInput(local, phiType, type);
|
| - tryBlock.locals.parent[local] = inputType;
|
| - }
|
| - // Update the current handler unconditionnally with the new
|
| - // type.
|
| - updateLocal();
|
| - } else {
|
| - updateLocal();
|
| - }
|
| - }
|
| -
|
| - void setCaptured(Local local, Element field) {
|
| - captured[local] = field;
|
| - }
|
| -
|
| - void setCapturedAndBoxed(Local local, Element field) {
|
| - capturedAndBoxed[local] = field;
|
| - }
|
| -
|
| - void mergeDiamondFlow(LocalsHandler<T> thenBranch,
|
| - LocalsHandler<T> elseBranch) {
|
| - if (fieldScope != null && elseBranch != null) {
|
| - fieldScope.mergeDiamondFlow(thenBranch.fieldScope, elseBranch.fieldScope);
|
| - }
|
| - seenReturnOrThrow = thenBranch.seenReturnOrThrow
|
| - && elseBranch != null
|
| - && elseBranch.seenReturnOrThrow;
|
| - seenBreakOrContinue = thenBranch.seenBreakOrContinue
|
| - && elseBranch != null
|
| - && elseBranch.seenBreakOrContinue;
|
| - if (aborts) return;
|
| -
|
| - void mergeOneBranch(LocalsHandler<T> other) {
|
| - other.locals.forEachOwnLocal((Local local, T type) {
|
| - T myType = locals[local];
|
| - if (myType == null) return; // Variable is only defined in [other].
|
| - if (type == myType) return;
|
| - locals[local] = types.allocateDiamondPhi(myType, type);
|
| - });
|
| - }
|
| -
|
| - void inPlaceUpdateOneBranch(LocalsHandler<T> other) {
|
| - other.locals.forEachOwnLocal((Local local, T type) {
|
| - T myType = locals[local];
|
| - if (myType == null) return; // Variable is only defined in [other].
|
| - if (type == myType) return;
|
| - locals[local] = type;
|
| - });
|
| - }
|
| -
|
| - if (thenBranch.aborts) {
|
| - if (elseBranch == null) return;
|
| - inPlaceUpdateOneBranch(elseBranch);
|
| - } else if (elseBranch == null) {
|
| - mergeOneBranch(thenBranch);
|
| - } else if (elseBranch.aborts) {
|
| - inPlaceUpdateOneBranch(thenBranch);
|
| - } else {
|
| - void mergeLocal(Local local) {
|
| - T myType = locals[local];
|
| - if (myType == null) return;
|
| - T elseType = elseBranch.locals[local];
|
| - T thenType = thenBranch.locals[local];
|
| - if (thenType == elseType) {
|
| - locals[local] = thenType;
|
| - } else {
|
| - locals[local] = types.allocateDiamondPhi(thenType, elseType);
|
| - }
|
| - }
|
| -
|
| - thenBranch.locals.forEachOwnLocal((Local local, _) {
|
| - mergeLocal(local);
|
| - });
|
| - elseBranch.locals.forEachOwnLocal((Local local, _) {
|
| - // Discard locals we already processed when iterating over
|
| - // [thenBranch]'s locals.
|
| - if (!thenBranch.locals.updates(local)) mergeLocal(local);
|
| - });
|
| - }
|
| - }
|
| -
|
| - /**
|
| - * Merge all [LocalsHandler] in [handlers] into [:this:].
|
| - *
|
| - * If [keepOwnLocals] is true, the types of locals in this
|
| - * [LocalsHandler] are being used in the merge. [keepOwnLocals]
|
| - * should be true if this [LocalsHandler], the dominator of
|
| - * all [handlers], also direclty flows into the join point,
|
| - * that is the code after all [handlers]. For example, consider:
|
| - *
|
| - * [: switch (...) {
|
| - * case 1: ...; break;
|
| - * }
|
| - * :]
|
| - *
|
| - * The [LocalsHandler] at entry of the switch also flows into the
|
| - * exit of the switch, because there is no default case. So the
|
| - * types of locals at entry of the switch have to take part to the
|
| - * merge.
|
| - *
|
| - * The above situation is also true for labeled statements like
|
| - *
|
| - * [: L: {
|
| - * if (...) break;
|
| - * ...
|
| - * }
|
| - * :]
|
| - *
|
| - * where [:this:] is the [LocalsHandler] for the paths through the
|
| - * labeled statement that do not break out.
|
| - */
|
| - void mergeAfterBreaks(List<LocalsHandler<T>> handlers,
|
| - {bool keepOwnLocals: true}) {
|
| - Node level = locals.block;
|
| - Set<Local> seenLocals = new Setlet<Local>();
|
| - // If we want to keep the locals, we first merge [this] into itself to
|
| - // create the required Phi nodes.
|
| - if (keepOwnLocals && !seenReturnOrThrow) {
|
| - mergeHandler(this, seenLocals);
|
| - }
|
| - bool allBranchesAbort = true;
|
| - // Merge all other handlers.
|
| - for (LocalsHandler handler in handlers) {
|
| - allBranchesAbort = allBranchesAbort && handler.seenReturnOrThrow;
|
| - mergeHandler(handler, seenLocals);
|
| - }
|
| - // Clean up Phi nodes with single input.
|
| - locals.forEachLocal((Local variable, T type) {
|
| - if (!seenLocals.contains(variable)) return;
|
| - T newType = types.simplifyPhi(level, variable, type);
|
| - if (newType != type) {
|
| - locals[variable] = newType;
|
| - }
|
| - });
|
| - seenReturnOrThrow = allBranchesAbort &&
|
| - (!keepOwnLocals || seenReturnOrThrow);
|
| - }
|
| -
|
| - /**
|
| - * Merge [other] into this handler. Returns whether a local in this
|
| - * has changed. If [seen] is not null, we allocate new Phi nodes
|
| - * unless the local is already present in the set [seen]. This effectively
|
| - * overwrites the current type knowledge in this handler.
|
| - */
|
| - bool mergeHandler(LocalsHandler<T> other, [Set<Local> seen]) {
|
| - if (other.seenReturnOrThrow) return false;
|
| - bool changed = false;
|
| - other.locals.forEachLocalUntilNode(locals.block, (local, otherType) {
|
| - T myType = locals[local];
|
| - if (myType == null) return;
|
| - T newType;
|
| - if (seen != null && !seen.contains(local)) {
|
| - newType = types.allocatePhi(locals.block, local, otherType);
|
| - seen.add(local);
|
| - } else {
|
| - newType = types.addPhiInput(local, myType, otherType);
|
| - }
|
| - if (newType != myType) {
|
| - changed = true;
|
| - locals[local] = newType;
|
| - }
|
| - });
|
| - return changed;
|
| - }
|
| -
|
| - /**
|
| - * Merge all [LocalsHandler] in [handlers] into this handler.
|
| - * Returns whether a local in this handler has changed.
|
| - */
|
| - bool mergeAll(List<LocalsHandler<T>> handlers) {
|
| - bool changed = false;
|
| - assert(!seenReturnOrThrow);
|
| - handlers.forEach((other) {
|
| - changed = mergeHandler(other) || changed;
|
| - });
|
| - return changed;
|
| - }
|
| -
|
| - void startLoop(Node loop) {
|
| - locals.forEachLocal((Local variable, T type) {
|
| - T newType = types.allocateLoopPhi(loop, variable, type);
|
| - if (newType != type) {
|
| - locals[variable] = newType;
|
| - }
|
| - });
|
| - }
|
| -
|
| - void endLoop(Node loop) {
|
| - locals.forEachLocal((Local variable, T type) {
|
| - T newType = types.simplifyPhi(loop, variable, type);
|
| - if (newType != type) {
|
| - locals[variable] = newType;
|
| - }
|
| - });
|
| - }
|
| -
|
| - void updateField(Element element, T type) {
|
| - fieldScope.updateField(element, type);
|
| - }
|
| -}
|
| -
|
| -abstract class InferrerVisitor
|
| - <T, E extends MinimalInferrerEngine<T>> extends ResolvedVisitor<T> {
|
| - final Compiler compiler;
|
| - final AstElement analyzedElement;
|
| - final TypeSystem<T> types;
|
| - final E inferrer;
|
| - final Map<JumpTarget, List<LocalsHandler<T>>> breaksFor =
|
| - new Map<JumpTarget, List<LocalsHandler<T>>>();
|
| - final Map<JumpTarget, List<LocalsHandler>> continuesFor =
|
| - new Map<JumpTarget, List<LocalsHandler<T>>>();
|
| - LocalsHandler<T> locals;
|
| - final List<T> cascadeReceiverStack = new List<T>();
|
| -
|
| - bool accumulateIsChecks = false;
|
| - bool conditionIsSimple = false;
|
| - List<Send> isChecks;
|
| - int loopLevel = 0;
|
| -
|
| - bool get inLoop => loopLevel > 0;
|
| - bool get isThisExposed {
|
| - return analyzedElement.isGenerativeConstructor
|
| - ? locals.fieldScope.isThisExposed
|
| - : true;
|
| - }
|
| - void set isThisExposed(value) {
|
| - if (analyzedElement.isGenerativeConstructor) {
|
| - locals.fieldScope.isThisExposed = value;
|
| - }
|
| - }
|
| -
|
| - InferrerVisitor(AstElement analyzedElement,
|
| - this.inferrer,
|
| - this.types,
|
| - this.compiler,
|
| - [LocalsHandler<T> handler])
|
| - : this.analyzedElement = analyzedElement,
|
| - this.locals = handler,
|
| - super(analyzedElement.resolvedAst.elements) {
|
| - if (handler != null) return;
|
| - Node node = analyzedElement.node;
|
| - FieldInitializationScope<T> fieldScope =
|
| - analyzedElement.isGenerativeConstructor
|
| - ? new FieldInitializationScope<T>(types)
|
| - : null;
|
| - locals = new LocalsHandler<T>(inferrer, types, compiler, node, fieldScope);
|
| - }
|
| -
|
| - T visitSendSet(SendSet node);
|
| -
|
| - T visitSuperSend(Send node);
|
| -
|
| - T visitStaticSend(Send node);
|
| -
|
| - T visitGetterSend(Send node);
|
| -
|
| - T visitClosureSend(Send node);
|
| -
|
| - T visitDynamicSend(Send node);
|
| -
|
| - T visitForIn(ForIn node);
|
| -
|
| - T visitReturn(Return node);
|
| -
|
| - T visitFunctionExpression(FunctionExpression node);
|
| -
|
| - T visitAssert(Send node) {
|
| - if (!compiler.enableUserAssertions) {
|
| - return types.nullType;
|
| - }
|
| - // TODO(johnniwinther): Don't handle assert like a regular static call since
|
| - // it break the selector name check.
|
| - return visitStaticSend(node);
|
| - }
|
| -
|
| - T visitNode(Node node) {
|
| - return node.visitChildren(this);
|
| - }
|
| -
|
| - T visitNewExpression(NewExpression node) {
|
| - return node.send.accept(this);
|
| - }
|
| -
|
| - T visit(Node node) {
|
| - return node == null ? null : node.accept(this);
|
| - }
|
| -
|
| - T visitFunctionDeclaration(FunctionDeclaration node) {
|
| - locals.update(elements[node], types.functionType, node);
|
| - return visit(node.function);
|
| - }
|
| -
|
| - T visitLiteralString(LiteralString node) {
|
| - return types.stringLiteralType(node.dartString);
|
| - }
|
| -
|
| - T visitStringInterpolation(StringInterpolation node) {
|
| - node.visitChildren(this);
|
| - return types.stringType;
|
| - }
|
| -
|
| - T visitStringJuxtaposition(StringJuxtaposition node) {
|
| - node.visitChildren(this);
|
| - return types.stringType;
|
| - }
|
| -
|
| - T visitLiteralBool(LiteralBool node) {
|
| - return types.boolType;
|
| - }
|
| -
|
| - T visitLiteralDouble(LiteralDouble node) {
|
| - ConstantSystem constantSystem = compiler.backend.constantSystem;
|
| - // The JavaScript backend may turn this literal into an integer at
|
| - // runtime.
|
| - return types.getConcreteTypeFor(
|
| - constantSystem.createDouble(node.value).computeMask(compiler));
|
| - }
|
| -
|
| - T visitLiteralInt(LiteralInt node) {
|
| - ConstantSystem constantSystem = compiler.backend.constantSystem;
|
| - // The JavaScript backend may turn this literal into a double at
|
| - // runtime.
|
| - return types.getConcreteTypeFor(
|
| - constantSystem.createInt(node.value).computeMask(compiler));
|
| - }
|
| -
|
| - T visitLiteralList(LiteralList node) {
|
| - node.visitChildren(this);
|
| - return node.isConst ? types.constListType : types.growableListType;
|
| - }
|
| -
|
| - T visitLiteralMap(LiteralMap node) {
|
| - node.visitChildren(this);
|
| - return node.isConst ? types.constMapType : types.mapType;
|
| - }
|
| -
|
| - T visitLiteralNull(LiteralNull node) {
|
| - return types.nullType;
|
| - }
|
| -
|
| - T visitLiteralSymbol(LiteralSymbol node) {
|
| - // TODO(kasperl): We should be able to tell that the type of a literal
|
| - // symbol is always a non-null exact symbol implementation -- not just
|
| - // any non-null subtype of the symbol interface.
|
| - return types.nonNullSubtype(compiler.symbolClass);
|
| - }
|
| -
|
| - T visitTypePrefixSend(Send node) {
|
| - // TODO(johnniwinther): Remove the need for handling this node.
|
| - return types.dynamicType;
|
| - }
|
| -
|
| - T visitTypeLiteralSend(Send node) {
|
| - return types.typeType;
|
| - }
|
| -
|
| - bool isThisOrSuper(Node node) => node.isThis() || node.isSuper();
|
| -
|
| - Element get outermostElement {
|
| - return analyzedElement.outermostEnclosingMemberOrTopLevel.implementation;
|
| - }
|
| -
|
| - T _thisType;
|
| - T get thisType {
|
| - if (_thisType != null) return _thisType;
|
| - ClassElement cls = outermostElement.enclosingClass;
|
| - ClassWorld classWorld = compiler.world;
|
| - if (classWorld.isUsedAsMixin(cls)) {
|
| - return _thisType = types.nonNullSubtype(cls);
|
| - } else {
|
| - return _thisType = types.nonNullSubclass(cls);
|
| - }
|
| - }
|
| -
|
| - T _superType;
|
| - T get superType {
|
| - if (_superType != null) return _superType;
|
| - return _superType = types.nonNullExact(
|
| - outermostElement.enclosingClass.superclass);
|
| - }
|
| -
|
| - T visitIdentifier(Identifier node) {
|
| - if (node.isThis()) {
|
| - return thisType;
|
| - } else if (node.isSuper()) {
|
| - return superType;
|
| - } else {
|
| - Element element = elements[node];
|
| - if (Elements.isLocal(element)) {
|
| - LocalElement local = element;
|
| - return locals.use(local);
|
| - }
|
| - return null;
|
| - }
|
| - }
|
| -
|
| - void potentiallyAddIsCheck(Send node) {
|
| - if (!accumulateIsChecks) return;
|
| - if (!Elements.isLocal(elements[node.receiver])) return;
|
| - isChecks.add(node);
|
| - }
|
| -
|
| - void potentiallyAddNullCheck(Send node, Node receiver) {
|
| - if (!accumulateIsChecks) return;
|
| - if (!Elements.isLocal(elements[receiver])) return;
|
| - isChecks.add(node);
|
| - }
|
| -
|
| - void updateIsChecks(List<Node> tests, {bool usePositive}) {
|
| - void narrow(Element element, DartType type, Node node) {
|
| - if (element is LocalElement) {
|
| - T existing = locals.use(element);
|
| - T newType = types.narrowType(existing, type, isNullable: false);
|
| - locals.update(element, newType, node);
|
| - }
|
| - }
|
| -
|
| - if (tests == null) return;
|
| - for (Send node in tests) {
|
| - if (node.isTypeTest) {
|
| - if (node.isIsNotCheck) {
|
| - if (usePositive) continue;
|
| - } else {
|
| - if (!usePositive) continue;
|
| - }
|
| - DartType type = elements.getType(node.typeAnnotationFromIsCheckOrCast);
|
| - narrow(elements[node.receiver], type, node);
|
| - } else {
|
| - Element receiverElement = elements[node.receiver];
|
| - Element argumentElement = elements[node.arguments.first];
|
| - String operator = node.selector.asOperator().source;
|
| - if ((operator == '==' && usePositive)
|
| - || (operator == '!=' && !usePositive)) {
|
| - // Type the elements as null.
|
| - if (Elements.isLocal(receiverElement)) {
|
| - locals.update(receiverElement, types.nullType, node);
|
| - }
|
| - if (Elements.isLocal(argumentElement)) {
|
| - locals.update(argumentElement, types.nullType, node);
|
| - }
|
| - } else {
|
| - // Narrow the elements to a non-null type.
|
| - DartType objectType = compiler.objectClass.rawType;
|
| - if (Elements.isLocal(receiverElement)) {
|
| - narrow(receiverElement, objectType, node);
|
| - }
|
| - if (Elements.isLocal(argumentElement)) {
|
| - narrow(argumentElement, objectType, node);
|
| - }
|
| - }
|
| - }
|
| - }
|
| - }
|
| -
|
| - T visitOperatorSend(Send node) {
|
| - Operator op = node.selector;
|
| - if ("[]" == op.source) {
|
| - return visitDynamicSend(node);
|
| - } else if ("&&" == op.source) {
|
| - conditionIsSimple = false;
|
| - bool oldAccumulateIsChecks = accumulateIsChecks;
|
| - List<Send> oldIsChecks = isChecks;
|
| - if (!accumulateIsChecks) {
|
| - accumulateIsChecks = true;
|
| - isChecks = <Send>[];
|
| - }
|
| - visit(node.receiver);
|
| - LocalsHandler<T> saved = locals;
|
| - locals = new LocalsHandler<T>.from(locals, node);
|
| - updateIsChecks(isChecks, usePositive: true);
|
| - if (!oldAccumulateIsChecks) {
|
| - accumulateIsChecks = false;
|
| - isChecks = oldIsChecks;
|
| - }
|
| - visit(node.arguments.head);
|
| - saved.mergeDiamondFlow(locals, null);
|
| - locals = saved;
|
| - return types.boolType;
|
| - } else if ("||" == op.source) {
|
| - conditionIsSimple = false;
|
| - List<Send> tests = <Send>[];
|
| - bool isSimple = handleCondition(node.receiver, tests);
|
| - LocalsHandler<T> saved = locals;
|
| - locals = new LocalsHandler<T>.from(locals, node);
|
| - if (isSimple) updateIsChecks(tests, usePositive: false);
|
| - bool oldAccumulateIsChecks = accumulateIsChecks;
|
| - accumulateIsChecks = false;
|
| - visit(node.arguments.head);
|
| - accumulateIsChecks = oldAccumulateIsChecks;
|
| - saved.mergeDiamondFlow(locals, null);
|
| - locals = saved;
|
| - return types.boolType;
|
| - } else if ("!" == op.source) {
|
| - bool oldAccumulateIsChecks = accumulateIsChecks;
|
| - accumulateIsChecks = false;
|
| - node.visitChildren(this);
|
| - accumulateIsChecks = oldAccumulateIsChecks;
|
| - return types.boolType;
|
| - } else if ("is" == op.source) {
|
| - potentiallyAddIsCheck(node);
|
| - node.visitChildren(this);
|
| - return types.boolType;
|
| - } else if ("as" == op.source) {
|
| - T receiverType = visit(node.receiver);
|
| - DartType type = elements.getType(node.arguments.head);
|
| - return types.narrowType(receiverType, type);
|
| - } else if (node.argumentsNode is Prefix) {
|
| - // Unary operator.
|
| - return visitDynamicSend(node);
|
| - } else if ('===' == op.source
|
| - || '!==' == op.source) {
|
| - node.visitChildren(this);
|
| - return types.boolType;
|
| - } else if ('!=' == op.source) {
|
| - visitDynamicSend(node);
|
| - return types.boolType;
|
| - } else {
|
| - // Binary operator.
|
| - return visitDynamicSend(node);
|
| - }
|
| - }
|
| -
|
| - // Because some nodes just visit their children, we may end up
|
| - // visiting a type annotation, that may contain a send in case of a
|
| - // prefixed type. Therefore we explicitly visit the type annotation
|
| - // to avoid confusing the [ResolvedVisitor].
|
| - visitTypeAnnotation(TypeAnnotation node) {}
|
| -
|
| - T visitConditional(Conditional node) {
|
| - List<Send> tests = <Send>[];
|
| - bool simpleCondition = handleCondition(node.condition, tests);
|
| - LocalsHandler<T> saved = locals;
|
| - locals = new LocalsHandler<T>.from(locals, node);
|
| - updateIsChecks(tests, usePositive: true);
|
| - T firstType = visit(node.thenExpression);
|
| - LocalsHandler<T> thenLocals = locals;
|
| - locals = new LocalsHandler<T>.from(saved, node);
|
| - if (simpleCondition) updateIsChecks(tests, usePositive: false);
|
| - T secondType = visit(node.elseExpression);
|
| - saved.mergeDiamondFlow(thenLocals, locals);
|
| - locals = saved;
|
| - T type = types.allocateDiamondPhi(firstType, secondType);
|
| - return type;
|
| - }
|
| -
|
| - T visitVariableDefinitions(VariableDefinitions node) {
|
| - for (Link<Node> link = node.definitions.nodes;
|
| - !link.isEmpty;
|
| - link = link.tail) {
|
| - Node definition = link.head;
|
| - if (definition is Identifier) {
|
| - locals.update(elements[definition], types.nullType, node);
|
| - } else {
|
| - assert(definition.asSendSet() != null);
|
| - visit(definition);
|
| - }
|
| - }
|
| - return null;
|
| - }
|
| -
|
| - bool handleCondition(Node node, List<Send> tests) {
|
| - bool oldConditionIsSimple = conditionIsSimple;
|
| - bool oldAccumulateIsChecks = accumulateIsChecks;
|
| - List<Send> oldIsChecks = isChecks;
|
| - accumulateIsChecks = true;
|
| - conditionIsSimple = true;
|
| - isChecks = tests;
|
| - visit(node);
|
| - bool simpleCondition = conditionIsSimple;
|
| - accumulateIsChecks = oldAccumulateIsChecks;
|
| - isChecks = oldIsChecks;
|
| - conditionIsSimple = oldConditionIsSimple;
|
| - return simpleCondition;
|
| - }
|
| -
|
| - T visitIf(If node) {
|
| - List<Send> tests = <Send>[];
|
| - bool simpleCondition = handleCondition(node.condition, tests);
|
| - LocalsHandler<T> saved = locals;
|
| - locals = new LocalsHandler<T>.from(locals, node);
|
| - updateIsChecks(tests, usePositive: true);
|
| - visit(node.thenPart);
|
| - LocalsHandler<T> thenLocals = locals;
|
| - locals = new LocalsHandler<T>.from(saved, node);
|
| - if (simpleCondition) updateIsChecks(tests, usePositive: false);
|
| - visit(node.elsePart);
|
| - saved.mergeDiamondFlow(thenLocals, locals);
|
| - locals = saved;
|
| - return null;
|
| - }
|
| -
|
| - void setupBreaksAndContinues(JumpTarget element) {
|
| - if (element == null) return;
|
| - if (element.isContinueTarget) continuesFor[element] = <LocalsHandler>[];
|
| - if (element.isBreakTarget) breaksFor[element] = <LocalsHandler>[];
|
| - }
|
| -
|
| - void clearBreaksAndContinues(JumpTarget element) {
|
| - continuesFor.remove(element);
|
| - breaksFor.remove(element);
|
| - }
|
| -
|
| - List<LocalsHandler<T>> getBreaks(JumpTarget element) {
|
| - List<LocalsHandler<T>> list = <LocalsHandler<T>>[locals];
|
| - if (element == null) return list;
|
| - if (!element.isBreakTarget) return list;
|
| - return list..addAll(breaksFor[element]);
|
| - }
|
| -
|
| - List<LocalsHandler<T>> getLoopBackEdges(JumpTarget element) {
|
| - List<LocalsHandler<T>> list = <LocalsHandler<T>>[locals];
|
| - if (element == null) return list;
|
| - if (!element.isContinueTarget) return list;
|
| - return list..addAll(continuesFor[element]);
|
| - }
|
| -
|
| - T handleLoop(Node node, void logic()) {
|
| - loopLevel++;
|
| - bool changed = false;
|
| - JumpTarget target = elements.getTargetDefinition(node);
|
| - LocalsHandler<T> saved = locals;
|
| - saved.startLoop(node);
|
| - do {
|
| - // Setup (and clear in case of multiple iterations of the loop)
|
| - // the lists of breaks and continues seen in the loop.
|
| - setupBreaksAndContinues(target);
|
| - locals = new LocalsHandler<T>.from(saved, node);
|
| - logic();
|
| - changed = saved.mergeAll(getLoopBackEdges(target));
|
| - } while (changed);
|
| - loopLevel--;
|
| - saved.endLoop(node);
|
| - bool keepOwnLocals = node.asDoWhile() == null;
|
| - saved.mergeAfterBreaks(
|
| - getBreaks(target), keepOwnLocals: keepOwnLocals);
|
| - locals = saved;
|
| - clearBreaksAndContinues(target);
|
| - return null;
|
| - }
|
| -
|
| - T visitWhile(While node) {
|
| - return handleLoop(node, () {
|
| - List<Send> tests = <Send>[];
|
| - handleCondition(node.condition, tests);
|
| - updateIsChecks(tests, usePositive: true);
|
| - visit(node.body);
|
| - });
|
| - }
|
| -
|
| - T visitDoWhile(DoWhile node) {
|
| - return handleLoop(node, () {
|
| - visit(node.body);
|
| - List<Send> tests = <Send>[];
|
| - handleCondition(node.condition, tests);
|
| - updateIsChecks(tests, usePositive: true);
|
| - });
|
| - }
|
| -
|
| - T visitFor(For node) {
|
| - visit(node.initializer);
|
| - return handleLoop(node, () {
|
| - List<Send> tests = <Send>[];
|
| - handleCondition(node.condition, tests);
|
| - updateIsChecks(tests, usePositive: true);
|
| - visit(node.body);
|
| - visit(node.update);
|
| - });
|
| - }
|
| -
|
| - T visitTryStatement(TryStatement node) {
|
| - LocalsHandler<T> saved = locals;
|
| - locals = new LocalsHandler<T>.from(
|
| - locals, node, useOtherTryBlock: false);
|
| - visit(node.tryBlock);
|
| - saved.mergeDiamondFlow(locals, null);
|
| - locals = saved;
|
| - for (Node catchBlock in node.catchBlocks) {
|
| - saved = locals;
|
| - locals = new LocalsHandler<T>.from(locals, catchBlock);
|
| - visit(catchBlock);
|
| - saved.mergeDiamondFlow(locals, null);
|
| - locals = saved;
|
| - }
|
| - visit(node.finallyBlock);
|
| - return null;
|
| - }
|
| -
|
| - T visitThrow(Throw node) {
|
| - node.visitChildren(this);
|
| - locals.seenReturnOrThrow = true;
|
| - return types.nonNullEmpty();
|
| - }
|
| -
|
| - T visitCatchBlock(CatchBlock node) {
|
| - Node exception = node.exception;
|
| - if (exception != null) {
|
| - DartType type = elements.getType(node.type);
|
| - T mask = type == null ||
|
| - type.treatAsDynamic ||
|
| - type.isTypeVariable
|
| - ? types.dynamicType
|
| - : types.nonNullSubtype(type.element);
|
| - locals.update(elements[exception], mask, node);
|
| - }
|
| - Node trace = node.trace;
|
| - if (trace != null) {
|
| - locals.update(elements[trace], types.dynamicType, node);
|
| - }
|
| - visit(node.block);
|
| - return null;
|
| - }
|
| -
|
| - T visitParenthesizedExpression(ParenthesizedExpression node) {
|
| - return visit(node.expression);
|
| - }
|
| -
|
| - T visitBlock(Block node) {
|
| - if (node.statements != null) {
|
| - for (Node statement in node.statements) {
|
| - visit(statement);
|
| - if (locals.aborts) break;
|
| - }
|
| - }
|
| - return null;
|
| - }
|
| -
|
| - T visitLabeledStatement(LabeledStatement node) {
|
| - Statement body = node.statement;
|
| - if (body is Loop
|
| - || body is SwitchStatement
|
| - || Elements.isUnusedLabel(node, elements)) {
|
| - // Loops and switches handle their own labels.
|
| - visit(body);
|
| - } else {
|
| - JumpTarget targetElement = elements.getTargetDefinition(body);
|
| - setupBreaksAndContinues(targetElement);
|
| - visit(body);
|
| - locals.mergeAfterBreaks(getBreaks(targetElement));
|
| - clearBreaksAndContinues(targetElement);
|
| - }
|
| - return null;
|
| - }
|
| -
|
| - T visitBreakStatement(BreakStatement node) {
|
| - JumpTarget target = elements.getTargetOf(node);
|
| - locals.seenBreakOrContinue = true;
|
| - // Do a deep-copy of the locals, because the code following the
|
| - // break will change them.
|
| - breaksFor[target].add(new LocalsHandler<T>.deepCopyOf(locals));
|
| - return null;
|
| - }
|
| -
|
| - T visitContinueStatement(ContinueStatement node) {
|
| - JumpTarget target = elements.getTargetOf(node);
|
| - locals.seenBreakOrContinue = true;
|
| - // Do a deep-copy of the locals, because the code following the
|
| - // continue will change them.
|
| - continuesFor[target].add(new LocalsHandler<T>.deepCopyOf(locals));
|
| - return null;
|
| - }
|
| -
|
| - void internalError(String reason, {Node node}) {
|
| - compiler.internalError(node, reason);
|
| - }
|
| -
|
| - T visitSwitchStatement(SwitchStatement node) {
|
| - visit(node.parenthesizedExpression);
|
| -
|
| - setupBreaksAndContinues(elements.getTargetDefinition(node));
|
| - if (Elements.switchStatementHasContinue(node, elements)) {
|
| - void forEachLabeledCase(void action(JumpTarget target)) {
|
| - for (SwitchCase switchCase in node.cases) {
|
| - for (Node labelOrCase in switchCase.labelsAndCases) {
|
| - if (labelOrCase.asLabel() == null) continue;
|
| - LabelDefinition labelElement =
|
| - elements.getLabelDefinition(labelOrCase);
|
| - if (labelElement != null) {
|
| - action(labelElement.target);
|
| - }
|
| - }
|
| - }
|
| - }
|
| -
|
| - forEachLabeledCase((JumpTarget target) {
|
| - setupBreaksAndContinues(target);
|
| - });
|
| -
|
| - // If the switch statement has a continue, we conservatively
|
| - // visit all cases and update [locals] until we have reached a
|
| - // fixed point.
|
| - bool changed;
|
| - locals.startLoop(node);
|
| - do {
|
| - changed = false;
|
| - for (Node switchCase in node.cases) {
|
| - LocalsHandler<T> saved = locals;
|
| - locals = new LocalsHandler<T>.from(locals, switchCase);
|
| - visit(switchCase);
|
| - changed = saved.mergeAll([locals]) || changed;
|
| - locals = saved;
|
| - }
|
| - } while (changed);
|
| - locals.endLoop(node);
|
| -
|
| - forEachLabeledCase((JumpTarget target) {
|
| - clearBreaksAndContinues(target);
|
| - });
|
| - } else {
|
| - LocalsHandler<T> saved = locals;
|
| - List<LocalsHandler<T>> localsToMerge = <LocalsHandler<T>>[];
|
| - bool hasDefaultCase = false;
|
| -
|
| - for (SwitchCase switchCase in node.cases) {
|
| - if (switchCase.isDefaultCase) {
|
| - hasDefaultCase = true;
|
| - }
|
| - locals = new LocalsHandler<T>.from(saved, switchCase);
|
| - visit(switchCase);
|
| - localsToMerge.add(locals);
|
| - }
|
| - saved.mergeAfterBreaks(localsToMerge, keepOwnLocals: !hasDefaultCase);
|
| - locals = saved;
|
| - }
|
| - clearBreaksAndContinues(elements.getTargetDefinition(node));
|
| - return null;
|
| - }
|
| -
|
| - T visitCascadeReceiver(CascadeReceiver node) {
|
| - var type = visit(node.expression);
|
| - cascadeReceiverStack.add(type);
|
| - return type;
|
| - }
|
| -
|
| - T visitCascade(Cascade node) {
|
| - // Ignore the result of the cascade send and return the type of the cascade
|
| - // receiver.
|
| - visit(node.expression);
|
| - return cascadeReceiverStack.removeLast();
|
| - }
|
| -}
|
|
|