mirror of
https://github.com/flutter/flutter.git
synced 2025-06-03 00:51:18 +00:00
[flutter_driver] Add SceneDisplayLag stats to timeline summary (#54218)
This commit is contained in:
parent
f7de5aa5b0
commit
ce92333bf7
@ -189,7 +189,6 @@ TaskFunction createColorFilterAndFadePerfTest() {
|
|||||||
).run;
|
).run;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// Measure application startup performance.
|
/// Measure application startup performance.
|
||||||
class StartupTest {
|
class StartupTest {
|
||||||
const StartupTest(this.testDirectory, { this.reportMetrics = true });
|
const StartupTest(this.testDirectory, { this.reportMetrics = true });
|
||||||
@ -281,6 +280,9 @@ class PerfTest {
|
|||||||
'worst_frame_rasterizer_time_millis',
|
'worst_frame_rasterizer_time_millis',
|
||||||
'90th_percentile_frame_rasterizer_time_millis',
|
'90th_percentile_frame_rasterizer_time_millis',
|
||||||
'99th_percentile_frame_rasterizer_time_millis',
|
'99th_percentile_frame_rasterizer_time_millis',
|
||||||
|
'average_vsync_transitions_missed',
|
||||||
|
'90th_percentile_vsync_transitions_missed',
|
||||||
|
'99th_percentile_vsync_transitions_missed',
|
||||||
if (needsMeasureCpuGPu) 'cpu_percentage',
|
if (needsMeasureCpuGPu) 'cpu_percentage',
|
||||||
if (needsMeasureCpuGPu) 'gpu_percentage',
|
if (needsMeasureCpuGPu) 'gpu_percentage',
|
||||||
]);
|
]);
|
||||||
|
@ -0,0 +1,80 @@
|
|||||||
|
// Copyright 2014 The Flutter Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
import 'timeline.dart';
|
||||||
|
|
||||||
|
/// Key for SceneDisplayLag timeline events.
|
||||||
|
const String kSceneDisplayLagEvent = 'SceneDisplayLag';
|
||||||
|
|
||||||
|
const String _kVsyncTransitionsMissed = 'vsync_transitions_missed';
|
||||||
|
|
||||||
|
/// Returns the [p]-th percentile element from the [doubles].
|
||||||
|
///
|
||||||
|
/// Note: [doubles] will be sorted.
|
||||||
|
double findPercentile(List<double> doubles, double p) {
|
||||||
|
assert(doubles.isNotEmpty);
|
||||||
|
doubles.sort();
|
||||||
|
return doubles[((doubles.length - 1) * (p / 100)).round()];
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Summarizes [TimelineEvents]s corresponding to [kSceneDisplayLagEvent] events.
|
||||||
|
///
|
||||||
|
/// A sample event (some fields have been omitted for brewity):
|
||||||
|
/// ```
|
||||||
|
/// {
|
||||||
|
/// "name": "SceneDisplayLag",
|
||||||
|
/// "ts": 408920509340,
|
||||||
|
/// "ph": "b", (this can be 'b' or 'e' for begin or end)
|
||||||
|
/// "args": {
|
||||||
|
/// "frame_target_time": "408920509340458",
|
||||||
|
/// "current_frame_target_time": "408920542689291",
|
||||||
|
/// "vsync_transitions_missed": "2"
|
||||||
|
/// }
|
||||||
|
/// },
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// `vsync_transitions_missed` corresponds to the elapsed number of frame budget
|
||||||
|
/// durations between when the frame was scheduled to be displayed, i.e, the
|
||||||
|
/// `frame_target_time` and the next vsync pulse timestamp, i.e, the
|
||||||
|
/// `current_frame_target_time`.
|
||||||
|
class SceneDisplayLagSummarizer {
|
||||||
|
/// Creates a SceneDisplayLagSummarizer given the timeline events.
|
||||||
|
SceneDisplayLagSummarizer(this.sceneDisplayLagEvents) {
|
||||||
|
for (final TimelineEvent event in sceneDisplayLagEvents) {
|
||||||
|
assert(event.name == kSceneDisplayLagEvent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The scene display lag events.
|
||||||
|
final List<TimelineEvent> sceneDisplayLagEvents;
|
||||||
|
|
||||||
|
/// Computes the average of the `vsync_transitions_missed` over the lag events.
|
||||||
|
double computeAverageVsyncTransitionsMissed() {
|
||||||
|
if (sceneDisplayLagEvents.isEmpty) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
final double total = sceneDisplayLagEvents
|
||||||
|
.map(_getVsyncTransitionsMissed)
|
||||||
|
.reduce((double a, double b) => a + b);
|
||||||
|
return total / sceneDisplayLagEvents.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The [percentile]-th percentile `vsync_transitions_missed` over the lag events.
|
||||||
|
double computePercentileVsyncTransitionsMissed(double percentile) {
|
||||||
|
if (sceneDisplayLagEvents.isEmpty) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
final List<double> doubles =
|
||||||
|
sceneDisplayLagEvents.map(_getVsyncTransitionsMissed).toList();
|
||||||
|
return findPercentile(doubles, percentile);
|
||||||
|
}
|
||||||
|
|
||||||
|
double _getVsyncTransitionsMissed(TimelineEvent e) {
|
||||||
|
assert(e.name == kSceneDisplayLagEvent);
|
||||||
|
assert(e.arguments.containsKey(_kVsyncTransitionsMissed));
|
||||||
|
return (e.arguments[_kVsyncTransitionsMissed] as num).toDouble();
|
||||||
|
}
|
||||||
|
}
|
@ -10,6 +10,7 @@ import 'package:file/file.dart';
|
|||||||
import 'package:path/path.dart' as path;
|
import 'package:path/path.dart' as path;
|
||||||
|
|
||||||
import 'common.dart';
|
import 'common.dart';
|
||||||
|
import 'scene_display_lag_summarizer.dart';
|
||||||
import 'timeline.dart';
|
import 'timeline.dart';
|
||||||
|
|
||||||
const JsonEncoder _prettyEncoder = JsonEncoder.withIndent(' ');
|
const JsonEncoder _prettyEncoder = JsonEncoder.withIndent(' ');
|
||||||
@ -85,6 +86,8 @@ class TimelineSummary {
|
|||||||
|
|
||||||
/// Encodes this summary as JSON.
|
/// Encodes this summary as JSON.
|
||||||
Map<String, dynamic> get summaryJson {
|
Map<String, dynamic> get summaryJson {
|
||||||
|
final SceneDisplayLagSummarizer sceneDisplayLagSummarizer = _sceneDisplayLagSummarizer();
|
||||||
|
|
||||||
return <String, dynamic>{
|
return <String, dynamic>{
|
||||||
'average_frame_build_time_millis': computeAverageFrameBuildTimeMillis(),
|
'average_frame_build_time_millis': computeAverageFrameBuildTimeMillis(),
|
||||||
'90th_percentile_frame_build_time_millis': computePercentileFrameBuildTimeMillis(90.0),
|
'90th_percentile_frame_build_time_millis': computePercentileFrameBuildTimeMillis(90.0),
|
||||||
@ -105,7 +108,10 @@ class TimelineSummary {
|
|||||||
.toList(),
|
.toList(),
|
||||||
'frame_begin_times': _extractBeginTimestamps('Frame')
|
'frame_begin_times': _extractBeginTimestamps('Frame')
|
||||||
.map<int>((Duration duration) => duration.inMicroseconds)
|
.map<int>((Duration duration) => duration.inMicroseconds)
|
||||||
.toList()
|
.toList(),
|
||||||
|
'average_vsync_transitions_missed': sceneDisplayLagSummarizer.computeAverageVsyncTransitionsMissed(),
|
||||||
|
'90th_percentile_vsync_transitions_missed': sceneDisplayLagSummarizer.computePercentileVsyncTransitionsMissed(90.0),
|
||||||
|
'99th_percentile_vsync_transitions_missed': sceneDisplayLagSummarizer.computePercentileVsyncTransitionsMissed(99.0)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -208,9 +214,7 @@ class TimelineSummary {
|
|||||||
throw ArgumentError('durations is empty!');
|
throw ArgumentError('durations is empty!');
|
||||||
assert(percentile >= 0.0 && percentile <= 100.0);
|
assert(percentile >= 0.0 && percentile <= 100.0);
|
||||||
final List<double> doubles = durations.map<double>((Duration duration) => duration.inMicroseconds.toDouble() / 1000.0).toList();
|
final List<double> doubles = durations.map<double>((Duration duration) => duration.inMicroseconds.toDouble() / 1000.0).toList();
|
||||||
doubles.sort();
|
return findPercentile(doubles, percentile);
|
||||||
return doubles[((doubles.length - 1) * (percentile / 100)).round()];
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
double _maxInMillis(Iterable<Duration> durations) {
|
double _maxInMillis(Iterable<Duration> durations) {
|
||||||
@ -221,6 +225,8 @@ class TimelineSummary {
|
|||||||
.reduce(math.max);
|
.reduce(math.max);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SceneDisplayLagSummarizer _sceneDisplayLagSummarizer() => SceneDisplayLagSummarizer(_extractNamedEvents(kSceneDisplayLagEvent));
|
||||||
|
|
||||||
List<Duration> _extractGpuRasterizerDrawDurations() => _extractBeginEndEvents('GPURasterizer::Draw');
|
List<Duration> _extractGpuRasterizerDrawDurations() => _extractBeginEndEvents('GPURasterizer::Draw');
|
||||||
|
|
||||||
List<Duration> _extractFrameDurations() => _extractBeginEndEvents('Frame');
|
List<Duration> _extractFrameDurations() => _extractBeginEndEvents('Frame');
|
||||||
|
@ -7,6 +7,7 @@ import 'dart:convert' show json;
|
|||||||
import 'package:file/file.dart';
|
import 'package:file/file.dart';
|
||||||
import 'package:flutter_driver/flutter_driver.dart';
|
import 'package:flutter_driver/flutter_driver.dart';
|
||||||
import 'package:flutter_driver/src/driver/common.dart';
|
import 'package:flutter_driver/src/driver/common.dart';
|
||||||
|
import 'package:flutter_driver/src/driver/scene_display_lag_summarizer.dart';
|
||||||
import 'package:path/path.dart' as path;
|
import 'package:path/path.dart' as path;
|
||||||
|
|
||||||
import '../../common.dart';
|
import '../../common.dart';
|
||||||
@ -44,6 +45,24 @@ void main() {
|
|||||||
'ts': timeStamp,
|
'ts': timeStamp,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Map<String, dynamic> lagBegin(int timeStamp, int vsyncsMissed) => <String, dynamic>{
|
||||||
|
'name': 'SceneDisplayLag',
|
||||||
|
'ph': 'b',
|
||||||
|
'ts': timeStamp,
|
||||||
|
'args': <String, int>{
|
||||||
|
'vsync_transitions_missed': vsyncsMissed
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Map<String, dynamic> lagEnd(int timeStamp, int vsyncsMissed) => <String, dynamic>{
|
||||||
|
'name': 'SceneDisplayLag',
|
||||||
|
'ph': 'e',
|
||||||
|
'ts': timeStamp,
|
||||||
|
'args': <String, int>{
|
||||||
|
'vsync_transitions_missed': vsyncsMissed
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
List<Map<String, dynamic>> rasterizeTimeSequenceInMillis(List<int> sequence) {
|
List<Map<String, dynamic>> rasterizeTimeSequenceInMillis(List<int> sequence) {
|
||||||
final List<Map<String, dynamic>> result = <Map<String, dynamic>>[];
|
final List<Map<String, dynamic>> result = <Map<String, dynamic>>[];
|
||||||
int t = 0;
|
int t = 0;
|
||||||
@ -356,6 +375,9 @@ void main() {
|
|||||||
'frame_build_times': <int>[17000, 9000, 19000],
|
'frame_build_times': <int>[17000, 9000, 19000],
|
||||||
'frame_rasterizer_times': <int>[18000, 10000, 20000],
|
'frame_rasterizer_times': <int>[18000, 10000, 20000],
|
||||||
'frame_begin_times': <int>[0, 18000, 28000],
|
'frame_begin_times': <int>[0, 18000, 28000],
|
||||||
|
'average_vsync_transitions_missed': 0.0,
|
||||||
|
'90th_percentile_vsync_transitions_missed': 0.0,
|
||||||
|
'99th_percentile_vsync_transitions_missed': 0.0
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
@ -391,6 +413,9 @@ void main() {
|
|||||||
frameBegin(1000), frameEnd(18000),
|
frameBegin(1000), frameEnd(18000),
|
||||||
frameBegin(19000), frameEnd(28000),
|
frameBegin(19000), frameEnd(28000),
|
||||||
frameBegin(29000), frameEnd(48000),
|
frameBegin(29000), frameEnd(48000),
|
||||||
|
lagBegin(1000, 4), lagEnd(2000, 4),
|
||||||
|
lagBegin(1200, 12), lagEnd(2400, 12),
|
||||||
|
lagBegin(4200, 8), lagEnd(9400, 8),
|
||||||
]).writeSummaryToFile('test', destinationDirectory: tempDir.path);
|
]).writeSummaryToFile('test', destinationDirectory: tempDir.path);
|
||||||
final String written =
|
final String written =
|
||||||
await fs.file(path.join(tempDir.path, 'test.timeline_summary.json')).readAsString();
|
await fs.file(path.join(tempDir.path, 'test.timeline_summary.json')).readAsString();
|
||||||
@ -409,8 +434,62 @@ void main() {
|
|||||||
'frame_build_times': <int>[17000, 9000, 19000],
|
'frame_build_times': <int>[17000, 9000, 19000],
|
||||||
'frame_rasterizer_times': <int>[18000, 10000, 20000],
|
'frame_rasterizer_times': <int>[18000, 10000, 20000],
|
||||||
'frame_begin_times': <int>[0, 18000, 28000],
|
'frame_begin_times': <int>[0, 18000, 28000],
|
||||||
|
'average_vsync_transitions_missed': 8.0,
|
||||||
|
'90th_percentile_vsync_transitions_missed': 12.0,
|
||||||
|
'99th_percentile_vsync_transitions_missed': 12.0
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
group('SceneDisplayLagSummarizer tests', () {
|
||||||
|
SceneDisplayLagSummarizer summarize(List<Map<String, dynamic>> traceEvents) {
|
||||||
|
final Timeline timeline = Timeline.fromJson(<String, dynamic>{
|
||||||
|
'traceEvents': traceEvents,
|
||||||
|
});
|
||||||
|
return SceneDisplayLagSummarizer(timeline.events);
|
||||||
|
}
|
||||||
|
|
||||||
|
test('average_vsyncs_missed', () async {
|
||||||
|
final SceneDisplayLagSummarizer summarizer = summarize(<Map<String, dynamic>>[
|
||||||
|
lagBegin(1000, 4), lagEnd(2000, 4),
|
||||||
|
lagBegin(1200, 12), lagEnd(2400, 12),
|
||||||
|
lagBegin(4200, 8), lagEnd(9400, 8),
|
||||||
|
]);
|
||||||
|
expect(summarizer.computeAverageVsyncTransitionsMissed(), 8.0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('all stats are 0 for 0 missed transitions', () async {
|
||||||
|
final SceneDisplayLagSummarizer summarizer = summarize(<Map<String, dynamic>>[]);
|
||||||
|
expect(summarizer.computeAverageVsyncTransitionsMissed(), 0.0);
|
||||||
|
expect(summarizer.computePercentileVsyncTransitionsMissed(90.0), 0.0);
|
||||||
|
expect(summarizer.computePercentileVsyncTransitionsMissed(99.0), 0.0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('90th_percentile_vsyncs_missed', () async {
|
||||||
|
final SceneDisplayLagSummarizer summarizer = summarize(<Map<String, dynamic>>[
|
||||||
|
lagBegin(1000, 4), lagEnd(2000, 4),
|
||||||
|
lagBegin(1200, 12), lagEnd(2400, 12),
|
||||||
|
lagBegin(4200, 8), lagEnd(9400, 8),
|
||||||
|
lagBegin(6100, 14), lagEnd(11000, 14),
|
||||||
|
lagBegin(7100, 16), lagEnd(11500, 16),
|
||||||
|
lagBegin(7400, 11), lagEnd(13000, 11),
|
||||||
|
lagBegin(8200, 27), lagEnd(14100, 27),
|
||||||
|
lagBegin(8700, 7), lagEnd(14300, 7),
|
||||||
|
lagBegin(24200, 4187), lagEnd(39400, 4187),
|
||||||
|
]);
|
||||||
|
expect(summarizer.computePercentileVsyncTransitionsMissed(90), 27.0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('99th_percentile_vsyncs_missed', () async {
|
||||||
|
final SceneDisplayLagSummarizer summarizer = summarize(<Map<String, dynamic>>[
|
||||||
|
lagBegin(1000, 4), lagEnd(2000, 4),
|
||||||
|
lagBegin(1200, 12), lagEnd(2400, 12),
|
||||||
|
lagBegin(4200, 8), lagEnd(9400, 8),
|
||||||
|
lagBegin(6100, 14), lagEnd(11000, 14),
|
||||||
|
lagBegin(24200, 4187), lagEnd(39400, 4187),
|
||||||
|
]);
|
||||||
|
expect(summarizer.computePercentileVsyncTransitionsMissed(99), 4187.0);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user