Overwrite
Complete Overwrite of the Folder with the free shard. ServUO 57.3 has been added.
This commit is contained in:
263
Server/Persistence/FileQueue.cs
Normal file
263
Server/Persistence/FileQueue.cs
Normal file
@@ -0,0 +1,263 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Threading;
|
||||
using Server.Network;
|
||||
|
||||
namespace Server
|
||||
{
|
||||
public delegate void FileCommitCallback(FileQueue.Chunk chunk);
|
||||
|
||||
public sealed class FileQueue : IDisposable
|
||||
{
|
||||
private static int bufferSize;
|
||||
private static BufferPool bufferPool;
|
||||
private readonly object syncRoot;
|
||||
private readonly Chunk[] active;
|
||||
private readonly Queue<Page> pending;
|
||||
private Page buffered;
|
||||
private readonly FileCommitCallback callback;
|
||||
private int activeCount;
|
||||
private ManualResetEvent idle;
|
||||
private long position;
|
||||
public FileQueue(int concurrentWrites, FileCommitCallback callback)
|
||||
{
|
||||
if (concurrentWrites < 1)
|
||||
{
|
||||
throw new ArgumentOutOfRangeException("concurrentWrites");
|
||||
}
|
||||
else if (bufferSize < 1)
|
||||
{
|
||||
throw new ArgumentOutOfRangeException("bufferSize");
|
||||
}
|
||||
else if (callback == null)
|
||||
{
|
||||
throw new ArgumentNullException("callback");
|
||||
}
|
||||
|
||||
this.syncRoot = new object();
|
||||
|
||||
this.active = new Chunk[concurrentWrites];
|
||||
this.pending = new Queue<Page>();
|
||||
|
||||
this.callback = callback;
|
||||
|
||||
this.idle = new ManualResetEvent(true);
|
||||
}
|
||||
|
||||
static FileQueue()
|
||||
{
|
||||
bufferSize = FileOperations.BufferSize;
|
||||
bufferPool = new BufferPool("File Buffers", 64, bufferSize);
|
||||
}
|
||||
|
||||
public long Position
|
||||
{
|
||||
get
|
||||
{
|
||||
return this.position;
|
||||
}
|
||||
}
|
||||
public void Dispose()
|
||||
{
|
||||
if (this.idle != null)
|
||||
{
|
||||
this.idle.Close();
|
||||
this.idle = null;
|
||||
}
|
||||
}
|
||||
|
||||
public void Flush()
|
||||
{
|
||||
if (this.buffered.buffer != null)
|
||||
{
|
||||
this.Append(this.buffered);
|
||||
|
||||
this.buffered.buffer = null;
|
||||
this.buffered.length = 0;
|
||||
}
|
||||
|
||||
/*lock ( syncRoot ) {
|
||||
if ( pending.Count > 0 ) {
|
||||
idle.Reset();
|
||||
}
|
||||
|
||||
for ( int slot = 0; slot < active.Length && pending.Count > 0; ++slot ) {
|
||||
if ( active[slot] == null ) {
|
||||
Page page = pending.Dequeue();
|
||||
|
||||
active[slot] = new Chunk( this, slot, page.buffer, 0, page.length );
|
||||
|
||||
++activeCount;
|
||||
|
||||
callback( active[slot] );
|
||||
}
|
||||
}
|
||||
}*/
|
||||
|
||||
this.idle.WaitOne();
|
||||
}
|
||||
|
||||
public void Enqueue(byte[] buffer, int offset, int size)
|
||||
{
|
||||
if (buffer == null)
|
||||
{
|
||||
throw new ArgumentNullException("buffer");
|
||||
}
|
||||
else if (offset < 0)
|
||||
{
|
||||
throw new ArgumentOutOfRangeException("offset");
|
||||
}
|
||||
else if (size < 0)
|
||||
{
|
||||
throw new ArgumentOutOfRangeException("size");
|
||||
}
|
||||
else if ((buffer.Length - offset) < size)
|
||||
{
|
||||
throw new ArgumentException();
|
||||
}
|
||||
|
||||
this.position += size;
|
||||
|
||||
while (size > 0)
|
||||
{
|
||||
if (this.buffered.buffer == null)
|
||||
{ // nothing yet buffered
|
||||
this.buffered.buffer = bufferPool.AcquireBuffer();
|
||||
}
|
||||
|
||||
byte[] page = this.buffered.buffer; // buffer page
|
||||
int pageSpace = page.Length - this.buffered.length; // available bytes in page
|
||||
int byteCount = (size > pageSpace ? pageSpace : size); // how many bytes we can copy over
|
||||
|
||||
Buffer.BlockCopy(buffer, offset, page, this.buffered.length, byteCount);
|
||||
|
||||
this.buffered.length += byteCount;
|
||||
offset += byteCount;
|
||||
size -= byteCount;
|
||||
|
||||
if (this.buffered.length == page.Length)
|
||||
{ // page full
|
||||
this.Append(this.buffered);
|
||||
|
||||
this.buffered.buffer = null;
|
||||
this.buffered.length = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void Append(Page page)
|
||||
{
|
||||
lock (this.syncRoot)
|
||||
{
|
||||
if (this.activeCount == 0)
|
||||
{
|
||||
this.idle.Reset();
|
||||
}
|
||||
|
||||
++this.activeCount;
|
||||
|
||||
for (int slot = 0; slot < this.active.Length; ++slot)
|
||||
{
|
||||
if (this.active[slot] == null)
|
||||
{
|
||||
this.active[slot] = new Chunk(this, slot, page.buffer, 0, page.length);
|
||||
|
||||
this.callback(this.active[slot]);
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
this.pending.Enqueue(page);
|
||||
}
|
||||
}
|
||||
|
||||
private void Commit(Chunk chunk, int slot)
|
||||
{
|
||||
if (slot < 0 || slot >= this.active.Length)
|
||||
{
|
||||
throw new ArgumentOutOfRangeException("slot");
|
||||
}
|
||||
|
||||
lock (this.syncRoot)
|
||||
{
|
||||
if (this.active[slot] != chunk)
|
||||
{
|
||||
throw new ArgumentException();
|
||||
}
|
||||
|
||||
bufferPool.ReleaseBuffer(chunk.Buffer);
|
||||
|
||||
if (this.pending.Count > 0)
|
||||
{
|
||||
Page page = this.pending.Dequeue();
|
||||
|
||||
this.active[slot] = new Chunk(this, slot, page.buffer, 0, page.length);
|
||||
|
||||
this.callback(this.active[slot]);
|
||||
}
|
||||
else
|
||||
{
|
||||
this.active[slot] = null;
|
||||
}
|
||||
|
||||
--this.activeCount;
|
||||
|
||||
if (this.activeCount == 0)
|
||||
{
|
||||
this.idle.Set();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private struct Page
|
||||
{
|
||||
public byte[] buffer;
|
||||
public int length;
|
||||
}
|
||||
|
||||
public sealed class Chunk
|
||||
{
|
||||
private readonly FileQueue owner;
|
||||
private readonly int slot;
|
||||
private readonly byte[] buffer;
|
||||
private readonly int offset;
|
||||
private readonly int size;
|
||||
public Chunk(FileQueue owner, int slot, byte[] buffer, int offset, int size)
|
||||
{
|
||||
this.owner = owner;
|
||||
this.slot = slot;
|
||||
|
||||
this.buffer = buffer;
|
||||
this.offset = offset;
|
||||
this.size = size;
|
||||
}
|
||||
|
||||
public byte[] Buffer
|
||||
{
|
||||
get
|
||||
{
|
||||
return this.buffer;
|
||||
}
|
||||
}
|
||||
public int Offset
|
||||
{
|
||||
get
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
public int Size
|
||||
{
|
||||
get
|
||||
{
|
||||
return this.size;
|
||||
}
|
||||
}
|
||||
public void Commit()
|
||||
{
|
||||
this.owner.Commit(this, this.slot);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user