-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathSevenZipHelper.cs
116 lines (96 loc) · 3.75 KB
/
SevenZipHelper.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
using System;
using System.IO;
using System.Text;
namespace SevenZip
{
public static class Helper
{
#region Settings
/// <summary>
/// 2 MB of memory will be reserved for dictionary.
/// </summary>
static Int32 dictionary = 1 << 21; // 2 MB
static Int32 posStateBits = 2;
static Int32 litContextBits = 3;
static Int32 litPosBits = 0;
static Int32 algorithm = 2;
/// <summary>
/// Incerease numFastBytes to get better compression ratios.
/// 32 - moderate compression
/// 128 - extreme compression
/// </summary>
static Int32 numFastBytes = 32;
static bool eos = false;
static CoderPropID[] propIDs =
{
CoderPropID.DictionarySize,
CoderPropID.PosStateBits,
CoderPropID.LitContextBits,
CoderPropID.LitPosBits,
CoderPropID.Algorithm,
CoderPropID.NumFastBytes,
CoderPropID.MatchFinder,
CoderPropID.EndMarker
};
static object[] properties =
{
(Int32)(dictionary),
(Int32)(posStateBits),
(Int32)(litContextBits),
(Int32)(litPosBits),
(Int32)(algorithm),
(Int32)(numFastBytes),
"bt4",
eos
};
#endregion
/// <summary>
/// Compress input stream into output stream using LZMA.
/// Remeber to set desirable positions input and output streams.
/// Input reading starts at inStream.Position and ends at inStream.Length.
/// Output writing starts at outStream.Position.
/// </summary>
/// <param name="inStream">Input stream</param>
/// <param name="outStream">Output stream</param>
public static void Compress(Stream inStream, Stream outStream)
{
Compression.LZMA.Encoder encoder = new Compression.LZMA.Encoder();
encoder.SetCoderProperties(propIDs, properties);
encoder.WriteCoderProperties(outStream);
var writer = new BinaryWriter(outStream, Encoding.UTF8);
// Write original size.
writer.Write(inStream.Length - inStream.Position);
// Save position with compressed size.
long positionForCompressedSize = outStream.Position;
// Leave placeholder for size after compression.
writer.Write((Int64)0);
long positionForCompressedDataStart = outStream.Position;
encoder.Code(inStream, outStream, -1, -1, null);
long positionAfterCompression = outStream.Position;
// Seek back to the placeholder for compressed size.
outStream.Position = positionForCompressedSize;
// Write size after compression.
writer.Write((Int64)(positionAfterCompression - positionForCompressedDataStart));
// Restore position.
outStream.Position = positionAfterCompression;
}
/// <summary>
/// Decompress LZMA compressed stream into outStream.
/// Remeber to set desirable positions in input and output streams.
/// </summary>
/// <param name="inStream">Input stream</param>
/// <param name="outStream">Output stream</param>
public static void Decompress(Stream inStream, Stream outStream)
{
byte[] properties = new byte[5];
if (inStream.Read(properties, 0, 5) != 5)
throw (new Exception("Input stream is too short."));
Compression.LZMA.Decoder decoder = new Compression.LZMA.Decoder();
decoder.SetDecoderProperties(properties);
var br = new BinaryReader(inStream, Encoding.UTF8);
long decompressedSize = br.ReadInt64();
long compressedSize = br.ReadInt64();
decoder.Code(inStream, outStream, compressedSize, decompressedSize, null);
}
}
}