Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
7b9aeb2f
Unverified
Commit
7b9aeb2f
authored
Apr 11, 2020
by
Kaushik Iska
Committed by
GitHub
Apr 11, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[flutter_driver] Reland add SceneDisplayLag stats to timeline summary (#54490)
parent
da343dcc
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
174 additions
and
5 deletions
+174
-5
perf_tests.dart
dev/devicelab/lib/tasks/perf_tests.dart
+3
-1
scene_display_lag_summarizer.dart
...r_driver/lib/src/driver/scene_display_lag_summarizer.dart
+82
-0
timeline_summary.dart
packages/flutter_driver/lib/src/driver/timeline_summary.dart
+10
-4
timeline_summary_test.dart
...ter_driver/test/src/real_tests/timeline_summary_test.dart
+79
-0
No files found.
dev/devicelab/lib/tasks/perf_tests.dart
View file @
7b9aeb2f
...
@@ -189,7 +189,6 @@ TaskFunction createColorFilterAndFadePerfTest() {
...
@@ -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 {
...
@@ -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'
,
]);
]);
...
...
packages/flutter_driver/lib/src/driver/scene_display_lag_summarizer.dart
0 → 100644
View file @
7b9aeb2f
// 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
));
final
dynamic
transitionsMissed
=
e
.
arguments
[
_kVsyncTransitionsMissed
];
assert
(
transitionsMissed
is
String
);
return
double
.
parse
(
transitionsMissed
as
String
);
}
}
packages/flutter_driver/lib/src/driver/timeline_summary.dart
View file @
7b9aeb2f
...
@@ -10,6 +10,7 @@ import 'package:file/file.dart';
...
@@ -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 {
...
@@ -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 {
...
@@ -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 {
...
@@ -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 {
...
@@ -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'
);
...
...
packages/flutter_driver/test/src/real_tests/timeline_summary_test.dart
View file @
7b9aeb2f
...
@@ -7,6 +7,7 @@ import 'dart:convert' show json;
...
@@ -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() {
...
@@ -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
,
String
>{
'vsync_transitions_missed'
:
vsyncsMissed
.
toString
()
}
};
Map
<
String
,
dynamic
>
lagEnd
(
int
timeStamp
,
int
vsyncsMissed
)
=>
<
String
,
dynamic
>{
'name'
:
'SceneDisplayLag'
,
'ph'
:
'e'
,
'ts'
:
timeStamp
,
'args'
:
<
String
,
String
>{
'vsync_transitions_missed'
:
vsyncsMissed
.
toString
()
}
};
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() {
...
@@ -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() {
...
@@ -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,7 +434,61 @@ void main() {
...
@@ -409,7 +434,61 @@ 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
);
});
});
});
});
});
});
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment