blob: dd483eab4aa24e0a62e894d8aa4cd6b4f77b12c9 [file] [log] [blame]
// Copyright (c) 2017, 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 kernel.transformations.precompiler;
import '../ast.dart'
show
DirectMethodInvocation,
DirectPropertyGet,
DirectPropertySet,
Field,
Library,
Member,
MethodInvocation,
Procedure,
Program,
PropertyGet,
PropertySet,
TreeNode;
import '../core_types.dart' show CoreTypes;
import '../class_hierarchy.dart' show ClosedWorldClassHierarchy;
import '../visitor.dart' show Transformer;
/// Performs whole-program transformations for Dart VM precompiler.
/// Assumes strong mode and closed world.
Program transformProgram(CoreTypes coreTypes, Program program) {
new _DevirtualizationTransformer(program).visitProgram(program);
return program;
}
/// Transforms instance method invocations into direct using strong mode
/// types / interface targets and closed-world class hierarchy analysis.
class _DevirtualizationTransformer extends Transformer {
/// Toggles tracing (useful for debugging).
static const _trace = const bool.fromEnvironment('trace.devirtualization');
ClosedWorldClassHierarchy _hierarchy;
_DevirtualizationTransformer(Program program)
: _hierarchy = new ClosedWorldClassHierarchy(program) {}
@override
TreeNode visitLibrary(Library node) {
if (_trace) {
String external = node.isExternal ? " (external)" : "";
print("[devirt] Processing library ${node.name}${external}");
}
return super.visitLibrary(node);
}
@override
TreeNode visitMethodInvocation(MethodInvocation node) {
node = super.visitMethodInvocation(node);
Member target = node.interfaceTarget;
if ((target != null) && (target is! Field)) {
Member singleTarget =
_hierarchy.getSingleTargetForInterfaceInvocation(target);
if ((singleTarget is Procedure) && !singleTarget.isGetter) {
if (_trace) {
print("[devirt] Replacing ${target} with ${singleTarget}");
}
// TODO(dartbug.com/30480): add annotation to check for null
return new DirectMethodInvocation(
node.receiver, singleTarget, node.arguments);
}
}
return node;
}
@override
TreeNode visitPropertyGet(PropertyGet node) {
node = super.visitPropertyGet(node);
Member target = node.interfaceTarget;
if (target != null) {
Member singleTarget =
_hierarchy.getSingleTargetForInterfaceInvocation(target);
if (singleTarget != null) {
if (_trace) {
print("[devirt] Replacing ${target} with ${singleTarget}");
}
// TODO(dartbug.com/30480): add annotation to check for null
return new DirectPropertyGet(node.receiver, singleTarget);
}
}
return node;
}
@override
TreeNode visitPropertySet(PropertySet node) {
node = super.visitPropertySet(node);
Member target = node.interfaceTarget;
if (target != null) {
Member singleTarget = _hierarchy
.getSingleTargetForInterfaceInvocation(target, setter: true);
if (singleTarget != null) {
if (_trace) {
print("[devirt] Replacing ${target} with ${singleTarget}");
}
// TODO(dartbug.com/30480): add annotation to check for null
return new DirectPropertySet(node.receiver, singleTarget, node.value);
}
}
return node;
}
}