| // CRC32.cs | 
 | // ------------------------------------------------------------------ | 
 | // | 
 | // Copyright (c) 2011 Dino Chiesa. | 
 | // All rights reserved. | 
 | // | 
 | // This code module is part of DotNetZip, a zipfile class library. | 
 | // | 
 | // ------------------------------------------------------------------ | 
 | // | 
 | // This code is licensed under the Microsoft Public License. | 
 | // See the file License.txt for the license details. | 
 | // More info on: http://dotnetzip.codeplex.com | 
 | // | 
 | // ------------------------------------------------------------------ | 
 | // | 
 | // Last Saved: <2011-August-02 18:25:54> | 
 | // | 
 | // ------------------------------------------------------------------ | 
 | // | 
 | // This module defines the CRC32 class, which can do the CRC32 algorithm, using | 
 | // arbitrary starting polynomials, and bit reversal. The bit reversal is what | 
 | // distinguishes this CRC-32 used in BZip2 from the CRC-32 that is used in PKZIP | 
 | // files, or GZIP files. This class does both. | 
 | // | 
 | // ------------------------------------------------------------------ | 
 |  | 
 |  | 
 | using System; | 
 | using Interop = System.Runtime.InteropServices; | 
 |  | 
 | namespace OfficeOpenXml.Packaging.Ionic.Crc | 
 | { | 
 |     /// <summary> | 
 |     ///   Computes a CRC-32. The CRC-32 algorithm is parameterized - you | 
 |     ///   can set the polynomial and enable or disable bit | 
 |     ///   reversal. This can be used for GZIP, BZip2, or ZIP. | 
 |     /// </summary> | 
 |     /// <remarks> | 
 |     ///   This type is used internally by DotNetZip; it is generally not used | 
 |     ///   directly by applications wishing to create, read, or manipulate zip | 
 |     ///   archive files. | 
 |     /// </remarks> | 
 |  | 
 |     [Interop.GuidAttribute("ebc25cf6-9120-4283-b972-0e5520d0000C")] | 
 |     [Interop.ComVisible(true)] | 
 | #if !NETCF | 
 |     [Interop.ClassInterface(Interop.ClassInterfaceType.AutoDispatch)] | 
 | #endif | 
 |     internal class CRC32 | 
 |     { | 
 |         /// <summary> | 
 |         ///   Indicates the total number of bytes applied to the CRC. | 
 |         /// </summary> | 
 |         public Int64 TotalBytesRead | 
 |         { | 
 |             get | 
 |             { | 
 |                 return _TotalBytesRead; | 
 |             } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Indicates the current CRC for all blocks slurped in. | 
 |         /// </summary> | 
 |         public Int32 Crc32Result | 
 |         { | 
 |             get | 
 |             { | 
 |                 return unchecked((Int32)(~_register)); | 
 |             } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Returns the CRC32 for the specified stream. | 
 |         /// </summary> | 
 |         /// <param name="input">The stream over which to calculate the CRC32</param> | 
 |         /// <returns>the CRC32 calculation</returns> | 
 |         public Int32 GetCrc32(System.IO.Stream input) | 
 |         { | 
 |             return GetCrc32AndCopy(input, null); | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Returns the CRC32 for the specified stream, and writes the input into the | 
 |         /// output stream. | 
 |         /// </summary> | 
 |         /// <param name="input">The stream over which to calculate the CRC32</param> | 
 |         /// <param name="output">The stream into which to deflate the input</param> | 
 |         /// <returns>the CRC32 calculation</returns> | 
 |         public Int32 GetCrc32AndCopy(System.IO.Stream input, System.IO.Stream output) | 
 |         { | 
 |             if (input == null) | 
 |                 throw new Exception("The input stream must not be null."); | 
 |  | 
 |             unchecked | 
 |             { | 
 |                 byte[] buffer = new byte[BUFFER_SIZE]; | 
 |                 int readSize = BUFFER_SIZE; | 
 |  | 
 |                 _TotalBytesRead = 0; | 
 |                 int count = input.Read(buffer, 0, readSize); | 
 |                 if (output != null) output.Write(buffer, 0, count); | 
 |                 _TotalBytesRead += count; | 
 |                 while (count > 0) | 
 |                 { | 
 |                     SlurpBlock(buffer, 0, count); | 
 |                     count = input.Read(buffer, 0, readSize); | 
 |                     if (output != null) output.Write(buffer, 0, count); | 
 |                     _TotalBytesRead += count; | 
 |                 } | 
 |  | 
 |                 return (Int32)(~_register); | 
 |             } | 
 |         } | 
 |  | 
 |  | 
 |         /// <summary> | 
 |         ///   Get the CRC32 for the given (word,byte) combo.  This is a | 
 |         ///   computation defined by PKzip for PKZIP 2.0 (weak) encryption. | 
 |         /// </summary> | 
 |         /// <param name="W">The word to start with.</param> | 
 |         /// <param name="B">The byte to combine it with.</param> | 
 |         /// <returns>The CRC-ized result.</returns> | 
 |         public Int32 ComputeCrc32(Int32 W, byte B) | 
 |         { | 
 |             return _InternalComputeCrc32((UInt32)W, B); | 
 |         } | 
 |  | 
 |         internal Int32 _InternalComputeCrc32(UInt32 W, byte B) | 
 |         { | 
 |             return (Int32)(crc32Table[(W ^ B) & 0xFF] ^ (W >> 8)); | 
 |         } | 
 |  | 
 |  | 
 |         /// <summary> | 
 |         /// Update the value for the running CRC32 using the given block of bytes. | 
 |         /// This is useful when using the CRC32() class in a Stream. | 
 |         /// </summary> | 
 |         /// <param name="block">block of bytes to slurp</param> | 
 |         /// <param name="offset">starting point in the block</param> | 
 |         /// <param name="count">how many bytes within the block to slurp</param> | 
 |         public void SlurpBlock(byte[] block, int offset, int count) | 
 |         { | 
 |             if (block == null) | 
 |                 throw new Exception("The data buffer must not be null."); | 
 |  | 
 |             // bzip algorithm | 
 |             for (int i = 0; i < count; i++) | 
 |             { | 
 |                 int x = offset + i; | 
 |                 byte b = block[x]; | 
 |                 if (this.reverseBits) | 
 |                 { | 
 |                     UInt32 temp = (_register >> 24) ^ b; | 
 |                     _register = (_register << 8) ^ crc32Table[temp]; | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     UInt32 temp = (_register & 0x000000FF) ^ b; | 
 |                     _register = (_register >> 8) ^ crc32Table[temp]; | 
 |                 } | 
 |             } | 
 |             _TotalBytesRead += count; | 
 |         } | 
 |  | 
 |  | 
 |         /// <summary> | 
 |         ///   Process one byte in the CRC. | 
 |         /// </summary> | 
 |         /// <param name = "b">the byte to include into the CRC .  </param> | 
 |         public void UpdateCRC(byte b) | 
 |         { | 
 |             if (this.reverseBits) | 
 |             { | 
 |                 UInt32 temp = (_register >> 24) ^ b; | 
 |                 _register = (_register << 8) ^ crc32Table[temp]; | 
 |             } | 
 |             else | 
 |             { | 
 |                 UInt32 temp = (_register & 0x000000FF) ^ b; | 
 |                 _register = (_register >> 8) ^ crc32Table[temp]; | 
 |             } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   Process a run of N identical bytes into the CRC. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     This method serves as an optimization for updating the CRC when a | 
 |         ///     run of identical bytes is found. Rather than passing in a buffer of | 
 |         ///     length n, containing all identical bytes b, this method accepts the | 
 |         ///     byte value and the length of the (virtual) buffer - the length of | 
 |         ///     the run. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         /// <param name = "b">the byte to include into the CRC.  </param> | 
 |         /// <param name = "n">the number of times that byte should be repeated. </param> | 
 |         public void UpdateCRC(byte b, int n) | 
 |         { | 
 |             while (n-- > 0) | 
 |             { | 
 |                 if (this.reverseBits) | 
 |                 { | 
 |                     uint temp = (_register >> 24) ^ b; | 
 |                     _register = (_register << 8) ^ crc32Table[(temp >= 0) | 
 |                                                               ? temp | 
 |                                                               : (temp + 256)]; | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     UInt32 temp = (_register & 0x000000FF) ^ b; | 
 |                     _register = (_register >> 8) ^ crc32Table[(temp >= 0) | 
 |                                                               ? temp | 
 |                                                               : (temp + 256)]; | 
 |  | 
 |                 } | 
 |             } | 
 |         } | 
 |  | 
 |  | 
 |  | 
 |         private static uint ReverseBits(uint data) | 
 |         { | 
 |             unchecked | 
 |             { | 
 |                 uint ret = data; | 
 |                 ret = (ret & 0x55555555) << 1 | (ret >> 1) & 0x55555555; | 
 |                 ret = (ret & 0x33333333) << 2 | (ret >> 2) & 0x33333333; | 
 |                 ret = (ret & 0x0F0F0F0F) << 4 | (ret >> 4) & 0x0F0F0F0F; | 
 |                 ret = (ret << 24) | ((ret & 0xFF00) << 8) | ((ret >> 8) & 0xFF00) | (ret >> 24); | 
 |                 return ret; | 
 |             } | 
 |         } | 
 |  | 
 |         private static byte ReverseBits(byte data) | 
 |         { | 
 |             unchecked | 
 |             { | 
 |                 uint u = (uint)data * 0x00020202; | 
 |                 uint m = 0x01044010; | 
 |                 uint s = u & m; | 
 |                 uint t = (u << 2) & (m << 1); | 
 |                 return (byte)((0x01001001 * (s + t)) >> 24); | 
 |             } | 
 |         } | 
 |  | 
 |  | 
 |  | 
 |         private void GenerateLookupTable() | 
 |         { | 
 |             crc32Table = new UInt32[256]; | 
 |             unchecked | 
 |             { | 
 |                 UInt32 dwCrc; | 
 |                 byte i = 0; | 
 |                 do | 
 |                 { | 
 |                     dwCrc = i; | 
 |                     for (byte j = 8; j > 0; j--) | 
 |                     { | 
 |                         if ((dwCrc & 1) == 1) | 
 |                         { | 
 |                             dwCrc = (dwCrc >> 1) ^ dwPolynomial; | 
 |                         } | 
 |                         else | 
 |                         { | 
 |                             dwCrc >>= 1; | 
 |                         } | 
 |                     } | 
 |                     if (reverseBits) | 
 |                     { | 
 |                         crc32Table[ReverseBits(i)] = ReverseBits(dwCrc); | 
 |                     } | 
 |                     else | 
 |                     { | 
 |                         crc32Table[i] = dwCrc; | 
 |                     } | 
 |                     i++; | 
 |                 } while (i!=0); | 
 |             } | 
 |  | 
 | #if VERBOSE | 
 |             Console.WriteLine(); | 
 |             Console.WriteLine("private static readonly UInt32[] crc32Table = {"); | 
 |             for (int i = 0; i < crc32Table.Length; i+=4) | 
 |             { | 
 |                 Console.Write("   "); | 
 |                 for (int j=0; j < 4; j++) | 
 |                 { | 
 |                     Console.Write(" 0x{0:X8}U,", crc32Table[i+j]); | 
 |                 } | 
 |                 Console.WriteLine(); | 
 |             } | 
 |             Console.WriteLine("};"); | 
 |             Console.WriteLine(); | 
 | #endif | 
 |         } | 
 |  | 
 |  | 
 |         private uint gf2_matrix_times(uint[] matrix, uint vec) | 
 |         { | 
 |             uint sum = 0; | 
 |             int i=0; | 
 |             while (vec != 0) | 
 |             { | 
 |                 if ((vec & 0x01)== 0x01) | 
 |                     sum ^= matrix[i]; | 
 |                 vec >>= 1; | 
 |                 i++; | 
 |             } | 
 |             return sum; | 
 |         } | 
 |  | 
 |         private void gf2_matrix_square(uint[] square, uint[] mat) | 
 |         { | 
 |             for (int i = 0; i < 32; i++) | 
 |                 square[i] = gf2_matrix_times(mat, mat[i]); | 
 |         } | 
 |  | 
 |  | 
 |  | 
 |         /// <summary> | 
 |         ///   Combines the given CRC32 value with the current running total. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   This is useful when using a divide-and-conquer approach to | 
 |         ///   calculating a CRC.  Multiple threads can each calculate a | 
 |         ///   CRC32 on a segment of the data, and then combine the | 
 |         ///   individual CRC32 values at the end. | 
 |         /// </remarks> | 
 |         /// <param name="crc">the crc value to be combined with this one</param> | 
 |         /// <param name="length">the length of data the CRC value was calculated on</param> | 
 |         public void Combine(int crc, int length) | 
 |         { | 
 |             uint[] even = new uint[32];     // even-power-of-two zeros operator | 
 |             uint[] odd = new uint[32];      // odd-power-of-two zeros operator | 
 |  | 
 |             if (length == 0) | 
 |                 return; | 
 |  | 
 |             uint crc1= ~_register; | 
 |             uint crc2= (uint) crc; | 
 |  | 
 |             // put operator for one zero bit in odd | 
 |             odd[0] = this.dwPolynomial;  // the CRC-32 polynomial | 
 |             uint row = 1; | 
 |             for (int i = 1; i < 32; i++) | 
 |             { | 
 |                 odd[i] = row; | 
 |                 row <<= 1; | 
 |             } | 
 |  | 
 |             // put operator for two zero bits in even | 
 |             gf2_matrix_square(even, odd); | 
 |  | 
 |             // put operator for four zero bits in odd | 
 |             gf2_matrix_square(odd, even); | 
 |  | 
 |             uint len2 = (uint) length; | 
 |  | 
 |             // apply len2 zeros to crc1 (first square will put the operator for one | 
 |             // zero byte, eight zero bits, in even) | 
 |             do { | 
 |                 // apply zeros operator for this bit of len2 | 
 |                 gf2_matrix_square(even, odd); | 
 |  | 
 |                 if ((len2 & 1)== 1) | 
 |                     crc1 = gf2_matrix_times(even, crc1); | 
 |                 len2 >>= 1; | 
 |  | 
 |                 if (len2 == 0) | 
 |                     break; | 
 |  | 
 |                 // another iteration of the loop with odd and even swapped | 
 |                 gf2_matrix_square(odd, even); | 
 |                 if ((len2 & 1)==1) | 
 |                     crc1 = gf2_matrix_times(odd, crc1); | 
 |                 len2 >>= 1; | 
 |  | 
 |  | 
 |             } while (len2 != 0); | 
 |  | 
 |             crc1 ^= crc2; | 
 |  | 
 |             _register= ~crc1; | 
 |  | 
 |             //return (int) crc1; | 
 |             return; | 
 |         } | 
 |  | 
 |  | 
 |         /// <summary> | 
 |         ///   Create an instance of the CRC32 class using the default settings: no | 
 |         ///   bit reversal, and a polynomial of 0xEDB88320. | 
 |         /// </summary> | 
 |         public CRC32() : this(false) | 
 |         { | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   Create an instance of the CRC32 class, specifying whether to reverse | 
 |         ///   data bits or not. | 
 |         /// </summary> | 
 |         /// <param name='reverseBits'> | 
 |         ///   specify true if the instance should reverse data bits. | 
 |         /// </param> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     In the CRC-32 used by BZip2, the bits are reversed. Therefore if you | 
 |         ///     want a CRC32 with compatibility with BZip2, you should pass true | 
 |         ///     here. In the CRC-32 used by GZIP and PKZIP, the bits are not | 
 |         ///     reversed; Therefore if you want a CRC32 with compatibility with | 
 |         ///     those, you should pass false. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         public CRC32(bool reverseBits) : | 
 |             this( unchecked((int)0xEDB88320), reverseBits) | 
 |         { | 
 |         } | 
 |  | 
 |  | 
 |         /// <summary> | 
 |         ///   Create an instance of the CRC32 class, specifying the polynomial and | 
 |         ///   whether to reverse data bits or not. | 
 |         /// </summary> | 
 |         /// <param name='polynomial'> | 
 |         ///   The polynomial to use for the CRC, expressed in the reversed (LSB) | 
 |         ///   format: the highest ordered bit in the polynomial value is the | 
 |         ///   coefficient of the 0th power; the second-highest order bit is the | 
 |         ///   coefficient of the 1 power, and so on. Expressed this way, the | 
 |         ///   polynomial for the CRC-32C used in IEEE 802.3, is 0xEDB88320. | 
 |         /// </param> | 
 |         /// <param name='reverseBits'> | 
 |         ///   specify true if the instance should reverse data bits. | 
 |         /// </param> | 
 |         /// | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     In the CRC-32 used by BZip2, the bits are reversed. Therefore if you | 
 |         ///     want a CRC32 with compatibility with BZip2, you should pass true | 
 |         ///     here for the <c>reverseBits</c> parameter. In the CRC-32 used by | 
 |         ///     GZIP and PKZIP, the bits are not reversed; Therefore if you want a | 
 |         ///     CRC32 with compatibility with those, you should pass false for the | 
 |         ///     <c>reverseBits</c> parameter. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         public CRC32(int polynomial, bool reverseBits) | 
 |         { | 
 |             this.reverseBits = reverseBits; | 
 |             this.dwPolynomial = (uint) polynomial; | 
 |             this.GenerateLookupTable(); | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   Reset the CRC-32 class - clear the CRC "remainder register." | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     Use this when employing a single instance of this class to compute | 
 |         ///     multiple, distinct CRCs on multiple, distinct data blocks. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         public void Reset() | 
 |         { | 
 |             _register = 0xFFFFFFFFU; | 
 |         } | 
 |  | 
 |         // private member vars | 
 |         private UInt32 dwPolynomial; | 
 |         private Int64 _TotalBytesRead; | 
 |         private bool reverseBits; | 
 |         private UInt32[] crc32Table; | 
 |         private const int BUFFER_SIZE = 8192; | 
 |         private UInt32 _register = 0xFFFFFFFFU; | 
 |     } | 
 |  | 
 |  | 
 |     /// <summary> | 
 |     /// A Stream that calculates a CRC32 (a checksum) on all bytes read, | 
 |     /// or on all bytes written. | 
 |     /// </summary> | 
 |     /// | 
 |     /// <remarks> | 
 |     /// <para> | 
 |     /// This class can be used to verify the CRC of a ZipEntry when | 
 |     /// reading from a stream, or to calculate a CRC when writing to a | 
 |     /// stream.  The stream should be used to either read, or write, but | 
 |     /// not both.  If you intermix reads and writes, the results are not | 
 |     /// defined. | 
 |     /// </para> | 
 |     /// | 
 |     /// <para> | 
 |     /// This class is intended primarily for use internally by the | 
 |     /// DotNetZip library. | 
 |     /// </para> | 
 |     /// </remarks> | 
 |     internal class CrcCalculatorStream : System.IO.Stream, System.IDisposable | 
 |     { | 
 |         private static readonly Int64 UnsetLengthLimit = -99; | 
 |  | 
 |         internal System.IO.Stream _innerStream; | 
 |         private CRC32 _Crc32; | 
 |         private Int64 _lengthLimit = -99; | 
 |         private bool _leaveOpen; | 
 |  | 
 |         /// <summary> | 
 |         /// The default constructor. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     Instances returned from this constructor will leave the underlying | 
 |         ///     stream open upon Close().  The stream uses the default CRC32 | 
 |         ///     algorithm, which implies a polynomial of 0xEDB88320. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         /// <param name="stream">The underlying stream</param> | 
 |         public CrcCalculatorStream(System.IO.Stream stream) | 
 |             : this(true, CrcCalculatorStream.UnsetLengthLimit, stream, null) | 
 |         { | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   The constructor allows the caller to specify how to handle the | 
 |         ///   underlying stream at close. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     The stream uses the default CRC32 algorithm, which implies a | 
 |         ///     polynomial of 0xEDB88320. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         /// <param name="stream">The underlying stream</param> | 
 |         /// <param name="leaveOpen">true to leave the underlying stream | 
 |         /// open upon close of the <c>CrcCalculatorStream</c>; false otherwise.</param> | 
 |         public CrcCalculatorStream(System.IO.Stream stream, bool leaveOpen) | 
 |             : this(leaveOpen, CrcCalculatorStream.UnsetLengthLimit, stream, null) | 
 |         { | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   A constructor allowing the specification of the length of the stream | 
 |         ///   to read. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     The stream uses the default CRC32 algorithm, which implies a | 
 |         ///     polynomial of 0xEDB88320. | 
 |         ///   </para> | 
 |         ///   <para> | 
 |         ///     Instances returned from this constructor will leave the underlying | 
 |         ///     stream open upon Close(). | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         /// <param name="stream">The underlying stream</param> | 
 |         /// <param name="length">The length of the stream to slurp</param> | 
 |         public CrcCalculatorStream(System.IO.Stream stream, Int64 length) | 
 |             : this(true, length, stream, null) | 
 |         { | 
 |             if (length < 0) | 
 |                 throw new ArgumentException("length"); | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   A constructor allowing the specification of the length of the stream | 
 |         ///   to read, as well as whether to keep the underlying stream open upon | 
 |         ///   Close(). | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     The stream uses the default CRC32 algorithm, which implies a | 
 |         ///     polynomial of 0xEDB88320. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         /// <param name="stream">The underlying stream</param> | 
 |         /// <param name="length">The length of the stream to slurp</param> | 
 |         /// <param name="leaveOpen">true to leave the underlying stream | 
 |         /// open upon close of the <c>CrcCalculatorStream</c>; false otherwise.</param> | 
 |         public CrcCalculatorStream(System.IO.Stream stream, Int64 length, bool leaveOpen) | 
 |             : this(leaveOpen, length, stream, null) | 
 |         { | 
 |             if (length < 0) | 
 |                 throw new ArgumentException("length"); | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   A constructor allowing the specification of the length of the stream | 
 |         ///   to read, as well as whether to keep the underlying stream open upon | 
 |         ///   Close(), and the CRC32 instance to use. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     The stream uses the specified CRC32 instance, which allows the | 
 |         ///     application to specify how the CRC gets calculated. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         /// <param name="stream">The underlying stream</param> | 
 |         /// <param name="length">The length of the stream to slurp</param> | 
 |         /// <param name="leaveOpen">true to leave the underlying stream | 
 |         /// open upon close of the <c>CrcCalculatorStream</c>; false otherwise.</param> | 
 |         /// <param name="crc32">the CRC32 instance to use to calculate the CRC32</param> | 
 |         public CrcCalculatorStream(System.IO.Stream stream, Int64 length, bool leaveOpen, | 
 |                                    CRC32 crc32) | 
 |             : this(leaveOpen, length, stream, crc32) | 
 |         { | 
 |             if (length < 0) | 
 |                 throw new ArgumentException("length"); | 
 |         } | 
 |  | 
 |  | 
 |         // This ctor is private - no validation is done here.  This is to allow the use | 
 |         // of a (specific) negative value for the _lengthLimit, to indicate that there | 
 |         // is no length set.  So we validate the length limit in those ctors that use an | 
 |         // explicit param, otherwise we don't validate, because it could be our special | 
 |         // value. | 
 |         private CrcCalculatorStream | 
 |             (bool leaveOpen, Int64 length, System.IO.Stream stream, CRC32 crc32) | 
 |             : base() | 
 |         { | 
 |             _innerStream = stream; | 
 |             _Crc32 = crc32 ?? new CRC32(); | 
 |             _lengthLimit = length; | 
 |             _leaveOpen = leaveOpen; | 
 |         } | 
 |  | 
 |  | 
 |         /// <summary> | 
 |         ///   Gets the total number of bytes run through the CRC32 calculator. | 
 |         /// </summary> | 
 |         /// | 
 |         /// <remarks> | 
 |         ///   This is either the total number of bytes read, or the total number of | 
 |         ///   bytes written, depending on the direction of this stream. | 
 |         /// </remarks> | 
 |         public Int64 TotalBytesSlurped | 
 |         { | 
 |             get { return _Crc32.TotalBytesRead; } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   Provides the current CRC for all blocks slurped in. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     The running total of the CRC is kept as data is written or read | 
 |         ///     through the stream.  read this property after all reads or writes to | 
 |         ///     get an accurate CRC for the entire stream. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         public Int32 Crc | 
 |         { | 
 |             get { return _Crc32.Crc32Result; } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   Indicates whether the underlying stream will be left open when the | 
 |         ///   <c>CrcCalculatorStream</c> is Closed. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     Set this at any point before calling <see cref="Close()"/>. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         public bool LeaveOpen | 
 |         { | 
 |             get { return _leaveOpen; } | 
 |             set { _leaveOpen = value; } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Read from the stream | 
 |         /// </summary> | 
 |         /// <param name="buffer">the buffer to read</param> | 
 |         /// <param name="offset">the offset at which to start</param> | 
 |         /// <param name="count">the number of bytes to read</param> | 
 |         /// <returns>the number of bytes actually read</returns> | 
 |         public override int Read(byte[] buffer, int offset, int count) | 
 |         { | 
 |             int bytesToRead = count; | 
 |  | 
 |             // Need to limit the # of bytes returned, if the stream is intended to have | 
 |             // a definite length.  This is especially useful when returning a stream for | 
 |             // the uncompressed data directly to the application.  The app won't | 
 |             // necessarily read only the UncompressedSize number of bytes.  For example | 
 |             // wrapping the stream returned from OpenReader() into a StreadReader() and | 
 |             // calling ReadToEnd() on it, We can "over-read" the zip data and get a | 
 |             // corrupt string.  The length limits that, prevents that problem. | 
 |  | 
 |             if (_lengthLimit != CrcCalculatorStream.UnsetLengthLimit) | 
 |             { | 
 |                 if (_Crc32.TotalBytesRead >= _lengthLimit) return 0; // EOF | 
 |                 Int64 bytesRemaining = _lengthLimit - _Crc32.TotalBytesRead; | 
 |                 if (bytesRemaining < count) bytesToRead = (int)bytesRemaining; | 
 |             } | 
 |             int n = _innerStream.Read(buffer, offset, bytesToRead); | 
 |             if (n > 0) _Crc32.SlurpBlock(buffer, offset, n); | 
 |             return n; | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Write to the stream. | 
 |         /// </summary> | 
 |         /// <param name="buffer">the buffer from which to write</param> | 
 |         /// <param name="offset">the offset at which to start writing</param> | 
 |         /// <param name="count">the number of bytes to write</param> | 
 |         public override void Write(byte[] buffer, int offset, int count) | 
 |         { | 
 |             if (count > 0) _Crc32.SlurpBlock(buffer, offset, count); | 
 |             _innerStream.Write(buffer, offset, count); | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Indicates whether the stream supports reading. | 
 |         /// </summary> | 
 |         public override bool CanRead | 
 |         { | 
 |             get { return _innerStream.CanRead; } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   Indicates whether the stream supports seeking. | 
 |         /// </summary> | 
 |         /// <remarks> | 
 |         ///   <para> | 
 |         ///     Always returns false. | 
 |         ///   </para> | 
 |         /// </remarks> | 
 |         public override bool CanSeek | 
 |         { | 
 |             get { return false; } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Indicates whether the stream supports writing. | 
 |         /// </summary> | 
 |         public override bool CanWrite | 
 |         { | 
 |             get { return _innerStream.CanWrite; } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Flush the stream. | 
 |         /// </summary> | 
 |         public override void Flush() | 
 |         { | 
 |             _innerStream.Flush(); | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   Returns the length of the underlying stream. | 
 |         /// </summary> | 
 |         public override long Length | 
 |         { | 
 |             get | 
 |             { | 
 |                 if (_lengthLimit == CrcCalculatorStream.UnsetLengthLimit) | 
 |                     return _innerStream.Length; | 
 |                 else return _lengthLimit; | 
 |             } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         ///   The getter for this property returns the total bytes read. | 
 |         ///   If you use the setter, it will throw | 
 |         /// <see cref="NotSupportedException"/>. | 
 |         /// </summary> | 
 |         public override long Position | 
 |         { | 
 |             get { return _Crc32.TotalBytesRead; } | 
 |             set { throw new NotSupportedException(); } | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Seeking is not supported on this stream. This method always throws | 
 |         /// <see cref="NotSupportedException"/> | 
 |         /// </summary> | 
 |         /// <param name="offset">N/A</param> | 
 |         /// <param name="origin">N/A</param> | 
 |         /// <returns>N/A</returns> | 
 |         public override long Seek(long offset, System.IO.SeekOrigin origin) | 
 |         { | 
 |             throw new NotSupportedException(); | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// This method always throws | 
 |         /// <see cref="NotSupportedException"/> | 
 |         /// </summary> | 
 |         /// <param name="value">N/A</param> | 
 |         public override void SetLength(long value) | 
 |         { | 
 |             throw new NotSupportedException(); | 
 |         } | 
 |  | 
 |  | 
 |         void IDisposable.Dispose() | 
 |         { | 
 |             Close(); | 
 |         } | 
 |  | 
 |         /// <summary> | 
 |         /// Closes the stream. | 
 |         /// </summary> | 
 |         public override void Close() | 
 |         { | 
 |             base.Close(); | 
 |             if (!_leaveOpen) | 
 |                 _innerStream.Close(); | 
 |         } | 
 |  | 
 |     } | 
 |  | 
 | } |