Skip to content

Commit ab29695

Browse files
committedDec 18, 2020
Defer more field detachments to passive phase
This allows us to use those fields during passive unmount traversal.
1 parent d37d7a4 commit ab29695

File tree

4 files changed

+80
-46
lines changed

4 files changed

+80
-46
lines changed
 

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

+27-15
Original file line numberDiff line numberDiff line change
@@ -1069,30 +1069,42 @@ function commitNestedUnmounts(
10691069
}
10701070

10711071
function detachFiberMutation(fiber: Fiber) {
1072-
// Cut off the return pointers to disconnect it from the tree. Ideally, we
1073-
// should clear the child pointer of the parent alternate to let this
1072+
// Cut off the return pointer to disconnect it from the tree.
1073+
// This enables us to detect and warn against state updates on an unmounted component.
1074+
// It also prevents events from bubbling from within disconnected components.
1075+
//
1076+
// Ideally, we should also clear the child pointer of the parent alternate to let this
10741077
// get GC:ed but we don't know which for sure which parent is the current
1075-
// one so we'll settle for GC:ing the subtree of this child. This child
1076-
// itself will be GC:ed when the parent updates the next time.
1077-
// Note: we cannot null out sibling here, otherwise it can cause issues
1078-
// with findDOMNode and how it requires the sibling field to carry out
1079-
// traversal in a later effect. See PR #16820. We now clear the sibling
1080-
// field after effects, see: detachFiberAfterEffects.
1078+
// one so we'll settle for GC:ing the subtree of this child.
1079+
// This child itself will be GC:ed when the parent updates the next time.
10811080
//
1082-
// Don't disconnect stateNode now; it will be detached in detachFiberAfterEffects.
1083-
// It may be required if the current component is an error boundary,
1084-
// and one of its descendants throws while unmounting a passive effect.
1085-
fiber.alternate = null;
1081+
// Note that we can't clear child or sibling pointers yet.
1082+
// They're needed for passive effects and for findDOMNode.
1083+
// We defer those fields, and all other cleanup, to the passive phase (see detachFiberAfterEffects).
1084+
const alternate = fiber.alternate;
1085+
if (alternate !== null) {
1086+
alternate.return = null;
1087+
fiber.alternate = null;
1088+
}
1089+
fiber.return = null;
1090+
}
1091+
1092+
export function detachFiberAfterEffects(fiber: Fiber): void {
1093+
// Null out fields to improve GC for references that may be lingering (e.g. DevTools).
1094+
// Note that we already cleared the return pointer in detachFiberMutation().
10861095
fiber.child = null;
10871096
fiber.deletions = null;
10881097
fiber.dependencies = null;
1089-
fiber.firstEffect = null;
1090-
fiber.lastEffect = null;
10911098
fiber.memoizedProps = null;
10921099
fiber.memoizedState = null;
10931100
fiber.pendingProps = null;
1094-
fiber.return = null;
1101+
fiber.sibling = null;
1102+
fiber.stateNode = null;
10951103
fiber.updateQueue = null;
1104+
fiber.nextEffect = null;
1105+
fiber.firstEffect = null;
1106+
fiber.lastEffect = null;
1107+
10961108
if (__DEV__) {
10971109
fiber._debugOwner = null;
10981110
}

‎packages/react-reconciler/src/ReactFiberCommitWork.old.js

+27-15
Original file line numberDiff line numberDiff line change
@@ -1069,30 +1069,42 @@ function commitNestedUnmounts(
10691069
}
10701070

10711071
function detachFiberMutation(fiber: Fiber) {
1072-
// Cut off the return pointers to disconnect it from the tree. Ideally, we
1073-
// should clear the child pointer of the parent alternate to let this
1072+
// Cut off the return pointer to disconnect it from the tree.
1073+
// This enables us to detect and warn against state updates on an unmounted component.
1074+
// It also prevents events from bubbling from within disconnected components.
1075+
//
1076+
// Ideally, we should also clear the child pointer of the parent alternate to let this
10741077
// get GC:ed but we don't know which for sure which parent is the current
1075-
// one so we'll settle for GC:ing the subtree of this child. This child
1076-
// itself will be GC:ed when the parent updates the next time.
1077-
// Note: we cannot null out sibling here, otherwise it can cause issues
1078-
// with findDOMNode and how it requires the sibling field to carry out
1079-
// traversal in a later effect. See PR #16820. We now clear the sibling
1080-
// field after effects, see: detachFiberAfterEffects.
1078+
// one so we'll settle for GC:ing the subtree of this child.
1079+
// This child itself will be GC:ed when the parent updates the next time.
10811080
//
1082-
// Don't disconnect stateNode now; it will be detached in detachFiberAfterEffects.
1083-
// It may be required if the current component is an error boundary,
1084-
// and one of its descendants throws while unmounting a passive effect.
1085-
fiber.alternate = null;
1081+
// Note that we can't clear child or sibling pointers yet.
1082+
// They're needed for passive effects and for findDOMNode.
1083+
// We defer those fields, and all other cleanup, to the passive phase (see detachFiberAfterEffects).
1084+
const alternate = fiber.alternate;
1085+
if (alternate !== null) {
1086+
alternate.return = null;
1087+
fiber.alternate = null;
1088+
}
1089+
fiber.return = null;
1090+
}
1091+
1092+
export function detachFiberAfterEffects(fiber: Fiber): void {
1093+
// Null out fields to improve GC for references that may be lingering (e.g. DevTools).
1094+
// Note that we already cleared the return pointer in detachFiberMutation().
10861095
fiber.child = null;
10871096
fiber.deletions = null;
10881097
fiber.dependencies = null;
1089-
fiber.firstEffect = null;
1090-
fiber.lastEffect = null;
10911098
fiber.memoizedProps = null;
10921099
fiber.memoizedState = null;
10931100
fiber.pendingProps = null;
1094-
fiber.return = null;
1101+
fiber.sibling = null;
1102+
fiber.stateNode = null;
10951103
fiber.updateQueue = null;
1104+
fiber.nextEffect = null;
1105+
fiber.firstEffect = null;
1106+
fiber.lastEffect = null;
1107+
10961108
if (__DEV__) {
10971109
fiber._debugOwner = null;
10981110
}

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

+13-8
Original file line numberDiff line numberDiff line change
@@ -122,6 +122,7 @@ import {
122122
Update,
123123
PlacementAndUpdate,
124124
Deletion,
125+
ChildDeletion,
125126
Ref,
126127
ContentReset,
127128
Snapshot,
@@ -194,6 +195,7 @@ import {
194195
commitResetTextContent,
195196
isSuspenseBoundaryBeingHidden,
196197
commitPassiveMountEffects,
198+
detachFiberAfterEffects,
197199
} from './ReactFiberCommitWork.new';
198200
import {enqueueUpdate} from './ReactUpdateQueue.new';
199201
import {resetContextDependencies} from './ReactFiberNewContext.new';
@@ -2129,13 +2131,21 @@ function commitRootImpl(root, renderPriorityLevel) {
21292131
} else {
21302132
// We are done with the effect chain at this point so let's clear the
21312133
// nextEffect pointers to assist with GC. If we have passive effects, we'll
2132-
// clear this in flushPassiveEffects.
2134+
// clear this in flushPassiveEffects
2135+
// TODO: We should always do this in the passive phase, by scheduling
2136+
// a passive callback for every deletion.
21332137
nextEffect = firstEffect;
21342138
while (nextEffect !== null) {
21352139
const nextNextEffect = nextEffect.nextEffect;
21362140
nextEffect.nextEffect = null;
2137-
if (nextEffect.flags & Deletion) {
2138-
detachFiberAfterEffects(nextEffect);
2141+
if (nextEffect.flags & ChildDeletion) {
2142+
const deletions = nextEffect.deletions;
2143+
if (deletions !== null) {
2144+
for (let i = 0; i < deletions.length; i++) {
2145+
const deletion = deletions[i];
2146+
detachFiberAfterEffects(deletion);
2147+
}
2148+
}
21392149
}
21402150
nextEffect = nextNextEffect;
21412151
}
@@ -3708,8 +3718,3 @@ export function act(callback: () => Thenable<mixed>): Thenable<void> {
37083718
};
37093719
}
37103720
}
3711-
3712-
function detachFiberAfterEffects(fiber: Fiber): void {
3713-
fiber.sibling = null;
3714-
fiber.stateNode = null;
3715-
}

‎packages/react-reconciler/src/ReactFiberWorkLoop.old.js

+13-8
Original file line numberDiff line numberDiff line change
@@ -122,6 +122,7 @@ import {
122122
Update,
123123
PlacementAndUpdate,
124124
Deletion,
125+
ChildDeletion,
125126
Ref,
126127
ContentReset,
127128
Snapshot,
@@ -194,6 +195,7 @@ import {
194195
commitResetTextContent,
195196
isSuspenseBoundaryBeingHidden,
196197
commitPassiveMountEffects,
198+
detachFiberAfterEffects,
197199
} from './ReactFiberCommitWork.old';
198200
import {enqueueUpdate} from './ReactUpdateQueue.old';
199201
import {resetContextDependencies} from './ReactFiberNewContext.old';
@@ -2129,13 +2131,21 @@ function commitRootImpl(root, renderPriorityLevel) {
21292131
} else {
21302132
// We are done with the effect chain at this point so let's clear the
21312133
// nextEffect pointers to assist with GC. If we have passive effects, we'll
2132-
// clear this in flushPassiveEffects.
2134+
// clear this in flushPassiveEffects
2135+
// TODO: We should always do this in the passive phase, by scheduling
2136+
// a passive callback for every deletion.
21332137
nextEffect = firstEffect;
21342138
while (nextEffect !== null) {
21352139
const nextNextEffect = nextEffect.nextEffect;
21362140
nextEffect.nextEffect = null;
2137-
if (nextEffect.flags & Deletion) {
2138-
detachFiberAfterEffects(nextEffect);
2141+
if (nextEffect.flags & ChildDeletion) {
2142+
const deletions = nextEffect.deletions;
2143+
if (deletions !== null) {
2144+
for (let i = 0; i < deletions.length; i++) {
2145+
const deletion = deletions[i];
2146+
detachFiberAfterEffects(deletion);
2147+
}
2148+
}
21392149
}
21402150
nextEffect = nextNextEffect;
21412151
}
@@ -3708,8 +3718,3 @@ export function act(callback: () => Thenable<mixed>): Thenable<void> {
37083718
};
37093719
}
37103720
}
3711-
3712-
function detachFiberAfterEffects(fiber: Fiber): void {
3713-
fiber.sibling = null;
3714-
fiber.stateNode = null;
3715-
}

0 commit comments

Comments
 (0)
Please sign in to comment.