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