From a047ee1fdb7631b6aac63976890eb01a508b223b Mon Sep 17 00:00:00 2001 From: ZeroLing Date: Thu, 31 Mar 2022 18:44:40 +0800 Subject: [PATCH 1/4] feat: add idle request callback --- kraken/lib/src/dom/background_tasks.dart | 190 +++++++++++++++++++++++ kraken/lib/src/dom/window.dart | 4 +- 2 files changed, 193 insertions(+), 1 deletion(-) create mode 100644 kraken/lib/src/dom/background_tasks.dart diff --git a/kraken/lib/src/dom/background_tasks.dart b/kraken/lib/src/dom/background_tasks.dart new file mode 100644 index 0000000000..443adabd70 --- /dev/null +++ b/kraken/lib/src/dom/background_tasks.dart @@ -0,0 +1,190 @@ +/* + * Copyright (C) 2022-present The Kraken authors. All rights reserved. + */ + +import 'dart:async'; + +import 'package:flutter/foundation.dart'; +import 'package:flutter/scheduler.dart'; + +typedef IdleRequestCallback = void Function(IdleDeadline deadline); + +class IdleDeadline { + IdleDeadline._(double time) : _time = time; + + // Each IdleDeadline has an associated time which holds a DOMHighResTimeStamp representing the absolute time in + // milliseconds of the deadline. This must be populated when the IdleDeadline is created. + final double _time; + + // Each IdleDeadline has an associated timeout, which is initially false. + bool _timeout = false; + + // The didTimeout getter must return timeout. + bool get didTimeout => _timeout; + + // When the timeRemaining() method is invoked on an IdleDeadline object it must return the duration, + // as a DOMHighResTimeStamp, between the current time and the time associated with the IdleDeadline object. + // The value should be accurate to 5 microseconds - see "Privacy and Security" section of [HR-TIME]. + // This value is calculated by performing the following steps: + // + // 1. Let now be a DOMHighResTimeStamp representing current high resolution time in milliseconds. + // 2. Let deadline be the time associated with the IdleDeadline object. + // 3. Let timeRemaining be deadline - now. + // 4. If timeRemaining is negative, set it to 0. + // 5. Return timeRemaining. + double timeRemaining() { + double currentTime = _currentTime(); + return _time - currentTime; + } +} + +class IdleRequestOptions { + IdleRequestOptions(this.timeout); + int? timeout; +} + +// https://www.w3.org/TR/requestidlecallback +mixin ScheduleBackgroundTasks { + // The list must be initially empty and each entry in this list is identified by a number, which must + // be unique within the list for the lifetime of the Window object. + final Map _idleRequestCallbacks = {}; + // A list of runnable idle callbacks. The list must be initially empty and each entry in this list is + // identified by a number, which must be unique within the list of the lifetime of the Window object. + final Map _runnableIdleCallback = {}; + // An idle callback identifier, which is a number which must initially be zero. + int _idleCallbackIdentifier = 0; + // A last idle period deadline, which is a [DOMHighResTimeStamp] which must initially be zero. + double _lastIdlePeriodDeadline = 0; + + // https://www.w3.org/TR/requestidlecallback/#dom-window-requestidlecallback + int requestIdleCallback(IdleRequestCallback callback, [IdleRequestOptions? options]) { + final int id = ++_idleCallbackIdentifier; + + final startIdlePeriod = _idleRequestCallbacks.isEmpty && _runnableIdleCallback.isEmpty; + _idleRequestCallbacks[id] = callback; + if (startIdlePeriod) { + _queueIdleTask(_startIdlePeriod); + } + + final int? timeout = options?.timeout; + if (timeout != null && timeout > 0) { + Timer(Duration(milliseconds: timeout), () { + _queueIdleTask(() { + _invokeIdleCallbackTimeout(id); + }); + }); + } + + return id; + } + + // https://www.w3.org/TR/requestidlecallback/#dom-window-cancelidlecallback + // 1. Let window be this Window object. + // 2. Find the entry in either the window's list of idle request callbacks or list of runnable + // idle callbacks that is associated with the value handle. + // 3. If there is such an entry, remove it from both window's list of idle request + // callbacks and the list of runnable idle callbacks. + void cancelIdleCallback(int handle) { + IdleRequestCallback? entry = _idleRequestCallbacks.remove(handle); + if (entry == null) { + _runnableIdleCallback.remove(handle); + } + } + + // https://www.w3.org/TR/requestidlecallback/#dfn-start-an-idle-period-algorithm + void _startIdlePeriod() async { + double lastDeadline = _lastIdlePeriodDeadline; + double now = _currentTime(); + if (lastDeadline > now) { + await _wait((lastDeadline - now).floor()); + } + await _waitUntilNextMicrotask(); + now = _currentTime(); + double deadline = _expectedNextDeadline; + if (deadline - now > 50) { + deadline = now + 50; + } + Map pendingList = _idleRequestCallbacks; + Map runList = _runnableIdleCallback; + runList.addAll(pendingList); + pendingList.clear(); + _queueIdleTask(() { + _invokeIdleCallback(deadline); + }); + _lastIdlePeriodDeadline = deadline; + } + + // The user agent should choose deadline to ensure that no time-critical tasks will be delayed + // even if a callback runs for the whole time period from now to deadline. As such, it should + // be set to the minimum of: the closest timeout in the list of active timers as set via setTimeout + // and setInterval; the scheduled runtime for pending animation callbacks posted via requestAnimationFrame; + // pending internal timeouts such as deadlines to start rendering the next frame, process audio + // or any other internal task the user agent deems important. + double get _expectedNextDeadline { + // @TODO: Only supported 60fps. + return SchedulerBinding.instance!.currentFrameTimeStamp.inMicroseconds + 1000 / 60; + } + + void _queueIdleTask(VoidCallback task) { + SchedulerBinding.instance!.scheduleTask(task, Priority.idle); + } + + // https://www.w3.org/TR/requestidlecallback/#dfn-invoke-idle-callbacks-algorithm + void _invokeIdleCallback(double deadline) { + double now = _currentTime(); + if (now < deadline && _runnableIdleCallback.isNotEmpty) { + int first = _runnableIdleCallback.keys.first; + IdleRequestCallback callback = _runnableIdleCallback.remove(first)!; + IdleDeadline idleDeadline = IdleDeadline._(deadline) + .._timeout = false; + _invokeCallback(callback, idleDeadline); + if (_runnableIdleCallback.isNotEmpty) { + Timer.run(() { + _invokeIdleCallback(deadline); + }); + } + } else { + if (_idleRequestCallbacks.isNotEmpty || _runnableIdleCallback.isNotEmpty) { + Timer.run(_startIdlePeriod); + } + } + } + + // https://www.w3.org/TR/requestidlecallback/#dfn-invoke-idle-callback-timeout-algorithm + void _invokeIdleCallbackTimeout(int id) { + IdleRequestCallback? callback = _idleRequestCallbacks[id] ?? _runnableIdleCallback[id]; + if (callback != null) { + _idleRequestCallbacks.remove(id); + _runnableIdleCallback.remove(id); + double now = _currentTime(); + IdleDeadline deadline = IdleDeadline._(now) + .._timeout = true; + _invokeCallback(callback, deadline); + } + } + + void _invokeCallback(IdleRequestCallback callback, IdleDeadline idleDeadline) { + try { + callback(idleDeadline); + } catch (exception, exceptionStack) { + FlutterError.reportError(FlutterErrorDetails( + exception: exception, + stack: exceptionStack, + library: 'scheduler library', + context: ErrorDescription('during a task callback'), + )); + } + } +} + +Future _wait(int millisecond) { + return Future.delayed(Duration(milliseconds: millisecond)); +} + +FutureOr _waitUntilNextMicrotask() { + return Future.microtask(() => null); +} + +double _currentTime() { + return DateTime.now().microsecond / 1000; +} diff --git a/kraken/lib/src/dom/window.dart b/kraken/lib/src/dom/window.dart index 38a67b44de..11075938f6 100644 --- a/kraken/lib/src/dom/window.dart +++ b/kraken/lib/src/dom/window.dart @@ -9,9 +9,11 @@ import 'package:kraken/launcher.dart'; import 'package:kraken/module.dart'; import 'package:kraken/bridge.dart'; +import 'background_tasks.dart'; + const String WINDOW = 'WINDOW'; -class Window extends EventTarget { +class Window extends EventTarget with ScheduleBackgroundTasks { final Document document; @override From d1da9fe880cc4def27f77c84aca9cc91675d4685 Mon Sep 17 00:00:00 2001 From: ZeroLing Date: Thu, 31 Mar 2022 18:48:26 +0800 Subject: [PATCH 2/4] refactor: add frame duration --- kraken/lib/src/dom/background_tasks.dart | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/kraken/lib/src/dom/background_tasks.dart b/kraken/lib/src/dom/background_tasks.dart index 443adabd70..5effa3f359 100644 --- a/kraken/lib/src/dom/background_tasks.dart +++ b/kraken/lib/src/dom/background_tasks.dart @@ -45,6 +45,11 @@ class IdleRequestOptions { // https://www.w3.org/TR/requestidlecallback mixin ScheduleBackgroundTasks { + // @TODO: Current only supported 60fps. + // Initial frame duration is 60fps, after negotiation it should be the time between each frame. + // ignore: prefer_final_fields + double _frameDuration = 1000 / 60; + // The list must be initially empty and each entry in this list is identified by a number, which must // be unique within the list for the lifetime of the Window object. final Map _idleRequestCallbacks = {}; @@ -121,8 +126,7 @@ mixin ScheduleBackgroundTasks { // pending internal timeouts such as deadlines to start rendering the next frame, process audio // or any other internal task the user agent deems important. double get _expectedNextDeadline { - // @TODO: Only supported 60fps. - return SchedulerBinding.instance!.currentFrameTimeStamp.inMicroseconds + 1000 / 60; + return SchedulerBinding.instance!.currentFrameTimeStamp.inMicroseconds + _frameDuration; } void _queueIdleTask(VoidCallback task) { From df406384c4eee5e0c303d0e4cccfe520f7c2f4b2 Mon Sep 17 00:00:00 2001 From: ZeroLing Date: Thu, 31 Mar 2022 19:04:03 +0800 Subject: [PATCH 3/4] fix: currentFrameTimeStamp may be null --- kraken/lib/src/dom/background_tasks.dart | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kraken/lib/src/dom/background_tasks.dart b/kraken/lib/src/dom/background_tasks.dart index 5effa3f359..e5a4bb1e96 100644 --- a/kraken/lib/src/dom/background_tasks.dart +++ b/kraken/lib/src/dom/background_tasks.dart @@ -126,7 +126,7 @@ mixin ScheduleBackgroundTasks { // pending internal timeouts such as deadlines to start rendering the next frame, process audio // or any other internal task the user agent deems important. double get _expectedNextDeadline { - return SchedulerBinding.instance!.currentFrameTimeStamp.inMicroseconds + _frameDuration; + return SchedulerBinding.instance!.currentSystemFrameTimeStamp.inMicroseconds + _frameDuration; } void _queueIdleTask(VoidCallback task) { From 667f439de058652b52ee788997744bd94ed19cab Mon Sep 17 00:00:00 2001 From: ZeroLing Date: Fri, 1 Apr 2022 17:33:41 +0800 Subject: [PATCH 4/4] refator: frame --- kraken/lib/src/dom/background_tasks.dart | 46 ++++++++++++++++++++++-- 1 file changed, 43 insertions(+), 3 deletions(-) diff --git a/kraken/lib/src/dom/background_tasks.dart b/kraken/lib/src/dom/background_tasks.dart index e5a4bb1e96..efa38f5dea 100644 --- a/kraken/lib/src/dom/background_tasks.dart +++ b/kraken/lib/src/dom/background_tasks.dart @@ -3,7 +3,10 @@ */ import 'dart:async'; +import 'dart:collection'; +import 'dart:ui'; +import 'package:flutter/cupertino.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/scheduler.dart'; @@ -43,12 +46,15 @@ class IdleRequestOptions { int? timeout; } +const double _DEFAULT_FRAME_DURATION = 1000 / 60; + // https://www.w3.org/TR/requestidlecallback mixin ScheduleBackgroundTasks { // @TODO: Current only supported 60fps. // Initial frame duration is 60fps, after negotiation it should be the time between each frame. // ignore: prefer_final_fields - double _frameDuration = 1000 / 60; + double _frameDuration = _DEFAULT_FRAME_DURATION; + int _frameCount = 0; // The list must be initially empty and each entry in this list is identified by a number, which must // be unique within the list for the lifetime of the Window object. @@ -108,7 +114,10 @@ mixin ScheduleBackgroundTasks { double deadline = _expectedNextDeadline; if (deadline - now > 50) { deadline = now + 50; + } else if (deadline < now) { + deadline = now + 1; } + assert(deadline > now, 'deadline should greater than now.'); Map pendingList = _idleRequestCallbacks; Map runList = _runnableIdleCallback; runList.addAll(pendingList); @@ -126,9 +135,40 @@ mixin ScheduleBackgroundTasks { // pending internal timeouts such as deadlines to start rendering the next frame, process audio // or any other internal task the user agent deems important. double get _expectedNextDeadline { - return SchedulerBinding.instance!.currentSystemFrameTimeStamp.inMicroseconds + _frameDuration; + _ensureBeginFrameHooked(); + if (_diffBetweenEpochTimeStamp == 0) { + return _currentTime() + _frameDuration; + } else { + return _currentFrameBegin + _diffBetweenEpochTimeStamp + _frameDuration; + } } + double _diffBetweenEpochTimeStamp = 0; + static double get _currentFrameBegin => SchedulerBinding.instance!.currentSystemFrameTimeStamp.inMicroseconds / Duration.microsecondsPerMillisecond; + + bool _beginFrameHooked = false; + void _ensureBeginFrameHooked() { + if (!_beginFrameHooked) { + _beginFrameHooked = true; + FrameCallback? prevOnBeginFrame = window.onBeginFrame; + window.onBeginFrame = (Duration duration) { + var now = _currentTime(); + _diffBetweenEpochTimeStamp = now - _currentFrameBegin; + var period = now - _lastBeginFrameTime; + if (period < _DEFAULT_FRAME_DURATION) { + _frameCount++; + _frameDuration = (_frameDuration * (_frameCount - 1) + period) / _frameCount; + } + _lastBeginFrameTime = now; + if (prevOnBeginFrame != null) { + prevOnBeginFrame(duration); + } + }; + } + } + + double _lastBeginFrameTime = 0; + void _queueIdleTask(VoidCallback task) { SchedulerBinding.instance!.scheduleTask(task, Priority.idle); } @@ -190,5 +230,5 @@ FutureOr _waitUntilNextMicrotask() { } double _currentTime() { - return DateTime.now().microsecond / 1000; + return DateTime.now().microsecondsSinceEpoch / 1000; }