mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-11-23 18:14:13 +08:00
Introduce OP_VAR_MSYM_VALUE
The previous patch left GDB with an inconsistency. While with normal expression evaluation the "unknown return type" error shows the name of the function that misses debug info: (gdb) p getenv ("PATH") 'getenv' has unknown return type; cast the call to its declared return type ^^^^^^ which can by handy in more complicated expressions, "ptype" does not: (gdb) ptype getenv ("PATH") function has unknown return type; cast the call to its declared return type ^^^^^^^^ This commit is a step toward fixing it. The problem is that while evaluating the expression above, we have no reference to the minimal symbol where we could extract the name from. This is because the resulting expression tree has no reference to the minsym at all. During parsing, the type and address of the minsym are extracted and an UNOP_MEMVAL / UNOP_MEMVAL_TLS operator is generated (see write_exp_elt_msym). With "set debug expression", here's what you see: 0 OP_FUNCALL Number of args: 0 3 UNOP_MEMVAL Type @0x565334a51930 (<text variable, no debug info>) 6 OP_LONG Type @0x565334a51c60 (__CORE_ADDR), value 140737345035648 (0x7ffff7751d80) The "print" case finds the function name, because call_function_by_hand looks up the function by address again. However, for "ptype", we don't reach that code, because obviously we don't really call the function. Unlike minsym references, references to variables with debug info have a pointer to the variable's symbol in the expression tree, with OP_VAR_VALUE: (gdb) ptype main() ... 0 OP_FUNCALL Number of args: 0 3 OP_VAR_VALUE Block @0x0, symbol @0x559bbbd9b358 (main(int, char**)) ... so I don't see why do minsyms need to be different. So to prepare for fixing the missing function name issue, this commit adds a new OP_VAR_MSYM_VALUE operator that mimics OP_VAR_VALUE, except that it's for minsyms instead of debug symbols. For infcalls, we now get expressions like these: 0 OP_FUNCALL Number of args: 0 3 OP_VAR_MSYM_VALUE Objfile @0x1e41bf0, msymbol @0x7fffe599b000 (getenv) In the following patch, we'll make OP_FUNCALL extract the function name from the symbol stored in OP_VAR_VALUE/OP_VAR_MSYM_VALUE. OP_VAR_MSYM_VALUE will be used more in a later patch in the series too. gdb/ChangeLog: 2017-09-04 Pedro Alves <palves@redhat.com> * ada-lang.c (resolve_subexp): Handle OP_VAR_MSYM_VALUE. * ax-gdb.c (gen_msym_var_ref): New function. (gen_expr): Handle OP_VAR_MSYM_VALUE. * eval.c (evaluate_var_msym_value): New function. * eval.c (evaluate_subexp_standard): Handle OP_VAR_MSYM_VALUE. <OP_FUNCALL>: Extract function name from symbol/minsym and pass it to call_function_by_hand. * expprint.c (print_subexp_standard, dump_subexp_body_standard): Handle OP_VAR_MSYM_VALUE. (union exp_element) <msymbol>: New field. * minsyms.h (struct type): Forward declare. (find_minsym_type_and_address): Declare. * parse.c (write_exp_elt_msym): New function. (write_exp_msymbol): Delete, refactored as ... (find_minsym_type_and_address): ... this new function. (write_exp_msymbol): Reimplement using OP_VAR_MSYM_VALUE. (operator_length_standard, operator_check_standard): Handle OP_VAR_MSYM_VALUE. * std-operator.def (OP_VAR_MSYM_VALUE): New.
This commit is contained in:
parent
7022349d5c
commit
74ea4be48e
@ -1,3 +1,25 @@
|
|||||||
|
2017-09-04 Pedro Alves <palves@redhat.com>
|
||||||
|
|
||||||
|
* ada-lang.c (resolve_subexp): Handle OP_VAR_MSYM_VALUE.
|
||||||
|
* ax-gdb.c (gen_msym_var_ref): New function.
|
||||||
|
(gen_expr): Handle OP_VAR_MSYM_VALUE.
|
||||||
|
* eval.c (evaluate_var_msym_value): New function.
|
||||||
|
* eval.c (evaluate_subexp_standard): Handle OP_VAR_MSYM_VALUE.
|
||||||
|
<OP_FUNCALL>: Extract function name from symbol/minsym and pass it
|
||||||
|
to call_function_by_hand.
|
||||||
|
* expprint.c (print_subexp_standard, dump_subexp_body_standard):
|
||||||
|
Handle OP_VAR_MSYM_VALUE.
|
||||||
|
(union exp_element) <msymbol>: New field.
|
||||||
|
* minsyms.h (struct type): Forward declare.
|
||||||
|
(find_minsym_type_and_address): Declare.
|
||||||
|
* parse.c (write_exp_elt_msym): New function.
|
||||||
|
(write_exp_msymbol): Delete, refactored as ...
|
||||||
|
(find_minsym_type_and_address): ... this new function.
|
||||||
|
(write_exp_msymbol): Reimplement using OP_VAR_MSYM_VALUE.
|
||||||
|
(operator_length_standard, operator_check_standard): Handle
|
||||||
|
OP_VAR_MSYM_VALUE.
|
||||||
|
* std-operator.def (OP_VAR_MSYM_VALUE): New.
|
||||||
|
|
||||||
2017-09-04 Pedro Alves <palves@redhat.com>
|
2017-09-04 Pedro Alves <palves@redhat.com>
|
||||||
|
|
||||||
* ada-lang.c (ada_evaluate_subexp) <TYPE_CODE_FUNC>: Don't handle
|
* ada-lang.c (ada_evaluate_subexp) <TYPE_CODE_FUNC>: Don't handle
|
||||||
|
@ -3385,6 +3385,7 @@ resolve_subexp (struct expression **expp, int *pos, int deprocedure_p,
|
|||||||
case OP_LONG:
|
case OP_LONG:
|
||||||
case OP_DOUBLE:
|
case OP_DOUBLE:
|
||||||
case OP_VAR_VALUE:
|
case OP_VAR_VALUE:
|
||||||
|
case OP_VAR_MSYM_VALUE:
|
||||||
*pos += 4;
|
*pos += 4;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
22
gdb/ax-gdb.c
22
gdb/ax-gdb.c
@ -733,6 +733,23 @@ gen_var_ref (struct agent_expr *ax, struct axs_value *value, struct symbol *var)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Generate code for a minimal symbol variable reference to AX. The
|
||||||
|
variable is the symbol MINSYM, of OBJFILE. Set VALUE to describe
|
||||||
|
the result. */
|
||||||
|
|
||||||
|
static void
|
||||||
|
gen_msym_var_ref (agent_expr *ax, axs_value *value,
|
||||||
|
minimal_symbol *msymbol, objfile *objf)
|
||||||
|
{
|
||||||
|
CORE_ADDR address;
|
||||||
|
type *t = find_minsym_type_and_address (msymbol, objf, &address);
|
||||||
|
value->type = t;
|
||||||
|
value->optimized_out = false;
|
||||||
|
ax_const_l (ax, address);
|
||||||
|
value->kind = axs_lvalue_memory;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -1964,6 +1981,11 @@ gen_expr (struct expression *exp, union exp_element **pc,
|
|||||||
(*pc) += 4;
|
(*pc) += 4;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case OP_VAR_MSYM_VALUE:
|
||||||
|
gen_msym_var_ref (ax, value, (*pc)[2].msymbol, (*pc)[1].objfile);
|
||||||
|
(*pc) += 4;
|
||||||
|
break;
|
||||||
|
|
||||||
case OP_REGISTER:
|
case OP_REGISTER:
|
||||||
{
|
{
|
||||||
const char *name = &(*pc)[2].string;
|
const char *name = &(*pc)[2].string;
|
||||||
|
24
gdb/eval.c
24
gdb/eval.c
@ -676,6 +676,25 @@ make_params (int num_types, struct type **param_types)
|
|||||||
return type;
|
return type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Helper for evaluating an OP_VAR_MSYM_VALUE. */
|
||||||
|
|
||||||
|
static value *
|
||||||
|
evaluate_var_msym_value (enum noside noside,
|
||||||
|
struct objfile *objfile, minimal_symbol *msymbol)
|
||||||
|
{
|
||||||
|
if (noside == EVAL_AVOID_SIDE_EFFECTS)
|
||||||
|
{
|
||||||
|
type *the_type = find_minsym_type_and_address (msymbol, objfile, NULL);
|
||||||
|
return value_zero (the_type, not_lval);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CORE_ADDR address;
|
||||||
|
type *the_type = find_minsym_type_and_address (msymbol, objfile, &address);
|
||||||
|
return value_at_lazy (the_type, address);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
struct value *
|
struct value *
|
||||||
evaluate_subexp_standard (struct type *expect_type,
|
evaluate_subexp_standard (struct type *expect_type,
|
||||||
struct expression *exp, int *pos,
|
struct expression *exp, int *pos,
|
||||||
@ -766,6 +785,11 @@ evaluate_subexp_standard (struct type *expect_type,
|
|||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
case OP_VAR_MSYM_VALUE:
|
||||||
|
(*pos) += 3;
|
||||||
|
return evaluate_var_msym_value (noside,
|
||||||
|
exp->elts[pc + 1].objfile,
|
||||||
|
exp->elts[pc + 2].msymbol);
|
||||||
|
|
||||||
case OP_VAR_ENTRY_VALUE:
|
case OP_VAR_ENTRY_VALUE:
|
||||||
(*pos) += 2;
|
(*pos) += 2;
|
||||||
|
@ -134,6 +134,13 @@ print_subexp_standard (struct expression *exp, int *pos,
|
|||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
case OP_VAR_MSYM_VALUE:
|
||||||
|
{
|
||||||
|
(*pos) += 3;
|
||||||
|
fputs_filtered (MSYMBOL_PRINT_NAME (exp->elts[pc + 2].msymbol), stream);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
|
||||||
case OP_VAR_ENTRY_VALUE:
|
case OP_VAR_ENTRY_VALUE:
|
||||||
{
|
{
|
||||||
(*pos) += 2;
|
(*pos) += 2;
|
||||||
@ -876,6 +883,15 @@ dump_subexp_body_standard (struct expression *exp,
|
|||||||
SYMBOL_PRINT_NAME (exp->elts[elt + 1].symbol));
|
SYMBOL_PRINT_NAME (exp->elts[elt + 1].symbol));
|
||||||
elt += 3;
|
elt += 3;
|
||||||
break;
|
break;
|
||||||
|
case OP_VAR_MSYM_VALUE:
|
||||||
|
fprintf_filtered (stream, "Objfile @");
|
||||||
|
gdb_print_host_address (exp->elts[elt].objfile, stream);
|
||||||
|
fprintf_filtered (stream, ", msymbol @");
|
||||||
|
gdb_print_host_address (exp->elts[elt + 1].msymbol, stream);
|
||||||
|
fprintf_filtered (stream, " (%s)",
|
||||||
|
MSYMBOL_PRINT_NAME (exp->elts[elt + 1].msymbol));
|
||||||
|
elt += 3;
|
||||||
|
break;
|
||||||
case OP_VAR_ENTRY_VALUE:
|
case OP_VAR_ENTRY_VALUE:
|
||||||
fprintf_filtered (stream, "Entry value of symbol @");
|
fprintf_filtered (stream, "Entry value of symbol @");
|
||||||
gdb_print_host_address (exp->elts[elt].symbol, stream);
|
gdb_print_host_address (exp->elts[elt].symbol, stream);
|
||||||
|
@ -64,6 +64,7 @@ union exp_element
|
|||||||
{
|
{
|
||||||
enum exp_opcode opcode;
|
enum exp_opcode opcode;
|
||||||
struct symbol *symbol;
|
struct symbol *symbol;
|
||||||
|
struct minimal_symbol *msymbol;
|
||||||
LONGEST longconst;
|
LONGEST longconst;
|
||||||
DOUBLEST doubleconst;
|
DOUBLEST doubleconst;
|
||||||
gdb_byte decfloatconst[16];
|
gdb_byte decfloatconst[16];
|
||||||
|
@ -20,6 +20,8 @@
|
|||||||
#ifndef MINSYMS_H
|
#ifndef MINSYMS_H
|
||||||
#define MINSYMS_H
|
#define MINSYMS_H
|
||||||
|
|
||||||
|
struct type;
|
||||||
|
|
||||||
/* Several lookup functions return both a minimal symbol and the
|
/* Several lookup functions return both a minimal symbol and the
|
||||||
objfile in which it is found. This structure is used in these
|
objfile in which it is found. This structure is used in these
|
||||||
cases. */
|
cases. */
|
||||||
@ -275,4 +277,11 @@ void iterate_over_minimal_symbols (struct objfile *objf,
|
|||||||
|
|
||||||
CORE_ADDR minimal_symbol_upper_bound (struct bound_minimal_symbol minsym);
|
CORE_ADDR minimal_symbol_upper_bound (struct bound_minimal_symbol minsym);
|
||||||
|
|
||||||
|
/* Return the type of MSYMBOL, a minimal symbol of OBJFILE. If
|
||||||
|
ADDRESS_P is not NULL, set it to the MSYMBOL's resolved
|
||||||
|
address. */
|
||||||
|
|
||||||
|
type *find_minsym_type_and_address (minimal_symbol *msymbol, objfile *objf,
|
||||||
|
CORE_ADDR *address_p);
|
||||||
|
|
||||||
#endif /* MINSYMS_H */
|
#endif /* MINSYMS_H */
|
||||||
|
81
gdb/parse.c
81
gdb/parse.c
@ -255,6 +255,16 @@ write_exp_elt_sym (struct parser_state *ps, struct symbol *expelt)
|
|||||||
write_exp_elt (ps, &tmp);
|
write_exp_elt (ps, &tmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
write_exp_elt_msym (struct parser_state *ps, minimal_symbol *expelt)
|
||||||
|
{
|
||||||
|
union exp_element tmp;
|
||||||
|
|
||||||
|
memset (&tmp, 0, sizeof (union exp_element));
|
||||||
|
tmp.msymbol = expelt;
|
||||||
|
write_exp_elt (ps, &tmp);
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
write_exp_elt_block (struct parser_state *ps, const struct block *b)
|
write_exp_elt_block (struct parser_state *ps, const struct block *b)
|
||||||
{
|
{
|
||||||
@ -470,17 +480,17 @@ write_exp_bitstring (struct parser_state *ps, struct stoken str)
|
|||||||
write_exp_elt_longcst (ps, (LONGEST) bits);
|
write_exp_elt_longcst (ps, (LONGEST) bits);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Add the appropriate elements for a minimal symbol to the end of
|
/* Return the type of MSYMBOL, a minimal symbol of OBJFILE. If
|
||||||
the expression. */
|
ADDRESS_P is not NULL, set it to the MSYMBOL's resolved
|
||||||
|
address. */
|
||||||
|
|
||||||
void
|
type *
|
||||||
write_exp_msymbol (struct parser_state *ps,
|
find_minsym_type_and_address (minimal_symbol *msymbol,
|
||||||
struct bound_minimal_symbol bound_msym)
|
struct objfile *objfile,
|
||||||
|
CORE_ADDR *address_p)
|
||||||
{
|
{
|
||||||
struct minimal_symbol *msymbol = bound_msym.minsym;
|
bound_minimal_symbol bound_msym = {msymbol, objfile};
|
||||||
struct objfile *objfile = bound_msym.objfile;
|
|
||||||
struct gdbarch *gdbarch = get_objfile_arch (objfile);
|
struct gdbarch *gdbarch = get_objfile_arch (objfile);
|
||||||
|
|
||||||
CORE_ADDR addr = BMSYMBOL_VALUE_ADDRESS (bound_msym);
|
CORE_ADDR addr = BMSYMBOL_VALUE_ADDRESS (bound_msym);
|
||||||
struct obj_section *section = MSYMBOL_OBJ_SECTION (objfile, msymbol);
|
struct obj_section *section = MSYMBOL_OBJ_SECTION (objfile, msymbol);
|
||||||
enum minimal_symbol_type type = MSYMBOL_TYPE (msymbol);
|
enum minimal_symbol_type type = MSYMBOL_TYPE (msymbol);
|
||||||
@ -515,51 +525,54 @@ write_exp_msymbol (struct parser_state *ps,
|
|||||||
if (overlay_debugging)
|
if (overlay_debugging)
|
||||||
addr = symbol_overlayed_address (addr, section);
|
addr = symbol_overlayed_address (addr, section);
|
||||||
|
|
||||||
write_exp_elt_opcode (ps, OP_LONG);
|
|
||||||
/* Let's make the type big enough to hold a 64-bit address. */
|
|
||||||
write_exp_elt_type (ps, objfile_type (objfile)->builtin_core_addr);
|
|
||||||
write_exp_elt_longcst (ps, (LONGEST) addr);
|
|
||||||
write_exp_elt_opcode (ps, OP_LONG);
|
|
||||||
|
|
||||||
if (section && section->the_bfd_section->flags & SEC_THREAD_LOCAL)
|
if (section && section->the_bfd_section->flags & SEC_THREAD_LOCAL)
|
||||||
{
|
{
|
||||||
write_exp_elt_opcode (ps, UNOP_MEMVAL_TLS);
|
/* Skip translation if caller does not need the address. */
|
||||||
write_exp_elt_objfile (ps, objfile);
|
if (address_p != NULL)
|
||||||
write_exp_elt_type (ps, objfile_type (objfile)->nodebug_tls_symbol);
|
*address_p = target_translate_tls_address (objfile, addr);
|
||||||
write_exp_elt_opcode (ps, UNOP_MEMVAL_TLS);
|
return objfile_type (objfile)->nodebug_tls_symbol;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
write_exp_elt_opcode (ps, UNOP_MEMVAL);
|
if (address_p != NULL)
|
||||||
|
*address_p = addr;
|
||||||
|
|
||||||
|
struct type *the_type;
|
||||||
|
|
||||||
switch (type)
|
switch (type)
|
||||||
{
|
{
|
||||||
case mst_text:
|
case mst_text:
|
||||||
case mst_file_text:
|
case mst_file_text:
|
||||||
case mst_solib_trampoline:
|
case mst_solib_trampoline:
|
||||||
write_exp_elt_type (ps, objfile_type (objfile)->nodebug_text_symbol);
|
return objfile_type (objfile)->nodebug_text_symbol;
|
||||||
break;
|
|
||||||
|
|
||||||
case mst_text_gnu_ifunc:
|
case mst_text_gnu_ifunc:
|
||||||
write_exp_elt_type (ps, objfile_type (objfile)
|
return objfile_type (objfile)->nodebug_text_gnu_ifunc_symbol;
|
||||||
->nodebug_text_gnu_ifunc_symbol);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case mst_data:
|
case mst_data:
|
||||||
case mst_file_data:
|
case mst_file_data:
|
||||||
case mst_bss:
|
case mst_bss:
|
||||||
case mst_file_bss:
|
case mst_file_bss:
|
||||||
write_exp_elt_type (ps, objfile_type (objfile)->nodebug_data_symbol);
|
return objfile_type (objfile)->nodebug_data_symbol;
|
||||||
break;
|
|
||||||
|
|
||||||
case mst_slot_got_plt:
|
case mst_slot_got_plt:
|
||||||
write_exp_elt_type (ps, objfile_type (objfile)->nodebug_got_plt_symbol);
|
return objfile_type (objfile)->nodebug_got_plt_symbol;
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
write_exp_elt_type (ps, objfile_type (objfile)->nodebug_unknown_symbol);
|
return objfile_type (objfile)->nodebug_unknown_symbol;
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
write_exp_elt_opcode (ps, UNOP_MEMVAL);
|
}
|
||||||
|
|
||||||
|
/* Add the appropriate elements for a minimal symbol to the end of
|
||||||
|
the expression. */
|
||||||
|
|
||||||
|
void
|
||||||
|
write_exp_msymbol (struct parser_state *ps,
|
||||||
|
struct bound_minimal_symbol bound_msym)
|
||||||
|
{
|
||||||
|
write_exp_elt_opcode (ps, OP_VAR_MSYM_VALUE);
|
||||||
|
write_exp_elt_objfile (ps, bound_msym.objfile);
|
||||||
|
write_exp_elt_msym (ps, bound_msym.minsym);
|
||||||
|
write_exp_elt_opcode (ps, OP_VAR_MSYM_VALUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Mark the current index as the starting location of a structure
|
/* Mark the current index as the starting location of a structure
|
||||||
@ -884,6 +897,7 @@ operator_length_standard (const struct expression *expr, int endpos,
|
|||||||
case OP_DOUBLE:
|
case OP_DOUBLE:
|
||||||
case OP_DECFLOAT:
|
case OP_DECFLOAT:
|
||||||
case OP_VAR_VALUE:
|
case OP_VAR_VALUE:
|
||||||
|
case OP_VAR_MSYM_VALUE:
|
||||||
oplen = 4;
|
oplen = 4;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1840,6 +1854,9 @@ operator_check_standard (struct expression *exp, int pos,
|
|||||||
type = SYMBOL_TYPE (symbol);
|
type = SYMBOL_TYPE (symbol);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case OP_VAR_MSYM_VALUE:
|
||||||
|
objfile = elts[pos + 1].objfile;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Invoke callbacks for TYPE and OBJFILE if they were set as non-NULL. */
|
/* Invoke callbacks for TYPE and OBJFILE if they were set as non-NULL. */
|
||||||
|
@ -132,6 +132,12 @@ OP (OP_VAR_VALUE)
|
|||||||
current function. Implemented via DW_OP_entry_value. */
|
current function. Implemented via DW_OP_entry_value. */
|
||||||
OP (OP_VAR_ENTRY_VALUE)
|
OP (OP_VAR_ENTRY_VALUE)
|
||||||
|
|
||||||
|
/* OP_VAR_MSYM_VALUE takes one struct objfile * in the following
|
||||||
|
element, and one struct minimal_symbol * in the following
|
||||||
|
exp_element, followed by another OP_VAR_MSYM_VALUE, making four
|
||||||
|
exp_elements. */
|
||||||
|
OP (OP_VAR_MSYM_VALUE)
|
||||||
|
|
||||||
/* OP_LAST is followed by an integer in the next exp_element.
|
/* OP_LAST is followed by an integer in the next exp_element.
|
||||||
The integer is zero for the last value printed,
|
The integer is zero for the last value printed,
|
||||||
or it is the absolute number of a history element.
|
or it is the absolute number of a history element.
|
||||||
|
Loading…
Reference in New Issue
Block a user