Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(244)

Unified Diff: sdk/lib/_internal/compiler/implementation/inferrer/inferrer_visitor.dart

Issue 694353007: Move dart2js from sdk/lib/_internal/compiler to pkg/compiler (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 6 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
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();
- }
-}

Powered by Google App Engine
This is Rietveld 408576698