dbghelp: Fix location computation when attribute has a constant form instead of a block form.
This commit is contained in:
parent
4432a1fa91
commit
baf8dfd8c8
|
@ -571,24 +571,39 @@ static void dwarf2_load_one_entry(dwarf2_parse_context_t*, dwarf2_debug_info_t*,
|
||||||
#define Wine_DW_frame_register 0x7FFFFFFE
|
#define Wine_DW_frame_register 0x7FFFFFFE
|
||||||
#define Wine_DW_register_deref 0x80000000
|
#define Wine_DW_register_deref 0x80000000
|
||||||
|
|
||||||
static unsigned long dwarf2_compute_location(dwarf2_parse_context_t* ctx,
|
static BOOL dwarf2_compute_location(dwarf2_parse_context_t* ctx,
|
||||||
struct dwarf2_block* block,
|
dwarf2_debug_info_t* di,
|
||||||
int* in_register)
|
unsigned long dw,
|
||||||
|
unsigned long* offset, int* in_register)
|
||||||
{
|
{
|
||||||
unsigned long loc[64];
|
unsigned long loc[64];
|
||||||
unsigned stk;
|
unsigned stk;
|
||||||
|
struct attribute xloc;
|
||||||
|
|
||||||
|
if (!dwarf2_find_attribute(ctx, di, dw, &xloc)) return FALSE;
|
||||||
|
|
||||||
if (in_register) *in_register = Wine_DW_no_register;
|
if (in_register) *in_register = Wine_DW_no_register;
|
||||||
|
|
||||||
|
switch (xloc.form)
|
||||||
|
{
|
||||||
|
case DW_FORM_data1: case DW_FORM_data2:
|
||||||
|
case DW_FORM_data4: case DW_FORM_data8:
|
||||||
|
case DW_FORM_udata: case DW_FORM_sdata:
|
||||||
|
/* we've got a constant */
|
||||||
|
*offset = xloc.u.uvalue;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
/* assume we have a block form */
|
||||||
loc[stk = 0] = 0;
|
loc[stk = 0] = 0;
|
||||||
|
|
||||||
if (block->size)
|
if (xloc.u.block.size)
|
||||||
{
|
{
|
||||||
dwarf2_traverse_context_t lctx;
|
dwarf2_traverse_context_t lctx;
|
||||||
unsigned char op;
|
unsigned char op;
|
||||||
BOOL piece_found = FALSE;
|
BOOL piece_found = FALSE;
|
||||||
|
|
||||||
lctx.data = block->ptr;
|
lctx.data = xloc.u.block.ptr;
|
||||||
lctx.end_data = block->ptr + block->size;
|
lctx.end_data = xloc.u.block.ptr + xloc.u.block.size;
|
||||||
lctx.word_size = ctx->word_size;
|
lctx.word_size = ctx->word_size;
|
||||||
|
|
||||||
while (lctx.data < lctx.end_data)
|
while (lctx.data < lctx.end_data)
|
||||||
|
@ -667,7 +682,8 @@ static unsigned long dwarf2_compute_location(dwarf2_parse_context_t* ctx,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return loc[stk];
|
*offset = loc[stk];
|
||||||
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct symt* dwarf2_lookup_type(dwarf2_parse_context_t* ctx,
|
static struct symt* dwarf2_lookup_type(dwarf2_parse_context_t* ctx,
|
||||||
|
@ -919,7 +935,6 @@ static void dwarf2_parse_udt_member(dwarf2_parse_context_t* ctx,
|
||||||
{
|
{
|
||||||
struct symt* elt_type;
|
struct symt* elt_type;
|
||||||
struct attribute name;
|
struct attribute name;
|
||||||
struct attribute loc;
|
|
||||||
unsigned long offset = 0;
|
unsigned long offset = 0;
|
||||||
struct attribute bit_size;
|
struct attribute bit_size;
|
||||||
struct attribute bit_offset;
|
struct attribute bit_offset;
|
||||||
|
@ -930,12 +945,8 @@ static void dwarf2_parse_udt_member(dwarf2_parse_context_t* ctx,
|
||||||
|
|
||||||
dwarf2_find_name(ctx, di, &name, "udt_member");
|
dwarf2_find_name(ctx, di, &name, "udt_member");
|
||||||
elt_type = dwarf2_lookup_type(ctx, di);
|
elt_type = dwarf2_lookup_type(ctx, di);
|
||||||
if (dwarf2_find_attribute(ctx, di, DW_AT_data_member_location, &loc))
|
if (dwarf2_compute_location(ctx, di, DW_AT_data_member_location, &offset, NULL))
|
||||||
{
|
TRACE("found member_location at %s -> %lu\n", dwarf2_debug_ctx(ctx), offset);
|
||||||
TRACE("found member_location at %s\n", dwarf2_debug_ctx(ctx));
|
|
||||||
offset = dwarf2_compute_location(ctx, &loc.u.block, NULL);
|
|
||||||
TRACE("found offset:%lu\n", offset);
|
|
||||||
}
|
|
||||||
if (!dwarf2_find_attribute(ctx, di, DW_AT_bit_size, &bit_size)) bit_size.u.uvalue = 0;
|
if (!dwarf2_find_attribute(ctx, di, DW_AT_bit_size, &bit_size)) bit_size.u.uvalue = 0;
|
||||||
if (dwarf2_find_attribute(ctx, di, DW_AT_bit_offset, &bit_offset))
|
if (dwarf2_find_attribute(ctx, di, DW_AT_bit_offset, &bit_offset))
|
||||||
{
|
{
|
||||||
|
@ -1117,7 +1128,7 @@ typedef struct dwarf2_subprogram_s
|
||||||
dwarf2_parse_context_t* ctx;
|
dwarf2_parse_context_t* ctx;
|
||||||
struct symt_compiland* compiland;
|
struct symt_compiland* compiland;
|
||||||
struct symt_function* func;
|
struct symt_function* func;
|
||||||
long frame_offset;
|
unsigned long frame_offset;
|
||||||
int frame_reg;
|
int frame_reg;
|
||||||
} dwarf2_subprogram_t;
|
} dwarf2_subprogram_t;
|
||||||
|
|
||||||
|
@ -1131,20 +1142,19 @@ static void dwarf2_parse_variable(dwarf2_subprogram_t* subpgm,
|
||||||
dwarf2_debug_info_t* di)
|
dwarf2_debug_info_t* di)
|
||||||
{
|
{
|
||||||
struct symt* param_type;
|
struct symt* param_type;
|
||||||
struct attribute name, loc, value;
|
struct attribute name, value;
|
||||||
|
unsigned long offset;
|
||||||
|
int in_reg;
|
||||||
BOOL is_pmt = di->abbrev->tag == DW_TAG_formal_parameter;
|
BOOL is_pmt = di->abbrev->tag == DW_TAG_formal_parameter;
|
||||||
|
|
||||||
TRACE("%s, for %s\n", dwarf2_debug_ctx(subpgm->ctx), dwarf2_debug_di(di));
|
TRACE("%s, for %s\n", dwarf2_debug_ctx(subpgm->ctx), dwarf2_debug_di(di));
|
||||||
|
|
||||||
param_type = dwarf2_lookup_type(subpgm->ctx, di);
|
param_type = dwarf2_lookup_type(subpgm->ctx, di);
|
||||||
dwarf2_find_name(subpgm->ctx, di, &name, "parameter");
|
dwarf2_find_name(subpgm->ctx, di, &name, "parameter");
|
||||||
if (dwarf2_find_attribute(subpgm->ctx, di, DW_AT_location, &loc))
|
if (dwarf2_compute_location(subpgm->ctx, di, DW_AT_location, &offset, &in_reg))
|
||||||
{
|
{
|
||||||
struct attribute ext;
|
struct attribute ext;
|
||||||
long offset;
|
|
||||||
int in_reg;
|
|
||||||
|
|
||||||
offset = dwarf2_compute_location(subpgm->ctx, &loc.u.block, &in_reg);
|
|
||||||
TRACE("found parameter %s/%ld (reg=%d) at %s\n",
|
TRACE("found parameter %s/%ld (reg=%d) at %s\n",
|
||||||
name.u.string, offset, in_reg, dwarf2_debug_ctx(subpgm->ctx));
|
name.u.string, offset, in_reg, dwarf2_debug_ctx(subpgm->ctx));
|
||||||
switch (in_reg & ~Wine_DW_register_deref)
|
switch (in_reg & ~Wine_DW_register_deref)
|
||||||
|
@ -1331,7 +1341,6 @@ static struct symt* dwarf2_parse_subprogram(dwarf2_parse_context_t* ctx,
|
||||||
struct attribute high_pc;
|
struct attribute high_pc;
|
||||||
struct attribute is_decl;
|
struct attribute is_decl;
|
||||||
struct attribute inline_flags;
|
struct attribute inline_flags;
|
||||||
struct attribute frame;
|
|
||||||
struct symt* ret_type;
|
struct symt* ret_type;
|
||||||
struct symt_function_signature* sig_type;
|
struct symt_function_signature* sig_type;
|
||||||
dwarf2_subprogram_t subpgm;
|
dwarf2_subprogram_t subpgm;
|
||||||
|
@ -1368,11 +1377,8 @@ static struct symt* dwarf2_parse_subprogram(dwarf2_parse_context_t* ctx,
|
||||||
|
|
||||||
subpgm.ctx = ctx;
|
subpgm.ctx = ctx;
|
||||||
subpgm.compiland = compiland;
|
subpgm.compiland = compiland;
|
||||||
if (dwarf2_find_attribute(ctx, di, DW_AT_frame_base, &frame))
|
if (dwarf2_compute_location(ctx, di, DW_AT_frame_base, &subpgm.frame_offset, &subpgm.frame_reg))
|
||||||
{
|
|
||||||
subpgm.frame_offset = dwarf2_compute_location(ctx, &frame.u.block, &subpgm.frame_reg);
|
|
||||||
TRACE("For %s got %ld/%d\n", name.u.string, subpgm.frame_offset, subpgm.frame_reg);
|
TRACE("For %s got %ld/%d\n", name.u.string, subpgm.frame_offset, subpgm.frame_reg);
|
||||||
}
|
|
||||||
else /* on stack !! */
|
else /* on stack !! */
|
||||||
{
|
{
|
||||||
subpgm.frame_reg = 0;
|
subpgm.frame_reg = 0;
|
||||||
|
|
Loading…
Reference in New Issue