/* * File symbol.c - management of symbols (lexical tree) * * Copyright (C) 1993, Eric Youngdale. * 2004, Eric Pouech * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #define NONAMELESSUNION #define NONAMELESSSTRUCT #include "config.h" #include #include #include #include #include #include #ifdef HAVE_REGEX_H # include #endif #include "wine/debug.h" #include "dbghelp_private.h" WINE_DEFAULT_DEBUG_CHANNEL(dbghelp); WINE_DECLARE_DEBUG_CHANNEL(dbghelp_symt); struct line_info { unsigned long is_first : 1, is_last : 1, is_source_file : 1, line_number; union { unsigned long pc_offset; /* if is_source_file isn't set */ unsigned source_file; /* if is_source_file is set */ } u; }; inline static int cmp_addr(ULONG64 a1, ULONG64 a2) { if (a1 > a2) return 1; if (a1 < a2) return -1; return 0; } inline static int cmp_sorttab_addr(const struct module* module, int idx, ULONG64 addr) { ULONG64 ref; symt_get_info(&module->addr_sorttab[idx]->symt, TI_GET_ADDRESS, &ref); return cmp_addr(ref, addr); } int symt_cmp_addr(const void* p1, const void* p2) { const struct symt* sym1 = *(const struct symt* const *)p1; const struct symt* sym2 = *(const struct symt* const *)p2; ULONG64 a1, a2; symt_get_info(sym1, TI_GET_ADDRESS, &a1); symt_get_info(sym2, TI_GET_ADDRESS, &a2); return cmp_addr(a1, a2); } static inline void re_append(char** mask, unsigned* len, char ch) { *mask = HeapReAlloc(GetProcessHeap(), 0, *mask, ++(*len)); (*mask)[*len - 2] = ch; } /* transforms a dbghelp's regular expression into a POSIX one * Here are the valid dbghelp reg ex characters: * * 0 or more characters * ? a single character * [] list * # 0 or more of preceding char * + 1 or more of preceding char * escapes \ on #, ?, [, ], *, +. don't work on - */ static void compile_regex(const char* str, int numchar, regex_t* re) { char* mask = HeapAlloc(GetProcessHeap(), 0, 1); unsigned len = 1; BOOL in_escape = FALSE; re_append(&mask, &len, '^'); while (*str && numchar--) { /* FIXME: this shouldn't be valid on '-' */ if (in_escape) { re_append(&mask, &len, '\\'); re_append(&mask, &len, *str); in_escape = FALSE; } else switch (*str) { case '\\': in_escape = TRUE; break; case '*': re_append(&mask, &len, '.'); re_append(&mask, &len, '*'); break; case '?': re_append(&mask, &len, '.'); break; case '#': re_append(&mask, &len, '*'); break; /* escape some valid characters in dbghelp reg exp:s */ case '$': re_append(&mask, &len, '\\'); re_append(&mask, &len, '$'); break; /* +, [, ], - are the same in dbghelp & POSIX, use them as any other char */ default: re_append(&mask, &len, *str); break; } str++; } if (in_escape) { re_append(&mask, &len, '\\'); re_append(&mask, &len, '\\'); } re_append(&mask, &len, '$'); mask[len - 1] = '\0'; if (regcomp(re, mask, REG_NOSUB)) FIXME("Couldn't compile %s\n", mask); HeapFree(GetProcessHeap(), 0, mask); } struct symt_compiland* symt_new_compiland(struct module* module, const char* name) { struct symt_compiland* sym; TRACE_(dbghelp_symt)("Adding compiland symbol %s:%s\n", module->module.ModuleName, name); if ((sym = pool_alloc(&module->pool, sizeof(*sym)))) { sym->symt.tag = SymTagCompiland; sym->source = source_new(module, name); vector_init(&sym->vchildren, sizeof(struct symt*), 32); } return sym; } struct symt_public* symt_new_public(struct module* module, struct symt_compiland* compiland, const char* name, unsigned long address, unsigned size, BOOL in_code, BOOL is_func) { struct symt_public* sym; struct symt** p; TRACE_(dbghelp_symt)("Adding public symbol %s:%s @%lx\n", module->module.ModuleName, name, address); if ((dbghelp_options & SYMOPT_AUTO_PUBLICS) && symt_find_nearest(module, address) != -1) return NULL; if ((sym = pool_alloc(&module->pool, sizeof(*sym)))) { sym->symt.tag = SymTagPublicSymbol; sym->hash_elt.name = pool_strdup(&module->pool, name); hash_table_add(&module->ht_symbols, &sym->hash_elt); module->sortlist_valid = FALSE; sym->container = compiland ? &compiland->symt : NULL; sym->address = address; sym->size = size; sym->in_code = in_code; sym->is_function = is_func; if (compiland) { p = vector_add(&compiland->vchildren, &module->pool); *p = &sym->symt; } } return sym; } struct symt_data* symt_new_global_variable(struct module* module, struct symt_compiland* compiland, const char* name, unsigned is_static, unsigned long addr, unsigned long size, struct symt* type) { struct symt_data* sym; struct symt** p; DWORD tsz; TRACE_(dbghelp_symt)("Adding global symbol %s:%s @%lx %p\n", module->module.ModuleName, name, addr, type); if ((sym = pool_alloc(&module->pool, sizeof(*sym)))) { sym->symt.tag = SymTagData; sym->hash_elt.name = pool_strdup(&module->pool, name); hash_table_add(&module->ht_symbols, &sym->hash_elt); module->sortlist_valid = FALSE; sym->kind = is_static ? DataIsFileStatic : DataIsGlobal; sym->container = compiland ? &compiland->symt : NULL; sym->type = type; sym->u.address = addr; if (type && size && symt_get_info(type, TI_GET_LENGTH, &tsz)) { if (tsz != size) FIXME("Size mismatch for %s.%s between type (%lu) and src (%lu)\n", module->module.ModuleName, name, tsz, size); } if (compiland) { p = vector_add(&compiland->vchildren, &module->pool); *p = &sym->symt; } } return sym; } struct symt_function* symt_new_function(struct module* module, struct symt_compiland* compiland, const char* name, unsigned long addr, unsigned long size, struct symt* sig_type) { struct symt_function* sym; struct symt** p; TRACE_(dbghelp_symt)("Adding global function %s:%s @%lx-%lx\n", module->module.ModuleName, name, addr, addr + size - 1); assert(!sig_type || sig_type->tag == SymTagFunctionType); if ((sym = pool_alloc(&module->pool, sizeof(*sym)))) { sym->symt.tag = SymTagFunction; sym->hash_elt.name = pool_strdup(&module->pool, name); hash_table_add(&module->ht_symbols, &sym->hash_elt); module->sortlist_valid = FALSE; sym->container = &compiland->symt; sym->address = addr; sym->type = sig_type; sym->size = size; vector_init(&sym->vlines, sizeof(struct line_info), 64); vector_init(&sym->vchildren, sizeof(struct symt*), 8); if (compiland) { p = vector_add(&compiland->vchildren, &module->pool); *p = &sym->symt; } } return sym; } void symt_add_func_line(struct module* module, struct symt_function* func, unsigned source_idx, int line_num, unsigned long offset) { struct line_info* dli; BOOL last_matches = FALSE; if (func == NULL || !(dbghelp_options & SYMOPT_LOAD_LINES)) return; TRACE_(dbghelp_symt)("(%p)%s:%lx %s:%u\n", func, func->hash_elt.name, offset, source_get(module, source_idx), line_num); assert(func->symt.tag == SymTagFunction); dli = NULL; while ((dli = vector_iter_down(&func->vlines, dli))) { if (dli->is_source_file) { last_matches = (source_idx == dli->u.source_file); break; } } if (!last_matches) { /* we shouldn't have line changes on first line of function */ dli = vector_add(&func->vlines, &module->pool); dli->is_source_file = 1; dli->is_first = dli->is_last = 0; dli->line_number = 0; dli->u.source_file = source_idx; } dli = vector_add(&func->vlines, &module->pool); dli->is_source_file = 0; dli->is_first = dli->is_last = 0; dli->line_number = line_num; dli->u.pc_offset = func->address + offset; } struct symt_data* symt_add_func_local(struct module* module, struct symt_function* func, int regno, int offset, struct symt_block* block, struct symt* type, const char* name) { struct symt_data* locsym; struct symt** p; assert(func); assert(func->symt.tag == SymTagFunction); TRACE_(dbghelp_symt)("Adding local symbol (%s:%s): %s %p\n", module->module.ModuleName, func->hash_elt.name, name, type); locsym = pool_alloc(&module->pool, sizeof(*locsym)); locsym->symt.tag = SymTagData; locsym->hash_elt.name = pool_strdup(&module->pool, name); locsym->hash_elt.next = NULL; locsym->kind = (offset < 0) ? DataIsParam : DataIsLocal; locsym->container = &block->symt; locsym->type = type; if (regno) { locsym->u.s.reg_id = regno; locsym->u.s.offset = 0; locsym->u.s.length = 0; } else { locsym->u.s.reg_id = 0; locsym->u.s.offset = offset * 8; locsym->u.s.length = 0; } if (block) p = vector_add(&block->vchildren, &module->pool); else p = vector_add(&func->vchildren, &module->pool); *p = &locsym->symt; return locsym; } struct symt_block* symt_open_func_block(struct module* module, struct symt_function* func, struct symt_block* parent_block, unsigned pc, unsigned len) { struct symt_block* block; struct symt** p; assert(func); assert(func->symt.tag == SymTagFunction); assert(!parent_block || parent_block->symt.tag == SymTagBlock); block = pool_alloc(&module->pool, sizeof(*block)); block->symt.tag = SymTagBlock; block->address = func->address + pc; block->size = len; block->container = parent_block ? &parent_block->symt : &func->symt; vector_init(&block->vchildren, sizeof(struct symt*), 4); if (parent_block) p = vector_add(&parent_block->vchildren, &module->pool); else p = vector_add(&func->vchildren, &module->pool); *p = &block->symt; return block; } struct symt_block* symt_close_func_block(struct module* module, struct symt_function* func, struct symt_block* block, unsigned pc) { assert(func->symt.tag == SymTagFunction); if (pc) block->size = func->address + pc - block->address; return (block->container->tag == SymTagBlock) ? GET_ENTRY(block->container, struct symt_block, symt) : NULL; } struct symt_function_point* symt_add_function_point(struct module* module, struct symt_function* func, enum SymTagEnum point, unsigned offset, const char* name) { struct symt_function_point* sym; struct symt** p; if ((sym = pool_alloc(&module->pool, sizeof(*sym)))) { sym->symt.tag = point; sym->parent = func; sym->offset = offset; sym->name = name ? pool_strdup(&module->pool, name) : NULL; p = vector_add(&func->vchildren, &module->pool); *p = &sym->symt; } return sym; } BOOL symt_normalize_function(struct module* module, struct symt_function* func) { unsigned len; struct line_info* dli; assert(func); /* We aren't adding any more locals or line numbers to this function. * Free any spare memory that we might have allocated. */ assert(func->symt.tag == SymTagFunction); /* EPP vector_pool_normalize(&func->vlines, &module->pool); */ /* EPP vector_pool_normalize(&func->vchildren, &module->pool); */ len = vector_length(&func->vlines); if (len--) { dli = vector_at(&func->vlines, 0); dli->is_first = 1; dli = vector_at(&func->vlines, len); dli->is_last = 1; } return TRUE; } struct symt_thunk* symt_new_thunk(struct module* module, struct symt_compiland* compiland, const char* name, THUNK_ORDINAL ord, unsigned long addr, unsigned long size) { struct symt_thunk* sym; TRACE_(dbghelp_symt)("Adding global thunk %s:%s @%lx-%lx\n", module->module.ModuleName, name, addr, addr + size - 1); if ((sym = pool_alloc(&module->pool, sizeof(*sym)))) { sym->symt.tag = SymTagThunk; sym->hash_elt.name = pool_strdup(&module->pool, name); hash_table_add(&module->ht_symbols, &sym->hash_elt); module->sortlist_valid = FALSE; sym->container = &compiland->symt; sym->address = addr; sym->size = size; sym->ordinal = ord; if (compiland) { struct symt** p; p = vector_add(&compiland->vchildren, &module->pool); *p = &sym->symt; } } return sym; } /* expect sym_info->MaxNameLen to be set before being called */ static void symt_fill_sym_info(const struct module* module, const struct symt* sym, SYMBOL_INFO* sym_info) { const char* name; sym_info->TypeIndex = (DWORD)sym; sym_info->info = 0; /* TBD */ symt_get_info(sym, TI_GET_LENGTH, &sym_info->Size); sym_info->ModBase = module->module.BaseOfImage; sym_info->Flags = 0; switch (sym->tag) { case SymTagData: { const struct symt_data* data = (const struct symt_data*)sym; switch (data->kind) { case DataIsLocal: case DataIsParam: if (data->u.s.reg_id) { sym_info->Flags |= SYMFLAG_LOCAL | SYMFLAG_REGISTER; sym_info->Register = data->u.s.reg_id; sym_info->Address = 0; } else { if (data->u.s.offset < 0) sym_info->Flags |= SYMFLAG_LOCAL | SYMFLAG_FRAMEREL; else sym_info->Flags |= SYMFLAG_PARAMETER | SYMFLAG_FRAMEREL; /* FIXME: needed ? moreover, it's i386 dependent !!! */ sym_info->Register = CV_REG_EBP; sym_info->Address = data->u.s.offset; } break; case DataIsGlobal: case DataIsFileStatic: symt_get_info(sym, TI_GET_ADDRESS, &sym_info->Address); sym_info->Register = 0; break; case DataIsConstant: sym_info->Flags |= SYMFLAG_VALUEPRESENT; switch (data->u.value.n1.n2.vt) { case VT_I4: sym_info->Value = (ULONG)data->u.value.n1.n2.n3.lVal; break; case VT_I2: sym_info->Value = (ULONG)(long)data->u.value.n1.n2.n3.iVal; break; case VT_I1: sym_info->Value = (ULONG)(long)data->u.value.n1.n2.n3.cVal; break; case VT_UI4: sym_info->Value = (ULONG)data->u.value.n1.n2.n3.ulVal; break; case VT_UI2: sym_info->Value = (ULONG)data->u.value.n1.n2.n3.uiVal; break; case VT_UI1: sym_info->Value = (ULONG)data->u.value.n1.n2.n3.bVal; break; default: FIXME("Unsupported variant type (%u)\n", data->u.value.n1.n2.vt); } break; default: FIXME("Unhandled kind (%u) in sym data\n", data->kind); } } break; case SymTagPublicSymbol: sym_info->Flags |= SYMFLAG_EXPORT; symt_get_info(sym, TI_GET_ADDRESS, &sym_info->Address); break; case SymTagFunction: sym_info->Flags |= SYMFLAG_FUNCTION; symt_get_info(sym, TI_GET_ADDRESS, &sym_info->Address); break; case SymTagThunk: sym_info->Flags |= SYMFLAG_THUNK; symt_get_info(sym, TI_GET_ADDRESS, &sym_info->Address); break; default: symt_get_info(sym, TI_GET_ADDRESS, &sym_info->Address); sym_info->Register = 0; break; } sym_info->Scope = 0; /* FIXME */ sym_info->Tag = sym->tag; name = symt_get_name(sym); if (sym_info->MaxNameLen) { if (sym->tag != SymTagPublicSymbol || !(dbghelp_options & SYMOPT_UNDNAME) || (sym_info->NameLen = UnDecorateSymbolName(name, sym_info->Name, sym_info->MaxNameLen, UNDNAME_COMPLETE) == 0)) { sym_info->NameLen = min(strlen(name), sym_info->MaxNameLen - 1); memcpy(sym_info->Name, name, sym_info->NameLen); sym_info->Name[sym_info->NameLen] = '\0'; } } TRACE_(dbghelp_symt)("%p => %s %lu %s\n", sym, sym_info->Name, sym_info->Size, wine_dbgstr_longlong(sym_info->Address)); } static BOOL symt_enum_module(struct module* module, regex_t* regex, PSYM_ENUMERATESYMBOLS_CALLBACK cb, PVOID user) { char buffer[sizeof(SYMBOL_INFO) + 256]; SYMBOL_INFO* sym_info = (SYMBOL_INFO*)buffer; void* ptr; struct symt_ht* sym = NULL; struct hash_table_iter hti; hash_table_iter_init(&module->ht_symbols, &hti, NULL); while ((ptr = hash_table_iter_up(&hti))) { sym = GET_ENTRY(ptr, struct symt_ht, hash_elt); if (sym->hash_elt.name && regexec(regex, sym->hash_elt.name, 0, NULL, 0) == 0) { sym_info->SizeOfStruct = sizeof(SYMBOL_INFO); sym_info->MaxNameLen = sizeof(buffer) - sizeof(SYMBOL_INFO); symt_fill_sym_info(module, &sym->symt, sym_info); if (!cb(sym_info, sym_info->Size, user)) return TRUE; } } return FALSE; } /*********************************************************************** * resort_symbols * * Rebuild sorted list of symbols for a module. */ static BOOL resort_symbols(struct module* module) { int nsym = 0; void* ptr; struct symt_ht* sym; struct hash_table_iter hti; hash_table_iter_init(&module->ht_symbols, &hti, NULL); while ((ptr = hash_table_iter_up(&hti))) nsym++; if (!(module->module.NumSyms = nsym)) return FALSE; if (module->addr_sorttab) module->addr_sorttab = HeapReAlloc(GetProcessHeap(), 0, module->addr_sorttab, nsym * sizeof(struct symt_ht*)); else module->addr_sorttab = HeapAlloc(GetProcessHeap(), 0, nsym * sizeof(struct symt_ht*)); if (!module->addr_sorttab) return FALSE; nsym = 0; hash_table_iter_init(&module->ht_symbols, &hti, NULL); while ((ptr = hash_table_iter_up(&hti))) { sym = GET_ENTRY(ptr, struct symt_ht, hash_elt); assert(sym); module->addr_sorttab[nsym++] = sym; } qsort(module->addr_sorttab, nsym, sizeof(struct symt_ht*), symt_cmp_addr); return module->sortlist_valid = TRUE; } /* assume addr is in module */ int symt_find_nearest(struct module* module, DWORD addr) { int mid, high, low; ULONG64 ref_addr; DWORD ref_size; if (!module->sortlist_valid || !module->addr_sorttab) { if (!resort_symbols(module)) return -1; } /* * Binary search to find closest symbol. */ low = 0; high = module->module.NumSyms; symt_get_info(&module->addr_sorttab[0]->symt, TI_GET_ADDRESS, &ref_addr); if (addr < ref_addr) return -1; if (high) { symt_get_info(&module->addr_sorttab[high - 1]->symt, TI_GET_ADDRESS, &ref_addr); if (!symt_get_info(&module->addr_sorttab[high - 1]->symt, TI_GET_LENGTH, &ref_size) || !ref_size) ref_size = 0x1000; /* arbitrary value */ if (addr >= ref_addr + ref_size) return -1; } while (high > low + 1) { mid = (high + low) / 2; if (cmp_sorttab_addr(module, mid, addr) < 0) low = mid; else high = mid; } if (low != high && high != module->module.NumSyms && cmp_sorttab_addr(module, high, addr) <= 0) low = high; /* If found symbol is a public symbol, check if there are any other entries that * might also have the same address, but would get better information */ if (module->addr_sorttab[low]->symt.tag == SymTagPublicSymbol) { symt_get_info(&module->addr_sorttab[low]->symt, TI_GET_ADDRESS, &ref_addr); if (low > 0 && module->addr_sorttab[low - 1]->symt.tag != SymTagPublicSymbol && !cmp_sorttab_addr(module, low - 1, ref_addr)) low--; else if (low < module->module.NumSyms - 1 && module->addr_sorttab[low + 1]->symt.tag != SymTagPublicSymbol && !cmp_sorttab_addr(module, low + 1, ref_addr)) low++; } /* finally check that we fit into the found symbol */ symt_get_info(&module->addr_sorttab[low]->symt, TI_GET_ADDRESS, &ref_addr); if (addr < ref_addr) return -1; if (!symt_get_info(&module->addr_sorttab[high - 1]->symt, TI_GET_LENGTH, &ref_size) || !ref_size) ref_size = 0x1000; /* arbitrary value */ if (addr >= ref_addr + ref_size) return -1; return low; } static BOOL symt_enum_locals_helper(struct process* pcs, struct module* module, regex_t* preg, PSYM_ENUMERATESYMBOLS_CALLBACK cb, PVOID user, SYMBOL_INFO* sym_info, struct vector* v) { struct symt** plsym = NULL; struct symt* lsym = NULL; DWORD pc = pcs->ctx_frame.InstructionOffset; while ((plsym = vector_iter_up(v, plsym))) { lsym = *plsym; switch (lsym->tag) { case SymTagBlock: { struct symt_block* block = (struct symt_block*)lsym; if (pc < block->address || block->address + block->size <= pc) continue; if (!symt_enum_locals_helper(pcs, module, preg, cb, user, sym_info, &block->vchildren)) return FALSE; } break; case SymTagData: if (regexec(preg, symt_get_name(lsym), 0, NULL, 0) == 0) { symt_fill_sym_info(module, lsym, sym_info); if (!cb(sym_info, sym_info->Size, user)) return FALSE; } break; case SymTagLabel: case SymTagFuncDebugStart: case SymTagFuncDebugEnd: break; default: FIXME("Unknown type: %u (%x)\n", lsym->tag, lsym->tag); assert(0); } } return TRUE; } static BOOL symt_enum_locals(struct process* pcs, const char* mask, PSYM_ENUMERATESYMBOLS_CALLBACK EnumSymbolsCallback, PVOID UserContext) { struct module* module; struct symt_ht* sym; char buffer[sizeof(SYMBOL_INFO) + 256]; SYMBOL_INFO* sym_info = (SYMBOL_INFO*)buffer; DWORD pc = pcs->ctx_frame.InstructionOffset; int idx; sym_info->SizeOfStruct = sizeof(*sym_info); sym_info->MaxNameLen = sizeof(buffer) - sizeof(SYMBOL_INFO); module = module_find_by_addr(pcs, pc, DMT_UNKNOWN); if (!(module = module_get_debug(pcs, module))) return FALSE; if ((idx = symt_find_nearest(module, pc)) == -1) return FALSE; sym = module->addr_sorttab[idx]; if (sym->symt.tag == SymTagFunction) { BOOL ret; regex_t preg; compile_regex(mask ? mask : "*", -1, &preg); ret = symt_enum_locals_helper(pcs, module, &preg, EnumSymbolsCallback, UserContext, sym_info, &((struct symt_function*)sym)->vchildren); regfree(&preg); return ret; } symt_fill_sym_info(module, &sym->symt, sym_info); return EnumSymbolsCallback(sym_info, sym_info->Size, UserContext); } /****************************************************************** * SymEnumSymbols (DBGHELP.@) * * cases BaseOfDll = 0 * !foo fails always (despite what MSDN states) * RE1!RE2 looks up all modules matching RE1, and in all these modules, lookup RE2 * no ! in Mask, lookup in local Context * cases BaseOfDll != 0 * !foo fails always (despite what MSDN states) * RE1!RE2 gets RE2 from BaseOfDll (whatever RE1 is) */ BOOL WINAPI SymEnumSymbols(HANDLE hProcess, ULONG64 BaseOfDll, PCSTR Mask, PSYM_ENUMERATESYMBOLS_CALLBACK EnumSymbolsCallback, PVOID UserContext) { struct process* pcs = process_find_by_handle(hProcess); struct module* module; struct module* dbg_module; const char* bang; regex_t mod_regex, sym_regex; TRACE("(%p %s %s %p %p)\n", hProcess, wine_dbgstr_longlong(BaseOfDll), debugstr_a(Mask), EnumSymbolsCallback, UserContext); if (!pcs) return FALSE; if (BaseOfDll == 0) { /* do local variables ? */ if (!Mask || !(bang = strchr(Mask, '!'))) return symt_enum_locals(pcs, Mask, EnumSymbolsCallback, UserContext); if (bang == Mask) return FALSE; compile_regex(Mask, bang - Mask, &mod_regex); compile_regex(bang + 1, -1, &sym_regex); for (module = pcs->lmodules; module; module = module->next) { if (module->type == DMT_PE && (dbg_module = module_get_debug(pcs, module))) { if (regexec(&mod_regex, module->module.ModuleName, 0, NULL, 0) == 0 && symt_enum_module(dbg_module, &sym_regex, EnumSymbolsCallback, UserContext)) break; } } /* not found in PE modules, retry on the ELF ones */ if (!module && (dbghelp_options & SYMOPT_WINE_WITH_ELF_MODULES)) { for (module = pcs->lmodules; module; module = module->next) { if (module->type == DMT_ELF && !module_get_containee(pcs, module) && (dbg_module = module_get_debug(pcs, module))) { if (regexec(&mod_regex, module->module.ModuleName, 0, NULL, 0) == 0 && symt_enum_module(dbg_module, &sym_regex, EnumSymbolsCallback, UserContext)) break; } } } regfree(&mod_regex); regfree(&sym_regex); return TRUE; } module = module_find_by_addr(pcs, BaseOfDll, DMT_UNKNOWN); if (!(module = module_get_debug(pcs, module))) return FALSE; /* we always ignore module name from Mask when BaseOfDll is defined */ if (Mask && (bang = strchr(Mask, '!'))) { if (bang == Mask) return FALSE; Mask = bang + 1; } compile_regex(Mask ? Mask : "*", -1, &sym_regex); symt_enum_module(module, &sym_regex, EnumSymbolsCallback, UserContext); regfree(&sym_regex); return TRUE; } struct sym_enumerate { void* ctx; PSYM_ENUMSYMBOLS_CALLBACK cb; }; static BOOL CALLBACK sym_enumerate_cb(PSYMBOL_INFO syminfo, ULONG size, void* ctx) { struct sym_enumerate* se = (struct sym_enumerate*)ctx; return (se->cb)(syminfo->Name, syminfo->Address, syminfo->Size, se->ctx); } /*********************************************************************** * SymEnumerateSymbols (DBGHELP.@) */ BOOL WINAPI SymEnumerateSymbols(HANDLE hProcess, DWORD BaseOfDll, PSYM_ENUMSYMBOLS_CALLBACK EnumSymbolsCallback, PVOID UserContext) { struct sym_enumerate se; se.ctx = UserContext; se.cb = EnumSymbolsCallback; return SymEnumSymbols(hProcess, BaseOfDll, NULL, sym_enumerate_cb, &se); } /****************************************************************** * SymFromAddr (DBGHELP.@) * */ BOOL WINAPI SymFromAddr(HANDLE hProcess, DWORD64 Address, DWORD64* Displacement, PSYMBOL_INFO Symbol) { struct process* pcs = process_find_by_handle(hProcess); struct module* module; struct symt_ht* sym; int idx; if (!pcs) return FALSE; module = module_find_by_addr(pcs, Address, DMT_UNKNOWN); if (!(module = module_get_debug(pcs, module))) return FALSE; if ((idx = symt_find_nearest(module, Address)) == -1) return FALSE; sym = module->addr_sorttab[idx]; symt_fill_sym_info(module, &sym->symt, Symbol); *Displacement = Address - Symbol->Address; return TRUE; } /****************************************************************** * SymGetSymFromAddr (DBGHELP.@) * */ BOOL WINAPI SymGetSymFromAddr(HANDLE hProcess, DWORD Address, PDWORD Displacement, PIMAGEHLP_SYMBOL Symbol) { char buffer[sizeof(SYMBOL_INFO) + 256]; SYMBOL_INFO*si = (SYMBOL_INFO*)buffer; size_t len; DWORD64 Displacement64; if (Symbol->SizeOfStruct < sizeof(*Symbol)) return FALSE; si->SizeOfStruct = sizeof(*si); si->MaxNameLen = 256; if (!SymFromAddr(hProcess, Address, &Displacement64, si)) return FALSE; if (Displacement) *Displacement = Displacement64; Symbol->Address = si->Address; Symbol->Size = si->Size; Symbol->Flags = si->Flags; len = min(Symbol->MaxNameLength, si->MaxNameLen); lstrcpynA(Symbol->Name, si->Name, len); return TRUE; } /****************************************************************** * SymFromName (DBGHELP.@) * */ BOOL WINAPI SymFromName(HANDLE hProcess, LPSTR Name, PSYMBOL_INFO Symbol) { struct process* pcs = process_find_by_handle(hProcess); struct module* module; struct hash_table_iter hti; void* ptr; struct symt_ht* sym = NULL; const char* name; TRACE("(%p, %s, %p)\n", hProcess, Name, Symbol); if (!pcs) return FALSE; if (Symbol->SizeOfStruct < sizeof(*Symbol)) return FALSE; name = strchr(Name, '!'); if (name) { char tmp[128]; assert(name - Name < sizeof(tmp)); memcpy(tmp, Name, name - Name); tmp[name - Name] = '\0'; module = module_find_by_name(pcs, tmp, DMT_UNKNOWN); if (!module) return FALSE; Name = (char*)(name + 1); } else module = pcs->lmodules; /* FIXME: Name could be made out of a regular expression */ for (; module; module = (name) ? NULL : module->next) { if (module->module.SymType == SymNone) continue; if (module->module.SymType == SymDeferred) { struct module* xmodule = module_get_debug(pcs, module); if (!xmodule || xmodule != module) continue; } hash_table_iter_init(&module->ht_symbols, &hti, Name); while ((ptr = hash_table_iter_up(&hti))) { sym = GET_ENTRY(ptr, struct symt_ht, hash_elt); if (!strcmp(sym->hash_elt.name, Name)) { symt_fill_sym_info(module, &sym->symt, Symbol); return TRUE; } } } return FALSE; } /*********************************************************************** * SymGetSymFromName (DBGHELP.@) */ BOOL WINAPI SymGetSymFromName(HANDLE hProcess, LPSTR Name, PIMAGEHLP_SYMBOL Symbol) { char buffer[sizeof(SYMBOL_INFO) + 256]; SYMBOL_INFO*si = (SYMBOL_INFO*)buffer; size_t len; if (Symbol->SizeOfStruct < sizeof(*Symbol)) return FALSE; si->SizeOfStruct = sizeof(*si); si->MaxNameLen = 256; if (!SymFromName(hProcess, Name, si)) return FALSE; Symbol->Address = si->Address; Symbol->Size = si->Size; Symbol->Flags = si->Flags; len = min(Symbol->MaxNameLength, si->MaxNameLen); lstrcpynA(Symbol->Name, si->Name, len); return TRUE; } /****************************************************************** * sym_fill_func_line_info * * fills information about a file */ BOOL symt_fill_func_line_info(struct module* module, struct symt_function* func, DWORD addr, IMAGEHLP_LINE* line) { struct line_info* dli = NULL; BOOL found = FALSE; assert(func->symt.tag == SymTagFunction); while ((dli = vector_iter_down(&func->vlines, dli))) { if (!dli->is_source_file) { if (found || dli->u.pc_offset > addr) continue; line->LineNumber = dli->line_number; line->Address = dli->u.pc_offset; line->Key = dli; found = TRUE; continue; } if (found) { line->FileName = (char*)source_get(module, dli->u.source_file); return TRUE; } } return FALSE; } /*********************************************************************** * SymGetSymNext (DBGHELP.@) */ BOOL WINAPI SymGetSymNext(HANDLE hProcess, PIMAGEHLP_SYMBOL Symbol) { /* algo: * get module from Symbol.Address * get index in module.addr_sorttab of Symbol.Address * increment index * if out of module bounds, move to next module in process address space */ FIXME("(%p, %p): stub\n", hProcess, Symbol); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return FALSE; } /*********************************************************************** * SymGetSymPrev (DBGHELP.@) */ BOOL WINAPI SymGetSymPrev(HANDLE hProcess, PIMAGEHLP_SYMBOL Symbol) { FIXME("(%p, %p): stub\n", hProcess, Symbol); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return FALSE; } /****************************************************************** * SymGetLineFromAddr (DBGHELP.@) * */ BOOL WINAPI SymGetLineFromAddr(HANDLE hProcess, DWORD dwAddr, PDWORD pdwDisplacement, PIMAGEHLP_LINE Line) { struct process* pcs = process_find_by_handle(hProcess); struct module* module; int idx; TRACE("%p %08lx %p %p\n", hProcess, dwAddr, pdwDisplacement, Line); if (Line->SizeOfStruct < sizeof(*Line)) return FALSE; if (!pcs) return FALSE; module = module_find_by_addr(pcs, dwAddr, DMT_UNKNOWN); if (!(module = module_get_debug(pcs, module))) return FALSE; if ((idx = symt_find_nearest(module, dwAddr)) == -1) return FALSE; if (module->addr_sorttab[idx]->symt.tag != SymTagFunction) return FALSE; if (!symt_fill_func_line_info(module, (struct symt_function*)module->addr_sorttab[idx], dwAddr, Line)) return FALSE; *pdwDisplacement = dwAddr - Line->Address; return TRUE; } /****************************************************************** * SymGetLinePrev (DBGHELP.@) * */ BOOL WINAPI SymGetLinePrev(HANDLE hProcess, PIMAGEHLP_LINE Line) { struct process* pcs = process_find_by_handle(hProcess); struct module* module; struct line_info* li; BOOL in_search = FALSE; TRACE("(%p %p)\n", hProcess, Line); if (Line->SizeOfStruct < sizeof(*Line)) return FALSE; if (!pcs) return FALSE; module = module_find_by_addr(pcs, Line->Address, DMT_UNKNOWN); if (!(module = module_get_debug(pcs, module))) return FALSE; if (Line->Key == 0) return FALSE; li = (struct line_info*)Line->Key; /* things are a bit complicated because when we encounter a DLIT_SOURCEFILE * element we have to go back until we find the prev one to get the real * source file name for the DLIT_OFFSET element just before * the first DLIT_SOURCEFILE */ while (!li->is_first) { li--; if (!li->is_source_file) { Line->LineNumber = li->line_number; Line->Address = li->u.pc_offset; Line->Key = li; if (!in_search) return TRUE; } else { if (in_search) { Line->FileName = (char*)source_get(module, li->u.source_file); return TRUE; } in_search = TRUE; } } SetLastError(ERROR_NO_MORE_ITEMS); /* FIXME */ return FALSE; } BOOL symt_get_func_line_next(struct module* module, PIMAGEHLP_LINE line) { struct line_info* li; if (line->Key == 0) return FALSE; li = (struct line_info*)line->Key; while (!li->is_last) { li++; if (!li->is_source_file) { line->LineNumber = li->line_number; line->Address = li->u.pc_offset; line->Key = li; return TRUE; } line->FileName = (char*)source_get(module, li->u.source_file); } return FALSE; } /****************************************************************** * SymGetLineNext (DBGHELP.@) * */ BOOL WINAPI SymGetLineNext(HANDLE hProcess, PIMAGEHLP_LINE Line) { struct process* pcs = process_find_by_handle(hProcess); struct module* module; TRACE("(%p %p)\n", hProcess, Line); if (Line->SizeOfStruct < sizeof(*Line)) return FALSE; if (!pcs) return FALSE; module = module_find_by_addr(pcs, Line->Address, DMT_UNKNOWN); if (!(module = module_get_debug(pcs, module))) return FALSE; if (symt_get_func_line_next(module, Line)) return TRUE; SetLastError(ERROR_NO_MORE_ITEMS); /* FIXME */ return FALSE; } /*********************************************************************** * SymFunctionTableAccess (DBGHELP.@) */ PVOID WINAPI SymFunctionTableAccess(HANDLE hProcess, DWORD AddrBase) { FIXME("(%p, 0x%08lx): stub\n", hProcess, AddrBase); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return FALSE; } /*********************************************************************** * SymUnDName (DBGHELP.@) */ BOOL WINAPI SymUnDName(PIMAGEHLP_SYMBOL sym, LPSTR UnDecName, DWORD UnDecNameLength) { TRACE("(%p %s %lu): stub\n", sym, UnDecName, UnDecNameLength); return UnDecorateSymbolName(sym->Name, UnDecName, UnDecNameLength, UNDNAME_COMPLETE) != 0; } static void* und_alloc(size_t len) { return HeapAlloc(GetProcessHeap(), 0, len); } static void und_free (void* ptr) { HeapFree(GetProcessHeap(), 0, ptr); } /*********************************************************************** * UnDecorateSymbolName (DBGHELP.@) */ DWORD WINAPI UnDecorateSymbolName(LPCSTR DecoratedName, LPSTR UnDecoratedName, DWORD UndecoratedLength, DWORD Flags) { /* undocumented from msvcrt */ static char* (*p_undname)(char*, const char*, int, void* (*)(size_t), void (*)(void*), unsigned short); static WCHAR szMsvcrt[] = {'m','s','v','c','r','t','.','d','l','l',0}; TRACE("(%s, %p, %ld, 0x%08lx): stub\n", debugstr_a(DecoratedName), UnDecoratedName, UndecoratedLength, Flags); if (!p_undname) { if (!hMsvcrt) hMsvcrt = LoadLibraryW(szMsvcrt); if (hMsvcrt) p_undname = (void*)GetProcAddress(hMsvcrt, "__unDName"); if (!p_undname) return 0; } if (!UnDecoratedName) return 0; if (!p_undname(UnDecoratedName, DecoratedName, UndecoratedLength, und_alloc, und_free, Flags)) return 0; return strlen(UnDecoratedName); }