Skip to content

Commit 765e89b

Browse files
authoredNov 13, 2020
Reset new fork to old fork (#20254)
* Fix typo This typo was fixed in the new fork but not the old. * Reset new fork to old fork Something in the new fork is causing a topline metrics regression. We're not sure what it is, so we're going to split it into steps and bisect. As a first step, this resets the new fork back to the contents of the old fork. We will land this to confirm that the fork infra itself is not causing a regression. * Fix tests: Add `dfsEffectsRefactor` flag Some of the tests that gated on the effects refactor used the `new` flag. In order to bisect, we'll need to decompose the new fork changes into multiple steps. So I added a hardcoded test flag called `dfsEffectsRefactor` and set it to false. Will turn back on when we switch back to traversing the finished tree using DFS and `subtreeTag`.
1 parent 7548dd5 commit 765e89b

19 files changed

+1380
-2552
lines changed
 

‎packages/react-dom/src/__tests__/ReactDOMServerPartialHydration-test.internal.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -367,7 +367,7 @@ describe('ReactDOMServerPartialHydration', () => {
367367
// This is a new node.
368368
expect(span).not.toBe(span2);
369369

370-
if (gate(flags => flags.new)) {
370+
if (gate(flags => flags.dfsEffectsRefactor)) {
371371
// The effects list refactor causes this to be null because the Suspense Offscreen's child
372372
// is null. However, since we can't hydrate Suspense in legacy this change in behavior is ok
373373
expect(ref.current).toBe(null);

‎packages/react-dom/src/__tests__/ReactWrongReturnPointer-test.js

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ beforeEach(() => {
1616
});
1717

1818
// Don't feel too guilty if you have to delete this test.
19+
// @gate dfsEffectsRefactor
1920
// @gate new
2021
// @gate __DEV__
2122
test('warns in DEV if return pointer is inconsistent', async () => {

‎packages/react-reconciler/src/ReactChildFiber.new.js

+13-6
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ import type {Fiber} from './ReactInternalTypes';
1313
import type {Lanes} from './ReactFiberLane';
1414

1515
import getComponentName from 'shared/getComponentName';
16-
import {Deletion, Placement} from './ReactFiberFlags';
16+
import {Placement, Deletion} from './ReactFiberFlags';
1717
import {
1818
getIteratorFn,
1919
REACT_ELEMENT_TYPE,
@@ -256,13 +256,20 @@ function ChildReconciler(shouldTrackSideEffects) {
256256
// Noop.
257257
return;
258258
}
259-
const deletions = returnFiber.deletions;
260-
if (deletions === null) {
261-
returnFiber.deletions = [childToDelete];
262-
returnFiber.flags |= Deletion;
259+
// Deletions are added in reversed order so we add it to the front.
260+
// At this point, the return fiber's effect list is empty except for
261+
// deletions, so we can just append the deletion to the list. The remaining
262+
// effects aren't added until the complete phase. Once we implement
263+
// resuming, this may not be true.
264+
const last = returnFiber.lastEffect;
265+
if (last !== null) {
266+
last.nextEffect = childToDelete;
267+
returnFiber.lastEffect = childToDelete;
263268
} else {
264-
deletions.push(childToDelete);
269+
returnFiber.firstEffect = returnFiber.lastEffect = childToDelete;
265270
}
271+
childToDelete.nextEffect = null;
272+
childToDelete.flags = Deletion;
266273
}
267274

268275
function deleteRemainingChildren(

‎packages/react-reconciler/src/ReactFiber.new.js

+13-4
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ import {
2828
enableFundamentalAPI,
2929
enableScopeAPI,
3030
} from 'shared/ReactFeatureFlags';
31-
import {NoFlags, Placement, StaticMask} from './ReactFiberFlags';
31+
import {NoFlags, Placement} from './ReactFiberFlags';
3232
import {ConcurrentRoot, BlockingRoot} from './ReactRootTags';
3333
import {
3434
IndeterminateComponent,
@@ -279,6 +279,13 @@ export function createWorkInProgress(current: Fiber, pendingProps: any): Fiber {
279279
workInProgress.type = current.type;
280280

281281
// We already have an alternate.
282+
// Reset the effect tag.
283+
workInProgress.flags = NoFlags;
284+
285+
// The effect list is no longer valid.
286+
workInProgress.nextEffect = null;
287+
workInProgress.firstEffect = null;
288+
workInProgress.lastEffect = null;
282289
workInProgress.subtreeFlags = NoFlags;
283290
workInProgress.deletions = null;
284291

@@ -292,9 +299,6 @@ export function createWorkInProgress(current: Fiber, pendingProps: any): Fiber {
292299
}
293300
}
294301

295-
// Reset all effects except static ones.
296-
// Static effects are not specific to a render.
297-
workInProgress.flags = current.flags & StaticMask;
298302
workInProgress.childLanes = current.childLanes;
299303
workInProgress.lanes = current.lanes;
300304

@@ -360,6 +364,11 @@ export function resetWorkInProgress(workInProgress: Fiber, renderLanes: Lanes) {
360364
// that child fiber is setting, not the reconciliation.
361365
workInProgress.flags &= Placement;
362366

367+
// The effect list is no longer valid.
368+
workInProgress.nextEffect = null;
369+
workInProgress.firstEffect = null;
370+
workInProgress.lastEffect = null;
371+
363372
const current = workInProgress.alternate;
364373
if (current === null) {
365374
// Reset to createFiber's initial values.

‎packages/react-reconciler/src/ReactFiberBeginWork.new.js

+42-26
Original file line numberDiff line numberDiff line change
@@ -58,10 +58,10 @@ import {
5858
Hydrating,
5959
ContentReset,
6060
DidCapture,
61+
Update,
6162
Ref,
6263
Deletion,
6364
ForceUpdateForLegacySuspense,
64-
StaticMask,
6565
} from './ReactFiberFlags';
6666
import ReactSharedInternals from 'shared/ReactSharedInternals';
6767
import {
@@ -671,6 +671,8 @@ function updateProfiler(
671671
renderLanes: Lanes,
672672
) {
673673
if (enableProfilerTimer) {
674+
workInProgress.flags |= Update;
675+
674676
// Reset effect durations for the next eventual effect phase.
675677
// These are reset during render to allow the DevTools commit hook a chance to read them,
676678
const stateNode = workInProgress.stateNode;
@@ -1077,9 +1079,6 @@ function updateHostComponent(
10771079
workInProgress.flags |= ContentReset;
10781080
}
10791081

1080-
// React DevTools reads this flag.
1081-
workInProgress.flags |= PerformedWork;
1082-
10831082
markRef(current, workInProgress);
10841083
reconcileChildren(current, workInProgress, nextChildren, renderLanes);
10851084
return workInProgress.child;
@@ -2005,14 +2004,9 @@ function updateSuspensePrimaryChildren(
20052004
primaryChildFragment.sibling = null;
20062005
if (currentFallbackChildFragment !== null) {
20072006
// Delete the fallback child fragment
2008-
const deletions = workInProgress.deletions;
2009-
if (deletions === null) {
2010-
workInProgress.deletions = [currentFallbackChildFragment];
2011-
// TODO (effects) Rename this to better reflect its new usage (e.g. ChildDeletions)
2012-
workInProgress.flags |= Deletion;
2013-
} else {
2014-
deletions.push(currentFallbackChildFragment);
2015-
}
2007+
currentFallbackChildFragment.nextEffect = null;
2008+
currentFallbackChildFragment.flags = Deletion;
2009+
workInProgress.firstEffect = workInProgress.lastEffect = currentFallbackChildFragment;
20162010
}
20172011

20182012
workInProgress.child = primaryChildFragment;
@@ -2069,19 +2063,24 @@ function updateSuspenseFallbackChildren(
20692063

20702064
// The fallback fiber was added as a deletion effect during the first pass.
20712065
// However, since we're going to remain on the fallback, we no longer want
2072-
// to delete it.
2073-
workInProgress.deletions = null;
2066+
// to delete it. So we need to remove it from the list. Deletions are stored
2067+
// on the same list as effects. We want to keep the effects from the primary
2068+
// tree. So we copy the primary child fragment's effect list, which does not
2069+
// include the fallback deletion effect.
2070+
const progressedLastEffect = primaryChildFragment.lastEffect;
2071+
if (progressedLastEffect !== null) {
2072+
workInProgress.firstEffect = primaryChildFragment.firstEffect;
2073+
workInProgress.lastEffect = progressedLastEffect;
2074+
progressedLastEffect.nextEffect = null;
2075+
} else {
2076+
// TODO: Reset this somewhere else? Lol legacy mode is so weird.
2077+
workInProgress.firstEffect = workInProgress.lastEffect = null;
2078+
}
20742079
} else {
20752080
primaryChildFragment = createWorkInProgressOffscreenFiber(
20762081
currentPrimaryChildFragment,
20772082
primaryChildProps,
20782083
);
2079-
2080-
// Since we're reusing a current tree, we need to reuse the flags, too.
2081-
// (We don't do this in legacy mode, because in legacy mode we don't re-use
2082-
// the current tree; see previous branch.)
2083-
primaryChildFragment.subtreeFlags =
2084-
currentPrimaryChildFragment.subtreeFlags & StaticMask;
20852084
}
20862085
let fallbackChildFragment;
20872086
if (currentFallbackChildFragment !== null) {
@@ -2566,6 +2565,7 @@ function initSuspenseListRenderState(
25662565
tail: null | Fiber,
25672566
lastContentRow: null | Fiber,
25682567
tailMode: SuspenseListTailMode,
2568+
lastEffectBeforeRendering: null | Fiber,
25692569
): void {
25702570
const renderState: null | SuspenseListRenderState =
25712571
workInProgress.memoizedState;
@@ -2577,6 +2577,7 @@ function initSuspenseListRenderState(
25772577
last: lastContentRow,
25782578
tail: tail,
25792579
tailMode: tailMode,
2580+
lastEffect: lastEffectBeforeRendering,
25802581
}: SuspenseListRenderState);
25812582
} else {
25822583
// We can reuse the existing object from previous renders.
@@ -2586,6 +2587,7 @@ function initSuspenseListRenderState(
25862587
renderState.last = lastContentRow;
25872588
renderState.tail = tail;
25882589
renderState.tailMode = tailMode;
2590+
renderState.lastEffect = lastEffectBeforeRendering;
25892591
}
25902592
}
25912593

@@ -2667,6 +2669,7 @@ function updateSuspenseListComponent(
26672669
tail,
26682670
lastContentRow,
26692671
tailMode,
2672+
workInProgress.lastEffect,
26702673
);
26712674
break;
26722675
}
@@ -2698,6 +2701,7 @@ function updateSuspenseListComponent(
26982701
tail,
26992702
null, // last
27002703
tailMode,
2704+
workInProgress.lastEffect,
27012705
);
27022706
break;
27032707
}
@@ -2708,6 +2712,7 @@ function updateSuspenseListComponent(
27082712
null, // tail
27092713
null, // last
27102714
undefined,
2715+
workInProgress.lastEffect,
27112716
);
27122717
break;
27132718
}
@@ -2967,14 +2972,15 @@ function remountFiber(
29672972

29682973
// Delete the old fiber and place the new one.
29692974
// Since the old fiber is disconnected, we have to schedule it manually.
2970-
const deletions = returnFiber.deletions;
2971-
if (deletions === null) {
2972-
returnFiber.deletions = [current];
2973-
// TODO (effects) Rename this to better reflect its new usage (e.g. ChildDeletions)
2974-
returnFiber.flags |= Deletion;
2975+
const last = returnFiber.lastEffect;
2976+
if (last !== null) {
2977+
last.nextEffect = current;
2978+
returnFiber.lastEffect = current;
29752979
} else {
2976-
deletions.push(current);
2980+
returnFiber.firstEffect = returnFiber.lastEffect = current;
29772981
}
2982+
current.nextEffect = null;
2983+
current.flags = Deletion;
29782984

29792985
newWorkInProgress.flags |= Placement;
29802986

@@ -3059,6 +3065,15 @@ function beginWork(
30593065
}
30603066
case Profiler:
30613067
if (enableProfilerTimer) {
3068+
// Profiler should only call onRender when one of its descendants actually rendered.
3069+
const hasChildWork = includesSomeLane(
3070+
renderLanes,
3071+
workInProgress.childLanes,
3072+
);
3073+
if (hasChildWork) {
3074+
workInProgress.flags |= Update;
3075+
}
3076+
30623077
// Reset effect durations for the next eventual effect phase.
30633078
// These are reset during render to allow the DevTools commit hook a chance to read them,
30643079
const stateNode = workInProgress.stateNode;
@@ -3165,6 +3180,7 @@ function beginWork(
31653180
// update in the past but didn't complete it.
31663181
renderState.rendering = null;
31673182
renderState.tail = null;
3183+
renderState.lastEffect = null;
31683184
}
31693185
pushSuspenseContext(workInProgress, suspenseStackCursor.current);
31703186

‎packages/react-reconciler/src/ReactFiberClassComponent.new.js

+6-46
Original file line numberDiff line numberDiff line change
@@ -12,14 +12,13 @@ import type {Lanes} from './ReactFiberLane';
1212
import type {UpdateQueue} from './ReactUpdateQueue.new';
1313

1414
import * as React from 'react';
15-
import {Update, Snapshot, MountLayoutDev} from './ReactFiberFlags';
15+
import {Update, Snapshot} from './ReactFiberFlags';
1616
import {
1717
debugRenderPhaseSideEffectsForStrictMode,
1818
disableLegacyContext,
1919
enableDebugTracing,
2020
enableSchedulingProfiler,
2121
warnAboutDeprecatedLifecycles,
22-
enableDoubleInvokingEffects,
2322
} from 'shared/ReactFeatureFlags';
2423
import ReactStrictModeWarnings from './ReactStrictModeWarnings.new';
2524
import {isMounted} from './ReactFiberTreeReflection';
@@ -30,13 +29,7 @@ import invariant from 'shared/invariant';
3029
import {REACT_CONTEXT_TYPE, REACT_PROVIDER_TYPE} from 'shared/ReactSymbols';
3130

3231
import {resolveDefaultProps} from './ReactFiberLazyComponent.new';
33-
import {
34-
BlockingMode,
35-
ConcurrentMode,
36-
DebugTracingMode,
37-
NoMode,
38-
StrictMode,
39-
} from './ReactTypeOfMode';
32+
import {DebugTracingMode, StrictMode} from './ReactTypeOfMode';
4033

4134
import {
4235
enqueueUpdate,
@@ -897,16 +890,7 @@ function mountClassInstance(
897890
}
898891

899892
if (typeof instance.componentDidMount === 'function') {
900-
if (
901-
__DEV__ &&
902-
enableDoubleInvokingEffects &&
903-
(workInProgress.mode & (BlockingMode | ConcurrentMode)) !== NoMode
904-
) {
905-
// Never double-invoke effects for legacy roots.
906-
workInProgress.flags |= MountLayoutDev | Update;
907-
} else {
908-
workInProgress.flags |= Update;
909-
}
893+
workInProgress.flags |= Update;
910894
}
911895
}
912896

@@ -976,15 +960,7 @@ function resumeMountClassInstance(
976960
// If an update was already in progress, we should schedule an Update
977961
// effect even though we're bailing out, so that cWU/cDU are called.
978962
if (typeof instance.componentDidMount === 'function') {
979-
if (
980-
__DEV__ &&
981-
enableDoubleInvokingEffects &&
982-
(workInProgress.mode & (BlockingMode | ConcurrentMode)) !== NoMode
983-
) {
984-
workInProgress.flags |= MountLayoutDev | Update;
985-
} else {
986-
workInProgress.flags |= Update;
987-
}
963+
workInProgress.flags |= Update;
988964
}
989965
return false;
990966
}
@@ -1027,29 +1003,13 @@ function resumeMountClassInstance(
10271003
}
10281004
}
10291005
if (typeof instance.componentDidMount === 'function') {
1030-
if (
1031-
__DEV__ &&
1032-
enableDoubleInvokingEffects &&
1033-
(workInProgress.mode & (BlockingMode | ConcurrentMode)) !== NoMode
1034-
) {
1035-
workInProgress.flags |= MountLayoutDev | Update;
1036-
} else {
1037-
workInProgress.flags |= Update;
1038-
}
1006+
workInProgress.flags |= Update;
10391007
}
10401008
} else {
10411009
// If an update was already in progress, we should schedule an Update
10421010
// effect even though we're bailing out, so that cWU/cDU are called.
10431011
if (typeof instance.componentDidMount === 'function') {
1044-
if (
1045-
__DEV__ &&
1046-
enableDoubleInvokingEffects &&
1047-
(workInProgress.mode & (BlockingMode | ConcurrentMode)) !== NoMode
1048-
) {
1049-
workInProgress.flags |= MountLayoutDev | Update;
1050-
} else {
1051-
workInProgress.flags |= Update;
1052-
}
1012+
workInProgress.flags |= Update;
10531013
}
10541014

10551015
// If shouldComponentUpdate returned false, we should still update the

0 commit comments

Comments
 (0)
Please sign in to comment.