814 lines
34 KiB
C#
814 lines
34 KiB
C#
|
//------------------------------------------------------------------------------
|
|||
|
// <copyright file="ZLibNative.cs" company="Microsoft">
|
|||
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|||
|
// </copyright>
|
|||
|
//
|
|||
|
// <owner>gpaperin</owner>
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
|
|||
|
using System.Text;
|
|||
|
using System.Runtime.InteropServices;
|
|||
|
using System.Runtime.Versioning;
|
|||
|
using System.Runtime.CompilerServices;
|
|||
|
using System.Runtime.ConstrainedExecution;
|
|||
|
using Microsoft.Win32.SafeHandles;
|
|||
|
using System.Security.Permissions;
|
|||
|
using System.Diagnostics.Contracts;
|
|||
|
using System.Security;
|
|||
|
|
|||
|
|
|||
|
namespace System.IO.Compression {
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// This class provides declaration for constants and PInvokes as well as some basic tools for exposing the
|
|||
|
/// native CLRCompression.dll (effectively, ZLib) library to managed code.
|
|||
|
///
|
|||
|
/// This file can be included into any managed project that needs to reference CLRCompression.dll.
|
|||
|
/// Thus, it is located together with the CLRCompression build files that produce the unmanaged CLRCompression.dll
|
|||
|
///
|
|||
|
/// ** How to refer to this file from your build project: **
|
|||
|
///
|
|||
|
/// - The master copy of this file is located under
|
|||
|
/// ndp\fx\src\CLRCompression\System\IO\Compression\ZLibNative.cs .
|
|||
|
/// (together with all native CLRCompression sources which are under ndp\fx\src\CLRCompression\zlib-1.2.3\)
|
|||
|
///
|
|||
|
/// - The CLRCompression assembly build project for will publish this file to
|
|||
|
/// <binaries>\<build_flavour>\InterAPIsCandidates\NDP_FX\inc\ .
|
|||
|
/// Projects within the NDP_FX partition can consume this file from there.
|
|||
|
///
|
|||
|
/// - After every change, this file must also be published to
|
|||
|
/// InternalApis\NDP_FX\inc\ .
|
|||
|
/// Projects outside NDP_FX can consume this files from that location.
|
|||
|
///
|
|||
|
/// In either case, the msbuild variable $(NDP_FXIncPath) should include the one of the above directories that
|
|||
|
/// is appropriate for the project.
|
|||
|
///
|
|||
|
/// Projects outside the FX partition can declare a dependency on NDP_FX in their
|
|||
|
/// partition.settings.targets, which will cause the right path to be automatically added to their include path list.
|
|||
|
///
|
|||
|
/// Projects inside FX can also use $(InterAPIsCandidatesPath)\inc\ instead of $(NDP_FXIncPath) if the latter is broken.
|
|||
|
///
|
|||
|
/// If some additional ZLib functionality needs to be exposed in future, there are (at least) 3 files that need
|
|||
|
/// to be updated:
|
|||
|
/// 1) clrcompression.def : To list the DLL entry points.
|
|||
|
/// 2) clrcompression.nativeproj : To list all the required ZLib source files
|
|||
|
/// (sources that are not required for the currently exposed functionality are commented out).
|
|||
|
/// 3) This file (ZLibNative.cs) : To include PInvoke declarations / delegates for all necessary routines.
|
|||
|
///
|
|||
|
///
|
|||
|
/// See also: How to choose a compression level (in comments to <code>CompressionLevel</code>.
|
|||
|
/// </summary>
|
|||
|
internal static class ZLibNative {
|
|||
|
|
|||
|
#region File name constants
|
|||
|
|
|||
|
public const string ZLibNativeDllName = "clrcompression.dll";
|
|||
|
private const string Kernel32DllName = "kernel32.dll";
|
|||
|
|
|||
|
#endregion // File name constants
|
|||
|
|
|||
|
|
|||
|
#region Constants defined in zlib.h
|
|||
|
|
|||
|
public const string ZLibVersion = "1.2.3";
|
|||
|
|
|||
|
// This is the NULL pointer for using with ZLib pointers;
|
|||
|
// we prefer it to IntPtr.Zero to mimic the definition of Z_NULL in zlib.h:
|
|||
|
internal static readonly IntPtr ZNullPtr = (IntPtr) ((Int32) 0);
|
|||
|
|
|||
|
|
|||
|
public enum FlushCode : int {
|
|||
|
NoFlush = 0,
|
|||
|
PartialFlush = 1,
|
|||
|
SyncFlush = 2,
|
|||
|
FullFlush = 3,
|
|||
|
Finish = 4,
|
|||
|
Block = 5,
|
|||
|
//Trees = 6 // Only in ZLib 1.2.4 and later
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
public enum ErrorCode : int {
|
|||
|
Ok = 0,
|
|||
|
StreamEnd = 1,
|
|||
|
NeedDictionary = 2,
|
|||
|
ErrorNo = -1,
|
|||
|
StreamError = -2,
|
|||
|
DataError = -3,
|
|||
|
MemError = -4,
|
|||
|
BufError = -5,
|
|||
|
VersionError = -6
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// <p>ZLib can accept any integer value between 0 and 9 (inclusive) as a valid compression level parameter:
|
|||
|
/// 1 gives best speed, 9 gives best compression, 0 gives no compression at all (the input data is simply copied a block at a time).
|
|||
|
/// <code>CompressionLevel.DefaultCompression</code> = -1 requests a default compromise between speed and compression
|
|||
|
/// (currently equivalent to level 6).</p>
|
|||
|
///
|
|||
|
/// <p><strong>How to choose a compression level:</strong></p>
|
|||
|
///
|
|||
|
/// <p>The names <code>NoCompression</code>, <code>BestSpeed</code>, <code>BestCompression</code> are taken over from the corresponding
|
|||
|
/// ZLib definitions. However, extensive compression performance tests on real data show that they do not describe the reality well.
|
|||
|
/// We have run a large number of tests on different data sets including binary data, English language text, JPEG images and source code.
|
|||
|
/// The results show:</p>
|
|||
|
/// <ul>
|
|||
|
/// <li><code>CompressionStrategy.DefaultStrategy</code> is the best strategy in most scenarios.</li>
|
|||
|
/// <li><code>CompressionLevel</code> values over 6 do not significantly improve the compression rate,
|
|||
|
/// however such values require additional compression time.</li>
|
|||
|
/// <li>Thus it is not recommended to use a compression level higher than 6, including the
|
|||
|
/// value <code>CompressionLevel.BestCompression</code>.</li>
|
|||
|
/// <li>The optimal compression performance (time/rate ratio) tends to occur at compression level 6
|
|||
|
/// (<code>CompressionLevel.DefaultCompression</code>).</li>
|
|||
|
/// <li>In scenarios where runtime performance takes precedence over compression rate, smaller compression level values
|
|||
|
/// can be considered.</li>
|
|||
|
/// </ul>
|
|||
|
/// <p>We recommend using one of the three following combinations:<br />
|
|||
|
/// (See also the constants <code>Deflate_DefaultWindowBits</code> and <code>Deflate_DefaultMemLevel</code> below.)</p>
|
|||
|
///
|
|||
|
/// <p><em>Optimal Compression:</em></p>
|
|||
|
/// <p><code>ZLibNative.CompressionLevel compressionLevel = (ZLibNative.CompressionLevel) 6;</code> <br />
|
|||
|
/// <code>Int32 windowBits = 15; // or -15 if no headers required</code> <br />
|
|||
|
/// <code>Int32 memLevel = 8;</code> <br />
|
|||
|
/// <code>ZLibNative.CompressionStrategy strategy = ZLibNative.CompressionStrategy.DefaultStrategy;</code> </p>
|
|||
|
///
|
|||
|
///<p><em>Fastest compression:</em></p>
|
|||
|
///<p><code>ZLibNative.CompressionLevel compressionLevel = (ZLibNative.CompressionLevel) 1;</code> <br />
|
|||
|
/// <code>Int32 windowBits = 15; // or -15 if no headers required</code> <br />
|
|||
|
/// <code>Int32 memLevel = 8; </code> <br />
|
|||
|
/// <code>ZLibNative.CompressionStrategy strategy = ZLibNative.CompressionStrategy.DefaultStrategy;</code> </p>
|
|||
|
///
|
|||
|
/// <p><em>No compression (even faster, useful for data that cannot be compressed such some image formats):</em></p>
|
|||
|
/// <p><code>ZLibNative.CompressionLevel compressionLevel = (ZLibNative.CompressionLevel) 0;</code> <br />
|
|||
|
/// <code>Int32 windowBits = 15; // or -15 if no headers required</code> <br />
|
|||
|
/// <code>Int32 memLevel = 7;</code> <br />
|
|||
|
/// <code>ZLibNative.CompressionStrategy strategy = ZLibNative.CompressionStrategy.DefaultStrategy;</code> </p>
|
|||
|
/// </summary>
|
|||
|
public enum CompressionLevel : int {
|
|||
|
NoCompression = 0,
|
|||
|
BestSpeed = 1,
|
|||
|
BestCompression = 9, // Refer to "How to choose a compression level" above.
|
|||
|
DefaultCompression = -1
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// <p><strong>From the ZLib manual:</strong></p>
|
|||
|
/// <p><code>CompressionStrategy</code> is used to tune the compression algorithm.<br />
|
|||
|
/// Use the value <code>DefaultStrategy</code> for normal data, <code>Filtered</code> for data produced by a filter (or predictor),
|
|||
|
/// <code>HuffmanOnly</code> to force Huffman encoding only (no string match), or <code>Rle</code> to limit match distances to one
|
|||
|
/// (run-length encoding). Filtered data consists mostly of small values with a somewhat random distribution. In this case, the
|
|||
|
/// compression algorithm is tuned to compress them better. The effect of <code>Filtered</code> is to force more Huffman coding and]
|
|||
|
/// less string matching; it is somewhat intermediate between <code>DefaultStrategy</code> and <code>HuffmanOnly</code>.
|
|||
|
/// <code>Rle</code> is designed to be almost as fast as <code>HuffmanOnly</code>, but give better compression for PNG image data.
|
|||
|
/// The strategy parameter only affects the compression ratio but not the correctness of the compressed output even if it is not set
|
|||
|
/// appropriately. <code>Fixed</code> prevents the use of dynamic Huffman codes, allowing for a simpler decoder for special applications.</p>
|
|||
|
///
|
|||
|
/// <p><strong>For NetFx use:</strong></p>
|
|||
|
/// <p>We have investigated compression scenarios for a bunch of different requently occuring compression data and found that in all
|
|||
|
/// cases we invesigated so far, <code>DefaultStrategy</code> provided best results</p>
|
|||
|
/// <p>See also: How to choose a compression level (in comments to <code>CompressionLevel</code>.</p>
|
|||
|
/// </summary>
|
|||
|
public enum CompressionStrategy : int {
|
|||
|
Filtered = 1,
|
|||
|
HuffmanOnly = 2,
|
|||
|
Rle = 3,
|
|||
|
Fixed = 4,
|
|||
|
DefaultStrategy = 0
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// In version 1.2.3, ZLib provides on the <code>Deflated</code>-<code>CompressionMethod</code>.
|
|||
|
/// </summary>
|
|||
|
public enum CompressionMethod : int {
|
|||
|
Deflated = 8
|
|||
|
}
|
|||
|
|
|||
|
#endregion // Constants defined in zlib.h
|
|||
|
|
|||
|
|
|||
|
#region Defaults for ZLib parameters
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// <p><strong>From the ZLib manual:</strong></p>
|
|||
|
/// <p>ZLib's <code>windowBits</code> parameter is the base two logarithm of the window size (the size of the history buffer).
|
|||
|
/// It should be in the range 8..15 for this version of the library. Larger values of this parameter result in better compression
|
|||
|
/// at the expense of memory usage. The default value is 15 if deflateInit is used instead.<br />
|
|||
|
/// <strong>Note</strong>:
|
|||
|
/// <code>windowBits</code> can also be <20>8..<2E>15 for raw deflate. In this case, -windowBits determines the window size.
|
|||
|
/// <code>Deflate</code> will then generate raw deflate data with no ZLib header or trailer, and will not compute an adler32 check value.<br />
|
|||
|
/// <code>windowBits</code> can also be greater than 15 for optional gzip encoding. Add 16 to <code>windowBits</code> to write a simple
|
|||
|
/// GZip header and trailer around the compressed data instead of a ZLib wrapper. The GZip header will have no file name, no extra data,
|
|||
|
/// no comment, no modification time (set to zero), no header crc, and the operating system will be set to 255 (unknown).
|
|||
|
/// If a GZip stream is being written, <code>ZStream.adler</code> is a crc32 instead of an adler32.</p>
|
|||
|
/// <p>See also: How to choose a compression level (in comments to <code>CompressionLevel</code>.</p>
|
|||
|
/// </summary>
|
|||
|
public const int Deflate_DefaultWindowBits = -15; // Leagl values are 8..15 and -8..-15. 15 is the window size,
|
|||
|
// negative val causes deflate to produce raw deflate data (no zlib header).
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// <p><strong>From the ZLib manual:</strong></p>
|
|||
|
/// <p>The <code>memLevel</code> parameter specifies how much memory should be allocated for the internal compression state.
|
|||
|
/// <code>memLevel</code> = 1 uses minimum memory but is slow and reduces compression ratio; <code>memLevel</code> = 9 uses maximum
|
|||
|
/// memory for optimal speed. The default value is 8.</p>
|
|||
|
/// <p>See also: How to choose a compression level (in comments to <code>CompressionLevel</code>.</p>
|
|||
|
/// </summary>
|
|||
|
public const int Deflate_DefaultMemLevel = 8; // Memory usage by deflate. Legal range: [1..9]. 8 is ZLib default.
|
|||
|
// More is faster and better compression with more memory usage.
|
|||
|
#endregion // Defaults for ZLib parameters
|
|||
|
|
|||
|
|
|||
|
#region ZLib stream descriptor data structure
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Do not construct instances of <code>ZStream</code> explicitly.
|
|||
|
/// Always use <code>ZLibNative.DeflateInit2_</code> or <code>ZLibNative.InflateInit2_</code> instead.
|
|||
|
/// Those methods will wrap this structure into a <code>SafeHandle</code> and thus make sure that it is always disposed correctly.
|
|||
|
/// </summary>
|
|||
|
[StructLayout(LayoutKind.Sequential, CharSet=CharSet.Ansi)]
|
|||
|
internal struct ZStream {
|
|||
|
internal IntPtr nextIn; //Bytef *next_in; /* next input byte */
|
|||
|
internal UInt32 availIn; //uInt avail_in; /* number of bytes available at next_in */
|
|||
|
internal UInt32 totalIn; //uLong total_in; /* total nb of input bytes read so far */
|
|||
|
|
|||
|
internal IntPtr nextOut; //Bytef *next_out; /* next output byte should be put there */
|
|||
|
internal UInt32 availOut; //uInt avail_out; /* remaining free space at next_out */
|
|||
|
internal UInt32 totalOut; //uLong total_out; /* total nb of bytes output so far */
|
|||
|
|
|||
|
internal IntPtr msg; //char *msg; /* last error message, NULL if no error */
|
|||
|
|
|||
|
internal IntPtr state; //struct internal_state FAR *state; /* not visible by applications */
|
|||
|
|
|||
|
internal IntPtr zalloc; //alloc_func zalloc; /* used to allocate the internal state */
|
|||
|
internal IntPtr zfree; //free_func zfree; /* used to free the internal state */
|
|||
|
internal IntPtr opaque; //voidpf opaque; /* private data object passed to zalloc and zfree */
|
|||
|
|
|||
|
internal Int32 dataType; //int data_type; /* best guess about the data type: binary or text */
|
|||
|
internal UInt32 adler; //uLong adler; /* adler32 value of the uncompressed data */
|
|||
|
internal UInt32 reserved; //uLong reserved; /* reserved for future use */
|
|||
|
}
|
|||
|
|
|||
|
#endregion // ZLib stream descriptor data structure
|
|||
|
|
|||
|
|
|||
|
#region PInvoke declarations for wrapped native functions
|
|||
|
|
|||
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
private delegate ErrorCode DeflateInit2_Delegate(ref ZStream stream, CompressionLevel level, CompressionMethod method,
|
|||
|
int windowBits, int memLevel, CompressionStrategy strategy,
|
|||
|
[MarshalAs(UnmanagedType.LPStr)] string version, int streamSize);
|
|||
|
|
|||
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
private delegate ErrorCode DeflateDelegate(ref ZStream stream, FlushCode flush);
|
|||
|
|
|||
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
private delegate ErrorCode DeflateEndDelegate(ref ZStream stream);
|
|||
|
|
|||
|
|
|||
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
private delegate ErrorCode InflateInit2_Delegate(ref ZStream stream,
|
|||
|
int windowBits,
|
|||
|
[MarshalAs(UnmanagedType.LPStr)] string version, int streamSize);
|
|||
|
|
|||
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
private delegate ErrorCode InflateDelegate(ref ZStream stream, FlushCode flush);
|
|||
|
|
|||
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
private delegate ErrorCode InflateEndDelegate(ref ZStream stream);
|
|||
|
|
|||
|
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
private delegate Int32 ZlibCompileFlagsDelegate();
|
|||
|
|
|||
|
#endregion // PInvoke declarations for wrapped native functions
|
|||
|
|
|||
|
|
|||
|
#region Declarations of Windows API needed to load the native library
|
|||
|
// As described at the top of this file, this file may be used in several managed DLLs that require ZLib
|
|||
|
// functionality. Thus we cannot rely on a specific internal XXXNativeMethod class to declare Windows APIs.
|
|||
|
// Instead, let us declare them explicitly:
|
|||
|
|
|||
|
private class NativeMethods {
|
|||
|
|
|||
|
[DllImport(Kernel32DllName, CharSet=CharSet.Ansi, BestFitMapping=false)]
|
|||
|
[ResourceExposure(ResourceScope.Process)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
internal static extern IntPtr GetProcAddress(SafeLibraryHandle moduleHandle, String procName);
|
|||
|
|
|||
|
[DllImport(Kernel32DllName, CharSet=CharSet.Unicode, SetLastError=true)]
|
|||
|
[ResourceExposure(ResourceScope.Machine)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
internal static extern SafeLibraryHandle LoadLibrary(String libPath);
|
|||
|
|
|||
|
#if !FEATURE_NETCORE
|
|||
|
[ReliabilityContract(Consistency.WillNotCorruptState, Cer.Success)]
|
|||
|
#endif
|
|||
|
[DllImport(Kernel32DllName, ExactSpelling=true)]
|
|||
|
[SuppressUnmanagedCodeSecurity]
|
|||
|
[SecurityCritical]
|
|||
|
internal static extern bool FreeLibrary(IntPtr moduleHandle);
|
|||
|
}
|
|||
|
#endregion // Declarations of Windows API needed to load the native library
|
|||
|
|
|||
|
|
|||
|
#region Handle to native DLL in memory
|
|||
|
|
|||
|
|
|||
|
// Handle type:
|
|||
|
[SecurityCritical]
|
|||
|
private class SafeLibraryHandle : SafeHandleZeroOrMinusOneIsInvalid {
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
internal SafeLibraryHandle()
|
|||
|
: base(true) {
|
|||
|
}
|
|||
|
|
|||
|
#if !FEATURE_NETCORE
|
|||
|
[ReliabilityContract(Consistency.WillNotCorruptState, Cer.MayFail)]
|
|||
|
#endif
|
|||
|
[SecurityCritical]
|
|||
|
protected override bool ReleaseHandle() {
|
|||
|
|
|||
|
bool res = ZLibNative.NativeMethods.FreeLibrary(handle);
|
|||
|
handle = IntPtr.Zero;
|
|||
|
return res;
|
|||
|
}
|
|||
|
} // private class SafeLibraryHandle
|
|||
|
|
|||
|
#endregion // Handle to native DLL in memory
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* Do not remove the nested typing of types inside of <code>System.IO.Compression.ZLibNative</code>.
|
|||
|
* This was done on purpose to:
|
|||
|
*
|
|||
|
* - Achieve the right encapsulation in a situation where <code>ZLibNative</code> may be compiled division-wide
|
|||
|
* into different assemblies that wish to consume <code>CLRCompression</code>. Since <code>internal</code>
|
|||
|
* scope is effectively like <code>public</code> scope when compiling <code>ZLibNative</code> into a higher
|
|||
|
* level assembly, we need a combination of inner types and <code>private</code>-scope members to achieve
|
|||
|
* the right encapsulation.
|
|||
|
*
|
|||
|
* - Achieve late dynamic loading of <code>CLRCompression.dll</code> at the right time.
|
|||
|
* The native assembly will not be loaded unless it is actually used since the loading is performed by a static
|
|||
|
* constructor of an inner type that is not directly referenced by user code.
|
|||
|
*
|
|||
|
* In Dev12 we would like to create a proper feature for loading native assemblies from user-specified
|
|||
|
* directories in order to PInvoke into them. This would preferably happen in the native interop/PInvoke
|
|||
|
* layer; if not we can add a Framework level feature.
|
|||
|
*/
|
|||
|
|
|||
|
#region ZLib Stream Handle type
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// The <code>ZLibStreamHandle</code> could be a <code>CriticalFinalizerObject</code> rather than a
|
|||
|
/// <code>SafeHandleMinusOneIsInvalid</code>. This would save an <code>IntPtr</code> field since
|
|||
|
/// <code>ZLibStreamHandle</code> does not actually use its <code>handle</code> field.
|
|||
|
/// Instead it uses a <code>private ZStream zStream</code> field which is the actual handle data
|
|||
|
/// structure requiring critical finalization.
|
|||
|
/// However, we would like to take advantage if the better debugability offered by the fact that a
|
|||
|
/// <em>releaseHandleFailed MDA</em> is raised if the <code>ReleaseHandle</code> method returns
|
|||
|
/// <code>false</code>, which can for instance happen if the underlying ZLib <code>XxxxEnd</code>
|
|||
|
/// routines return an failure error code.
|
|||
|
/// </summary>
|
|||
|
[SecurityCritical]
|
|||
|
public sealed class ZLibStreamHandle : SafeHandleMinusOneIsInvalid {
|
|||
|
|
|||
|
|
|||
|
#region Library loading and initialisation
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
private static class NativeZLibDLLStub {
|
|||
|
|
|||
|
#region Function pointers to native functions:
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
internal static DeflateInit2_Delegate deflateInit2_Delegate;
|
|||
|
[SecurityCritical]
|
|||
|
internal static DeflateDelegate deflateDelegate;
|
|||
|
[SecurityCritical]
|
|||
|
internal static DeflateEndDelegate deflateEndDelegate;
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
internal static InflateInit2_Delegate inflateInit2_Delegate;
|
|||
|
[SecurityCritical]
|
|||
|
internal static InflateDelegate inflateDelegate;
|
|||
|
[SecurityCritical]
|
|||
|
internal static InflateEndDelegate inflateEndDelegate;
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
internal static ZlibCompileFlagsDelegate zlibCompileFlagsDelegate;
|
|||
|
|
|||
|
#endregion // Function pointers to native functions:
|
|||
|
|
|||
|
|
|||
|
#region Initialisation code
|
|||
|
|
|||
|
#if !FEATURE_NETCORE // Security rules for CoreSys say the class is security critical, therefore this cannot be security safe critical.
|
|||
|
[SecuritySafeCritical]
|
|||
|
#endif
|
|||
|
private static void LoadZLibDLL() {
|
|||
|
|
|||
|
#if !FEATURE_NETCORE
|
|||
|
new FileIOPermission(PermissionState.Unrestricted).Assert();
|
|||
|
#endif
|
|||
|
|
|||
|
String fxDir = RuntimeEnvironment.GetRuntimeDirectory();
|
|||
|
String zlibDllPath = Path.Combine(fxDir, ZLibNativeDllName);
|
|||
|
|
|||
|
if (!File.Exists(zlibDllPath))
|
|||
|
throw new DllNotFoundException(ZLibNativeDllName);
|
|||
|
|
|||
|
SafeLibraryHandle libHndl = ZLibNative.NativeMethods.LoadLibrary(zlibDllPath);
|
|||
|
|
|||
|
if (libHndl.IsInvalid) {
|
|||
|
|
|||
|
Int32 hresult = Marshal.GetHRForLastWin32Error();
|
|||
|
Marshal.ThrowExceptionForHR(hresult, new IntPtr(-1));
|
|||
|
|
|||
|
// If Marshal.ThrowExceptionForHR did not throw, we still need to make sure to throw:
|
|||
|
throw new InvalidOperationException();
|
|||
|
}
|
|||
|
|
|||
|
ZLibStreamHandle.zlibLibraryHandle = libHndl;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
private static DT CreateDelegate<DT>(String entryPointName) {
|
|||
|
|
|||
|
IntPtr entryPoint = ZLibNative.NativeMethods.GetProcAddress(ZLibStreamHandle.zlibLibraryHandle, entryPointName);
|
|||
|
|
|||
|
if (IntPtr.Zero == entryPoint)
|
|||
|
throw new EntryPointNotFoundException(ZLibNativeDllName + "!" + entryPointName);
|
|||
|
|
|||
|
return (DT) (Object) Marshal.GetDelegateForFunctionPointer(entryPoint, typeof(DT));
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
#if !FEATURE_NETCORE // Security rules for CoreSys say the class is security critical, therefore this cannot be security safe critical.
|
|||
|
[SecuritySafeCritical]
|
|||
|
#endif
|
|||
|
private static void InitDelegates() {
|
|||
|
|
|||
|
Contract.Assert(null != ZLibStreamHandle.zlibLibraryHandle);
|
|||
|
Contract.Assert(!ZLibStreamHandle.zlibLibraryHandle.IsInvalid);
|
|||
|
|
|||
|
deflateInit2_Delegate = CreateDelegate<DeflateInit2_Delegate>("deflateInit2_");
|
|||
|
|
|||
|
deflateDelegate = CreateDelegate<DeflateDelegate>("deflate");
|
|||
|
|
|||
|
deflateEndDelegate = CreateDelegate<DeflateEndDelegate>("deflateEnd");
|
|||
|
|
|||
|
inflateInit2_Delegate = CreateDelegate<InflateInit2_Delegate>("inflateInit2_");
|
|||
|
|
|||
|
inflateDelegate = CreateDelegate<InflateDelegate>("inflate");
|
|||
|
|
|||
|
inflateEndDelegate = CreateDelegate<InflateEndDelegate>("inflateEnd");
|
|||
|
|
|||
|
zlibCompileFlagsDelegate = CreateDelegate<ZlibCompileFlagsDelegate>("zlibCompileFlags");
|
|||
|
|
|||
|
#if !SILVERLIGHT
|
|||
|
RuntimeHelpers.PrepareDelegate(deflateInit2_Delegate);
|
|||
|
RuntimeHelpers.PrepareDelegate(deflateDelegate);
|
|||
|
RuntimeHelpers.PrepareDelegate(deflateEndDelegate);
|
|||
|
RuntimeHelpers.PrepareDelegate(inflateInit2_Delegate);
|
|||
|
RuntimeHelpers.PrepareDelegate(inflateDelegate);
|
|||
|
RuntimeHelpers.PrepareDelegate(inflateEndDelegate);
|
|||
|
RuntimeHelpers.PrepareDelegate(zlibCompileFlagsDelegate);
|
|||
|
#endif // !SILVERLIGHT
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecuritySafeCritical]
|
|||
|
static NativeZLibDLLStub() {
|
|||
|
|
|||
|
LoadZLibDLL();
|
|||
|
InitDelegates();
|
|||
|
}
|
|||
|
|
|||
|
#endregion // Initialisation code
|
|||
|
|
|||
|
} // private static class NativeZLibDLLStub
|
|||
|
|
|||
|
|
|||
|
// Handle reference:
|
|||
|
[SecurityCritical]
|
|||
|
private static SafeLibraryHandle zlibLibraryHandle;
|
|||
|
|
|||
|
#endregion // Library loading and initialisation
|
|||
|
|
|||
|
|
|||
|
#region ZLibStream-SafeHandle-related routines
|
|||
|
|
|||
|
public enum State { NotInitialized, InitializedForDeflate, InitializedForInflate, Disposed }
|
|||
|
|
|||
|
private ZStream zStream;
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
private volatile State initializationState;
|
|||
|
|
|||
|
|
|||
|
public ZLibStreamHandle()
|
|||
|
|
|||
|
: base(true) {
|
|||
|
|
|||
|
this.zStream = new ZStream();
|
|||
|
this.zStream.zalloc = ZNullPtr;
|
|||
|
this.zStream.zfree = ZNullPtr;
|
|||
|
this.zStream.opaque = ZNullPtr;
|
|||
|
|
|||
|
this.initializationState = State.NotInitialized;
|
|||
|
this.handle = IntPtr.Zero;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
public State InitializationState {
|
|||
|
|
|||
|
[Pure]
|
|||
|
[SecurityCritical]
|
|||
|
get { return initializationState; }
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
#if !FEATURE_NETCORE
|
|||
|
[ReliabilityContract(Consistency.WillNotCorruptState, Cer.MayFail)]
|
|||
|
#endif
|
|||
|
[SecurityCritical]
|
|||
|
protected override bool ReleaseHandle() {
|
|||
|
|
|||
|
// We are in a finalizer thread at the end of the App and the finalization of the dynamically loaded ZLib happend
|
|||
|
// to be scheduled first. In such case we have no hope of properly freeing zStream. If the process is dying - we
|
|||
|
// do not care. In other cases somethign went badly wrong anyway:
|
|||
|
if (zlibLibraryHandle == null || zlibLibraryHandle.IsInvalid)
|
|||
|
return false;
|
|||
|
|
|||
|
switch (InitializationState) {
|
|||
|
case State.NotInitialized: return true;
|
|||
|
case State.InitializedForDeflate: return (DeflateEnd() == ZLibNative.ErrorCode.Ok);
|
|||
|
case State.InitializedForInflate: return (InflateEnd() == ZLibNative.ErrorCode.Ok);
|
|||
|
case State.Disposed: return true;
|
|||
|
default: return false; // This should never happen. Did we forget one of the State enum values in the switch?
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
#endregion // ZLibStream-SafeHandle-related routines
|
|||
|
|
|||
|
|
|||
|
#region Expose fields on ZStream for use by user / Fx code (add more as required)
|
|||
|
|
|||
|
public IntPtr NextIn { [SecurityCritical] get { return zStream.nextIn; }
|
|||
|
[SecurityCritical] set { zStream.nextIn = value; } }
|
|||
|
|
|||
|
public UInt32 AvailIn { [SecurityCritical] get { return zStream.availIn; }
|
|||
|
[SecurityCritical] set { zStream.availIn = value; } }
|
|||
|
|
|||
|
public UInt32 TotalIn { [SecurityCritical] get { return zStream.totalIn; } }
|
|||
|
|
|||
|
public IntPtr NextOut { [SecurityCritical] get { return zStream.nextOut; }
|
|||
|
[SecurityCritical] set { zStream.nextOut = value; } }
|
|||
|
|
|||
|
public UInt32 AvailOut { [SecurityCritical] get { return zStream.availOut; }
|
|||
|
[SecurityCritical] set { zStream.availOut = value; } }
|
|||
|
|
|||
|
public UInt32 TotalOut { [SecurityCritical] get { return zStream.totalOut; } }
|
|||
|
|
|||
|
public Int32 DataType { [SecurityCritical] get { return zStream.dataType; } }
|
|||
|
|
|||
|
public UInt32 Adler { [SecurityCritical] get { return zStream.adler; } }
|
|||
|
|
|||
|
#endregion // Expose fields on ZStream for use by user / Fx code (add more as required)
|
|||
|
|
|||
|
|
|||
|
#region Expose ZLib functions for use by user / Fx code (add more as required)
|
|||
|
|
|||
|
[Pure]
|
|||
|
[SecurityCritical]
|
|||
|
private void EnsureNotDisposed() {
|
|||
|
|
|||
|
if (InitializationState == State.Disposed)
|
|||
|
throw new ObjectDisposedException(this.GetType().Name);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[Pure]
|
|||
|
[SecurityCritical]
|
|||
|
private void EnsureState(State requiredState) {
|
|||
|
|
|||
|
if (InitializationState != requiredState)
|
|||
|
throw new InvalidOperationException("InitializationState != " + requiredState.ToString());
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public ErrorCode DeflateInit2_(CompressionLevel level, int windowBits, int memLevel, CompressionStrategy strategy) {
|
|||
|
|
|||
|
EnsureNotDisposed();
|
|||
|
EnsureState(State.NotInitialized);
|
|||
|
|
|||
|
ErrorCode errC;
|
|||
|
bool addRefSuccess = false;
|
|||
|
|
|||
|
#if !FEATURE_NETCORE
|
|||
|
RuntimeHelpers.PrepareConstrainedRegions();
|
|||
|
#endif
|
|||
|
try { } finally {
|
|||
|
|
|||
|
errC = NativeZLibDLLStub.deflateInit2_Delegate(ref zStream, level, CompressionMethod.Deflated, windowBits, memLevel,
|
|||
|
strategy, ZLibVersion, Marshal.SizeOf(zStream));
|
|||
|
initializationState = State.InitializedForDeflate;
|
|||
|
zlibLibraryHandle.DangerousAddRef(ref addRefSuccess);
|
|||
|
}
|
|||
|
|
|||
|
Contract.Assert(addRefSuccess, "zlibLibraryHandle.DangerousAddRef in DeflateInit2_ should always succeed, but it did not.");
|
|||
|
|
|||
|
return errC;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public ErrorCode Deflate(FlushCode flush) {
|
|||
|
|
|||
|
EnsureNotDisposed();
|
|||
|
EnsureState(State.InitializedForDeflate);
|
|||
|
|
|||
|
return NativeZLibDLLStub.deflateDelegate(ref zStream, flush);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public ErrorCode DeflateEnd() {
|
|||
|
|
|||
|
EnsureNotDisposed();
|
|||
|
EnsureState(State.InitializedForDeflate);
|
|||
|
|
|||
|
ErrorCode errC;
|
|||
|
|
|||
|
#if !FEATURE_NETCORE
|
|||
|
RuntimeHelpers.PrepareConstrainedRegions();
|
|||
|
#endif
|
|||
|
try { } finally {
|
|||
|
|
|||
|
errC = NativeZLibDLLStub.deflateEndDelegate(ref zStream);
|
|||
|
initializationState = State.Disposed;
|
|||
|
zlibLibraryHandle.DangerousRelease();
|
|||
|
}
|
|||
|
return errC;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public ErrorCode InflateInit2_(int windowBits) {
|
|||
|
|
|||
|
EnsureNotDisposed();
|
|||
|
EnsureState(State.NotInitialized);
|
|||
|
|
|||
|
ErrorCode errC;
|
|||
|
bool addRefSuccess = false;
|
|||
|
|
|||
|
#if !FEATURE_NETCORE
|
|||
|
RuntimeHelpers.PrepareConstrainedRegions();
|
|||
|
#endif
|
|||
|
try { } finally {
|
|||
|
|
|||
|
errC = NativeZLibDLLStub.inflateInit2_Delegate(ref zStream, windowBits, ZLibVersion, Marshal.SizeOf(zStream));
|
|||
|
|
|||
|
initializationState = State.InitializedForInflate;
|
|||
|
zlibLibraryHandle.DangerousAddRef(ref addRefSuccess);
|
|||
|
}
|
|||
|
|
|||
|
Contract.Assert(addRefSuccess, "zlibLibraryHandle.DangerousAddRef in InflateInit2_ should always succeed, but it did not.");
|
|||
|
|
|||
|
return errC;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public ErrorCode Inflate(FlushCode flush) {
|
|||
|
|
|||
|
EnsureNotDisposed();
|
|||
|
EnsureState(State.InitializedForInflate);
|
|||
|
|
|||
|
return NativeZLibDLLStub.inflateDelegate(ref zStream, flush);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public ErrorCode InflateEnd() {
|
|||
|
|
|||
|
EnsureNotDisposed();
|
|||
|
EnsureState(State.InitializedForInflate);
|
|||
|
|
|||
|
ErrorCode errC;
|
|||
|
|
|||
|
#if !FEATURE_NETCORE
|
|||
|
RuntimeHelpers.PrepareConstrainedRegions();
|
|||
|
#endif
|
|||
|
try { } finally {
|
|||
|
|
|||
|
errC = NativeZLibDLLStub.inflateEndDelegate(ref zStream);
|
|||
|
initializationState = State.Disposed;
|
|||
|
zlibLibraryHandle.DangerousRelease();
|
|||
|
}
|
|||
|
|
|||
|
return errC;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public string GetErrorMessage() {
|
|||
|
|
|||
|
// This can work even after XxflateEnd().
|
|||
|
|
|||
|
if (ZNullPtr.Equals(zStream.msg))
|
|||
|
return String.Empty;
|
|||
|
|
|||
|
unsafe {
|
|||
|
String msgStr = new String((SByte*) zStream.msg);
|
|||
|
return msgStr;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
#endregion // Expose ZLib functions for use by user / Fx code (add more as required)
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
internal static Int32 ZLibCompileFlags() {
|
|||
|
|
|||
|
return NativeZLibDLLStub.zlibCompileFlagsDelegate();
|
|||
|
}
|
|||
|
|
|||
|
} // class ZLibStreamHandle
|
|||
|
|
|||
|
#endregion // ZLib Stream Handle type
|
|||
|
|
|||
|
|
|||
|
#region public factory methods for ZLibStreamHandle
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public static ErrorCode CreateZLibStreamForDeflate(out ZLibStreamHandle zLibStreamHandle) {
|
|||
|
|
|||
|
return CreateZLibStreamForDeflate(out zLibStreamHandle,
|
|||
|
CompressionLevel.DefaultCompression, Deflate_DefaultWindowBits,
|
|||
|
Deflate_DefaultMemLevel, CompressionStrategy.DefaultStrategy);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public static ErrorCode CreateZLibStreamForDeflate(out ZLibStreamHandle zLibStreamHandle,
|
|||
|
CompressionLevel level, int windowBits, int memLevel, CompressionStrategy strategy) {
|
|||
|
|
|||
|
zLibStreamHandle = new ZLibStreamHandle();
|
|||
|
return zLibStreamHandle.DeflateInit2_(level, windowBits, memLevel, strategy);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public static ErrorCode CreateZLibStreamForInflate(out ZLibStreamHandle zLibStreamHandle) {
|
|||
|
|
|||
|
return CreateZLibStreamForInflate(out zLibStreamHandle, Deflate_DefaultWindowBits);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public static ErrorCode CreateZLibStreamForInflate(out ZLibStreamHandle zLibStreamHandle, int windowBits) {
|
|||
|
|
|||
|
zLibStreamHandle = new ZLibStreamHandle();
|
|||
|
return zLibStreamHandle.InflateInit2_(windowBits);
|
|||
|
}
|
|||
|
|
|||
|
#endregion // public factory methods for ZLibStreamHandle
|
|||
|
|
|||
|
|
|||
|
#region public utility APIs
|
|||
|
|
|||
|
[SecurityCritical]
|
|||
|
public static Int32 ZLibCompileFlags() {
|
|||
|
|
|||
|
return ZLibStreamHandle.ZLibCompileFlags();
|
|||
|
}
|
|||
|
|
|||
|
#endregion // public utility APIs
|
|||
|
|
|||
|
} // internal class ZLibNative
|
|||
|
|
|||
|
} // namespace System.IO.Compression
|
|||
|
|
|||
|
// file ZLibNative.cs
|