blob: cd76db6f4ff669ebec32a7b7fffc3b65da6a9f9e [file] [log] [blame]
// 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 compiler.src.inferrer.closure_tracer;
import '../common/names.dart' show Names;
import '../elements/elements.dart';
import '../js_backend/backend_helpers.dart';
import '../types/types.dart' show TypeMask;
import '../universe/selector.dart' show Selector;
import 'debug.dart' as debug;
import 'inferrer_engine.dart';
import 'node_tracer.dart';
import 'type_graph_nodes.dart';
class ClosureTracerVisitor extends TracerVisitor {
final Iterable<FunctionElement> tracedElements;
final List<CallSiteTypeInformation> _callsToAnalyze =
new List<CallSiteTypeInformation>();
ClosureTracerVisitor(this.tracedElements, ApplyableTypeInformation tracedType,
InferrerEngine inferrer)
: super(tracedType, inferrer);
ApplyableTypeInformation get tracedType => super.tracedType;
void run() {
analyze();
if (!continueAnalyzing) return;
_callsToAnalyze.forEach(_analyzeCall);
for (FunctionElement e in tracedElements) {
e.functionSignature.forEachParameter((Element parameter) {
ElementTypeInformation info =
inferrer.types.getInferredTypeOf(parameter);
info.disableInferenceForClosures = false;
});
}
}
void _tagAsFunctionApplyTarget([String reason]) {
tracedType.mightBePassedToFunctionApply = true;
if (debug.VERBOSE) {
print("Closure $tracedType might be passed to apply: $reason");
}
}
void _registerCallForLaterAnalysis(CallSiteTypeInformation info) {
_callsToAnalyze.add(info);
}
void _analyzeCall(CallSiteTypeInformation info) {
Selector selector = info.selector;
TypeMask mask = info.mask;
tracedElements.forEach((FunctionElement functionElement) {
if (!selector.callStructure.signatureApplies(functionElement.type)) {
return;
}
inferrer.updateParameterAssignments(
info, functionElement, info.arguments, selector, mask,
remove: false, addToQueue: false);
});
}
@override
visitClosureCallSiteTypeInformation(ClosureCallSiteTypeInformation info) {
super.visitClosureCallSiteTypeInformation(info);
if (info.closure == currentUser) {
_registerCallForLaterAnalysis(info);
} else {
bailout('Passed to a closure');
}
}
@override
visitStaticCallSiteTypeInformation(StaticCallSiteTypeInformation info) {
super.visitStaticCallSiteTypeInformation(info);
Element called = info.calledElement;
if (compiler.backend.isForeign(called)) {
String name = called.name;
if (name == BackendHelpers.JS || name == 'DART_CLOSURE_TO_JS') {
bailout('Used in JS ${info.call}');
}
}
if (called.isGetter &&
info.selector != null &&
info.selector.isCall &&
inferrer.types.getInferredTypeOf(called) == currentUser) {
// This node can be a closure call as well. For example, `foo()`
// where `foo` is a getter.
_registerCallForLaterAnalysis(info);
}
if (_checkIfFunctionApply(called) &&
info.arguments != null &&
info.arguments.contains(currentUser)) {
_tagAsFunctionApplyTarget("static call");
}
}
bool _checkIfCurrentUser(element) =>
inferrer.types.getInferredTypeOf(element) == currentUser;
bool _checkIfFunctionApply(Element element) {
return element is MemberElement &&
compiler.commonElements.isFunctionApplyMethod(element);
}
@override
visitDynamicCallSiteTypeInformation(DynamicCallSiteTypeInformation info) {
super.visitDynamicCallSiteTypeInformation(info);
if (info.selector.isCall) {
if (info.arguments.contains(currentUser)) {
if (!info.targets.every((element) => element.isFunction)) {
bailout('Passed to a closure');
}
if (info.targets.any(_checkIfFunctionApply)) {
_tagAsFunctionApplyTarget("dynamic call");
}
} else if (info.targets.any((element) => _checkIfCurrentUser(element))) {
_registerCallForLaterAnalysis(info);
}
} else if (info.selector.isGetter &&
info.selector.memberName == Names.call) {
// We are potentially tearing off ourself here
addNewEscapeInformation(info);
}
}
}
class StaticTearOffClosureTracerVisitor extends ClosureTracerVisitor {
StaticTearOffClosureTracerVisitor(tracedElement, tracedType, inferrer)
: super([tracedElement], tracedType, inferrer);
@override
visitStaticCallSiteTypeInformation(StaticCallSiteTypeInformation info) {
super.visitStaticCallSiteTypeInformation(info);
if (info.calledElement == tracedElements.first &&
info.selector != null &&
info.selector.isGetter) {
addNewEscapeInformation(info);
}
}
}