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
122
|
// 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"
extern "C"
{
#include "../../../../C/7zCrc.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, UInt32 &crc)
{
RINOK(ReadBytes(inStream, &value, 1));
crc = CRC_UPDATE_BYTE(crc, value);
return S_OK;
}
HRESULT CInArchive::ReadUInt16(ISequentialInStream *inStream, UInt16 &value, UInt32 &crc)
{
value = 0;
for (int i = 0; i < 2; i++)
{
Byte b;
RINOK(ReadByte(inStream, b, crc));
value |= (UInt16(b) << (8 * i));
}
return S_OK;
}
HRESULT CInArchive::ReadUInt32(ISequentialInStream *inStream, UInt32 &value, UInt32 &crc)
{
value = 0;
for (int i = 0; i < 4; i++)
{
Byte b;
RINOK(ReadByte(inStream, b, crc));
value |= (UInt32(b) << (8 * i));
}
return S_OK;
}
HRESULT CInArchive::ReadZeroTerminatedString(ISequentialInStream *inStream, AString &resString, UInt32 &crc)
{
resString.Empty();
for (;;)
{
Byte c;
RINOK(ReadByte(inStream, c, crc));
if (c == 0)
return S_OK;
resString += char(c);
}
}
HRESULT CInArchive::ReadHeader(ISequentialInStream *inStream, CItem &item)
{
item.Clear();
m_Position = 0;
UInt16 signature;
UInt32 crc = CRC_INIT_VAL;;
RINOK(ReadUInt16(inStream, signature, crc));
if (signature != kSignature)
return S_FALSE;
RINOK(ReadByte(inStream, item.CompressionMethod, crc));
RINOK(ReadByte(inStream, item.Flags, crc));
RINOK(ReadUInt32(inStream, item.Time, crc));
RINOK(ReadByte(inStream, item.ExtraFlags, crc));
RINOK(ReadByte(inStream, item.HostOS, crc));
if (item.ExtraFieldIsPresent())
{
UInt16 extraSize;
RINOK(ReadUInt16(inStream, extraSize, crc));
item.Extra.SetCapacity(extraSize);
RINOK(ReadBytes(inStream, item.Extra, extraSize));
crc = CrcUpdate(crc, 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;
UInt32 dummy = 0;
RINOK(ReadUInt16(inStream, headerCRC, dummy));
if ((UInt16)CRC_GET_DIGEST(crc) != headerCRC)
return S_FALSE;
}
return S_OK;
}
HRESULT CInArchive::ReadPostHeader(ISequentialInStream *inStream, CItem &item)
{
UInt32 dummy = 0;
RINOK(ReadUInt32(inStream, item.FileCRC, dummy));
return ReadUInt32(inStream, item.UnPackSize32, dummy);
}
}}
|