Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
wine-mirror
GitHub Repository: wine-mirror/wine
Path: blob/master/tools/winedump/dump.c
4389 views
1
/*
2
* File dumping utility
3
*
4
* Copyright 2001,2007 Eric Pouech
5
*
6
* This library is free software; you can redistribute it and/or
7
* modify it under the terms of the GNU Lesser General Public
8
* License as published by the Free Software Foundation; either
9
* version 2.1 of the License, or (at your option) any later version.
10
*
11
* This library is distributed in the hope that it will be useful,
12
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14
* Lesser General Public License for more details.
15
*
16
* You should have received a copy of the GNU Lesser General Public
17
* License along with this library; if not, write to the Free Software
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19
*/
20
21
#include "config.h"
22
23
#include <stdlib.h>
24
#include <stdarg.h>
25
#include <stdio.h>
26
#include <time.h>
27
28
#include "../tools.h"
29
#include "windef.h"
30
#include "winbase.h"
31
#include "winedump.h"
32
33
void *dump_base = NULL;
34
size_t dump_total_len = 0;
35
36
void dump_data_offset( const unsigned char *ptr, unsigned int size, unsigned int offset, const char *prefix )
37
{
38
unsigned int i, j;
39
40
printf( "%s%08x: ", prefix, offset );
41
if (!ptr)
42
{
43
printf("NULL\n");
44
return;
45
}
46
for (i = 0; i < size; i++)
47
{
48
printf( "%02x%c", ptr[i], (i % 16 == 7) ? '-' : ' ' );
49
if ((i % 16) == 15)
50
{
51
printf( " " );
52
for (j = 0; j < 16; j++)
53
printf( "%c", isprint(ptr[i-15+j]) ? ptr[i-15+j] : '.' );
54
if (i < size-1) printf( "\n%s%08x: ", prefix, offset + i + 1 );
55
}
56
}
57
if (i % 16)
58
{
59
printf( "%*s ", 3 * (16-(i%16)), "" );
60
for (j = 0; j < i % 16; j++)
61
printf( "%c", isprint(ptr[i-(i%16)+j]) ? ptr[i-(i%16)+j] : '.' );
62
}
63
printf( "\n" );
64
}
65
66
void dump_data( const unsigned char *ptr, unsigned int size, const char *prefix )
67
{
68
dump_data_offset( ptr, size, 0, prefix );
69
}
70
71
static char* dump_want_n(unsigned sz)
72
{
73
static char buffer[64 * 1024];
74
static unsigned idx;
75
char* ret;
76
77
assert(sz < sizeof(buffer));
78
if (idx + sz >= sizeof(buffer)) idx = 0;
79
ret = &buffer[idx];
80
idx += sz;
81
return ret;
82
}
83
84
const char *get_time_str(unsigned long _t)
85
{
86
const time_t t = (const time_t)_t;
87
const char *str = ctime(&t);
88
size_t len;
89
char* buf;
90
91
if (!str) return "not valid time";
92
93
len = strlen(str);
94
/* FIXME: I don't get the same values from MS' pedump running under Wine...
95
* I wonder if Wine isn't broken wrt to GMT settings...
96
*/
97
if (len && str[len-1] == '\n') len--;
98
buf = dump_want_n(len + 1);
99
if (buf)
100
{
101
memcpy( buf, str, len );
102
buf[len] = 0;
103
}
104
return buf;
105
}
106
107
unsigned int strlenW( const WCHAR *str )
108
{
109
const WCHAR *s = str;
110
while (*s) s++;
111
return s - str;
112
}
113
114
void dump_unicode_str( const WCHAR *str, int len )
115
{
116
if (len == -1) len = strlenW( str );
117
printf( "L\"");
118
while (len-- > 0 && *str)
119
{
120
WCHAR c = *str++;
121
switch (c)
122
{
123
case '\n': printf( "\\n" ); break;
124
case '\r': printf( "\\r" ); break;
125
case '\t': printf( "\\t" ); break;
126
case '"': printf( "\\\"" ); break;
127
case '\\': printf( "\\\\" ); break;
128
default:
129
if (c >= ' ' && c <= 126) putchar(c);
130
else printf( "\\u%04x",c);
131
}
132
}
133
printf( "\"" );
134
}
135
136
const char *get_hexint64_str( DWORD64 l )
137
{
138
char *buf = dump_want_n(2 + 16 + 1);
139
if (sizeof(l) > sizeof(unsigned long) && l >> 32)
140
sprintf(buf, "%#lx%08lx", (unsigned long)(l >> 32), (unsigned long)l);
141
else
142
sprintf(buf, "%#lx", (unsigned long)l);
143
assert(strlen(buf) <= 18);
144
return buf;
145
}
146
147
const char *get_uint64_str( DWORD64 l )
148
{
149
char *buf = dump_want_n( 32 );
150
char *ptr = buf + 31;
151
*ptr = '\0';
152
for ( ; l; l /= 10)
153
*--ptr = '0' + (l % 10);
154
if (ptr == buf + 31) *--ptr = '0';
155
assert(ptr >= buf);
156
return ptr;
157
}
158
159
const char* get_symbol_str(const char* symname)
160
{
161
const char* ret = NULL;
162
163
if (!symname) return "(nil)";
164
if (globals.do_demangle) ret = demangle( symname );
165
return ret ? ret : symname;
166
}
167
168
const char* get_guid_str(const GUID* guid)
169
{
170
char* str;
171
172
str = dump_want_n(39);
173
if (str)
174
sprintf(str, "{%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}",
175
(unsigned int)guid->Data1, guid->Data2, guid->Data3,
176
guid->Data4[0], guid->Data4[1], guid->Data4[2], guid->Data4[3],
177
guid->Data4[4], guid->Data4[5], guid->Data4[6], guid->Data4[7]);
178
return str;
179
}
180
181
const char *get_unicode_str( const WCHAR *str, int len )
182
{
183
char *buffer;
184
int i = 0;
185
186
if (!str) return "(null)";
187
if (len == -1) len = strlenW( str );
188
buffer = dump_want_n( len * 6 + 3);
189
buffer[i++] = '"';
190
while (len-- > 0 && *str)
191
{
192
WCHAR c = *str++;
193
switch (c)
194
{
195
case '\n': strcpy( buffer + i, "\\n" ); i += 2; break;
196
case '\r': strcpy( buffer + i, "\\r" ); i += 2; break;
197
case '\t': strcpy( buffer + i, "\\t" ); i += 2; break;
198
case '"': strcpy( buffer + i, "\\\"" ); i += 2; break;
199
case '\\': strcpy( buffer + i, "\\\\" ); i += 2; break;
200
default:
201
if (c >= ' ' && c <= 126) buffer[i++] = c;
202
else i += sprintf( buffer + i, "\\u%04x",c);
203
}
204
}
205
buffer[i++] = '"';
206
buffer[i] = 0;
207
return buffer;
208
}
209
210
const void* PRD(unsigned long prd, unsigned long len)
211
{
212
return (prd + len > dump_total_len) ? NULL : (const char*)dump_base + prd;
213
}
214
215
unsigned long Offset(const void* ptr)
216
{
217
if (ptr < dump_base) {printf("<<<<<ptr below\n");return 0;}
218
if ((const char *)ptr >= (const char*)dump_base + dump_total_len) {printf("<<<<<ptr above\n");return 0;}
219
return (const char *)ptr - (const char *)dump_base;
220
}
221
222
static const struct dumper
223
{
224
enum FileSig kind;
225
enum FileSig (*get_kind)(void);
226
file_dumper dumper; /* default dump tool */
227
enum FileSig (*alt_get_kind)( int fd );
228
void (*alt_dumper)( int fd );
229
}
230
dumpers[] =
231
{
232
{SIG_DOS, get_kind_exec, dos_dump},
233
{SIG_PE, get_kind_exec, pe_dump},
234
{SIG_DBG, get_kind_dbg, dbg_dump},
235
{SIG_PDB, .alt_get_kind = get_kind_pdb, .alt_dumper = pdb_dump},
236
{SIG_NE, get_kind_exec, ne_dump},
237
{SIG_LE, get_kind_exec, le_dump},
238
{SIG_COFFLIB, get_kind_lib, lib_dump},
239
{SIG_MDMP, get_kind_mdmp, mdmp_dump},
240
{SIG_LNK, get_kind_lnk, lnk_dump},
241
{SIG_EMF, get_kind_emf, emf_dump},
242
{SIG_EMFSPOOL, get_kind_emfspool, emfspool_dump},
243
{SIG_MF, get_kind_mf, mf_dump},
244
{SIG_FNT, get_kind_fnt, fnt_dump},
245
{SIG_TLB, get_kind_tlb, tlb_dump},
246
{SIG_NLS, get_kind_nls, nls_dump},
247
{SIG_REG, get_kind_reg, reg_dump},
248
{SIG_UNKNOWN, NULL, NULL} /* sentinel */
249
};
250
251
BOOL dump_analysis(const char *name, file_dumper fn, enum FileSig wanted_sig)
252
{
253
BOOL ret = TRUE;
254
const struct dumper*dpr;
255
int fd;
256
struct stat st;
257
258
setbuf(stdout, NULL);
259
260
if ((fd = open( name, O_RDONLY | O_BINARY )) == -1) fatal( "Cannot read file" );
261
fstat( fd, &st );
262
printf("Contents of %s: %llu bytes\n\n", name, (unsigned long long)st.st_size);
263
264
for (dpr = dumpers; dpr->kind != SIG_UNKNOWN; dpr++)
265
{
266
if (!dpr->alt_get_kind) continue;
267
/* alt interface isn't compatible with incoming file_dumper */
268
if (wanted_sig == dpr->kind)
269
assert( !fn );
270
271
lseek( fd, 0, SEEK_SET );
272
if (dpr->alt_get_kind( fd ) == dpr->kind &&
273
(wanted_sig == SIG_UNKNOWN || wanted_sig == dpr->kind))
274
{
275
lseek( fd, 0, SEEK_SET );
276
dpr->alt_dumper( fd );
277
break;
278
}
279
}
280
close(fd);
281
282
if (dpr->kind == SIG_UNKNOWN)
283
{
284
if (!(dump_base = read_file( name, &dump_total_len ))) fatal( "Cannot read file" );
285
286
for (dpr = dumpers; dpr->kind != SIG_UNKNOWN; dpr++)
287
{
288
if (!dpr->get_kind) continue;
289
if (dpr->get_kind() == dpr->kind &&
290
(wanted_sig == SIG_UNKNOWN || wanted_sig == dpr->kind))
291
{
292
if (fn) fn(); else dpr->dumper();
293
break;
294
}
295
}
296
if (dpr->kind == SIG_UNKNOWN)
297
{
298
printf("Can't get a suitable file signature, aborting\n");
299
ret = FALSE;
300
}
301
302
free( dump_base );
303
}
304
if (ret) printf("Done dumping %s\n", name);
305
306
return ret;
307
}
308
309
void dump_file(const char* name)
310
{
311
dump_analysis(name, NULL, SIG_UNKNOWN);
312
}
313
314