blob: a41730bdaa65a08f610658f6cc16251fb6253d07 [file] [log] [blame] [edit]
// Copyright (c) 2018, 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:_fe_analyzer_shared/src/testing/features.dart';
import 'package:expect/async_helper.dart';
import 'package:compiler/src/closure.dart';
import 'package:compiler/src/common.dart';
import 'package:compiler/src/compiler.dart';
import 'package:compiler/src/elements/entities.dart';
import 'package:compiler/src/js_backend/runtime_types.dart';
import 'package:compiler/src/js_emitter/model.dart';
import 'package:compiler/src/js_model/element_map.dart';
import 'package:compiler/src/js_model/js_strategy.dart';
import 'package:compiler/src/js_model/js_world.dart';
import 'package:kernel/ast.dart' as ir;
import '../equivalence/id_equivalence.dart';
import '../equivalence/id_equivalence_helper.dart';
import '../helpers/program_lookup.dart';
main(List<String> args) {
runTests(args);
}
runTests(List<String> args, [int? shardIndex]) {
asyncTest(() async {
Directory dataDir = Directory.fromUri(Platform.script.resolve('emission'));
await checkTests(
dataDir,
const RtiEmissionDataComputer(),
args: args,
shardIndex: shardIndex ?? 0,
shards: shardIndex != null ? 4 : 1,
);
});
}
class Tags {
static const String isChecks = 'checks';
static const String indirectInstance = 'indirectInstance';
static const String directInstance = 'instance';
static const String onlyForRti = 'onlyForRti';
static const String onlyForConstructor = 'onlyForConstructor';
static const String checkedInstance = 'checkedInstance';
static const String typeArgument = 'typeArgument';
static const String checkedTypeArgument = 'checkedTypeArgument';
static const String typeLiteral = 'typeLiteral';
static const String functionType = 'functionType';
}
mixin ComputeValueMixin {
Compiler get compiler;
late final ProgramLookup lookup = ProgramLookup(backendStrategy);
JsBackendStrategy get backendStrategy => compiler.backendStrategy;
RuntimeTypesImpl get checksBuilder =>
backendStrategy.rtiChecksBuilderForTesting as RuntimeTypesImpl;
String getClassValue(ClassEntity element) {
Class? cls = lookup.getClass(element);
Features features = Features();
if (cls != null) {
features.addElement(Tags.isChecks);
for (StubMethod stub in cls.isChecks) {
features.addElement(Tags.isChecks, stub.name!.key);
}
if (cls.functionTypeIndex != null) {
features.add(Tags.functionType);
}
if (cls.onlyForRti) {
features.add(Tags.onlyForRti);
}
if (cls.onlyForConstructor) {
features.add(Tags.onlyForConstructor);
}
}
ClassUse? classUse = checksBuilder.classUseMapForTesting![element];
if (classUse != null) {
if (classUse.directInstance) {
features.add(Tags.directInstance);
} else if (classUse.instance) {
features.add(Tags.indirectInstance);
}
if (classUse.checkedInstance) {
features.add(Tags.checkedInstance);
}
if (classUse.typeArgument) {
features.add(Tags.typeArgument);
}
if (classUse.checkedTypeArgument) {
features.add(Tags.checkedTypeArgument);
}
if (classUse.typeLiteral) {
features.add(Tags.typeLiteral);
}
}
return features.getText();
}
String? getMemberValue(MemberEntity member) {
if (member.enclosingClass != null && member.enclosingClass!.isClosure) {
return getClassValue(member.enclosingClass!);
}
return null;
}
}
class RtiEmissionDataComputer extends DataComputer<String> {
const RtiEmissionDataComputer();
@override
void computeMemberData(
Compiler compiler,
MemberEntity member,
Map<Id, ActualData<String>> actualMap, {
bool verbose = false,
}) {
JClosedWorld closedWorld = compiler.backendClosedWorldForTesting!;
JsToElementMap elementMap = closedWorld.elementMap;
MemberDefinition definition = elementMap.getMemberDefinition(member);
RtiEmissionIrComputer(
compiler.reporter,
actualMap,
elementMap,
compiler,
closedWorld.closureDataLookup,
).run(definition.node);
}
@override
void computeClassData(
Compiler compiler,
ClassEntity cls,
Map<Id, ActualData<String>> actualMap, {
bool verbose = false,
}) {
JClosedWorld closedWorld = compiler.backendClosedWorldForTesting!;
JsToElementMap elementMap = closedWorld.elementMap;
RtiEmissionIrComputer(
compiler.reporter,
actualMap,
elementMap,
compiler,
closedWorld.closureDataLookup,
).computeForClass(elementMap.getClassDefinition(cls).node as ir.Class);
}
@override
DataInterpreter<String> get dataValidator => const StringDataInterpreter();
}
class RtiEmissionIrComputer extends IrDataExtractor<String>
with ComputeValueMixin {
final JsToElementMap _elementMap;
final ClosureData _closureDataLookup;
@override
final Compiler compiler;
RtiEmissionIrComputer(
DiagnosticReporter reporter,
Map<Id, ActualData<String>> actualMap,
this._elementMap,
this.compiler,
this._closureDataLookup,
) : super(reporter, actualMap);
@override
String computeClassValue(Id id, ir.Class node) {
return getClassValue(_elementMap.getClass(node));
}
@override
String? computeMemberValue(Id id, ir.Member node) {
return getMemberValue(_elementMap.getMember(node));
}
@override
String? computeNodeValue(Id id, ir.TreeNode node) {
if (node is ir.FunctionExpression || node is ir.FunctionDeclaration) {
ClosureRepresentationInfo info = _closureDataLookup.getClosureInfo(
node as ir.LocalFunction,
);
return getMemberValue(info.callMethod!);
}
return null;
}
}