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
2004afae
Unverified
Commit
2004afae
authored
Sep 10, 2021
by
Jason Simmons
Committed by
GitHub
Sep 10, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Use async timeline events for the phases of the scheduler binding (#88825)
parent
7736b9ee
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
38 additions
and
12 deletions
+38
-12
transitions_perf_test.dart
...ts/flutter_gallery/test_driver/transitions_perf_test.dart
+2
-2
binding.dart
packages/flutter/lib/src/foundation/binding.dart
+2
-2
binding.dart
packages/flutter/lib/src/scheduler/binding.dart
+12
-7
timeline_summary.dart
packages/flutter_driver/lib/src/driver/timeline_summary.dart
+1
-1
timeline_test.dart
...s/flutter_tools/test/integration.shard/timeline_test.dart
+21
-0
No files found.
dev/integration_tests/flutter_gallery/test_driver/transitions_perf_test.dart
View file @
2004afae
...
...
@@ -37,11 +37,11 @@ Future<void> saveDurationsHistogram(List<Map<String, dynamic>> events, String ou
}
else
if
(
startEvent
!=
null
&&
eventName
==
'Frame'
)
{
final
String
phase
=
event
[
'ph'
]
as
String
;
final
int
timestamp
=
event
[
'ts'
]
as
int
;
if
(
phase
==
'B'
)
{
if
(
phase
==
'B'
||
phase
==
'b'
)
{
assert
(
frameStart
==
null
);
frameStart
=
timestamp
;
}
else
{
assert
(
phase
==
'E'
);
assert
(
phase
==
'E'
||
phase
==
'e'
);
final
String
routeName
=
(
startEvent
[
'args'
]
as
Map
<
String
,
dynamic
>)[
'to'
]
as
String
;
durations
[
routeName
]
??=
<
int
>[];
durations
[
routeName
]!.
add
(
timestamp
-
frameStart
!);
...
...
packages/flutter/lib/src/foundation/binding.dart
View file @
2004afae
...
...
@@ -298,7 +298,7 @@ abstract class BindingBase {
/// The [Future] returned by the `callback` argument is returned by [lockEvents].
@protected
Future
<
void
>
lockEvents
(
Future
<
void
>
Function
()
callback
)
{
developer
.
Timeline
.
startSync
(
'Lock events'
);
final
developer
.
TimelineTask
timelineTask
=
developer
.
TimelineTask
()..
start
(
'Lock events'
);
assert
(
callback
!=
null
);
_lockCount
+=
1
;
...
...
@@ -307,7 +307,7 @@ abstract class BindingBase {
future
.
whenComplete
(()
{
_lockCount
-=
1
;
if
(!
locked
)
{
developer
.
Timeline
.
finishSync
();
timelineTask
.
finish
();
unlocked
();
}
});
...
...
packages/flutter/lib/src/scheduler/binding.dart
View file @
2004afae
...
...
@@ -4,7 +4,7 @@
import
'dart:async'
;
import
'dart:collection'
;
import
'dart:developer'
show
Flow
,
Timeline
;
import
'dart:developer'
show
Flow
,
Timeline
,
TimelineTask
;
import
'dart:ui'
show
AppLifecycleState
,
FramePhase
,
FrameTiming
,
TimingsCallback
,
PlatformDispatcher
;
import
'package:collection/collection.dart'
show
PriorityQueue
,
HeapPriorityQueue
;
...
...
@@ -851,7 +851,7 @@ mixin SchedulerBinding on BindingBase {
return
;
_warmUpFrame
=
true
;
Timeline
.
startSync
(
'Warm-up frame'
);
final
TimelineTask
timelineTask
=
TimelineTask
()..
start
(
'Warm-up frame'
);
final
bool
hadScheduledFrame
=
_hasScheduledFrame
;
// We use timers here to ensure that microtasks flush in between.
Timer
.
run
(()
{
...
...
@@ -879,7 +879,7 @@ mixin SchedulerBinding on BindingBase {
// scheduled frame has finished.
lockEvents
(()
async
{
await
endOfFrame
;
Timeline
.
finishSync
();
timelineTask
.
finish
();
});
}
...
...
@@ -996,6 +996,9 @@ mixin SchedulerBinding on BindingBase {
handleDrawFrame
();
}
TimelineTask
?
_frameTimelineTask
;
TimelineTask
?
_animateTimelineTask
;
/// Called by the engine to prepare the framework to produce a new frame.
///
/// This function calls all the transient frame callbacks registered by
...
...
@@ -1020,7 +1023,7 @@ mixin SchedulerBinding on BindingBase {
/// statements printed during a frame from those printed between frames (e.g.
/// in response to events or timers).
void
handleBeginFrame
(
Duration
?
rawTimeStamp
)
{
Timeline
.
startSync
(
'Frame'
,
arguments:
timelineArgumentsIndicatingLandmarkEvent
);
_frameTimelineTask
=
TimelineTask
()..
start
(
'Frame'
,
arguments:
timelineArgumentsIndicatingLandmarkEvent
);
_firstRawTimeStampInEpoch
??=
rawTimeStamp
;
_currentFrameTimeStamp
=
_adjustForEpoch
(
rawTimeStamp
??
_lastRawTimeStamp
);
if
(
rawTimeStamp
!=
null
)
...
...
@@ -1047,7 +1050,7 @@ mixin SchedulerBinding on BindingBase {
_hasScheduledFrame
=
false
;
try
{
// TRANSIENT FRAME CALLBACKS
Timeline
.
startSync
(
'Animate'
,
arguments:
timelineArgumentsIndicatingLandmarkEvent
);
_animateTimelineTask
=
TimelineTask
()..
start
(
'Animate'
,
arguments:
timelineArgumentsIndicatingLandmarkEvent
);
_schedulerPhase
=
SchedulerPhase
.
transientCallbacks
;
final
Map
<
int
,
_FrameCallbackEntry
>
callbacks
=
_transientCallbacks
;
_transientCallbacks
=
<
int
,
_FrameCallbackEntry
>{};
...
...
@@ -1072,7 +1075,8 @@ mixin SchedulerBinding on BindingBase {
/// useful when working with frame callbacks.
void
handleDrawFrame
()
{
assert
(
_schedulerPhase
==
SchedulerPhase
.
midFrameMicrotasks
);
Timeline
.
finishSync
();
// end the "Animate" phase
_animateTimelineTask
?.
finish
();
// end the "Animate" phase
_animateTimelineTask
=
null
;
try
{
// PERSISTENT FRAME CALLBACKS
_schedulerPhase
=
SchedulerPhase
.
persistentCallbacks
;
...
...
@@ -1088,7 +1092,8 @@ mixin SchedulerBinding on BindingBase {
_invokeFrameCallback
(
callback
,
_currentFrameTimeStamp
!);
}
finally
{
_schedulerPhase
=
SchedulerPhase
.
idle
;
Timeline
.
finishSync
();
// end the Frame
_frameTimelineTask
?.
finish
();
// end the Frame
_frameTimelineTask
=
null
;
assert
(()
{
if
(
debugPrintEndFrameBanner
)
debugPrint
(
'▀'
*
_debugBanner
!.
length
);
...
...
packages/flutter_driver/lib/src/driver/timeline_summary.dart
View file @
2004afae
...
...
@@ -351,7 +351,7 @@ class TimelineSummary {
// Timeline does not guarantee that the first event is the "begin" event.
TimelineEvent
?
begin
;
for
(
final
TimelineEvent
event
in
events
)
{
if
(
event
.
phase
==
'B'
)
{
if
(
event
.
phase
==
'B'
||
event
.
phase
==
'b'
)
{
begin
=
event
;
}
else
{
if
(
begin
!=
null
)
{
...
...
packages/flutter_tools/test/integration.shard/timeline_test.dart
View file @
2004afae
...
...
@@ -79,5 +79,26 @@ void main() {
expect
(
response
.
json
[
'value'
],
'Brightness.light'
);
}
timer
.
cancel
();
// Verify that all duration events on the timeline are properly nested.
final
Response
response
=
await
vmService
.
callServiceExtension
(
'getVMTimeline'
);
final
List
<
TimelineEvent
>
events
=
(
response
as
Timeline
).
traceEvents
;
final
Map
<
int
,
List
<
String
>>
threadDurationEventStack
=
<
int
,
List
<
String
>>{};
for
(
final
TimelineEvent
e
in
events
)
{
final
Map
<
String
,
dynamic
>
event
=
e
.
json
;
final
String
phase
=
event
[
'ph'
]
as
String
;
final
int
tid
=
event
[
'tid'
]
as
int
;
final
String
name
=
event
[
'name'
]
as
String
;
final
List
<
String
>
stack
=
threadDurationEventStack
.
putIfAbsent
(
tid
,
()
=>
<
String
>[]);
if
(
phase
==
'B'
)
{
stack
.
add
(
name
);
}
else
if
(
phase
==
'E'
)
{
// The downloaded part of the timeline may contain an end event whose
// corresponding begin event happened before the start of the timeline.
if
(
stack
.
isNotEmpty
)
{
expect
(
stack
.
removeLast
(),
name
);
}
}
}
});
}
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