mirror of
https://github.com/chylex/.NET-Community-Toolkit.git
synced 2024-11-23 22:42:47 +01:00
160 lines
5.3 KiB
C#
160 lines
5.3 KiB
C#
// 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.ComponentModel;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Runtime.InteropServices;
|
|
|
|
namespace CommunityToolkit.HighPerformance.Enumerables;
|
|
|
|
/// <summary>
|
|
/// A <see langword="ref"/> <see langword="struct"/> that enumerates the items in a given <see cref="Span{T}"/> instance.
|
|
/// </summary>
|
|
/// <typeparam name="T">The type of items to enumerate.</typeparam>
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
public ref struct SpanEnumerable<T>
|
|
{
|
|
/// <summary>
|
|
/// The source <see cref="Span{T}"/> instance.
|
|
/// </summary>
|
|
private readonly Span<T> span;
|
|
|
|
/// <summary>
|
|
/// The current index within <see cref="span"/>.
|
|
/// </summary>
|
|
private int index;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="SpanEnumerable{T}"/> struct.
|
|
/// </summary>
|
|
/// <param name="span">The source <see cref="Span{T}"/> instance.</param>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public SpanEnumerable(Span<T> span)
|
|
{
|
|
this.span = span;
|
|
this.index = -1;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Implements the duck-typed <see cref="IEnumerable{T}.GetEnumerator"/> method.
|
|
/// </summary>
|
|
/// <returns>An <see cref="SpanEnumerable{T}"/> instance targeting the current <see cref="Span{T}"/> value.</returns>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public readonly SpanEnumerable<T> GetEnumerator() => this;
|
|
|
|
/// <summary>
|
|
/// Implements the duck-typed <see cref="System.Collections.IEnumerator.MoveNext"/> method.
|
|
/// </summary>
|
|
/// <returns><see langword="true"/> whether a new element is available, <see langword="false"/> otherwise</returns>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public bool MoveNext()
|
|
{
|
|
return ++this.index < this.span.Length;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the duck-typed <see cref="IEnumerator{T}.Current"/> property.
|
|
/// </summary>
|
|
public readonly Item Current
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
ref T r0 = ref MemoryMarshal.GetReference(this.span);
|
|
ref T ri = ref Unsafe.Add(ref r0, (nint)(uint)this.index);
|
|
|
|
// On .NET Standard 2.1 and .NET Core (or on any target that offers runtime
|
|
// support for the Span<T> types), we can save 4 bytes by piggybacking the
|
|
// current index in the length of the wrapped span. We're going to use the
|
|
// first item as the target reference, and the length as a host for the
|
|
// current original offset. This is not possible on eg. .NET Standard 2.0,
|
|
// as we lack the API to create Span<T>-s from arbitrary references.
|
|
return new(ref ri, this.index);
|
|
#else
|
|
return new(this.span, this.index);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// An item from a source <see cref="Span{T}"/> instance.
|
|
/// </summary>
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
public readonly ref struct Item
|
|
{
|
|
/// <summary>
|
|
/// The source <see cref="Span{T}"/> instance.
|
|
/// </summary>
|
|
private readonly Span<T> span;
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="Item"/> struct.
|
|
/// </summary>
|
|
/// <param name="value">A reference to the target value.</param>
|
|
/// <param name="index">The index of the target value.</param>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public Item(ref T value, int index)
|
|
{
|
|
this.span = MemoryMarshal.CreateSpan(ref value, index);
|
|
}
|
|
#else
|
|
/// <summary>
|
|
/// The current index within <see cref="span"/>.
|
|
/// </summary>
|
|
private readonly int index;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="Item"/> struct.
|
|
/// </summary>
|
|
/// <param name="span">The source <see cref="Span{T}"/> instance.</param>
|
|
/// <param name="index">The current index within <paramref name="span"/>.</param>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public Item(Span<T> span, int index)
|
|
{
|
|
this.span = span;
|
|
this.index = index;
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Gets the reference to the current value.
|
|
/// </summary>
|
|
public ref T Value
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
return ref MemoryMarshal.GetReference(this.span);
|
|
#else
|
|
ref T r0 = ref MemoryMarshal.GetReference(this.span);
|
|
ref T ri = ref Unsafe.Add(ref r0, (nint)(uint)this.index);
|
|
|
|
return ref ri;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the current index.
|
|
/// </summary>
|
|
public int Index
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
return this.span.Length;
|
|
#else
|
|
return this.index;
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|