| // 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(); |
| } |
| |
| } |
| |
| } |