File: Syntax\SyntaxTokenList.Reversed.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;
using System.Collections.Generic;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Runtime.InteropServices;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis
{
    public partial struct SyntaxTokenList
    {
        /// <summary>
        /// Reversed enumerable.
        /// </summary>
        public readonly struct Reversed : IEnumerable<SyntaxToken>, IEquatable<Reversed>
        {
            private readonly SyntaxTokenList _list;
 
            public Reversed(SyntaxTokenList list)
            {
                _list = list;
            }
 
            public Enumerator GetEnumerator()
            {
                return new Enumerator(in _list);
            }
 
            IEnumerator<SyntaxToken> IEnumerable<SyntaxToken>.GetEnumerator()
            {
                if (_list.Count == 0)
                {
                    return SpecializedCollections.EmptyEnumerator<SyntaxToken>();
                }
 
                return new EnumeratorImpl(in _list);
            }
 
            IEnumerator IEnumerable.GetEnumerator()
            {
                if (_list.Count == 0)
                {
                    return SpecializedCollections.EmptyEnumerator<SyntaxToken>();
                }
 
                return new EnumeratorImpl(in _list);
            }
 
            public override bool Equals(object? obj)
            {
                return obj is Reversed r && Equals(r);
            }
 
            public bool Equals(Reversed other)
            {
                return _list.Equals(other._list);
            }
 
            public override int GetHashCode()
            {
                return _list.GetHashCode();
            }
 
            [SuppressMessage("Performance", "CA1067", Justification = "Equality not actually implemented")]
            [StructLayout(LayoutKind.Auto)]
            public struct Enumerator
            {
                private readonly SyntaxNode? _parent;
                private readonly GreenNode? _singleNodeOrList;
                private readonly int _baseIndex;
                private readonly int _count;
 
                private int _index;
                private GreenNode? _current;
                private int _position;
 
                internal Enumerator(in SyntaxTokenList list)
                    : this()
                {
                    if (list.Any())
                    {
                        _parent = list._parent;
                        _singleNodeOrList = list.Node;
                        _baseIndex = list._index;
                        _count = list.Count;
 
                        _index = _count;
                        _current = null;
 
                        var last = list.Last();
                        _position = last.Position + last.FullWidth;
                    }
                }
 
                public bool MoveNext()
                {
                    if (_count == 0 || _index <= 0)
                    {
                        _current = null;
                        return false;
                    }
 
                    _index--;
 
                    Debug.Assert(_singleNodeOrList is object);
                    _current = GetGreenNodeAt(_singleNodeOrList, _index);
                    Debug.Assert(_current is object);
                    _position -= _current.FullWidth;
 
                    return true;
                }
 
                public SyntaxToken Current
                {
                    get
                    {
                        if (_current == null)
                        {
                            throw new InvalidOperationException();
                        }
 
                        return new SyntaxToken(_parent, _current, _position, _baseIndex + _index);
                    }
                }
 
                public override bool Equals(object? obj)
                {
                    throw new NotSupportedException();
                }
 
                public override int GetHashCode()
                {
                    throw new NotSupportedException();
                }
            }
 
            private class EnumeratorImpl : IEnumerator<SyntaxToken>
            {
                private Enumerator _enumerator;
 
                // SyntaxTriviaList is a relatively big struct so is passed as ref
                internal EnumeratorImpl(in SyntaxTokenList list)
                {
                    _enumerator = new Enumerator(in list);
                }
 
                public SyntaxToken Current => _enumerator.Current;
 
                object IEnumerator.Current => _enumerator.Current;
 
                public bool MoveNext()
                {
                    return _enumerator.MoveNext();
                }
 
                public void Reset()
                {
                    throw new NotSupportedException();
                }
 
                public void Dispose()
                {
                }
            }
        }
    }
}