// $Id$
//
// Copyright 2007-2008 Cisco Systems Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy
// of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations
// under the License.
using System;
using System.Diagnostics;
using System.IO;
namespace Etch.Util
{
///
/// A FlexBuffer wraps a byte array and manages the active region of
/// it (0..length). It also supports dynamically extending the buffer
/// as needed.
///
/// A FlexBuffer also has an index (read or write cursor). The various
/// Get and put operations always occur at the current index, with the
/// index adjusted appropriately afterward. Get will not move index past
/// length. If put needs to move index past length, length is also
/// adjusted. This may cause the byte array to be re-allocated to a
/// larger size.
///
public sealed class FlexBuffer
{
///
/// Constructs a FlexBuffer with initial length and index of 0.
///
public FlexBuffer()
: this(new byte[INIT_BUFFER_LEN], 0, 0)
{
}
///
/// Constructs the FlexBuffer out of an existing buffer, ready to
/// read, with the index set to 0 and length set to buffer.length.
/// The buffer is adopted, not copied. If you want to copy, use one
/// of the put methods instead.
///
/// Note: this is the same as FlexBuffer( buffer, 0, buffer.length ).
///
/// the buffer to adopt.
///
public FlexBuffer(byte[] buffer)
: this(buffer, 0, buffer.Length)
{
}
///
/// Constructs the FlexBuffer out of an existing buffer, ready to
/// read, with the index set to 0 and length set to buffer.length.
/// The buffer is adopted, not copied. If you want to copy, use one
/// of the put methods instead.
///
/// Note: this is the same as FlexBuffer( buffer, 0, length ).
///
/// the buffer to adopt.
/// the length of the data in the buffer (data presumed
/// to start at 0).
///
public FlexBuffer(byte[] buffer, int length)
: this(buffer, 0, length)
{
}
///
/// Constructs the FlexBuffer out of an existing buffer, ready to
/// read, with the index set to 0 and length set to buffer.length.
/// The buffer is adopted, not copied. If you want to copy, use one
/// of the put methods instead.
///
/// Note: if you want to start off writing to the end of the buffer
/// and not reading it, specify index as the place to start writing,
/// and length as the amount of data that is valid after index ( which
/// might reasonably be 0 )
///
/// the buffer to adopt.
/// /// the index to start reading or writing.
/// the length of the valid data in the buffer,
/// starting at the index.
///
public FlexBuffer(byte[] buffer, int index, int length )
{
if ( buffer == null )
throw new NullReferenceException( "buffer == null" );
if ( index < 0 )
throw new ArgumentException( " index < 0 " );
if ( length < 0 )
throw new ArgumentException( "length < 0 " );
if ( index + length > buffer.Length )
throw new ArgumentException( "index + length > buffer.Length" );
this.buffer = buffer;
this.index = index;
this.length = index + length;
}
///
///
///
/// the current byte array. Might change if any operation
/// needs to extend length past the end of the array.
public byte[] GetBuf()
{
return buffer;
}
///
///
///
///
/// Exception:
/// throws IOException
private void EnsureLength( int len )
{
int n = buffer.Length;
if (len <= n)
return;
// the buffer is not big enough, expand it
int k = n;
if ( k < INIT_BUFFER_LEN )
k = INIT_BUFFER_LEN;
while ( len > k && k < MAX_BUFFER_LEN )
k = k << 1;
if (len > k)
throw new IOException( "buffer overflow" );
byte[] b = new byte[k];
Array.Copy( buffer, 0, b, 0, n );
buffer = b;
}
private byte[] buffer;
private const int INIT_BUFFER_LEN = 32;
private const int TRIM_BUFFER_LEN = 16*1024;
private const int MAX_BUFFER_LEN = 4*1024*1024;
///
///
///
/// the current value of length. This is the last
/// index of valid data in the buffer.
public int Length()
{
return length;
}
///
///
///
/// length the new value of length. Length must be >= 0.
/// If length is less than index, index is also set to length. If length
/// is larger than the current buffer, the buffer is expanded.
/// this flex buffer object.
/// Exception:
/// throws ArgumentOutOfRangeException, IOException
public FlexBuffer SetLength( int length )
{
if (length < 0)
throw new ArgumentOutOfRangeException( "length < 0" );
EnsureLength( length );
this.length = length;
if (index > length)
index = length;
return this;
}
private int length;
///
///
///
/// the current value of index.
public int Index()
{
return index;
}
///
///
///
/// index the new value of index. Index must be >= 0.
/// this flex buffer object.
/// Exception:
/// throws ArgumentOutOfRangeException
public FlexBuffer SetIndex( int index )
{
if (index < 0 || index > length)
throw new ArgumentOutOfRangeException( "index < 0 || index > length" );
this.index = index;
return this;
}
private int index;
///
///
///
/// length() - index(). Result is always >= 0. This is the amount
/// of data that could be read using the various forms of Get. It doesn't
/// really mean anything in relation to put.
public int Avail()
{
return length - index;
}
///
/// Sets both length and index to 0.
///
/// Shorthand for SetLength( 0 ).
///
/// this flex buffer object.
public FlexBuffer Reset()
{
index = 0;
length = 0;
if (buffer.Length > TRIM_BUFFER_LEN)
buffer = new byte[TRIM_BUFFER_LEN];
return this;
}
///
/// Compacts the buffer by moving remaining data (from index to length)
/// to the front of the buffer. Sets index to 0, and sets length to
/// Avail (before index was changed).
///
/// this flex buffer object.
public FlexBuffer Compact()
{
if(index == 0)
return this;
int n = Avail();
if(n == 0)
{
Reset();
return this;
}
Array.Copy(buffer, index, buffer, 0, n);
index = 0;
length = n;
return this;
}
///
///
///
/// the byte value at index, and adjusts index
/// by adding one.
/// Exception:
/// End of File exception / NullReferenceException
public int Get()
{
CheckAvail( 1 );
return buffer[index++] & 255;
}
///
/// Copies data from the byte array to buf as if by repeated
/// calls to Get().
///
/// a buffer to receive the data. At most
/// min( buf.length, Avail() ) bytes are transferred.
/// the amount of data transferred.
/// Exception:
/// End of File exception / NullReferenceException
public int Get( byte[] buf )
{
return Get( buf, 0, buf.Length );
}
///
/// Copies data from the byte array to buf as if by repeated
/// calls to Get().
///
/// a buffer to receive the data. At most
/// min( len, Avail() ) bytes are transferred, starting
/// at off.
/// the index in buf to receive the data.
/// off must be >= 0 && less than or equal to buf.length.
/// the max amount of data to transfer. Len
/// must be >= 0 and less than or equal to buf.length - off.
/// the amount of data transferred.
/// Exception:
/// End of File exception / NullReferenceException
public int Get( byte[] buf, int off, int len )
{
CheckBuf( buf, off, len );
if (len == 0)
return 0;
CheckAvail( 1 );
int n = Math.Min( len, Avail() );
// Array.Copy( buffer, index, buf, off, n );
Buffer.BlockCopy(buffer, index, buf, off, n);
index += n;
return n;
}
public sbyte GetByte()
{
CheckAvail( 1 );
return (sbyte)buffer[ index++ ];
}
public readonly static bool littleEndian = false;
///
///
///
/// a short composed from the next 2 bytes. Little-endian.
/// Exception:
/// throws Exception if avail() < 2
public short GetShort()
{
CheckAvail( 2 );
if ( littleEndian )
{
// little-endian
int value = buffer[ index++ ] & 255;
return ( short ) ( value + ( ( buffer[ index++ ] & 255 ) << 8 ) );
}
else
{
// big-endian
int value = buffer[ index++ ];
return ( short ) ( ( value << 8 ) + ( buffer[ index++ ] & 255 ) );
}
}
///
///
///
/// an integer composed of 4 bytes from
/// the buffer, with the first byte being the least
/// significant and the last being the most significant.
/// Exception:
/// End of File exception / NullReferenceException
public int GetInt()
{
CheckAvail( 4 );
if ( littleEndian )
{
// little-endian
int value = buffer[ index++ ] & 255;
value += ( ( buffer[ index++ ] & 255 ) << 8 );
value += ( ( buffer[ index++ ] & 255 ) << 16 );
return value + ( ( buffer[ index++ ] & 255 ) << 24 );
}
else
{
// big-endian
int value = buffer[ index++ ];
value = ( value << 8 ) + ( buffer[ index++ ] & 255 );
value = ( value << 8 ) + ( buffer[ index++ ] & 255 );
return ( value << 8 ) + ( buffer[ index++ ] & 255 );
}
}
///
///
///
/// a long comprised of the next 8 bytes. Little-endian
public long GetLong()
{
CheckAvail( 8 );
if ( littleEndian )
{
// little-endian
long value = buffer[ index++ ] & 255;
value += ( ( ( long ) ( buffer[ index++ ] & 255 ) ) << 8 );
value += ( ( ( long ) ( buffer[ index++ ] & 255 ) ) << 16 );
value += ( ( ( long ) ( buffer[ index++ ] & 255 ) ) << 24 );
value += ( ( ( long ) ( buffer[ index++ ] & 255 ) ) << 32 );
value += ( ( ( long ) ( buffer[ index++ ] & 255 ) ) << 40 );
value += ( ( ( long ) ( buffer[ index++ ] & 255 ) ) << 48 );
return value + ( ( ( long ) ( buffer[ index++ ] & 255 ) ) << 56 );
}
else
{
// big-endian
long value = buffer[ index++ ];
value = ( value << 8 ) + ( buffer[ index++ ] & 255 );
value = ( value << 8 ) + ( buffer[ index++ ] & 255 );
value = ( value << 8 ) + ( buffer[ index++ ] & 255 );
value = ( value << 8 ) + ( buffer[ index++ ] & 255 );
value = ( value << 8 ) + ( buffer[ index++ ] & 255 );
value = ( value << 8 ) + ( buffer[ index++ ] & 255 );
return ( value << 8 ) + ( buffer[ index++ ] & 255 );
}
}
///
///
///
/// a float from the next available bytes
public float GetFloat()
{
return BitConverter.ToSingle( BitConverter.GetBytes( GetInt() ), 0 );
}
///
///
///
/// a double from the next available bytes
public double GetDouble()
{
return BitConverter.Int64BitsToDouble( GetLong() );
}
public void GetFully( byte[] b )
{
CheckAvail( b.Length );
int n = Get( b, 0, b.Length );
Debug.Assert( n == b.Length );
}
///
/// Puts a byte into the buffer at the current index,
/// then adjusts the index by one. Adjusts the length
/// as necessary. The buffer is expanded if needed.
///
/// byte to be put
/// this flex buffer object.
/// Exception:
/// IOException if the buffer overflows its max length.
public FlexBuffer Put( int b )
{
EnsureLength( index+1 );
buffer[index++] = (byte) b;
FixLength();
return this;
}
///
/// Puts some bytes into the buffer as if by repeated
/// calls to put().
///
/// the source of the bytes to put. The entire
/// array of bytes is put.
/// flex buffer object.
/// Exception:
/// IOException if the buffer overflows its max length.
public FlexBuffer Put( byte[] buf )
{
return Put( buf, 0, buf.Length );
}
///
/// Puts some bytes into the buffer as if by repeated
/// calls to Put().
///
/// the source of the bytes to put.
/// the index to the first byte to put.
/// the number of bytes to put.
/// flex buffer object.
/// Exception:
/// IOException if the buffer overflows its max length.
public FlexBuffer Put( byte[] buf, int off, int len )
{
CheckBuf( buf, off, len );
if (len == 0)
return this;
EnsureLength( index+len );
// Array.Copy( buf, off, buffer, index, len );
Buffer.BlockCopy(buf, off, buffer, index, len);
index += len;
FixLength();
return this;
}
///
/// Copies the Available bytes from buf into buffer as if by
/// repeated execution of put( buf.Get() ).
///
/// the source of the bytes to put. All Available
/// bytes are copied.
/// flex buffer object.
/// Exception:
/// IOException if the buffer overflows its max length.
public FlexBuffer Put( FlexBuffer buf )
{
int n = buf.Avail();
Put( buf.buffer, buf.index, n );
buf.Skip( n, false );
return this;
}
///
/// Copies the specified number of bytes from buf into buffer
/// as if by repeated execution of Put( buf.Get() ).
///
/// the source of the bytes to put.
/// len the number of bytes to put. Len must be
/// less than or equal to buf.Avail().
/// flex buffer object.
/// Exception:
/// IOException if the buffer overflows its max length.
public FlexBuffer Put( FlexBuffer buf, int len )
{
if (len > buf.Avail())
throw new ArgumentOutOfRangeException( "len > buf.Avail()" );
Put( buf.buffer, buf.index, len );
buf.Skip( len, false );
return this;
}
public void PutByte( byte value )
{
EnsureLength( index + 1 );
buffer[ index++ ] = ( byte ) value;
FixLength();
}
public void PutByte( sbyte value )
{
EnsureLength( index + 1 );
buffer[ index++ ] = ( byte ) value;
FixLength();
}
///
/// Put short as the next 2 bytes. Little-endian
///
///
public void PutShort( short value )
{
EnsureLength( index + 2 );
if ( littleEndian )
{
buffer[ index++ ] = ( byte ) value;
buffer[ index++ ] = ( byte ) ( value >> 8 );
}
else
{
/// In C#, since we're using the byte (which is unsigned),
/// it doesn't matter whether you do logical or arithmetic
/// shift. Hence, an equivalent of the Java >>> operator is
/// not required here.
buffer[ index++ ] = ( byte ) ( value >> 8 );
buffer[ index++ ] = ( byte ) value;
}
FixLength();
}
public void PutInt( int value )
{
EnsureLength( length + 4 );
if ( littleEndian )
{
buffer[ index++ ] = ( byte ) value;
buffer[ index++ ] = ( byte ) ( value >> 8 );
buffer[ index++ ] = ( byte ) ( value >> 16 );
buffer[ index++ ] = ( byte ) ( value >> 24 );
}
else
{
buffer[ index++ ] = ( byte ) ( value >> 24 );
buffer[ index++ ] = ( byte ) ( value >> 16 );
buffer[ index++ ] = ( byte ) ( value >> 8 );
buffer[ index++ ] = ( byte ) value;
}
FixLength();
}
public void PutLong( long value )
{
EnsureLength( index+8 );
if ( littleEndian )
{
buffer[ index++ ] = ( byte ) value;
buffer[ index++ ] = ( byte ) ( value >> 8 );
buffer[ index++ ] = ( byte ) ( value >> 16 );
buffer[ index++ ] = ( byte ) ( value >> 24 );
buffer[ index++ ] = ( byte ) ( value >> 32 );
buffer[ index++ ] = ( byte ) ( value >> 40 );
buffer[ index++ ] = ( byte ) ( value >> 48 );
buffer[ index++ ] = ( byte ) ( value >> 56 );
}
else
{
buffer[ index++ ] = ( byte ) ( value >> 56 );
buffer[ index++ ] = ( byte ) ( value >> 48 );
buffer[ index++ ] = ( byte ) ( value >> 40 );
buffer[ index++ ] = ( byte ) ( value >> 32 );
buffer[ index++ ] = ( byte ) ( value >> 24 );
buffer[ index++ ] = ( byte ) ( value >> 16 );
buffer[ index++ ] = ( byte ) ( value >> 8 );
buffer[ index++ ] = ( byte ) value;
}
FixLength();
}
public void PutFloat( float value )
{
PutInt( BitConverter.ToInt32( BitConverter.GetBytes( value ), 0 ) );
}
public void PutDouble( double value )
{
PutLong( BitConverter.DoubleToInt64Bits( value ) );
}
///
/// Adjusts index as if by a Get or put but without transferring
/// any data. This could be used to skip over a data item in an
/// input or output buffer.
///
/// len the number of bytes to skip over. Len must be
/// greater than or equal to 0. If put is false, it is an error if len >Avail().
/// If put is true, the buffer
/// may be extended (and the buffer length adjusted).
/// put if true it is ok to extend the length of the
/// buffer.
/// this Flexbuffer object.
/// Exception:
/// IOException
public FlexBuffer Skip( int len, bool put )
{
if (len < 0)
throw new ArgumentException( "count < 0" );
if (len == 0)
return this;
if (put)
{
EnsureLength( index+len );
index += len;
FixLength();
return this;
}
CheckAvail( len );
index += len;
return this;
}
///
/// If index has moved past length during a put, then adjust length
/// to track index.
///
private void FixLength()
{
if(index > length)
length = index;
}
private void CheckBuf(byte[] buf, int off, int len)
{
if(buf == null)
throw new NullReferenceException("buf == null");
if(off < 0 || off > buf.Length)
throw new ArgumentOutOfRangeException("off < 0 || off > buf.length");
if(len < 0)
throw new ArgumentOutOfRangeException("len < 0");
if(off+len > buf.Length)
throw new ArgumentOutOfRangeException("off+len > buf.length");
}
///
/// Return the currently Available bytes.
///
/// the currently Available bytes.
/// Exception:
/// throws an IO Exception
public byte[] GetAvailBytes()
{
byte[] buf = new byte[Avail()];
Get( buf );
return buf;
}
///
/// Checks that there are enough bytes to for a read.
///
/// the length required by a read operation.
private void CheckAvail( int len )
{
if ( len > Avail() )
throw new EndOfStreamException( " len > Avail() " );
}
}
}