File: System\Text\Json\BitStack.cs
Web Access
Project: src\src\libraries\System.Text.Json\src\System.Text.Json.csproj (System.Text.Json)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.Diagnostics;
using System.Runtime.CompilerServices;
 
namespace System.Text.Json
{
    internal struct BitStack
    {
        // We are using a ulong to represent our nested state, so we can only
        // go 64 levels deep without having to allocate.
        private const int AllocationFreeMaxDepth = sizeof(ulong) * 8;
 
        private const int DefaultInitialArraySize = 2;
 
        // The backing array for the stack used when the depth exceeds AllocationFreeMaxDepth.
        private int[]? _array;
 
        // This ulong container represents a tiny stack to track the state during nested transitions.
        // The first bit represents the state of the current depth (1 == object, 0 == array).
        // Each subsequent bit is the parent / containing type (object or array). Since this
        // reader does a linear scan, we only need to keep a single path as we go through the data.
        // This is primarily used as an optimization to avoid having to allocate an object for
        // depths up to 64 (which is the default max depth).
        private ulong _allocationFreeContainer;
 
        private int _currentDepth;
 
        /// <summary>
        /// Gets the number of elements in the stack.
        /// </summary>
        public readonly int CurrentDepth => _currentDepth;
 
        /// <summary>
        /// Pushes <see langword="true"/> onto the stack.
        /// </summary>
        [MethodImpl(MethodImplOptions.AggressiveInlining)]
        public void PushTrue()
        {
            if (_currentDepth < AllocationFreeMaxDepth)
            {
                _allocationFreeContainer = (_allocationFreeContainer << 1) | 1;
            }
            else
            {
                PushToArray(true);
            }
            _currentDepth++;
        }
 
        /// <summary>
        /// Pushes <see langword="false"/> onto the stack.
        /// </summary>
        [MethodImpl(MethodImplOptions.AggressiveInlining)]
        public void PushFalse()
        {
            if (_currentDepth < AllocationFreeMaxDepth)
            {
                _allocationFreeContainer <<= 1;
            }
            else
            {
                PushToArray(false);
            }
            _currentDepth++;
        }
 
        /// <summary>
        /// Pushes a bit onto the stack. Allocate the bit array lazily only when it is absolutely necessary.
        /// </summary>
        /// <param name="value">The bit to push onto the stack.</param>
        [MethodImpl(MethodImplOptions.NoInlining)]
        private void PushToArray(bool value)
        {
            _array ??= new int[DefaultInitialArraySize];
 
            int index = _currentDepth - AllocationFreeMaxDepth;
 
            Debug.Assert(index >= 0, $"Set - Negative - index: {index}, arrayLength: {_array.Length}");
 
            // Maximum possible array length if bitLength was int.MaxValue (i.e. 67_108_864)
            Debug.Assert(_array.Length <= int.MaxValue / 32 + 1, $"index: {index}, arrayLength: {_array.Length}");
 
            int elementIndex = Div32Rem(index, out int extraBits);
 
            // Grow the array when setting a bit if it isn't big enough
            // This way the caller doesn't have to check.
            if (elementIndex >= _array.Length)
            {
                // This multiplication can overflow, so cast to uint first.
                Debug.Assert(index >= 0 && index > (int)((uint)_array.Length * 32 - 1), $"Only grow when necessary - index: {index}, arrayLength: {_array.Length}");
                DoubleArray(elementIndex);
            }
 
            Debug.Assert(elementIndex < _array.Length, $"Set - index: {index}, elementIndex: {elementIndex}, arrayLength: {_array.Length}, extraBits: {extraBits}");
 
            int newValue = _array[elementIndex];
            if (value)
            {
                newValue |= 1 << extraBits;
            }
            else
            {
                newValue &= ~(1 << extraBits);
            }
            _array[elementIndex] = newValue;
        }
 
        /// <summary>
        /// Pops the bit at the top of the stack and returns its value.
        /// </summary>
        /// <returns>The bit that was popped.</returns>
        [MethodImpl(MethodImplOptions.AggressiveInlining)]
        public bool Pop()
        {
            _currentDepth--;
            bool inObject;
            if (_currentDepth < AllocationFreeMaxDepth)
            {
                _allocationFreeContainer >>= 1;
                inObject = (_allocationFreeContainer & 1) != 0;
            }
            else if (_currentDepth == AllocationFreeMaxDepth)
            {
                inObject = (_allocationFreeContainer & 1) != 0;
            }
            else
            {
                // Decrementing depth above effectively pops the last element in the array-backed case.
                inObject = PeekInArray();
            }
            return inObject;
        }
 
        /// <summary>
        /// If the stack has a backing array allocated, this method will find the topmost bit in the array and return its value.
        /// This should only be called if the depth is greater than AllocationFreeMaxDepth and an array has been allocated.
        /// </summary>
        [MethodImpl(MethodImplOptions.NoInlining)]
        private readonly bool PeekInArray()
        {
            int index = _currentDepth - AllocationFreeMaxDepth - 1;
            Debug.Assert(_array != null);
            Debug.Assert(index >= 0, $"Get - Negative - index: {index}, arrayLength: {_array.Length}");
 
            int elementIndex = Div32Rem(index, out int extraBits);
 
            Debug.Assert(elementIndex < _array.Length, $"Get - index: {index}, elementIndex: {elementIndex}, arrayLength: {_array.Length}, extraBits: {extraBits}");
 
            return (_array[elementIndex] & (1 << extraBits)) != 0;
        }
 
        /// <summary>
        /// Peeks at the bit at the top of the stack.
        /// </summary>
        /// <returns>The bit at the top of the stack.</returns>
        public readonly bool Peek()
            // If the stack is small enough, we can use the allocation-free container, otherwise check the allocated array.
            => _currentDepth <= AllocationFreeMaxDepth ? (_allocationFreeContainer & 1) != 0 : PeekInArray();
 
        private void DoubleArray(int minSize)
        {
            Debug.Assert(_array != null);
            Debug.Assert(_array.Length < int.MaxValue / 2, $"Array too large - arrayLength: {_array.Length}");
            Debug.Assert(minSize >= 0 && minSize >= _array.Length);
 
            int nextDouble = Math.Max(minSize + 1, _array.Length * 2);
            Debug.Assert(nextDouble > minSize);
 
            Array.Resize(ref _array, nextDouble);
        }
 
        /// <summary>
        /// Optimization to push <see langword="true"/> as the first bit when the stack is empty.
        /// </summary>
        public void SetFirstBit()
        {
            Debug.Assert(_currentDepth == 0, "Only call SetFirstBit when depth is 0");
            _currentDepth++;
            _allocationFreeContainer = 1;
        }
 
        /// <summary>
        /// Optimization to push <see langword="false"/> as the first bit when the stack is empty.
        /// </summary>
        public void ResetFirstBit()
        {
            Debug.Assert(_currentDepth == 0, "Only call ResetFirstBit when depth is 0");
            _currentDepth++;
            _allocationFreeContainer = 0;
        }
 
        [MethodImpl(MethodImplOptions.AggressiveInlining)]
        private static int Div32Rem(int number, out int remainder)
        {
            uint quotient = (uint)number / 32;
            remainder = number & (32 - 1);   // equivalent to number % 32, since 32 is a power of 2
            return (int)quotient;
        }
    }
}