527 lines
20 KiB
C
527 lines
20 KiB
C
/*
|
|
* File dbghelp_private.h - dbghelp internal definitions
|
|
*
|
|
* Copyright (C) 1995, Alexandre Julliard
|
|
* Copyright (C) 1996, Eric Youngdale.
|
|
* Copyright (C) 1999-2000, Ulrich Weigand.
|
|
* Copyright (C) 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
|
|
*/
|
|
|
|
#include <stdarg.h>
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winver.h"
|
|
#include "dbghelp.h"
|
|
#include "objbase.h"
|
|
#include "oaidl.h"
|
|
|
|
#include "cvconst.h"
|
|
|
|
/* #define USE_STATS */
|
|
|
|
struct pool /* poor's man */
|
|
{
|
|
struct pool_arena* first;
|
|
unsigned arena_size;
|
|
};
|
|
|
|
void pool_init(struct pool* a, unsigned arena_size);
|
|
void pool_destroy(struct pool* a);
|
|
void* pool_alloc(struct pool* a, unsigned len);
|
|
/* void* pool_realloc(struct pool* a, void* p,
|
|
unsigned old_size, unsigned new_size); */
|
|
char* pool_strdup(struct pool* a, const char* str);
|
|
|
|
struct vector
|
|
{
|
|
void** buckets;
|
|
unsigned elt_size;
|
|
unsigned shift;
|
|
unsigned num_elts;
|
|
unsigned num_buckets;
|
|
};
|
|
|
|
void vector_init(struct vector* v, unsigned elt_sz, unsigned bucket_sz);
|
|
unsigned vector_length(const struct vector* v);
|
|
void* vector_at(const struct vector* v, unsigned pos);
|
|
void* vector_add(struct vector* v, struct pool* pool);
|
|
/*void vector_pool_normalize(struct vector* v, struct pool* pool); */
|
|
void* vector_iter_up(const struct vector* v, void* elt);
|
|
void* vector_iter_down(const struct vector* v, void* elt);
|
|
|
|
struct hash_table_elt
|
|
{
|
|
const char* name;
|
|
struct hash_table_elt* next;
|
|
};
|
|
|
|
struct hash_table
|
|
{
|
|
unsigned num_buckets;
|
|
struct hash_table_elt** buckets;
|
|
};
|
|
|
|
void hash_table_init(struct pool* pool, struct hash_table* ht,
|
|
unsigned num_buckets);
|
|
void hash_table_destroy(struct hash_table* ht);
|
|
void hash_table_add(struct hash_table* ht, struct hash_table_elt* elt);
|
|
void* hash_table_find(const struct hash_table* ht, const char* name);
|
|
unsigned hash_table_hash(const char* name, unsigned num_buckets);
|
|
|
|
struct hash_table_iter
|
|
{
|
|
const struct hash_table* ht;
|
|
struct hash_table_elt* element;
|
|
int index;
|
|
int last;
|
|
};
|
|
|
|
void hash_table_iter_init(const struct hash_table* ht,
|
|
struct hash_table_iter* hti, const char* name);
|
|
void* hash_table_iter_up(struct hash_table_iter* hti);
|
|
|
|
#define GET_ENTRY(__i, __t, __f) \
|
|
((__t*)((char*)(__i) - (unsigned int)(&((__t*)0)->__f)))
|
|
|
|
|
|
extern unsigned dbghelp_options;
|
|
/* some more Wine extensions */
|
|
#define SYMOPT_WINE_WITH_ELF_MODULES 0x40000000
|
|
|
|
struct symt
|
|
{
|
|
enum SymTagEnum tag;
|
|
};
|
|
|
|
struct symt_ht
|
|
{
|
|
struct symt symt;
|
|
struct hash_table_elt hash_elt; /* if global symbol or type */
|
|
};
|
|
|
|
/* lexical tree */
|
|
struct symt_block
|
|
{
|
|
struct symt symt;
|
|
unsigned long address;
|
|
unsigned long size;
|
|
struct symt* container; /* block, or func */
|
|
struct vector vchildren; /* sub-blocks & local variables */
|
|
};
|
|
|
|
struct symt_compiland
|
|
{
|
|
struct symt symt;
|
|
unsigned source;
|
|
struct vector vchildren; /* global variables & functions */
|
|
};
|
|
|
|
struct symt_data
|
|
{
|
|
struct symt symt;
|
|
struct hash_table_elt hash_elt; /* if global symbol */
|
|
enum DataKind kind;
|
|
struct symt* container;
|
|
struct symt* type;
|
|
union /* depends on kind */
|
|
{
|
|
unsigned long address; /* DataIs{Global, FileStatic} */
|
|
struct
|
|
{
|
|
long offset; /* DataIs{Member,Local,Param} in bits */
|
|
unsigned long length; /* DataIs{Member} in bits */
|
|
unsigned long reg_id; /* DataIs{Local} (0 if frame relative) */
|
|
} s;
|
|
VARIANT value; /* DataIsConstant */
|
|
} u;
|
|
};
|
|
|
|
struct symt_function
|
|
{
|
|
struct symt symt;
|
|
struct hash_table_elt hash_elt; /* if global symbol */
|
|
unsigned long address;
|
|
struct symt* container; /* compiland */
|
|
struct symt* type; /* points to function_signature */
|
|
unsigned long size;
|
|
struct vector vlines;
|
|
struct vector vchildren; /* locals, params, blocks, start/end, labels */
|
|
};
|
|
|
|
struct symt_function_point
|
|
{
|
|
struct symt symt; /* either SymTagFunctionDebugStart, SymTagFunctionDebugEnd, SymTagLabel */
|
|
struct symt_function* parent;
|
|
unsigned long offset;
|
|
const char* name; /* for labels */
|
|
};
|
|
|
|
struct symt_public
|
|
{
|
|
struct symt symt;
|
|
struct hash_table_elt hash_elt;
|
|
struct symt* container; /* compiland */
|
|
unsigned long address;
|
|
unsigned long size;
|
|
unsigned in_code : 1,
|
|
is_function : 1;
|
|
};
|
|
|
|
struct symt_thunk
|
|
{
|
|
struct symt symt;
|
|
struct hash_table_elt hash_elt;
|
|
struct symt* container; /* compiland */
|
|
unsigned long address;
|
|
unsigned long size;
|
|
THUNK_ORDINAL ordinal; /* FIXME: doesn't seem to be accessible */
|
|
};
|
|
|
|
/* class tree */
|
|
struct symt_array
|
|
{
|
|
struct symt symt;
|
|
int start;
|
|
int end;
|
|
struct symt* base_type;
|
|
struct symt* index_type;
|
|
};
|
|
|
|
struct symt_basic
|
|
{
|
|
struct symt symt;
|
|
struct hash_table_elt hash_elt;
|
|
enum BasicType bt;
|
|
unsigned long size;
|
|
};
|
|
|
|
struct symt_enum
|
|
{
|
|
struct symt symt;
|
|
const char* name;
|
|
struct vector vchildren;
|
|
};
|
|
|
|
struct symt_function_signature
|
|
{
|
|
struct symt symt;
|
|
struct symt* rettype;
|
|
struct vector vchildren;
|
|
enum CV_call_e call_conv;
|
|
};
|
|
|
|
struct symt_function_arg_type
|
|
{
|
|
struct symt symt;
|
|
struct symt* arg_type;
|
|
struct symt* container;
|
|
};
|
|
|
|
struct symt_pointer
|
|
{
|
|
struct symt symt;
|
|
struct symt* pointsto;
|
|
};
|
|
|
|
struct symt_typedef
|
|
{
|
|
struct symt symt;
|
|
struct hash_table_elt hash_elt;
|
|
struct symt* type;
|
|
};
|
|
|
|
struct symt_udt
|
|
{
|
|
struct symt symt;
|
|
struct hash_table_elt hash_elt;
|
|
enum UdtKind kind;
|
|
int size;
|
|
struct vector vchildren;
|
|
};
|
|
|
|
enum module_type
|
|
{
|
|
DMT_UNKNOWN, /* for lookup, not actually used for a module */
|
|
DMT_ELF, /* a real ELF shared module */
|
|
DMT_PE, /* a native or builtin PE module */
|
|
DMT_PDB, /* PDB file */
|
|
};
|
|
|
|
struct module
|
|
{
|
|
IMAGEHLP_MODULE module;
|
|
struct module* next;
|
|
enum module_type type : 16;
|
|
unsigned short is_virtual : 1;
|
|
struct elf_module_info* elf_info;
|
|
|
|
/* memory allocation pool */
|
|
struct pool pool;
|
|
|
|
/* symbol tables */
|
|
int sortlist_valid;
|
|
struct symt_ht** addr_sorttab;
|
|
struct hash_table ht_symbols;
|
|
|
|
/* types */
|
|
struct hash_table ht_types;
|
|
struct vector vtypes;
|
|
|
|
/* source files */
|
|
unsigned sources_used;
|
|
unsigned sources_alloc;
|
|
char* sources;
|
|
};
|
|
|
|
struct process
|
|
{
|
|
struct process* next;
|
|
HANDLE handle;
|
|
WCHAR* search_path;
|
|
|
|
PSYMBOL_REGISTERED_CALLBACK64 reg_cb;
|
|
BOOL reg_is_unicode;
|
|
DWORD64 reg_user;
|
|
|
|
struct module* lmodules;
|
|
unsigned long dbg_hdr_addr;
|
|
|
|
IMAGEHLP_STACK_FRAME ctx_frame;
|
|
|
|
unsigned buffer_size;
|
|
void* buffer;
|
|
};
|
|
|
|
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;
|
|
};
|
|
|
|
struct module_pair
|
|
{
|
|
struct module* requested; /* in: to module_get_debug() */
|
|
struct module* effective; /* out: module with debug info */
|
|
};
|
|
|
|
enum pdb_kind {PDB_JG, PDB_DS};
|
|
|
|
struct pdb_lookup
|
|
{
|
|
const char* filename;
|
|
DWORD age;
|
|
enum pdb_kind kind;
|
|
union
|
|
{
|
|
struct
|
|
{
|
|
DWORD timestamp;
|
|
struct PDB_JG_TOC* toc;
|
|
} jg;
|
|
struct
|
|
{
|
|
GUID guid;
|
|
struct PDB_DS_TOC* toc;
|
|
} ds;
|
|
} u;
|
|
};
|
|
|
|
/* dbghelp.c */
|
|
extern struct process* process_find_by_handle(HANDLE hProcess);
|
|
extern HANDLE hMsvcrt;
|
|
extern BOOL validate_addr64(DWORD64 addr);
|
|
extern BOOL pcs_callback(const struct process* pcs, ULONG action, void* data);
|
|
extern void* fetch_buffer(struct process* pcs, unsigned size);
|
|
|
|
/* elf_module.c */
|
|
typedef BOOL (*elf_enum_modules_cb)(const char*, unsigned long addr, void* user);
|
|
extern BOOL elf_enum_modules(HANDLE hProc, elf_enum_modules_cb, void*);
|
|
extern BOOL elf_fetch_file_info(const char* name, DWORD* base, DWORD* size, DWORD* checksum);
|
|
struct elf_file_map;
|
|
extern BOOL elf_load_debug_info(struct module* module, struct elf_file_map* fmap);
|
|
extern struct module*
|
|
elf_load_module(struct process* pcs, const char* name, unsigned long);
|
|
extern BOOL elf_read_wine_loader_dbg_info(struct process* pcs);
|
|
extern BOOL elf_synchronize_module_list(struct process* pcs);
|
|
|
|
extern DWORD WINAPI addr_to_linear(HANDLE hProcess, HANDLE hThread, ADDRESS* addr);
|
|
|
|
/* module.c */
|
|
extern int module_compute_num_syms(struct module* module);
|
|
extern struct module*
|
|
module_find_by_addr(const struct process* pcs, unsigned long addr,
|
|
enum module_type type);
|
|
extern struct module*
|
|
module_find_by_name(const struct process* pcs,
|
|
const char* name, enum module_type type);
|
|
extern BOOL module_get_debug(const struct process* pcs, struct module_pair*);
|
|
extern struct module*
|
|
module_new(struct process* pcs, const char* name,
|
|
enum module_type type, BOOL virtual,
|
|
unsigned long addr, unsigned long size,
|
|
unsigned long stamp, unsigned long checksum);
|
|
extern struct module*
|
|
module_get_container(const struct process* pcs,
|
|
const struct module* inner);
|
|
extern struct module*
|
|
module_get_containee(const struct process* pcs,
|
|
const struct module* inner);
|
|
extern enum module_type
|
|
module_get_type_by_name(const char* name);
|
|
extern void module_reset_debug_info(struct module* module);
|
|
extern BOOL module_remove(struct process* pcs,
|
|
struct module* module);
|
|
/* msc.c */
|
|
extern BOOL pe_load_debug_directory(const struct process* pcs,
|
|
struct module* module,
|
|
const BYTE* mapping,
|
|
const IMAGE_SECTION_HEADER* sectp, DWORD nsect,
|
|
const IMAGE_DEBUG_DIRECTORY* dbg, int nDbg);
|
|
extern BOOL pdb_fetch_file_info(struct pdb_lookup* pdb_lookup);
|
|
|
|
/* pe_module.c */
|
|
extern BOOL pe_load_nt_header(HANDLE hProc, DWORD base, IMAGE_NT_HEADERS* nth);
|
|
extern struct module*
|
|
pe_load_module(struct process* pcs, const char* name,
|
|
HANDLE hFile, DWORD base, DWORD size);
|
|
extern struct module*
|
|
pe_load_module_from_pcs(struct process* pcs, const char* name,
|
|
const char* mod_name, DWORD base, DWORD size);
|
|
extern BOOL pe_load_debug_info(const struct process* pcs,
|
|
struct module* module);
|
|
/* source.c */
|
|
extern unsigned source_new(struct module* module, const char* source);
|
|
extern const char* source_get(const struct module* module, unsigned idx);
|
|
|
|
/* stabs.c */
|
|
extern BOOL stabs_parse(struct module* module, unsigned long load_offset,
|
|
const void* stabs, int stablen,
|
|
const char* strs, int strtablen);
|
|
|
|
/* dwarf.c */
|
|
extern BOOL dwarf2_parse(struct module* module, unsigned long load_offset,
|
|
const unsigned char* debug, unsigned int debug_size,
|
|
const unsigned char* abbrev, unsigned int abbrev_size,
|
|
const unsigned char* str, unsigned int str_sz);
|
|
|
|
/* symbol.c */
|
|
extern const char* symt_get_name(const struct symt* sym);
|
|
extern int symt_cmp_addr(const void* p1, const void* p2);
|
|
extern int symt_find_nearest(struct module* module, DWORD addr);
|
|
extern struct symt_compiland*
|
|
symt_new_compiland(struct module* module,
|
|
const char* filename);
|
|
extern struct symt_public*
|
|
symt_new_public(struct module* module,
|
|
struct symt_compiland* parent,
|
|
const char* typename,
|
|
unsigned long address, unsigned size,
|
|
BOOL in_code, BOOL is_func);
|
|
extern struct symt_data*
|
|
symt_new_global_variable(struct module* module,
|
|
struct symt_compiland* parent,
|
|
const char* name, unsigned is_static,
|
|
unsigned long addr, unsigned long size,
|
|
struct symt* type);
|
|
extern struct symt_function*
|
|
symt_new_function(struct module* module,
|
|
struct symt_compiland* parent,
|
|
const char* name,
|
|
unsigned long addr, unsigned long size,
|
|
struct symt* type);
|
|
extern BOOL symt_normalize_function(struct module* module,
|
|
struct symt_function* func);
|
|
extern void symt_add_func_line(struct module* module,
|
|
struct symt_function* func,
|
|
unsigned source_idx, int line_num,
|
|
unsigned long offset);
|
|
extern 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);
|
|
extern struct symt_block*
|
|
symt_open_func_block(struct module* module,
|
|
struct symt_function* func,
|
|
struct symt_block* block,
|
|
unsigned pc, unsigned len);
|
|
extern struct symt_block*
|
|
symt_close_func_block(struct module* module,
|
|
struct symt_function* func,
|
|
struct symt_block* block, unsigned pc);
|
|
extern struct symt_function_point*
|
|
symt_add_function_point(struct module* module,
|
|
struct symt_function* func,
|
|
enum SymTagEnum point,
|
|
unsigned offset, const char* name);
|
|
extern BOOL symt_fill_func_line_info(struct module* module,
|
|
struct symt_function* func,
|
|
DWORD addr, IMAGEHLP_LINE* line);
|
|
extern BOOL symt_get_func_line_next(struct module* module, PIMAGEHLP_LINE line);
|
|
extern struct symt_thunk*
|
|
symt_new_thunk(struct module* module,
|
|
struct symt_compiland* parent,
|
|
const char* name, THUNK_ORDINAL ord,
|
|
unsigned long addr, unsigned long size);
|
|
|
|
/* type.c */
|
|
extern void symt_init_basic(struct module* module);
|
|
extern BOOL symt_get_info(const struct symt* type,
|
|
IMAGEHLP_SYMBOL_TYPE_INFO req, void* pInfo);
|
|
extern struct symt_basic*
|
|
symt_new_basic(struct module* module, enum BasicType,
|
|
const char* typename, unsigned size);
|
|
extern struct symt_udt*
|
|
symt_new_udt(struct module* module, const char* typename,
|
|
unsigned size, enum UdtKind kind);
|
|
extern BOOL symt_set_udt_size(struct module* module,
|
|
struct symt_udt* type, unsigned size);
|
|
extern BOOL symt_add_udt_element(struct module* module,
|
|
struct symt_udt* udt_type,
|
|
const char* name,
|
|
struct symt* elt_type, unsigned offset,
|
|
unsigned size);
|
|
extern struct symt_enum*
|
|
symt_new_enum(struct module* module, const char* typename);
|
|
extern BOOL symt_add_enum_element(struct module* module,
|
|
struct symt_enum* enum_type,
|
|
const char* name, int value);
|
|
extern struct symt_array*
|
|
symt_new_array(struct module* module, int min, int max,
|
|
struct symt* base, struct symt* index);
|
|
extern struct symt_function_signature*
|
|
symt_new_function_signature(struct module* module,
|
|
struct symt* ret_type,
|
|
enum CV_call_e call_conv);
|
|
extern BOOL symt_add_function_signature_parameter(struct module* module,
|
|
struct symt_function_signature* sig,
|
|
struct symt* param);
|
|
extern struct symt_pointer*
|
|
symt_new_pointer(struct module* module,
|
|
struct symt* ref_type);
|
|
extern struct symt_typedef*
|
|
symt_new_typedef(struct module* module, struct symt* ref,
|
|
const char* name);
|