pedump 0.5.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/Gemfile +16 -0
- data/Gemfile.lock +90 -0
- data/LICENSE.txt +20 -0
- data/README.md +410 -0
- data/Rakefile +179 -0
- data/VERSION +1 -0
- data/bin/pedump +7 -0
- data/data/fs.txt +224 -0
- data/data/jc-userdb.txt +14371 -0
- data/data/sig.bin +0 -0
- data/data/signatures.txt +678 -0
- data/data/userdb.txt +14083 -0
- data/lib/pedump.rb +868 -0
- data/lib/pedump/cli.rb +804 -0
- data/lib/pedump/comparer.rb +147 -0
- data/lib/pedump/composite_io.rb +56 -0
- data/lib/pedump/core.rb +38 -0
- data/lib/pedump/core_ext/try.rb +57 -0
- data/lib/pedump/loader.rb +393 -0
- data/lib/pedump/loader/minidump.rb +351 -0
- data/lib/pedump/loader/section.rb +57 -0
- data/lib/pedump/logger.rb +67 -0
- data/lib/pedump/ne.rb +425 -0
- data/lib/pedump/ne/version_info.rb +171 -0
- data/lib/pedump/packer.rb +173 -0
- data/lib/pedump/pe.rb +121 -0
- data/lib/pedump/resources.rb +436 -0
- data/lib/pedump/security.rb +58 -0
- data/lib/pedump/sig_parser.rb +507 -0
- data/lib/pedump/tls.rb +17 -0
- data/lib/pedump/unpacker.rb +26 -0
- data/lib/pedump/unpacker/aspack.rb +858 -0
- data/lib/pedump/unpacker/upx.rb +13 -0
- data/lib/pedump/version.rb +10 -0
- data/lib/pedump/version_info.rb +171 -0
- data/misc/aspack/Makefile +3 -0
- data/misc/aspack/aspack_unlzx.c +92 -0
- data/misc/aspack/lzxdec.c +479 -0
- data/misc/aspack/lzxdec.h +56 -0
- data/misc/nedump.c +751 -0
- data/pedump.gemspec +109 -0
- metadata +227 -0
@@ -0,0 +1,13 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
# coding: binary
|
3
|
+
require 'pedump/loader'
|
4
|
+
require 'pedump/cli'
|
5
|
+
|
6
|
+
module PEdump::Unpacker; end
|
7
|
+
|
8
|
+
class PEdump::Unpacker::UPX
|
9
|
+
def self.unpack src_fname, dst_fname, log = ''
|
10
|
+
log << `upx -dqq #{src_fname} -o #{dst_fname} 2>&1`
|
11
|
+
$?.success?
|
12
|
+
end
|
13
|
+
end
|
@@ -0,0 +1,171 @@
|
|
1
|
+
class PEdump
|
2
|
+
class VS_VERSIONINFO < IOStruct.new( 'v3a32v',
|
3
|
+
:wLength,
|
4
|
+
:wValueLength,
|
5
|
+
:wType,
|
6
|
+
:szKey, # The Unicode string L"VS_VERSION_INFO".
|
7
|
+
:Padding1,
|
8
|
+
# manual:
|
9
|
+
:Value, # VS_FIXEDFILEINFO
|
10
|
+
:Padding2,
|
11
|
+
:Children
|
12
|
+
)
|
13
|
+
def self.read f, size = SIZE
|
14
|
+
super.tap do |vi|
|
15
|
+
vi.szKey.force_encoding('UTF-16LE').encode!('UTF-8').sub!(/\u0000$/,'') rescue nil
|
16
|
+
vi.Padding1 = f.tell%4 > 0 ? f.read(4 - f.tell%4) : nil
|
17
|
+
vi.Value = VS_FIXEDFILEINFO.read(f,vi.wValueLength)
|
18
|
+
# As many zero words as necessary to align the Children member on a 32-bit boundary.
|
19
|
+
# These bytes are not included in wValueLength. This member is optional.
|
20
|
+
vi.Padding2 = f.tell%4 > 0 ? f.read(4 - f.tell%4) : nil
|
21
|
+
vi.Children = [] # An array of zero or one StringFileInfo structures,
|
22
|
+
# and zero or one VarFileInfo structures
|
23
|
+
|
24
|
+
2.times do
|
25
|
+
pos = f.tell
|
26
|
+
f.seek(pos+6) # seek 6 bytes forward
|
27
|
+
t = f.read(6)
|
28
|
+
f.seek(pos) # return back
|
29
|
+
case t
|
30
|
+
when "V\x00a\x00r\x00"
|
31
|
+
vi.Children << VarFileInfo.read(f)
|
32
|
+
when "S\x00t\x00r\x00"
|
33
|
+
vi.Children << StringFileInfo.read(f)
|
34
|
+
else
|
35
|
+
PEdump.logger.warn "[?] invalid VS_VERSIONINFO child type #{t.inspect}"
|
36
|
+
break
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
class VS_FIXEDFILEINFO < IOStruct.new( 'V13',
|
44
|
+
:dwSignature,
|
45
|
+
:dwStrucVersion,
|
46
|
+
:dwFileVersionMS,
|
47
|
+
:dwFileVersionLS,
|
48
|
+
:dwProductVersionMS,
|
49
|
+
:dwProductVersionLS,
|
50
|
+
:dwFileFlagsMask,
|
51
|
+
:dwFileFlags,
|
52
|
+
:dwFileOS,
|
53
|
+
:dwFileType,
|
54
|
+
:dwFileSubtype,
|
55
|
+
:dwFileDateMS,
|
56
|
+
:dwFileDateLS,
|
57
|
+
# manual:
|
58
|
+
:valid
|
59
|
+
)
|
60
|
+
def self.read f, size = SIZE
|
61
|
+
super.tap do |ffi|
|
62
|
+
ffi.valid = (ffi.dwSignature == 0xFEEF04BD)
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
66
|
+
|
67
|
+
class StringFileInfo < IOStruct.new( 'v3a30',
|
68
|
+
:wLength,
|
69
|
+
:wValueLength, # always 0
|
70
|
+
:wType, # 1 => text data, 0 => binary data
|
71
|
+
:szKey, # The Unicode string L"StringFileInfo"
|
72
|
+
:Padding, # As many zero words as necessary to align the Children member on a 32-bit boundary
|
73
|
+
:Children # An array of one or more StringTable structures
|
74
|
+
)
|
75
|
+
def self.read f, size = SIZE
|
76
|
+
pos0 = f.tell
|
77
|
+
super.tap do |x|
|
78
|
+
x.szKey.force_encoding('UTF-16LE').encode!('UTF-8').sub!(/\u0000$/,'') rescue nil
|
79
|
+
x.Padding = f.tell%4 > 0 ? f.read(4 - f.tell%4) : nil
|
80
|
+
x.Children = []
|
81
|
+
while !f.eof? && f.tell < pos0+x.wLength
|
82
|
+
x.Children << StringTable.read(f)
|
83
|
+
end
|
84
|
+
end
|
85
|
+
end
|
86
|
+
end
|
87
|
+
|
88
|
+
class StringTable < IOStruct.new( 'v3a16v',
|
89
|
+
:wLength, # The length, in bytes, of this StringTable structure,
|
90
|
+
# including all structures indicated by the Children member.
|
91
|
+
:wValueLength, # always 0
|
92
|
+
:wType, # 1 => text data, 0 => binary data
|
93
|
+
:szKey, # An 8-digit hexadecimal number stored as a Unicode string
|
94
|
+
:Padding, # As many zero words as necessary to align the Children member on a 32-bit boundary
|
95
|
+
:Children # An array of one or more String structures.
|
96
|
+
)
|
97
|
+
def self.read f, size = SIZE
|
98
|
+
pos0 = f.tell
|
99
|
+
super.tap do |x|
|
100
|
+
x.szKey.force_encoding('UTF-16LE').encode!('UTF-8').sub!(/\u0000$/,'') rescue nil
|
101
|
+
x.Padding = f.tell%4 > 0 ? f.read(4 - f.tell%4) : nil
|
102
|
+
x.Children = []
|
103
|
+
while !f.eof? && f.tell < pos0+x.wLength
|
104
|
+
x.Children << VersionString.read(f)
|
105
|
+
end
|
106
|
+
end
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
class VersionString < IOStruct.new( 'v3',
|
111
|
+
:wLength, # The length, in bytes, of this String structure.
|
112
|
+
:wValueLength, # The size, in words, of the Value member
|
113
|
+
:wType, # 1 => text data, 0 => binary data
|
114
|
+
:szKey, # An arbitrary Unicode string
|
115
|
+
:Padding, # As many zero words as necessary to align the Value member on a 32-bit boundary
|
116
|
+
:Value # A zero-terminated string. See the szKey member description for more information
|
117
|
+
)
|
118
|
+
def self.read f, size = SIZE
|
119
|
+
pos = f.tell
|
120
|
+
super.tap do |x|
|
121
|
+
x.szKey = ''
|
122
|
+
x.szKey << f.read(2) until x.szKey[-2..-1] == "\x00\x00" || f.eof?
|
123
|
+
x.Padding = f.tell%4 > 0 ? f.read(4 - f.tell%4) : nil
|
124
|
+
|
125
|
+
value_len = [x.wValueLength*2, x.wLength - (f.tell-pos)].min
|
126
|
+
value_len = 0 if value_len < 0
|
127
|
+
|
128
|
+
x.Value = f.read(value_len)
|
129
|
+
if f.tell%4 > 0
|
130
|
+
f.read(4-f.tell%4) # undoc padding?
|
131
|
+
end
|
132
|
+
x.szKey.force_encoding('UTF-16LE').encode!('UTF-8').sub!(/\u0000+$/,'') rescue nil
|
133
|
+
x.Value.force_encoding('UTF-16LE').encode!('UTF-8').sub!(/\u0000+$/,'') rescue nil
|
134
|
+
end
|
135
|
+
end
|
136
|
+
end
|
137
|
+
|
138
|
+
class VarFileInfo < IOStruct.new( 'v3a24v',
|
139
|
+
:wLength,
|
140
|
+
:wValueLength, # always 0
|
141
|
+
:wType, # 1 => text data, 0 => binary data
|
142
|
+
:szKey, # The Unicode string L"VarFileInfo"
|
143
|
+
:Padding, # As many zero words as necessary to align the Children member on a 32-bit boundary
|
144
|
+
:Children # Typically contains a list of languages that the application or DLL supports
|
145
|
+
)
|
146
|
+
def self.read f, size = SIZE
|
147
|
+
super.tap do |x|
|
148
|
+
x.szKey.force_encoding('UTF-16LE').encode!('UTF-8').sub!(/\u0000$/,'') rescue nil
|
149
|
+
x.Padding = f.tell%4 > 0 ? f.read(4 - f.tell%4) : nil
|
150
|
+
x.Children = Var.read(f)
|
151
|
+
end
|
152
|
+
end
|
153
|
+
end
|
154
|
+
|
155
|
+
class Var < IOStruct.new( 'v3a24',
|
156
|
+
:wLength,
|
157
|
+
:wValueLength, # The length, in bytes, of the Value member
|
158
|
+
:wType, # 1 => text data, 0 => binary data
|
159
|
+
:szKey, # The Unicode string L"Translation"
|
160
|
+
:Padding, # As many zero words as necessary to align the Children member on a 32-bit boundary
|
161
|
+
:Value # An array of one or more values that are language and code page identifier pairs
|
162
|
+
)
|
163
|
+
def self.read f, size = SIZE
|
164
|
+
super.tap do |x|
|
165
|
+
x.szKey.force_encoding('UTF-16LE').encode!('UTF-8').sub!(/\u0000$/,'') rescue nil
|
166
|
+
x.Padding = f.tell%4 > 0 ? f.read(4 - f.tell%4) : nil
|
167
|
+
x.Value = f.read(x.wValueLength).unpack('v*')
|
168
|
+
end
|
169
|
+
end
|
170
|
+
end
|
171
|
+
end
|
@@ -0,0 +1,92 @@
|
|
1
|
+
#include <stdlib.h>
|
2
|
+
#include <stdio.h>
|
3
|
+
#include <strings.h> // for bzero()
|
4
|
+
|
5
|
+
#include "lzxdec.c"
|
6
|
+
|
7
|
+
// based on ASPack Unpacker v1.00 by Dr.Golova (thx! :)
|
8
|
+
|
9
|
+
/* error codes for unpacker thread */
|
10
|
+
#define ERR_OK (0)
|
11
|
+
#define ERR_NO_FILE (1)
|
12
|
+
#define ERR_FILE_OPEN (2)
|
13
|
+
#define ERR_NO_MEM (3)
|
14
|
+
#define ERR_CANT_MAP (4)
|
15
|
+
#define ERR_FILE_CREATE (5)
|
16
|
+
#define ERR_FILE_WRITE (6)
|
17
|
+
#define ERR_COPY_OVL (7)
|
18
|
+
#define ERR_CORRUPT (8)
|
19
|
+
#define ERR_UNKNOWN (9)
|
20
|
+
#define ERR_SKIPPED (10)
|
21
|
+
#define ERR_UNPACK (11)
|
22
|
+
#define ERR_FAILED (12)
|
23
|
+
|
24
|
+
|
25
|
+
void write_result(void*buf, int size){
|
26
|
+
fwrite(buf,1,size,stdout);
|
27
|
+
}
|
28
|
+
|
29
|
+
int unpack(BYTE*packed_data, size_t packed_size, size_t unpacked_size){
|
30
|
+
LZX_CONTEXT LZX;
|
31
|
+
BYTE* unpacked_data = NULL;
|
32
|
+
size_t decoded_size;
|
33
|
+
|
34
|
+
bzero(&LZX, sizeof(LZX));
|
35
|
+
|
36
|
+
if ( NULL == (unpacked_data = calloc(1, unpacked_size + 300)) ){
|
37
|
+
perror("no mem");
|
38
|
+
return(ERR_NO_MEM);
|
39
|
+
}
|
40
|
+
|
41
|
+
decoded_size = DecodeLZX(&LZX, packed_data, unpacked_data, packed_size, unpacked_size);
|
42
|
+
if ( decoded_size < 0 || decoded_size < unpacked_size ) {
|
43
|
+
free(unpacked_data);
|
44
|
+
fprintf(stderr,"ERR_UNPACK\n");
|
45
|
+
return(ERR_UNPACK);
|
46
|
+
}
|
47
|
+
|
48
|
+
write_result(unpacked_data, decoded_size);
|
49
|
+
free(unpacked_data);
|
50
|
+
return 0;
|
51
|
+
}
|
52
|
+
|
53
|
+
int main(int argc, char*argv[]){
|
54
|
+
size_t packed_size, unpacked_size;
|
55
|
+
BYTE* packed_data = NULL;
|
56
|
+
int r;
|
57
|
+
|
58
|
+
if(argc != 3){
|
59
|
+
fprintf(stderr, "ASPack unLZX\n");
|
60
|
+
fprintf(stderr, "usage: %s <packed_size> <unpacked_size>\n", argv[0]);
|
61
|
+
fprintf(stderr, "(data is read from stdin and written to stdout)\n", argv[0]);
|
62
|
+
return 1;
|
63
|
+
}
|
64
|
+
|
65
|
+
sscanf(argv[1],"%zu",&packed_size);
|
66
|
+
if( packed_size < 1 || packed_size > 0x10000000 ){ // 256 Mb max
|
67
|
+
fprintf(stderr, "invalid packed_size: %zu\n", packed_size);
|
68
|
+
return 1;
|
69
|
+
}
|
70
|
+
|
71
|
+
sscanf(argv[2],"%zu",&unpacked_size);
|
72
|
+
if( unpacked_size < 1 || unpacked_size > 0x10000000 ){ // 256 Mb max
|
73
|
+
fprintf(stderr, "invalid unpacked_size: %zu\n", unpacked_size);
|
74
|
+
return 1;
|
75
|
+
}
|
76
|
+
|
77
|
+
/* alloc buffer */
|
78
|
+
if ( NULL == (packed_data = calloc(1, packed_size + 300)) ){
|
79
|
+
perror("no mem");
|
80
|
+
return(ERR_NO_MEM);
|
81
|
+
}
|
82
|
+
|
83
|
+
if( packed_size != fread(packed_data, 1, packed_size, stdin)){
|
84
|
+
free(packed_data);
|
85
|
+
perror("read");
|
86
|
+
return(ERR_NO_FILE);
|
87
|
+
}
|
88
|
+
|
89
|
+
r = unpack(packed_data, packed_size, unpacked_size);
|
90
|
+
free(packed_data);
|
91
|
+
return r;
|
92
|
+
}
|
@@ -0,0 +1,479 @@
|
|
1
|
+
//#include <windows.h>
|
2
|
+
|
3
|
+
typedef unsigned char BYTE;
|
4
|
+
typedef int INT;
|
5
|
+
typedef unsigned int UINT;
|
6
|
+
typedef unsigned long DWORD;
|
7
|
+
typedef int BOOL;
|
8
|
+
|
9
|
+
#define FALSE 0
|
10
|
+
#define TRUE 1
|
11
|
+
|
12
|
+
#include "lzxdec.h"
|
13
|
+
|
14
|
+
/* #define LZX_DEBUG_LOG */
|
15
|
+
|
16
|
+
#if defined _DEBUG && defined LZX_DEBUG_LOG
|
17
|
+
# include <stdio.h>
|
18
|
+
static FILE* FDebLog = NULL;
|
19
|
+
#endif
|
20
|
+
|
21
|
+
/* ##### *** COMMON FUNCTIONS *** ########################################## */
|
22
|
+
|
23
|
+
/* memset local redefinition */
|
24
|
+
static void lzx_memset(void* dst, int val, unsigned int count)
|
25
|
+
{
|
26
|
+
while( count-- )
|
27
|
+
{
|
28
|
+
*(char*)(dst) = val;
|
29
|
+
dst = (char*)(dst) + 1;
|
30
|
+
}
|
31
|
+
}
|
32
|
+
|
33
|
+
/* memcpy local redefinition */
|
34
|
+
static void lzx_memcpy(void* dst, void* src, unsigned int count)
|
35
|
+
{
|
36
|
+
while( count-- )
|
37
|
+
{
|
38
|
+
*(char*)(dst) = *(char*)(src);
|
39
|
+
dst = (char*)(dst) + 1;
|
40
|
+
src = (char*)(src) + 1;
|
41
|
+
}
|
42
|
+
}
|
43
|
+
|
44
|
+
/* ##### *** BIT READER FUNCTIONS *** ###################################### */
|
45
|
+
|
46
|
+
/* init bit reader */
|
47
|
+
#define BITRDR_INIT(BitRdr, RawSrc, RawLen) { \
|
48
|
+
(BitRdr)->SrcData = (RawSrc); \
|
49
|
+
(BitRdr)->SrcSize = (RawLen); \
|
50
|
+
(BitRdr)->CurOffs = 0; \
|
51
|
+
(BitRdr)->BitBuff = 0; \
|
52
|
+
(BitRdr)->BitFree = 32; \
|
53
|
+
}
|
54
|
+
|
55
|
+
/* read input byte (can be rewritten for file io) */
|
56
|
+
#define BITRDR_GETBYTE(BitRdr, RetVal) { \
|
57
|
+
if ( (BitRdr)->CurOffs < (BitRdr)->SrcSize ) \
|
58
|
+
RetVal = (BitRdr)->SrcData[(BitRdr)->CurOffs++]; \
|
59
|
+
else \
|
60
|
+
return(-1); /* error */ \
|
61
|
+
}
|
62
|
+
|
63
|
+
/* remove number of bits from bit buffer */
|
64
|
+
#define BITRDR_DROPBITS(BitRdr, BitNum) { \
|
65
|
+
(BitRdr)->BitFree += (BitNum); \
|
66
|
+
}
|
67
|
+
|
68
|
+
/* extract integer from bit buffer */
|
69
|
+
#define BITRDR_GETINT(BitRdr, RetVal) { \
|
70
|
+
DWORD BitBuff = (BitRdr)->BitBuff; \
|
71
|
+
UINT BitFree = (BitRdr)->BitFree; \
|
72
|
+
UINT UVal; \
|
73
|
+
while( BitFree >= 8 ) { \
|
74
|
+
BITRDR_GETBYTE(BitRdr, UVal); \
|
75
|
+
BitBuff = (BitBuff << 8) | UVal; \
|
76
|
+
BitFree -= 8; \
|
77
|
+
} \
|
78
|
+
RetVal = (BitBuff >> (8 - BitFree)) & 0x00ffffff; \
|
79
|
+
(BitRdr)->BitFree = BitFree; \
|
80
|
+
(BitRdr)->BitBuff = BitBuff; \
|
81
|
+
}
|
82
|
+
|
83
|
+
/* read number of bits from bit buffer */
|
84
|
+
#define BITRDR_GETBITS(BitRdr, BitNum, RetVal) { \
|
85
|
+
DWORD BitBuff = (BitRdr)->BitBuff; \
|
86
|
+
UINT BitFree = (BitRdr)->BitFree; \
|
87
|
+
UINT UVal; \
|
88
|
+
while( BitFree >= 8 ) { \
|
89
|
+
BITRDR_GETBYTE(BitRdr, UVal); \
|
90
|
+
BitBuff = (BitBuff << 8) | UVal; \
|
91
|
+
BitFree -= 8; \
|
92
|
+
} \
|
93
|
+
RetVal = (BitBuff >> (8 - BitFree)); \
|
94
|
+
RetVal = (RetVal & 0x00ffffff) >> (24 - (BitNum)); \
|
95
|
+
(BitRdr)->BitFree = (BitFree + (BitNum)); \
|
96
|
+
(BitRdr)->BitBuff = BitBuff; \
|
97
|
+
}
|
98
|
+
|
99
|
+
/* ##### *** SLIDING WINDOW DICTIONARY FUNCTIONS *** ####################### */
|
100
|
+
|
101
|
+
/* init swd context */
|
102
|
+
#define SWD_INIT(Wnd, Mem, Len) { \
|
103
|
+
(Wnd)->Window = (Mem); \
|
104
|
+
(Wnd)->WndLen = (Len); \
|
105
|
+
(Wnd)->CurOfs = 0; \
|
106
|
+
}
|
107
|
+
|
108
|
+
/* put byte to dictionary */
|
109
|
+
#define SWD_PUTBYTE(Wnd, DVal) { \
|
110
|
+
if ( (Wnd)->CurOfs < (Wnd)->WndLen ) \
|
111
|
+
(Wnd)->Window[(Wnd)->CurOfs++] = (BYTE)(DVal); \
|
112
|
+
else \
|
113
|
+
return(-1); /* error */ \
|
114
|
+
}
|
115
|
+
|
116
|
+
/* copy lz phrase in window */
|
117
|
+
#define SWD_DELTACOPY(Wnd, DOfs, DLen) { \
|
118
|
+
BYTE* WndPtr; \
|
119
|
+
if ( (Wnd)->CurOfs < (DOfs) ) \
|
120
|
+
return(-1); /* error */ \
|
121
|
+
if ( (Wnd)->CurOfs + (DLen) > (Wnd)->WndLen ) \
|
122
|
+
return(-1); /* error */ \
|
123
|
+
WndPtr = (Wnd)->Window + (Wnd)->CurOfs; \
|
124
|
+
lzx_memcpy(WndPtr, WndPtr - (DOfs), (DLen)); \
|
125
|
+
(Wnd)->CurOfs += (DLen); \
|
126
|
+
}
|
127
|
+
|
128
|
+
/* ##### *** HUFFMAN DECODERS FUNCTIONS *** ################################ */
|
129
|
+
|
130
|
+
/* init huffman decoder, return next table address */
|
131
|
+
static BYTE* LzxHuf_Init(LZX_HUFF* Huf, LZX_BITRDR* BitRdr, UINT SymNum,
|
132
|
+
BYTE* HufTbl)
|
133
|
+
{
|
134
|
+
Huf->BitRdr = BitRdr;
|
135
|
+
Huf->SymNum = SymNum;
|
136
|
+
Huf->Symbol = (UINT*)(HufTbl);
|
137
|
+
HufTbl += (SymNum * sizeof(Huf->Symbol[0]));
|
138
|
+
Huf->Length = HufTbl;
|
139
|
+
HufTbl += (256 * sizeof(Huf->Length[0]));
|
140
|
+
return(HufTbl);
|
141
|
+
}
|
142
|
+
|
143
|
+
/* decode one huffman symbol */
|
144
|
+
static INT LzxHuf_DecodeSymbol(LZX_HUFF* Huf)
|
145
|
+
{
|
146
|
+
UINT HVal, BNum, IOfs;
|
147
|
+
|
148
|
+
BITRDR_GETINT(Huf->BitRdr, HVal);
|
149
|
+
HVal &= 0x00fffe00;
|
150
|
+
|
151
|
+
if ( HVal < Huf->SymLim[8] )
|
152
|
+
BNum = Huf->Length[HVal >> 16];
|
153
|
+
else if ( HVal < Huf->SymLim[10] )
|
154
|
+
BNum = (HVal < Huf->SymLim[9]) ? (9) : (10);
|
155
|
+
else if ( HVal < Huf->SymLim[11] )
|
156
|
+
BNum = 11;
|
157
|
+
else if ( HVal < Huf->SymLim[12] )
|
158
|
+
BNum = 12;
|
159
|
+
else if ( HVal < Huf->SymLim[13] )
|
160
|
+
BNum = 13;
|
161
|
+
else
|
162
|
+
BNum = (HVal < Huf->SymLim[14]) ? (14) : (15);
|
163
|
+
|
164
|
+
BITRDR_DROPBITS(Huf->BitRdr, BNum);
|
165
|
+
IOfs = (HVal - Huf->SymLim[BNum-1]) >> (24 - BNum);
|
166
|
+
return(Huf->Symbol[Huf->SymIdx[BNum] + IOfs]);
|
167
|
+
}
|
168
|
+
|
169
|
+
/* construct huffman tables */
|
170
|
+
static INT LzxHuf_HufTblBuild(LZX_HUFF* Huf, BYTE* CodeLen)
|
171
|
+
{
|
172
|
+
UINT LenCnt[16];
|
173
|
+
UINT CurIdx[16];
|
174
|
+
UINT SymIdx, I;
|
175
|
+
UINT Lim, Idx;
|
176
|
+
UINT Ofs;
|
177
|
+
|
178
|
+
for ( I = 0; I < 16; I++ )
|
179
|
+
LenCnt[I] = 0;
|
180
|
+
for ( SymIdx = 0; SymIdx < Huf->SymNum; SymIdx++ )
|
181
|
+
LenCnt[CodeLen[SymIdx]]++;
|
182
|
+
|
183
|
+
#ifdef LZX_DEBUG_LOG
|
184
|
+
if ( NULL == FDebLog )
|
185
|
+
FDebLog = fopen("lzxdeb.log", "wt");
|
186
|
+
#endif
|
187
|
+
|
188
|
+
LenCnt[0] = 0;
|
189
|
+
CurIdx[0] = 0;
|
190
|
+
Huf->SymIdx[0] = 0;
|
191
|
+
Huf->SymLim[0] = 0;
|
192
|
+
Lim = 0;
|
193
|
+
Idx = 0;
|
194
|
+
|
195
|
+
for ( I = 1; I < 16; I++ )
|
196
|
+
{
|
197
|
+
Lim += (LenCnt[I] << (24 - I));
|
198
|
+
if ( Lim > 0x1000000 )
|
199
|
+
return(-1); /* overrun */
|
200
|
+
|
201
|
+
Huf->SymLim[I] = Lim;
|
202
|
+
Huf->SymIdx[I] = Huf->SymIdx[I-1] + LenCnt[I-1];
|
203
|
+
CurIdx[I] = Huf->SymIdx[I];
|
204
|
+
|
205
|
+
if ( I <= 8 )
|
206
|
+
{
|
207
|
+
Ofs = (Huf->SymLim[I] >> 16);
|
208
|
+
lzx_memset(&Huf->Length[Idx], I, Ofs - Idx);
|
209
|
+
Idx = Ofs;
|
210
|
+
}
|
211
|
+
}
|
212
|
+
|
213
|
+
if ( Lim != 0x1000000 )
|
214
|
+
return(-1); /* not full set */
|
215
|
+
|
216
|
+
#ifdef LZX_DEBUG_LOG
|
217
|
+
fprintf(FDebLog, "Huf->SymNum == %u\n", Huf->SymNum);
|
218
|
+
fprintf(FDebLog, "Huf->SymIdx == ");
|
219
|
+
for ( I = 0; I < 16; I++ )
|
220
|
+
fprintf(FDebLog, "%u ", Huf->SymIdx[I]);
|
221
|
+
fprintf(FDebLog, "\n");
|
222
|
+
fprintf(FDebLog, "Huf->SymLim == ");
|
223
|
+
for ( I = 0; I < 16; I++ )
|
224
|
+
fprintf(FDebLog, "%u ", Huf->SymLim[I]);
|
225
|
+
fprintf(FDebLog, "\n");
|
226
|
+
fflush(FDebLog);
|
227
|
+
#endif
|
228
|
+
|
229
|
+
for ( SymIdx = 0; SymIdx < Huf->SymNum; SymIdx++ )
|
230
|
+
{
|
231
|
+
if ( CodeLen[SymIdx] )
|
232
|
+
{
|
233
|
+
#ifdef LZX_DEBUG_LOG
|
234
|
+
fprintf(FDebLog, "%u\n", SymIdx);
|
235
|
+
fflush(FDebLog);
|
236
|
+
#endif
|
237
|
+
Huf->Symbol[CurIdx[CodeLen[SymIdx]]++] = SymIdx;
|
238
|
+
}
|
239
|
+
}
|
240
|
+
|
241
|
+
return(0); /* all ok */
|
242
|
+
}
|
243
|
+
|
244
|
+
/* ##### *** LZX DECODER FUNCTIONS *** ##################################### */
|
245
|
+
|
246
|
+
/* basic lzx tables */
|
247
|
+
static const UINT LzxTblLenBits[28] = {
|
248
|
+
0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4,
|
249
|
+
5, 5, 5, 5
|
250
|
+
};
|
251
|
+
static const UINT LzxTblLenBase[28] = {
|
252
|
+
0, 1, 2, 3, 4, 5, 6, 7, 8, 10, 12, 14, 16, 20, 24, 28, 32,
|
253
|
+
40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224
|
254
|
+
};
|
255
|
+
static const UINT LzxTblOfsBits[58] = {
|
256
|
+
0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7,
|
257
|
+
8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13, 14, 14, 15, 15, 16, 16,
|
258
|
+
17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 18, 18, 18, 18,
|
259
|
+
18, 18, 18, 18
|
260
|
+
};
|
261
|
+
static const UINT LzxTblOfsBase[58] = {
|
262
|
+
0x00000000, 0x00000001, 0x00000002, 0x00000003, 0x00000004, 0x00000006,
|
263
|
+
0x00000008, 0x0000000C, 0x00000010, 0x00000018, 0x00000020, 0x00000030,
|
264
|
+
0x00000040, 0x00000060, 0x00000080, 0x000000C0, 0x00000100, 0x00000180,
|
265
|
+
0x00000200, 0x00000300, 0x00000400, 0x00000600, 0x00000800, 0x00000C00,
|
266
|
+
0x00001000, 0x00001800, 0x00002000, 0x00003000, 0x00004000, 0x00006000,
|
267
|
+
0x00008000, 0x0000C000, 0x00010000, 0x00018000, 0x00020000, 0x00030000,
|
268
|
+
0x00040000, 0x00060000, 0x00080000, 0x000A0000, 0x000C0000, 0x000E0000,
|
269
|
+
0x00100000, 0x00120000, 0x00140000, 0x00160000, 0x00180000, 0x001A0000,
|
270
|
+
0x001C0000, 0x001E0000, 0x00200000, 0x00240000, 0x00280000, 0x002C0000,
|
271
|
+
0x00300000, 0x00340000, 0x00380000, 0x003C0000
|
272
|
+
};
|
273
|
+
|
274
|
+
/* ------------------------------------------------------------------------- */
|
275
|
+
|
276
|
+
/* build lzx tables */
|
277
|
+
static INT Lzx_LzxTblBuild(LZX_CONTEXT* Ctx)
|
278
|
+
{
|
279
|
+
BYTE* Heap;
|
280
|
+
BYTE LenTbl[19];
|
281
|
+
BYTE HufSym[757];
|
282
|
+
UINT Val, I, Sym;
|
283
|
+
INT ISym;
|
284
|
+
|
285
|
+
Heap = Ctx->LstMem;
|
286
|
+
BITRDR_GETBITS(&Ctx->BitRdr, 1, Val);
|
287
|
+
if ( !Val ) lzx_memset(Heap, 0, 757);
|
288
|
+
|
289
|
+
for ( I = 0; I < 19; I++ )
|
290
|
+
{
|
291
|
+
BITRDR_GETBITS(&Ctx->BitRdr, 4, Val);
|
292
|
+
LenTbl[I] = (BYTE)(Val);
|
293
|
+
}
|
294
|
+
|
295
|
+
if ( LzxHuf_HufTblBuild(&Ctx->HufSpec, LenTbl) )
|
296
|
+
return(-1); /* error */
|
297
|
+
|
298
|
+
I = 0;
|
299
|
+
while ( I < 757 )
|
300
|
+
{
|
301
|
+
if ( (ISym = LzxHuf_DecodeSymbol(&Ctx->HufSpec)) == -1 )
|
302
|
+
return(-1); /* error */
|
303
|
+
|
304
|
+
if ( (Sym = (UINT)(ISym)) < 16 )
|
305
|
+
{
|
306
|
+
HufSym[I] = (BYTE)((Heap[I] + Sym) & 15); I++;
|
307
|
+
continue;
|
308
|
+
}
|
309
|
+
|
310
|
+
if ( Sym == 16 )
|
311
|
+
{
|
312
|
+
BITRDR_GETBITS(&Ctx->BitRdr, 2, Val); Val += 3;
|
313
|
+
while ( Val > 0 && I < 757 )
|
314
|
+
{
|
315
|
+
HufSym[I] = HufSym[I-1];
|
316
|
+
Val--; I++;
|
317
|
+
}
|
318
|
+
continue;
|
319
|
+
}
|
320
|
+
|
321
|
+
if ( Sym == 17 )
|
322
|
+
{
|
323
|
+
BITRDR_GETBITS(&Ctx->BitRdr, 3, Val);
|
324
|
+
Val += 3;
|
325
|
+
}
|
326
|
+
else
|
327
|
+
{
|
328
|
+
BITRDR_GETBITS(&Ctx->BitRdr, 7, Val);
|
329
|
+
Val += 11;
|
330
|
+
}
|
331
|
+
|
332
|
+
while ( Val > 0 && I < 757 )
|
333
|
+
{
|
334
|
+
HufSym[I] = 0;
|
335
|
+
Val--; I++;
|
336
|
+
}
|
337
|
+
}
|
338
|
+
|
339
|
+
if ( LzxHuf_HufTblBuild(&Ctx->HufBase, &HufSym[0]) )
|
340
|
+
return(-1);
|
341
|
+
if ( LzxHuf_HufTblBuild(&Ctx->HufLens, &HufSym[721]) )
|
342
|
+
return(-1);
|
343
|
+
if ( LzxHuf_HufTblBuild(&Ctx->HufOffs, &HufSym[749]) )
|
344
|
+
return(-1);
|
345
|
+
|
346
|
+
Ctx->HasOffs = FALSE;
|
347
|
+
for ( I = 0; I < 8; I++ )
|
348
|
+
{
|
349
|
+
if ( HufSym[749+I] != 3 )
|
350
|
+
{
|
351
|
+
Ctx->HasOffs = TRUE;
|
352
|
+
break;
|
353
|
+
}
|
354
|
+
}
|
355
|
+
|
356
|
+
lzx_memcpy(Heap, &HufSym[0], 757);
|
357
|
+
return(0); /* all ok */
|
358
|
+
}
|
359
|
+
|
360
|
+
/* ------------------------------------------------------------------------- */
|
361
|
+
|
362
|
+
INT DecodeLZX(LZX_CONTEXT* Ctx, BYTE* Src, BYTE* Dst, UINT PSize, UINT USize)
|
363
|
+
{
|
364
|
+
UINT Written;
|
365
|
+
UINT Symbol;
|
366
|
+
BYTE* HufMem;
|
367
|
+
UINT Ofs;
|
368
|
+
UINT Len;
|
369
|
+
UINT Num;
|
370
|
+
UINT Val;
|
371
|
+
INT ISym;
|
372
|
+
|
373
|
+
/* check params */
|
374
|
+
if ( !Ctx || !Src || !Dst || !USize )
|
375
|
+
return(-1); /* invalid param */
|
376
|
+
|
377
|
+
/* init lzx context */
|
378
|
+
BITRDR_INIT(&Ctx->BitRdr, Src, PSize);
|
379
|
+
SWD_INIT(&Ctx->Window, Dst, USize);
|
380
|
+
Ctx->HasOffs = FALSE;
|
381
|
+
Ctx->LstOfs[0] = 0;
|
382
|
+
Ctx->LstOfs[1] = 0;
|
383
|
+
Ctx->LstOfs[2] = 0;
|
384
|
+
|
385
|
+
/* init huffman coders */
|
386
|
+
HufMem = &Ctx->HufTbl[0];
|
387
|
+
HufMem = LzxHuf_Init(&Ctx->HufBase, &Ctx->BitRdr, 721, HufMem);
|
388
|
+
HufMem = LzxHuf_Init(&Ctx->HufLens, &Ctx->BitRdr, 28, HufMem);
|
389
|
+
HufMem = LzxHuf_Init(&Ctx->HufOffs, &Ctx->BitRdr, 8, HufMem);
|
390
|
+
HufMem = LzxHuf_Init(&Ctx->HufSpec, &Ctx->BitRdr, 19, HufMem);
|
391
|
+
lzx_memset(Ctx->LstMem = HufMem, 0, 757);
|
392
|
+
|
393
|
+
/* build lzx tables */
|
394
|
+
if ( Lzx_LzxTblBuild(Ctx) )
|
395
|
+
return(-1);
|
396
|
+
|
397
|
+
/* decode */
|
398
|
+
Written = 0;
|
399
|
+
while ( Written < USize )
|
400
|
+
{
|
401
|
+
if ( (ISym = LzxHuf_DecodeSymbol(&Ctx->HufBase)) == -1 )
|
402
|
+
return(-1);
|
403
|
+
|
404
|
+
if ( (Symbol = (UINT)(ISym)) < 256 )
|
405
|
+
{
|
406
|
+
/* literal */
|
407
|
+
SWD_PUTBYTE(&Ctx->Window, Symbol);
|
408
|
+
Written++;
|
409
|
+
continue;
|
410
|
+
}
|
411
|
+
|
412
|
+
if ( Symbol < 720 )
|
413
|
+
{
|
414
|
+
/* phrase */
|
415
|
+
Symbol -= 256;
|
416
|
+
Ofs = (Symbol >> 3);
|
417
|
+
Len = (Symbol & 7) + 2;
|
418
|
+
|
419
|
+
if ( Len == 9 )
|
420
|
+
{
|
421
|
+
if ( (ISym = LzxHuf_DecodeSymbol(&Ctx->HufLens)) == -1 )
|
422
|
+
return(-1);
|
423
|
+
|
424
|
+
Symbol = (UINT)(ISym);
|
425
|
+
Num = LzxTblLenBits[Symbol];
|
426
|
+
BITRDR_GETBITS(&Ctx->BitRdr, Num, Val);
|
427
|
+
Len += (Val + LzxTblLenBase[Symbol]);
|
428
|
+
}
|
429
|
+
|
430
|
+
Num = LzxTblOfsBits[Ofs];
|
431
|
+
Ofs = LzxTblOfsBase[Ofs];
|
432
|
+
|
433
|
+
if ( Num < 3 || !Ctx->HasOffs )
|
434
|
+
{
|
435
|
+
BITRDR_GETBITS(&Ctx->BitRdr, Num, Val);
|
436
|
+
Ofs += Val;
|
437
|
+
}
|
438
|
+
else
|
439
|
+
{
|
440
|
+
Num -= 3;
|
441
|
+
BITRDR_GETBITS(&Ctx->BitRdr, Num, Val);
|
442
|
+
if ( (ISym = LzxHuf_DecodeSymbol(&Ctx->HufOffs)) == -1 )
|
443
|
+
return(-1);
|
444
|
+
Ofs += ((UINT)(ISym) + (Val << 3));
|
445
|
+
}
|
446
|
+
|
447
|
+
if ( Ofs < 3 )
|
448
|
+
{
|
449
|
+
/* use saved last offset */
|
450
|
+
Ofs = Ctx->LstOfs[Num = Ofs];
|
451
|
+
if ( Num )
|
452
|
+
{
|
453
|
+
Ctx->LstOfs[Num] = Ctx->LstOfs[0];
|
454
|
+
Ctx->LstOfs[0] = Ofs;
|
455
|
+
}
|
456
|
+
}
|
457
|
+
else
|
458
|
+
{
|
459
|
+
/* update last offset */
|
460
|
+
Ctx->LstOfs[2] = Ctx->LstOfs[1];
|
461
|
+
Ctx->LstOfs[1] = Ctx->LstOfs[0];
|
462
|
+
Ctx->LstOfs[0] = (Ofs -= 3);
|
463
|
+
}
|
464
|
+
|
465
|
+
/* copy phrase */
|
466
|
+
SWD_DELTACOPY(&Ctx->Window, Ofs+1, Len);
|
467
|
+
Written += Len;
|
468
|
+
continue;
|
469
|
+
}
|
470
|
+
|
471
|
+
/* update trees */
|
472
|
+
if ( Lzx_LzxTblBuild(Ctx) )
|
473
|
+
return(-1);
|
474
|
+
}
|
475
|
+
|
476
|
+
return(Written); /* all ok */
|
477
|
+
}
|
478
|
+
|
479
|
+
/* ------------------------------------------------------------------------- */
|