mirror of
https://github.com/chylex/.NET-Community-Toolkit.git
synced 2024-11-24 07:42:45 +01:00
346 lines
14 KiB
C#
346 lines
14 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.Buffers;
|
|
using System.Diagnostics;
|
|
using System.Runtime.CompilerServices;
|
|
#if NETCOREAPP3_1_OR_GREATER
|
|
using System.Runtime.InteropServices;
|
|
#endif
|
|
using CommunityToolkit.HighPerformance.Buffers.Views;
|
|
|
|
namespace CommunityToolkit.HighPerformance.Buffers;
|
|
|
|
/// <summary>
|
|
/// An <see cref="IMemoryOwner{T}"/> implementation with an embedded length and a fast <see cref="Span{T}"/> accessor.
|
|
/// </summary>
|
|
/// <typeparam name="T">The type of items to store in the current instance.</typeparam>
|
|
[DebuggerTypeProxy(typeof(MemoryDebugView<>))]
|
|
[DebuggerDisplay("{ToString(),raw}")]
|
|
public sealed class MemoryOwner<T> : IMemoryOwner<T>
|
|
{
|
|
/// <summary>
|
|
/// The starting offset within <see cref="array"/>.
|
|
/// </summary>
|
|
private readonly int start;
|
|
|
|
#pragma warning disable IDE0032
|
|
/// <summary>
|
|
/// The usable length within <see cref="array"/> (starting from <see cref="start"/>).
|
|
/// </summary>
|
|
private readonly int length;
|
|
#pragma warning restore IDE0032
|
|
|
|
/// <summary>
|
|
/// The <see cref="ArrayPool{T}"/> instance used to rent <see cref="array"/>.
|
|
/// </summary>
|
|
private readonly ArrayPool<T> pool;
|
|
|
|
/// <summary>
|
|
/// The underlying <typeparamref name="T"/> array.
|
|
/// </summary>
|
|
private T[]? array;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="MemoryOwner{T}"/> class with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="length">The length of the new memory buffer to use.</param>
|
|
/// <param name="pool">The <see cref="ArrayPool{T}"/> instance to use.</param>
|
|
/// <param name="mode">Indicates the allocation mode to use for the new buffer to rent.</param>
|
|
private MemoryOwner(int length, ArrayPool<T> pool, AllocationMode mode)
|
|
{
|
|
this.start = 0;
|
|
this.length = length;
|
|
this.pool = pool;
|
|
this.array = pool.Rent(length);
|
|
|
|
if (mode == AllocationMode.Clear)
|
|
{
|
|
this.array.AsSpan(0, length).Clear();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="MemoryOwner{T}"/> class with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="start">The starting offset within <paramref name="array"/>.</param>
|
|
/// <param name="length">The length of the array to use.</param>
|
|
/// <param name="pool">The <see cref="ArrayPool{T}"/> instance currently in use.</param>
|
|
/// <param name="array">The input <typeparamref name="T"/> array to use.</param>
|
|
private MemoryOwner(int start, int length, ArrayPool<T> pool, T[] array)
|
|
{
|
|
this.start = start;
|
|
this.length = length;
|
|
this.pool = pool;
|
|
this.array = array;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Finalizes an instance of the <see cref="MemoryOwner{T}"/> class.
|
|
/// </summary>
|
|
~MemoryOwner() => Dispose();
|
|
|
|
/// <summary>
|
|
/// Gets an empty <see cref="MemoryOwner{T}"/> instance.
|
|
/// </summary>
|
|
public static MemoryOwner<T> Empty
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => new(0, ArrayPool<T>.Shared, AllocationMode.Default);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a new <see cref="MemoryOwner{T}"/> instance with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="size">The length of the new memory buffer to use.</param>
|
|
/// <returns>A <see cref="MemoryOwner{T}"/> instance of the requested length.</returns>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when <paramref name="size"/> is not valid.</exception>
|
|
/// <remarks>This method is just a proxy for the <see langword="private"/> constructor, for clarity.</remarks>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static MemoryOwner<T> Allocate(int size) => new(size, ArrayPool<T>.Shared, AllocationMode.Default);
|
|
|
|
/// <summary>
|
|
/// Creates a new <see cref="MemoryOwner{T}"/> instance with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="size">The length of the new memory buffer to use.</param>
|
|
/// <param name="pool">The <see cref="ArrayPool{T}"/> instance currently in use.</param>
|
|
/// <returns>A <see cref="MemoryOwner{T}"/> instance of the requested length.</returns>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when <paramref name="size"/> is not valid.</exception>
|
|
/// <remarks>This method is just a proxy for the <see langword="private"/> constructor, for clarity.</remarks>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static MemoryOwner<T> Allocate(int size, ArrayPool<T> pool) => new(size, pool, AllocationMode.Default);
|
|
|
|
/// <summary>
|
|
/// Creates a new <see cref="MemoryOwner{T}"/> instance with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="size">The length of the new memory buffer to use.</param>
|
|
/// <param name="mode">Indicates the allocation mode to use for the new buffer to rent.</param>
|
|
/// <returns>A <see cref="MemoryOwner{T}"/> instance of the requested length.</returns>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when <paramref name="size"/> is not valid.</exception>
|
|
/// <remarks>This method is just a proxy for the <see langword="private"/> constructor, for clarity.</remarks>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static MemoryOwner<T> Allocate(int size, AllocationMode mode) => new(size, ArrayPool<T>.Shared, mode);
|
|
|
|
/// <summary>
|
|
/// Creates a new <see cref="MemoryOwner{T}"/> instance with the specified parameters.
|
|
/// </summary>
|
|
/// <param name="size">The length of the new memory buffer to use.</param>
|
|
/// <param name="pool">The <see cref="ArrayPool{T}"/> instance currently in use.</param>
|
|
/// <param name="mode">Indicates the allocation mode to use for the new buffer to rent.</param>
|
|
/// <returns>A <see cref="MemoryOwner{T}"/> instance of the requested length.</returns>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when <paramref name="size"/> is not valid.</exception>
|
|
/// <remarks>This method is just a proxy for the <see langword="private"/> constructor, for clarity.</remarks>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public static MemoryOwner<T> Allocate(int size, ArrayPool<T> pool, AllocationMode mode) => new(size, pool, mode);
|
|
|
|
/// <summary>
|
|
/// Gets the number of items in the current instance
|
|
/// </summary>
|
|
public int Length
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => this.length;
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public Memory<T> Memory
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
T[]? array = this.array;
|
|
|
|
if (array is null)
|
|
{
|
|
ThrowObjectDisposedException();
|
|
}
|
|
|
|
return new(array!, this.start, this.length);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="Span{T}"/> wrapping the memory belonging to the current instance.
|
|
/// </summary>
|
|
public Span<T> Span
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
T[]? array = this.array;
|
|
|
|
if (array is null)
|
|
{
|
|
ThrowObjectDisposedException();
|
|
}
|
|
|
|
#if NETCOREAPP3_1_OR_GREATER
|
|
ref T r0 = ref array!.DangerousGetReferenceAt(this.start);
|
|
|
|
// On .NET Core runtimes, we can manually create a span from the starting reference to
|
|
// skip the argument validations, which include an explicit null check, covariance check
|
|
// for the array and the actual validation for the starting offset and target length. We
|
|
// only do this on .NET Core as we can leverage the runtime-specific array layout to get
|
|
// a fast access to the initial element, which makes this trick worth it. Otherwise, on
|
|
// runtimes where we would need to at least access a static field to retrieve the base
|
|
// byte offset within an SZ array object, we can get better performance by just using the
|
|
// default Span<T> constructor and paying the cost of the extra conditional branches,
|
|
// especially if T is a value type, in which case the covariance check is JIT removed.
|
|
return MemoryMarshal.CreateSpan(ref r0, this.length);
|
|
#else
|
|
return new(array!, this.start, this.length);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/// <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>
|
|
/// <exception cref="ObjectDisposedException">Thrown when the buffer in use has already been disposed.</exception>
|
|
/// <remarks>
|
|
/// This method does not perform bounds checks on the underlying buffer, but does check whether
|
|
/// the buffer itself has been disposed or not. This check should not be removed, and it's also
|
|
/// the reason why the method to get a reference at a specified offset is not present.
|
|
/// </remarks>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public ref T DangerousGetReference()
|
|
{
|
|
T[]? array = this.array;
|
|
|
|
if (array is null)
|
|
{
|
|
ThrowObjectDisposedException();
|
|
}
|
|
|
|
return ref array!.DangerousGetReferenceAt(this.start);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets an <see cref="ArraySegment{T}"/> instance wrapping the underlying <typeparamref name="T"/> array in use.
|
|
/// </summary>
|
|
/// <returns>An <see cref="ArraySegment{T}"/> instance wrapping the underlying <typeparamref name="T"/> array in use.</returns>
|
|
/// <exception cref="ObjectDisposedException">Thrown when the buffer in use has already been disposed.</exception>
|
|
/// <remarks>
|
|
/// This method is meant to be used when working with APIs that only accept an array as input, and should be used with caution.
|
|
/// In particular, the returned array is rented from an array pool, and it is responsibility of the caller to ensure that it's
|
|
/// not used after the current <see cref="MemoryOwner{T}"/> instance is disposed. Doing so is considered undefined behavior,
|
|
/// as the same array might be in use within another <see cref="MemoryOwner{T}"/> instance.
|
|
/// </remarks>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public ArraySegment<T> DangerousGetArray()
|
|
{
|
|
T[]? array = this.array;
|
|
|
|
if (array is null)
|
|
{
|
|
ThrowObjectDisposedException();
|
|
}
|
|
|
|
return new(array!, this.start, this.length);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Slices the buffer currently in use and returns a new <see cref="MemoryOwner{T}"/> instance.
|
|
/// </summary>
|
|
/// <param name="start">The starting offset within the current buffer.</param>
|
|
/// <param name="length">The length of the buffer to use.</param>
|
|
/// <returns>A new <see cref="MemoryOwner{T}"/> instance using the target range of items.</returns>
|
|
/// <exception cref="ObjectDisposedException">Thrown when the buffer in use has already been disposed.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">Thrown when <paramref name="start"/> or <paramref name="length"/> are not valid.</exception>
|
|
/// <remarks>
|
|
/// Using this method will dispose the current instance, and should only be used when an oversized
|
|
/// buffer is rented and then adjusted in size, to avoid having to rent a new buffer of the new
|
|
/// size and copy the previous items into the new one, or needing an additional variable/field
|
|
/// to manually handle to track the used range within a given <see cref="MemoryOwner{T}"/> instance.
|
|
/// </remarks>
|
|
public MemoryOwner<T> Slice(int start, int length)
|
|
{
|
|
T[]? array = this.array;
|
|
|
|
if (array is null)
|
|
{
|
|
ThrowObjectDisposedException();
|
|
}
|
|
|
|
this.array = null;
|
|
|
|
if ((uint)start > this.length)
|
|
{
|
|
ThrowInvalidOffsetException();
|
|
}
|
|
|
|
if ((uint)length > (this.length - start))
|
|
{
|
|
ThrowInvalidLengthException();
|
|
}
|
|
|
|
// We're transferring the ownership of the underlying array, so the current
|
|
// instance no longer needs to be disposed. Because of this, we can manually
|
|
// suppress the finalizer to reduce the overhead on the garbage collector.
|
|
GC.SuppressFinalize(this);
|
|
|
|
return new(start, length, this.pool, array!);
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public void Dispose()
|
|
{
|
|
T[]? array = this.array;
|
|
|
|
if (array is null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
this.array = null;
|
|
|
|
this.pool.Return(array);
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public override string ToString()
|
|
{
|
|
// Normally we would throw if the array has been disposed,
|
|
// but in this case we'll just return the non formatted
|
|
// representation as a fallback, since the ToString method
|
|
// is generally expected not to throw exceptions.
|
|
if (typeof(T) == typeof(char) &&
|
|
this.array is char[] chars)
|
|
{
|
|
return new(chars, this.start, this.length);
|
|
}
|
|
|
|
// Same representation used in Span<T>
|
|
return $"CommunityToolkit.HighPerformance.Buffers.MemoryOwner<{typeof(T)}>[{this.length}]";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Throws an <see cref="ObjectDisposedException"/> when <see cref="array"/> is <see langword="null"/>.
|
|
/// </summary>
|
|
private static void ThrowObjectDisposedException()
|
|
{
|
|
throw new ObjectDisposedException(nameof(MemoryOwner<T>), "The current buffer has already been disposed");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Throws an <see cref="ArgumentOutOfRangeException"/> when the <see cref="start"/> is invalid.
|
|
/// </summary>
|
|
private static void ThrowInvalidOffsetException()
|
|
{
|
|
throw new ArgumentOutOfRangeException(nameof(start), "The input start parameter was not valid");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Throws an <see cref="ArgumentOutOfRangeException"/> when the <see cref="length"/> is invalid.
|
|
/// </summary>
|
|
private static void ThrowInvalidLengthException()
|
|
{
|
|
throw new ArgumentOutOfRangeException(nameof(length), "The input length parameter was not valid");
|
|
}
|
|
}
|