From de75315d7ed388493f6d42fccd15ae3638adebe0 Mon Sep 17 00:00:00 2001 From: Andrew Clark Date: Mon, 16 Nov 2020 16:01:26 -0600 Subject: [PATCH] Track deletions using an array on the parent Adds back the `deletions` array and uses it in the commit phase. We use a trick where the first time we hit a deletion effect, we commit all the deletion effects that belong to that parent. This is an incremental step away from using the effect list and toward a DFS + subtreeFlags traversal. This will help determine whether the regression is caused by, say, pushing the same fiber into the deletions array multiple times. --- .../src/ReactChildFiber.new.js | 19 ++++++- .../src/ReactChildFiber.old.js | 20 +++++++- .../src/ReactFiberBeginWork.new.js | 48 ++++++++++++------ .../src/ReactFiberBeginWork.old.js | 48 ++++++++++++------ .../src/ReactFiberHooks.old.js | 3 -- .../src/ReactFiberHydrationContext.new.js | 11 ++++- .../src/ReactFiberHydrationContext.old.js | 11 ++++- .../src/ReactFiberWorkLoop.new.js | 33 ++++++++++++- .../src/ReactFiberWorkLoop.old.js | 49 ++++++++++++++++++- 9 files changed, 203 insertions(+), 39 deletions(-) diff --git a/packages/react-reconciler/src/ReactChildFiber.new.js b/packages/react-reconciler/src/ReactChildFiber.new.js index a92f6a6790d57..d8894768c6d91 100644 --- a/packages/react-reconciler/src/ReactChildFiber.new.js +++ b/packages/react-reconciler/src/ReactChildFiber.new.js @@ -13,7 +13,7 @@ import type {Fiber} from './ReactInternalTypes'; import type {Lanes} from './ReactFiberLane.new'; import getComponentName from 'shared/getComponentName'; -import {Placement, Deletion} from './ReactFiberFlags'; +import {Deletion, ChildDeletion, Placement} from './ReactFiberFlags'; import { getIteratorFn, REACT_ELEMENT_TYPE, @@ -276,6 +276,23 @@ function ChildReconciler(shouldTrackSideEffects) { } childToDelete.nextEffect = null; childToDelete.flags = Deletion; + + let deletions = returnFiber.deletions; + if (deletions === null) { + deletions = returnFiber.deletions = [childToDelete]; + returnFiber.flags |= ChildDeletion; + } else { + deletions.push(childToDelete); + } + // Stash a reference to the return fiber's deletion array on each of the + // deleted children. This is really weird, but it's a temporary workaround + // while we're still using the effect list to traverse effect fibers. A + // better workaround would be to follow the `.return` pointer in the commit + // phase, but unfortunately we can't assume that `.return` points to the + // correct fiber, even in the commit phase, because `findDOMNode` might + // mutate it. + // TODO: Remove this line. + childToDelete.deletions = deletions; } function deleteRemainingChildren( diff --git a/packages/react-reconciler/src/ReactChildFiber.old.js b/packages/react-reconciler/src/ReactChildFiber.old.js index 730f08663878c..268f71dd5308f 100644 --- a/packages/react-reconciler/src/ReactChildFiber.old.js +++ b/packages/react-reconciler/src/ReactChildFiber.old.js @@ -13,7 +13,7 @@ import type {Fiber} from './ReactInternalTypes'; import type {Lanes} from './ReactFiberLane.old'; import getComponentName from 'shared/getComponentName'; -import {Placement, Deletion} from './ReactFiberFlags'; +import {Deletion, ChildDeletion, Placement} from './ReactFiberFlags'; import { getIteratorFn, REACT_ELEMENT_TYPE, @@ -276,6 +276,23 @@ function ChildReconciler(shouldTrackSideEffects) { } childToDelete.nextEffect = null; childToDelete.flags = Deletion; + + let deletions = returnFiber.deletions; + if (deletions === null) { + deletions = returnFiber.deletions = [childToDelete]; + returnFiber.flags |= ChildDeletion; + } else { + deletions.push(childToDelete); + } + // Stash a reference to the return fiber's deletion array on each of the + // deleted children. This is really weird, but it's a temporary workaround + // while we're still using the effect list to traverse effect fibers. A + // better workaround would be to follow the `.return` pointer in the commit + // phase, but unfortunately we can't assume that `.return` points to the + // correct fiber, even in the commit phase, because `findDOMNode` might + // mutate it. + // TODO: Remove this line. + childToDelete.deletions = deletions; } function deleteRemainingChildren( @@ -1125,6 +1142,7 @@ function ChildReconciler(shouldTrackSideEffects) { } else { if ( child.elementType === elementType || + // Keep this check inline so it only runs on the false path: (__DEV__ ? isCompatibleFamilyForHotReloading(child, element) : false) || diff --git a/packages/react-reconciler/src/ReactFiberBeginWork.new.js b/packages/react-reconciler/src/ReactFiberBeginWork.new.js index 99aa9193e2f20..3ae9945a5707f 100644 --- a/packages/react-reconciler/src/ReactFiberBeginWork.new.js +++ b/packages/react-reconciler/src/ReactFiberBeginWork.new.js @@ -61,6 +61,7 @@ import { Update, Ref, Deletion, + ChildDeletion, ForceUpdateForLegacySuspense, } from './ReactFiberFlags'; import ReactSharedInternals from 'shared/ReactSharedInternals'; @@ -2007,6 +2008,14 @@ function updateSuspensePrimaryChildren( currentFallbackChildFragment.nextEffect = null; currentFallbackChildFragment.flags = Deletion; workInProgress.firstEffect = workInProgress.lastEffect = currentFallbackChildFragment; + let deletions = workInProgress.deletions; + if (deletions === null) { + deletions = workInProgress.deletions = [currentFallbackChildFragment]; + workInProgress.flags |= ChildDeletion; + } else { + deletions.push(currentFallbackChildFragment); + } + currentFallbackChildFragment.deletions = deletions; } workInProgress.child = primaryChildFragment; @@ -2061,21 +2070,23 @@ function updateSuspenseFallbackChildren( currentPrimaryChildFragment.treeBaseDuration; } - // The fallback fiber was added as a deletion effect during the first pass. - // However, since we're going to remain on the fallback, we no longer want - // to delete it. So we need to remove it from the list. Deletions are stored - // on the same list as effects. We want to keep the effects from the primary - // tree. So we copy the primary child fragment's effect list, which does not - // include the fallback deletion effect. - const progressedLastEffect = primaryChildFragment.lastEffect; - if (progressedLastEffect !== null) { - workInProgress.firstEffect = primaryChildFragment.firstEffect; - workInProgress.lastEffect = progressedLastEffect; - progressedLastEffect.nextEffect = null; - } else { - // TODO: Reset this somewhere else? Lol legacy mode is so weird. - workInProgress.firstEffect = workInProgress.lastEffect = null; + if (currentFallbackChildFragment !== null) { + // The fallback fiber was added as a deletion effect during the first + // pass. However, since we're going to remain on the fallback, we no + // longer want to delete it. So we need to remove it from the list. + // Deletions are stored on the same list as effects, and are always added + // to the front. So we know that the first effect must be the fallback + // deletion effect, and everything after that is from the primary free. + const firstPrimaryTreeEffect = currentFallbackChildFragment.nextEffect; + if (firstPrimaryTreeEffect !== null) { + workInProgress.firstEffect = firstPrimaryTreeEffect; + } else { + // TODO: Reset this somewhere else? Lol legacy mode is so weird. + workInProgress.firstEffect = workInProgress.lastEffect = null; + } } + + workInProgress.deletions = null; } else { primaryChildFragment = createWorkInProgressOffscreenFiber( currentPrimaryChildFragment, @@ -2982,6 +2993,15 @@ function remountFiber( current.nextEffect = null; current.flags = Deletion; + let deletions = returnFiber.deletions; + if (deletions === null) { + deletions = returnFiber.deletions = [current]; + returnFiber.flags |= ChildDeletion; + } else { + deletions.push(current); + } + current.deletions = deletions; + newWorkInProgress.flags |= Placement; // Restart work from the new fiber. diff --git a/packages/react-reconciler/src/ReactFiberBeginWork.old.js b/packages/react-reconciler/src/ReactFiberBeginWork.old.js index f875db5944954..7cd86c488c044 100644 --- a/packages/react-reconciler/src/ReactFiberBeginWork.old.js +++ b/packages/react-reconciler/src/ReactFiberBeginWork.old.js @@ -61,6 +61,7 @@ import { Update, Ref, Deletion, + ChildDeletion, ForceUpdateForLegacySuspense, } from './ReactFiberFlags'; import ReactSharedInternals from 'shared/ReactSharedInternals'; @@ -2007,6 +2008,14 @@ function updateSuspensePrimaryChildren( currentFallbackChildFragment.nextEffect = null; currentFallbackChildFragment.flags = Deletion; workInProgress.firstEffect = workInProgress.lastEffect = currentFallbackChildFragment; + let deletions = workInProgress.deletions; + if (deletions === null) { + deletions = workInProgress.deletions = [currentFallbackChildFragment]; + workInProgress.flags |= ChildDeletion; + } else { + deletions.push(currentFallbackChildFragment); + } + currentFallbackChildFragment.deletions = deletions; } workInProgress.child = primaryChildFragment; @@ -2061,21 +2070,23 @@ function updateSuspenseFallbackChildren( currentPrimaryChildFragment.treeBaseDuration; } - // The fallback fiber was added as a deletion effect during the first pass. - // However, since we're going to remain on the fallback, we no longer want - // to delete it. So we need to remove it from the list. Deletions are stored - // on the same list as effects. We want to keep the effects from the primary - // tree. So we copy the primary child fragment's effect list, which does not - // include the fallback deletion effect. - const progressedLastEffect = primaryChildFragment.lastEffect; - if (progressedLastEffect !== null) { - workInProgress.firstEffect = primaryChildFragment.firstEffect; - workInProgress.lastEffect = progressedLastEffect; - progressedLastEffect.nextEffect = null; - } else { - // TODO: Reset this somewhere else? Lol legacy mode is so weird. - workInProgress.firstEffect = workInProgress.lastEffect = null; + if (currentFallbackChildFragment !== null) { + // The fallback fiber was added as a deletion effect during the first + // pass. However, since we're going to remain on the fallback, we no + // longer want to delete it. So we need to remove it from the list. + // Deletions are stored on the same list as effects, and are always added + // to the front. So we know that the first effect must be the fallback + // deletion effect, and everything after that is from the primary free. + const firstPrimaryTreeEffect = currentFallbackChildFragment.nextEffect; + if (firstPrimaryTreeEffect !== null) { + workInProgress.firstEffect = firstPrimaryTreeEffect; + } else { + // TODO: Reset this somewhere else? Lol legacy mode is so weird. + workInProgress.firstEffect = workInProgress.lastEffect = null; + } } + + workInProgress.deletions = null; } else { primaryChildFragment = createWorkInProgressOffscreenFiber( currentPrimaryChildFragment, @@ -2982,6 +2993,15 @@ function remountFiber( current.nextEffect = null; current.flags = Deletion; + let deletions = returnFiber.deletions; + if (deletions === null) { + deletions = returnFiber.deletions = [current]; + returnFiber.flags |= ChildDeletion; + } else { + deletions.push(current); + } + current.deletions = deletions; + newWorkInProgress.flags |= Placement; // Restart work from the new fiber. diff --git a/packages/react-reconciler/src/ReactFiberHooks.old.js b/packages/react-reconciler/src/ReactFiberHooks.old.js index 0a7a99079c963..8206eb00f67e0 100644 --- a/packages/react-reconciler/src/ReactFiberHooks.old.js +++ b/packages/react-reconciler/src/ReactFiberHooks.old.js @@ -1864,9 +1864,6 @@ const HooksDispatcherOnMount: Dispatcher = { unstable_isNewReconciler: enableNewReconciler, }; -if (enableCache) { - (HooksDispatcherOnMount: Dispatcher).getCacheForType = getCacheForType; -} const HooksDispatcherOnUpdate: Dispatcher = { readContext, diff --git a/packages/react-reconciler/src/ReactFiberHydrationContext.new.js b/packages/react-reconciler/src/ReactFiberHydrationContext.new.js index 6ed0f6c8cbc55..85ddcca5f08eb 100644 --- a/packages/react-reconciler/src/ReactFiberHydrationContext.new.js +++ b/packages/react-reconciler/src/ReactFiberHydrationContext.new.js @@ -24,7 +24,7 @@ import { HostRoot, SuspenseComponent, } from './ReactWorkTags'; -import {Deletion, Placement, Hydrating} from './ReactFiberFlags'; +import {Deletion, ChildDeletion, Placement, Hydrating} from './ReactFiberFlags'; import invariant from 'shared/invariant'; import { @@ -137,6 +137,15 @@ function deleteHydratableInstance( } else { returnFiber.firstEffect = returnFiber.lastEffect = childToDelete; } + + let deletions = returnFiber.deletions; + if (deletions === null) { + deletions = returnFiber.deletions = [childToDelete]; + returnFiber.flags |= ChildDeletion; + } else { + deletions.push(childToDelete); + } + childToDelete.deletions = deletions; } function insertNonHydratedInstance(returnFiber: Fiber, fiber: Fiber) { diff --git a/packages/react-reconciler/src/ReactFiberHydrationContext.old.js b/packages/react-reconciler/src/ReactFiberHydrationContext.old.js index 5d084bf97badd..b9a5a22298998 100644 --- a/packages/react-reconciler/src/ReactFiberHydrationContext.old.js +++ b/packages/react-reconciler/src/ReactFiberHydrationContext.old.js @@ -24,7 +24,7 @@ import { HostRoot, SuspenseComponent, } from './ReactWorkTags'; -import {Deletion, Placement, Hydrating} from './ReactFiberFlags'; +import {Deletion, ChildDeletion, Placement, Hydrating} from './ReactFiberFlags'; import invariant from 'shared/invariant'; import { @@ -137,6 +137,15 @@ function deleteHydratableInstance( } else { returnFiber.firstEffect = returnFiber.lastEffect = childToDelete; } + + let deletions = returnFiber.deletions; + if (deletions === null) { + deletions = returnFiber.deletions = [childToDelete]; + returnFiber.flags |= ChildDeletion; + } else { + deletions.push(childToDelete); + } + childToDelete.deletions = deletions; } function insertNonHydratedInstance(returnFiber: Fiber, fiber: Fiber) { diff --git a/packages/react-reconciler/src/ReactFiberWorkLoop.new.js b/packages/react-reconciler/src/ReactFiberWorkLoop.new.js index dc1ff874b8825..2db6d5e7554cf 100644 --- a/packages/react-reconciler/src/ReactFiberWorkLoop.new.js +++ b/packages/react-reconciler/src/ReactFiberWorkLoop.new.js @@ -1821,6 +1821,7 @@ function completeUnitOfWork(unitOfWork: Fiber): void { // Mark the parent fiber as incomplete and clear its effect list. returnFiber.firstEffect = returnFiber.lastEffect = null; returnFiber.flags |= Incomplete; + returnFiber.deletions = null; } } @@ -2384,7 +2385,7 @@ function commitMutationEffects(root: FiberRoot, renderPriorityLevel) { // bitmap value, we remove the secondary effects from the effect tag and // switch on that value. const primaryFlags = flags & (Placement | Update | Deletion | Hydrating); - switch (primaryFlags) { + outer: switch (primaryFlags) { case Placement: { commitPlacement(nextEffect); // Clear the "placement" from effect tag so that we know that this is @@ -2424,7 +2425,35 @@ function commitMutationEffects(root: FiberRoot, renderPriorityLevel) { break; } case Deletion: { - commitDeletion(root, nextEffect, renderPriorityLevel); + // Reached a deletion effect. Instead of commit this effect like we + // normally do, we're going to use the `deletions` array of the parent. + // However, because the effect list is sorted in depth-first order, we + // can't wait until we reach the parent node, because the child effects + // will have run in the meantime. + // + // So instead, we use a trick where the first time we hit a deletion + // effect, we commit all the deletion effects that belong to that parent. + // + // This is an incremental step away from using the effect list and + // toward a DFS + subtreeFlags traversal. + // + // A reference to the deletion array of the parent is also stored on + // each of the deletions. This is really weird. It would be better to + // follow the `.return` pointer, but unfortunately we can't assume that + // `.return` points to the correct fiber, even in the commit phase, + // because `findDOMNode` might mutate it. + const deletedChild = nextEffect; + const deletions = deletedChild.deletions; + if (deletions !== null) { + for (let i = 0; i < deletions.length; i++) { + const deletion = deletions[i]; + // Clear the deletion effect so that we don't delete this node more + // than once. + deletion.flags &= ~Deletion; + deletion.deletions = null; + commitDeletion(root, deletion, renderPriorityLevel); + } + } break; } } diff --git a/packages/react-reconciler/src/ReactFiberWorkLoop.old.js b/packages/react-reconciler/src/ReactFiberWorkLoop.old.js index 57fce214dc47d..72c580dcf48d2 100644 --- a/packages/react-reconciler/src/ReactFiberWorkLoop.old.js +++ b/packages/react-reconciler/src/ReactFiberWorkLoop.old.js @@ -1821,6 +1821,7 @@ function completeUnitOfWork(unitOfWork: Fiber): void { // Mark the parent fiber as incomplete and clear its effect list. returnFiber.firstEffect = returnFiber.lastEffect = null; returnFiber.flags |= Incomplete; + returnFiber.deletions = null; } } @@ -2384,7 +2385,7 @@ function commitMutationEffects(root: FiberRoot, renderPriorityLevel) { // bitmap value, we remove the secondary effects from the effect tag and // switch on that value. const primaryFlags = flags & (Placement | Update | Deletion | Hydrating); - switch (primaryFlags) { + outer: switch (primaryFlags) { case Placement: { commitPlacement(nextEffect); // Clear the "placement" from effect tag so that we know that this is @@ -2424,7 +2425,35 @@ function commitMutationEffects(root: FiberRoot, renderPriorityLevel) { break; } case Deletion: { - commitDeletion(root, nextEffect, renderPriorityLevel); + // Reached a deletion effect. Instead of commit this effect like we + // normally do, we're going to use the `deletions` array of the parent. + // However, because the effect list is sorted in depth-first order, we + // can't wait until we reach the parent node, because the child effects + // will have run in the meantime. + // + // So instead, we use a trick where the first time we hit a deletion + // effect, we commit all the deletion effects that belong to that parent. + // + // This is an incremental step away from using the effect list and + // toward a DFS + subtreeFlags traversal. + // + // A reference to the deletion array of the parent is also stored on + // each of the deletions. This is really weird. It would be better to + // follow the `.return` pointer, but unfortunately we can't assume that + // `.return` points to the correct fiber, even in the commit phase, + // because `findDOMNode` might mutate it. + const deletedChild = nextEffect; + const deletions = deletedChild.deletions; + if (deletions !== null) { + for (let i = 0; i < deletions.length; i++) { + const deletion = deletions[i]; + // Clear the deletion effect so that we don't delete this node more + // than once. + deletion.flags &= ~Deletion; + deletion.deletions = null; + commitDeletion(root, deletion, renderPriorityLevel); + } + } break; } } @@ -2844,6 +2873,22 @@ export function captureCommitPhaseError(sourceFiber: Fiber, error: mixed) { } fiber = fiber.return; } + + if (__DEV__) { + // TODO: Until we re-land skipUnmountedBoundaries (see #20147), this warning + // will fire for errors that are thrown by destroy functions inside deleted + // trees. What it should instead do is propagate the error to the parent of + // the deleted tree. In the meantime, do not add this warning to the + // allowlist; this is only for our internal use. + console.error( + 'Internal React error: Attempted to capture a commit phase error ' + + 'inside a detached tree. This indicates a bug in React. Likely ' + + 'causes include deleting the same fiber more than once, committing an ' + + 'already-finished tree, or an inconsistent return pointer.\n\n' + + 'Error message:\n\n%s', + error, + ); + } } export function pingSuspendedRoot(