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
117
118
119
120
121
|
// Archive/GZipIn.cpp
#include "StdAfx.h"
#include "GZipIn.h"
#include "Common/Defs.h"
#include "Common/MyCom.h"
#include "Windows/Defs.h"
#include "../../Common/StreamUtils.h"
namespace NArchive {
namespace NGZip {
HRESULT CInArchive::ReadBytes(ISequentialInStream *inStream, void *data, UInt32 size)
{
UInt32 realProcessedSize;
RINOK(ReadStream(inStream, data, size, &realProcessedSize));
m_Position += realProcessedSize;
if(realProcessedSize != size)
return S_FALSE;
return S_OK;
}
HRESULT CInArchive::ReadByte(ISequentialInStream *inStream, Byte &value)
{
return ReadBytes(inStream, &value, 1);
}
HRESULT CInArchive::ReadUInt16(ISequentialInStream *inStream, UInt16 &value)
{
value = 0;
for (int i = 0; i < 2; i++)
{
Byte b;
RINOK(ReadByte(inStream, b));
value |= (UInt16(b) << (8 * i));
}
return S_OK;
}
HRESULT CInArchive::ReadUInt32(ISequentialInStream *inStream, UInt32 &value)
{
value = 0;
for (int i = 0; i < 4; i++)
{
Byte b;
RINOK(ReadByte(inStream, b));
value |= (UInt32(b) << (8 * i));
}
return S_OK;
}
HRESULT CInArchive::ReadZeroTerminatedString(ISequentialInStream *inStream, AString &resString, CCRC &crc)
{
resString.Empty();
while(true)
{
Byte c;
RINOK(ReadByte(inStream, c));
crc.UpdateByte(c);
if (c == 0)
return S_OK;
resString += char(c);
}
}
HRESULT CInArchive::ReadHeader(ISequentialInStream *inStream, CItem &item)
{
item.Clear();
m_Position = 0;
UInt16 signature;
RINOK(ReadUInt16(inStream, signature));
if (signature != kSignature)
return S_FALSE;
RINOK(ReadByte(inStream, item.CompressionMethod));
RINOK(ReadByte(inStream, item.Flags));
RINOK(ReadUInt32(inStream, item.Time));
RINOK(ReadByte(inStream, item.ExtraFlags));
RINOK(ReadByte(inStream, item.HostOS));
CCRC crc;
crc.Update(&signature, 2);
crc.UpdateByte(item.CompressionMethod);
crc.UpdateByte(item.Flags);
crc.UpdateUInt32(item.Time);
crc.UpdateByte(item.ExtraFlags);
crc.UpdateByte(item.HostOS);
if (item.ExtraFieldIsPresent())
{
UInt16 extraSize;
RINOK(ReadUInt16(inStream, extraSize));
crc.UpdateUInt16(extraSize);
item.Extra.SetCapacity(extraSize);
RINOK(ReadBytes(inStream, item.Extra, extraSize));
crc.Update(item.Extra, extraSize);
}
if (item.NameIsPresent())
RINOK(ReadZeroTerminatedString(inStream, item.Name, crc));
if (item.CommentIsPresent())
RINOK(ReadZeroTerminatedString(inStream, item.Comment, crc));
if (item.HeaderCRCIsPresent())
{
UInt16 headerCRC;
RINOK(ReadUInt16(inStream, headerCRC));
if ((UInt16)crc.GetDigest() != headerCRC)
return S_FALSE;
}
return S_OK;
}
HRESULT CInArchive::ReadPostHeader(ISequentialInStream *inStream, CItem &item)
{
RINOK(ReadUInt32(inStream, item.FileCRC));
return ReadUInt32(inStream, item.UnPackSize32);
}
}}
|