StreamWrappers.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  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. using System.Globalization;
  16. using System.IO;
  17. using System.Runtime.InteropServices;
  18. #if MONO
  19. using SevenZip.Mono.COM;
  20. #endif
  21. namespace SevenZip
  22. {
  23. #if UNMANAGED
  24. /// <summary>
  25. /// A class that has DisposeStream property.
  26. /// </summary>
  27. internal class DisposeVariableWrapper
  28. {
  29. public bool DisposeStream { protected get; set; }
  30. protected DisposeVariableWrapper(bool disposeStream) { DisposeStream = disposeStream; }
  31. }
  32. /// <summary>
  33. /// Stream wrapper used in InStreamWrapper
  34. /// </summary>
  35. internal class StreamWrapper : DisposeVariableWrapper, IDisposable
  36. {
  37. /// <summary>
  38. /// File name associated with the stream (for date fix)
  39. /// </summary>
  40. private readonly string _fileName;
  41. private readonly DateTime _fileTime;
  42. /// <summary>
  43. /// Worker stream for reading, writing and seeking.
  44. /// </summary>
  45. private Stream _baseStream;
  46. /// <summary>
  47. /// Initializes a new instance of the StreamWrapper class
  48. /// </summary>
  49. /// <param name="baseStream">Worker stream for reading, writing and seeking</param>
  50. /// <param name="fileName">File name associated with the stream (for attributes fix)</param>
  51. /// <param name="time">File last write time (for attributes fix)</param>
  52. /// <param name="disposeStream">Indicates whether to dispose the baseStream</param>
  53. protected StreamWrapper(Stream baseStream, string fileName, DateTime time, bool disposeStream)
  54. : base(disposeStream)
  55. {
  56. _baseStream = baseStream;
  57. _fileName = fileName;
  58. _fileTime = time;
  59. }
  60. /// <summary>
  61. /// Initializes a new instance of the StreamWrapper class
  62. /// </summary>
  63. /// <param name="baseStream">Worker stream for reading, writing and seeking</param>
  64. /// <param name="disposeStream">Indicates whether to dispose the baseStream</param>
  65. protected StreamWrapper(Stream baseStream, bool disposeStream)
  66. : base(disposeStream)
  67. {
  68. _baseStream = baseStream;
  69. }
  70. /// <summary>
  71. /// Gets the worker stream for reading, writing and seeking.
  72. /// </summary>
  73. protected Stream BaseStream
  74. {
  75. get
  76. {
  77. return _baseStream;
  78. }
  79. }
  80. #region IDisposable Members
  81. /// <summary>
  82. /// Cleans up any resources used and fixes file attributes.
  83. /// </summary>
  84. public void Dispose()
  85. {
  86. if (_baseStream != null && DisposeStream)
  87. {
  88. try
  89. {
  90. _baseStream.Dispose();
  91. }
  92. catch (ObjectDisposedException) { }
  93. _baseStream = null;
  94. }
  95. if (!String.IsNullOrEmpty(_fileName) && File.Exists(_fileName))
  96. {
  97. try
  98. {
  99. #if !WINCE
  100. File.SetLastWriteTime(_fileName, _fileTime);
  101. File.SetLastAccessTime(_fileName, _fileTime);
  102. File.SetCreationTime(_fileName, _fileTime);
  103. #elif WINCE
  104. OpenNETCF.IO.FileHelper.SetLastWriteTime(_fileName, _fileTime);
  105. OpenNETCF.IO.FileHelper.SetLastAccessTime(_fileName, _fileTime);
  106. OpenNETCF.IO.FileHelper.SetCreationTime(_fileName, _fileTime);
  107. #endif
  108. //TODO: time support for Windows Phone
  109. }
  110. catch (ArgumentOutOfRangeException) {}
  111. }
  112. GC.SuppressFinalize(this);
  113. }
  114. #endregion
  115. public virtual void Seek(long offset, SeekOrigin seekOrigin, IntPtr newPosition)
  116. {
  117. if (BaseStream != null)
  118. {
  119. long position = BaseStream.Seek(offset, seekOrigin);
  120. if (newPosition != IntPtr.Zero)
  121. {
  122. Marshal.WriteInt64(newPosition, position);
  123. }
  124. }
  125. }
  126. }
  127. /// <summary>
  128. /// IInStream wrapper used in stream read operations.
  129. /// </summary>
  130. internal sealed class InStreamWrapper : StreamWrapper, ISequentialInStream, IInStream
  131. {
  132. /// <summary>
  133. /// Initializes a new instance of the InStreamWrapper class.
  134. /// </summary>
  135. /// <param name="baseStream">Stream for writing data</param>
  136. /// <param name="disposeStream">Indicates whether to dispose the baseStream</param>
  137. public InStreamWrapper(Stream baseStream, bool disposeStream) : base(baseStream, disposeStream) { }
  138. #region ISequentialInStream Members
  139. /// <summary>
  140. /// Reads data from the stream.
  141. /// </summary>
  142. /// <param name="data">A data array.</param>
  143. /// <param name="size">The array size.</param>
  144. /// <returns>The read bytes count.</returns>
  145. public int Read(byte[] data, uint size)
  146. {
  147. int readCount = 0;
  148. if (BaseStream != null)
  149. {
  150. readCount = BaseStream.Read(data, 0, (int) size);
  151. if (readCount > 0)
  152. {
  153. OnBytesRead(new IntEventArgs(readCount));
  154. }
  155. }
  156. return readCount;
  157. }
  158. #endregion
  159. /// <summary>
  160. /// Occurs when IntEventArgs.Value bytes were read from the source.
  161. /// </summary>
  162. public event EventHandler<IntEventArgs> BytesRead;
  163. private void OnBytesRead(IntEventArgs e)
  164. {
  165. if (BytesRead != null)
  166. {
  167. BytesRead(this, e);
  168. }
  169. }
  170. }
  171. /// <summary>
  172. /// IOutStream wrapper used in stream write operations.
  173. /// </summary>
  174. internal sealed class OutStreamWrapper : StreamWrapper, ISequentialOutStream, IOutStream
  175. {
  176. /// <summary>
  177. /// Initializes a new instance of the OutStreamWrapper class
  178. /// </summary>
  179. /// <param name="baseStream">Stream for writing data</param>
  180. /// <param name="fileName">File name (for attributes fix)</param>
  181. /// <param name="time">Time of the file creation (for attributes fix)</param>
  182. /// <param name="disposeStream">Indicates whether to dispose the baseStream</param>
  183. public OutStreamWrapper(Stream baseStream, string fileName, DateTime time, bool disposeStream) :
  184. base(baseStream, fileName, time, disposeStream) {}
  185. /// <summary>
  186. /// Initializes a new instance of the OutStreamWrapper class
  187. /// </summary>
  188. /// <param name="baseStream">Stream for writing data</param>
  189. /// <param name="disposeStream">Indicates whether to dispose the baseStream</param>
  190. public OutStreamWrapper(Stream baseStream, bool disposeStream) :
  191. base(baseStream, disposeStream) {}
  192. #region IOutStream Members
  193. public int SetSize(long newSize)
  194. {
  195. BaseStream.SetLength(newSize);
  196. return 0;
  197. }
  198. #endregion
  199. #region ISequentialOutStream Members
  200. /// <summary>
  201. /// Writes data to the stream
  202. /// </summary>
  203. /// <param name="data">Data array</param>
  204. /// <param name="size">Array size</param>
  205. /// <param name="processedSize">Count of written bytes</param>
  206. /// <returns>Zero if Ok</returns>
  207. public int Write(byte[] data, uint size, IntPtr processedSize)
  208. {
  209. BaseStream.Write(data, 0, (int) size);
  210. if (processedSize != IntPtr.Zero)
  211. {
  212. Marshal.WriteInt32(processedSize, (int) size);
  213. }
  214. OnBytesWritten(new IntEventArgs((int) size));
  215. return 0;
  216. }
  217. #endregion
  218. /// <summary>
  219. /// Occurs when IntEventArgs.Value bytes were written.
  220. /// </summary>
  221. public event EventHandler<IntEventArgs> BytesWritten;
  222. private void OnBytesWritten(IntEventArgs e)
  223. {
  224. if (BytesWritten != null)
  225. {
  226. BytesWritten(this, e);
  227. }
  228. }
  229. }
  230. /// <summary>
  231. /// Base multi volume stream wrapper class.
  232. /// </summary>
  233. internal class MultiStreamWrapper : DisposeVariableWrapper, IDisposable
  234. {
  235. protected readonly Dictionary<int, KeyValuePair<long, long>> StreamOffsets =
  236. new Dictionary<int, KeyValuePair<long, long>>();
  237. protected readonly List<Stream> Streams = new List<Stream>();
  238. protected int CurrentStream;
  239. protected long Position;
  240. protected long StreamLength;
  241. /// <summary>
  242. /// Initializes a new instance of the MultiStreamWrapper class.
  243. /// </summary>
  244. /// <param name="dispose">Perform Dispose() if requested to.</param>
  245. protected MultiStreamWrapper(bool dispose) : base(dispose) {}
  246. /// <summary>
  247. /// Gets the total length of input data.
  248. /// </summary>
  249. public long Length
  250. {
  251. get
  252. {
  253. return StreamLength;
  254. }
  255. }
  256. #region IDisposable Members
  257. /// <summary>
  258. /// Cleans up any resources used and fixes file attributes.
  259. /// </summary>
  260. public virtual void Dispose()
  261. {
  262. if (DisposeStream)
  263. {
  264. foreach (Stream stream in Streams)
  265. {
  266. try
  267. {
  268. stream.Dispose();
  269. }
  270. catch (ObjectDisposedException) {}
  271. }
  272. Streams.Clear();
  273. }
  274. GC.SuppressFinalize(this);
  275. }
  276. #endregion
  277. protected static string VolumeNumber(int num)
  278. {
  279. if (num < 10)
  280. {
  281. return ".00" + num.ToString(CultureInfo.InvariantCulture);
  282. }
  283. if (num > 9 && num < 100)
  284. {
  285. return ".0" + num.ToString(CultureInfo.InvariantCulture);
  286. }
  287. if (num > 99 && num < 1000)
  288. {
  289. return "." + num.ToString(CultureInfo.InvariantCulture);
  290. }
  291. return String.Empty;
  292. }
  293. private int StreamNumberByOffset(long offset)
  294. {
  295. foreach (int number in StreamOffsets.Keys)
  296. {
  297. if (StreamOffsets[number].Key <= offset &&
  298. StreamOffsets[number].Value >= offset)
  299. {
  300. return number;
  301. }
  302. }
  303. return -1;
  304. }
  305. public void Seek(long offset, SeekOrigin seekOrigin, IntPtr newPosition)
  306. {
  307. long absolutePosition = (seekOrigin == SeekOrigin.Current)
  308. ? Position + offset
  309. : offset;
  310. CurrentStream = StreamNumberByOffset(absolutePosition);
  311. long delta = Streams[CurrentStream].Seek(
  312. absolutePosition - StreamOffsets[CurrentStream].Key, SeekOrigin.Begin);
  313. Position = StreamOffsets[CurrentStream].Key + delta;
  314. if (newPosition != IntPtr.Zero)
  315. {
  316. Marshal.WriteInt64(newPosition, Position);
  317. }
  318. }
  319. }
  320. /// <summary>
  321. /// IInStream wrapper used in stream multi volume read operations.
  322. /// </summary>
  323. internal sealed class InMultiStreamWrapper : MultiStreamWrapper, ISequentialInStream, IInStream
  324. {
  325. /// <summary>
  326. /// Initializes a new instance of the InMultiStreamWrapper class.
  327. /// </summary>
  328. /// <param name="fileName">The archive file name.</param>
  329. /// <param name="dispose">Perform Dispose() if requested to.</param>
  330. public InMultiStreamWrapper(string fileName, bool dispose) :
  331. base(dispose)
  332. {
  333. string baseName = fileName.Substring(0, fileName.Length - 4);
  334. int i = 0;
  335. while (File.Exists(fileName))
  336. {
  337. Streams.Add(new FileStream(fileName, FileMode.Open));
  338. long length = Streams[i].Length;
  339. StreamOffsets.Add(i++, new KeyValuePair<long, long>(StreamLength, StreamLength + length));
  340. StreamLength += length;
  341. fileName = baseName + VolumeNumber(i + 1);
  342. }
  343. }
  344. #region ISequentialInStream Members
  345. /// <summary>
  346. /// Reads data from the stream.
  347. /// </summary>
  348. /// <param name="data">A data array.</param>
  349. /// <param name="size">The array size.</param>
  350. /// <returns>The read bytes count.</returns>
  351. public int Read(byte[] data, uint size)
  352. {
  353. var readSize = (int) size;
  354. int readCount = Streams[CurrentStream].Read(data, 0, readSize);
  355. readSize -= readCount;
  356. Position += readCount;
  357. while (readCount < (int) size)
  358. {
  359. if (CurrentStream == Streams.Count - 1)
  360. {
  361. return readCount;
  362. }
  363. CurrentStream++;
  364. Streams[CurrentStream].Seek(0, SeekOrigin.Begin);
  365. int count = Streams[CurrentStream].Read(data, readCount, readSize);
  366. readCount += count;
  367. readSize -= count;
  368. Position += count;
  369. }
  370. return readCount;
  371. }
  372. #endregion
  373. }
  374. #if COMPRESS
  375. /// <summary>
  376. /// IOutStream wrapper used in multi volume stream write operations.
  377. /// </summary>
  378. internal sealed class OutMultiStreamWrapper : MultiStreamWrapper, ISequentialOutStream, IOutStream
  379. {
  380. private readonly string _archiveName;
  381. private readonly int _volumeSize;
  382. private long _overallLength;
  383. /// <summary>
  384. /// Initializes a new instance of the OutMultiStreamWrapper class.
  385. /// </summary>
  386. /// <param name="archiveName">The archive name.</param>
  387. /// <param name="volumeSize">The volume size.</param>
  388. public OutMultiStreamWrapper(string archiveName, int volumeSize) :
  389. base(true)
  390. {
  391. _archiveName = archiveName;
  392. _volumeSize = volumeSize;
  393. CurrentStream = -1;
  394. NewVolumeStream();
  395. }
  396. #region IOutStream Members
  397. public int SetSize(long newSize)
  398. {
  399. return 0;
  400. }
  401. #endregion
  402. #region ISequentialOutStream Members
  403. public int Write(byte[] data, uint size, IntPtr processedSize)
  404. {
  405. int offset = 0;
  406. var originalSize = (int) size;
  407. Position += size;
  408. _overallLength = Math.Max(Position + 1, _overallLength);
  409. while (size > _volumeSize - Streams[CurrentStream].Position)
  410. {
  411. var count = (int) (_volumeSize - Streams[CurrentStream].Position);
  412. Streams[CurrentStream].Write(data, offset, count);
  413. size -= (uint) count;
  414. offset += count;
  415. NewVolumeStream();
  416. }
  417. Streams[CurrentStream].Write(data, offset, (int) size);
  418. if (processedSize != IntPtr.Zero)
  419. {
  420. Marshal.WriteInt32(processedSize, originalSize);
  421. }
  422. return 0;
  423. }
  424. #endregion
  425. public override void Dispose()
  426. {
  427. int lastIndex = Streams.Count - 1;
  428. Streams[lastIndex].SetLength(lastIndex > 0? Streams[lastIndex].Position : _overallLength);
  429. base.Dispose();
  430. }
  431. private void NewVolumeStream()
  432. {
  433. CurrentStream++;
  434. Streams.Add(File.Create(_archiveName + VolumeNumber(CurrentStream + 1)));
  435. Streams[CurrentStream].SetLength(_volumeSize);
  436. StreamOffsets.Add(CurrentStream, new KeyValuePair<long, long>(0, _volumeSize - 1));
  437. }
  438. }
  439. #endif
  440. internal sealed class FakeOutStreamWrapper : ISequentialOutStream, IDisposable
  441. {
  442. #region IDisposable Members
  443. public void Dispose()
  444. {
  445. GC.SuppressFinalize(this);
  446. }
  447. #endregion
  448. #region ISequentialOutStream Members
  449. /// <summary>
  450. /// Does nothing except calling the BytesWritten event
  451. /// </summary>
  452. /// <param name="data">Data array</param>
  453. /// <param name="size">Array size</param>
  454. /// <param name="processedSize">Count of written bytes</param>
  455. /// <returns>Zero if Ok</returns>
  456. public int Write(byte[] data, uint size, IntPtr processedSize)
  457. {
  458. OnBytesWritten(new IntEventArgs((int) size));
  459. if (processedSize != IntPtr.Zero)
  460. {
  461. Marshal.WriteInt32(processedSize, (int) size);
  462. }
  463. return 0;
  464. }
  465. #endregion
  466. /// <summary>
  467. /// Occurs when IntEventArgs.Value bytes were written
  468. /// </summary>
  469. public event EventHandler<IntEventArgs> BytesWritten;
  470. private void OnBytesWritten(IntEventArgs e)
  471. {
  472. if (BytesWritten != null)
  473. {
  474. BytesWritten(this, e);
  475. }
  476. }
  477. }
  478. #endif
  479. }