From dcf2a6f6651c0791c2e87253a8815fcf3d53e4d2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebastian=20Markb=C3=A5ge?= Date: Wed, 30 Jul 2025 09:55:09 -0400 Subject: [PATCH 1/2] [DevTools] Keep a Suspense Tree Parellel to the Instance tree in the Backend (#34050) This keeps a data structure of Suspense boundaries and the root which can keep track which boundaries might participate in a loading sequence and everything that suspends them. This will power the Suspense tab. Now when you select a `` boundary the "suspended by" section shows the whole boundary instead of just that component. In the future, we'll likely need to add "Activity" boundaries to this tree as well, so that we can track what suspended the root of an Activity when filtering a subtree. Similar to how the root SuspenseNode now tracks suspending at the root. Maybe it's ok to just traverse to collect this information on-demand when you select one though since this doesn't contribute to the deduping. We'll also need to add implicit Suspense boundaries for the rows of a SuspenseList with `tail=hidden/collapsed`. --- .../src/backend/fiber/renderer.js | 358 +++++++++++++++++- 1 file changed, 340 insertions(+), 18 deletions(-) diff --git a/packages/react-devtools-shared/src/backend/fiber/renderer.js b/packages/react-devtools-shared/src/backend/fiber/renderer.js index b5b7fe9658ff3..46af24f27d1cb 100644 --- a/packages/react-devtools-shared/src/backend/fiber/renderer.js +++ b/packages/react-devtools-shared/src/backend/fiber/renderer.js @@ -11,6 +11,7 @@ import type { ReactComponentInfo, ReactDebugInfo, ReactAsyncInfo, + ReactIOInfo, } from 'shared/ReactTypes'; import { @@ -173,6 +174,7 @@ type FiberInstance = { logCount: number, // total number of errors/warnings last seen treeBaseDuration: number, // the profiled time of the last render of this subtree suspendedBy: null | Array, // things that suspended in the children position of this component + suspenseNode: null | SuspenseNode, data: Fiber, // one of a Fiber pair }; @@ -187,6 +189,7 @@ function createFiberInstance(fiber: Fiber): FiberInstance { logCount: 0, treeBaseDuration: 0, suspendedBy: null, + suspenseNode: null, data: fiber, }; } @@ -203,6 +206,7 @@ type FilteredFiberInstance = { logCount: number, // total number of errors/warnings last seen treeBaseDuration: number, // the profiled time of the last render of this subtree suspendedBy: null | Array, // not used + suspenseNode: null | SuspenseNode, data: Fiber, // one of a Fiber pair }; @@ -218,6 +222,7 @@ function createFilteredFiberInstance(fiber: Fiber): FilteredFiberInstance { logCount: 0, treeBaseDuration: 0, suspendedBy: null, + suspenseNode: null, data: fiber, }: any); } @@ -237,6 +242,7 @@ type VirtualInstance = { logCount: number, // total number of errors/warnings last seen treeBaseDuration: number, // the profiled time of the last render of this subtree suspendedBy: null | Array, // things that blocked the server component's child from rendering + suspenseNode: null, // The latest info for this instance. This can be updated over time and the // same info can appear in more than once ServerComponentInstance. data: ReactComponentInfo, @@ -255,12 +261,39 @@ function createVirtualInstance( logCount: 0, treeBaseDuration: 0, suspendedBy: null, + suspenseNode: null, data: debugEntry, }; } type DevToolsInstance = FiberInstance | VirtualInstance | FilteredFiberInstance; +type SuspenseNode = { + // The Instance can be a Suspense boundary, a SuspenseList Row, or HostRoot. + // It can also be disconnected from the main tree if it's a Filtered Instance. + instance: FiberInstance | FilteredFiberInstance, + parent: null | SuspenseNode, + firstChild: null | SuspenseNode, + nextSibling: null | SuspenseNode, + suspendedBy: Map>, // Tracks which data we're suspended by and the children that suspend it. + // Track whether any of the items in suspendedBy are unique this this Suspense boundaries or if they're all + // also in the parent sets. This determine whether this could contribute in the loading sequence. + hasUniqueSuspenders: boolean, +}; + +function createSuspenseNode( + instance: FiberInstance | FilteredFiberInstance, +): SuspenseNode { + return (instance.suspenseNode = { + instance: instance, + parent: null, + firstChild: null, + nextSibling: null, + suspendedBy: new Map(), + hasUniqueSuspenders: false, + }); +} + type getDisplayNameForFiberType = (fiber: Fiber) => string | null; type getTypeSymbolType = (type: any) => symbol | string | number; @@ -2367,18 +2400,162 @@ export function attach( // the current parent here as well. let reconcilingParent: null | DevToolsInstance = null; + let remainingReconcilingChildrenSuspenseNodes: null | SuspenseNode = null; + // The previously placed child. + let previouslyReconciledSiblingSuspenseNode: null | SuspenseNode = null; + // To save on stack allocation and ensure that they are updated as a pair, we also store + // the current parent here as well. + let reconcilingParentSuspenseNode: null | SuspenseNode = null; + + function isSuspenseInFallback(suspenseNode: SuspenseNode) { + const fiber = suspenseNode.instance.data; + return fiber.tag === SuspenseComponent && fiber.memoizedState !== null; + } + + function ioExistsInSuspenseAncestor( + suspenseNode: SuspenseNode, + ioInfo: ReactIOInfo, + ): boolean { + let ancestor = suspenseNode.parent; + while (ancestor !== null) { + if (ancestor.suspendedBy.has(ioInfo)) { + return true; + } + ancestor = ancestor.parent; + } + return false; + } + function insertSuspendedBy(asyncInfo: ReactAsyncInfo): void { + let parentSuspenseNode = reconcilingParentSuspenseNode; + while ( + parentSuspenseNode !== null && + isSuspenseInFallback(parentSuspenseNode) + ) { + // If we have something that suspends inside the fallback tree of a Suspense boundary, then + // we bubble that up to the nearest parent Suspense boundary that isn't in fallback mode. + parentSuspenseNode = parentSuspenseNode.parent; + } const parentInstance = reconcilingParent; - if (parentInstance === null) { - // Suspending at the root is not attributed to any particular component - // TODO: It should be attributed to the shell. + if (parentSuspenseNode !== null) { + const suspendedBy = parentSuspenseNode.suspendedBy; + const ioInfo = asyncInfo.awaited; + let suspendedBySet = suspendedBy.get(ioInfo); + if (suspendedBySet === undefined) { + suspendedBySet = new Set(); + suspendedBy.set(asyncInfo.awaited, suspendedBySet); + } + // The child of the Suspense boundary that was suspended on this, or null if suspended at the root. + // This is used to keep track of how many dependents are still alive and also to get information + // like owner instances to link down into the tree. + if (!suspendedBySet.has(parentInstance)) { + suspendedBySet.add(parentInstance); + if ( + !parentSuspenseNode.hasUniqueSuspenders && + !ioExistsInSuspenseAncestor(parentSuspenseNode, ioInfo) + ) { + // This didn't exist in the parent before, so let's mark this boundary as having a unique suspender. + parentSuspenseNode.hasUniqueSuspenders = true; + } + } + } + if (parentInstance !== null) { + // Suspending at the root is not attributed to any particular component other than the SuspenseNode. + const suspendedBy = parentInstance.suspendedBy; + if (suspendedBy === null) { + parentInstance.suspendedBy = [asyncInfo]; + } else if (suspendedBy.indexOf(asyncInfo) === -1) { + suspendedBy.push(asyncInfo); + } + } + } + + function getAwaitInSuspendedByFromIO( + suspensedBy: Array, + ioInfo: ReactIOInfo, + ): null | ReactAsyncInfo { + for (let i = 0; i < suspensedBy.length; i++) { + const asyncInfo = suspensedBy[i]; + if (asyncInfo.awaited === ioInfo) { + return asyncInfo; + } + } + return null; + } + + function unblockSuspendedBy( + parentSuspenseNode: SuspenseNode, + ioInfo: ReactIOInfo, + ): void { + const firstChild = parentSuspenseNode.firstChild; + if (firstChild === null) { return; } - const suspendedBy = parentInstance.suspendedBy; - if (suspendedBy === null) { - parentInstance.suspendedBy = [asyncInfo]; - } else if (suspendedBy.indexOf(asyncInfo) === -1) { - suspendedBy.push(asyncInfo); + let node: SuspenseNode = firstChild; + while (node !== null) { + if (node.suspendedBy.has(ioInfo)) { + // We have found a child boundary that depended on the unblocked I/O. + // It can now be marked as having unique suspenders. We can skip its children + // since they'll still be blocked by this one. + node.hasUniqueSuspenders = true; + } else if (node.firstChild !== null) { + node = node.firstChild; + continue; + } + while (node.nextSibling === null) { + if (node.parent === null || node.parent === parentSuspenseNode) { + return; + } + node = node.parent; + } + node = node.nextSibling; + } + } + + function removePreviousSuspendedBy( + instance: DevToolsInstance, + previousSuspendedBy: null | Array, + ): void { + // Remove any async info from the parent, if they were in the previous set but + // is no longer in the new set. + const parentSuspenseNode = reconcilingParentSuspenseNode; + if (previousSuspendedBy !== null && parentSuspenseNode !== null) { + const nextSuspendedBy = instance.suspendedBy; + for (let i = 0; i < previousSuspendedBy.length; i++) { + const asyncInfo = previousSuspendedBy[i]; + if ( + nextSuspendedBy === null || + (nextSuspendedBy.indexOf(asyncInfo) === -1 && + getAwaitInSuspendedByFromIO(nextSuspendedBy, asyncInfo.awaited) === + null) + ) { + // This IO entry is no longer blocking the current tree. + // Let's remove it from the parent SuspenseNode. + const ioInfo = asyncInfo.awaited; + const suspendedBySet = parentSuspenseNode.suspendedBy.get(ioInfo); + if ( + suspendedBySet === undefined || + !suspendedBySet.delete(instance) + ) { + throw new Error( + 'We are cleaning up async info that was not on the parent Suspense boundary. ' + + 'This is a bug in React.', + ); + } + if (suspendedBySet.size === 0) { + parentSuspenseNode.suspendedBy.delete(asyncInfo.awaited); + } + if ( + parentSuspenseNode.hasUniqueSuspenders && + !ioExistsInSuspenseAncestor(parentSuspenseNode, ioInfo) + ) { + // This entry wasn't in any ancestor and is no longer in this suspense boundary. + // This means that a child might now be the unique suspender for this IO. + // Search the child boundaries to see if we can reveal any of them. + unblockSuspendedBy(parentSuspenseNode, ioInfo); + } + } + } } } @@ -2398,6 +2575,22 @@ export function attach( previouslyReconciledSibling = instance; } instance.nextSibling = null; + // Insert any SuspenseNode into its parent Node. + const suspenseNode = instance.suspenseNode; + if (suspenseNode !== null) { + const parentNode = reconcilingParentSuspenseNode; + if (parentNode !== null) { + suspenseNode.parent = parentNode; + if (previouslyReconciledSiblingSuspenseNode === null) { + previouslyReconciledSiblingSuspenseNode = suspenseNode; + parentNode.firstChild = suspenseNode; + } else { + previouslyReconciledSiblingSuspenseNode.nextSibling = suspenseNode; + previouslyReconciledSiblingSuspenseNode = suspenseNode; + } + suspenseNode.nextSibling = null; + } + } } function moveChild( @@ -2447,6 +2640,36 @@ export function attach( } instance.nextSibling = null; instance.parent = null; + + // Remove any SuspenseNode from its parent. + const suspenseNode = instance.suspenseNode; + if (suspenseNode !== null && suspenseNode.parent !== null) { + const parentNode = reconcilingParentSuspenseNode; + if (parentNode === null) { + throw new Error('Should not have a parent if we are at the root'); + } + if (suspenseNode.parent !== parentNode) { + throw new Error( + 'Cannot remove a node from a different parent than is being reconciled.', + ); + } + let previousSuspenseSibling = remainingReconcilingChildrenSuspenseNodes; + if (previousSuspenseSibling === suspenseNode) { + // We're first in the remaining set. Remove us. + remainingReconcilingChildrenSuspenseNodes = suspenseNode.nextSibling; + } else { + // Search for our previous sibling and remove us. + while (previousSuspenseSibling !== null) { + if (previousSuspenseSibling.nextSibling === suspenseNode) { + previousSuspenseSibling.nextSibling = suspenseNode.nextSibling; + break; + } + previousSuspenseSibling = previousSuspenseSibling.nextSibling; + } + } + suspenseNode.nextSibling = null; + suspenseNode.parent = null; + } } function unmountRemainingChildren() { @@ -2654,20 +2877,27 @@ export function attach( ): void { const shouldIncludeInTree = !shouldFilterFiber(fiber); let newInstance = null; + let newSuspenseNode = null; if (shouldIncludeInTree) { newInstance = recordMount(fiber, reconcilingParent); + if (fiber.tag === SuspenseComponent || fiber.tag === HostRoot) { + newSuspenseNode = createSuspenseNode(newInstance); + } insertChild(newInstance); if (__DEBUG__) { debug('mountFiberRecursively()', newInstance, reconcilingParent); } } else if ( - reconcilingParent !== null && - reconcilingParent.kind === VIRTUAL_INSTANCE + (reconcilingParent !== null && + reconcilingParent.kind === VIRTUAL_INSTANCE) || + fiber.tag === SuspenseComponent ) { // If the parent is a Virtual Instance and we filtered this Fiber we include a - // hidden node. - + // hidden node. We also include this if it's a Suspense boundary so we can track those + // in the Suspense tree. if ( + reconcilingParent !== null && + reconcilingParent.kind === VIRTUAL_INSTANCE && reconcilingParent.data === fiber._debugOwner && fiber._debugStack != null && reconcilingParent.source === null @@ -2678,6 +2908,9 @@ export function attach( } newInstance = createFilteredFiberInstance(fiber); + if (fiber.tag === SuspenseComponent) { + newSuspenseNode = createSuspenseNode(newInstance); + } insertChild(newInstance); if (__DEBUG__) { debug('mountFiberRecursively()', newInstance, reconcilingParent); @@ -2694,12 +2927,20 @@ export function attach( const stashedParent = reconcilingParent; const stashedPrevious = previouslyReconciledSibling; const stashedRemaining = remainingReconcilingChildren; + const stashedSuspenseParent = reconcilingParentSuspenseNode; + const stashedSuspensePrevious = previouslyReconciledSiblingSuspenseNode; + const stashedSuspenseRemaining = remainingReconcilingChildrenSuspenseNodes; if (newInstance !== null) { // Push a new DevTools instance parent while reconciling this subtree. reconcilingParent = newInstance; previouslyReconciledSibling = null; remainingReconcilingChildren = null; } + if (newSuspenseNode !== null) { + reconcilingParentSuspenseNode = newSuspenseNode; + previouslyReconciledSiblingSuspenseNode = null; + remainingReconcilingChildrenSuspenseNodes = null; + } try { if (traceUpdatesEnabled) { if (traceNearestHostComponentUpdate) { @@ -2753,6 +2994,7 @@ export function attach( ); } } + // TODO: Track SuspenseNode in resuspended trees. } else { let primaryChild: Fiber | null = null; const areSuspenseChildrenConditionallyWrapped = @@ -2784,6 +3026,11 @@ export function attach( previouslyReconciledSibling = stashedPrevious; remainingReconcilingChildren = stashedRemaining; } + if (newSuspenseNode !== null) { + reconcilingParentSuspenseNode = stashedSuspenseParent; + previouslyReconciledSiblingSuspenseNode = stashedSuspensePrevious; + remainingReconcilingChildrenSuspenseNodes = stashedSuspenseRemaining; + } } // We're exiting this Fiber now, and entering its siblings. @@ -2801,6 +3048,10 @@ export function attach( const stashedParent = reconcilingParent; const stashedPrevious = previouslyReconciledSibling; const stashedRemaining = remainingReconcilingChildren; + const stashedSuspenseParent = reconcilingParentSuspenseNode; + const stashedSuspensePrevious = previouslyReconciledSiblingSuspenseNode; + const stashedSuspenseRemaining = remainingReconcilingChildrenSuspenseNodes; + const previousSuspendedBy = instance.suspendedBy; // Push a new DevTools instance parent while reconciling this subtree. reconcilingParent = instance; previouslyReconciledSibling = null; @@ -2808,13 +3059,24 @@ export function attach( remainingReconcilingChildren = instance.firstChild; instance.firstChild = null; instance.suspendedBy = null; + + if (instance.suspenseNode !== null) { + reconcilingParentSuspenseNode = instance.suspenseNode; + previouslyReconciledSiblingSuspenseNode = null; + remainingReconcilingChildrenSuspenseNodes = null; + } + try { // Unmount the remaining set. unmountRemainingChildren(); + removePreviousSuspendedBy(instance, previousSuspendedBy); } finally { reconcilingParent = stashedParent; previouslyReconciledSibling = stashedPrevious; remainingReconcilingChildren = stashedRemaining; + reconcilingParentSuspenseNode = stashedSuspenseParent; + previouslyReconciledSiblingSuspenseNode = stashedSuspensePrevious; + remainingReconcilingChildrenSuspenseNodes = stashedSuspenseRemaining; } if (instance.kind === FIBER_INSTANCE) { recordUnmount(instance); @@ -3001,6 +3263,7 @@ export function attach( const stashedParent = reconcilingParent; const stashedPrevious = previouslyReconciledSibling; const stashedRemaining = remainingReconcilingChildren; + const previousSuspendedBy = virtualInstance.suspendedBy; // Push a new DevTools instance parent while reconciling this subtree. reconcilingParent = virtualInstance; previouslyReconciledSibling = null; @@ -3021,6 +3284,7 @@ export function attach( ) { recordResetChildren(virtualInstance); } + removePreviousSuspendedBy(virtualInstance, previousSuspendedBy); // Update the errors/warnings count. If this Instance has switched to a different // ReactComponentInfo instance, such as when refreshing Server Components, then // we replace all the previous logs with the ones associated with the new ones rather @@ -3376,7 +3640,12 @@ export function attach( const stashedParent = reconcilingParent; const stashedPrevious = previouslyReconciledSibling; const stashedRemaining = remainingReconcilingChildren; + const stashedSuspenseParent = reconcilingParentSuspenseNode; + const stashedSuspensePrevious = previouslyReconciledSiblingSuspenseNode; + const stashedSuspenseRemaining = remainingReconcilingChildrenSuspenseNodes; + let previousSuspendedBy = null; if (fiberInstance !== null) { + previousSuspendedBy = fiberInstance.suspendedBy; // Update the Fiber so we that we always keep the current Fiber on the data. fiberInstance.data = nextFiber; if ( @@ -3396,6 +3665,12 @@ export function attach( remainingReconcilingChildren = fiberInstance.firstChild; fiberInstance.firstChild = null; fiberInstance.suspendedBy = null; + + if (fiberInstance.suspenseNode !== null) { + reconcilingParentSuspenseNode = fiberInstance.suspenseNode; + previouslyReconciledSiblingSuspenseNode = null; + remainingReconcilingChildrenSuspenseNodes = null; + } } try { if ( @@ -3550,6 +3825,8 @@ export function attach( } if (fiberInstance !== null) { + removePreviousSuspendedBy(fiberInstance, previousSuspendedBy); + let componentLogsEntry = fiberToComponentLogsMap.get( fiberInstance.data, ); @@ -3587,6 +3864,9 @@ export function attach( reconcilingParent = stashedParent; previouslyReconciledSibling = stashedPrevious; remainingReconcilingChildren = stashedRemaining; + reconcilingParentSuspenseNode = stashedSuspenseParent; + previouslyReconciledSiblingSuspenseNode = stashedSuspensePrevious; + remainingReconcilingChildrenSuspenseNodes = stashedSuspenseRemaining; } } } @@ -4104,6 +4384,43 @@ export function attach( return null; } + function getSuspendedByOfSuspenseNode( + suspenseNode: SuspenseNode, + ): Array { + // Collect all ReactAsyncInfo that was suspending this SuspenseNode but + // isn't also in any parent set. + const result: Array = []; + if (!suspenseNode.hasUniqueSuspenders) { + return result; + } + suspenseNode.suspendedBy.forEach((set, ioInfo) => { + let parentNode = suspenseNode.parent; + while (parentNode !== null) { + if (parentNode.suspendedBy.has(ioInfo)) { + return; + } + parentNode = parentNode.parent; + } + // We have the ioInfo but we need to find at least one corresponding await + // to go along with it. We don't really need to show every child that awaits the same + // thing so we just pick the first one that is still alive. + if (set.size === 0) { + return; + } + const firstInstance: DevToolsInstance = (set.values().next().value: any); + if (firstInstance.suspendedBy !== null) { + const asyncInfo = getAwaitInSuspendedByFromIO( + firstInstance.suspendedBy, + ioInfo, + ); + if (asyncInfo !== null) { + result.push(asyncInfo); + } + } + }); + return result; + } + function serializeAsyncInfo( asyncInfo: ReactAsyncInfo, index: number, @@ -4448,12 +4765,17 @@ export function attach( nativeTag = getNativeTag(fiber.stateNode); } - // This set is an edge case where if you pass a promise to a Client Component into a children - // position without a Server Component as the direct parent. E.g.
{promise}
- // In this case, this becomes associated with the Client/Host Component where as normally - // you'd expect these to be associated with the Server Component that awaited the data. - // TODO: Prepend other suspense sources like css, images and use(). - const suspendedBy = fiberInstance.suspendedBy; + const suspendedBy = + fiberInstance.suspenseNode !== null + ? // If this is a Suspense boundary, then we include everything in the subtree that might suspend + // this boundary down to the next Suspense boundary. + getSuspendedByOfSuspenseNode(fiberInstance.suspenseNode) + : // This set is an edge case where if you pass a promise to a Client Component into a children + // position without a Server Component as the direct parent. E.g.
{promise}
+ // In this case, this becomes associated with the Client/Host Component where as normally + // you'd expect these to be associated with the Server Component that awaited the data. + // TODO: Prepend other suspense sources like css, images and use(). + fiberInstance.suspendedBy; return { id: fiberInstance.id, From 9784cb379e249a5495cde5ba3037521207144e91 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebastian=20Markb=C3=A5ge?= Date: Wed, 30 Jul 2025 11:31:27 -0400 Subject: [PATCH 2/2] [DevTools] No suspending above the root (#34055) Follow up to #34050. It's not actually possible to suspend *above* the root since even if you suspend in the first child position, you're still suspending the HostRoot which always has a corresponding FiberInstance and SuspenseNode. --- .../src/backend/fiber/renderer.js | 61 ++++++++++--------- 1 file changed, 32 insertions(+), 29 deletions(-) diff --git a/packages/react-devtools-shared/src/backend/fiber/renderer.js b/packages/react-devtools-shared/src/backend/fiber/renderer.js index 46af24f27d1cb..7dac18289e88a 100644 --- a/packages/react-devtools-shared/src/backend/fiber/renderer.js +++ b/packages/react-devtools-shared/src/backend/fiber/renderer.js @@ -275,7 +275,7 @@ type SuspenseNode = { parent: null | SuspenseNode, firstChild: null | SuspenseNode, nextSibling: null | SuspenseNode, - suspendedBy: Map>, // Tracks which data we're suspended by and the children that suspend it. + suspendedBy: Map>, // Tracks which data we're suspended by and the children that suspend it. // Track whether any of the items in suspendedBy are unique this this Suspense boundaries or if they're all // also in the parent sets. This determine whether this could contribute in the loading sequence. hasUniqueSuspenders: boolean, @@ -2437,37 +2437,40 @@ export function attach( parentSuspenseNode = parentSuspenseNode.parent; } const parentInstance = reconcilingParent; - if (parentSuspenseNode !== null) { - const suspendedBy = parentSuspenseNode.suspendedBy; - const ioInfo = asyncInfo.awaited; - let suspendedBySet = suspendedBy.get(ioInfo); - if (suspendedBySet === undefined) { - suspendedBySet = new Set(); - suspendedBy.set(asyncInfo.awaited, suspendedBySet); - } - // The child of the Suspense boundary that was suspended on this, or null if suspended at the root. - // This is used to keep track of how many dependents are still alive and also to get information - // like owner instances to link down into the tree. - if (!suspendedBySet.has(parentInstance)) { - suspendedBySet.add(parentInstance); - if ( - !parentSuspenseNode.hasUniqueSuspenders && - !ioExistsInSuspenseAncestor(parentSuspenseNode, ioInfo) - ) { - // This didn't exist in the parent before, so let's mark this boundary as having a unique suspender. - parentSuspenseNode.hasUniqueSuspenders = true; - } - } + if (parentInstance === null || parentSuspenseNode === null) { + throw new Error( + 'It should not be possible to have suspended data outside the root. ' + + 'Even suspending at the first position is still a child of the root.', + ); } - if (parentInstance !== null) { - // Suspending at the root is not attributed to any particular component other than the SuspenseNode. - const suspendedBy = parentInstance.suspendedBy; - if (suspendedBy === null) { - parentInstance.suspendedBy = [asyncInfo]; - } else if (suspendedBy.indexOf(asyncInfo) === -1) { - suspendedBy.push(asyncInfo); + const suspenseNodeSuspendedBy = parentSuspenseNode.suspendedBy; + const ioInfo = asyncInfo.awaited; + let suspendedBySet = suspenseNodeSuspendedBy.get(ioInfo); + if (suspendedBySet === undefined) { + suspendedBySet = new Set(); + suspenseNodeSuspendedBy.set(asyncInfo.awaited, suspendedBySet); + } + // The child of the Suspense boundary that was suspended on this, or null if suspended at the root. + // This is used to keep track of how many dependents are still alive and also to get information + // like owner instances to link down into the tree. + if (!suspendedBySet.has(parentInstance)) { + suspendedBySet.add(parentInstance); + if ( + !parentSuspenseNode.hasUniqueSuspenders && + !ioExistsInSuspenseAncestor(parentSuspenseNode, ioInfo) + ) { + // This didn't exist in the parent before, so let's mark this boundary as having a unique suspender. + parentSuspenseNode.hasUniqueSuspenders = true; } } + // Suspending right below the root is not attributed to any particular component in UI + // other than the SuspenseNode and the HostRoot's FiberInstance. + const suspendedBy = parentInstance.suspendedBy; + if (suspendedBy === null) { + parentInstance.suspendedBy = [asyncInfo]; + } else if (suspendedBy.indexOf(asyncInfo) === -1) { + suspendedBy.push(asyncInfo); + } } function getAwaitInSuspendedByFromIO(