blob: b18da0edbc15a8dee501359d72a642489e7d948a [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.
import 'dart:io';
import 'package:async_helper/async_helper.dart';
import 'package:compiler/src/commandline_options.dart';
import 'package:compiler/src/common.dart';
import 'package:compiler/src/compiler.dart';
import 'package:compiler/src/diagnostics/diagnostic_listener.dart';
import 'package:compiler/src/elements/elements.dart';
import 'package:compiler/src/elements/entities.dart';
import 'package:compiler/src/elements/jumps.dart';
import 'package:compiler/src/js_model/locals.dart';
import 'package:compiler/src/kernel/element_map.dart';
import 'package:compiler/src/kernel/kernel_backend_strategy.dart';
import 'package:compiler/src/tree/nodes.dart' as ast;
import '../equivalence/id_equivalence.dart';
import '../equivalence/id_equivalence_helper.dart';
import 'package:kernel/ast.dart' as ir;
main(List<String> args) {
asyncTest(() async {
Directory dataDir = new Directory.fromUri(Platform.script.resolve('data'));
await checkTests(dataDir, computeJumpsData, computeKernelJumpsData,
options: [Flags.disableTypeInference, stopAfterTypeInference],
args: args);
});
}
/// Compute closure data mapping for [_member] as a [MemberElement].
///
/// Fills [actualMap] with the data and [sourceSpanMap] with the source spans
/// for the data origin.
void computeJumpsData(
Compiler compiler, MemberEntity _member, Map<Id, ActualData> actualMap,
{bool verbose: false}) {
MemberElement member = _member;
new JumpsAstComputer(compiler.reporter, actualMap, member.resolvedAst).run();
}
/// Compute closure data mapping for [member] as a kernel based element.
///
/// Fills [actualMap] with the data and [sourceSpanMap] with the source spans
/// for the data origin.
void computeKernelJumpsData(
Compiler compiler, MemberEntity member, Map<Id, ActualData> actualMap,
{bool verbose: false}) {
KernelBackendStrategy backendStrategy = compiler.backendStrategy;
KernelToElementMapForBuilding elementMap = backendStrategy.elementMap;
GlobalLocalsMap localsMap = backendStrategy.globalLocalsMapForTesting;
MemberDefinition definition = elementMap.getMemberDefinition(member);
new JumpsIrChecker(
compiler.reporter, actualMap, localsMap.getLocalsMap(member))
.run(definition.node);
}
class TargetData {
final int index;
final NodeId id;
final SourceSpan sourceSpan;
final JumpTarget target;
TargetData(this.index, this.id, this.sourceSpan, this.target);
String toString() => 'TargetData(index=$index,id=$id,'
'sourceSpan=$sourceSpan,target=$target)';
}
class GotoData {
final NodeId id;
final SourceSpan sourceSpan;
final JumpTarget target;
GotoData(this.id, this.sourceSpan, this.target);
String toString() => 'GotoData(id=$id,sourceSpan=$sourceSpan,target=$target)';
}
abstract class JumpsMixin {
int index = 0;
Map<JumpTarget, TargetData> targets = <JumpTarget, TargetData>{};
List<GotoData> gotos = <GotoData>[];
void registerValue(SourceSpan sourceSpan, Id id, String value, Object object);
void processData() {
targets.forEach((JumpTarget target, TargetData data) {
StringBuffer sb = new StringBuffer();
sb.write(data.index);
sb.write('@');
bool needsComma = false;
if (target.isBreakTarget) {
sb.write('break');
needsComma = true;
}
if (target.isContinueTarget) {
if (needsComma) {
sb.write(',');
}
sb.write('continue');
needsComma = true;
}
String value = sb.toString();
registerValue(data.sourceSpan, data.id, value, target);
});
gotos.forEach((GotoData data) {
StringBuffer sb = new StringBuffer();
sb.write('target=');
TargetData targetData = targets[data.target];
assert(targetData != null, "No TargetData for ${data.target}");
sb.write(targetData.index);
String value = sb.toString();
registerValue(data.sourceSpan, data.id, value, data);
});
}
}
/// Ast visitor for computing jump data.
class JumpsAstComputer extends AstDataExtractor with JumpsMixin {
JumpsAstComputer(DiagnosticReporter reporter, Map<Id, ActualData> actualMap,
ResolvedAst resolvedAst)
: super(reporter, actualMap, resolvedAst);
void run() {
super.run();
processData();
}
@override
String computeNodeValue(Id id, ast.Node node, [AstElement element]) {
// Node values are computed post-visit in [processData].
return null;
}
@override
String computeElementValue(Id id, AstElement element) {
return null;
}
@override
visitLoop(ast.Loop node) {
JumpTarget target = elements.getTargetDefinition(node);
if (target != null) {
NodeId id = createLoopId(node);
SourceSpan sourceSpan = computeSourceSpan(node);
targets[target] = new TargetData(index++, id, sourceSpan, target);
}
super.visitLoop(node);
}
@override
visitGotoStatement(ast.GotoStatement node) {
JumpTarget target = elements.getTargetOf(node);
assert(target != null, 'No target for $node.');
NodeId id = createGotoId(node);
SourceSpan sourceSpan = computeSourceSpan(node);
gotos.add(new GotoData(id, sourceSpan, target));
super.visitGotoStatement(node);
}
@override
visitLabeledStatement(ast.LabeledStatement node) {
if (node.statement is! ast.Loop && node.statement is! ast.SwitchStatement) {
JumpTarget target = elements.getTargetDefinition(node.statement);
if (target != null) {
NodeId id = createLabeledStatementId(node);
SourceSpan sourceSpan = computeSourceSpan(node);
targets[target] = new TargetData(index++, id, sourceSpan, target);
}
}
super.visitLabeledStatement(node);
}
@override
visitSwitchStatement(ast.SwitchStatement node) {
JumpTarget target = elements.getTargetDefinition(node);
if (target != null) {
NodeId id = createLoopId(node);
SourceSpan sourceSpan = computeSourceSpan(node);
targets[target] = new TargetData(index++, id, sourceSpan, target);
}
super.visitSwitchStatement(node);
}
@override
visitSwitchCase(ast.SwitchCase node) {
JumpTarget target = elements.getTargetDefinition(node);
if (target != null) {
NodeId id = createSwitchCaseId(node);
SourceSpan sourceSpan = computeSourceSpan(node);
targets[target] = new TargetData(index++, id, sourceSpan, target);
}
super.visitSwitchCase(node);
}
}
/// Kernel IR visitor for computing jump data.
class JumpsIrChecker extends IrDataExtractor with JumpsMixin {
final KernelToLocalsMap _localsMap;
JumpsIrChecker(DiagnosticReporter reporter, Map<Id, ActualData> actualMap,
this._localsMap)
: super(reporter, actualMap);
void run(ir.Node root) {
super.run(root);
processData();
}
@override
String computeNodeValue(Id id, ir.Node node) {
// Node values are computed post-visit in [processData].
return null;
}
@override
String computeMemberValue(Id id, ir.Member member) {
return null;
}
void addTargetData(ir.TreeNode node, NodeId id, JumpTarget target) {
if (target != null) {
SourceSpan sourceSpan = computeSourceSpan(node);
targets[target] = new TargetData(index++, id, sourceSpan, target);
}
}
visitForStatement(ir.ForStatement node) {
addTargetData(
node, createLoopId(node), _localsMap.getJumpTargetForFor(node));
super.visitForStatement(node);
}
visitForInStatement(ir.ForInStatement node) {
addTargetData(
node, createLoopId(node), _localsMap.getJumpTargetForForIn(node));
super.visitForInStatement(node);
}
visitWhileStatement(ir.WhileStatement node) {
addTargetData(
node, createLoopId(node), _localsMap.getJumpTargetForWhile(node));
super.visitWhileStatement(node);
}
visitDoStatement(ir.DoStatement node) {
addTargetData(
node, createLoopId(node), _localsMap.getJumpTargetForDo(node));
super.visitDoStatement(node);
}
visitBreakStatement(ir.BreakStatement node) {
JumpTarget target = _localsMap.getJumpTargetForBreak(node);
assert(target != null, 'No target for $node.');
NodeId id = createGotoId(node);
SourceSpan sourceSpan = computeSourceSpan(node);
gotos.add(new GotoData(id, sourceSpan, target));
super.visitBreakStatement(node);
}
@override
visitLabeledStatement(ir.LabeledStatement node) {
JumpTarget target = _localsMap.getJumpTargetForLabel(node);
if (target != null) {
NodeId id = createLabeledStatementId(node);
SourceSpan sourceSpan = computeSourceSpan(node);
targets[target] = new TargetData(index++, id, sourceSpan, target);
}
super.visitLabeledStatement(node);
}
visitSwitchStatement(ir.SwitchStatement node) {
addTargetData(
node, createSwitchId(node), _localsMap.getJumpTargetForSwitch(node));
super.visitSwitchStatement(node);
}
visitSwitchCase(ir.SwitchCase node) {
addTargetData(node, createSwitchCaseId(node),
_localsMap.getJumpTargetForSwitchCase(node));
super.visitSwitchCase(node);
}
visitContinueSwitchStatement(ir.ContinueSwitchStatement node) {
JumpTarget target = _localsMap.getJumpTargetForContinueSwitch(node);
assert(target != null, 'No target for $node.');
NodeId id = createGotoId(node);
SourceSpan sourceSpan = computeSourceSpan(node);
gotos.add(new GotoData(id, sourceSpan, target));
super.visitContinueSwitchStatement(node);
}
}