1*aaf4ece6Schristos // 2*aaf4ece6Schristos // � Copyright Henrik Ravn 2004 3*aaf4ece6Schristos // 4*aaf4ece6Schristos // Use, modification and distribution are subject to the Boost Software License, Version 1.0. 5*aaf4ece6Schristos // (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) 6*aaf4ece6Schristos // 7*aaf4ece6Schristos 8*aaf4ece6Schristos using System; 9*aaf4ece6Schristos using System.Diagnostics; 10*aaf4ece6Schristos using System.Runtime.InteropServices; 11*aaf4ece6Schristos 12*aaf4ece6Schristos namespace DotZLib 13*aaf4ece6Schristos { 14*aaf4ece6Schristos 15*aaf4ece6Schristos /// <summary> 16*aaf4ece6Schristos /// Implements a data compressor, using the deflate algorithm in the ZLib dll 17*aaf4ece6Schristos /// </summary> 18*aaf4ece6Schristos public sealed class Deflater : CodecBase 19*aaf4ece6Schristos { 20*aaf4ece6Schristos #region Dll imports 21*aaf4ece6Schristos [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl, CharSet=CharSet.Ansi)] deflateInit_(ref ZStream sz, int level, string vs, int size)22*aaf4ece6Schristos private static extern int deflateInit_(ref ZStream sz, int level, string vs, int size); 23*aaf4ece6Schristos 24*aaf4ece6Schristos [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)] deflate(ref ZStream sz, int flush)25*aaf4ece6Schristos private static extern int deflate(ref ZStream sz, int flush); 26*aaf4ece6Schristos 27*aaf4ece6Schristos [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)] deflateReset(ref ZStream sz)28*aaf4ece6Schristos private static extern int deflateReset(ref ZStream sz); 29*aaf4ece6Schristos 30*aaf4ece6Schristos [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)] deflateEnd(ref ZStream sz)31*aaf4ece6Schristos private static extern int deflateEnd(ref ZStream sz); 32*aaf4ece6Schristos #endregion 33*aaf4ece6Schristos 34*aaf4ece6Schristos /// <summary> 35*aaf4ece6Schristos /// Constructs an new instance of the <c>Deflater</c> 36*aaf4ece6Schristos /// </summary> 37*aaf4ece6Schristos /// <param name="level">The compression level to use for this <c>Deflater</c></param> Deflater(CompressLevel level)38*aaf4ece6Schristos public Deflater(CompressLevel level) : base() 39*aaf4ece6Schristos { 40*aaf4ece6Schristos int retval = deflateInit_(ref _ztream, (int)level, Info.Version, Marshal.SizeOf(_ztream)); 41*aaf4ece6Schristos if (retval != 0) 42*aaf4ece6Schristos throw new ZLibException(retval, "Could not initialize deflater"); 43*aaf4ece6Schristos 44*aaf4ece6Schristos resetOutput(); 45*aaf4ece6Schristos } 46*aaf4ece6Schristos 47*aaf4ece6Schristos /// <summary> 48*aaf4ece6Schristos /// Adds more data to the codec to be processed. 49*aaf4ece6Schristos /// </summary> 50*aaf4ece6Schristos /// <param name="data">Byte array containing the data to be added to the codec</param> 51*aaf4ece6Schristos /// <param name="offset">The index of the first byte to add from <c>data</c></param> 52*aaf4ece6Schristos /// <param name="count">The number of bytes to add</param> 53*aaf4ece6Schristos /// <remarks>Adding data may, or may not, raise the <c>DataAvailable</c> event</remarks> Add(byte[] data, int offset, int count)54*aaf4ece6Schristos public override void Add(byte[] data, int offset, int count) 55*aaf4ece6Schristos { 56*aaf4ece6Schristos if (data == null) throw new ArgumentNullException(); 57*aaf4ece6Schristos if (offset < 0 || count < 0) throw new ArgumentOutOfRangeException(); 58*aaf4ece6Schristos if ((offset+count) > data.Length) throw new ArgumentException(); 59*aaf4ece6Schristos 60*aaf4ece6Schristos int total = count; 61*aaf4ece6Schristos int inputIndex = offset; 62*aaf4ece6Schristos int err = 0; 63*aaf4ece6Schristos 64*aaf4ece6Schristos while (err >= 0 && inputIndex < total) 65*aaf4ece6Schristos { 66*aaf4ece6Schristos copyInput(data, inputIndex, Math.Min(total - inputIndex, kBufferSize)); 67*aaf4ece6Schristos while (err >= 0 && _ztream.avail_in > 0) 68*aaf4ece6Schristos { 69*aaf4ece6Schristos err = deflate(ref _ztream, (int)FlushTypes.None); 70*aaf4ece6Schristos if (err == 0) 71*aaf4ece6Schristos while (_ztream.avail_out == 0) 72*aaf4ece6Schristos { 73*aaf4ece6Schristos OnDataAvailable(); 74*aaf4ece6Schristos err = deflate(ref _ztream, (int)FlushTypes.None); 75*aaf4ece6Schristos } 76*aaf4ece6Schristos inputIndex += (int)_ztream.total_in; 77*aaf4ece6Schristos } 78*aaf4ece6Schristos } 79*aaf4ece6Schristos setChecksum( _ztream.adler ); 80*aaf4ece6Schristos } 81*aaf4ece6Schristos 82*aaf4ece6Schristos 83*aaf4ece6Schristos /// <summary> 84*aaf4ece6Schristos /// Finishes up any pending data that needs to be processed and handled. 85*aaf4ece6Schristos /// </summary> Finish()86*aaf4ece6Schristos public override void Finish() 87*aaf4ece6Schristos { 88*aaf4ece6Schristos int err; 89*aaf4ece6Schristos do 90*aaf4ece6Schristos { 91*aaf4ece6Schristos err = deflate(ref _ztream, (int)FlushTypes.Finish); 92*aaf4ece6Schristos OnDataAvailable(); 93*aaf4ece6Schristos } 94*aaf4ece6Schristos while (err == 0); 95*aaf4ece6Schristos setChecksum( _ztream.adler ); 96*aaf4ece6Schristos deflateReset(ref _ztream); 97*aaf4ece6Schristos resetOutput(); 98*aaf4ece6Schristos } 99*aaf4ece6Schristos 100*aaf4ece6Schristos /// <summary> 101*aaf4ece6Schristos /// Closes the internal zlib deflate stream 102*aaf4ece6Schristos /// </summary> CleanUp()103*aaf4ece6Schristos protected override void CleanUp() { deflateEnd(ref _ztream); } 104*aaf4ece6Schristos 105*aaf4ece6Schristos } 106*aaf4ece6Schristos } 107