ValueStringBuilder
struct ValueStringBuilder
using System.Buffers;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
namespace System.Text
{
internal ref struct ValueStringBuilder
{
private char[] _arrayToReturnToPool;
private Span<char> _chars;
private int _pos;
public int Length {
get {
return _pos;
}
set {
_pos = value;
}
}
public int Capacity => _chars.Length;
public ref char this[int index] {
get {
return ref _chars[index];
}
}
public Span<char> RawChars => _chars;
public ValueStringBuilder(Span<char> initialBuffer)
{
_arrayToReturnToPool = null;
_chars = initialBuffer;
_pos = 0;
}
public ValueStringBuilder(int initialCapacity)
{
_arrayToReturnToPool = ArrayPool<char>.Shared.Rent(initialCapacity);
_chars = _arrayToReturnToPool;
_pos = 0;
}
public void EnsureCapacity(int capacity)
{
if ((uint)capacity > (uint)_chars.Length)
Grow(capacity - _pos);
}
public ref char GetPinnableReference()
{
return ref MemoryMarshal.GetReference(_chars);
}
public ref char GetPinnableReference(bool terminate)
{
if (terminate) {
EnsureCapacity(Length + 1);
_chars[Length] = ' ';
}
return ref MemoryMarshal.GetReference(_chars);
}
public override string ToString()
{
string result = _chars.Slice(0, _pos).ToString();
Dispose();
return result;
}
public ReadOnlySpan<char> AsSpan(bool terminate)
{
if (terminate) {
EnsureCapacity(Length + 1);
_chars[Length] = ' ';
}
return _chars.Slice(0, _pos);
}
public ReadOnlySpan<char> AsSpan()
{
return _chars.Slice(0, _pos);
}
public ReadOnlySpan<char> AsSpan(int start)
{
return _chars.Slice(start, _pos - start);
}
public ReadOnlySpan<char> AsSpan(int start, int length)
{
return _chars.Slice(start, length);
}
public bool TryCopyTo(Span<char> destination, out int charsWritten)
{
if (_chars.Slice(0, _pos).TryCopyTo(destination)) {
charsWritten = _pos;
Dispose();
return true;
}
charsWritten = 0;
Dispose();
return false;
}
public void Insert(int index, char value, int count)
{
if (_pos > _chars.Length - count)
Grow(count);
int length = _pos - index;
Span<char> span = _chars.Slice(index, length);
span.CopyTo(_chars.Slice(index + count));
span = _chars.Slice(index, count);
span.Fill(value);
_pos += count;
}
public void Insert(int index, string s)
{
if (s != null) {
int length = s.Length;
if (_pos > _chars.Length - length)
Grow(length);
int length2 = _pos - index;
_chars.Slice(index, length2).CopyTo(_chars.Slice(index + length));
s.AsSpan().CopyTo(_chars.Slice(index));
_pos += length;
}
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public void Append(char c)
{
int pos = _pos;
Span<char> chars = _chars;
if ((uint)pos < (uint)chars.Length) {
chars[pos] = c;
_pos = pos + 1;
} else
GrowAndAppend(c);
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public void Append(string s)
{
if (s != null) {
int pos = _pos;
if (s.Length == 1 && (uint)pos < (uint)_chars.Length) {
_chars[pos] = s[0];
_pos = pos + 1;
} else
AppendSlow(s);
}
}
private void AppendSlow(string s)
{
int pos = _pos;
if (pos > _chars.Length - s.Length)
Grow(s.Length);
s.AsSpan().CopyTo(_chars.Slice(pos));
_pos += s.Length;
}
public void Append(char c, int count)
{
if (_pos > _chars.Length - count)
Grow(count);
Span<char> span = _chars.Slice(_pos, count);
for (int i = 0; i < span.Length; i++) {
span[i] = c;
}
_pos += count;
}
public unsafe void Append(char* value, int length)
{
if (_pos > _chars.Length - length)
Grow(length);
Span<char> span = _chars.Slice(_pos, length);
for (int i = 0; i < span.Length; i++) {
ref char reference = ref span[i];
char* intPtr = value;
value = intPtr + 1;
reference = *intPtr;
}
_pos += length;
}
public void Append([System.Runtime.CompilerServices.ScopedRef] ReadOnlySpan<char> value)
{
if (_pos > _chars.Length - value.Length)
Grow(value.Length);
value.CopyTo(_chars.Slice(_pos));
_pos += value.Length;
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public Span<char> AppendSpan(int length)
{
int pos = _pos;
if (pos > _chars.Length - length)
Grow(length);
_pos = pos + length;
return _chars.Slice(pos, length);
}
[MethodImpl(MethodImplOptions.NoInlining)]
private void GrowAndAppend(char c)
{
Grow(1);
Append(c);
}
[MethodImpl(MethodImplOptions.NoInlining)]
private void Grow(int additionalCapacityBeyondPos)
{
int minimumLength = (int)Math.Max((uint)(_pos + additionalCapacityBeyondPos), Math.Min((uint)(_chars.Length * 2), 2147483591));
char[] array = ArrayPool<char>.Shared.Rent(minimumLength);
_chars.Slice(0, _pos).CopyTo(array);
char[] arrayToReturnToPool = _arrayToReturnToPool;
_chars = (_arrayToReturnToPool = array);
if (arrayToReturnToPool != null)
ArrayPool<char>.Shared.Return(arrayToReturnToPool, false);
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public void Dispose()
{
char[] arrayToReturnToPool = _arrayToReturnToPool;
this = default(System.Text.ValueStringBuilder);
if (arrayToReturnToPool != null)
ArrayPool<char>.Shared.Return(arrayToReturnToPool, false);
}
}
}