// Copyright (C) 2004-2007 MySQL AB // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License version 2 as published by // the Free Software Foundation // // There are special exceptions to the terms and conditions of the GPL // as it is applied to this software. View the full text of the // exception in file EXCEPTIONS in the directory of this software // distribution. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA using System; using System.IO; using zlib; using MySql.Data.MySqlClient.Properties; using MySql.Data.Common; namespace MySql.Data.MySqlClient { /// /// Summary description for CompressedStream. /// internal class CompressedStream : Stream { // writing fields private Stream baseStream; private MemoryStream cache; // reading fields private byte[] localByte; private byte[] inBuffer; private byte[] lengthBytes; private WeakReference inBufferRef; private int inPos; private int maxInPos; private ZInputStream zInStream; public CompressedStream(Stream baseStream) { this.baseStream = baseStream; localByte = new byte[1]; lengthBytes = new byte[7]; cache = new MemoryStream(); inBufferRef = new WeakReference(inBuffer, false); } #region Properties public override bool CanRead { get { return baseStream.CanRead; } } public override bool CanWrite { get { return baseStream.CanWrite; } } public override bool CanSeek { get { return baseStream.CanSeek; } } public override long Length { get { return baseStream.Length; } } public override long Position { get { return baseStream.Position; } set { baseStream.Position = value; } } #endregion public override void Close() { baseStream.Close(); base.Close(); } public override void SetLength(long value) { throw new NotSupportedException(Resources.CSNoSetLength); } public override int ReadByte() { try { Read(localByte, 0, 1); return localByte[0]; } catch (EndOfStreamException) { return -1; } } public override bool CanTimeout { get { return baseStream.CanTimeout; } } public override int ReadTimeout { get { return baseStream.ReadTimeout; } set { baseStream.ReadTimeout = value; } } public override int WriteTimeout { get { return baseStream.WriteTimeout; } set { baseStream.WriteTimeout = value; } } public override int Read(byte[] buffer, int offset, int count) { if (buffer == null) throw new ArgumentNullException("buffer", Resources.BufferCannotBeNull); if (offset < 0 || offset >= buffer.Length) throw new ArgumentOutOfRangeException("offset", Resources.OffsetMustBeValid); if ((offset + count) > buffer.Length) throw new ArgumentException(Resources.BufferNotLargeEnough, "buffer"); if (inPos == maxInPos) PrepareNextPacket(); int countToRead = Math.Min(count, maxInPos - inPos); int countRead; if (zInStream != null) countRead = zInStream.read(buffer, offset, countToRead); else countRead = baseStream.Read(buffer, offset, countToRead); inPos += countRead; // release the weak reference if (inPos == maxInPos) { zInStream = null; if (!Platform.IsMono()) { inBufferRef = new WeakReference(inBuffer, false); inBuffer = null; } } return countRead; } private void PrepareNextPacket() { MySqlStream.ReadFully(baseStream, lengthBytes, 0, 7); int compressedLength = lengthBytes[0] + (lengthBytes[1] << 8) + (lengthBytes[2] << 16); // lengthBytes[3] is seq int unCompressedLength = lengthBytes[4] + (lengthBytes[5] << 8) + (lengthBytes[6] << 16); if (unCompressedLength == 0) { unCompressedLength = compressedLength; zInStream = null; } else { ReadNextPacket(compressedLength); MemoryStream ms = new MemoryStream(inBuffer); zInStream = new ZInputStream(ms); zInStream.maxInput = compressedLength; } inPos = 0; maxInPos = unCompressedLength; } private void ReadNextPacket(int len) { if (!Platform.IsMono()) inBuffer = inBufferRef.Target as byte[]; if (inBuffer == null || inBuffer.Length < len) inBuffer = new byte[len]; MySqlStream.ReadFully(baseStream, inBuffer, 0, len); } private MemoryStream CompressCache() { // small arrays almost never yeild a benefit from compressing if (cache.Length < 50) return null; byte[] cacheBytes = cache.GetBuffer(); MemoryStream compressedBuffer = new MemoryStream(); ZOutputStream zos = new ZOutputStream(compressedBuffer, zlibConst.Z_DEFAULT_COMPRESSION); zos.Write(cacheBytes, 0, (int) cache.Length); zos.finish(); // if the compression hasn't helped, then just return null if (compressedBuffer.Length >= cache.Length) return null; return compressedBuffer; } private void CompressAndSendCache() { long compressedLength, uncompressedLength; // we need to save the sequence byte that is written byte[] cacheBuffer = cache.GetBuffer(); byte seq = cacheBuffer[3]; cacheBuffer[3] = 0; // first we compress our current cache MemoryStream compressedBuffer = CompressCache(); // now we set our compressed and uncompressed lengths // based on if our compression is going to help or not if (compressedBuffer == null) { compressedLength = cache.Length; uncompressedLength = 0; } else { compressedLength = compressedBuffer.Length; uncompressedLength = cache.Length; } baseStream.WriteByte((byte) (compressedLength & 0xff)); baseStream.WriteByte((byte) ((compressedLength >> 8) & 0xff)); baseStream.WriteByte((byte) ((compressedLength >> 16) & 0Xff)); baseStream.WriteByte(seq); baseStream.WriteByte((byte) (uncompressedLength & 0xff)); baseStream.WriteByte((byte) ((uncompressedLength >> 8) & 0xff)); baseStream.WriteByte((byte) ((uncompressedLength >> 16) & 0Xff)); if (compressedBuffer == null) baseStream.Write(cacheBuffer, 0, (int) cache.Length); else { byte[] compressedBytes = compressedBuffer.GetBuffer(); baseStream.Write(compressedBytes, 0, (int) compressedBuffer.Length); } baseStream.Flush(); cache.SetLength(0); } public override void Flush() { if (!InputDone()) return; CompressAndSendCache(); } private bool InputDone() { // if we have not done so yet, see if we can calculate how many bytes we are expecting if (cache.Length < 4) return false; byte[] buf = cache.GetBuffer(); int expectedLen = buf[0] + (buf[1] << 8) + (buf[2] << 16); if (cache.Length < (expectedLen + 4)) return false; return true; } public override void WriteByte(byte value) { cache.WriteByte(value); } public override void Write(byte[] buffer, int offset, int count) { cache.Write(buffer, offset, count); } public override long Seek(long offset, SeekOrigin origin) { return baseStream.Seek(offset, origin); } } }