LzmaEncodeStream.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  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.IO;
  15. using SevenZip.Sdk.Compression.Lzma;
  16. namespace SevenZip
  17. {
  18. #if LZMA_STREAM
  19. #if COMPRESS
  20. /// <summary>
  21. /// The stream which compresses data with LZMA on the fly.
  22. /// </summary>
  23. public class LzmaEncodeStream : Stream
  24. {
  25. private const int MAX_BUFFER_CAPACITY = 1 << 30; //1 Gb
  26. private readonly MemoryStream _buffer = new MemoryStream();
  27. private readonly int _bufferCapacity = 1 << 18; //256 kb
  28. private readonly bool _ownOutput;
  29. private bool _disposed;
  30. private Encoder _lzmaEncoder;
  31. private Stream _output;
  32. /// <summary>
  33. /// Initializes a new instance of the LzmaEncodeStream class.
  34. /// </summary>
  35. public LzmaEncodeStream()
  36. {
  37. _output = new MemoryStream();
  38. _ownOutput = true;
  39. Init();
  40. }
  41. /// <summary>
  42. /// Initializes a new instance of the LzmaEncodeStream class.
  43. /// </summary>
  44. /// <param name="bufferCapacity">The buffer size. The bigger size, the better compression.</param>
  45. public LzmaEncodeStream(int bufferCapacity)
  46. {
  47. _output = new MemoryStream();
  48. _ownOutput = true;
  49. if (bufferCapacity > MAX_BUFFER_CAPACITY)
  50. {
  51. throw new ArgumentException("Too large capacity.", "bufferCapacity");
  52. }
  53. _bufferCapacity = bufferCapacity;
  54. Init();
  55. }
  56. /// <summary>
  57. /// Initializes a new instance of the LzmaEncodeStream class.
  58. /// </summary>
  59. /// <param name="outputStream">An output stream which supports writing.</param>
  60. public LzmaEncodeStream(Stream outputStream)
  61. {
  62. if (!outputStream.CanWrite)
  63. {
  64. throw new ArgumentException("The specified stream can not write.", "outputStream");
  65. }
  66. _output = outputStream;
  67. Init();
  68. }
  69. /// <summary>
  70. /// Initializes a new instance of the LzmaEncodeStream class.
  71. /// </summary>
  72. /// <param name="outputStream">An output stream which supports writing.</param>
  73. /// <param name="bufferCapacity">A buffer size. The bigger size, the better compression.</param>
  74. public LzmaEncodeStream(Stream outputStream, int bufferCapacity)
  75. {
  76. if (!outputStream.CanWrite)
  77. {
  78. throw new ArgumentException("The specified stream can not write.", "outputStream");
  79. }
  80. _output = outputStream;
  81. if (bufferCapacity > 1 << 30)
  82. {
  83. throw new ArgumentException("Too large capacity.", "bufferCapacity");
  84. }
  85. _bufferCapacity = bufferCapacity;
  86. Init();
  87. }
  88. /// <summary>
  89. /// Gets a value indicating whether the current stream supports reading.
  90. /// </summary>
  91. public override bool CanRead
  92. {
  93. get
  94. {
  95. return false;
  96. }
  97. }
  98. /// <summary>
  99. /// Gets a value indicating whether the current stream supports seeking.
  100. /// </summary>
  101. public override bool CanSeek
  102. {
  103. get
  104. {
  105. return false;
  106. }
  107. }
  108. /// <summary>
  109. /// Gets a value indicating whether the current stream supports writing.
  110. /// </summary>
  111. public override bool CanWrite
  112. {
  113. get
  114. {
  115. DisposedCheck();
  116. return _buffer.CanWrite;
  117. }
  118. }
  119. /// <summary>
  120. /// Gets the length in bytes of the output stream.
  121. /// </summary>
  122. public override long Length
  123. {
  124. get
  125. {
  126. DisposedCheck();
  127. if (_output.CanSeek)
  128. {
  129. return _output.Length;
  130. }
  131. return _buffer.Position;
  132. }
  133. }
  134. /// <summary>
  135. /// Gets or sets the position within the output stream.
  136. /// </summary>
  137. public override long Position
  138. {
  139. get
  140. {
  141. DisposedCheck();
  142. if (_output.CanSeek)
  143. {
  144. return _output.Position;
  145. }
  146. return _buffer.Position;
  147. }
  148. set
  149. {
  150. throw new NotSupportedException();
  151. }
  152. }
  153. private void Init()
  154. {
  155. _buffer.Capacity = _bufferCapacity;
  156. SevenZipCompressor.LzmaDictionarySize = _bufferCapacity;
  157. _lzmaEncoder = new Encoder();
  158. SevenZipCompressor.WriteLzmaProperties(_lzmaEncoder);
  159. }
  160. /// <summary>
  161. /// Checked whether the class was disposed.
  162. /// </summary>
  163. /// <exception cref="System.ObjectDisposedException" />
  164. private void DisposedCheck()
  165. {
  166. if (_disposed)
  167. {
  168. throw new ObjectDisposedException("SevenZipExtractor");
  169. }
  170. }
  171. private void WriteChunk()
  172. {
  173. _lzmaEncoder.WriteCoderProperties(_output);
  174. long streamSize = _buffer.Position;
  175. if (_buffer.Length != _buffer.Position)
  176. {
  177. _buffer.SetLength(_buffer.Position);
  178. }
  179. _buffer.Position = 0;
  180. for (int i = 0; i < 8; i++)
  181. {
  182. _output.WriteByte((byte) (streamSize >> (8*i)));
  183. }
  184. _lzmaEncoder.Code(_buffer, _output, -1, -1, null);
  185. _buffer.Position = 0;
  186. }
  187. /// <summary>
  188. /// Converts the LzmaEncodeStream to the LzmaDecodeStream to read data.
  189. /// </summary>
  190. /// <returns></returns>
  191. public LzmaDecodeStream ToDecodeStream()
  192. {
  193. DisposedCheck();
  194. Flush();
  195. return new LzmaDecodeStream(_output);
  196. }
  197. /// <summary>
  198. /// Clears all buffers for this stream and causes any buffered data to be compressed and written.
  199. /// </summary>
  200. public override void Flush()
  201. {
  202. DisposedCheck();
  203. WriteChunk();
  204. }
  205. /// <summary>
  206. /// Releases all unmanaged resources used by LzmaEncodeStream.
  207. /// </summary>
  208. protected override void Dispose(bool disposing)
  209. {
  210. if (!_disposed)
  211. {
  212. if (disposing)
  213. {
  214. Flush();
  215. _buffer.Close();
  216. if (_ownOutput)
  217. {
  218. _output.Dispose();
  219. }
  220. _output = null;
  221. }
  222. _disposed = true;
  223. }
  224. }
  225. /// <summary>
  226. /// Reads a sequence of bytes from the current stream and advances the position within the stream by the number of bytes read.
  227. /// </summary>
  228. /// <param name="buffer">An array of bytes.</param>
  229. /// <param name="offset">The zero-based byte offset in buffer at which to begin storing the data read from the current stream.</param>
  230. /// <param name="count">The maximum number of bytes to be read from the current stream.</param>
  231. /// <returns>The total number of bytes read into the buffer.</returns>
  232. public override int Read(byte[] buffer, int offset, int count)
  233. {
  234. DisposedCheck();
  235. throw new NotSupportedException();
  236. }
  237. /// <summary>
  238. /// Sets the position within the current stream.
  239. /// </summary>
  240. /// <param name="offset">A byte offset relative to the origin parameter.</param>
  241. /// <param name="origin">A value of type System.IO.SeekOrigin indicating the reference point used to obtain the new position.</param>
  242. /// <returns>The new position within the current stream.</returns>
  243. public override long Seek(long offset, SeekOrigin origin)
  244. {
  245. DisposedCheck();
  246. throw new NotSupportedException();
  247. }
  248. /// <summary>
  249. /// Sets the length of the current stream.
  250. /// </summary>
  251. /// <param name="value">The desired length of the current stream in bytes.</param>
  252. public override void SetLength(long value)
  253. {
  254. DisposedCheck();
  255. throw new NotSupportedException();
  256. }
  257. /// <summary>
  258. /// Writes a sequence of bytes to the current stream and compresses it if necessary.
  259. /// </summary>
  260. /// <param name="buffer">An array of bytes.</param>
  261. /// <param name="offset">The zero-based byte offset in buffer at which to begin storing the data read from the current stream.</param>
  262. /// <param name="count">The maximum number of bytes to be read from the current stream.</param>
  263. public override void Write(byte[] buffer, int offset, int count)
  264. {
  265. DisposedCheck();
  266. int dataLength = Math.Min(buffer.Length - offset, count);
  267. while (_buffer.Position + dataLength >= _bufferCapacity)
  268. {
  269. int length = _bufferCapacity - (int) _buffer.Position;
  270. _buffer.Write(buffer, offset, length);
  271. offset = length + offset;
  272. dataLength -= length;
  273. WriteChunk();
  274. }
  275. _buffer.Write(buffer, offset, dataLength);
  276. }
  277. }
  278. #endif
  279. #endif
  280. }