using System; using System.Threading; using System.Threading.Tasks; namespace NadekoBot.Modules.Music.Classes { /// /// 💩 /// public class PoopyBuffer { private readonly byte[] ringBuffer; public int WritePosition { get; private set; } = 0; public int ReadPosition { get; private set; } = 0; public int ContentLength => (WritePosition >= ReadPosition ? WritePosition - ReadPosition : (BufferSize - ReadPosition) + WritePosition); public int BufferSize { get; } private readonly object readWriteLock = new object(); public PoopyBuffer(int size) { if (size <= 0) throw new ArgumentException(); BufferSize = size; ringBuffer = new byte[size]; } public int Read(byte[] buffer, int count) { if (buffer.Length < count) throw new ArgumentException(); //Console.WriteLine($"***\nRead: {ReadPosition}\nWrite: {WritePosition}\nContentLength:{ContentLength}\n***"); lock (readWriteLock) { //read as much as you can if you're reading too much if (count > ContentLength) count = ContentLength; //if nothing to read, return 0 if (WritePosition == ReadPosition) return 0; // if buffer is in the "normal" state, just read if (WritePosition > ReadPosition) { Buffer.BlockCopy(ringBuffer, ReadPosition, buffer, 0, count); ReadPosition += count; //Console.WriteLine($"Read only normally1 {count}[{ReadPosition - count} to {ReadPosition}]"); return count; } //else ReadPos buffer.Length) throw new ArgumentException(); while (ContentLength + count > BufferSize) { await Task.Delay(20, cancelToken).ConfigureAwait(false); if (cancelToken.IsCancellationRequested) return; } //the while above assures that i cannot write past readposition with my write, so i don't have to check // *unless its multithreaded or task is not awaited lock (readWriteLock) { // if i can just write without hitting buffer.length, do it if (WritePosition + count < BufferSize) { Buffer.BlockCopy(buffer, 0, ringBuffer, WritePosition, count); WritePosition += count; //Console.WriteLine($"Wrote only normally {count}[{WritePosition - count} to {WritePosition}]"); return; } // otherwise, i have to write to the end, then write the rest from the start var wroteNormaly = BufferSize - WritePosition; Buffer.BlockCopy(buffer, 0, ringBuffer, WritePosition, wroteNormaly); //Console.WriteLine($"Wrote normally {wroteNormaly}[{WritePosition} to {BufferSize}]"); var wroteFromStart = count - wroteNormaly; Buffer.BlockCopy(buffer, wroteNormaly, ringBuffer, 0, wroteFromStart); //Console.WriteLine($"and from start {wroteFromStart} [0 to {wroteFromStart}"); WritePosition = wroteFromStart; } } } }