forked from crosire/blink
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpdb_reader.cpp
326 lines (272 loc) · 9.85 KB
/
pdb_reader.cpp
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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
/**
* Copyright (C) 2016 Patrick Mours. All rights reserved.
* License: https://github.com/crosire/blink#license
*/
#include "pdb_reader.hpp"
#include <unordered_set>
/**
* Microsoft program debug database file
*
* File is a multi-stream file with various different data streams. Some streams are located at a fixed index:
* - Stream 1: PDB headers and list of named streams
* - Stream 2: Type info (TPI stream)
* - Stream 3: Debug info (DBI stream)
*/
#pragma region PDB Headers
#pragma pack(1)
struct pdb_header
{
uint32_t version;
uint32_t time_date_stamp;
uint32_t age;
blink::guid guid;
uint32_t names_map_offset;
};
struct pdb_names_header
{
uint32_t signature;
uint32_t version;
uint32_t names_map_offset;
};
struct pdb_link_info_header
{
uint32_t cb;
uint32_t version;
uint32_t cwd_offset;
uint32_t command_offset; // Example: link.exe -re -out:foo.exe
uint32_t out_file_begin_in_command; // Example: 18 (index of 'foo.exe' in command)
uint32_t libs_offset;
};
struct pdb_dbi_header
{
uint32_t signature;
uint32_t version;
uint32_t age;
uint16_t global_symbol_info_stream;
uint16_t toolchain_major : 8;
uint16_t toolchain_minor : 7;
uint16_t new_version_format : 1;
uint16_t public_symbol_info_stream;
uint16_t pdb_dll_build_major;
uint16_t symbol_record_stream;
uint16_t pdb_dll_build_minor;
uint32_t module_info_size;
uint32_t section_contribution_size;
uint32_t section_map_size;
uint32_t file_info_size;
uint32_t ts_map_size;
uint32_t mfc_index;
uint32_t debug_header_size;
uint32_t ec_info_size;
uint16_t incrementally_linked : 1;
uint16_t private_symbols_stripped : 1;
uint16_t has_conflicting_types : 1;
uint16_t padding1 : 13;
uint16_t machine;
uint32_t padding2;
};
struct pdb_dbi_module_info
{
uint32_t is_opened;
struct {
uint16_t index;
uint16_t padding1;
uint32_t offset;
uint32_t size;
uint32_t characteristics;
uint16_t module_index;
uint16_t padding2;
uint32_t data_crc;
uint32_t relocation_crc;
} section;
uint16_t is_dirty : 1;
uint16_t has_ec_info : 1;
uint16_t padding1 : 6;
uint16_t type_server_index : 8;
uint16_t symbol_stream;
uint32_t symbol_byte_size;
uint32_t old_lines_byte_size;
uint32_t lines_byte_size;
uint16_t num_source_files;
uint16_t padding2;
uint32_t offsets;
uint32_t source_file_name_index;
uint32_t pdb_file_name_index;
};
struct pdb_dbi_debug_header
{
uint16_t fpo; // IMAGE_DEBUG_TYPE_FPO
uint16_t exception; // IMAGE_DEBUG_TYPE_EXCEPTION
uint16_t fixup; // IMAGE_DEBUG_TYPE_FIXUP
uint16_t omap_to_src; // IMAGE_DEBUG_TYPE_OMAP_TO_SRC
uint16_t omap_from_src; // IMAGE_DEBUG_TYPE_OMAP_FROM_SRC
uint16_t section_header; // A dump of all section headers from the executable
uint16_t token_rid_map;
uint16_t xdata; // A dump of the .xdata section from the executable
uint16_t pdata;
uint16_t new_fpo;
uint16_t section_header_orig;
};
struct pdb_dbi_section_header
{
char name[8];
uint32_t size;
uint32_t virtual_address;
uint32_t data_size;
uint32_t raw_data_rva;
uint32_t relocation_table_rva;
uint32_t line_numbers_rva;
uint16_t num_relocations;
uint16_t num_line_numbers;
uint32_t flags;
};
#pragma endregion
blink::pdb_reader::pdb_reader(const std::string &path) : msf_reader(path)
{
// PDB files should have 4 streams at the beginning that are always at the same index
_is_valid &= stream_count() > 4;
if (!is_valid())
return;
// Read PDB info stream
stream_reader pdb_stream(msf_reader::stream(1));
if (pdb_stream.size() == 0)
return;
const pdb_header &header = pdb_stream.read<pdb_header>();
_version = header.version;
_timestamp = header.time_date_stamp;
_guid = header.guid;
// Read stream names from string hash map
pdb_stream.skip(header.names_map_offset);
const auto count = pdb_stream.read<uint32_t>();
const auto hash_table_size = pdb_stream.read<uint32_t>();
_named_streams.reserve(count);
const auto num_bitset_present = pdb_stream.read<uint32_t>();
std::vector<uint32_t> bitset_present(num_bitset_present);
pdb_stream.read(bitset_present.data(), num_bitset_present * sizeof(uint32_t));
const auto num_bitset_deleted = pdb_stream.read<uint32_t>();
pdb_stream.skip(num_bitset_deleted * sizeof(uint32_t));
for (uint32_t i = 0; i < hash_table_size; i++)
{
if ((bitset_present[i / 32] & (1 << (i % 32))) == 0)
continue;
const auto name_offset = pdb_stream.read<uint32_t>();
const auto stream_index = pdb_stream.read<uint32_t>();
const auto pos = pdb_stream.tell();
pdb_stream.seek(sizeof(header) + name_offset); // Seek into the string table that stores the name
const std::string name(pdb_stream.read_string());
pdb_stream.seek(pos); // Seek previous position in stream to read next name offset in the next iteration of this loop
_named_streams.insert({ name, stream_index });
}
}
void blink::pdb_reader::read_symbol_table(uint8_t *image_base, std::unordered_map<std::string, void *> &symbols)
{
stream_reader stream(msf_reader::stream(3));
const pdb_dbi_header &header = stream.read<pdb_dbi_header>();
if (header.signature != 0xFFFFFFFF)
return;
// Find debug header stream (https://llvm.org/docs/PDB/DbiStream.html#optional-debug-header-stream)
stream.skip(header.module_info_size + header.section_contribution_size + header.section_map_size + header.file_info_size + header.ts_map_size + header.ec_info_size);
const pdb_dbi_debug_header &debug_header = stream.read<pdb_dbi_debug_header>();
// Read section headers
stream_reader section_stream(msf_reader::stream(debug_header.section_header));
const size_t num_sections = section_stream.size() / sizeof(pdb_dbi_section_header);
const pdb_dbi_section_header *sections = section_stream.data<pdb_dbi_section_header>();
// Read symbol table records in CodeView format
stream = msf_reader::stream(header.symbol_record_stream);
parse_code_view_records(stream, [&](uint16_t tag) {
if (tag != 0x110E) // S_PUB32
return; // Skip all records that are not about public symbols
const struct PUBSYM32 {
uint32_t flags;
uint32_t offset;
uint16_t section;
const char name[1];
} &sym = *stream.data<const PUBSYM32>();
if (sym.section == 0 || sym.section > num_sections)
symbols[sym.name] = reinterpret_cast<void *>(static_cast<uintptr_t>(sym.offset)); // Relative address
else
symbols[sym.name] = image_base + sections[sym.section - 1].virtual_address + sym.offset; // Absolute address
}, 4);
}
void blink::pdb_reader::read_object_files(std::vector<std::filesystem::path> &object_files)
{
stream_reader stream(msf_reader::stream(3));
const pdb_dbi_header &header = stream.read<pdb_dbi_header>();
if (header.signature != 0xFFFFFFFF)
return;
// Read module information stream (https://llvm.org/docs/PDB/DbiStream.html#dbi-mod-info-substream)
while (stream.tell() < sizeof(pdb_dbi_header) + header.module_info_size)
{
stream.skip(sizeof(pdb_dbi_module_info));
const auto module_name = stream.read_string();
const auto obj_file_name = stream.read_string(); // Contains the name of the ".lib" if this object file is part of a library
object_files.push_back(module_name);
stream.align(4);
}
}
void blink::pdb_reader::read_source_files(std::vector<std::vector<std::filesystem::path>> &source_files)
{
stream_reader stream(msf_reader::stream(3));
const pdb_dbi_header &header = stream.read<pdb_dbi_header>();
if (header.signature != 0xFFFFFFFF)
return;
// Find file information stream (https://llvm.org/docs/PDB/DbiStream.html#file-info-substream)
stream.skip(header.module_info_size + header.section_contribution_size + header.section_map_size);
const uint16_t num_modules = stream.read<uint16_t>();
stream.skip(2); // Skip old number of file names (see comment on counting the number below)
const uint16_t *const module_file_offsets = stream.data<uint16_t>();
const uint16_t *const module_num_source_files = stream.data<uint16_t>(num_modules * sizeof(uint16_t));
const uint32_t *const file_name_offsets = stream.data<uint32_t>(num_modules * sizeof(uint16_t) * 2);
// Count number of source files instead of reading the value from the header, since there may be more source files that would fit into a 16-bit value
uint32_t num_source_files = 0;
for (uint16_t i = 0; i < num_modules; ++i)
num_source_files += module_num_source_files[i];
stream.skip(num_modules * sizeof(uint16_t) * 2 + num_source_files * sizeof(uint32_t));
const auto offset = stream.tell();
source_files.resize(num_modules);
for (uint32_t k = 0; k < num_modules; ++k)
{
for (uint32_t i = 0; i < module_num_source_files[k]; ++i)
{
stream.seek(offset + file_name_offsets[module_file_offsets[k] + i]);
source_files[k].push_back(stream.read_string());
}
}
}
void blink::pdb_reader::read_link_info(std::filesystem::path &cwd, std::string &cmd)
{
stream_reader stream(this->stream("/LinkInfo"));
if (!is_valid() || stream.size() == 0)
return;
// See https://github.com/Microsoft/microsoft-pdb/blob/master/langapi/include/pdb.h#L500
stream.skip(sizeof(pdb_link_info_header));
cwd = stream.read_string();
cmd = stream.read_string();
// Followed by another null-terminated string with all linked libraries
}
void blink::pdb_reader::read_name_hash_table(std::unordered_map<uint32_t, std::string> &names)
{
stream_reader stream(this->stream("/names"));
if (!is_valid() || stream.size() == 0)
return;
// Read names stream
const pdb_names_header &header = stream.read<pdb_names_header>();
if (header.signature != 0xEFFEEFFE || header.version != 1)
return;
// Read the string table
stream.skip(header.names_map_offset);
const auto size = stream.read<uint32_t>();
names.reserve(size);
for (uint32_t i = 0; i < size; i++)
{
const auto name_offset = stream.read<uint32_t>();
// Skip empty entries
if (name_offset == 0)
continue;
const auto pos = stream.tell();
stream.seek(sizeof(header) + name_offset);
const std::string name(stream.read_string());
stream.seek(pos); // Seek previous position in stream to read next name offset in the next iteration of this loop
names.insert({ i, name });
}
}