123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563 |
- /* This file is part of SevenZipSharp.
- SevenZipSharp is free software: you can redistribute it and/or modify
- it under the terms of the GNU Lesser General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
- SevenZipSharp is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU Lesser General Public License for more details.
- You should have received a copy of the GNU Lesser General Public License
- along with SevenZipSharp. If not, see <http://www.gnu.org/licenses/>.
- */
- using System;
- using System.Collections.Generic;
- #if !WINCE && !MONO
- using System.Configuration;
- using System.Diagnostics;
- using System.Security.Permissions;
- #endif
- #if WINCE
- using OpenNETCF.Diagnostics;
- #endif
- using System.IO;
- using System.Reflection;
- using System.Runtime.InteropServices;
- using System.Text;
- #if MONO
- using SevenZip.Mono.COM;
- #endif
- namespace SevenZip
- {
- #if UNMANAGED
- /// <summary>
- /// 7-zip library low-level wrapper.
- /// </summary>
- internal static class SevenZipLibraryManager
- {
- #if !WINCE && !MONO
- /// <summary>
- /// Path to the 7-zip dll.
- /// </summary>
- /// <remarks>7zxa.dll supports only decoding from .7z archives.
- /// Features of 7za.dll:
- /// - Supporting 7z format;
- /// - Built encoders: LZMA, PPMD, BCJ, BCJ2, COPY, AES-256 Encryption.
- /// - Built decoders: LZMA, PPMD, BCJ, BCJ2, COPY, AES-256 Encryption, BZip2, Deflate.
- /// 7z.dll (from the 7-zip distribution) supports every InArchiveFormat for encoding and decoding.
- /// </remarks>
- private static string _libraryFileName = ConfigurationManager.AppSettings["7zLocation"] ??
- Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "7z.dll");
- #endif
- #if WINCE
- private static string _libraryFileName =
- Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().GetName().CodeBase), "7z.dll");
- #endif
- /// <summary>
- /// 7-zip library handle.
- /// </summary>
- private static IntPtr _modulePtr;
- /// <summary>
- /// 7-zip library features.
- /// </summary>
- private static LibraryFeature? _features;
- private static Dictionary<object, Dictionary<InArchiveFormat, IInArchive>> _inArchives;
- #if COMPRESS
- private static Dictionary<object, Dictionary<OutArchiveFormat, IOutArchive>> _outArchives;
- #endif
- private static int _totalUsers;
- // private static string _LibraryVersion;
- private static bool? _modifyCapabale;
- private static void InitUserInFormat(object user, InArchiveFormat format)
- {
- if (!_inArchives.ContainsKey(user))
- {
- _inArchives.Add(user, new Dictionary<InArchiveFormat, IInArchive>());
- }
- if (!_inArchives[user].ContainsKey(format))
- {
- _inArchives[user].Add(format, null);
- _totalUsers++;
- }
- }
- #if COMPRESS
- private static void InitUserOutFormat(object user, OutArchiveFormat format)
- {
- if (!_outArchives.ContainsKey(user))
- {
- _outArchives.Add(user, new Dictionary<OutArchiveFormat, IOutArchive>());
- }
- if (!_outArchives[user].ContainsKey(format))
- {
- _outArchives[user].Add(format, null);
- _totalUsers++;
- }
- }
- #endif
- private static void Init()
- {
- _inArchives = new Dictionary<object, Dictionary<InArchiveFormat, IInArchive>>();
- #if COMPRESS
- _outArchives = new Dictionary<object, Dictionary<OutArchiveFormat, IOutArchive>>();
- #endif
- }
- /// <summary>
- /// Loads the 7-zip library if necessary and adds user to the reference list
- /// </summary>
- /// <param name="user">Caller of the function</param>
- /// <param name="format">Archive format</param>
- public static void LoadLibrary(object user, Enum format)
- {
- if (_inArchives == null
- #if COMPRESS
- || _outArchives == null
- #endif
- )
- {
- Init();
- }
- #if !WINCE && !MONO
- if (_modulePtr == IntPtr.Zero)
- {
- if (!File.Exists(_libraryFileName))
- {
- throw new SevenZipLibraryException("DLL file does not exist.");
- }
- if ((_modulePtr = NativeMethods.LoadLibrary(_libraryFileName)) == IntPtr.Zero)
- {
- throw new SevenZipLibraryException("failed to load library.");
- }
- if (NativeMethods.GetProcAddress(_modulePtr, "GetHandlerProperty") == IntPtr.Zero)
- {
- NativeMethods.FreeLibrary(_modulePtr);
- throw new SevenZipLibraryException("library is invalid.");
- }
- }
- #endif
- if (format is InArchiveFormat)
- {
- InitUserInFormat(user, (InArchiveFormat) format);
- return;
- }
- #if COMPRESS
- if (format is OutArchiveFormat)
- {
- InitUserOutFormat(user, (OutArchiveFormat) format);
- return;
- }
- #endif
- throw new ArgumentException(
- "Enum " + format + " is not a valid archive format attribute!");
- }
- /*/// <summary>
- /// Gets the native 7zip library version string.
- /// </summary>
- public static string LibraryVersion
- {
- get
- {
- if (String.IsNullOrEmpty(_LibraryVersion))
- {
- FileVersionInfo dllVersionInfo = FileVersionInfo.GetVersionInfo(_libraryFileName);
- _LibraryVersion = String.Format(
- System.Globalization.CultureInfo.CurrentCulture,
- "{0}.{1}",
- dllVersionInfo.FileMajorPart, dllVersionInfo.FileMinorPart);
- }
- return _LibraryVersion;
- }
- }*/
- /// <summary>
- /// Gets the value indicating whether the library supports modifying archives.
- /// </summary>
- public static bool ModifyCapable
- {
- get
- {
- if (!_modifyCapabale.HasValue)
- {
- #if !WINCE && !MONO
- FileVersionInfo dllVersionInfo = FileVersionInfo.GetVersionInfo(_libraryFileName);
- _modifyCapabale = dllVersionInfo.FileMajorPart >= 9;
- #else
- _modifyCapabale = true;
- #endif
- }
- return _modifyCapabale.Value;
- }
- }
-
- static readonly string Namespace = Assembly.GetExecutingAssembly().GetManifestResourceNames()[0].Split('.')[0];
- private static string GetResourceString(string str)
- {
- return Namespace + ".arch." + str;
- }
- private static bool ExtractionBenchmark(string archiveFileName, Stream outStream,
- ref LibraryFeature? features, LibraryFeature testedFeature)
- {
- var stream = Assembly.GetExecutingAssembly().GetManifestResourceStream(
- GetResourceString(archiveFileName));
- try
- {
- using (var extr = new SevenZipExtractor(stream))
- {
- extr.ExtractFile(0, outStream);
- }
- }
- catch(Exception)
- {
- return false;
- }
- features |= testedFeature;
- return true;
- }
- private static bool CompressionBenchmark(Stream inStream, Stream outStream,
- OutArchiveFormat format, CompressionMethod method,
- ref LibraryFeature? features, LibraryFeature testedFeature)
- {
- try
- {
- var compr = new SevenZipCompressor {ArchiveFormat = format, CompressionMethod = method};
- compr.CompressStream(inStream, outStream);
- }
- catch (Exception)
- {
- return false;
- }
- features |= testedFeature;
- return true;
- }
- public static LibraryFeature CurrentLibraryFeatures
- {
- get
- {
- if (_features != null && _features.HasValue)
- {
- return _features.Value;
- }
- _features = LibraryFeature.None;
- #region Benchmark
- #region Extraction features
- using (var outStream = new MemoryStream())
- {
- ExtractionBenchmark("Test.lzma.7z", outStream, ref _features, LibraryFeature.Extract7z);
- ExtractionBenchmark("Test.lzma2.7z", outStream, ref _features, LibraryFeature.Extract7zLZMA2);
- int i = 0;
- if (ExtractionBenchmark("Test.bzip2.7z", outStream, ref _features, _features.Value))
- {
- i++;
- }
- if (ExtractionBenchmark("Test.ppmd.7z", outStream, ref _features, _features.Value))
- {
- i++;
- if (i == 2 && (_features & LibraryFeature.Extract7z) != 0 &&
- (_features & LibraryFeature.Extract7zLZMA2) != 0)
- {
- _features |= LibraryFeature.Extract7zAll;
- }
- }
- ExtractionBenchmark("Test.rar", outStream, ref _features, LibraryFeature.ExtractRar);
- ExtractionBenchmark("Test.tar", outStream, ref _features, LibraryFeature.ExtractTar);
- ExtractionBenchmark("Test.txt.bz2", outStream, ref _features, LibraryFeature.ExtractBzip2);
- ExtractionBenchmark("Test.txt.gz", outStream, ref _features, LibraryFeature.ExtractGzip);
- ExtractionBenchmark("Test.txt.xz", outStream, ref _features, LibraryFeature.ExtractXz);
- ExtractionBenchmark("Test.zip", outStream, ref _features, LibraryFeature.ExtractZip);
- }
- #endregion
- #region Compression features
- using (var inStream = new MemoryStream())
- {
- inStream.Write(Encoding.UTF8.GetBytes("Test"), 0, 4);
- using (var outStream = new MemoryStream())
- {
- CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.SevenZip, CompressionMethod.Lzma,
- ref _features, LibraryFeature.Compress7z);
- CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.SevenZip, CompressionMethod.Lzma2,
- ref _features, LibraryFeature.Compress7zLZMA2);
- int i = 0;
- if (CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.SevenZip, CompressionMethod.BZip2,
- ref _features, _features.Value))
- {
- i++;
- }
- if (CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.SevenZip, CompressionMethod.Ppmd,
- ref _features, _features.Value))
- {
- i++;
- if (i == 2 && (_features & LibraryFeature.Compress7z) != 0 &&
- (_features & LibraryFeature.Compress7zLZMA2) != 0)
- {
- _features |= LibraryFeature.Compress7zAll;
- }
- }
- CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.Zip, CompressionMethod.Default,
- ref _features, LibraryFeature.CompressZip);
- CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.BZip2, CompressionMethod.Default,
- ref _features, LibraryFeature.CompressBzip2);
- CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.GZip, CompressionMethod.Default,
- ref _features, LibraryFeature.CompressGzip);
- CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.Tar, CompressionMethod.Default,
- ref _features, LibraryFeature.CompressTar);
- CompressionBenchmark(inStream, outStream,
- OutArchiveFormat.XZ, CompressionMethod.Default,
- ref _features, LibraryFeature.CompressXz);
- }
- }
- #endregion
- #endregion
- if (ModifyCapable && (_features.Value & LibraryFeature.Compress7z) != 0)
- {
- _features |= LibraryFeature.Modify;
- }
- return _features.Value;
- }
- }
- /// <summary>
- /// Removes user from reference list and frees the 7-zip library if it becomes empty
- /// </summary>
- /// <param name="user">Caller of the function</param>
- /// <param name="format">Archive format</param>
- public static void FreeLibrary(object user, Enum format)
- {
- #if !WINCE && !MONO
- var sp = new SecurityPermission(SecurityPermissionFlag.UnmanagedCode);
- sp.Demand();
- #endif
- if (_modulePtr != IntPtr.Zero)
- {
- if (format is InArchiveFormat)
- {
- if (_inArchives != null && _inArchives.ContainsKey(user) &&
- _inArchives[user].ContainsKey((InArchiveFormat) format) &&
- _inArchives[user][(InArchiveFormat) format] != null)
- {
- try
- {
- Marshal.ReleaseComObject(_inArchives[user][(InArchiveFormat) format]);
- }
- catch (InvalidComObjectException) {}
- _inArchives[user].Remove((InArchiveFormat) format);
- _totalUsers--;
- if (_inArchives[user].Count == 0)
- {
- _inArchives.Remove(user);
- }
- }
- }
- #if COMPRESS
- if (format is OutArchiveFormat)
- {
- if (_outArchives != null && _outArchives.ContainsKey(user) &&
- _outArchives[user].ContainsKey((OutArchiveFormat) format) &&
- _outArchives[user][(OutArchiveFormat) format] != null)
- {
- try
- {
- Marshal.ReleaseComObject(_outArchives[user][(OutArchiveFormat) format]);
- }
- catch (InvalidComObjectException) {}
- _outArchives[user].Remove((OutArchiveFormat) format);
- _totalUsers--;
- if (_outArchives[user].Count == 0)
- {
- _outArchives.Remove(user);
- }
- }
- }
- #endif
- if ((_inArchives == null || _inArchives.Count == 0)
- #if COMPRESS
- && (_outArchives == null || _outArchives.Count == 0)
- #endif
- )
- {
- _inArchives = null;
- #if COMPRESS
- _outArchives = null;
- #endif
- if (_totalUsers == 0)
- {
- #if !WINCE && !MONO
- NativeMethods.FreeLibrary(_modulePtr);
- #endif
- _modulePtr = IntPtr.Zero;
- }
- }
- }
- }
- /// <summary>
- /// Gets IInArchive interface to extract 7-zip archives.
- /// </summary>
- /// <param name="format">Archive format.</param>
- /// <param name="user">Archive format user.</param>
- public static IInArchive InArchive(InArchiveFormat format, object user)
- {
- #if !WINCE && !MONO
- lock (_libraryFileName)
- {
- #endif
- if (_inArchives[user][format] == null)
- {
- #if !WINCE && !MONO
- var sp = new SecurityPermission(SecurityPermissionFlag.UnmanagedCode);
- sp.Demand();
- if (_modulePtr == IntPtr.Zero)
- {
- LoadLibrary(user, format);
- if (_modulePtr == IntPtr.Zero)
- {
- throw new SevenZipLibraryException();
- }
- }
- var createObject = (NativeMethods.CreateObjectDelegate)
- Marshal.GetDelegateForFunctionPointer(
- NativeMethods.GetProcAddress(_modulePtr, "CreateObject"),
- typeof(NativeMethods.CreateObjectDelegate));
- if (createObject == null)
- {
- throw new SevenZipLibraryException();
- }
- #endif
- object result;
- Guid interfaceId =
- #if !WINCE && !MONO
- typeof(IInArchive).GUID;
- #else
- new Guid(((GuidAttribute)typeof(IInArchive).GetCustomAttributes(typeof(GuidAttribute), false)[0]).Value);
- #endif
- Guid classID = Formats.InFormatGuids[format];
- try
- {
- #if !WINCE && !MONO
- createObject(ref classID, ref interfaceId, out result);
- #elif !MONO
- NativeMethods.CreateCOMObject(ref classID, ref interfaceId, out result);
- #else
- result = SevenZip.Mono.Factory.CreateInterface<IInArchive>(user, classID, interfaceId);
- #endif
- }
- catch (Exception)
- {
- throw new SevenZipLibraryException("Your 7-zip library does not support this archive type.");
- }
- InitUserInFormat(user, format);
- _inArchives[user][format] = result as IInArchive;
- }
- #if !WINCE && !MONO
- }
- #endif
- return _inArchives[user][format];
- }
- #if COMPRESS
- /// <summary>
- /// Gets IOutArchive interface to pack 7-zip archives.
- /// </summary>
- /// <param name="format">Archive format.</param>
- /// <param name="user">Archive format user.</param>
- public static IOutArchive OutArchive(OutArchiveFormat format, object user)
- {
- #if !WINCE && !MONO
- lock (_libraryFileName)
- {
- #endif
- if (_outArchives[user][format] == null)
- {
- #if !WINCE && !MONO
- var sp = new SecurityPermission(SecurityPermissionFlag.UnmanagedCode);
- sp.Demand();
- if (_modulePtr == IntPtr.Zero)
- {
- throw new SevenZipLibraryException();
- }
- var createObject = (NativeMethods.CreateObjectDelegate)
- Marshal.GetDelegateForFunctionPointer(
- NativeMethods.GetProcAddress(_modulePtr, "CreateObject"),
- typeof(NativeMethods.CreateObjectDelegate));
- if (createObject == null)
- {
- throw new SevenZipLibraryException();
- }
- #endif
- object result;
- Guid interfaceId =
- #if !WINCE && !MONO
- typeof(IOutArchive).GUID;
- #else
- new Guid(((GuidAttribute)typeof(IOutArchive).GetCustomAttributes(typeof(GuidAttribute), false)[0]).Value);
- #endif
- Guid classID = Formats.OutFormatGuids[format];
- try
- {
- #if !WINCE && !MONO
- createObject(ref classID, ref interfaceId, out result);
- #elif !MONO
- NativeMethods.CreateCOMObject(ref classID, ref interfaceId, out result);
- #else
- result = SevenZip.Mono.Factory.CreateInterface<IOutArchive>(classID, interfaceId, user);
- #endif
- }
- catch (Exception)
- {
- throw new SevenZipLibraryException("Your 7-zip library does not support this archive type.");
- }
- InitUserOutFormat(user, format);
- _outArchives[user][format] = result as IOutArchive;
- }
- #if !WINCE && !MONO
- }
- #endif
- return _outArchives[user][format];
- }
- #endif
- #if !WINCE && !MONO
- public static void SetLibraryPath(string libraryPath)
- {
- if (_modulePtr != IntPtr.Zero && !Path.GetFullPath(libraryPath).Equals(
- Path.GetFullPath(_libraryFileName), StringComparison.OrdinalIgnoreCase))
- {
- throw new SevenZipLibraryException(
- "can not change the library path while the library \"" + _libraryFileName + "\" is being used.");
- }
- if (!File.Exists(libraryPath))
- {
- throw new SevenZipLibraryException(
- "can not change the library path because the file \"" + libraryPath + "\" does not exist.");
- }
- _libraryFileName = libraryPath;
- _features = null;
- }
- #endif
- }
- #endif
- }
|