File: SourceGeneration\Nodes\NodeStateTable.cs
Web Access
Project: src\src\Compilers\Core\Portable\Microsoft.CodeAnalysis.csproj (Microsoft.CodeAnalysis)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
using System;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using Microsoft.CodeAnalysis.PooledObjects;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis
{
    // A node table is the fundamental structure we use to track changes through the incremental 
    // generator api. It can be thought of as a series of slots that take their input from an
    // upstream table and produce 0-or-more outputs. When viewed from a downstream table the outputs
    // are presented as a single unified list, with each output forming the new input to the downstream
    // table.
    // 
    // Each slot has an associated state which is used to inform the operation that should be performed
    // to create or update the outputs. States generally flow through from upstream to downstream tables.
    // For instance an Added state implies that the upstream table produced a value that was not seen 
    // in the previous iteration, and the table should run whatever transform it tracks on the input 
    // to produce the outputs. These new outputs will also have a state of Added. A cached input specifies
    // that the input has not changed, and thus the outputs will be the same as the previous run. Added,
    // and Modified inputs will always run a transform to produce new outputs. Cached and Removed
    // entries will always use the previous entries and perform no work.
    // 
    // It is important to track Removed entries while updating the downstream tables, as an upstream 
    // remove can result in multiple downstream entries being removed. However, once all tables are up 
    // to date, the removed entries are no longer needed, and the remaining entries can be considered to
    // be cached. This process is called 'compaction' and results in the actual tables which are stored
    // between runs, as opposed to the 'live' tables that exist during an update.
    //
    // Modified entries are similar to added inputs, but with a subtle difference. When an input is Added
    // all outputs are unconditionally added too. However when an input is modified, the outputs may still
    // be the same (for instance something changed elsewhere in a file that had no bearing on the produced
    // output). In this case, the state table checks the results against the previously produced values,
    // and any that are found to be the same instead get a cached state, meaning no new downstream work 
    // will be produced for them. Thus a modified input is the only slot that can have differing output 
    // states.
 
    internal enum EntryState { Added, Removed, Modified, Cached };
 
    internal interface IStateTable
    {
        IStateTable AsCached();
 
        bool HasTrackedSteps { get; }
        ImmutableArray<IncrementalGeneratorRunStep> Steps { get; }
    }
 
    internal readonly record struct NodeStateEntry<T>(T Item, EntryState State, int OutputIndex, IncrementalGeneratorRunStep? Step);
 
    /// <summary>
    /// A data structure that tracks the inputs and output of an execution node
    /// </summary>
    /// <typeparam name="T">The type of the items tracked by this table</typeparam>
    internal sealed class NodeStateTable<T> : IStateTable
    {
        internal static NodeStateTable<T> Empty { get; } = new NodeStateTable<T>(ImmutableArray<TableEntry>.Empty, ImmutableArray<IncrementalGeneratorRunStep>.Empty, hasTrackedSteps: true, isCached: false);
 
        private readonly ImmutableArray<TableEntry> _states;
 
        private NodeStateTable(ImmutableArray<TableEntry> states, ImmutableArray<IncrementalGeneratorRunStep> steps, bool hasTrackedSteps, bool isCached)
        {
            Debug.Assert(!hasTrackedSteps || steps.Length == states.Length);
 
            _states = states;
            Steps = steps;
            IsCached = isCached;
            HasTrackedSteps = hasTrackedSteps;
        }
 
        public int Count => _states.Length;
 
        /// <summary>
        /// Indicates that this table is unchanged from the previous version.
        /// </summary>
        public bool IsCached { get; }
 
        public bool IsEmpty => _states.IsEmpty;
 
        public bool HasTrackedSteps { get; }
 
        public ImmutableArray<IncrementalGeneratorRunStep> Steps { get; }
 
        public int GetTotalEntryItemCount()
            => _states.Sum(static e => e.Count);
 
        public struct Enumerator
        {
            private readonly NodeStateTable<T> _stateTable;
            private int _nextStatesIndex;
            private int _nextInputEntryIndex;
            private IncrementalGeneratorRunStep? _step;
            private TableEntry _inputEntry;
            private NodeStateEntry<T> _current;
 
            public Enumerator(NodeStateTable<T> stateTable)
            {
                _stateTable = stateTable;
                _nextStatesIndex = 0;
 
                UpdateAfterNextStatesIndexModification();
            }
 
            public NodeStateEntry<T> Current => _current;
 
            public bool MoveNext()
            {
                while (_nextStatesIndex < _stateTable._states.Length)
                {
                    if (_nextInputEntryIndex < _inputEntry.Count)
                    {
                        _current = new NodeStateEntry<T>(_inputEntry.GetItem(_nextInputEntryIndex), _inputEntry.GetState(_nextInputEntryIndex), _nextInputEntryIndex, _step);
                        _nextInputEntryIndex += 1;
 
                        return true;
                    }
 
                    _nextStatesIndex += 1;
 
                    UpdateAfterNextStatesIndexModification();
                }
 
                return false;
            }
 
            private void UpdateAfterNextStatesIndexModification()
            {
                _nextInputEntryIndex = 0;
 
                if (_nextStatesIndex < _stateTable._states.Length)
                {
                    _step = _stateTable.HasTrackedSteps ? _stateTable.Steps[_nextStatesIndex] : null;
                    _inputEntry = _stateTable._states[_nextStatesIndex];
                }
            }
        }
 
        public Enumerator GetEnumerator()
        {
            return new Enumerator(this);
        }
 
        public NodeStateTable<T> AsCached()
        {
            if (IsCached)
                return this;
 
            // If the input to an entry was removed, we also need to remove the entry.
            // However, if the input was present, but the entry didn't produce any values (or removed them all),
            // we need to keep the empty entry as a placeholder, so that on the next generation pass
            // we can retrieve it as the cached value of the input.
            var nonRemovedCount = _states.Count(static e => !e.IsRemovedDueToInputRemoval);
 
            var compacted = ArrayBuilder<TableEntry>.GetInstance(nonRemovedCount);
            foreach (var entry in _states)
            {
                if (!entry.IsRemovedDueToInputRemoval)
                    compacted.Add(entry.AsCached());
            }
 
            // When we're preparing a table for caching between runs, we drop the step information as we cannot guarantee the graph structure while also updating
            // the input states
 
            // Ensure we are completely full so that ToImmutable translates to a MoveToImmutable
            Debug.Assert(compacted.Count == nonRemovedCount);
            return new NodeStateTable<T>(compacted.ToImmutableAndFree(), ImmutableArray<IncrementalGeneratorRunStep>.Empty, hasTrackedSteps: false, isCached: true);
        }
 
        IStateTable IStateTable.AsCached() => AsCached();
 
        public (T item, IncrementalGeneratorRunStep? step) Single()
        {
            Debug.Assert((_states.Length == 1 || _states.Length == 2 && _states[0].IsRemovedDueToInputRemoval) && _states[^1].Count == 1);
            return (_states[^1].GetItem(0), HasTrackedSteps ? Steps[^1] : null);
        }
 
        public Builder ToBuilder(string? stepName, bool stepTrackingEnabled, IEqualityComparer<T>? equalityComparer = null, int? tableCapacity = null)
            => new(this, stepName, stepTrackingEnabled, equalityComparer, tableCapacity);
 
        public NodeStateTable<T> CreateCachedTableWithUpdatedSteps<TInput>(NodeStateTable<TInput> inputTable, string? stepName, IEqualityComparer<T>? equalityComparer)
        {
            Debug.Assert(inputTable.HasTrackedSteps && inputTable.IsCached);
            NodeStateTable<T>.Builder builder = ToBuilder(stepName, stepTrackingEnabled: true, equalityComparer);
            foreach (var entry in inputTable)
            {
                var inputs = ImmutableArray.Create((entry.Step!, entry.OutputIndex));
                bool usedCachedEntry = builder.TryUseCachedEntries(TimeSpan.Zero, inputs);
                Debug.Assert(usedCachedEntry);
            }
 
            return builder.ToImmutableAndFree();
        }
 
        public string GetPackedStates()
        {
            var pooled = PooledStringBuilder.GetInstance();
            foreach (var state in _states)
            {
                for (int i = 0; i < state.Count; i++)
                {
                    var packedChar = state.GetState(i) switch
                    {
                        EntryState.Added => 'A',
                        EntryState.Removed => 'R',
                        EntryState.Modified => 'M',
                        EntryState.Cached => 'C',
                        _ => throw ExceptionUtilities.Unreachable(),
                    };
 
                    pooled.Builder.Append(packedChar);
                }
                pooled.Builder.Append(',');
            }
            return pooled.ToStringAndFree();
        }
 
        /// <remarks>
        /// The builder is <b>not</b> threadsafe.
        /// </remarks>
        public sealed class Builder
        {
            private readonly ArrayBuilder<TableEntry> _states;
            private readonly NodeStateTable<T> _previous;
 
            private readonly string? _name;
            private readonly IEqualityComparer<T> _equalityComparer;
            private readonly ArrayBuilder<IncrementalGeneratorRunStep>? _steps;
 
            private int _insertedCount = 0;
 
            [MemberNotNullWhen(true, nameof(_steps))]
            public bool TrackIncrementalSteps => _steps is not null;
 
#if DEBUG
            private readonly int? _requestedTableCapacity;
#endif
 
            internal Builder(
                NodeStateTable<T> previous,
                string? name,
                bool stepTrackingEnabled,
                IEqualityComparer<T>? equalityComparer,
                int? tableCapacity)
            {
#if DEBUG
                _requestedTableCapacity = tableCapacity;
#endif
                // If the caller specified a desired capacity, then use that.  Otherwise, use the previous table's total
                // entry count as a reasonable approximation for what we will need.
                _states = ArrayBuilder<TableEntry>.GetInstance(tableCapacity ?? previous.GetTotalEntryItemCount());
                _previous = previous;
                _name = name;
                _equalityComparer = equalityComparer ?? WrappedUserComparer<T>.Default;
                if (stepTrackingEnabled)
                {
                    _steps = ArrayBuilder<IncrementalGeneratorRunStep>.GetInstance();
                }
            }
 
            public int Count => _states.Count;
 
            public bool TryRemoveEntries(TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs)
            {
                if (!TryGetPreviousEntry(out var previousEntry))
                {
                    // The previous table had less node executions than this one, so we don't have any entries from a previous corresponding node execution to remove.
                    return false;
                }
 
                // Mark the corresponding entries to this node execution in the previous table as removed.
                // Since they are removed due to their input having been removed, we won't have to keep placeholders for them.
                var previousEntries = previousEntry.AsRemovedDueToInputRemoval();
                _states.Add(previousEntries);
                RecordStepInfoForLastEntry(elapsedTime, stepInputs, EntryState.Removed);
                return true;
            }
 
            public bool TryRemoveEntries(TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs, out OneOrMany<T> entries)
            {
                if (!TryRemoveEntries(elapsedTime, stepInputs))
                {
                    entries = default;
                    return false;
                }
 
                entries = _states[^1].Items;
                return true;
            }
 
            public bool TryUseCachedEntries(TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs)
            {
                if (!TryGetPreviousEntry(out var previousEntries))
                {
                    // The previous table had less node executions than this one, so we don't have any entries from a previous corresponding node execution to copy as cached.
                    return false;
                }
 
                Debug.Assert(previousEntries.IsCached);
 
                _states.Add(previousEntries);
                RecordStepInfoForLastEntry(elapsedTime, stepInputs, EntryState.Cached);
                return true;
            }
 
            internal bool TryUseCachedEntries(TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs, out TableEntry entry)
            {
                if (!TryUseCachedEntries(elapsedTime, stepInputs))
                {
                    entry = default;
                    return false;
                }
 
                entry = _states[^1];
                return true;
            }
 
            public bool TryModifyEntry(T value, TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs, EntryState overallInputState)
            {
                if (!TryGetPreviousEntry(out var previousEntry))
                {
                    // The previous table had less node executions than this one, so we don't have any entries from a previous corresponding node execution to try to modify.
                    return false;
                }
 
                if (previousEntry.Count == 0)
                {
                    // it's possible that the previous execution removed this item, but we left in an empty entry as a placeholder. In which case, we can't modify it
                    return false;
                }
 
                Debug.Assert(previousEntry.Count == 1);
                var (chosen, state, _) = GetModifiedItemAndState(previousEntry.GetItem(0), value);
                _states.Add(new TableEntry(OneOrMany.Create(chosen), state));
                RecordStepInfoForLastEntry(elapsedTime, stepInputs, overallInputState);
                return true;
            }
 
            public bool TryModifyEntries(ImmutableArray<T> outputs, TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs, EntryState overallInputState)
            {
                // Semantics:
                // For each item in the row, we compare with the new matching new value.
                // - Cached when the same
                // - Modified when different
                // - Removed when old item position > outputs.length
                // - Added when new item position < previousTable.length
 
                if (!TryGetPreviousEntry(out var previousEntry))
                {
                    return false;
                }
 
                // when both entries have no items, we can short circuit
                if (previousEntry.Count == 0 && outputs.Length == 0)
                {
                    _states.Add(previousEntry);
                    if (TrackIncrementalSteps)
                    {
                        RecordStepInfoForLastEntry(elapsedTime, stepInputs, EntryState.Cached);
                    }
 
                    return true;
                }
 
                // We may be able to move the previous entry over wholesale.  So avoid creating an builder and doing any
                // expensive work there until necessary (e.g. we detected either a different item or a different state).
                // We can only do this if the counts of before/after are the same. If not, then obviously something
                // changed and we can't reuse the before item.
 
                var totalBuilderItems = Math.Max(previousEntry.Count, outputs.Length);
                var builder = previousEntry.Count == outputs.Length ? null : new TableEntry.Builder(capacity: totalBuilderItems);
 
                var sharedCount = Math.Min(previousEntry.Count, outputs.Length);
 
                // cached or modified items
                for (int i = 0; i < sharedCount; i++)
                {
                    var previousItem = previousEntry.GetItem(i);
                    var previousState = previousEntry.GetState(i);
                    var replacementItem = outputs[i];
 
                    var (chosenItem, state, chosePrevious) = GetModifiedItemAndState(previousItem, replacementItem);
 
                    if (builder != null)
                    {
                        // if we have a builder, then we're keeping track of all entries no matter what.
                        builder.Add(chosenItem, state);
                        continue;
                    }
 
                    if (!chosePrevious || state != previousState)
                    {
                        // We don't have a builder, but we also can't use the previous entry.  Make a builder, copy
                        // everything prior to this point to it, and then add the latest entry.
                        builder = new TableEntry.Builder(capacity: totalBuilderItems);
                        for (int j = 0; j < i; j++)
                            builder.Add(previousEntry.GetItem(j), previousEntry.GetState(j));
 
                        builder.Add(chosenItem, state);
                        continue;
                    }
 
                    // otherwise, we don't have a builder and we are still able to use the previous entry.  Keep going
                    // without constructing anything.
                }
 
                // removed
                for (int i = sharedCount; i < previousEntry.Count; i++)
                {
                    // We know we must have a builder because we only get into this path when the counts are different
                    // (and thus we created a builder at the start).
                    builder!.Add(previousEntry.GetItem(i), EntryState.Removed);
                }
 
                // added
                for (int i = sharedCount; i < outputs.Length; i++)
                {
                    // We know we must have a builder because we only get into this path when the counts are different
                    // (and thus we created a builder at the start).
                    builder!.Add(outputs[i], EntryState.Added);
                }
 
                // If we still don't have a builder, then we can reuse the previous table entry entirely.  Otherwise,
                // construct the new one from the values collected.
                _states.Add(builder == null ? previousEntry : builder.ToImmutableAndFree());
 
                RecordStepInfoForLastEntry(elapsedTime, stepInputs, overallInputState);
                return true;
            }
 
            public bool TryModifyEntries(ImmutableArray<T> outputs, TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs, EntryState overallInputState, out TableEntry entry)
            {
                if (!TryModifyEntries(outputs, elapsedTime, stepInputs, overallInputState))
                {
                    entry = default;
                    return false;
                }
 
                entry = _states[^1];
                return true;
            }
 
            public void AddEntry(T value, EntryState state, TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs, EntryState overallInputState)
            {
                _states.Add(new TableEntry(OneOrMany.Create(value), state));
                _insertedCount += state == EntryState.Added ? 1 : 0;
                RecordStepInfoForLastEntry(elapsedTime, stepInputs, overallInputState);
            }
 
            public TableEntry AddEntries(ImmutableArray<T> values, EntryState state, TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs, EntryState overallInputState)
            {
                var tableEntry = new TableEntry(OneOrMany.Create(values), state);
                _states.Add(tableEntry);
                _insertedCount += state == EntryState.Added ? 1 : 0;
                RecordStepInfoForLastEntry(elapsedTime, stepInputs, overallInputState);
                return tableEntry;
            }
 
            private bool TryGetPreviousEntry(out TableEntry previousEntry)
            {
                // When indexing into the previous table we need to subtract the number of entries that have been explicitly added
                // to the current table, as they didn't exist in the previous one.
                var previousTableEntryIndex = _states.Count - _insertedCount;
 
                var canUsePrevious = _previous._states.Length > previousTableEntryIndex;
                previousEntry = canUsePrevious ? _previous._states[previousTableEntryIndex] : default;
                return canUsePrevious;
            }
 
            private void RecordStepInfoForLastEntry(TimeSpan elapsedTime, ImmutableArray<(IncrementalGeneratorRunStep InputStep, int OutputIndex)> stepInputs, EntryState overallInputState)
            {
                Debug.Assert(stepInputs.IsDefault == !TrackIncrementalSteps);
                if (TrackIncrementalSteps)
                {
                    // We should have already recorded step information for all steps before the most recently recorded step.
                    Debug.Assert(_steps.Count + 1 == _states.Count);
 
                    TableEntry outputInfo = _states[^1];
 
                    var stepOutputBuilder = ArrayBuilder<(object, IncrementalStepRunReason)>.GetInstance(outputInfo.Count);
 
                    for (int i = 0; i < outputInfo.Count; i++)
                    {
                        stepOutputBuilder.Add((outputInfo.GetItem(i)!, AsStepState(overallInputState, outputInfo.GetState(i))));
                    }
 
                    _steps.Add(
                        new IncrementalGeneratorRunStep(
                            _name,
                            stepInputs,
                            stepOutputBuilder.ToImmutableAndFree(),
                            elapsedTime));
                }
            }
 
            public IReadOnlyList<IncrementalGeneratorRunStep> Steps => (IReadOnlyList<IncrementalGeneratorRunStep>?)_steps ?? ImmutableArray<IncrementalGeneratorRunStep>.Empty;
 
            private static IncrementalStepRunReason AsStepState(EntryState inputState, EntryState outputState)
            {
                return (inputState, outputState) switch
                {
                    (EntryState.Added, EntryState.Added) => IncrementalStepRunReason.New,
                    (EntryState.Modified, EntryState.Modified) => IncrementalStepRunReason.Modified,
                    (EntryState.Modified, EntryState.Cached) => IncrementalStepRunReason.Unchanged,
                    (EntryState.Cached, EntryState.Cached) => IncrementalStepRunReason.Cached,
                    (EntryState.Removed, EntryState.Removed) => IncrementalStepRunReason.Removed,
                    (EntryState.Modified, EntryState.Removed) => IncrementalStepRunReason.Removed,
                    (EntryState.Modified, EntryState.Added) => IncrementalStepRunReason.New,
                    _ => throw ExceptionUtilities.UnexpectedValue((inputState, outputState))
                };
            }
 
            public NodeStateTable<T> ToImmutableAndFree()
            {
                Debug.Assert(!TrackIncrementalSteps || _states.Count == _steps.Count);
 
                if (_states.Count == 0)
                {
                    _states.Free();
                    return NodeStateTable<T>.Empty;
                }
 
#if DEBUG
                // If the caller requested a specific capacity, then we should have added either that amount, or some
                // amount less than that.  It's possible to have added less as a Where clause will mean some amount of
                // states are filtered out.
                Debug.Assert(_requestedTableCapacity == null || _states.Count <= _requestedTableCapacity);
#endif
 
                // if we added the exact same entries as before, then we can directly embed previous' entry array,
                // avoiding a costly allocation of the same data.
                ImmutableArray<TableEntry> finalStates;
                if (_states.Count == _previous.Count && _states.SequenceEqual(_previous._states, (e1, e2) => e1.Matches(e2, _equalityComparer)))
                {
                    finalStates = _previous._states;
                    _states.Free();
                }
                else
                {
                    // Important to use ToImmutableAndFree so that we will MoveToImmutable when the requested capacity
                    // equals the count.
                    finalStates = _states.ToImmutableAndFree();
                }
 
                return new NodeStateTable<T>(
                    finalStates,
                    TrackIncrementalSteps ? _steps.ToImmutableAndFree() : default,
                    hasTrackedSteps: TrackIncrementalSteps,
                    isCached: finalStates.All(static s => s.IsCached) && _previous.GetTotalEntryItemCount() == finalStates.Sum(static s => s.Count));
            }
 
            private (T chosen, EntryState state, bool chosePrevious) GetModifiedItemAndState(T previous, T replacement)
            {
                // when comparing an item to check if its modified we explicitly cache the *previous* item in the case where its 
                // considered to be equal. This ensures that subsequent comparisons are stable across future generation passes.
                return _equalityComparer.Equals(previous, replacement)
                    ? (previous, EntryState.Cached, chosePrevious: true)
                    : (replacement, EntryState.Modified, chosePrevious: false);
            }
        }
 
        internal readonly struct TableEntry
        {
            private static readonly ImmutableArray<EntryState> s_allAddedEntries = ImmutableArray.Create(EntryState.Added);
            private static readonly ImmutableArray<EntryState> s_allCachedEntries = ImmutableArray.Create(EntryState.Cached);
            private static readonly ImmutableArray<EntryState> s_allModifiedEntries = ImmutableArray.Create(EntryState.Modified);
 
            /// <summary>
            /// All items removed as part of a transformation from non-empty input.
            /// </summary>
            private static readonly ImmutableArray<EntryState> s_allRemovedEntries = ImmutableArray.Create(EntryState.Removed);
 
            /// <summary>
            /// All items removed because the input has been removed.
            /// </summary>
            private static readonly ImmutableArray<EntryState> s_allRemovedDueToInputRemoval = ImmutableArray.Create(EntryState.Removed);
 
            private readonly OneOrMany<T> _items;
            private readonly bool _anyRemoved;
 
            /// <summary>
            /// Represents the corresponding state of each item in <see cref="_items"/>, or contains a single state when
            /// <see cref="_items"/> is populated or when every state of <see cref="_items"/> has the same value.
            /// </summary>
            private readonly ImmutableArray<EntryState> _states;
 
            public TableEntry(OneOrMany<T> items, EntryState state)
                : this(items, GetSingleArray(state), anyRemoved: state == EntryState.Removed) { }
 
            private TableEntry(OneOrMany<T> items, ImmutableArray<EntryState> states, bool anyRemoved)
            {
                Debug.Assert(!states.IsDefault);
                Debug.Assert(states.Length == 1 || states.Distinct().Length > 1);
 
                _items = items;
                _states = states;
                _anyRemoved = anyRemoved;
            }
 
            public bool Matches(TableEntry entry, IEqualityComparer<T> equalityComparer)
            {
                if (!_states.SequenceEqual(entry._states))
                    return false;
 
                if (this.Count != entry.Count)
                    return false;
 
                for (int i = 0, n = this.Count; i < n; i++)
                {
                    if (!equalityComparer.Equals(this.GetItem(i), entry.GetItem(i)))
                        return false;
                }
 
                return true;
            }
 
            public bool IsCached => this._states == s_allCachedEntries || this._states.All(s => s == EntryState.Cached);
 
            public bool IsRemovedDueToInputRemoval => this._states == s_allRemovedDueToInputRemoval;
 
            public int Count => _items.Count;
 
            public T GetItem(int index) => _items[index];
 
            public EntryState GetState(int index) => _states.Length == 1 ? _states[0] : _states[index];
 
            public OneOrMany<T> Items => _items;
 
            public TableEntry AsCached()
            {
                if (!_anyRemoved)
                {
                    return new TableEntry(_items, s_allCachedEntries, anyRemoved: false);
                }
 
                var itemBuilder = ArrayBuilder<T>.GetInstance();
                for (int i = 0; i < this.Count; i++)
                {
                    if (this.GetState(i) != EntryState.Removed)
                    {
                        itemBuilder.Add(this.GetItem(i));
                    }
                }
 
                Debug.Assert(itemBuilder.Count < this.Count);
                return new TableEntry(OneOrMany.Create(itemBuilder.ToImmutableArray()), s_allCachedEntries, anyRemoved: false);
            }
 
            public TableEntry AsRemovedDueToInputRemoval() => new(_items, s_allRemovedDueToInputRemoval, anyRemoved: true);
 
            private static ImmutableArray<EntryState> GetSingleArray(EntryState state) => state switch
            {
                EntryState.Added => s_allAddedEntries,
                EntryState.Cached => s_allCachedEntries,
                EntryState.Modified => s_allModifiedEntries,
                EntryState.Removed => s_allRemovedEntries,
                _ => throw ExceptionUtilities.Unreachable()
            };
 
            public Enumerator GetEnumerator()
                => new(this);
 
            public struct Enumerator
            {
                private readonly TableEntry _entry;
                private int _index = -1;
 
                public Enumerator(TableEntry tableEntry)
                {
                    _entry = tableEntry;
                }
 
                public bool MoveNext()
                {
                    _index++;
                    return _index < _entry.Count;
                }
 
                public T Current => _entry.GetItem(_index);
            }
 
#if DEBUG
            public override string ToString()
            {
                if (this.Count == 1)
                {
                    return $"{GetItem(0)}: {GetState(0)}";
                }
                else
                {
                    var sb = PooledStringBuilder.GetInstance();
                    sb.Builder.Append('{');
                    for (int i = 0; i < Count; i++)
                    {
                        if (i > 0)
                        {
                            sb.Builder.Append(',');
                        }
 
                        sb.Builder.Append(" (");
                        sb.Builder.Append(GetItem(i));
                        sb.Builder.Append(':');
                        sb.Builder.Append(GetState(i));
                        sb.Builder.Append(')');
                    }
 
                    sb.Builder.Append(" }");
                    return sb.ToStringAndFree();
                }
            }
#endif
 
            public sealed class Builder
            {
                private readonly ArrayBuilder<T> _items;
 
                private ArrayBuilder<EntryState>? _states;
                private EntryState? _currentState;
                private bool _anyRemoved;
 
                private readonly int _requestedCapacity;
 
                public Builder(int capacity)
                {
                    _items = ArrayBuilder<T>.GetInstance(capacity);
                    _requestedCapacity = capacity;
                }
 
                public void Add(T item, EntryState state)
                {
                    _items.Add(item);
                    _anyRemoved |= state == EntryState.Removed;
                    if (!_currentState.HasValue)
                    {
                        _currentState = state;
                    }
                    else if (_states is not null)
                    {
                        _states.Add(state);
                    }
                    else if (_currentState != state)
                    {
                        // Create a builder with the right capacity (so we don't waste scratch space). Copy all the same
                        // prior values all the way up to the last item we're about to add.
                        _states = ArrayBuilder<EntryState>.GetInstance(_requestedCapacity);
                        for (int i = 0, n = _items.Count - 1; i < n; i++)
                            _states.Add(_currentState.Value);
 
                        // then finally add the new value at the end.
                        _states.Add(state);
                    }
                }
 
                public TableEntry ToImmutableAndFree()
                {
                    Debug.Assert(_currentState.HasValue, "Created a builder with no values?");
                    Debug.Assert(_items.Count >= 1, "Created a builder with no values?");
 
                    Debug.Assert(_items.Count == _requestedCapacity);
                    Debug.Assert(_states == null || _states.Count == _requestedCapacity);
 
                    return new TableEntry(_items.ToOneOrManyAndFree(), _states?.ToImmutableAndFree() ?? GetSingleArray(_currentState.Value), anyRemoved: _anyRemoved);
                }
            }
        }
    }
}