| // 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(Uri base, Map json) { |
| String root = json['root']; |
| if (!root.endsWith('/')) { |
| print('Inside if!'); |
| root += '/'; |
| } |
| print(base.resolve(root)); |
| return 'some constant'; |
| } |
| |
| const optimizationCounterThreshold = 10; |
| |
| void testFunction() { |
| debugger(); |
| // Note that if we do `i < 1`` here optimization doesn't kick in |
| // (I'm not sure why it kicks in so soon though). |
| for (int i = 0; i < optimizationCounterThreshold; i++) { |
| leafFunction(Uri.base, {'root': 'foo/'}); |
| } |
| // Assuming `leafFunction` is optimized now, does coverage still work? |
| leafFunction(Uri.base, {'root': 'bar'}); |
| 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': 628, |
| 'compiled': true, |
| 'coverage': { |
| 'hits': [], |
| 'misses': [399, 488, 510, 531, 561, 575, 586], |
| }, |
| }; |
| |
| 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_instance_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': 628, |
| 'compiled': true, |
| 'coverage': { |
| 'hits': [399, 488, 510, 531, 561, 575, 586], |
| '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_instance_call_after_optimization_test.dart'), |
| ); |
| }, |
| ]; |
| |
| void main(List<String> args) => runIsolateTests( |
| args, |
| tests, |
| 'coverage_instance_call_after_optimization_test.dart', |
| testeeConcurrent: testFunction, |
| extraArgs: [ |
| '--deterministic', |
| '--optimization-counter-threshold=$optimizationCounterThreshold', |
| ], |
| ); |