2 * Read VC++ debug information from COFF and eventually
5 * Copyright (C) 1996, Eric Youngdale.
6 * Copyright (C) 1999-2000, Ulrich Weigand.
7 * Copyright (C) 2004, Eric Pouech.
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
25 * Note - this handles reading debug information for 32 bit applications
26 * that run under Windows-NT for example. I doubt that this would work well
27 * for 16 bit applications, but I don't think it really matters since the
28 * file format is different, and we should never get in here in such cases.
31 * Get 16 bit CV stuff working.
32 * Add symbol size to internal symbol table.
36 #include "wine/port.h"
46 #define PATH_MAX MAX_PATH
53 #include "wine/exception.h"
54 #include "wine/debug.h"
55 #include "dbghelp_private.h"
56 #include "wine/mscvpdb.h"
58 WINE_DEFAULT_DEBUG_CHANNEL(dbghelp_coff);
60 /*========================================================================
61 * Process COFF debug information.
66 unsigned int startaddr;
68 struct symt_compiland* compiland;
71 struct symt** entries;
78 struct CoffFile* files;
83 static const char* coff_get_name(const IMAGE_SYMBOL* coff_sym,
84 const char* coff_strtab)
86 static char namebuff[9];
89 if (coff_sym->N.Name.Short)
91 memcpy(namebuff, coff_sym->N.ShortName, 8);
93 nampnt = &namebuff[0];
97 nampnt = coff_strtab + coff_sym->N.Name.Long;
100 if (nampnt[0] == '_') nampnt++;
104 static int coff_add_file(struct CoffFileSet* coff_files, struct module* module,
105 const char* filename)
107 struct CoffFile* file;
109 if (coff_files->nfiles + 1 >= coff_files->nfiles_alloc)
111 if (coff_files->files)
113 coff_files->nfiles_alloc *= 2;
114 coff_files->files = HeapReAlloc(GetProcessHeap(), 0, coff_files->files,
115 coff_files->nfiles_alloc * sizeof(struct CoffFile));
119 coff_files->nfiles_alloc = 16;
120 coff_files->files = HeapAlloc(GetProcessHeap(), 0,
121 coff_files->nfiles_alloc * sizeof(struct CoffFile));
124 file = coff_files->files + coff_files->nfiles;
125 file->startaddr = 0xffffffff;
127 file->compiland = symt_new_compiland(module, 0,
128 source_new(module, NULL, filename));
129 file->linetab_offset = -1;
131 file->entries = NULL;
132 file->neps = file->neps_alloc = 0;
134 return coff_files->nfiles++;
137 static void coff_add_symbol(struct CoffFile* coff_file, struct symt* sym)
139 if (coff_file->neps + 1 >= coff_file->neps_alloc)
141 if (coff_file->entries)
143 coff_file->neps_alloc *= 2;
144 coff_file->entries = HeapReAlloc(GetProcessHeap(), 0, coff_file->entries,
145 coff_file->neps_alloc * sizeof(struct symt*));
149 coff_file->neps_alloc = 32;
150 coff_file->entries = HeapAlloc(GetProcessHeap(), 0,
151 coff_file->neps_alloc * sizeof(struct symt*));
154 coff_file->entries[coff_file->neps++] = sym;
157 DECLSPEC_HIDDEN BOOL coff_process_info(const struct msc_debug_info* msc_dbg)
159 const IMAGE_AUX_SYMBOL* aux;
160 const IMAGE_COFF_SYMBOLS_HEADER* coff;
161 const IMAGE_LINENUMBER* coff_linetab;
162 const IMAGE_LINENUMBER* linepnt;
163 const char* coff_strtab;
164 const IMAGE_SYMBOL* coff_sym;
165 const IMAGE_SYMBOL* coff_symbols;
166 struct CoffFileSet coff_files;
167 int curr_file_idx = -1;
178 TRACE("Processing COFF symbols...\n");
180 assert(sizeof(IMAGE_SYMBOL) == IMAGE_SIZEOF_SYMBOL);
181 assert(sizeof(IMAGE_LINENUMBER) == IMAGE_SIZEOF_LINENUMBER);
183 coff_files.files = NULL;
184 coff_files.nfiles = coff_files.nfiles_alloc = 0;
186 coff = (const IMAGE_COFF_SYMBOLS_HEADER*)msc_dbg->root;
188 coff_symbols = (const IMAGE_SYMBOL*)((const char *)coff + coff->LvaToFirstSymbol);
189 coff_linetab = (const IMAGE_LINENUMBER*)((const char *)coff + coff->LvaToFirstLinenumber);
190 coff_strtab = (const char*)(coff_symbols + coff->NumberOfSymbols);
194 for (i = 0; i < coff->NumberOfSymbols; i++)
196 coff_sym = coff_symbols + i;
197 naux = coff_sym->NumberOfAuxSymbols;
199 if (coff_sym->StorageClass == IMAGE_SYM_CLASS_FILE)
201 curr_file_idx = coff_add_file(&coff_files, msc_dbg->module,
202 (const char*)(coff_sym + 1));
203 TRACE("New file %s\n", (const char*)(coff_sym + 1));
208 if (curr_file_idx < 0)
210 assert(coff_files.nfiles == 0 && coff_files.nfiles_alloc == 0);
211 curr_file_idx = coff_add_file(&coff_files, msc_dbg->module, "<none>");
212 TRACE("New file <none>\n");
216 * This guy marks the size and location of the text section
217 * for the current file. We need to keep track of this so
218 * we can figure out what file the different global functions
221 if (coff_sym->StorageClass == IMAGE_SYM_CLASS_STATIC &&
222 naux != 0 && coff_sym->Type == 0 && coff_sym->SectionNumber == 1)
224 aux = (const IMAGE_AUX_SYMBOL*) (coff_sym + 1);
226 if (coff_files.files[curr_file_idx].linetab_offset != -1)
229 * Save this so we can still get the old name.
233 fn = source_get(msc_dbg->module,
234 coff_files.files[curr_file_idx].compiland->source);
236 TRACE("Duplicating sect from %s: %x %x %x %d %d\n",
237 fn, aux->Section.Length,
238 aux->Section.NumberOfRelocations,
239 aux->Section.NumberOfLinenumbers,
240 aux->Section.Number, aux->Section.Selection);
241 TRACE("More sect %d %s %08x %d %d %d\n",
242 coff_sym->SectionNumber,
243 coff_get_name(coff_sym, coff_strtab),
244 coff_sym->Value, coff_sym->Type,
245 coff_sym->StorageClass, coff_sym->NumberOfAuxSymbols);
248 * Duplicate the file entry. We have no way to describe
249 * multiple text sections in our current way of handling things.
251 coff_add_file(&coff_files, msc_dbg->module, fn);
255 TRACE("New text sect from %s: %x %x %x %d %d\n",
256 source_get(msc_dbg->module, coff_files.files[curr_file_idx].compiland->source),
258 aux->Section.NumberOfRelocations,
259 aux->Section.NumberOfLinenumbers,
260 aux->Section.Number, aux->Section.Selection);
263 if (coff_files.files[curr_file_idx].startaddr > coff_sym->Value)
265 coff_files.files[curr_file_idx].startaddr = coff_sym->Value;
268 if (coff_files.files[curr_file_idx].endaddr < coff_sym->Value + aux->Section.Length)
270 coff_files.files[curr_file_idx].endaddr = coff_sym->Value + aux->Section.Length;
273 coff_files.files[curr_file_idx].linetab_offset = linetab_indx;
274 coff_files.files[curr_file_idx].linecnt = aux->Section.NumberOfLinenumbers;
275 linetab_indx += aux->Section.NumberOfLinenumbers;
280 if (coff_sym->StorageClass == IMAGE_SYM_CLASS_STATIC && naux == 0 &&
281 coff_sym->SectionNumber == 1)
283 DWORD base = msc_dbg->sectp[coff_sym->SectionNumber - 1].VirtualAddress;
285 * This is a normal static function when naux == 0.
286 * Just register it. The current file is the correct
287 * one in this instance.
289 nampnt = coff_get_name(coff_sym, coff_strtab);
291 TRACE("\tAdding static symbol %s\n", nampnt);
293 /* FIXME: was adding symbol to this_file ??? */
294 coff_add_symbol(&coff_files.files[curr_file_idx],
295 &symt_new_function(msc_dbg->module,
296 coff_files.files[curr_file_idx].compiland,
298 msc_dbg->module->module.BaseOfImage + base + coff_sym->Value,
300 NULL /* FIXME */)->symt);
305 if (coff_sym->StorageClass == IMAGE_SYM_CLASS_EXTERNAL &&
306 ISFCN(coff_sym->Type) && coff_sym->SectionNumber > 0)
308 struct symt_compiland* compiland = NULL;
309 DWORD base = msc_dbg->sectp[coff_sym->SectionNumber - 1].VirtualAddress;
310 nampnt = coff_get_name(coff_sym, coff_strtab);
313 i, wine_dbgstr_longlong(msc_dbg->module->module.BaseOfImage + base + coff_sym->Value),
315 TRACE("\tAdding global symbol %s (sect=%s)\n",
316 nampnt, msc_dbg->sectp[coff_sym->SectionNumber - 1].Name);
319 * Now we need to figure out which file this guy belongs to.
321 for (j = 0; j < coff_files.nfiles; j++)
323 if (coff_files.files[j].startaddr <= base + coff_sym->Value
324 && coff_files.files[j].endaddr > base + coff_sym->Value)
326 compiland = coff_files.files[j].compiland;
330 if (j < coff_files.nfiles)
332 coff_add_symbol(&coff_files.files[j],
333 &symt_new_function(msc_dbg->module, compiland, nampnt,
334 msc_dbg->module->module.BaseOfImage + base + coff_sym->Value,
335 0 /* FIXME */, NULL /* FIXME */)->symt);
339 symt_new_function(msc_dbg->module, NULL, nampnt,
340 msc_dbg->module->module.BaseOfImage + base + coff_sym->Value,
341 0 /* FIXME */, NULL /* FIXME */);
347 if (coff_sym->StorageClass == IMAGE_SYM_CLASS_EXTERNAL &&
348 coff_sym->SectionNumber > 0)
350 DWORD base = msc_dbg->sectp[coff_sym->SectionNumber - 1].VirtualAddress;
354 * Similar to above, but for the case of data symbols.
355 * These aren't treated as entrypoints.
357 nampnt = coff_get_name(coff_sym, coff_strtab);
360 i, wine_dbgstr_longlong(msc_dbg->module->module.BaseOfImage + base + coff_sym->Value),
362 TRACE("\tAdding global data symbol %s\n", nampnt);
365 * Now we need to figure out which file this guy belongs to.
367 loc.kind = loc_absolute;
369 loc.offset = msc_dbg->module->module.BaseOfImage + base + coff_sym->Value;
370 symt_new_global_variable(msc_dbg->module, NULL, nampnt, TRUE /* FIXME */,
371 loc, 0 /* FIXME */, NULL /* FIXME */);
376 if (coff_sym->StorageClass == IMAGE_SYM_CLASS_STATIC && naux == 0)
379 * Ignore these. They don't have anything to do with
386 TRACE("Skipping unknown entry '%s' %d %d %d\n",
387 coff_get_name(coff_sym, coff_strtab),
388 coff_sym->StorageClass, coff_sym->SectionNumber, naux);
391 * For now, skip past the aux entries.
396 if (coff_files.files != NULL)
399 * OK, we now should have a list of files, and we should have a list
400 * of entrypoints. We need to sort the entrypoints so that we are
401 * able to tie the line numbers with the given functions within the
404 for (j = 0; j < coff_files.nfiles; j++)
406 if (coff_files.files[j].entries != NULL)
408 qsort(coff_files.files[j].entries, coff_files.files[j].neps,
409 sizeof(struct symt*), symt_cmp_addr);
414 * Now pick apart the line number tables, and attach the entries
415 * to the given functions.
417 for (j = 0; j < coff_files.nfiles; j++)
420 if (coff_files.files[j].neps != 0)
422 for (k = 0; k < coff_files.files[j].linecnt; k++)
424 linepnt = coff_linetab + coff_files.files[j].linetab_offset + k;
426 * If we have spilled onto the next entrypoint, then
431 if (l+1 >= coff_files.files[j].neps) break;
432 symt_get_address(coff_files.files[j].entries[l+1], &addr);
433 if (((msc_dbg->module->module.BaseOfImage + linepnt->Type.VirtualAddress) < addr))
438 if (coff_files.files[j].entries[l+1]->tag == SymTagFunction)
441 * Add the line number. This is always relative to the
442 * start of the function, so we need to subtract that offset
445 symt_get_address(coff_files.files[j].entries[l+1], &addr);
446 symt_add_func_line(msc_dbg->module, (struct symt_function*)coff_files.files[j].entries[l+1],
447 coff_files.files[j].compiland->source, linepnt->Linenumber,
448 msc_dbg->module->module.BaseOfImage + linepnt->Type.VirtualAddress - addr);
454 for (j = 0; j < coff_files.nfiles; j++)
456 HeapFree(GetProcessHeap(), 0, coff_files.files[j].entries);
458 HeapFree(GetProcessHeap(), 0, coff_files.files);
459 msc_dbg->module->module.SymType = SymCoff;
460 /* FIXME: we could have a finer grain here */
461 msc_dbg->module->module.LineNumbers = TRUE;
462 msc_dbg->module->module.GlobalSymbols = TRUE;
463 msc_dbg->module->module.TypeInfo = FALSE;
464 msc_dbg->module->module.SourceIndexed = TRUE;
465 msc_dbg->module->module.Publics = TRUE;