| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181 |
- using System;
- using System.Collections.Generic;
- namespace Model
- {
- public class CircularBuffer
- {
- public int ChunkSize = 8192;
- private readonly Queue<byte[]> bufferQueue = new Queue<byte[]>();
- private readonly Queue<byte[]> bufferCache = new Queue<byte[]>();
- public int LastIndex { get; set; }
- public int FirstIndex { get; set; }
-
- private byte[] lastBuffer;
- public CircularBuffer()
- {
- this.AddLast();
- }
- public CircularBuffer(int chunkSize)
- {
- this.ChunkSize = chunkSize;
- this.AddLast();
- }
- public int Count
- {
- get
- {
- int c = 0;
- if (this.bufferQueue.Count == 0)
- {
- c = 0;
- }
- else
- {
- c = (this.bufferQueue.Count - 1) * ChunkSize + this.LastIndex - this.FirstIndex;
- }
- if (c < 0)
- {
- Log.Error("TBuffer count < 0: {0}, {1}, {2}".Fmt(this.bufferQueue.Count, this.LastIndex, this.FirstIndex));
- }
- return c;
- }
- }
- public void AddLast()
- {
- byte[] buffer;
- if (this.bufferCache.Count > 0)
- {
- buffer = this.bufferCache.Dequeue();
- }
- else
- {
- buffer = new byte[ChunkSize];
- }
- this.bufferQueue.Enqueue(buffer);
- this.lastBuffer = buffer;
- }
- public void RemoveFirst()
- {
- this.bufferCache.Enqueue(bufferQueue.Dequeue());
- }
- public byte[] First
- {
- get
- {
- if (this.bufferQueue.Count == 0)
- {
- this.AddLast();
- }
- return this.bufferQueue.Peek();
- }
- }
- public byte[] Last
- {
- get
- {
- if (this.bufferQueue.Count == 0)
- {
- this.AddLast();
- }
- return this.lastBuffer;
- }
- }
- public void RecvFrom(byte[] buffer, int count)
- {
- if (this.Count < count)
- {
- throw new Exception($"bufferList size < n, bufferList: {this.Count} buffer length: {buffer.Length} {count}");
- }
- if (buffer.Length < count)
- {
- throw new Exception($"bufferList length < coutn, buffer length: {buffer.Length} {count}");
- }
- int alreadyCopyCount = 0;
- while (alreadyCopyCount < count)
- {
- int n = count - alreadyCopyCount;
- if (ChunkSize - this.FirstIndex > n)
- {
- Array.Copy(this.First, this.FirstIndex, buffer, alreadyCopyCount, n);
- this.FirstIndex += n;
- alreadyCopyCount += n;
- }
- else
- {
- Array.Copy(this.First, this.FirstIndex, buffer, alreadyCopyCount, ChunkSize - this.FirstIndex);
- alreadyCopyCount += ChunkSize - this.FirstIndex;
- this.FirstIndex = 0;
- this.RemoveFirst();
- }
- }
- }
- public void SendTo(byte[] buffer)
- {
- int alreadyCopyCount = 0;
- while (alreadyCopyCount < buffer.Length)
- {
- if (this.LastIndex == ChunkSize)
- {
- this.AddLast();
- this.LastIndex = 0;
- }
- int n = buffer.Length - alreadyCopyCount;
- if (ChunkSize - this.LastIndex > n)
- {
- Array.Copy(buffer, alreadyCopyCount, this.lastBuffer, this.LastIndex, n);
- this.LastIndex += buffer.Length - alreadyCopyCount;
- alreadyCopyCount += n;
- }
- else
- {
- Array.Copy(buffer, alreadyCopyCount, this.lastBuffer, this.LastIndex, ChunkSize - this.LastIndex);
- alreadyCopyCount += ChunkSize - this.LastIndex;
- this.LastIndex = ChunkSize;
- }
- }
- }
- public void SendTo(byte[] buffer, int offset, int count)
- {
- int alreadyCopyCount = 0;
- while (alreadyCopyCount < count)
- {
- if (this.LastIndex == ChunkSize)
- {
- this.AddLast();
- this.LastIndex = 0;
- }
- int n = count - alreadyCopyCount;
- if (ChunkSize - this.LastIndex > n)
- {
- Array.Copy(buffer, alreadyCopyCount + offset, this.lastBuffer, this.LastIndex, n);
- this.LastIndex += count - alreadyCopyCount;
- alreadyCopyCount += n;
- }
- else
- {
- Array.Copy(buffer, alreadyCopyCount + offset, this.lastBuffer, this.LastIndex, ChunkSize - this.LastIndex);
- alreadyCopyCount += ChunkSize - this.LastIndex;
- this.LastIndex = ChunkSize;
- }
- }
- }
- }
- }
|