blob: 618e97fa75c68df2f780f20151636f17fc3c44bc [file] [log] [blame]
// Copyright (c) 2024, 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:developer';
import 'package:test/test.dart';
import 'package:vm_service/vm_service.dart';
import 'common/service_test_common.dart';
import 'common/test_helper.dart';
@pragma('vm:never-inline')
String leafFunction(void Function() f, bool intoIf) {
if (intoIf) {
f();
}
return 'some constant';
}
const optimizationCounterThreshold = 10;
void testFunction() {
debugger();
// If we do `optimizationCounterThreshold - 2` here optimization doesn't kick
// in and the test (which otherwise currently fails) passes.
for (int i = 0; i < optimizationCounterThreshold; i++) {
leafFunction(() {}, false);
}
// Assuming `leafFunction` is optimized now, does coverage still work?
// Note that I via `--print_flow_graph --print_flow_graph_optimized \
// --print-flow-graph-filter=leafFunction` can see that it is, but that
// `func.code?.isOptimized` is false for whatever reason.
leafFunction(() {}, true);
debugger();
}
var tests = <IsolateTest>[
hasStoppedAtBreakpoint,
(VmService service, IsolateRef isolateRef) async {
final isolateId = isolateRef.id!;
final isolate = await service.getIsolate(isolateId);
final stack = await service.getStack(isolateId);
// Make sure we are in the right place.
final frames = stack.frames!;
expect(frames.length, greaterThanOrEqualTo(1));
expect(frames[0].function!.name, 'testFunction');
final rootLib =
await service.getObject(isolateId, isolate.rootLib!.id!) as Library;
final funcRef =
rootLib.functions!.singleWhere((f) => f.name == 'leafFunction');
final func = await service.getObject(isolateId, funcRef.id!) as Func;
final expectedRange = {
'scriptIndex': 0,
'startPos': 399,
'endPos': 535,
'compiled': true,
'coverage': {
'hits': [],
'misses': [399, 501],
},
};
final location = func.location!;
final report = await service.getSourceReport(
isolateId,
[SourceReportKind.kCoverage],
scriptId: location.script!.id!,
tokenPos: location.tokenPos,
endTokenPos: location.endTokenPos,
forceCompile: true,
);
final ranges = report.ranges!;
final scripts = report.scripts!;
expect(ranges.length, 1);
expect(ranges[0].toJson(), expectedRange);
expect(scripts.length, 1);
expect(
scripts[0].uri,
endsWith('coverage_closure_call_after_optimization_test.dart'),
);
},
resumeIsolate,
hasStoppedAtBreakpoint,
(VmService service, IsolateRef isolateRef) async {
final isolateId = isolateRef.id!;
final isolate = await service.getIsolate(isolateId);
final stack = await service.getStack(isolateId);
// Make sure we are in the right place.
final frames = stack.frames!;
expect(frames.length, greaterThanOrEqualTo(1));
expect(frames[0].function!.name, 'testFunction');
final rootLib =
await service.getObject(isolateId, isolate.rootLib!.id!) as Library;
final funcRef =
rootLib.functions!.singleWhere((f) => f.name == 'leafFunction');
final func = await service.getObject(isolateId, funcRef.id!) as Func;
final expectedRange = {
'scriptIndex': 0,
'startPos': 399,
'endPos': 535,
'compiled': true,
'coverage': {
'hits': [399, 501],
'misses': [],
},
};
final location = func.location!;
final report = await service.getSourceReport(
isolateId,
[SourceReportKind.kCoverage],
scriptId: location.script!.id!,
tokenPos: location.tokenPos,
endTokenPos: location.endTokenPos,
forceCompile: true,
);
final ranges = report.ranges!;
final scripts = report.scripts!;
expect(ranges.length, 1);
expect(ranges[0].toJson(), expectedRange);
expect(scripts.length, 1);
expect(
scripts[0].uri,
endsWith('coverage_closure_call_after_optimization_test.dart'),
);
},
];
void main(List<String> args) => runIsolateTests(
args,
tests,
'coverage_closure_call_after_optimization_test.dart',
testeeConcurrent: testFunction,
extraArgs: [
'--deterministic',
'--optimization-counter-threshold=$optimizationCounterThreshold',
],
);