/*************************************************************************** * 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 _pool = new Stack(); 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 _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(); } 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." ) { } } }