PooledBufferWriter<T>
Represents memory writer that uses pooled memory.
using System;
using System.Buffers;
using System.Runtime.CompilerServices;
namespace DotNext.Buffers
{
[System.Runtime.CompilerServices.Nullable(new byte[] {
0,
1
})]
public sealed class PooledBufferWriter<[System.Runtime.CompilerServices.Nullable(2)] T> : BufferWriter<T>, IMemoryOwner<T>, IDisposable
{
private readonly MemoryAllocator<T> allocator;
private MemoryOwner<T> buffer;
[System.Runtime.CompilerServices.Nullable(new byte[] {
2,
1
})]
public MemoryAllocator<T> BufferAllocator {
[param: System.Runtime.CompilerServices.Nullable(new byte[] {
2,
1
})]
set {
allocator = value;
}
}
public override int Capacity {
get {
ThrowIfDisposed();
return buffer.Length;
}
set {
if (value < 0)
throw new ArgumentOutOfRangeException("value");
if (value > 0)
buffer = MemoryAllocator.Invoke<T>(allocator, value, false);
}
}
[System.Runtime.CompilerServices.Nullable(new byte[] {
0,
1
})]
public override ReadOnlyMemory<T> WrittenMemory {
[return: System.Runtime.CompilerServices.Nullable(new byte[] {
0,
1
})]
get {
return GetWrittenMemory();
}
}
[System.Runtime.CompilerServices.Nullable(new byte[] {
0,
1
})]
Memory<T> IMemoryOwner<T>.Memory {
get {
return GetWrittenMemory();
}
}
[Obsolete("Use init-only properties to set the capacity and allocator")]
public PooledBufferWriter([System.Runtime.CompilerServices.Nullable(new byte[] {
2,
1
})] MemoryAllocator<T> allocator, int initialCapacity)
{
if (initialCapacity <= 0)
throw new ArgumentOutOfRangeException("initialCapacity");
this.allocator = allocator;
buffer = MemoryAllocator.Invoke<T>(allocator, initialCapacity, false);
}
[Obsolete("Use init-only properties to set the capacity and allocator")]
public PooledBufferWriter([System.Runtime.CompilerServices.Nullable(new byte[] {
2,
1
})] MemoryAllocator<T> allocator)
{
this.allocator = allocator;
}
public PooledBufferWriter()
{
}
private Memory<T> GetWrittenMemory()
{
ThrowIfDisposed();
return buffer.Memory.Slice(0, position);
}
public override void Clear(bool reuseBuffer = false)
{
ThrowIfDisposed();
if (!reuseBuffer)
buffer.Dispose();
else if (RuntimeHelpers.IsReferenceOrContainsReferences<T>()) {
buffer.Span.Clear();
}
position = 0;
}
[return: System.Runtime.CompilerServices.Nullable(new byte[] {
0,
1
})]
public override Memory<T> GetMemory(int sizeHint = 0)
{
if (sizeHint < 0)
throw new ArgumentOutOfRangeException("sizeHint");
CheckAndResizeBuffer(sizeHint);
return buffer.Memory.Slice(position);
}
[return: System.Runtime.CompilerServices.Nullable(new byte[] {
0,
1
})]
public override MemoryOwner<T> DetachBuffer()
{
ThrowIfDisposed();
MemoryOwner<T> result;
if (position > 0) {
result = buffer;
buffer = default(MemoryOwner<T>);
result.Truncate(position);
position = 0;
} else
result = default(MemoryOwner<T>);
return result;
}
private protected override void Resize(int newSize)
{
BufferHelpers.Resize<T>(ref buffer, newSize, false, allocator);
base.AllocationCounter?.WriteMetric((float)buffer.Length);
PooledBufferWriter.AllocationMeter.Record(buffer.Length, ref measurementTags);
}
protected override void Dispose(bool disposing)
{
if (disposing)
base.BufferSizeCallback?.Invoke(buffer.Length);
buffer.Dispose();
base.Dispose(disposing);
}
}
}