Skip to content

Commit

Permalink
[Scheduler] Always yield to native macro tasks when a virtual task co…
Browse files Browse the repository at this point in the history
…mpletes (#31787)

As an alternative to #31784.

We should really just always yield each virtual task to a native task.
So that it's 1:1 with native tasks. This affects when microtasks within
each task happens. This brings us closer to native `postTask` semantics
which makes it more seamless to just use that when available.

This still doesn't yield when a task expires to protect against
starvation.
  • Loading branch information
sebmarkbage authored Dec 17, 2024
1 parent 34ee391 commit f5077bc
Show file tree
Hide file tree
Showing 5 changed files with 53 additions and 11 deletions.
2 changes: 2 additions & 0 deletions packages/scheduler/src/SchedulerFeatureFlags.js
Original file line number Diff line number Diff line change
Expand Up @@ -15,3 +15,5 @@ export const userBlockingPriorityTimeout = 250;
export const normalPriorityTimeout = 5000;
export const lowPriorityTimeout = 10000;
export const enableRequestPaint = true;

export const enableAlwaysYieldScheduler = __EXPERIMENTAL__;
19 changes: 15 additions & 4 deletions packages/scheduler/src/__tests__/Scheduler-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -201,9 +201,10 @@ describe('SchedulerBrowser', () => {
runtime.assertLog([
'Message Event',
'Task',
SchedulerFeatureFlags.enableRequestPaint
? 'Yield at 0ms'
: `Yield at ${SchedulerFeatureFlags.frameYieldMs}ms`,
gate(flags => flags.enableAlwaysYieldScheduler) ||
!SchedulerFeatureFlags.enableRequestPaint
? gate(flags => (flags.www ? 'Yield at 10ms' : 'Yield at 5ms'))
: 'Yield at 0ms',
'Post Message',
]);

Expand All @@ -220,7 +221,13 @@ describe('SchedulerBrowser', () => {
});
runtime.assertLog(['Post Message']);
runtime.fireMessageEvent();
runtime.assertLog(['Message Event', 'A', 'B']);
if (gate(flags => flags.enableAlwaysYieldScheduler)) {
runtime.assertLog(['Message Event', 'A', 'Post Message']);
runtime.fireMessageEvent();
runtime.assertLog(['Message Event', 'B']);
} else {
runtime.assertLog(['Message Event', 'A', 'B']);
}
});

it('multiple tasks with a yield in between', () => {
Expand Down Expand Up @@ -267,6 +274,10 @@ describe('SchedulerBrowser', () => {
runtime.assertLog(['Message Event', 'Oops!', 'Post Message']);

runtime.fireMessageEvent();
if (gate(flags => flags.enableAlwaysYieldScheduler)) {
runtime.assertLog(['Message Event', 'Post Message']);
runtime.fireMessageEvent();
}
runtime.assertLog(['Message Event', 'Yay']);
});

Expand Down
24 changes: 21 additions & 3 deletions packages/scheduler/src/__tests__/SchedulerSetImmediate-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -188,7 +188,13 @@ describe('SchedulerDOMSetImmediate', () => {
});
runtime.assertLog(['Set Immediate']);
runtime.fireSetImmediate();
runtime.assertLog(['setImmediate Callback', 'A', 'B']);
if (gate(flags => flags.enableAlwaysYieldScheduler)) {
runtime.assertLog(['setImmediate Callback', 'A', 'Set Immediate']);
runtime.fireSetImmediate();
runtime.assertLog(['setImmediate Callback', 'B']);
} else {
runtime.assertLog(['setImmediate Callback', 'A', 'B']);
}
});

it('multiple tasks at different priority', () => {
Expand All @@ -200,7 +206,13 @@ describe('SchedulerDOMSetImmediate', () => {
});
runtime.assertLog(['Set Immediate']);
runtime.fireSetImmediate();
runtime.assertLog(['setImmediate Callback', 'B', 'A']);
if (gate(flags => flags.enableAlwaysYieldScheduler)) {
runtime.assertLog(['setImmediate Callback', 'B', 'Set Immediate']);
runtime.fireSetImmediate();
runtime.assertLog(['setImmediate Callback', 'A']);
} else {
runtime.assertLog(['setImmediate Callback', 'B', 'A']);
}
});

it('multiple tasks with a yield in between', () => {
Expand Down Expand Up @@ -246,7 +258,13 @@ describe('SchedulerDOMSetImmediate', () => {
runtime.assertLog(['setImmediate Callback', 'Oops!', 'Set Immediate']);

runtime.fireSetImmediate();
runtime.assertLog(['setImmediate Callback', 'Yay']);
if (gate(flags => flags.enableAlwaysYieldScheduler)) {
runtime.assertLog(['setImmediate Callback', 'Set Immediate']);
runtime.fireSetImmediate();
runtime.assertLog(['setImmediate Callback', 'Yay']);
} else {
runtime.assertLog(['setImmediate Callback', 'Yay']);
}
});

it('schedule new task after queue has emptied', () => {
Expand Down
17 changes: 13 additions & 4 deletions packages/scheduler/src/forks/Scheduler.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import {
lowPriorityTimeout,
normalPriorityTimeout,
enableRequestPaint,
enableAlwaysYieldScheduler,
} from '../SchedulerFeatureFlags';

import {push, pop, peek} from '../SchedulerMinHeap';
Expand Down Expand Up @@ -196,9 +197,11 @@ function workLoop(initialTime: number) {
currentTask !== null &&
!(enableSchedulerDebugging && isSchedulerPaused)
) {
if (currentTask.expirationTime > currentTime && shouldYieldToHost()) {
// This currentTask hasn't expired, and we've reached the deadline.
break;
if (!enableAlwaysYieldScheduler) {
if (currentTask.expirationTime > currentTime && shouldYieldToHost()) {
// This currentTask hasn't expired, and we've reached the deadline.
break;
}
}
// $FlowFixMe[incompatible-use] found when upgrading Flow
const callback = currentTask.callback;
Expand Down Expand Up @@ -242,6 +245,12 @@ function workLoop(initialTime: number) {
pop(taskQueue);
}
currentTask = peek(taskQueue);
if (enableAlwaysYieldScheduler) {
if (currentTask === null || currentTask.expirationTime > currentTime) {
// This currentTask hasn't expired we yield to the browser task.
break;
}
}
}
// Return whether there's additional work
if (currentTask !== null) {
Expand Down Expand Up @@ -459,7 +468,7 @@ let frameInterval = frameYieldMs;
let startTime = -1;

function shouldYieldToHost(): boolean {
if (enableRequestPaint && needsPaint) {
if (!enableAlwaysYieldScheduler && enableRequestPaint && needsPaint) {
// Yield now.
return true;
}
Expand Down
2 changes: 2 additions & 0 deletions packages/scheduler/src/forks/SchedulerFeatureFlags.www.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,3 +19,5 @@ export const frameYieldMs = 10;
export const userBlockingPriorityTimeout = 250;
export const normalPriorityTimeout = 5000;
export const lowPriorityTimeout = 10000;

export const enableAlwaysYieldScheduler = false;

0 comments on commit f5077bc

Please sign in to comment.