| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210 |
- //
- // System.IO.BufferedStream
- //
- // Author:
- // Matt Kimball ([email protected])
- // Ville Palo <[email protected]>
- //
- using System.Runtime.InteropServices;
- namespace System.IO {
- public sealed class BufferedStream : Stream {
- Stream m_stream;
- byte[] m_buffer;
- int m_buffer_pos;
- int m_buffer_read_ahead;
- bool m_buffer_reading;
- private bool disposed = false;
- public BufferedStream(Stream stream) : this(stream, 4096) {
- }
- public BufferedStream(Stream stream, int buffer_size) {
- if (stream == null)
- throw new ArgumentNullException ("stream was null");
- if (buffer_size < 0)
- throw new ArgumentOutOfRangeException ();
- if (!stream.CanRead && !stream.CanWrite)
- throw new ObjectDisposedException ("Cannot access a closed Stream.");
- m_stream = stream;
- m_buffer = new byte[buffer_size];
- }
- public override bool CanRead {
- get {
- return m_stream.CanRead;
- }
- }
- public override bool CanWrite {
- get {
- return m_stream.CanWrite;
- }
- }
- public override bool CanSeek {
- get {
- return m_stream.CanSeek;
- }
- }
- public override long Length {
- get {
- Flush ();
- return m_stream.Length;
- }
- }
-
- public override long Position {
- get {
- CheckObjectDisposedException ();
- return m_stream.Position - m_buffer_read_ahead + m_buffer_pos;
- }
- set {
- Flush();
- m_stream.Position = value;
- }
- }
- public override void Close() {
-
- if (m_buffer != null)
- Flush();
- m_stream.Close();
- m_buffer = null;
- disposed = true;
- }
- public override void Flush() {
-
- CheckObjectDisposedException ();
- if (m_buffer_reading) {
- if (CanSeek)
- m_stream.Position = Position;
- } else if (m_buffer_pos > 0) {
- m_stream.Write(m_buffer, 0, m_buffer_pos);
- }
- m_buffer_read_ahead = 0;
- m_buffer_pos = 0;
- }
- public override long Seek(long offset, SeekOrigin origin) {
- Flush();
- return m_stream.Seek(offset, origin);
- }
- public override void SetLength(long value) {
- m_stream.SetLength(value);
- }
- public override int ReadByte() {
- CheckObjectDisposedException ();
-
- byte[] b = new byte[1];
- if (Read(b, 0, 1) == 1) {
- return b[0];
- } else {
- return -1;
- }
- }
- public override void WriteByte(byte value) {
- CheckObjectDisposedException ();
- byte[] b = new byte[1];
- b[0] = value;
- Write(b, 0, 1);
- }
- public override int Read([In,Out] byte[] array, int offset, int count) {
- CheckObjectDisposedException ();
- if (array.Length < offset + count)
- throw new ArgumentException ();
- if (offset < 0)
- throw new ArgumentOutOfRangeException ("Offset was negative value.");
- if (!m_buffer_reading) {
- Flush();
- m_buffer_reading = true;
- }
- if (count <= m_buffer_read_ahead - m_buffer_pos) {
- Array.Copy(m_buffer, m_buffer_pos, array, offset, count);
- m_buffer_pos += count;
- if (m_buffer_pos == m_buffer_read_ahead) {
- m_buffer_pos = 0;
- m_buffer_read_ahead = 0;
- }
- return count;
- }
- int ret = m_buffer_read_ahead - m_buffer_pos;
- Array.Copy(m_buffer, m_buffer_pos, array, offset, ret);
- m_buffer_pos = 0;
- m_buffer_read_ahead = 0;
- offset += ret;
- count -= ret;
- if (count >= m_buffer.Length) {
- ret += m_stream.Read(array, offset, count);
- } else {
- m_buffer_read_ahead = m_stream.Read(m_buffer, 0, m_buffer.Length);
-
- if (count < m_buffer_read_ahead) {
- Array.Copy(m_buffer, 0, array, offset, count);
- m_buffer_pos = count;
- ret += count;
- } else {
- Array.Copy(m_buffer, 0, array, offset, m_buffer_read_ahead);
- ret += m_buffer_read_ahead;
- m_buffer_read_ahead = 0;
- }
- }
- return ret;
- }
- public override void Write(byte[] array, int offset, int count) {
- CheckObjectDisposedException ();
- if (!m_stream.CanWrite)
- throw new NotSupportedException ();
- if (offset < 0)
- throw new ArgumentOutOfRangeException ();
- if (m_buffer_reading) {
- Flush();
- m_buffer_reading = false;
- }
- if (m_buffer_pos + count >= m_buffer.Length) {
- Flush();
- m_stream.Write(array, offset, count);
- } else {
- Array.Copy(array, offset, m_buffer, m_buffer_pos, count);
- m_buffer_pos += count;
- }
- }
- private void CheckObjectDisposedException ()
- {
- if (disposed)
- throw new ObjectDisposedException ("BufferedStream", "Stream is closed");
- }
- }
- }
|