mirror of
https://github.com/chylex/.NET-Community-Toolkit.git
synced 2024-11-24 07:42:45 +01:00
1024 lines
37 KiB
C#
1024 lines
37 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.ComponentModel;
|
|
using System.Diagnostics;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Runtime.InteropServices;
|
|
#if !NETSTANDARD2_1_OR_GREATER
|
|
using CommunityToolkit.HighPerformance.Helpers;
|
|
#endif
|
|
using CommunityToolkit.HighPerformance.Memory.Internals;
|
|
using CommunityToolkit.HighPerformance.Memory.Views;
|
|
#if !NETSTANDARD2_1_OR_GREATER
|
|
using RuntimeHelpers = CommunityToolkit.HighPerformance.Helpers.Internals.RuntimeHelpers;
|
|
#endif
|
|
|
|
#pragma warning disable CS0809, CA1065
|
|
|
|
namespace CommunityToolkit.HighPerformance;
|
|
|
|
/// <summary>
|
|
/// A readonly version of <see cref="Span2D{T}"/>.
|
|
/// </summary>
|
|
/// <typeparam name="T">The type of items in the current <see cref="ReadOnlySpan2D{T}"/> instance.</typeparam>
|
|
[DebuggerTypeProxy(typeof(MemoryDebugView2D<>))]
|
|
[DebuggerDisplay("{ToString(),raw}")]
|
|
public readonly ref partial struct ReadOnlySpan2D<T>
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
/// <summary>
|
|
/// The <see cref="ReadOnlySpan{T}"/> instance pointing to the first item in the target memory area.
|
|
/// </summary>
|
|
private readonly ReadOnlySpan<T> span;
|
|
#else
|
|
/// <summary>
|
|
/// The target <see cref="object"/> instance, if present.
|
|
/// </summary>
|
|
private readonly object? instance;
|
|
|
|
/// <summary>
|
|
/// The initial offset within <see cref="instance"/>.
|
|
/// </summary>
|
|
private readonly IntPtr offset;
|
|
|
|
/// <summary>
|
|
/// The height of the specified 2D region.
|
|
/// </summary>
|
|
private readonly int height;
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// The width of the specified 2D region.
|
|
/// </summary>
|
|
private readonly int width;
|
|
|
|
/// <summary>
|
|
/// The stride of the specified 2D region.
|
|
/// </summary>
|
|
private readonly int stride;
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="value">The reference to the first <typeparamref name="T"/> item to map.</param>
|
|
/// <param name="height">The height of the 2D memory area to map.</param>
|
|
/// <param name="width">The width of the 2D memory area to map.</param>
|
|
/// <param name="pitch">The pitch of the 2D memory area to map (the distance between each row).</param>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
internal ReadOnlySpan2D(in T value, int height, int width, int pitch)
|
|
{
|
|
this.span = MemoryMarshal.CreateReadOnlySpan(ref Unsafe.AsRef(value), height);
|
|
this.width = width;
|
|
this.stride = width + pitch;
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="pointer">The pointer to the first <typeparamref name="T"/> item to map.</param>
|
|
/// <param name="height">The height of the 2D memory area to map.</param>
|
|
/// <param name="width">The width of the 2D memory area to map.</param>
|
|
/// <param name="pitch">The pitch of the 2D memory area to map (the distance between each row).</param>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when one of the parameters are negative.</exception>
|
|
public unsafe ReadOnlySpan2D(void* pointer, int height, int width, int pitch)
|
|
{
|
|
if (RuntimeHelpers.IsReferenceOrContainsReferences<T>())
|
|
{
|
|
ThrowHelper.ThrowArgumentExceptionForManagedType();
|
|
}
|
|
|
|
if (width < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForWidth();
|
|
}
|
|
|
|
if (height < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForHeight();
|
|
}
|
|
|
|
if (pitch < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForPitch();
|
|
}
|
|
|
|
OverflowHelper.EnsureIsInNativeIntRange(height, width, pitch);
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
this.span = new ReadOnlySpan<T>(pointer, height);
|
|
#else
|
|
this.instance = null;
|
|
this.offset = (IntPtr)pointer;
|
|
this.height = height;
|
|
#endif
|
|
this.width = width;
|
|
this.stride = width + pitch;
|
|
}
|
|
|
|
#if !NETSTANDARD2_1_OR_GREATER
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="instance">The target <see cref="object"/> instance.</param>
|
|
/// <param name="offset">The initial offset within the target instance.</param>
|
|
/// <param name="height">The height of the 2D memory area to map.</param>
|
|
/// <param name="width">The width of the 2D memory area to map.</param>
|
|
/// <param name="pitch">The pitch of the 2D memory area to map.</param>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
internal ReadOnlySpan2D(object? instance, IntPtr offset, int height, int width, int pitch)
|
|
{
|
|
this.instance = instance;
|
|
this.offset = offset;
|
|
this.height = height;
|
|
this.width = width;
|
|
this.stride = width + pitch;
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct.
|
|
/// </summary>
|
|
/// <param name="array">The target array to wrap.</param>
|
|
/// <param name="height">The height of the resulting 2D area.</param>
|
|
/// <param name="width">The width of each row in the resulting 2D area.</param>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown when either <paramref name="height"/> or <paramref name="width"/> are invalid.
|
|
/// </exception>
|
|
/// <remarks>The total area must match the length of <paramref name="array"/>.</remarks>
|
|
public ReadOnlySpan2D(T[] array, int height, int width)
|
|
: this(array, 0, height, width, 0)
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct.
|
|
/// </summary>
|
|
/// <param name="array">The target array to wrap.</param>
|
|
/// <param name="offset">The initial offset within <paramref name="array"/>.</param>
|
|
/// <param name="height">The height of the resulting 2D area.</param>
|
|
/// <param name="width">The width of each row in the resulting 2D area.</param>
|
|
/// <param name="pitch">The pitch in the resulting 2D area.</param>
|
|
/// <exception cref="ArgumentOutOfRangeException">
|
|
/// Thrown when one of the input parameters is out of range.
|
|
/// </exception>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown when the requested area is outside of bounds for <paramref name="array"/>.
|
|
/// </exception>
|
|
public ReadOnlySpan2D(T[] array, int offset, int height, int width, int pitch)
|
|
{
|
|
if ((uint)offset > (uint)array.Length)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForOffset();
|
|
}
|
|
|
|
if (height < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForHeight();
|
|
}
|
|
|
|
if (width < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForWidth();
|
|
}
|
|
|
|
if (pitch < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForPitch();
|
|
}
|
|
|
|
if (width == 0 || height == 0)
|
|
{
|
|
this = default;
|
|
|
|
return;
|
|
}
|
|
|
|
int area = OverflowHelper.ComputeInt32Area(height, width, pitch);
|
|
int remaining = array.Length - offset;
|
|
|
|
if (area > remaining)
|
|
{
|
|
ThrowHelper.ThrowArgumentException();
|
|
}
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
this.span = MemoryMarshal.CreateReadOnlySpan(ref array.DangerousGetReferenceAt(offset), height);
|
|
#else
|
|
this.instance = array;
|
|
this.offset = ObjectMarshal.DangerousGetObjectDataByteOffset(array, ref array.DangerousGetReferenceAt(offset));
|
|
this.height = height;
|
|
#endif
|
|
this.width = width;
|
|
this.stride = width + pitch;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct wrapping a 2D array.
|
|
/// </summary>
|
|
/// <param name="array">The given 2D array to wrap.</param>
|
|
public ReadOnlySpan2D(T[,]? array)
|
|
{
|
|
if (array is null)
|
|
{
|
|
this = default;
|
|
|
|
return;
|
|
}
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
this.span = MemoryMarshal.CreateReadOnlySpan(ref array.DangerousGetReference(), array.GetLength(0));
|
|
#else
|
|
this.instance = array;
|
|
this.offset = ObjectMarshal.DangerousGetObjectDataByteOffset(array, ref array.DangerousGetReferenceAt(0, 0));
|
|
this.height = array.GetLength(0);
|
|
#endif
|
|
this.width = this.stride = array.GetLength(1);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct wrapping a 2D array.
|
|
/// </summary>
|
|
/// <param name="array">The given 2D array to wrap.</param>
|
|
/// <param name="row">The target row to map within <paramref name="array"/>.</param>
|
|
/// <param name="column">The target column to map within <paramref name="array"/>.</param>
|
|
/// <param name="height">The height to map within <paramref name="array"/>.</param>
|
|
/// <param name="width">The width to map within <paramref name="array"/>.</param>
|
|
/// <exception cref="ArgumentOutOfRangeException">
|
|
/// Thrown when either <paramref name="height"/>, <paramref name="width"/> or <paramref name="height"/>
|
|
/// are negative or not within the bounds that are valid for <paramref name="array"/>.
|
|
/// </exception>
|
|
public ReadOnlySpan2D(T[,]? array, int row, int column, int height, int width)
|
|
{
|
|
if (array is null)
|
|
{
|
|
if (row != 0 || column != 0 || height != 0 || width != 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentException();
|
|
}
|
|
|
|
this = default;
|
|
|
|
return;
|
|
}
|
|
|
|
int rows = array.GetLength(0);
|
|
int columns = array.GetLength(1);
|
|
|
|
if ((uint)row >= (uint)rows)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForRow();
|
|
}
|
|
|
|
if ((uint)column >= (uint)columns)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForColumn();
|
|
}
|
|
|
|
if ((uint)height > (uint)(rows - row))
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForHeight();
|
|
}
|
|
|
|
if ((uint)width > (uint)(columns - column))
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForWidth();
|
|
}
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
this.span = MemoryMarshal.CreateReadOnlySpan(ref array.DangerousGetReferenceAt(row, column), height);
|
|
#else
|
|
this.instance = array;
|
|
this.offset = ObjectMarshal.DangerousGetObjectDataByteOffset(array, ref array.DangerousGetReferenceAt(row, column));
|
|
this.height = height;
|
|
#endif
|
|
this.width = width;
|
|
this.stride = columns;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct wrapping a layer in a 3D array.
|
|
/// </summary>
|
|
/// <param name="array">The given 3D array to wrap.</param>
|
|
/// <param name="depth">The target layer to map within <paramref name="array"/>.</param>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when a parameter is invalid.</exception>
|
|
public ReadOnlySpan2D(T[,,] array, int depth)
|
|
{
|
|
if ((uint)depth >= (uint)array.GetLength(0))
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForDepth();
|
|
}
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
this.span = MemoryMarshal.CreateReadOnlySpan(ref array.DangerousGetReferenceAt(depth, 0, 0), array.GetLength(1));
|
|
#else
|
|
this.instance = array;
|
|
this.offset = ObjectMarshal.DangerousGetObjectDataByteOffset(array, ref array.DangerousGetReferenceAt(depth, 0, 0));
|
|
this.height = array.GetLength(1);
|
|
#endif
|
|
this.width = this.stride = array.GetLength(2);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct wrapping a layer in a 3D array.
|
|
/// </summary>
|
|
/// <param name="array">The given 3D array to wrap.</param>
|
|
/// <param name="depth">The target layer to map within <paramref name="array"/>.</param>
|
|
/// <param name="row">The target row to map within <paramref name="array"/>.</param>
|
|
/// <param name="column">The target column to map within <paramref name="array"/>.</param>
|
|
/// <param name="height">The height to map within <paramref name="array"/>.</param>
|
|
/// <param name="width">The width to map within <paramref name="array"/>.</param>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when a parameter is invalid.</exception>
|
|
public ReadOnlySpan2D(T[,,] array, int depth, int row, int column, int height, int width)
|
|
{
|
|
if ((uint)depth >= (uint)array.GetLength(0))
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForDepth();
|
|
}
|
|
|
|
int rows = array.GetLength(1);
|
|
int columns = array.GetLength(2);
|
|
|
|
if ((uint)row >= (uint)rows)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForRow();
|
|
}
|
|
|
|
if ((uint)column >= (uint)columns)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForColumn();
|
|
}
|
|
|
|
if ((uint)height > (uint)(rows - row))
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForHeight();
|
|
}
|
|
|
|
if ((uint)width > (uint)(columns - column))
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForWidth();
|
|
}
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
this.span = MemoryMarshal.CreateReadOnlySpan(ref array.DangerousGetReferenceAt(depth, row, column), height);
|
|
#else
|
|
this.instance = array;
|
|
this.offset = ObjectMarshal.DangerousGetObjectDataByteOffset(array, ref array.DangerousGetReferenceAt(depth, row, column));
|
|
this.height = height;
|
|
#endif
|
|
this.width = width;
|
|
this.stride = columns;
|
|
}
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct.
|
|
/// </summary>
|
|
/// <param name="span">The target <see cref="ReadOnlySpan{T}"/> to wrap.</param>
|
|
/// <param name="height">The height of the resulting 2D area.</param>
|
|
/// <param name="width">The width of each row in the resulting 2D area.</param>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown when either <paramref name="height"/> or <paramref name="width"/> are invalid.
|
|
/// </exception>
|
|
/// <remarks>The total area must match the length of <paramref name="span"/>.</remarks>
|
|
internal ReadOnlySpan2D(ReadOnlySpan<T> span, int height, int width)
|
|
: this(span, 0, height, width, 0)
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct.
|
|
/// </summary>
|
|
/// <param name="span">The target <see cref="ReadOnlySpan{T}"/> to wrap.</param>
|
|
/// <param name="offset">The initial offset within <paramref name="span"/>.</param>
|
|
/// <param name="height">The height of the resulting 2D area.</param>
|
|
/// <param name="width">The width of each row in the resulting 2D area.</param>
|
|
/// <param name="pitch">The pitch in the resulting 2D area.</param>
|
|
/// <exception cref="ArgumentOutOfRangeException">
|
|
/// Thrown when one of the input parameters is out of range.
|
|
/// </exception>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown when the requested area is outside of bounds for <paramref name="span"/>.
|
|
/// </exception>
|
|
internal ReadOnlySpan2D(ReadOnlySpan<T> span, int offset, int height, int width, int pitch)
|
|
{
|
|
if ((uint)offset > (uint)span.Length)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForOffset();
|
|
}
|
|
|
|
if (height < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForHeight();
|
|
}
|
|
|
|
if (width < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForWidth();
|
|
}
|
|
|
|
if (pitch < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForPitch();
|
|
}
|
|
|
|
if (width == 0 || height == 0)
|
|
{
|
|
this = default;
|
|
|
|
return;
|
|
}
|
|
|
|
int area = OverflowHelper.ComputeInt32Area(height, width, pitch);
|
|
int remaining = span.Length - offset;
|
|
|
|
if (area > remaining)
|
|
{
|
|
ThrowHelper.ThrowArgumentException();
|
|
}
|
|
|
|
this.span = MemoryMarshal.CreateSpan(ref span.DangerousGetReferenceAt(offset), height);
|
|
this.width = width;
|
|
this.stride = width + pitch;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a new instance of the <see cref="ReadOnlySpan2D{T}"/> struct with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="value">The reference to the first <typeparamref name="T"/> item to map.</param>
|
|
/// <param name="height">The height of the 2D memory area to map.</param>
|
|
/// <param name="width">The width of the 2D memory area to map.</param>
|
|
/// <param name="pitch">The pitch of the 2D memory area to map (the distance between each row).</param>
|
|
/// <returns>A <see cref="ReadOnlySpan2D{T}"/> instance with the specified parameters.</returns>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when one of the parameters are negative.</exception>
|
|
public static ReadOnlySpan2D<T> DangerousCreate(in T value, int height, int width, int pitch)
|
|
{
|
|
if (width < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForWidth();
|
|
}
|
|
|
|
if (height < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForHeight();
|
|
}
|
|
|
|
if (pitch < 0)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForPitch();
|
|
}
|
|
|
|
OverflowHelper.EnsureIsInNativeIntRange(height, width, pitch);
|
|
|
|
return new(in value, height, width, pitch);
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Gets an empty <see cref="ReadOnlySpan2D{T}"/> instance.
|
|
/// </summary>
|
|
public static ReadOnlySpan2D<T> Empty => default;
|
|
|
|
/// <summary>
|
|
/// Gets a value indicating whether the current <see cref="ReadOnlySpan2D{T}"/> instance is empty.
|
|
/// </summary>
|
|
public bool IsEmpty
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => Height == 0 || this.width == 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the length of the current <see cref="ReadOnlySpan2D{T}"/> instance.
|
|
/// </summary>
|
|
public nint Length
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => (nint)(uint)Height * (nint)(uint)this.width;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the height of the underlying 2D memory area.
|
|
/// </summary>
|
|
public int Height
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
return this.span.Length;
|
|
#else
|
|
return this.height;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the width of the underlying 2D memory area.
|
|
/// </summary>
|
|
public int Width
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => this.width;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the element at the specified zero-based indices.
|
|
/// </summary>
|
|
/// <param name="row">The target row to get the element from.</param>
|
|
/// <param name="column">The target column to get the element from.</param>
|
|
/// <returns>A reference to the element at the specified indices.</returns>
|
|
/// <exception cref="IndexOutOfRangeException">
|
|
/// Thrown when either <paramref name="row"/> or <paramref name="column"/> are invalid.
|
|
/// </exception>
|
|
public ref readonly T this[int row, int column]
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
if ((uint)row >= (uint)Height ||
|
|
(uint)column >= (uint)Width)
|
|
{
|
|
ThrowHelper.ThrowIndexOutOfRangeException();
|
|
}
|
|
|
|
return ref DangerousGetReferenceAt(row, column);
|
|
}
|
|
}
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
/// <summary>
|
|
/// Gets the element at the specified zero-based indices.
|
|
/// </summary>
|
|
/// <param name="row">The target row to get the element from.</param>
|
|
/// <param name="column">The target column to get the element from.</param>
|
|
/// <returns>A reference to the element at the specified indices.</returns>
|
|
/// <exception cref="IndexOutOfRangeException">
|
|
/// Thrown when either <paramref name="row"/> or <paramref name="column"/> are invalid.
|
|
/// </exception>
|
|
public ref readonly T this[Index row, Index column]
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => ref this[row.GetOffset(Height), column.GetOffset(this.width)];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Slices the current instance with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="rows">The target range of rows to select.</param>
|
|
/// <param name="columns">The target range of columns to select.</param>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown when either <paramref name="rows"/> or <paramref name="columns"/> are invalid.
|
|
/// </exception>
|
|
/// <returns>A new <see cref="ReadOnlySpan2D{T}"/> instance representing a slice of the current one.</returns>
|
|
public ReadOnlySpan2D<T> this[Range rows, Range columns]
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
(int row, int height) = rows.GetOffsetAndLength(Height);
|
|
(int column, int width) = columns.GetOffsetAndLength(this.width);
|
|
|
|
return Slice(row, column, height, width);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Copies the contents of this <see cref="ReadOnlySpan2D{T}"/> into a destination <see cref="Span{T}"/> instance.
|
|
/// </summary>
|
|
/// <param name="destination">The destination <see cref="Span{T}"/> instance.</param>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown when <paramref name="destination" /> is shorter than the source <see cref="ReadOnlySpan2D{T}"/> instance.
|
|
/// </exception>
|
|
public void CopyTo(Span<T> destination)
|
|
{
|
|
if (IsEmpty)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (TryGetSpan(out ReadOnlySpan<T> span))
|
|
{
|
|
span.CopyTo(destination);
|
|
}
|
|
else
|
|
{
|
|
if (Length > destination.Length)
|
|
{
|
|
ThrowHelper.ThrowArgumentExceptionForDestinationTooShort();
|
|
}
|
|
|
|
// Copy each row individually
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
for (int i = 0, j = 0; i < Height; i++, j += this.width)
|
|
{
|
|
GetRowSpan(i).CopyTo(destination.Slice(j));
|
|
}
|
|
#else
|
|
int height = Height;
|
|
nint width = (nint)(uint)this.width;
|
|
|
|
ref T destinationRef = ref MemoryMarshal.GetReference(destination);
|
|
|
|
for (int i = 0; i < height; i++)
|
|
{
|
|
ref T sourceStart = ref DangerousGetReferenceAt(i, 0);
|
|
ref T sourceEnd = ref Unsafe.Add(ref sourceStart, width);
|
|
|
|
while (Unsafe.IsAddressLessThan(ref sourceStart, ref sourceEnd))
|
|
{
|
|
destinationRef = sourceStart;
|
|
|
|
sourceStart = ref Unsafe.Add(ref sourceStart, 1);
|
|
destinationRef = ref Unsafe.Add(ref destinationRef, 1);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Copies the contents of this <see cref="ReadOnlySpan2D{T}"/> into a destination <see cref="Span2D{T}"/> instance.
|
|
/// For this API to succeed, the target <see cref="Span2D{T}"/> has to have the same shape as the current one.
|
|
/// </summary>
|
|
/// <param name="destination">The destination <see cref="Span2D{T}"/> instance.</param>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown when <paramref name="destination" /> does not have the same shape as the source <see cref="ReadOnlySpan2D{T}"/> instance.
|
|
/// </exception>
|
|
public void CopyTo(Span2D<T> destination)
|
|
{
|
|
if (destination.Height != Height ||
|
|
destination.Width != Width)
|
|
{
|
|
ThrowHelper.ThrowArgumentExceptionForDestinationWithNotSameShape();
|
|
}
|
|
|
|
if (IsEmpty)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (destination.TryGetSpan(out Span<T> span))
|
|
{
|
|
CopyTo(span);
|
|
}
|
|
else
|
|
{
|
|
// Copy each row individually
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
for (int i = 0; i < Height; i++)
|
|
{
|
|
GetRowSpan(i).CopyTo(destination.GetRowSpan(i));
|
|
}
|
|
#else
|
|
int height = Height;
|
|
nint width = (nint)(uint)this.width;
|
|
|
|
for (int i = 0; i < height; i++)
|
|
{
|
|
ref T sourceStart = ref DangerousGetReferenceAt(i, 0);
|
|
ref T sourceEnd = ref Unsafe.Add(ref sourceStart, width);
|
|
ref T destinationRef = ref destination.DangerousGetReferenceAt(i, 0);
|
|
|
|
while (Unsafe.IsAddressLessThan(ref sourceStart, ref sourceEnd))
|
|
{
|
|
destinationRef = sourceStart;
|
|
|
|
sourceStart = ref Unsafe.Add(ref sourceStart, 1);
|
|
destinationRef = ref Unsafe.Add(ref destinationRef, 1);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempts to copy the current <see cref="ReadOnlySpan2D{T}"/> instance to a destination <see cref="Span{T}"/>.
|
|
/// </summary>
|
|
/// <param name="destination">The target <see cref="Span{T}"/> of the copy operation.</param>
|
|
/// <returns>Whether or not the operation was successful.</returns>
|
|
public bool TryCopyTo(Span<T> destination)
|
|
{
|
|
if (destination.Length >= Length)
|
|
{
|
|
CopyTo(destination);
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempts to copy the current <see cref="ReadOnlySpan2D{T}"/> instance to a destination <see cref="Span2D{T}"/>.
|
|
/// </summary>
|
|
/// <param name="destination">The target <see cref="Span2D{T}"/> of the copy operation.</param>
|
|
/// <returns>Whether or not the operation was successful.</returns>
|
|
public bool TryCopyTo(Span2D<T> destination)
|
|
{
|
|
if (destination.Height == Height &&
|
|
destination.Width == Width)
|
|
{
|
|
CopyTo(destination);
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a reference to the 0th element of the <see cref="ReadOnlySpan2D{T}"/> instance. If the current
|
|
/// instance is empty, returns a <see langword="null"/> reference. It can be used for pinning
|
|
/// and is required to support the use of span within a fixed statement.
|
|
/// </summary>
|
|
/// <returns>A reference to the 0th element, or a <see langword="null"/> reference.</returns>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
public unsafe ref T GetPinnableReference()
|
|
{
|
|
ref T r0 = ref Unsafe.AsRef<T>(null);
|
|
|
|
if (Length != 0)
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
r0 = ref MemoryMarshal.GetReference(this.span);
|
|
#else
|
|
r0 = ref RuntimeHelpers.GetObjectDataAtOffsetOrPointerReference<T>(this.instance, this.offset);
|
|
#endif
|
|
}
|
|
|
|
return ref r0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a reference to the first element within the current instance, with no bounds check.
|
|
/// </summary>
|
|
/// <returns>A reference to the first element within the current instance.</returns>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public ref T DangerousGetReference()
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
return ref MemoryMarshal.GetReference(this.span);
|
|
#else
|
|
return ref RuntimeHelpers.GetObjectDataAtOffsetOrPointerReference<T>(this.instance, this.offset);
|
|
#endif
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a reference to a specified element within the current instance, with no bounds check.
|
|
/// </summary>
|
|
/// <param name="i">The target row to get the element from.</param>
|
|
/// <param name="j">The target column to get the element from.</param>
|
|
/// <returns>A reference to the element at the specified indices.</returns>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public ref T DangerousGetReferenceAt(int i, int j)
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
ref T r0 = ref MemoryMarshal.GetReference(this.span);
|
|
#else
|
|
ref T r0 = ref RuntimeHelpers.GetObjectDataAtOffsetOrPointerReference<T>(this.instance, this.offset);
|
|
#endif
|
|
nint index = ((nint)(uint)i * (nint)(uint)this.stride) + (nint)(uint)j;
|
|
|
|
return ref Unsafe.Add(ref r0, index);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Slices the current instance with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="row">The target row to map within the current instance.</param>
|
|
/// <param name="column">The target column to map within the current instance.</param>
|
|
/// <param name="height">The height to map within the current instance.</param>
|
|
/// <param name="width">The width to map within the current instance.</param>
|
|
/// <exception cref="ArgumentException">
|
|
/// Thrown when either <paramref name="height"/>, <paramref name="width"/> or <paramref name="height"/>
|
|
/// are negative or not within the bounds that are valid for the current instance.
|
|
/// </exception>
|
|
/// <returns>A new <see cref="ReadOnlySpan2D{T}"/> instance representing a slice of the current one.</returns>
|
|
public ReadOnlySpan2D<T> Slice(int row, int column, int height, int width)
|
|
{
|
|
if ((uint)row >= Height)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForRow();
|
|
}
|
|
|
|
if ((uint)column >= this.width)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForColumn();
|
|
}
|
|
|
|
if ((uint)height > (Height - row))
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForHeight();
|
|
}
|
|
|
|
if ((uint)width > (this.width - column))
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForWidth();
|
|
}
|
|
|
|
nint shift = ((nint)(uint)this.stride * (nint)(uint)row) + (nint)(uint)column;
|
|
int pitch = this.stride - width;
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
ref T r0 = ref this.span.DangerousGetReferenceAt(shift);
|
|
|
|
return new(in r0, height, width, pitch);
|
|
#else
|
|
IntPtr offset = this.offset + (shift * (nint)(uint)Unsafe.SizeOf<T>());
|
|
|
|
return new(this.instance, offset, height, width, pitch);
|
|
#endif
|
|
}
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
/// <summary>
|
|
/// Gets a <see cref="ReadOnlySpan{T}"/> for a specified row.
|
|
/// </summary>
|
|
/// <param name="row">The index of the target row to retrieve.</param>
|
|
/// <exception cref="ArgumentOutOfRangeException">Throw when <paramref name="row"/> is out of range.</exception>
|
|
/// <returns>The resulting row <see cref="ReadOnlySpan{T}"/>.</returns>
|
|
public ReadOnlySpan<T> GetRowSpan(int row)
|
|
{
|
|
if ((uint)row >= (uint)Height)
|
|
{
|
|
ThrowHelper.ThrowArgumentOutOfRangeExceptionForRow();
|
|
}
|
|
|
|
ref T r0 = ref DangerousGetReferenceAt(row, 0);
|
|
|
|
return MemoryMarshal.CreateReadOnlySpan(ref r0, this.width);
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Tries to get a <see cref="ReadOnlySpan{T}"/> instance, if the underlying buffer is contiguous and small enough.
|
|
/// </summary>
|
|
/// <param name="span">The resulting <see cref="ReadOnlySpan{T}"/>, in case of success.</param>
|
|
/// <returns>Whether or not <paramref name="span"/> was correctly assigned.</returns>
|
|
public bool TryGetSpan(out ReadOnlySpan<T> span)
|
|
{
|
|
// We can only create a Span<T> if the buffer is contiguous
|
|
if (this.stride == this.width &&
|
|
Length <= int.MaxValue)
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
span = MemoryMarshal.CreateReadOnlySpan(ref MemoryMarshal.GetReference(this.span), (int)Length);
|
|
|
|
return true;
|
|
#else
|
|
// An empty Span2D<T> is still valid
|
|
if (IsEmpty)
|
|
{
|
|
span = default;
|
|
|
|
return true;
|
|
}
|
|
|
|
// Pinned ReadOnlySpan2D<T>
|
|
if (this.instance is null)
|
|
{
|
|
unsafe
|
|
{
|
|
span = new Span<T>((void*)this.offset, (int)Length);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// Without Span<T> runtime support, we can only get a Span<T> from a T[] instance
|
|
if (this.instance.GetType() == typeof(T[]))
|
|
{
|
|
T[] array = Unsafe.As<T[]>(this.instance)!;
|
|
int index = array.AsSpan().IndexOf(ref ObjectMarshal.DangerousGetObjectDataReferenceAt<T>(array, this.offset));
|
|
|
|
span = array.AsSpan(index, (int)Length);
|
|
|
|
return true;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
span = default;
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Copies the contents of the current <see cref="Span2D{T}"/> instance into a new 2D array.
|
|
/// </summary>
|
|
/// <returns>A 2D array containing the data in the current <see cref="Span2D{T}"/> instance.</returns>
|
|
public T[,] ToArray()
|
|
{
|
|
T[,] array = new T[Height, this.width];
|
|
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
CopyTo(array.AsSpan());
|
|
#else
|
|
// Skip the initialization if the array is empty
|
|
if (Length > 0)
|
|
{
|
|
int height = Height;
|
|
nint width = (nint)(uint)this.width;
|
|
|
|
ref T destinationRef = ref array.DangerousGetReference();
|
|
|
|
for (int i = 0; i < height; i++)
|
|
{
|
|
ref T sourceStart = ref DangerousGetReferenceAt(i, 0);
|
|
ref T sourceEnd = ref Unsafe.Add(ref sourceStart, width);
|
|
|
|
while (Unsafe.IsAddressLessThan(ref sourceStart, ref sourceEnd))
|
|
{
|
|
destinationRef = sourceStart;
|
|
|
|
sourceStart = ref Unsafe.Add(ref sourceStart, 1);
|
|
destinationRef = ref Unsafe.Add(ref destinationRef, 1);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return array;
|
|
}
|
|
|
|
/// <inheritdoc cref="ReadOnlySpan{T}.Equals(object)"/>
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
[Obsolete("Equals() on Span will always throw an exception. Use == instead.")]
|
|
public override bool Equals(object? obj)
|
|
{
|
|
throw new NotSupportedException("CommunityToolkit.HighPerformance.ReadOnlySpan2D<T>.Equals(object) is not supported.");
|
|
}
|
|
|
|
/// <inheritdoc cref="ReadOnlySpan{T}.GetHashCode()"/>
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
[Obsolete("GetHashCode() on Span will always throw an exception.")]
|
|
public override int GetHashCode()
|
|
{
|
|
throw new NotSupportedException("CommunityToolkit.HighPerformance.ReadOnlySpan2D<T>.GetHashCode() is not supported.");
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public override string ToString()
|
|
{
|
|
return $"CommunityToolkit.HighPerformance.ReadOnlySpan2D<{typeof(T)}>[{Height}, {this.width}]";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks whether two <see cref="ReadOnlySpan2D{T}"/> instances are equal.
|
|
/// </summary>
|
|
/// <param name="left">The first <see cref="ReadOnlySpan2D{T}"/> instance to compare.</param>
|
|
/// <param name="right">The second <see cref="ReadOnlySpan2D{T}"/> instance to compare.</param>
|
|
/// <returns>Whether or not <paramref name="left"/> and <paramref name="right"/> are equal.</returns>
|
|
public static bool operator ==(ReadOnlySpan2D<T> left, ReadOnlySpan2D<T> right)
|
|
{
|
|
return
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
left.span == right.span &&
|
|
#else
|
|
ReferenceEquals(
|
|
left.instance, right.instance) &&
|
|
left.offset == right.offset &&
|
|
left.height == right.height &&
|
|
#endif
|
|
left.width == right.width &&
|
|
left.stride == right.stride;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks whether two <see cref="ReadOnlySpan2D{T}"/> instances are not equal.
|
|
/// </summary>
|
|
/// <param name="left">The first <see cref="ReadOnlySpan2D{T}"/> instance to compare.</param>
|
|
/// <param name="right">The second <see cref="ReadOnlySpan2D{T}"/> instance to compare.</param>
|
|
/// <returns>Whether or not <paramref name="left"/> and <paramref name="right"/> are not equal.</returns>
|
|
public static bool operator !=(ReadOnlySpan2D<T> left, ReadOnlySpan2D<T> right)
|
|
{
|
|
return !(left == right);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Implicitly converts a given 2D array into a <see cref="ReadOnlySpan2D{T}"/> instance.
|
|
/// </summary>
|
|
/// <param name="array">The input 2D array to convert.</param>
|
|
public static implicit operator ReadOnlySpan2D<T>(T[,]? array) => new(array);
|
|
|
|
/// <summary>
|
|
/// Implicitly converts a given <see cref="Span2D{T}"/> into a <see cref="ReadOnlySpan2D{T}"/> instance.
|
|
/// </summary>
|
|
/// <param name="span">The input <see cref="Span2D{T}"/> to convert.</param>
|
|
public static implicit operator ReadOnlySpan2D<T>(Span2D<T> span)
|
|
{
|
|
#if NETSTANDARD2_1_OR_GREATER
|
|
return new(in span.DangerousGetReference(), span.Height, span.Width, span.Stride - span.Width);
|
|
#else
|
|
return new(span.Instance!, span.Offset, span.Height, span.Width, span.Stride - span.Width);
|
|
#endif
|
|
}
|
|
}
|