Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Fiber] Refactor Commit Phase into Separate Functions for Before Mutation/Mutation/Layout #31930

Merged
merged 7 commits into from
Jan 2, 2025

Conversation

sebmarkbage
Copy link
Collaborator

Stacked on #31922.

This is doing some general clean up to be able to split the commit root three phases into three separate async steps.

Copy link

vercel bot commented Dec 28, 2024

The latest updates on your projects. Learn more about Vercel for Git ↗︎

Name Status Preview Comments Updated (UTC)
react-compiler-playground ❌ Failed (Inspect) Jan 2, 2025 7:46pm

@react-sizebot
Copy link

react-sizebot commented Dec 28, 2024

Comparing: d8b903f...e953c4e

Critical size changes

Includes critical production bundles, as well as any change greater than 2%:

Name +/- Base Current +/- gzip Base gzip Current gzip
oss-stable/react-dom/cjs/react-dom.production.js = 6.68 kB 6.68 kB = 1.83 kB 1.83 kB
oss-stable/react-dom/cjs/react-dom-client.production.js +0.15% 511.67 kB 512.42 kB +0.26% 91.36 kB 91.59 kB
oss-experimental/react-dom/cjs/react-dom.production.js = 6.69 kB 6.69 kB = 1.83 kB 1.83 kB
oss-experimental/react-dom/cjs/react-dom-client.production.js +0.15% 516.43 kB 517.20 kB +0.26% 92.21 kB 92.45 kB
facebook-www/ReactDOM-prod.classic.js +0.19% 593.12 kB 594.22 kB +0.25% 104.42 kB 104.67 kB
facebook-www/ReactDOM-prod.modern.js +0.19% 583.38 kB 584.49 kB +0.25% 102.86 kB 103.13 kB

Significant size changes

Includes any change greater than 0.2%:

Expand to show
Name +/- Base Current +/- gzip Base gzip Current gzip
facebook-www/ReactART-prod.modern.js +0.79% 354.31 kB 357.10 kB +0.72% 59.38 kB 59.81 kB
facebook-www/ReactART-prod.classic.js +0.77% 364.03 kB 366.83 kB +0.74% 60.95 kB 61.41 kB
oss-stable-semver/react-art/cjs/react-art.production.js +0.39% 299.93 kB 301.11 kB +0.15% 51.21 kB 51.29 kB
oss-stable/react-art/cjs/react-art.production.js +0.39% 300.00 kB 301.19 kB +0.15% 51.24 kB 51.32 kB
facebook-react-native/react-test-renderer/cjs/ReactTestRenderer-prod.js +0.35% 332.31 kB 333.48 kB +0.16% 58.15 kB 58.24 kB
facebook-www/ReactART-dev.modern.js +0.34% 636.86 kB 639.04 kB +0.30% 100.95 kB 101.25 kB
oss-experimental/react-art/cjs/react-art.production.js +0.34% 304.53 kB 305.56 kB +0.17% 52.12 kB 52.21 kB
facebook-www/ReactART-dev.classic.js +0.34% 646.74 kB 648.92 kB +0.31% 102.90 kB 103.22 kB
react-native/implementations/ReactFabric-prod.js +0.33% 356.03 kB 357.20 kB +0.10% 62.07 kB 62.13 kB
react-native/implementations/ReactNativeRenderer-prod.js +0.32% 364.24 kB 365.41 kB +0.14% 63.38 kB 63.47 kB
facebook-react-native/react-test-renderer/cjs/ReactTestRenderer-profiling.js +0.32% 354.33 kB 355.45 kB +0.20% 61.24 kB 61.36 kB
react-native/implementations/ReactFabric-prod.fb.js +0.31% 380.73 kB 381.90 kB +0.10% 66.33 kB 66.40 kB
react-native/implementations/ReactNativeRenderer-prod.fb.js +0.30% 386.17 kB 387.34 kB +0.07% 67.30 kB 67.34 kB
oss-stable-semver/react-reconciler/cjs/react-reconciler.production.js +0.25% 387.46 kB 388.41 kB +0.20% 62.91 kB 63.04 kB
oss-stable/react-reconciler/cjs/react-reconciler.production.js +0.25% 387.48 kB 388.44 kB +0.19% 62.94 kB 63.06 kB
facebook-www/ReactReconciler-prod.modern.js +0.23% 448.43 kB 449.48 kB +0.17% 72.15 kB 72.28 kB
react-native/implementations/ReactFabric-profiling.js +0.23% 381.42 kB 382.30 kB +0.18% 65.72 kB 65.84 kB
facebook-www/ReactReconciler-prod.classic.js +0.23% 458.75 kB 459.80 kB +0.18% 73.71 kB 73.84 kB
react-native/implementations/ReactNativeRenderer-profiling.js +0.22% 389.67 kB 390.54 kB +0.18% 67.13 kB 67.25 kB
react-native/implementations/ReactFabric-profiling.fb.js +0.22% 406.37 kB 407.25 kB +0.06% 70.22 kB 70.26 kB
react-native/implementations/ReactNativeRenderer-profiling.fb.js +0.21% 411.78 kB 412.65 kB +0.05% 71.10 kB 71.13 kB
oss-experimental/react-reconciler/cjs/react-reconciler.production.js +0.20% 392.64 kB 393.43 kB +0.18% 63.85 kB 63.97 kB

Generated by 🚫 dangerJS against f472472

@@ -744,6 +744,7 @@ describe('ReactDeferredValue', () => {
</Container>,
);
// We should switch to pre-rendering the new preview.
await waitForPaint([]);
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This happens because the requestPaint() is now unconditional even if there weren't any effects. Which doesn't really hurt anything since it's actually a noop in anything but tests and should probably go away.

@@ -3460,7 +3471,8 @@ function commitRootImpl(
}
}

onCommitRootDevTools(finishedWork.stateNode, renderPriorityLevel);
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This one is a bit strange because the profiler in devtools tracks the priority level that the commit phase is running in rather than what priority was rendered and then committed. I think the idea is that it would be running in the scheduler priority of the render but that's not necessarily the case for the commit.

Copy link

@DuoAlly-AI-Recruiter DuoAlly-AI-Recruiter left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

All the codes function correctly.

Copy link

@anukaal anukaal left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

looks ok

Copy link
Member

@rickhanlonii rickhanlonii left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The refactor LGTM but also changing the finishedWork to cancelPendingCommit seems sus. Any way you could at least break that change into a separate PR so it's easier to revert if there's an unexpected issue?

@@ -1839,9 +1841,6 @@ function prepareFreshStack(root: FiberRoot, lanes: Lanes): Fiber {
}
}

root.finishedWork = null;
root.finishedLanes = NoLanes;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should root.cancelPendingCommit and timeoutHandle get reset?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

They do get reset right below here as they're cancelled.

'bug in React.',
);
}
}
}
root.finishedWork = null;
root.finishedLanes = NoLanes;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should root.cancelPendingCommit and timeoutHandle get reset?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

root.cancelPendingCommit gets reset in the beginning of commitRoot. timeoutHandle gets reset in the beginning of commitRootWhenReady.

Which is really just right at the beginning of the callback that was pending.

The idea is that it's reset right after the callback is fired, at the very beginning, since it's no longer pending. Or it gets reset when it's cancelled.

}
}
flushMutationEffects(root, finishedWork, lanes);
flushLayoutEffects(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Might be nice to move the before mutation logic above to flushBeforeMutationEffects to match these.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These won't remain exactly like this once they get split out into the async ones and there's more phases a bit in motion. For now it's to make it clear which parts are sync and which parts are async.

Once that settles we can see where we end up.

const prevTransition = ReactSharedInternals.T;
const previousUpdateLanePriority = getCurrentUpdatePriority();
setCurrentUpdatePriority(DiscreteEventPriority);
ReactSharedInternals.T = null;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do we need to set the priority here? Is this just so the reactor is more 1-1?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yea it's mainly to keep it 1-1. Since this is a user space callback you could have a setState in here such as updating an error dialog.

Therefore this must have some semantic. It used to be Discrete based on how it was factored but it makes sense that it would be.

Whether it always should be anyway is an interesting question though. It really shouldn't be possible to run a commit inside a startTransition so I'm not sure how it could ever be anything else.

We're already setting the update priority around user space code.
The execution scope of the commit isn't as useful for profiling information.
This aligns it with passive effects.

We can cancel these using the timeoutHandle or cancelPendingCommit.
We should never run a commit function that was cancelled.
We can check these to determine whether there are any pending commits.
Because ReactFiberLane is imported for its constants it cannot depend on
the config.
@sebmarkbage
Copy link
Collaborator Author

sebmarkbage commented Jan 2, 2025

The refactor LGTM but also changing the finishedWork to cancelPendingCommit

This part gives me a bit more confident that it'll work correctly with the async gaps inserted. I think we'll just have to revert this whole PR if something goes wrong. Which is why I wanted to land this piece as early as possible.

@sebmarkbage sebmarkbage merged commit c81312e into facebook:main Jan 2, 2025
186 of 187 checks passed
@sebmarkbage
Copy link
Collaborator Author

There's still an outstanding thing to resolve. It doesn't show up in this PR since it's still sync but once async gaps are inserted there could be setStates/renders that in the gap that needs to model a pending commit similar to passive but it's more like Execution Environment in that all sync updates need to be deferred until this commit finishes.

github-actions bot pushed a commit that referenced this pull request Jan 2, 2025
…tion/Mutation/Layout (#31930)

This is doing some general clean up to be able to split the commit root three phases into three separate async steps.

DiffTrain build for [c81312e](c81312e)
github-actions bot pushed a commit that referenced this pull request Jan 2, 2025
…tion/Mutation/Layout (#31930)

This is doing some general clean up to be able to split the commit root three phases into three separate async steps.

DiffTrain build for [c81312e](c81312e)
sebmarkbage added a commit that referenced this pull request Jan 6, 2025
…but flush it eagerly if we're sync (#31987)

This is a follow up to #31930 and a prerequisite for #31975.

With View Transitions, the commit phase becomes async which means that
other work can sneak in between. We need to be resilient to that.

This PR first refactors the flushMutationEffects and flushLayoutEffects
to use module scope variables to track its arguments so we can defer
them. It shares these with how we were already doing it for
flushPendingEffects.

We also track how far along the commit phase we are so we know what we
have left to flush.

Then callers of flushPassiveEffects become flushPendingEffects. That
helper synchronously flushes any remaining phases we've yet to commit.
That ensure that things are at least consistent if that happens.

Finally, when we are using a scheduled task, we don't do any work. This
ensures that we're not flushing any work too early if we could've
deferred it. This still ensures that we always do flush it before
starting any new work on any root so new roots observe the committed
state.

There are some unfortunate effects that could happen from allowing
things to flush eagerly. Such as if a flushSync sneaks in before
startViewTransition, it'll skip the animation. If it's during a
suspensey font it'll start the transition before the font has loaded
which might be better than breaking flushSync. It'll also potentially
flush passive effects inside the startViewTransition which should
typically be ok.
github-actions bot pushed a commit that referenced this pull request Jan 6, 2025
…but flush it eagerly if we're sync (#31987)

This is a follow up to #31930 and a prerequisite for #31975.

With View Transitions, the commit phase becomes async which means that
other work can sneak in between. We need to be resilient to that.

This PR first refactors the flushMutationEffects and flushLayoutEffects
to use module scope variables to track its arguments so we can defer
them. It shares these with how we were already doing it for
flushPendingEffects.

We also track how far along the commit phase we are so we know what we
have left to flush.

Then callers of flushPassiveEffects become flushPendingEffects. That
helper synchronously flushes any remaining phases we've yet to commit.
That ensure that things are at least consistent if that happens.

Finally, when we are using a scheduled task, we don't do any work. This
ensures that we're not flushing any work too early if we could've
deferred it. This still ensures that we always do flush it before
starting any new work on any root so new roots observe the committed
state.

There are some unfortunate effects that could happen from allowing
things to flush eagerly. Such as if a flushSync sneaks in before
startViewTransition, it'll skip the animation. If it's during a
suspensey font it'll start the transition before the font has loaded
which might be better than breaking flushSync. It'll also potentially
flush passive effects inside the startViewTransition which should
typically be ok.

DiffTrain build for [defffdb](defffdb)
github-actions bot pushed a commit that referenced this pull request Jan 6, 2025
…but flush it eagerly if we're sync (#31987)

This is a follow up to #31930 and a prerequisite for #31975.

With View Transitions, the commit phase becomes async which means that
other work can sneak in between. We need to be resilient to that.

This PR first refactors the flushMutationEffects and flushLayoutEffects
to use module scope variables to track its arguments so we can defer
them. It shares these with how we were already doing it for
flushPendingEffects.

We also track how far along the commit phase we are so we know what we
have left to flush.

Then callers of flushPassiveEffects become flushPendingEffects. That
helper synchronously flushes any remaining phases we've yet to commit.
That ensure that things are at least consistent if that happens.

Finally, when we are using a scheduled task, we don't do any work. This
ensures that we're not flushing any work too early if we could've
deferred it. This still ensures that we always do flush it before
starting any new work on any root so new roots observe the committed
state.

There are some unfortunate effects that could happen from allowing
things to flush eagerly. Such as if a flushSync sneaks in before
startViewTransition, it'll skip the animation. If it's during a
suspensey font it'll start the transition before the font has loaded
which might be better than breaking flushSync. It'll also potentially
flush passive effects inside the startViewTransition which should
typically be ok.

DiffTrain build for [defffdb](defffdb)
github-actions bot pushed a commit to code/lib-react that referenced this pull request Jan 6, 2025
…but flush it eagerly if we're sync (facebook#31987)

This is a follow up to facebook#31930 and a prerequisite for facebook#31975.

With View Transitions, the commit phase becomes async which means that
other work can sneak in between. We need to be resilient to that.

This PR first refactors the flushMutationEffects and flushLayoutEffects
to use module scope variables to track its arguments so we can defer
them. It shares these with how we were already doing it for
flushPendingEffects.

We also track how far along the commit phase we are so we know what we
have left to flush.

Then callers of flushPassiveEffects become flushPendingEffects. That
helper synchronously flushes any remaining phases we've yet to commit.
That ensure that things are at least consistent if that happens.

Finally, when we are using a scheduled task, we don't do any work. This
ensures that we're not flushing any work too early if we could've
deferred it. This still ensures that we always do flush it before
starting any new work on any root so new roots observe the committed
state.

There are some unfortunate effects that could happen from allowing
things to flush eagerly. Such as if a flushSync sneaks in before
startViewTransition, it'll skip the animation. If it's during a
suspensey font it'll start the transition before the font has loaded
which might be better than breaking flushSync. It'll also potentially
flush passive effects inside the startViewTransition which should
typically be ok.

DiffTrain build for [defffdb](facebook@defffdb)
github-actions bot pushed a commit to code/lib-react that referenced this pull request Jan 6, 2025
…but flush it eagerly if we're sync (facebook#31987)

This is a follow up to facebook#31930 and a prerequisite for facebook#31975.

With View Transitions, the commit phase becomes async which means that
other work can sneak in between. We need to be resilient to that.

This PR first refactors the flushMutationEffects and flushLayoutEffects
to use module scope variables to track its arguments so we can defer
them. It shares these with how we were already doing it for
flushPendingEffects.

We also track how far along the commit phase we are so we know what we
have left to flush.

Then callers of flushPassiveEffects become flushPendingEffects. That
helper synchronously flushes any remaining phases we've yet to commit.
That ensure that things are at least consistent if that happens.

Finally, when we are using a scheduled task, we don't do any work. This
ensures that we're not flushing any work too early if we could've
deferred it. This still ensures that we always do flush it before
starting any new work on any root so new roots observe the committed
state.

There are some unfortunate effects that could happen from allowing
things to flush eagerly. Such as if a flushSync sneaks in before
startViewTransition, it'll skip the animation. If it's during a
suspensey font it'll start the transition before the font has loaded
which might be better than breaking flushSync. It'll also potentially
flush passive effects inside the startViewTransition which should
typically be ok.

DiffTrain build for [defffdb](facebook@defffdb)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
CLA Signed React Core Team Opened by a member of the React Core Team
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants