diff --git a/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/JavaTimerManager.java b/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/JavaTimerManager.java deleted file mode 100644 index c155e7614dd495..00000000000000 --- a/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/JavaTimerManager.java +++ /dev/null @@ -1,428 +0,0 @@ -/* - * Copyright (c) Meta Platforms, Inc. and affiliates. - * - * This source code is licensed under the MIT license found in the - * LICENSE file in the root directory of this source tree. - */ - -package com.facebook.react.modules.core; - -import android.util.SparseArray; -import android.view.Choreographer; -import androidx.annotation.Nullable; -import com.facebook.proguard.annotations.DoNotStrip; -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.LifecycleEventListener; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.UiThreadUtil; -import com.facebook.react.bridge.WritableArray; -import com.facebook.react.common.SystemClock; -import com.facebook.react.devsupport.interfaces.DevSupportManager; -import com.facebook.react.jstasks.HeadlessJsTaskContext; -import com.facebook.react.jstasks.HeadlessJsTaskEventListener; -import java.util.Comparator; -import java.util.PriorityQueue; -import java.util.concurrent.atomic.AtomicBoolean; - -/** - * This class is the native implementation for JS timer execution on Android. It schedules JS timers - * to be invoked on frame boundaries using {@link ReactChoreographer}. - * - *

This is used by the NativeModule {@link TimingModule}. - */ -public class JavaTimerManager implements LifecycleEventListener, HeadlessJsTaskEventListener { - - // These timing constants should be kept in sync with the ones in `JSTimers.js`. - // The minimum time in milliseconds left in the frame to call idle callbacks. - private static final float IDLE_CALLBACK_FRAME_DEADLINE_MS = 1.f; - // The total duration of a frame in milliseconds, this assumes that devices run at 60 fps. - // TODO: Lower frame duration on devices that are too slow to run consistently - // at 60 fps. - private static final float FRAME_DURATION_MS = 1000.f / 60.f; - - private static class Timer { - private final int mCallbackID; - private final boolean mRepeat; - private final int mInterval; - private long mTargetTime; - - private Timer(int callbackID, long initialTargetTime, int duration, boolean repeat) { - mCallbackID = callbackID; - mTargetTime = initialTargetTime; - mInterval = duration; - mRepeat = repeat; - } - } - - private class TimerFrameCallback implements Choreographer.FrameCallback { - - // Temporary map for constructing the individual arrays of timers to call - private @Nullable WritableArray mTimersToCall = null; - - /** Calls all timers that have expired since the last time this frame callback was called. */ - @Override - public void doFrame(long frameTimeNanos) { - if (isPaused.get() && !isRunningTasks.get()) { - return; - } - - long frameTimeMillis = frameTimeNanos / 1000000; - synchronized (mTimerGuard) { - while (!mTimers.isEmpty() && mTimers.peek().mTargetTime < frameTimeMillis) { - Timer timer = mTimers.poll(); - if (mTimersToCall == null) { - mTimersToCall = Arguments.createArray(); - } - mTimersToCall.pushInt(timer.mCallbackID); - if (timer.mRepeat) { - timer.mTargetTime = frameTimeMillis + timer.mInterval; - mTimers.add(timer); - } else { - mTimerIdsToTimers.remove(timer.mCallbackID); - } - } - } - - if (mTimersToCall != null) { - mJavaScriptTimerExecutor.callTimers(mTimersToCall); - mTimersToCall = null; - } - - mReactChoreographer.postFrameCallback(ReactChoreographer.CallbackType.TIMERS_EVENTS, this); - } - } - - private class IdleFrameCallback implements Choreographer.FrameCallback { - - @Override - public void doFrame(long frameTimeNanos) { - if (isPaused.get() && !isRunningTasks.get()) { - return; - } - - // If the JS thread is busy for multiple frames we cancel any other pending runnable. - // We also capture the idleCallbackRunnable to tentatively fix: - // https://github.com/facebook/react-native/issues/44842 - IdleCallbackRunnable idleCallbackRunnable = mCurrentIdleCallbackRunnable; - if (idleCallbackRunnable != null) { - idleCallbackRunnable.cancel(); - } - - mCurrentIdleCallbackRunnable = new IdleCallbackRunnable(frameTimeNanos); - mReactApplicationContext.runOnJSQueueThread(mCurrentIdleCallbackRunnable); - - mReactChoreographer.postFrameCallback(ReactChoreographer.CallbackType.IDLE_EVENT, this); - } - } - - private class IdleCallbackRunnable implements Runnable { - private volatile boolean mCancelled = false; - private final long mFrameStartTime; - - public IdleCallbackRunnable(long frameStartTime) { - mFrameStartTime = frameStartTime; - } - - @Override - public void run() { - if (mCancelled) { - return; - } - - long frameTimeMillis = mFrameStartTime / 1000000; - long timeSinceBoot = SystemClock.uptimeMillis(); - long frameTimeElapsed = timeSinceBoot - frameTimeMillis; - long time = SystemClock.currentTimeMillis(); - long absoluteFrameStartTime = time - frameTimeElapsed; - - if (FRAME_DURATION_MS - (float) frameTimeElapsed < IDLE_CALLBACK_FRAME_DEADLINE_MS) { - return; - } - - boolean sendIdleEvents; - synchronized (mIdleCallbackGuard) { - sendIdleEvents = mSendIdleEvents; - } - - if (sendIdleEvents) { - mJavaScriptTimerExecutor.callIdleCallbacks(absoluteFrameStartTime); - } - - mCurrentIdleCallbackRunnable = null; - } - - public void cancel() { - mCancelled = true; - } - } - - private final ReactApplicationContext mReactApplicationContext; - private final JavaScriptTimerExecutor mJavaScriptTimerExecutor; - private final ReactChoreographer mReactChoreographer; - private final DevSupportManager mDevSupportManager; - private final Object mTimerGuard = new Object(); - private final Object mIdleCallbackGuard = new Object(); - private final PriorityQueue mTimers; - private final SparseArray mTimerIdsToTimers; - private final AtomicBoolean isPaused = new AtomicBoolean(true); - private final AtomicBoolean isRunningTasks = new AtomicBoolean(false); - private final TimerFrameCallback mTimerFrameCallback = new TimerFrameCallback(); - private final IdleFrameCallback mIdleFrameCallback = new IdleFrameCallback(); - private @Nullable IdleCallbackRunnable mCurrentIdleCallbackRunnable; - private boolean mFrameCallbackPosted = false; - private boolean mFrameIdleCallbackPosted = false; - private boolean mSendIdleEvents = false; - - public JavaTimerManager( - ReactApplicationContext reactContext, - JavaScriptTimerExecutor javaScriptTimerManager, - ReactChoreographer reactChoreographer, - DevSupportManager devSupportManager) { - mReactApplicationContext = reactContext; - mJavaScriptTimerExecutor = javaScriptTimerManager; - mReactChoreographer = reactChoreographer; - mDevSupportManager = devSupportManager; - - // We store timers sorted by finish time. - mTimers = - new PriorityQueue( - 11, // Default capacity: for some reason they don't expose a (Comparator) constructor - new Comparator() { - @Override - public int compare(Timer lhs, Timer rhs) { - long diff = lhs.mTargetTime - rhs.mTargetTime; - if (diff == 0) { - return 0; - } else if (diff < 0) { - return -1; - } else { - return 1; - } - } - }); - mTimerIdsToTimers = new SparseArray<>(); - - mReactApplicationContext.addLifecycleEventListener(this); - } - - @Override - public void onHostPause() { - isPaused.set(true); - clearFrameCallback(); - maybeIdleCallback(); - } - - @Override - public void onHostDestroy() { - clearFrameCallback(); - maybeIdleCallback(); - } - - @Override - public void onHostResume() { - isPaused.set(false); - // TODO(5195192) Investigate possible problems related to restarting all tasks at the same - // moment - setChoreographerCallback(); - maybeSetChoreographerIdleCallback(); - } - - @Override - public void onHeadlessJsTaskStart(int taskId) { - if (!isRunningTasks.getAndSet(true)) { - setChoreographerCallback(); - maybeSetChoreographerIdleCallback(); - } - } - - @Override - public void onHeadlessJsTaskFinish(int taskId) { - HeadlessJsTaskContext headlessJsTaskContext = - HeadlessJsTaskContext.getInstance(mReactApplicationContext); - if (!headlessJsTaskContext.hasActiveTasks()) { - isRunningTasks.set(false); - clearFrameCallback(); - maybeIdleCallback(); - } - } - - public void onInstanceDestroy() { - mReactApplicationContext.removeLifecycleEventListener(this); - - clearFrameCallback(); - clearChoreographerIdleCallback(); - } - - private void maybeSetChoreographerIdleCallback() { - synchronized (mIdleCallbackGuard) { - if (mSendIdleEvents) { - setChoreographerIdleCallback(); - } - } - } - - private void maybeIdleCallback() { - if (isPaused.get() && !isRunningTasks.get()) { - clearFrameCallback(); - } - } - - private void setChoreographerCallback() { - if (!mFrameCallbackPosted) { - mReactChoreographer.postFrameCallback( - ReactChoreographer.CallbackType.TIMERS_EVENTS, mTimerFrameCallback); - mFrameCallbackPosted = true; - } - } - - private void clearFrameCallback() { - HeadlessJsTaskContext headlessJsTaskContext = - HeadlessJsTaskContext.getInstance(mReactApplicationContext); - if (mFrameCallbackPosted && isPaused.get() && !headlessJsTaskContext.hasActiveTasks()) { - mReactChoreographer.removeFrameCallback( - ReactChoreographer.CallbackType.TIMERS_EVENTS, mTimerFrameCallback); - mFrameCallbackPosted = false; - } - } - - private void setChoreographerIdleCallback() { - if (!mFrameIdleCallbackPosted) { - mReactChoreographer.postFrameCallback( - ReactChoreographer.CallbackType.IDLE_EVENT, mIdleFrameCallback); - mFrameIdleCallbackPosted = true; - } - } - - private void clearChoreographerIdleCallback() { - if (mFrameIdleCallbackPosted) { - mReactChoreographer.removeFrameCallback( - ReactChoreographer.CallbackType.IDLE_EVENT, mIdleFrameCallback); - mFrameIdleCallbackPosted = false; - } - } - - /** - * A method to be used for synchronously creating a timer. The timer will not be invoked until the - * next frame, regardless of whether it has already expired (i.e. the delay is 0). - * - * @param callbackID An identifier for the callback that can be passed to JS or C++ to invoke it. - * @param delay The time in ms before the callback should be invoked. - * @param repeat Whether the timer should be repeated (used for setInterval). - */ - @DoNotStrip - public void createTimer(final int callbackID, final long delay, final boolean repeat) { - long initialTargetTime = SystemClock.nanoTime() / 1000000 + delay; - Timer timer = new Timer(callbackID, initialTargetTime, (int) delay, repeat); - synchronized (mTimerGuard) { - mTimers.add(timer); - mTimerIdsToTimers.put(callbackID, timer); - } - } - - /** - * A method to be used for asynchronously creating a timer. If the timer has already expired, - * (based on the provided jsSchedulingTime) then it will be immediately invoked. - * - * @param callbackID An identifier that can be passed back to JS to invoke the callback. - * @param duration The time in ms before the callback should be invoked. - * @param jsSchedulingTime The time (ms since epoch) when this timer was created in JS. - * @param repeat Whether the timer should be repeated (used for setInterval) - */ - public void createAndMaybeCallTimer( - final int callbackID, - final int duration, - final double jsSchedulingTime, - final boolean repeat) { - long deviceTime = SystemClock.currentTimeMillis(); - long remoteTime = (long) jsSchedulingTime; - - // If the times on the server and device have drifted throw an exception to warn the developer - // that things might not work or results may not be accurate. This is required only for - // developer builds. - if (mDevSupportManager.getDevSupportEnabled()) { - long driftTime = Math.abs(remoteTime - deviceTime); - if (driftTime > 60000) { - mJavaScriptTimerExecutor.emitTimeDriftWarning( - "Debugger and device times have drifted by more than 60s. Please correct this by " - + "running adb shell \"date `date +%m%d%H%M%Y.%S`\" on your debugger machine."); - } - } - - // Adjust for the amount of time it took for native to receive the timer registration call - long adjustedDuration = Math.max(0, remoteTime - deviceTime + duration); - if (duration == 0 && !repeat) { - WritableArray timerToCall = Arguments.createArray(); - timerToCall.pushInt(callbackID); - mJavaScriptTimerExecutor.callTimers(timerToCall); - return; - } - - createTimer(callbackID, adjustedDuration, repeat); - } - - @DoNotStrip - public void deleteTimer(int timerId) { - synchronized (mTimerGuard) { - Timer timer = mTimerIdsToTimers.get(timerId); - if (timer == null) { - return; - } - mTimerIdsToTimers.remove(timerId); - mTimers.remove(timer); - } - } - - @DoNotStrip - public void setSendIdleEvents(final boolean sendIdleEvents) { - synchronized (mIdleCallbackGuard) { - mSendIdleEvents = sendIdleEvents; - } - - UiThreadUtil.runOnUiThread( - new Runnable() { - @Override - public void run() { - synchronized (mIdleCallbackGuard) { - if (sendIdleEvents) { - setChoreographerIdleCallback(); - } else { - clearChoreographerIdleCallback(); - } - } - } - }); - } - - /** - * Returns a bool representing whether there are any active timers that will be fired within a - * certain period of time. Disregards repeating timers (setInterval). Used for testing to - * determine if RN is idle. - * - * @param rangeMs The time range, in ms, to check - * @return True if there are pending timers within the given range; false otherwise - */ - /* package */ boolean hasActiveTimersInRange(long rangeMs) { - synchronized (mTimerGuard) { - Timer nextTimer = mTimers.peek(); - if (nextTimer == null) { - // Timers queue is empty - return false; - } - if (isTimerInRange(nextTimer, rangeMs)) { - // First check the next timer, so we can avoid iterating over the entire queue if it's - // already within range. - return true; - } - for (Timer timer : mTimers) { - if (isTimerInRange(timer, rangeMs)) { - return true; - } - } - } - return false; - } - - private static boolean isTimerInRange(Timer timer, long rangeMs) { - return !timer.mRepeat && timer.mInterval < rangeMs; - } -} diff --git a/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/JavaTimerManager.kt b/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/JavaTimerManager.kt new file mode 100644 index 00000000000000..ecce67478dd9d0 --- /dev/null +++ b/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/JavaTimerManager.kt @@ -0,0 +1,363 @@ +/* + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */ + +package com.facebook.react.modules.core + +import android.util.SparseArray +import android.view.Choreographer +import com.facebook.proguard.annotations.DoNotStrip +import com.facebook.react.bridge.Arguments +import com.facebook.react.bridge.LifecycleEventListener +import com.facebook.react.bridge.ReactApplicationContext +import com.facebook.react.bridge.UiThreadUtil +import com.facebook.react.bridge.WritableArray +import com.facebook.react.common.SystemClock.currentTimeMillis +import com.facebook.react.common.SystemClock.nanoTime +import com.facebook.react.common.SystemClock.uptimeMillis +import com.facebook.react.devsupport.interfaces.DevSupportManager +import com.facebook.react.jstasks.HeadlessJsTaskContext +import com.facebook.react.jstasks.HeadlessJsTaskEventListener +import java.util.PriorityQueue +import java.util.concurrent.atomic.AtomicBoolean +import kotlin.concurrent.Volatile +import kotlin.math.abs +import kotlin.math.max +import kotlin.math.sign + +/** + * This class is the native implementation for JS timer execution on Android. It schedules JS timers + * to be invoked on frame boundaries using [ReactChoreographer]. + * + * This is used by the NativeModule [TimingModule]. + */ +public open class JavaTimerManager( + private val reactApplicationContext: ReactApplicationContext, + private val javaScriptTimerExecutor: JavaScriptTimerExecutor, + private val reactChoreographer: ReactChoreographer, + private val devSupportManager: DevSupportManager +) : LifecycleEventListener, HeadlessJsTaskEventListener { + private class Timer( + val timerId: Int, + var targetTime: Long, + val interval: Int, + val repeat: Boolean + ) + + private val timerGuard = Any() + private val idleCallbackGuard = Any() + private val timerIdsToTimers: SparseArray = SparseArray() + private val isPaused = AtomicBoolean(true) + private val isRunningTasks = AtomicBoolean(false) + private val timerFrameCallback = TimerFrameCallback() + private val idleFrameCallback = IdleFrameCallback() + private var currentIdleCallbackRunnable: IdleCallbackRunnable? = null + private var frameCallbackPosted = false + private var frameIdleCallbackPosted = false + private var sendIdleEvents = false + + // We store timers sorted by finish time. + private val timers: PriorityQueue = + PriorityQueue(TIMER_QUEUE_CAPACITY) { lhs, rhs -> (lhs.targetTime - rhs.targetTime).sign } + + init { + reactApplicationContext.addLifecycleEventListener(this) + } + + override fun onHostPause() { + isPaused.set(true) + clearFrameCallback() + maybeIdleCallback() + } + + override fun onHostDestroy() { + clearFrameCallback() + maybeIdleCallback() + } + + override fun onHostResume() { + isPaused.set(false) + // TODO(5195192) Investigate possible problems related to restarting all tasks at the same + // moment + setChoreographerCallback() + maybeSetChoreographerIdleCallback() + } + + override fun onHeadlessJsTaskStart(taskId: Int) { + if (!isRunningTasks.getAndSet(true)) { + setChoreographerCallback() + maybeSetChoreographerIdleCallback() + } + } + + override fun onHeadlessJsTaskFinish(taskId: Int) { + val headlessJsTaskContext = HeadlessJsTaskContext.getInstance(reactApplicationContext) + if (!headlessJsTaskContext.hasActiveTasks()) { + isRunningTasks.set(false) + clearFrameCallback() + maybeIdleCallback() + } + } + + public open fun onInstanceDestroy() { + reactApplicationContext.removeLifecycleEventListener(this) + clearFrameCallback() + clearChoreographerIdleCallback() + } + + private fun maybeSetChoreographerIdleCallback() { + synchronized(idleCallbackGuard) { + if (sendIdleEvents) { + setChoreographerIdleCallback() + } + } + } + + private fun maybeIdleCallback() { + if (isPaused.get() && !isRunningTasks.get()) { + clearFrameCallback() + } + } + + private fun setChoreographerCallback() { + if (!frameCallbackPosted) { + reactChoreographer.postFrameCallback( + ReactChoreographer.CallbackType.TIMERS_EVENTS, timerFrameCallback) + frameCallbackPosted = true + } + } + + private fun clearFrameCallback() { + val headlessJsTaskContext = HeadlessJsTaskContext.getInstance(reactApplicationContext) + if (frameCallbackPosted && isPaused.get() && !headlessJsTaskContext.hasActiveTasks()) { + reactChoreographer.removeFrameCallback( + ReactChoreographer.CallbackType.TIMERS_EVENTS, timerFrameCallback) + frameCallbackPosted = false + } + } + + private fun setChoreographerIdleCallback() { + if (!frameIdleCallbackPosted) { + reactChoreographer.postFrameCallback( + ReactChoreographer.CallbackType.IDLE_EVENT, idleFrameCallback) + frameIdleCallbackPosted = true + } + } + + private fun clearChoreographerIdleCallback() { + if (frameIdleCallbackPosted) { + reactChoreographer.removeFrameCallback( + ReactChoreographer.CallbackType.IDLE_EVENT, idleFrameCallback) + frameIdleCallbackPosted = false + } + } + + /** + * A method to be used for synchronously creating a timer. The timer will not be invoked until the + * next frame, regardless of whether it has already expired (i.e. the delay is 0). + * + * @param timerId An identifier for the callback that can be passed to JS or C++ to invoke it. + * @param delay The time in ms before the callback should be invoked. + * @param repeat Whether the timer should be repeated (used for setInterval). + */ + @DoNotStrip + public open fun createTimer(timerId: Int, delay: Long, repeat: Boolean) { + val initialTargetTime = nanoTime() / 1000000 + delay + val timer = Timer(timerId, initialTargetTime, delay.toInt(), repeat) + synchronized(timerGuard) { + timers.add(timer) + timerIdsToTimers.put(timerId, timer) + } + } + + /** + * A method to be used for asynchronously creating a timer. If the timer has already expired, + * (based on the provided jsSchedulingTime) then it will be immediately invoked. + * + * @param timerId An identifier that can be passed back to JS to invoke the callback. + * @param duration The time in ms before the callback should be invoked. + * @param jsSchedulingTime The time (ms since epoch) when this timer was created in JS. + * @param repeat Whether the timer should be repeated (used for setInterval) + */ + public open fun createAndMaybeCallTimer( + timerId: Int, + duration: Int, + jsSchedulingTime: Double, + repeat: Boolean + ) { + val deviceTime = currentTimeMillis() + val remoteTime = jsSchedulingTime.toLong() + + // If the times on the server and device have drifted throw an exception to warn the developer + // that things might not work or results may not be accurate. This is required only for + // developer builds. + if (devSupportManager.devSupportEnabled) { + val driftTime = abs(remoteTime - deviceTime) + if (driftTime > 60000) { + javaScriptTimerExecutor.emitTimeDriftWarning( + "Debugger and device times have drifted by more than 60s. Please correct this by " + + "running adb shell \"date `date +%m%d%H%M%Y.%S`\" on your debugger machine.") + } + } + + // Adjust for the amount of time it took for native to receive the timer registration call + val adjustedDuration = max(0, remoteTime - deviceTime + duration) + if (duration == 0 && !repeat) { + val timerToCall = Arguments.createArray() + timerToCall.pushInt(timerId) + javaScriptTimerExecutor.callTimers(timerToCall) + return + } + createTimer(timerId, adjustedDuration, repeat) + } + + @DoNotStrip + public open fun deleteTimer(timerId: Int) { + synchronized(timerGuard) { + val timer = timerIdsToTimers[timerId] ?: return + timerIdsToTimers.remove(timerId) + timers.remove(timer) + } + } + + @DoNotStrip + public open fun setSendIdleEvents(sendIdleEvents: Boolean) { + synchronized(idleCallbackGuard) { this.sendIdleEvents = sendIdleEvents } + UiThreadUtil.runOnUiThread { + synchronized(idleCallbackGuard) { + if (sendIdleEvents) { + setChoreographerIdleCallback() + } else { + clearChoreographerIdleCallback() + } + } + } + } + + /** + * Returns a bool representing whether there are any active timers that will be fired within a + * certain period of time. Disregards repeating timers (setInterval). Used for testing to + * determine if RN is idle. + * + * @param rangeMs The time range, in ms, to check + * @return True if there are pending timers within the given range; false otherwise + */ + internal fun hasActiveTimersInRange(rangeMs: Long): Boolean { + synchronized(timerGuard) { + val nextTimer = + timers.peek() + ?: // Timers queue is empty + return false + if (isTimerInRange(nextTimer, rangeMs)) { + // First check the next timer, so we can avoid iterating over the entire queue if it's + // already within range. + return true + } + for (timer in timers) { + if (isTimerInRange(timer, rangeMs)) { + return true + } + } + } + return false + } + + private inner class TimerFrameCallback : Choreographer.FrameCallback { + // Temporary map for constructing the individual arrays of timers to call + private var timersToCall: WritableArray? = null + + /** Calls all timers that have expired since the last time this frame callback was called. */ + override fun doFrame(frameTimeNanos: Long) { + if (isPaused.get() && !isRunningTasks.get()) { + return + } + val frameTimeMillis = frameTimeNanos / 1000000 + synchronized(timerGuard) { + while (!timers.isEmpty() && timers.peek()!!.targetTime < frameTimeMillis) { + var timer = timers.poll() + if (timer == null) { + break + } + if (timersToCall == null) { + timersToCall = Arguments.createArray() + } + timersToCall?.pushInt(timer.timerId) + if (timer.repeat) { + timer.targetTime = frameTimeMillis + timer.interval + timers.add(timer) + } else { + timerIdsToTimers.remove(timer.timerId) + } + } + } + timersToCall?.let { + javaScriptTimerExecutor.callTimers(it) + timersToCall = null + } + reactChoreographer.postFrameCallback(ReactChoreographer.CallbackType.TIMERS_EVENTS, this) + } + } + + private inner class IdleFrameCallback : Choreographer.FrameCallback { + override fun doFrame(frameTimeNanos: Long) { + if (isPaused.get() && !isRunningTasks.get()) { + return + } + + // If the JS thread is busy for multiple frames we cancel any other pending runnable. + // We also capture the idleCallbackRunnable to tentatively fix: + // https://github.com/facebook/react-native/issues/44842 + currentIdleCallbackRunnable?.cancel() + currentIdleCallbackRunnable = IdleCallbackRunnable(frameTimeNanos) + reactApplicationContext.runOnJSQueueThread(currentIdleCallbackRunnable) + reactChoreographer.postFrameCallback(ReactChoreographer.CallbackType.IDLE_EVENT, this) + } + } + + private inner class IdleCallbackRunnable(private val frameStartTime: Long) : Runnable { + @Volatile private var isCancelled = false + + override fun run() { + if (isCancelled) { + return + } + val frameTimeMillis = frameStartTime / 1000000 + val timeSinceBoot = uptimeMillis() + val frameTimeElapsed = timeSinceBoot - frameTimeMillis + val time = currentTimeMillis() + val absoluteFrameStartTime = time - frameTimeElapsed + if (FRAME_DURATION_MS - frameTimeElapsed.toFloat() < IDLE_CALLBACK_FRAME_DEADLINE_MS) { + return + } + var sendIdleEvents: Boolean + synchronized(idleCallbackGuard) { sendIdleEvents = this@JavaTimerManager.sendIdleEvents } + if (sendIdleEvents) { + javaScriptTimerExecutor.callIdleCallbacks(absoluteFrameStartTime.toDouble()) + } + currentIdleCallbackRunnable = null + } + + fun cancel() { + isCancelled = true + } + } + + private companion object { + // These timing constants should be kept in sync with the ones in `JSTimers.js`. + // The minimum time in milliseconds left in the frame to call idle callbacks. + private const val IDLE_CALLBACK_FRAME_DEADLINE_MS = 1f + + // The total duration of a frame in milliseconds, this assumes that devices run at 60 fps. + // TODO: Lower frame duration on devices that are too slow to run consistently + // at 60 fps. + private const val FRAME_DURATION_MS = 1000f / 60f + + private const val TIMER_QUEUE_CAPACITY = 11 + + private fun isTimerInRange(timer: Timer, rangeMs: Long): Boolean = + !timer.repeat && timer.interval < rangeMs + } +} diff --git a/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/TimingModule.kt b/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/TimingModule.kt index 5f797fdf7fcabe..bfa0219acf2454 100644 --- a/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/TimingModule.kt +++ b/packages/react-native/ReactAndroid/src/main/java/com/facebook/react/modules/core/TimingModule.kt @@ -18,8 +18,8 @@ import com.facebook.react.module.annotations.ReactModule /** Native module for JS timer execution. Timers fire on frame boundaries. */ @ReactModule(name = NativeTimingSpec.NAME) public class TimingModule( - reactContext: ReactApplicationContext?, - devSupportManager: DevSupportManager? + reactContext: ReactApplicationContext, + devSupportManager: DevSupportManager ) : com.facebook.fbreact.specs.NativeTimingSpec(reactContext), JavaScriptTimerExecutor { private val javaTimerManager: JavaTimerManager = JavaTimerManager(reactContext, this, ReactChoreographer.getInstance(), devSupportManager)