From 3fc4c496afdcfbbcea4bd52b0bf3287c7018ff4d Mon Sep 17 00:00:00 2001 From: Mark Plesko Date: Wed, 16 Oct 2024 12:30:23 -0700 Subject: [PATCH] ConditionalWeakTable - don't link Containers when no entries are transferred --- .../CompilerServices/ConditionalWeakTable.cs | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/src/libraries/System.Private.CoreLib/src/System/Runtime/CompilerServices/ConditionalWeakTable.cs b/src/libraries/System.Private.CoreLib/src/System/Runtime/CompilerServices/ConditionalWeakTable.cs index a50d4ff202a61..1b5c0894966d3 100644 --- a/src/libraries/System.Private.CoreLib/src/System/Runtime/CompilerServices/ConditionalWeakTable.cs +++ b/src/libraries/System.Private.CoreLib/src/System/Runtime/CompilerServices/ConditionalWeakTable.cs @@ -698,10 +698,13 @@ internal Container Resize(int newSize) Entry[] newEntries = new Entry[newSize]; int newEntriesIndex = 0; bool activeEnumerators = _parent != null && _parent._activeEnumeratorRefCount > 0; + bool transferredHandles; // Migrate existing entries to the new table. if (activeEnumerators) { + transferredHandles = true; + // There's at least one active enumerator, which means we don't want to // remove any expired/removed entries, in order to not affect existing // entries indices. Copy over the entries while rebuilding the buckets list, @@ -721,6 +724,8 @@ internal Container Resize(int newSize) } else { + transferredHandles = false; + // There are no active enumerators, which means we want to compact by // removing expired/removed entries. for (int entriesIndex = 0; entriesIndex < _entries.Length; entriesIndex++) @@ -732,6 +737,8 @@ internal Container Resize(int newSize) { if (depHnd.UnsafeGetTarget() is not null) { + transferredHandles = true; + ref Entry newEntry = ref newEntries[newEntriesIndex]; // Entry is used and has not expired. Link it into the appropriate bucket list. @@ -765,7 +772,13 @@ internal Container Resize(int newSize) // from being finalized, as it no longer has any responsibility for any cleanup. GC.SuppressFinalize(this); } - _oldKeepAlive = newContainer; // once this is set, the old container's finalizer will not free transferred dependent handles + + if (transferredHandles) + { + // Once this is set, the old container's finalizer will not free transferred dependent handles, + // and the new container's finalizer can't be run until this container is no longer in use. + _oldKeepAlive = newContainer; + } GC.KeepAlive(this); // ensure we don't get finalized while accessing DependentHandles.