您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
772 行
27 KiB
772 行
27 KiB
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Text;
|
|
using developer;
|
|
using Unity.UIWidgets.async;
|
|
using Unity.UIWidgets.engine;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.ui;
|
|
using UnityEngine;
|
|
using UnityEngine.Rendering;
|
|
using FrameTiming = Unity.UIWidgets.ui.FrameTiming;
|
|
|
|
namespace Unity.UIWidgets.scheduler {
|
|
public static partial class scheduler_ {
|
|
static float _timeDilation = 1.0f;
|
|
|
|
public static float timeDilation {
|
|
get { return _timeDilation; }
|
|
set {
|
|
D.assert(value > 0.0f);
|
|
if (_timeDilation == value) {
|
|
return;
|
|
}
|
|
|
|
SchedulerBinding.instance?.resetEpoch();
|
|
_timeDilation = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
public delegate void FrameCallback(TimeSpan timeStamp);
|
|
|
|
public delegate T TaskCallback<out T>();
|
|
|
|
public delegate bool SchedulingStrategy(int priority = 0, SchedulerBinding scheduler = null);
|
|
|
|
interface _TaskEntry : IComparable<_TaskEntry> {
|
|
int priority { get; }
|
|
string debugStack { get; }
|
|
void run();
|
|
}
|
|
|
|
class _TaskEntry<T> : _TaskEntry {
|
|
public readonly TaskCallback<T> task;
|
|
public Completer completer;
|
|
|
|
internal _TaskEntry(TaskCallback<T> task, int priority) {
|
|
this.task = task;
|
|
this.priority = priority;
|
|
|
|
D.assert(() => {
|
|
debugStack = StackTraceUtility.ExtractStackTrace();
|
|
return true;
|
|
});
|
|
completer = Completer.create();
|
|
}
|
|
|
|
public int priority { get; }
|
|
public string debugStack { get; private set; }
|
|
|
|
public void run() {
|
|
if (!foundation_.kReleaseMode) {
|
|
completer.complete(FutureOr.value(task()));
|
|
}
|
|
else {
|
|
completer.complete(FutureOr.value(task()));
|
|
}
|
|
}
|
|
|
|
public int CompareTo(_TaskEntry other) {
|
|
return -priority.CompareTo(value: other.priority);
|
|
}
|
|
}
|
|
|
|
class _FrameCallbackEntry {
|
|
public static string debugCurrentCallbackStack;
|
|
|
|
public readonly FrameCallback callback;
|
|
public string debugStack;
|
|
|
|
internal _FrameCallbackEntry(FrameCallback callback, bool rescheduling = false) {
|
|
this.callback = callback;
|
|
|
|
D.assert(() => {
|
|
if (rescheduling) {
|
|
D.assert(() => {
|
|
if (debugCurrentCallbackStack == null) {
|
|
throw new UIWidgetsError(
|
|
new List<DiagnosticsNode> {
|
|
new ErrorSummary(
|
|
"scheduleFrameCallback called with rescheduling true, but no callback is in scope."),
|
|
new ErrorDescription(
|
|
"The \"rescheduling\" argument should only be set to true if the " +
|
|
"callback is being reregistered from within the callback itself, " +
|
|
"and only then if the callback itself is entirely synchronous."),
|
|
new ErrorHint("If this is the initial registration of the callback, or if the " +
|
|
"callback is asynchronous, then do not use the \"rescheduling\" " +
|
|
"argument.")
|
|
});
|
|
}
|
|
|
|
return true;
|
|
});
|
|
debugStack = debugCurrentCallbackStack;
|
|
}
|
|
else {
|
|
debugStack = StackTraceUtility.ExtractStackTrace();
|
|
}
|
|
|
|
return true;
|
|
});
|
|
}
|
|
}
|
|
|
|
public enum SchedulerPhase {
|
|
idle,
|
|
transientCallbacks,
|
|
midFrameMicrotasks,
|
|
persistentCallbacks,
|
|
postFrameCallbacks
|
|
}
|
|
|
|
public class SchedulerBinding : PaintingBinding {
|
|
readonly List<FrameCallback> _persistentCallbacks = new List<FrameCallback>();
|
|
|
|
readonly List<FrameCallback> _postFrameCallbacks = new List<FrameCallback>();
|
|
readonly HashSet<int> _removedIds = new HashSet<int>();
|
|
|
|
readonly PriorityQueue<_TaskEntry> _taskQueue = new PriorityQueue<_TaskEntry>();
|
|
|
|
readonly List<TimingsCallback> _timingsCallbacks = new List<TimingsCallback>();
|
|
|
|
TimeSpan? _currentFrameTimeStamp;
|
|
string _debugBanner;
|
|
|
|
int _debugFrameNumber;
|
|
TimeSpan _epochStart = TimeSpan.Zero;
|
|
|
|
TimeSpan? _firstRawTimeStampInEpoch;
|
|
|
|
bool _hasRequestedAnEventLoopCallback;
|
|
bool _ignoreNextEngineDrawFrame;
|
|
|
|
int _nextFrameCallbackId;
|
|
|
|
Completer _nextFrameCompleter;
|
|
Dictionary<int, _FrameCallbackEntry> _transientCallbacks = new Dictionary<int, _FrameCallbackEntry>();
|
|
|
|
bool _warmUpFrame;
|
|
|
|
|
|
public SchedulingStrategy schedulingStrategy = scheduler_.defaultSchedulingStrategy;
|
|
|
|
public new static SchedulerBinding instance {
|
|
get { return (SchedulerBinding) Window.instance._binding; }
|
|
private set { Window.instance._binding = value; }
|
|
}
|
|
|
|
public AppLifecycleState? lifecycleState { get; private set; }
|
|
|
|
public int transientCallbackCount {
|
|
get { return _transientCallbacks.Count; }
|
|
}
|
|
|
|
public Future endOfFrame {
|
|
get {
|
|
if (_nextFrameCompleter == null) {
|
|
if (schedulerPhase == SchedulerPhase.idle) {
|
|
scheduleFrame();
|
|
}
|
|
|
|
_nextFrameCompleter = Completer.create();
|
|
addPostFrameCallback(timeStamp => {
|
|
_nextFrameCompleter.complete();
|
|
_nextFrameCompleter = null;
|
|
});
|
|
}
|
|
|
|
return _nextFrameCompleter.future;
|
|
}
|
|
}
|
|
|
|
public bool hasScheduledFrame { get; private set; }
|
|
|
|
public SchedulerPhase schedulerPhase { get; private set; } = SchedulerPhase.idle;
|
|
|
|
public bool framesEnabled { get; private set; } = true;
|
|
|
|
public TimeSpan currentFrameTimeStamp {
|
|
get {
|
|
D.assert(_currentFrameTimeStamp != null);
|
|
return _currentFrameTimeStamp.Value;
|
|
}
|
|
}
|
|
|
|
public TimeSpan currentSystemFrameTimeStamp { get; private set; } = TimeSpan.Zero;
|
|
|
|
protected override void initInstances() {
|
|
base.initInstances();
|
|
instance = this;
|
|
|
|
//SystemChannels.lifecycle.setMessageHandler(_handleLifecycleMessage);
|
|
readInitialLifecycleStateFromNativeWindow();
|
|
|
|
if (!foundation_.kReleaseMode) {
|
|
var frameNumber = 0;
|
|
addTimingsCallback(timings => {
|
|
foreach (var frameTiming in timings) {
|
|
frameNumber += 1;
|
|
_profileFramePostEvent(frameNumber: frameNumber, frameTiming: frameTiming);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
public void addTimingsCallback(TimingsCallback callback) {
|
|
_timingsCallbacks.Add(item: callback);
|
|
if (_timingsCallbacks.Count == 1) {
|
|
D.assert(window.onReportTimings == null);
|
|
window.onReportTimings = _executeTimingsCallbacks;
|
|
}
|
|
|
|
D.assert(window.onReportTimings == _executeTimingsCallbacks);
|
|
}
|
|
|
|
public void removeTimingsCallback(TimingsCallback callback) {
|
|
D.assert(_timingsCallbacks.Contains(item: callback));
|
|
_timingsCallbacks.Remove(item: callback);
|
|
if (_timingsCallbacks.isEmpty()) {
|
|
window.onReportTimings = null;
|
|
}
|
|
}
|
|
|
|
void _executeTimingsCallbacks(List<FrameTiming> timings) {
|
|
var clonedCallbacks =
|
|
new List<TimingsCallback>(collection: _timingsCallbacks);
|
|
foreach (var callback in clonedCallbacks) {
|
|
try {
|
|
if (_timingsCallbacks.Contains(item: callback)) {
|
|
callback(timings: timings);
|
|
}
|
|
}
|
|
catch (Exception ex) {
|
|
InformationCollector collector = null;
|
|
D.assert(() => {
|
|
IEnumerable<DiagnosticsNode> infoCollect() {
|
|
yield return new DiagnosticsProperty<TimingsCallback>(
|
|
"The TimingsCallback that gets executed was",
|
|
value: callback,
|
|
style: DiagnosticsTreeStyle.errorProperty);
|
|
}
|
|
|
|
collector = infoCollect;
|
|
return true;
|
|
});
|
|
|
|
UIWidgetsError.reportError(new UIWidgetsErrorDetails(
|
|
exception: ex,
|
|
context: new ErrorDescription("while executing callbacks for FrameTiming"),
|
|
informationCollector: collector
|
|
));
|
|
}
|
|
}
|
|
}
|
|
|
|
protected void readInitialLifecycleStateFromNativeWindow() {
|
|
if (lifecycleState == null) {
|
|
handleAppLifecycleStateChanged(_parseAppLifecycleMessage(message: window.initialLifecycleState));
|
|
}
|
|
}
|
|
|
|
protected virtual void handleAppLifecycleStateChanged(AppLifecycleState state) {
|
|
lifecycleState = state;
|
|
switch (state) {
|
|
case AppLifecycleState.resumed:
|
|
case AppLifecycleState.inactive:
|
|
_setFramesEnabledState(true);
|
|
break;
|
|
case AppLifecycleState.paused:
|
|
case AppLifecycleState.detached:
|
|
_setFramesEnabledState(false);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static AppLifecycleState _parseAppLifecycleMessage(string message) {
|
|
switch (message) {
|
|
case "AppLifecycleState.paused":
|
|
return AppLifecycleState.paused;
|
|
case "AppLifecycleState.resumed":
|
|
return AppLifecycleState.resumed;
|
|
case "AppLifecycleState.inactive":
|
|
return AppLifecycleState.inactive;
|
|
case "AppLifecycleState.detached":
|
|
return AppLifecycleState.detached;
|
|
}
|
|
|
|
throw new Exception("unknown AppLifecycleState: " + message);
|
|
}
|
|
|
|
public Future scheduleTask<T>(
|
|
TaskCallback<T> task,
|
|
Priority priority) {
|
|
var isFirstTask = _taskQueue.isEmpty;
|
|
var entry = new _TaskEntry<T>(
|
|
task: task,
|
|
priority: priority.value
|
|
);
|
|
_taskQueue.enqueue(item: entry);
|
|
if (isFirstTask && !locked) {
|
|
_ensureEventLoopCallback();
|
|
}
|
|
|
|
return entry.completer.future;
|
|
}
|
|
|
|
|
|
protected override void unlocked() {
|
|
base.unlocked();
|
|
if (_taskQueue.isNotEmpty) {
|
|
_ensureEventLoopCallback();
|
|
}
|
|
}
|
|
|
|
void _ensureEventLoopCallback() {
|
|
D.assert(result: !locked);
|
|
D.assert(_taskQueue.count != 0);
|
|
if (_hasRequestedAnEventLoopCallback) {
|
|
return;
|
|
}
|
|
|
|
_hasRequestedAnEventLoopCallback = true;
|
|
Timer.run(callback: _runTasks);
|
|
}
|
|
|
|
object _runTasks() {
|
|
_hasRequestedAnEventLoopCallback = false;
|
|
if (handleEventLoopCallback()) {
|
|
_ensureEventLoopCallback(); // runs next task when there's time
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
bool handleEventLoopCallback() {
|
|
if (_taskQueue.isEmpty || locked) {
|
|
return false;
|
|
}
|
|
|
|
var entry = _taskQueue.first;
|
|
if (schedulingStrategy(priority: entry.priority, this)) {
|
|
try {
|
|
_taskQueue.removeFirst();
|
|
entry.run();
|
|
}
|
|
catch (Exception exception) {
|
|
string callbackStack = null;
|
|
D.assert(() => {
|
|
callbackStack = entry.debugStack;
|
|
return true;
|
|
});
|
|
|
|
IEnumerable<DiagnosticsNode> infoCollector() {
|
|
yield return DiagnosticsNode.message(
|
|
"\nThis exception was thrown in the context of a scheduler callback. " +
|
|
"When the scheduler callback was _registered_ (as opposed to when the " +
|
|
"exception was thrown), this was the stack: " + callbackStack);
|
|
}
|
|
|
|
UIWidgetsError.reportError(new UIWidgetsErrorDetails(
|
|
exception: exception,
|
|
"scheduler library",
|
|
new ErrorDescription("during a task callback"),
|
|
informationCollector: callbackStack == null
|
|
? (InformationCollector) null
|
|
: infoCollector
|
|
));
|
|
}
|
|
|
|
return _taskQueue.isNotEmpty;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public int scheduleFrameCallback(FrameCallback callback, bool rescheduling = false) {
|
|
scheduleFrame();
|
|
_nextFrameCallbackId += 1;
|
|
_transientCallbacks[key: _nextFrameCallbackId] =
|
|
new _FrameCallbackEntry(callback: callback, rescheduling: rescheduling);
|
|
return _nextFrameCallbackId;
|
|
}
|
|
|
|
public void cancelFrameCallbackWithId(int id) {
|
|
D.assert(id > 0);
|
|
_transientCallbacks.Remove(key: id);
|
|
_removedIds.Add(item: id);
|
|
}
|
|
|
|
public void addPersistentFrameCallback(FrameCallback callback) {
|
|
_persistentCallbacks.Add(item: callback);
|
|
}
|
|
|
|
public void addPostFrameCallback(FrameCallback callback) {
|
|
_postFrameCallbacks.Add(item: callback);
|
|
}
|
|
|
|
void _setFramesEnabledState(bool enabled) {
|
|
if (framesEnabled == enabled) {
|
|
return;
|
|
}
|
|
|
|
framesEnabled = enabled;
|
|
if (enabled) {
|
|
scheduleFrame();
|
|
}
|
|
}
|
|
|
|
protected void ensureFrameCallbacksRegistered() {
|
|
window.onBeginFrame = window.onBeginFrame ?? _handleBeginFrame;
|
|
window.onDrawFrame = window.onDrawFrame ?? _handleDrawFrame;
|
|
}
|
|
|
|
public void ensureVisualUpdate() {
|
|
switch (schedulerPhase) {
|
|
case SchedulerPhase.idle:
|
|
case SchedulerPhase.postFrameCallbacks:
|
|
scheduleFrame();
|
|
return;
|
|
case SchedulerPhase.transientCallbacks:
|
|
case SchedulerPhase.midFrameMicrotasks:
|
|
case SchedulerPhase.persistentCallbacks:
|
|
return;
|
|
}
|
|
}
|
|
|
|
static readonly TimeSpan _coolDownDelay = new TimeSpan(0, 0, 0, 0, 200);
|
|
|
|
#pragma warning disable CS0414
|
|
static Timer frameCoolDownTimer = null;
|
|
#pragma warning restore CS0414
|
|
|
|
public void scheduleFrame() {
|
|
if (hasScheduledFrame || !framesEnabled) {
|
|
return;
|
|
}
|
|
|
|
D.assert(() => {
|
|
if (scheduler_.debugPrintScheduleFrameStacks) {
|
|
Debug.LogFormat("scheduleFrame() called. Current phase is {0}.", schedulerPhase);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
|
|
ensureFrameCallbacksRegistered();
|
|
Window.instance.scheduleFrame();
|
|
hasScheduledFrame = true;
|
|
|
|
#if !UNITY_EDITOR
|
|
adjustFrameRate();
|
|
#endif
|
|
}
|
|
|
|
#if !UNITY_EDITOR
|
|
void adjustFrameRate() {
|
|
if (!UIWidgetsGlobalConfiguration.EnableAutoAdjustFramerate) {
|
|
return;
|
|
}
|
|
|
|
onFrameRateSpeedUp();
|
|
frameCoolDownTimer?.cancel();
|
|
frameCoolDownTimer = Timer.create(_coolDownDelay,
|
|
() => {
|
|
onFrameRateCoolDown();
|
|
frameCoolDownTimer = null;
|
|
}
|
|
);
|
|
}
|
|
|
|
const int defaultMaxRenderFrameInterval = 200;
|
|
const int defaultMinRenderFrameInterval = 1;
|
|
|
|
void onFrameRateSpeedUp() {
|
|
OnDemandRendering.renderFrameInterval = defaultMinRenderFrameInterval;
|
|
}
|
|
|
|
void onFrameRateCoolDown() {
|
|
OnDemandRendering.renderFrameInterval = defaultMaxRenderFrameInterval;
|
|
}
|
|
#endif
|
|
|
|
public void scheduleForcedFrame() {
|
|
if (!framesEnabled) {
|
|
return;
|
|
}
|
|
|
|
if (hasScheduledFrame) {
|
|
return;
|
|
}
|
|
|
|
D.assert(() => {
|
|
if (scheduler_.debugPrintScheduleFrameStacks) {
|
|
Debug.LogFormat("scheduleForcedFrame() called. Current phase is {0}.", schedulerPhase);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
|
|
ensureFrameCallbacksRegistered();
|
|
Window.instance.scheduleFrame();
|
|
hasScheduledFrame = true;
|
|
|
|
#if !UNITY_EDITOR
|
|
adjustFrameRate();
|
|
#endif
|
|
}
|
|
|
|
public void scheduleWarmUpFrame() {
|
|
if (_warmUpFrame || schedulerPhase != SchedulerPhase.idle) {
|
|
return;
|
|
}
|
|
|
|
_warmUpFrame = true;
|
|
Timeline.startSync("Warm-up frame");
|
|
|
|
var hadScheduledFrame = hasScheduledFrame;
|
|
// We use timers here to ensure that microtasks flush in between.
|
|
Timer.run(() => {
|
|
D.assert(result: _warmUpFrame);
|
|
handleBeginFrame(null);
|
|
return null;
|
|
});
|
|
Timer.run(() => {
|
|
D.assert(result: _warmUpFrame);
|
|
handleDrawFrame();
|
|
// We call resetEpoch after this frame so that, in the hot reload case,
|
|
// the very next frame pretends to have occurred immediately after this
|
|
// warm-up frame. The warm-up frame's timestamp will typically be far in
|
|
// the past (the time of the last real frame), so if we didn't reset the
|
|
// epoch we would see a sudden jump from the old time in the warm-up frame
|
|
// to the new time in the "real" frame. The biggest problem with this is
|
|
// that implicit animations end up being triggered at the old time and
|
|
// then skipping every frame and finishing in the new time.
|
|
resetEpoch();
|
|
_warmUpFrame = false;
|
|
if (hadScheduledFrame) {
|
|
scheduleFrame();
|
|
}
|
|
|
|
return null;
|
|
});
|
|
|
|
// Lock events so touch events etc don't insert themselves until the
|
|
// scheduled frame has finished.
|
|
lockEvents(() => endOfFrame.then(v => {
|
|
Timeline.finishSync();
|
|
return FutureOr.nil;
|
|
}));
|
|
}
|
|
|
|
public void resetEpoch() {
|
|
_epochStart = _adjustForEpoch(rawTimeStamp: currentSystemFrameTimeStamp);
|
|
_firstRawTimeStampInEpoch = null;
|
|
}
|
|
|
|
TimeSpan _adjustForEpoch(TimeSpan rawTimeStamp) {
|
|
var rawDurationSinceEpoch = _firstRawTimeStampInEpoch == null
|
|
? TimeSpan.Zero
|
|
: rawTimeStamp - _firstRawTimeStampInEpoch.Value;
|
|
return new TimeSpan((long) (rawDurationSinceEpoch.Ticks / scheduler_.timeDilation) +
|
|
_epochStart.Ticks);
|
|
}
|
|
|
|
void _handleBeginFrame(TimeSpan rawTimeStamp) {
|
|
if (_warmUpFrame) {
|
|
D.assert(result: !_ignoreNextEngineDrawFrame);
|
|
_ignoreNextEngineDrawFrame = true;
|
|
return;
|
|
}
|
|
|
|
handleBeginFrame(rawTimeStamp: rawTimeStamp);
|
|
}
|
|
|
|
void _handleDrawFrame() {
|
|
if (_ignoreNextEngineDrawFrame) {
|
|
_ignoreNextEngineDrawFrame = false;
|
|
return;
|
|
}
|
|
|
|
handleDrawFrame();
|
|
}
|
|
|
|
public void handleBeginFrame(TimeSpan? rawTimeStamp) {
|
|
Timeline.startSync("Frame");
|
|
_firstRawTimeStampInEpoch = _firstRawTimeStampInEpoch ?? rawTimeStamp;
|
|
_currentFrameTimeStamp = _adjustForEpoch(rawTimeStamp ?? currentSystemFrameTimeStamp);
|
|
|
|
if (rawTimeStamp != null) {
|
|
currentSystemFrameTimeStamp = rawTimeStamp.Value;
|
|
}
|
|
|
|
D.assert(() => {
|
|
_debugFrameNumber += 1;
|
|
|
|
if (scheduler_.debugPrintBeginFrameBanner || scheduler_.debugPrintEndFrameBanner) {
|
|
var frameTimeStampDescription = new StringBuilder();
|
|
if (rawTimeStamp != null) {
|
|
_debugDescribeTimeStamp(timeStamp: _currentFrameTimeStamp.Value,
|
|
buffer: frameTimeStampDescription);
|
|
}
|
|
else {
|
|
frameTimeStampDescription.Append("(warm-up frame)");
|
|
}
|
|
|
|
_debugBanner =
|
|
$"▄▄▄▄▄▄▄▄ Frame {_debugFrameNumber.ToString().PadRight(7)} ${frameTimeStampDescription.ToString().PadLeft(18)} ▄▄▄▄▄▄▄▄";
|
|
if (scheduler_.debugPrintBeginFrameBanner) {
|
|
Debug.Log(message: _debugBanner);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
});
|
|
|
|
D.assert(schedulerPhase == SchedulerPhase.idle);
|
|
hasScheduledFrame = false;
|
|
|
|
try {
|
|
Timeline.startSync("Animate");
|
|
schedulerPhase = SchedulerPhase.transientCallbacks;
|
|
var callbacks = _transientCallbacks;
|
|
_transientCallbacks = new Dictionary<int, _FrameCallbackEntry>();
|
|
foreach (var entry in callbacks) {
|
|
if (!_removedIds.Contains(item: entry.Key)) {
|
|
_invokeFrameCallback(
|
|
callback: entry.Value.callback, timeStamp: _currentFrameTimeStamp.Value,
|
|
callbackStack: entry.Value.debugStack);
|
|
}
|
|
}
|
|
|
|
_removedIds.Clear();
|
|
}
|
|
finally {
|
|
schedulerPhase = SchedulerPhase.midFrameMicrotasks;
|
|
}
|
|
}
|
|
|
|
public void handleDrawFrame() {
|
|
|
|
D.assert(schedulerPhase == SchedulerPhase.midFrameMicrotasks);
|
|
Timeline.finishSync();
|
|
try {
|
|
schedulerPhase = SchedulerPhase.persistentCallbacks;
|
|
foreach (var callback in _persistentCallbacks) {
|
|
_invokeFrameCallback(callback: callback, timeStamp: _currentFrameTimeStamp.Value);
|
|
}
|
|
|
|
schedulerPhase = SchedulerPhase.postFrameCallbacks;
|
|
var localPostFrameCallbacks = new List<FrameCallback>(collection: _postFrameCallbacks);
|
|
_postFrameCallbacks.Clear();
|
|
foreach (var callback in localPostFrameCallbacks) {
|
|
_invokeFrameCallback(callback: callback, timeStamp: _currentFrameTimeStamp.Value);
|
|
}
|
|
}
|
|
finally {
|
|
schedulerPhase = SchedulerPhase.idle;
|
|
D.assert(() => {
|
|
if (scheduler_.debugPrintEndFrameBanner) {
|
|
Debug.Log(new string('▀', count: _debugBanner.Length));
|
|
}
|
|
|
|
_debugBanner = null;
|
|
return true;
|
|
});
|
|
_currentFrameTimeStamp = null;
|
|
}
|
|
}
|
|
|
|
void _profileFramePostEvent(int frameNumber, FrameTiming frameTiming) {
|
|
developer_.postEvent("Flutter.Frame", new Hashtable {
|
|
{"number", frameNumber},
|
|
{"startTime", frameTiming.timestampInMicroseconds(phase: FramePhase.buildStart)},
|
|
{"elapsed", (int) (frameTiming.totalSpan.TotalMilliseconds * 1000)},
|
|
{"build", (int) (frameTiming.buildDuration.TotalMilliseconds * 1000)},
|
|
{"raster", (int) (frameTiming.rasterDuration.TotalMilliseconds * 1000)}
|
|
});
|
|
}
|
|
|
|
|
|
static void _debugDescribeTimeStamp(TimeSpan timeStamp, StringBuilder buffer) {
|
|
if (timeStamp.TotalDays > 0) {
|
|
buffer.AppendFormat("{0}d ", arg0: timeStamp.Days);
|
|
}
|
|
|
|
if (timeStamp.TotalHours > 0) {
|
|
buffer.AppendFormat("{0}h ", arg0: timeStamp.Hours);
|
|
}
|
|
|
|
if (timeStamp.TotalMinutes > 0) {
|
|
buffer.AppendFormat("{0}m ", arg0: timeStamp.Minutes);
|
|
}
|
|
|
|
if (timeStamp.TotalSeconds > 0) {
|
|
buffer.AppendFormat("{0}s ", arg0: timeStamp.Seconds);
|
|
}
|
|
|
|
buffer.AppendFormat("{0}", arg0: timeStamp.Milliseconds);
|
|
|
|
var microseconds = (int) (timeStamp.Ticks % 10000 / 10);
|
|
if (microseconds > 0) {
|
|
buffer.AppendFormat(".{0}", microseconds.ToString().PadLeft(3, '0'));
|
|
}
|
|
|
|
buffer.Append("ms");
|
|
}
|
|
|
|
void _invokeFrameCallback(FrameCallback callback, TimeSpan timeStamp, string callbackStack = null) {
|
|
D.assert(callback != null);
|
|
D.assert(_FrameCallbackEntry.debugCurrentCallbackStack == null);
|
|
D.assert(() => {
|
|
_FrameCallbackEntry.debugCurrentCallbackStack = callbackStack;
|
|
return true;
|
|
});
|
|
|
|
try {
|
|
callback(timeStamp: timeStamp);
|
|
}
|
|
catch (Exception ex) {
|
|
IEnumerable<DiagnosticsNode> infoCollector() {
|
|
yield return DiagnosticsNode.message(
|
|
"\nThis exception was thrown in the context of a scheduler callback. " +
|
|
"When the scheduler callback was _registered_ (as opposed to when the " +
|
|
"exception was thrown), this was the stack:");
|
|
var builder = new StringBuilder();
|
|
foreach (var line in UIWidgetsError.defaultStackFilter(
|
|
callbackStack.TrimEnd().Split('\n'))) {
|
|
builder.AppendLine(value: line);
|
|
}
|
|
|
|
yield return DiagnosticsNode.message(builder.ToString());
|
|
}
|
|
|
|
UIWidgetsError.reportError(new UIWidgetsErrorDetails(
|
|
exception: ex,
|
|
"scheduler library",
|
|
new ErrorDescription("during a scheduler callback"),
|
|
informationCollector: callbackStack == null
|
|
? (InformationCollector) null
|
|
: infoCollector
|
|
));
|
|
}
|
|
|
|
D.assert(() => {
|
|
_FrameCallbackEntry.debugCurrentCallbackStack = null;
|
|
return true;
|
|
});
|
|
}
|
|
}
|
|
|
|
public static partial class scheduler_ {
|
|
public static bool defaultSchedulingStrategy(int priority, SchedulerBinding scheduler) {
|
|
if (scheduler.transientCallbackCount > 0) {
|
|
return priority >= Priority.animation.value;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
}
|