237 lines
No EOL
5.5 KiB
C#
237 lines
No EOL
5.5 KiB
C#
/***************************************************************************
|
|
* SendQueue.cs
|
|
* -------------------
|
|
* begin : May 1, 2002
|
|
* copyright : (C) The RunUO Software Team
|
|
* email : info@runuo.com
|
|
*
|
|
* $Id$
|
|
*
|
|
***************************************************************************/
|
|
|
|
/***************************************************************************
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
***************************************************************************/
|
|
|
|
using System;
|
|
using System.IO;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
|
|
namespace Server.Network {
|
|
public class SendQueue {
|
|
public class Gram {
|
|
private static Stack<Gram> _pool = new Stack<Gram>();
|
|
|
|
public static Gram Acquire() {
|
|
lock ( _pool ) {
|
|
Gram gram;
|
|
|
|
if ( _pool.Count > 0 ) {
|
|
gram = _pool.Pop();
|
|
} else {
|
|
gram = new Gram();
|
|
}
|
|
|
|
gram._buffer = AcquireBuffer();
|
|
gram._length = 0;
|
|
|
|
return gram;
|
|
}
|
|
}
|
|
|
|
private byte[] _buffer;
|
|
private int _length;
|
|
|
|
public byte[] Buffer {
|
|
get {
|
|
return _buffer;
|
|
}
|
|
}
|
|
|
|
public int Length {
|
|
get {
|
|
return _length;
|
|
}
|
|
}
|
|
|
|
public int Available {
|
|
get {
|
|
return ( _buffer.Length - _length );
|
|
}
|
|
}
|
|
|
|
public bool IsFull {
|
|
get {
|
|
return ( _length == _buffer.Length );
|
|
}
|
|
}
|
|
|
|
private Gram() {
|
|
}
|
|
|
|
public int Write( byte[] buffer, int offset, int length ) {
|
|
int write = Math.Min( length, this.Available );
|
|
|
|
System.Buffer.BlockCopy( buffer, offset, _buffer, _length, write );
|
|
|
|
_length += write;
|
|
|
|
return write;
|
|
}
|
|
|
|
public void Release() {
|
|
lock ( _pool ) {
|
|
_pool.Push( this );
|
|
ReleaseBuffer( _buffer );
|
|
}
|
|
}
|
|
}
|
|
|
|
private static int m_CoalesceBufferSize = 512;
|
|
private static BufferPool m_UnusedBuffers = new BufferPool( "Coalesced", 2048, m_CoalesceBufferSize );
|
|
|
|
public static int CoalesceBufferSize {
|
|
get {
|
|
return m_CoalesceBufferSize;
|
|
}
|
|
set {
|
|
if ( m_CoalesceBufferSize == value )
|
|
return;
|
|
|
|
if ( m_UnusedBuffers != null )
|
|
m_UnusedBuffers.Free();
|
|
|
|
m_CoalesceBufferSize = value;
|
|
m_UnusedBuffers = new BufferPool( "Coalesced", 2048, m_CoalesceBufferSize );
|
|
}
|
|
}
|
|
|
|
public static byte[] AcquireBuffer() {
|
|
return m_UnusedBuffers.AcquireBuffer();
|
|
}
|
|
|
|
public static void ReleaseBuffer( byte[] buffer ) {
|
|
if ( buffer != null && buffer.Length == m_CoalesceBufferSize ) {
|
|
m_UnusedBuffers.ReleaseBuffer( buffer );
|
|
}
|
|
}
|
|
|
|
private Queue<Gram> _pending;
|
|
|
|
private Gram _buffered;
|
|
|
|
public bool IsFlushReady {
|
|
get {
|
|
return ( _pending.Count == 0 && _buffered != null );
|
|
}
|
|
}
|
|
|
|
public bool IsEmpty {
|
|
get {
|
|
return ( _pending.Count == 0 && _buffered == null );
|
|
}
|
|
}
|
|
|
|
public SendQueue() {
|
|
_pending = new Queue<Gram>();
|
|
}
|
|
|
|
public Gram CheckFlushReady() {
|
|
Gram gram = null;
|
|
|
|
if ( _pending.Count == 0 && _buffered != null ) {
|
|
gram = _buffered;
|
|
|
|
_pending.Enqueue( _buffered );
|
|
_buffered = null;
|
|
}
|
|
|
|
return gram;
|
|
}
|
|
|
|
public Gram Dequeue() {
|
|
Gram gram = null;
|
|
|
|
if ( _pending.Count > 0 ) {
|
|
_pending.Dequeue().Release();
|
|
|
|
if ( _pending.Count > 0 ) {
|
|
gram = _pending.Peek();
|
|
}
|
|
}
|
|
|
|
return gram;
|
|
}
|
|
|
|
private const int PendingCap = 96 * 1024;
|
|
|
|
public Gram Enqueue( byte[] buffer, int length ) {
|
|
return Enqueue( buffer, 0, length );
|
|
}
|
|
|
|
public Gram Enqueue( byte[] buffer, int offset, int length ) {
|
|
if ( buffer == null ) {
|
|
throw new ArgumentNullException( "buffer" );
|
|
} else if ( !(offset >= 0 && offset < buffer.Length) ) {
|
|
throw new ArgumentOutOfRangeException( "offset", offset, "Offset must be greater than or equal to zero and less than the size of the buffer." );
|
|
} else if ( length < 0 || length > buffer.Length ) {
|
|
throw new ArgumentOutOfRangeException( "length", length, "Length cannot be less than zero or greater than the size of the buffer." );
|
|
} else if ( ( buffer.Length - offset ) < length ) {
|
|
throw new ArgumentException( "Offset and length do not point to a valid segment within the buffer." );
|
|
}
|
|
|
|
int existingBytes = ( _pending.Count * m_CoalesceBufferSize ) + ( _buffered == null ? 0 : _buffered.Length );
|
|
|
|
if ( ( existingBytes + length ) > PendingCap ) {
|
|
throw new CapacityExceededException();
|
|
}
|
|
|
|
Gram gram = null;
|
|
|
|
while ( length > 0 ) {
|
|
if ( _buffered == null ) { // nothing yet buffered
|
|
_buffered = Gram.Acquire();
|
|
}
|
|
|
|
int bytesWritten = _buffered.Write( buffer, offset, length );
|
|
|
|
offset += bytesWritten;
|
|
length -= bytesWritten;
|
|
|
|
if ( _buffered.IsFull ) {
|
|
if ( _pending.Count == 0 ) {
|
|
gram = _buffered;
|
|
}
|
|
|
|
_pending.Enqueue( _buffered );
|
|
_buffered = null;
|
|
}
|
|
}
|
|
|
|
return gram;
|
|
}
|
|
|
|
public void Clear() {
|
|
if ( _buffered != null ) {
|
|
_buffered.Release();
|
|
_buffered = null;
|
|
}
|
|
|
|
while ( _pending.Count > 0 ) {
|
|
_pending.Dequeue().Release();
|
|
}
|
|
}
|
|
}
|
|
|
|
public sealed class CapacityExceededException : Exception {
|
|
public CapacityExceededException()
|
|
: base( "Too much data pending." ) {
|
|
}
|
|
}
|
|
} |