LibraryManager.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563
  1. /* This file is part of SevenZipSharp.
  2. SevenZipSharp is free software: you can redistribute it and/or modify
  3. it under the terms of the GNU Lesser General Public License as published by
  4. the Free Software Foundation, either version 3 of the License, or
  5. (at your option) any later version.
  6. SevenZipSharp is distributed in the hope that it will be useful,
  7. but WITHOUT ANY WARRANTY; without even the implied warranty of
  8. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  9. GNU Lesser General Public License for more details.
  10. You should have received a copy of the GNU Lesser General Public License
  11. along with SevenZipSharp. If not, see <http://www.gnu.org/licenses/>.
  12. */
  13. using System;
  14. using System.Collections.Generic;
  15. #if !WINCE && !MONO
  16. using System.Configuration;
  17. using System.Diagnostics;
  18. using System.Security.Permissions;
  19. #endif
  20. #if WINCE
  21. using OpenNETCF.Diagnostics;
  22. #endif
  23. using System.IO;
  24. using System.Reflection;
  25. using System.Runtime.InteropServices;
  26. using System.Text;
  27. #if MONO
  28. using SevenZip.Mono.COM;
  29. #endif
  30. namespace SevenZip
  31. {
  32. #if UNMANAGED
  33. /// <summary>
  34. /// 7-zip library low-level wrapper.
  35. /// </summary>
  36. internal static class SevenZipLibraryManager
  37. {
  38. #if !WINCE && !MONO
  39. /// <summary>
  40. /// Path to the 7-zip dll.
  41. /// </summary>
  42. /// <remarks>7zxa.dll supports only decoding from .7z archives.
  43. /// Features of 7za.dll:
  44. /// - Supporting 7z format;
  45. /// - Built encoders: LZMA, PPMD, BCJ, BCJ2, COPY, AES-256 Encryption.
  46. /// - Built decoders: LZMA, PPMD, BCJ, BCJ2, COPY, AES-256 Encryption, BZip2, Deflate.
  47. /// 7z.dll (from the 7-zip distribution) supports every InArchiveFormat for encoding and decoding.
  48. /// </remarks>
  49. private static string _libraryFileName = ConfigurationManager.AppSettings["7zLocation"] ??
  50. Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "7z.dll");
  51. #endif
  52. #if WINCE
  53. private static string _libraryFileName =
  54. Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().GetName().CodeBase), "7z.dll");
  55. #endif
  56. /// <summary>
  57. /// 7-zip library handle.
  58. /// </summary>
  59. private static IntPtr _modulePtr;
  60. /// <summary>
  61. /// 7-zip library features.
  62. /// </summary>
  63. private static LibraryFeature? _features;
  64. private static Dictionary<object, Dictionary<InArchiveFormat, IInArchive>> _inArchives;
  65. #if COMPRESS
  66. private static Dictionary<object, Dictionary<OutArchiveFormat, IOutArchive>> _outArchives;
  67. #endif
  68. private static int _totalUsers;
  69. // private static string _LibraryVersion;
  70. private static bool? _modifyCapabale;
  71. private static void InitUserInFormat(object user, InArchiveFormat format)
  72. {
  73. if (!_inArchives.ContainsKey(user))
  74. {
  75. _inArchives.Add(user, new Dictionary<InArchiveFormat, IInArchive>());
  76. }
  77. if (!_inArchives[user].ContainsKey(format))
  78. {
  79. _inArchives[user].Add(format, null);
  80. _totalUsers++;
  81. }
  82. }
  83. #if COMPRESS
  84. private static void InitUserOutFormat(object user, OutArchiveFormat format)
  85. {
  86. if (!_outArchives.ContainsKey(user))
  87. {
  88. _outArchives.Add(user, new Dictionary<OutArchiveFormat, IOutArchive>());
  89. }
  90. if (!_outArchives[user].ContainsKey(format))
  91. {
  92. _outArchives[user].Add(format, null);
  93. _totalUsers++;
  94. }
  95. }
  96. #endif
  97. private static void Init()
  98. {
  99. _inArchives = new Dictionary<object, Dictionary<InArchiveFormat, IInArchive>>();
  100. #if COMPRESS
  101. _outArchives = new Dictionary<object, Dictionary<OutArchiveFormat, IOutArchive>>();
  102. #endif
  103. }
  104. /// <summary>
  105. /// Loads the 7-zip library if necessary and adds user to the reference list
  106. /// </summary>
  107. /// <param name="user">Caller of the function</param>
  108. /// <param name="format">Archive format</param>
  109. public static void LoadLibrary(object user, Enum format)
  110. {
  111. if (_inArchives == null
  112. #if COMPRESS
  113. || _outArchives == null
  114. #endif
  115. )
  116. {
  117. Init();
  118. }
  119. #if !WINCE && !MONO
  120. if (_modulePtr == IntPtr.Zero)
  121. {
  122. if (!File.Exists(_libraryFileName))
  123. {
  124. throw new SevenZipLibraryException("DLL file does not exist.");
  125. }
  126. if ((_modulePtr = NativeMethods.LoadLibrary(_libraryFileName)) == IntPtr.Zero)
  127. {
  128. throw new SevenZipLibraryException("failed to load library.");
  129. }
  130. if (NativeMethods.GetProcAddress(_modulePtr, "GetHandlerProperty") == IntPtr.Zero)
  131. {
  132. NativeMethods.FreeLibrary(_modulePtr);
  133. throw new SevenZipLibraryException("library is invalid.");
  134. }
  135. }
  136. #endif
  137. if (format is InArchiveFormat)
  138. {
  139. InitUserInFormat(user, (InArchiveFormat) format);
  140. return;
  141. }
  142. #if COMPRESS
  143. if (format is OutArchiveFormat)
  144. {
  145. InitUserOutFormat(user, (OutArchiveFormat) format);
  146. return;
  147. }
  148. #endif
  149. throw new ArgumentException(
  150. "Enum " + format + " is not a valid archive format attribute!");
  151. }
  152. /*/// <summary>
  153. /// Gets the native 7zip library version string.
  154. /// </summary>
  155. public static string LibraryVersion
  156. {
  157. get
  158. {
  159. if (String.IsNullOrEmpty(_LibraryVersion))
  160. {
  161. FileVersionInfo dllVersionInfo = FileVersionInfo.GetVersionInfo(_libraryFileName);
  162. _LibraryVersion = String.Format(
  163. System.Globalization.CultureInfo.CurrentCulture,
  164. "{0}.{1}",
  165. dllVersionInfo.FileMajorPart, dllVersionInfo.FileMinorPart);
  166. }
  167. return _LibraryVersion;
  168. }
  169. }*/
  170. /// <summary>
  171. /// Gets the value indicating whether the library supports modifying archives.
  172. /// </summary>
  173. public static bool ModifyCapable
  174. {
  175. get
  176. {
  177. if (!_modifyCapabale.HasValue)
  178. {
  179. #if !WINCE && !MONO
  180. FileVersionInfo dllVersionInfo = FileVersionInfo.GetVersionInfo(_libraryFileName);
  181. _modifyCapabale = dllVersionInfo.FileMajorPart >= 9;
  182. #else
  183. _modifyCapabale = true;
  184. #endif
  185. }
  186. return _modifyCapabale.Value;
  187. }
  188. }
  189. static readonly string Namespace = Assembly.GetExecutingAssembly().GetManifestResourceNames()[0].Split('.')[0];
  190. private static string GetResourceString(string str)
  191. {
  192. return Namespace + ".arch." + str;
  193. }
  194. private static bool ExtractionBenchmark(string archiveFileName, Stream outStream,
  195. ref LibraryFeature? features, LibraryFeature testedFeature)
  196. {
  197. var stream = Assembly.GetExecutingAssembly().GetManifestResourceStream(
  198. GetResourceString(archiveFileName));
  199. try
  200. {
  201. using (var extr = new SevenZipExtractor(stream))
  202. {
  203. extr.ExtractFile(0, outStream);
  204. }
  205. }
  206. catch(Exception)
  207. {
  208. return false;
  209. }
  210. features |= testedFeature;
  211. return true;
  212. }
  213. private static bool CompressionBenchmark(Stream inStream, Stream outStream,
  214. OutArchiveFormat format, CompressionMethod method,
  215. ref LibraryFeature? features, LibraryFeature testedFeature)
  216. {
  217. try
  218. {
  219. var compr = new SevenZipCompressor {ArchiveFormat = format, CompressionMethod = method};
  220. compr.CompressStream(inStream, outStream);
  221. }
  222. catch (Exception)
  223. {
  224. return false;
  225. }
  226. features |= testedFeature;
  227. return true;
  228. }
  229. public static LibraryFeature CurrentLibraryFeatures
  230. {
  231. get
  232. {
  233. if (_features != null && _features.HasValue)
  234. {
  235. return _features.Value;
  236. }
  237. _features = LibraryFeature.None;
  238. #region Benchmark
  239. #region Extraction features
  240. using (var outStream = new MemoryStream())
  241. {
  242. ExtractionBenchmark("Test.lzma.7z", outStream, ref _features, LibraryFeature.Extract7z);
  243. ExtractionBenchmark("Test.lzma2.7z", outStream, ref _features, LibraryFeature.Extract7zLZMA2);
  244. int i = 0;
  245. if (ExtractionBenchmark("Test.bzip2.7z", outStream, ref _features, _features.Value))
  246. {
  247. i++;
  248. }
  249. if (ExtractionBenchmark("Test.ppmd.7z", outStream, ref _features, _features.Value))
  250. {
  251. i++;
  252. if (i == 2 && (_features & LibraryFeature.Extract7z) != 0 &&
  253. (_features & LibraryFeature.Extract7zLZMA2) != 0)
  254. {
  255. _features |= LibraryFeature.Extract7zAll;
  256. }
  257. }
  258. ExtractionBenchmark("Test.rar", outStream, ref _features, LibraryFeature.ExtractRar);
  259. ExtractionBenchmark("Test.tar", outStream, ref _features, LibraryFeature.ExtractTar);
  260. ExtractionBenchmark("Test.txt.bz2", outStream, ref _features, LibraryFeature.ExtractBzip2);
  261. ExtractionBenchmark("Test.txt.gz", outStream, ref _features, LibraryFeature.ExtractGzip);
  262. ExtractionBenchmark("Test.txt.xz", outStream, ref _features, LibraryFeature.ExtractXz);
  263. ExtractionBenchmark("Test.zip", outStream, ref _features, LibraryFeature.ExtractZip);
  264. }
  265. #endregion
  266. #region Compression features
  267. using (var inStream = new MemoryStream())
  268. {
  269. inStream.Write(Encoding.UTF8.GetBytes("Test"), 0, 4);
  270. using (var outStream = new MemoryStream())
  271. {
  272. CompressionBenchmark(inStream, outStream,
  273. OutArchiveFormat.SevenZip, CompressionMethod.Lzma,
  274. ref _features, LibraryFeature.Compress7z);
  275. CompressionBenchmark(inStream, outStream,
  276. OutArchiveFormat.SevenZip, CompressionMethod.Lzma2,
  277. ref _features, LibraryFeature.Compress7zLZMA2);
  278. int i = 0;
  279. if (CompressionBenchmark(inStream, outStream,
  280. OutArchiveFormat.SevenZip, CompressionMethod.BZip2,
  281. ref _features, _features.Value))
  282. {
  283. i++;
  284. }
  285. if (CompressionBenchmark(inStream, outStream,
  286. OutArchiveFormat.SevenZip, CompressionMethod.Ppmd,
  287. ref _features, _features.Value))
  288. {
  289. i++;
  290. if (i == 2 && (_features & LibraryFeature.Compress7z) != 0 &&
  291. (_features & LibraryFeature.Compress7zLZMA2) != 0)
  292. {
  293. _features |= LibraryFeature.Compress7zAll;
  294. }
  295. }
  296. CompressionBenchmark(inStream, outStream,
  297. OutArchiveFormat.Zip, CompressionMethod.Default,
  298. ref _features, LibraryFeature.CompressZip);
  299. CompressionBenchmark(inStream, outStream,
  300. OutArchiveFormat.BZip2, CompressionMethod.Default,
  301. ref _features, LibraryFeature.CompressBzip2);
  302. CompressionBenchmark(inStream, outStream,
  303. OutArchiveFormat.GZip, CompressionMethod.Default,
  304. ref _features, LibraryFeature.CompressGzip);
  305. CompressionBenchmark(inStream, outStream,
  306. OutArchiveFormat.Tar, CompressionMethod.Default,
  307. ref _features, LibraryFeature.CompressTar);
  308. CompressionBenchmark(inStream, outStream,
  309. OutArchiveFormat.XZ, CompressionMethod.Default,
  310. ref _features, LibraryFeature.CompressXz);
  311. }
  312. }
  313. #endregion
  314. #endregion
  315. if (ModifyCapable && (_features.Value & LibraryFeature.Compress7z) != 0)
  316. {
  317. _features |= LibraryFeature.Modify;
  318. }
  319. return _features.Value;
  320. }
  321. }
  322. /// <summary>
  323. /// Removes user from reference list and frees the 7-zip library if it becomes empty
  324. /// </summary>
  325. /// <param name="user">Caller of the function</param>
  326. /// <param name="format">Archive format</param>
  327. public static void FreeLibrary(object user, Enum format)
  328. {
  329. #if !WINCE && !MONO
  330. var sp = new SecurityPermission(SecurityPermissionFlag.UnmanagedCode);
  331. sp.Demand();
  332. #endif
  333. if (_modulePtr != IntPtr.Zero)
  334. {
  335. if (format is InArchiveFormat)
  336. {
  337. if (_inArchives != null && _inArchives.ContainsKey(user) &&
  338. _inArchives[user].ContainsKey((InArchiveFormat) format) &&
  339. _inArchives[user][(InArchiveFormat) format] != null)
  340. {
  341. try
  342. {
  343. Marshal.ReleaseComObject(_inArchives[user][(InArchiveFormat) format]);
  344. }
  345. catch (InvalidComObjectException) {}
  346. _inArchives[user].Remove((InArchiveFormat) format);
  347. _totalUsers--;
  348. if (_inArchives[user].Count == 0)
  349. {
  350. _inArchives.Remove(user);
  351. }
  352. }
  353. }
  354. #if COMPRESS
  355. if (format is OutArchiveFormat)
  356. {
  357. if (_outArchives != null && _outArchives.ContainsKey(user) &&
  358. _outArchives[user].ContainsKey((OutArchiveFormat) format) &&
  359. _outArchives[user][(OutArchiveFormat) format] != null)
  360. {
  361. try
  362. {
  363. Marshal.ReleaseComObject(_outArchives[user][(OutArchiveFormat) format]);
  364. }
  365. catch (InvalidComObjectException) {}
  366. _outArchives[user].Remove((OutArchiveFormat) format);
  367. _totalUsers--;
  368. if (_outArchives[user].Count == 0)
  369. {
  370. _outArchives.Remove(user);
  371. }
  372. }
  373. }
  374. #endif
  375. if ((_inArchives == null || _inArchives.Count == 0)
  376. #if COMPRESS
  377. && (_outArchives == null || _outArchives.Count == 0)
  378. #endif
  379. )
  380. {
  381. _inArchives = null;
  382. #if COMPRESS
  383. _outArchives = null;
  384. #endif
  385. if (_totalUsers == 0)
  386. {
  387. #if !WINCE && !MONO
  388. NativeMethods.FreeLibrary(_modulePtr);
  389. #endif
  390. _modulePtr = IntPtr.Zero;
  391. }
  392. }
  393. }
  394. }
  395. /// <summary>
  396. /// Gets IInArchive interface to extract 7-zip archives.
  397. /// </summary>
  398. /// <param name="format">Archive format.</param>
  399. /// <param name="user">Archive format user.</param>
  400. public static IInArchive InArchive(InArchiveFormat format, object user)
  401. {
  402. #if !WINCE && !MONO
  403. lock (_libraryFileName)
  404. {
  405. #endif
  406. if (_inArchives[user][format] == null)
  407. {
  408. #if !WINCE && !MONO
  409. var sp = new SecurityPermission(SecurityPermissionFlag.UnmanagedCode);
  410. sp.Demand();
  411. if (_modulePtr == IntPtr.Zero)
  412. {
  413. LoadLibrary(user, format);
  414. if (_modulePtr == IntPtr.Zero)
  415. {
  416. throw new SevenZipLibraryException();
  417. }
  418. }
  419. var createObject = (NativeMethods.CreateObjectDelegate)
  420. Marshal.GetDelegateForFunctionPointer(
  421. NativeMethods.GetProcAddress(_modulePtr, "CreateObject"),
  422. typeof(NativeMethods.CreateObjectDelegate));
  423. if (createObject == null)
  424. {
  425. throw new SevenZipLibraryException();
  426. }
  427. #endif
  428. object result;
  429. Guid interfaceId =
  430. #if !WINCE && !MONO
  431. typeof(IInArchive).GUID;
  432. #else
  433. new Guid(((GuidAttribute)typeof(IInArchive).GetCustomAttributes(typeof(GuidAttribute), false)[0]).Value);
  434. #endif
  435. Guid classID = Formats.InFormatGuids[format];
  436. try
  437. {
  438. #if !WINCE && !MONO
  439. createObject(ref classID, ref interfaceId, out result);
  440. #elif !MONO
  441. NativeMethods.CreateCOMObject(ref classID, ref interfaceId, out result);
  442. #else
  443. result = SevenZip.Mono.Factory.CreateInterface<IInArchive>(user, classID, interfaceId);
  444. #endif
  445. }
  446. catch (Exception)
  447. {
  448. throw new SevenZipLibraryException("Your 7-zip library does not support this archive type.");
  449. }
  450. InitUserInFormat(user, format);
  451. _inArchives[user][format] = result as IInArchive;
  452. }
  453. #if !WINCE && !MONO
  454. }
  455. #endif
  456. return _inArchives[user][format];
  457. }
  458. #if COMPRESS
  459. /// <summary>
  460. /// Gets IOutArchive interface to pack 7-zip archives.
  461. /// </summary>
  462. /// <param name="format">Archive format.</param>
  463. /// <param name="user">Archive format user.</param>
  464. public static IOutArchive OutArchive(OutArchiveFormat format, object user)
  465. {
  466. #if !WINCE && !MONO
  467. lock (_libraryFileName)
  468. {
  469. #endif
  470. if (_outArchives[user][format] == null)
  471. {
  472. #if !WINCE && !MONO
  473. var sp = new SecurityPermission(SecurityPermissionFlag.UnmanagedCode);
  474. sp.Demand();
  475. if (_modulePtr == IntPtr.Zero)
  476. {
  477. throw new SevenZipLibraryException();
  478. }
  479. var createObject = (NativeMethods.CreateObjectDelegate)
  480. Marshal.GetDelegateForFunctionPointer(
  481. NativeMethods.GetProcAddress(_modulePtr, "CreateObject"),
  482. typeof(NativeMethods.CreateObjectDelegate));
  483. if (createObject == null)
  484. {
  485. throw new SevenZipLibraryException();
  486. }
  487. #endif
  488. object result;
  489. Guid interfaceId =
  490. #if !WINCE && !MONO
  491. typeof(IOutArchive).GUID;
  492. #else
  493. new Guid(((GuidAttribute)typeof(IOutArchive).GetCustomAttributes(typeof(GuidAttribute), false)[0]).Value);
  494. #endif
  495. Guid classID = Formats.OutFormatGuids[format];
  496. try
  497. {
  498. #if !WINCE && !MONO
  499. createObject(ref classID, ref interfaceId, out result);
  500. #elif !MONO
  501. NativeMethods.CreateCOMObject(ref classID, ref interfaceId, out result);
  502. #else
  503. result = SevenZip.Mono.Factory.CreateInterface<IOutArchive>(classID, interfaceId, user);
  504. #endif
  505. }
  506. catch (Exception)
  507. {
  508. throw new SevenZipLibraryException("Your 7-zip library does not support this archive type.");
  509. }
  510. InitUserOutFormat(user, format);
  511. _outArchives[user][format] = result as IOutArchive;
  512. }
  513. #if !WINCE && !MONO
  514. }
  515. #endif
  516. return _outArchives[user][format];
  517. }
  518. #endif
  519. #if !WINCE && !MONO
  520. public static void SetLibraryPath(string libraryPath)
  521. {
  522. if (_modulePtr != IntPtr.Zero && !Path.GetFullPath(libraryPath).Equals(
  523. Path.GetFullPath(_libraryFileName), StringComparison.OrdinalIgnoreCase))
  524. {
  525. throw new SevenZipLibraryException(
  526. "can not change the library path while the library \"" + _libraryFileName + "\" is being used.");
  527. }
  528. if (!File.Exists(libraryPath))
  529. {
  530. throw new SevenZipLibraryException(
  531. "can not change the library path because the file \"" + libraryPath + "\" does not exist.");
  532. }
  533. _libraryFileName = libraryPath;
  534. _features = null;
  535. }
  536. #endif
  537. }
  538. #endif
  539. }