a575963da9
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
275 lines
5.8 KiB
C#
275 lines
5.8 KiB
C#
// Transport Security Layer (TLS)
|
|
// Copyright (c) 2003-2004 Carlos Guzman Alvarez
|
|
// Copyright (C) 2006 Novell, Inc (http://www.novell.com)
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
|
|
using System;
|
|
using System.IO;
|
|
|
|
namespace Mono.Security.Protocol.Tls
|
|
{
|
|
internal class TlsStream : Stream
|
|
{
|
|
#region Fields
|
|
|
|
private bool canRead;
|
|
private bool canWrite;
|
|
private MemoryStream buffer;
|
|
private byte[] temp;
|
|
private const int temp_size = 4;
|
|
|
|
#endregion
|
|
|
|
#region Properties
|
|
|
|
public bool EOF
|
|
{
|
|
get
|
|
{
|
|
if (this.Position < this.Length)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Stream Properties
|
|
|
|
public override bool CanWrite
|
|
{
|
|
get { return this.canWrite; }
|
|
}
|
|
|
|
public override bool CanRead
|
|
{
|
|
get { return this.canRead; }
|
|
}
|
|
|
|
public override bool CanSeek
|
|
{
|
|
get { return this.buffer.CanSeek; }
|
|
}
|
|
|
|
public override long Position
|
|
{
|
|
get { return this.buffer.Position; }
|
|
set { this.buffer.Position = value; }
|
|
}
|
|
|
|
public override long Length
|
|
{
|
|
get { return this.buffer.Length; }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Constructors
|
|
|
|
public TlsStream() : base()
|
|
{
|
|
this.buffer = new MemoryStream(0);
|
|
this.canRead = false;
|
|
this.canWrite = true;
|
|
}
|
|
|
|
public TlsStream(byte[] data) : base()
|
|
{
|
|
if (data != null)
|
|
this.buffer = new MemoryStream(data);
|
|
else
|
|
this.buffer = new MemoryStream ();
|
|
this.canRead = true;
|
|
this.canWrite = false;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Specific Read Methods
|
|
|
|
// hack for reducing memory allocations
|
|
// returned value is valid only for the length asked *and*
|
|
// cannot be directly returned outside the class
|
|
// note: Mono's Stream.ReadByte does a 1 byte array allocation
|
|
private byte[] ReadSmallValue (int length)
|
|
{
|
|
if (length > temp_size)
|
|
throw new ArgumentException ("8 bytes maximum");
|
|
if (temp == null)
|
|
temp = new byte[temp_size];
|
|
|
|
if (this.Read (temp, 0, length) != length)
|
|
throw new TlsException (String.Format ("buffer underrun"));
|
|
return temp;
|
|
}
|
|
|
|
public new byte ReadByte()
|
|
{
|
|
byte[] result = ReadSmallValue (1);
|
|
return result [0];
|
|
}
|
|
|
|
public short ReadInt16()
|
|
{
|
|
byte[] result = ReadSmallValue (2);
|
|
return (short) (result[0] << 8 | result[1]);
|
|
}
|
|
|
|
public int ReadInt24()
|
|
{
|
|
byte[] result = ReadSmallValue (3);
|
|
return ((result[0] << 16) | (result[1] << 8) | result[2]);
|
|
}
|
|
|
|
public int ReadInt32()
|
|
{
|
|
byte[] result = ReadSmallValue (4);
|
|
return ((result[0] << 24) | (result[1] << 16) | (result[2] << 8) | result[3]);
|
|
}
|
|
|
|
public byte[] ReadBytes(int count)
|
|
{
|
|
byte[] bytes = new byte[count];
|
|
if (this.Read(bytes, 0, count) != count)
|
|
throw new TlsException ("buffer underrun");
|
|
|
|
return bytes;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Specific Write Methods
|
|
|
|
// note: Mono's Stream.WriteByte does a 1 byte array allocation
|
|
public void Write(byte value)
|
|
{
|
|
if (temp == null)
|
|
temp = new byte[temp_size];
|
|
temp[0] = value;
|
|
this.Write (temp, 0, 1);
|
|
}
|
|
|
|
public void Write(short value)
|
|
{
|
|
if (temp == null)
|
|
temp = new byte[temp_size];
|
|
temp[0] = ((byte)(value >> 8));
|
|
temp[1] = ((byte)value);
|
|
this.Write (temp, 0, 2);
|
|
}
|
|
|
|
public void WriteInt24(int value)
|
|
{
|
|
if (temp == null)
|
|
temp = new byte[temp_size];
|
|
temp[0] = ((byte)(value >> 16));
|
|
temp[1] = ((byte)(value >> 8));
|
|
temp[2] = ((byte)value);
|
|
this.Write (temp, 0, 3);
|
|
}
|
|
|
|
public void Write(int value)
|
|
{
|
|
if (temp == null)
|
|
temp = new byte[temp_size];
|
|
temp[0] = ((byte)(value >> 24));
|
|
temp[1] = ((byte)(value >> 16));
|
|
temp[2] = ((byte)(value >> 8));
|
|
temp[3] = ((byte)value);
|
|
this.Write (temp, 0, 4);
|
|
}
|
|
|
|
public void Write(ulong value)
|
|
{
|
|
Write ((int)(value >> 32));
|
|
Write ((int)value);
|
|
}
|
|
|
|
public void Write(byte[] buffer)
|
|
{
|
|
this.Write(buffer, 0, buffer.Length);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Methods
|
|
|
|
public void Reset()
|
|
{
|
|
this.buffer.SetLength(0);
|
|
this.buffer.Position = 0;
|
|
}
|
|
|
|
public byte[] ToArray()
|
|
{
|
|
return this.buffer.ToArray();
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Stream Methods
|
|
|
|
public override void Flush()
|
|
{
|
|
this.buffer.Flush();
|
|
}
|
|
|
|
public override void SetLength(long length)
|
|
{
|
|
this.buffer.SetLength(length);
|
|
}
|
|
|
|
public override long Seek(long offset, System.IO.SeekOrigin loc)
|
|
{
|
|
return this.buffer.Seek(offset, loc);
|
|
}
|
|
|
|
public override int Read(byte[] buffer, int offset, int count)
|
|
{
|
|
if (this.canRead)
|
|
{
|
|
return this.buffer.Read(buffer, offset, count);
|
|
}
|
|
throw new InvalidOperationException("Read operations are not allowed by this stream");
|
|
}
|
|
|
|
public override void Write(byte[] buffer, int offset, int count)
|
|
{
|
|
if (this.canWrite)
|
|
{
|
|
this.buffer.Write(buffer, offset, count);
|
|
}
|
|
else
|
|
{
|
|
throw new InvalidOperationException("Write operations are not allowed by this stream");
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|