4262abfb98
gdb/ 2013-09-24 Jan Kratochvil <jan.kratochvil@redhat.com> Code cleanup: Add objfile_name accessor function. * ada-lang.c (is_known_support_routine): Use objfile_name. * auto-load.c (source_gdb_script_for_objfile) (auto_load_objfile_script): Likewise. * coffread.c (coff_symtab_read, read_one_sym): Likewise. * dbxread.c (dbx_symfile_read): Likewise. * dwarf2-frame.c (dwarf2_build_frame_info): Likewise. * dwarf2loc.c (locexpr_describe_location_piece): Likewise. * dwarf2read.c (dwarf2_get_dwz_file, dwarf2_read_index) (dw2_symtab_iter_next, dw2_expand_symtabs_matching) (lookup_dwp_signatured_type, lookup_dwo_unit) (dwarf2_build_psymtabs_hard, scan_partial_symbols, process_queue) (fixup_go_packaging, process_imported_unit_die, dwarf2_physname) (read_import_statement, create_dwo_cu, open_and_init_dwp_file) (lookup_dwo_cutu, read_call_site_scope, dwarf2_ranges_read) (dwarf2_record_block_ranges, read_common_block, read_typedef) (read_subrange_type, load_partial_dies, read_partial_die) (read_addr_index_1, read_str_index, dwarf_decode_lines_1) (die_containing_type, build_error_marker_type, lookup_die_type) (follow_die_ref_or_sig, follow_die_ref, dwarf2_fetch_die_loc_sect_off) (dwarf2_fetch_constant_bytes, follow_die_sig, get_signatured_type) (get_DW_AT_signature_type, write_psymtabs_to_index) (save_gdb_index_command): Likewise. * elfread.c (find_separate_debug_file_by_buildid, elf_symfile_read): Likewise. * expprint.c (dump_subexp_body_standard): Likewise. * gdbtypes.c (type_name_no_tag_or_error): Likewise. * jit.c (jit_object_close_impl): Use the objfile field name renamed to original_name. * linux-thread-db.c (try_thread_db_load_from_pdir_1): New variable obj_name, use objfile_name for it, use the variable. (try_thread_db_load_from_pdir, has_libpthread, thread_db_new_objfile): Use objfile_name. * machoread.c (macho_symtab_read, macho_check_dsym) (macho_symfile_relocate): Likewise. * maint.c (maintenance_translate_address): Likewise. * minidebug.c (find_separate_debug_file_in_section): Likewise. * minsyms.c (install_minimal_symbols): Likewise. * objfiles.c (allocate_objfile): Use the objfile field name renamed to original_name. (filter_overlapping_sections): Use objfile_name. (objfile_name): New function. * objfiles.h (struct objfile): Rename field name to original_name. (objfile_name): New prototype. * printcmd.c (sym_info, address_info): Use objfile_name. * probe.c (parse_probes, collect_probes, compare_probes) (info_probes_for_ops): Likewise. * progspace.c (clone_program_space): Likewise. * psymtab.c (require_partial_symbols, dump_psymtab, allocate_psymtab) (maintenance_info_psymtabs): Likewise. * python/py-auto-load.c (gdbpy_load_auto_script_for_objfile) (source_section_scripts): Likewise. * python/py-objfile.c (objfpy_get_filename): Likewise. * python/py-progspace.c (pspy_get_filename): Likewise. * solib-aix.c (solib_aix_get_toc_value): Likewise. * solib-som.c (match_main, som_solib_section_offsets): Likewise. * solib.c (solib_read_symbols): Likewise. * stabsread.c (scan_file_globals): Likewise. * stap-probe.c (handle_stap_probe): Likewise. * symfile.c (symbol_file_clear, separate_debug_file_exists) (find_separate_debug_file_by_debuglink): Likewise. (reread_symbols): Likewise. Use the objfile field name renamed to original_name. (allocate_symtab): Use objfile_name. * symmisc.c (print_symbol_bcache_statistics, print_objfile_statistics) (dump_objfile, dump_msymbols, dump_symtab_1) (maintenance_print_msymbols, maintenance_print_objfiles) (maintenance_info_symtabs, maintenance_check_symtabs): Likewise. * target.c (target_translate_tls_address, target_info): Likewise. * xcoffread.c (xcoff_initial_scan): Make variable name const. Use objfile_name.
786 lines
20 KiB
C
786 lines
20 KiB
C
/* Generic static probe support for GDB.
|
||
|
||
Copyright (C) 2012-2013 Free Software Foundation, Inc.
|
||
|
||
This file is part of GDB.
|
||
|
||
This program is free software; you can redistribute it and/or modify
|
||
it under the terms of the GNU General Public License as published by
|
||
the Free Software Foundation; either version 3 of the License, or
|
||
(at your option) any later version.
|
||
|
||
This program 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 General Public License for more details.
|
||
|
||
You should have received a copy of the GNU General Public License
|
||
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||
|
||
#include "defs.h"
|
||
#include "probe.h"
|
||
#include "command.h"
|
||
#include "cli/cli-cmds.h"
|
||
#include "cli/cli-utils.h"
|
||
#include "objfiles.h"
|
||
#include "symtab.h"
|
||
#include "progspace.h"
|
||
#include "filenames.h"
|
||
#include "exceptions.h"
|
||
#include "linespec.h"
|
||
#include "gdb_regex.h"
|
||
#include "frame.h"
|
||
#include "arch-utils.h"
|
||
#include <ctype.h>
|
||
|
||
|
||
|
||
/* See definition in probe.h. */
|
||
|
||
struct symtabs_and_lines
|
||
parse_probes (char **argptr, struct linespec_result *canonical)
|
||
{
|
||
char *arg_start, *arg_end, *arg;
|
||
char *objfile_namestr = NULL, *provider = NULL, *name, *p;
|
||
struct cleanup *cleanup;
|
||
struct symtabs_and_lines result;
|
||
struct objfile *objfile;
|
||
struct program_space *pspace;
|
||
const struct probe_ops *probe_ops;
|
||
const char *cs;
|
||
|
||
result.sals = NULL;
|
||
result.nelts = 0;
|
||
|
||
arg_start = *argptr;
|
||
|
||
cs = *argptr;
|
||
probe_ops = probe_linespec_to_ops (&cs);
|
||
gdb_assert (probe_ops != NULL);
|
||
|
||
arg = (char *) cs;
|
||
arg = skip_spaces (arg);
|
||
if (!*arg)
|
||
error (_("argument to `%s' missing"), arg_start);
|
||
|
||
arg_end = skip_to_space (arg);
|
||
|
||
/* We make a copy here so we can write over parts with impunity. */
|
||
arg = savestring (arg, arg_end - arg);
|
||
cleanup = make_cleanup (xfree, arg);
|
||
|
||
/* Extract each word from the argument, separated by ":"s. */
|
||
p = strchr (arg, ':');
|
||
if (p == NULL)
|
||
{
|
||
/* This is `-p name'. */
|
||
name = arg;
|
||
}
|
||
else
|
||
{
|
||
char *hold = p + 1;
|
||
|
||
*p = '\0';
|
||
p = strchr (hold, ':');
|
||
if (p == NULL)
|
||
{
|
||
/* This is `-p provider:name'. */
|
||
provider = arg;
|
||
name = hold;
|
||
}
|
||
else
|
||
{
|
||
/* This is `-p objfile:provider:name'. */
|
||
*p = '\0';
|
||
objfile_namestr = arg;
|
||
provider = hold;
|
||
name = p + 1;
|
||
}
|
||
}
|
||
|
||
if (*name == '\0')
|
||
error (_("no probe name specified"));
|
||
if (provider && *provider == '\0')
|
||
error (_("invalid provider name"));
|
||
if (objfile_namestr && *objfile_namestr == '\0')
|
||
error (_("invalid objfile name"));
|
||
|
||
ALL_PSPACES (pspace)
|
||
ALL_PSPACE_OBJFILES (pspace, objfile)
|
||
{
|
||
VEC (probe_p) *probes;
|
||
struct probe *probe;
|
||
int ix;
|
||
|
||
if (!objfile->sf || !objfile->sf->sym_probe_fns)
|
||
continue;
|
||
|
||
if (objfile_namestr
|
||
&& FILENAME_CMP (objfile_name (objfile), objfile_namestr) != 0
|
||
&& FILENAME_CMP (lbasename (objfile_name (objfile)),
|
||
objfile_namestr) != 0)
|
||
continue;
|
||
|
||
probes = objfile->sf->sym_probe_fns->sym_get_probes (objfile);
|
||
|
||
for (ix = 0; VEC_iterate (probe_p, probes, ix, probe); ix++)
|
||
{
|
||
struct symtab_and_line *sal;
|
||
|
||
if (probe_ops != &probe_ops_any && probe->pops != probe_ops)
|
||
continue;
|
||
|
||
if (provider && strcmp (probe->provider, provider) != 0)
|
||
continue;
|
||
|
||
if (strcmp (probe->name, name) != 0)
|
||
continue;
|
||
|
||
++result.nelts;
|
||
result.sals = xrealloc (result.sals,
|
||
result.nelts
|
||
* sizeof (struct symtab_and_line));
|
||
sal = &result.sals[result.nelts - 1];
|
||
|
||
init_sal (sal);
|
||
|
||
sal->pc = probe->address;
|
||
sal->explicit_pc = 1;
|
||
sal->section = find_pc_overlay (sal->pc);
|
||
sal->pspace = pspace;
|
||
sal->probe = probe;
|
||
}
|
||
}
|
||
|
||
if (result.nelts == 0)
|
||
{
|
||
throw_error (NOT_FOUND_ERROR,
|
||
_("No probe matching objfile=`%s', provider=`%s', name=`%s'"),
|
||
objfile_namestr ? objfile_namestr : _("<any>"),
|
||
provider ? provider : _("<any>"),
|
||
name);
|
||
}
|
||
|
||
if (canonical)
|
||
{
|
||
canonical->special_display = 1;
|
||
canonical->pre_expanded = 1;
|
||
canonical->addr_string = savestring (*argptr, arg_end - *argptr);
|
||
}
|
||
|
||
*argptr = arg_end;
|
||
do_cleanups (cleanup);
|
||
|
||
return result;
|
||
}
|
||
|
||
/* See definition in probe.h. */
|
||
|
||
VEC (probe_p) *
|
||
find_probes_in_objfile (struct objfile *objfile, const char *provider,
|
||
const char *name)
|
||
{
|
||
VEC (probe_p) *probes, *result = NULL;
|
||
int ix;
|
||
struct probe *probe;
|
||
|
||
if (!objfile->sf || !objfile->sf->sym_probe_fns)
|
||
return NULL;
|
||
|
||
probes = objfile->sf->sym_probe_fns->sym_get_probes (objfile);
|
||
for (ix = 0; VEC_iterate (probe_p, probes, ix, probe); ix++)
|
||
{
|
||
if (strcmp (probe->provider, provider) != 0)
|
||
continue;
|
||
|
||
if (strcmp (probe->name, name) != 0)
|
||
continue;
|
||
|
||
VEC_safe_push (probe_p, result, probe);
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
/* See definition in probe.h. */
|
||
|
||
struct probe *
|
||
find_probe_by_pc (CORE_ADDR pc)
|
||
{
|
||
struct objfile *objfile;
|
||
|
||
ALL_OBJFILES (objfile)
|
||
{
|
||
VEC (probe_p) *probes;
|
||
int ix;
|
||
struct probe *probe;
|
||
|
||
if (!objfile->sf || !objfile->sf->sym_probe_fns)
|
||
continue;
|
||
|
||
/* If this proves too inefficient, we can replace with a hash. */
|
||
probes = objfile->sf->sym_probe_fns->sym_get_probes (objfile);
|
||
for (ix = 0; VEC_iterate (probe_p, probes, ix, probe); ix++)
|
||
if (probe->address == pc)
|
||
return probe;
|
||
}
|
||
|
||
return NULL;
|
||
}
|
||
|
||
|
||
|
||
/* Make a vector of probes matching OBJNAME, PROVIDER, and PROBE_NAME.
|
||
If POPS is not NULL, only probes of this certain probe_ops will match.
|
||
Each argument is a regexp, or NULL, which matches anything. */
|
||
|
||
static VEC (probe_p) *
|
||
collect_probes (char *objname, char *provider, char *probe_name,
|
||
const struct probe_ops *pops)
|
||
{
|
||
struct objfile *objfile;
|
||
VEC (probe_p) *result = NULL;
|
||
struct cleanup *cleanup, *cleanup_temps;
|
||
regex_t obj_pat, prov_pat, probe_pat;
|
||
|
||
cleanup = make_cleanup (VEC_cleanup (probe_p), &result);
|
||
|
||
cleanup_temps = make_cleanup (null_cleanup, NULL);
|
||
if (provider != NULL)
|
||
compile_rx_or_error (&prov_pat, provider, _("Invalid provider regexp"));
|
||
if (probe_name != NULL)
|
||
compile_rx_or_error (&probe_pat, probe_name, _("Invalid probe regexp"));
|
||
if (objname != NULL)
|
||
compile_rx_or_error (&obj_pat, objname, _("Invalid object file regexp"));
|
||
|
||
ALL_OBJFILES (objfile)
|
||
{
|
||
VEC (probe_p) *probes;
|
||
struct probe *probe;
|
||
int ix;
|
||
|
||
if (! objfile->sf || ! objfile->sf->sym_probe_fns)
|
||
continue;
|
||
|
||
if (objname)
|
||
{
|
||
if (regexec (&obj_pat, objfile_name (objfile), 0, NULL, 0) != 0)
|
||
continue;
|
||
}
|
||
|
||
probes = objfile->sf->sym_probe_fns->sym_get_probes (objfile);
|
||
|
||
for (ix = 0; VEC_iterate (probe_p, probes, ix, probe); ix++)
|
||
{
|
||
if (pops != NULL && probe->pops != pops)
|
||
continue;
|
||
|
||
if (provider
|
||
&& regexec (&prov_pat, probe->provider, 0, NULL, 0) != 0)
|
||
continue;
|
||
|
||
if (probe_name
|
||
&& regexec (&probe_pat, probe->name, 0, NULL, 0) != 0)
|
||
continue;
|
||
|
||
VEC_safe_push (probe_p, result, probe);
|
||
}
|
||
}
|
||
|
||
do_cleanups (cleanup_temps);
|
||
discard_cleanups (cleanup);
|
||
return result;
|
||
}
|
||
|
||
/* A qsort comparison function for probe_p objects. */
|
||
|
||
static int
|
||
compare_probes (const void *a, const void *b)
|
||
{
|
||
const struct probe *pa = *((const struct probe **) a);
|
||
const struct probe *pb = *((const struct probe **) b);
|
||
int v;
|
||
|
||
v = strcmp (pa->provider, pb->provider);
|
||
if (v)
|
||
return v;
|
||
|
||
v = strcmp (pa->name, pb->name);
|
||
if (v)
|
||
return v;
|
||
|
||
if (pa->address < pb->address)
|
||
return -1;
|
||
if (pa->address > pb->address)
|
||
return 1;
|
||
|
||
return strcmp (objfile_name (pa->objfile), objfile_name (pb->objfile));
|
||
}
|
||
|
||
/* Helper function that generate entries in the ui_out table being
|
||
crafted by `info_probes_for_ops'. */
|
||
|
||
static void
|
||
gen_ui_out_table_header_info (VEC (probe_p) *probes,
|
||
const struct probe_ops *p)
|
||
{
|
||
/* `headings' refers to the names of the columns when printing `info
|
||
probes'. */
|
||
VEC (info_probe_column_s) *headings = NULL;
|
||
struct cleanup *c;
|
||
info_probe_column_s *column;
|
||
size_t headings_size;
|
||
int ix;
|
||
|
||
gdb_assert (p != NULL);
|
||
|
||
if (p->gen_info_probes_table_header == NULL
|
||
&& p->gen_info_probes_table_values == NULL)
|
||
return;
|
||
|
||
gdb_assert (p->gen_info_probes_table_header != NULL
|
||
&& p->gen_info_probes_table_values != NULL);
|
||
|
||
c = make_cleanup (VEC_cleanup (info_probe_column_s), &headings);
|
||
p->gen_info_probes_table_header (&headings);
|
||
|
||
headings_size = VEC_length (info_probe_column_s, headings);
|
||
|
||
for (ix = 0;
|
||
VEC_iterate (info_probe_column_s, headings, ix, column);
|
||
++ix)
|
||
{
|
||
struct probe *probe;
|
||
int jx;
|
||
size_t size_max = strlen (column->print_name);
|
||
|
||
for (jx = 0; VEC_iterate (probe_p, probes, jx, probe); ++jx)
|
||
{
|
||
/* `probe_fields' refers to the values of each new field that this
|
||
probe will display. */
|
||
VEC (const_char_ptr) *probe_fields = NULL;
|
||
struct cleanup *c2;
|
||
const char *val;
|
||
int kx;
|
||
|
||
if (probe->pops != p)
|
||
continue;
|
||
|
||
c2 = make_cleanup (VEC_cleanup (const_char_ptr), &probe_fields);
|
||
p->gen_info_probes_table_values (probe, &probe_fields);
|
||
|
||
gdb_assert (VEC_length (const_char_ptr, probe_fields)
|
||
== headings_size);
|
||
|
||
for (kx = 0; VEC_iterate (const_char_ptr, probe_fields, kx, val);
|
||
++kx)
|
||
{
|
||
/* It is valid to have a NULL value here, which means that the
|
||
backend does not have something to write and this particular
|
||
field should be skipped. */
|
||
if (val == NULL)
|
||
continue;
|
||
|
||
size_max = max (strlen (val), size_max);
|
||
}
|
||
do_cleanups (c2);
|
||
}
|
||
|
||
ui_out_table_header (current_uiout, size_max, ui_left,
|
||
column->field_name, column->print_name);
|
||
}
|
||
|
||
do_cleanups (c);
|
||
}
|
||
|
||
/* Helper function to print extra information about a probe and an objfile
|
||
represented by PROBE. */
|
||
|
||
static void
|
||
print_ui_out_info (struct probe *probe)
|
||
{
|
||
int ix;
|
||
int j = 0;
|
||
/* `values' refers to the actual values of each new field in the output
|
||
of `info probe'. `headings' refers to the names of each new field. */
|
||
VEC (const_char_ptr) *values = NULL;
|
||
VEC (info_probe_column_s) *headings = NULL;
|
||
info_probe_column_s *column;
|
||
struct cleanup *c;
|
||
|
||
gdb_assert (probe != NULL);
|
||
gdb_assert (probe->pops != NULL);
|
||
|
||
if (probe->pops->gen_info_probes_table_header == NULL
|
||
&& probe->pops->gen_info_probes_table_values == NULL)
|
||
return;
|
||
|
||
gdb_assert (probe->pops->gen_info_probes_table_header != NULL
|
||
&& probe->pops->gen_info_probes_table_values != NULL);
|
||
|
||
c = make_cleanup (VEC_cleanup (info_probe_column_s), &headings);
|
||
make_cleanup (VEC_cleanup (const_char_ptr), &values);
|
||
|
||
probe->pops->gen_info_probes_table_header (&headings);
|
||
probe->pops->gen_info_probes_table_values (probe, &values);
|
||
|
||
gdb_assert (VEC_length (info_probe_column_s, headings)
|
||
== VEC_length (const_char_ptr, values));
|
||
|
||
for (ix = 0;
|
||
VEC_iterate (info_probe_column_s, headings, ix, column);
|
||
++ix)
|
||
{
|
||
const char *val = VEC_index (const_char_ptr, values, j++);
|
||
|
||
if (val == NULL)
|
||
ui_out_field_skip (current_uiout, column->field_name);
|
||
else
|
||
ui_out_field_string (current_uiout, column->field_name, val);
|
||
}
|
||
|
||
do_cleanups (c);
|
||
}
|
||
|
||
/* Helper function that returns the number of extra fields which POPS will
|
||
need. */
|
||
|
||
static int
|
||
get_number_extra_fields (const struct probe_ops *pops)
|
||
{
|
||
VEC (info_probe_column_s) *headings = NULL;
|
||
struct cleanup *c;
|
||
int n;
|
||
|
||
if (pops->gen_info_probes_table_header == NULL)
|
||
return 0;
|
||
|
||
c = make_cleanup (VEC_cleanup (info_probe_column_s), &headings);
|
||
pops->gen_info_probes_table_header (&headings);
|
||
|
||
n = VEC_length (info_probe_column_s, headings);
|
||
|
||
do_cleanups (c);
|
||
|
||
return n;
|
||
}
|
||
|
||
/* See comment in probe.h. */
|
||
|
||
void
|
||
info_probes_for_ops (char *arg, int from_tty, const struct probe_ops *pops)
|
||
{
|
||
char *provider, *probe_name = NULL, *objname = NULL;
|
||
struct cleanup *cleanup = make_cleanup (null_cleanup, NULL);
|
||
VEC (probe_p) *probes;
|
||
int i, any_found;
|
||
int ui_out_extra_fields = 0;
|
||
size_t size_addr;
|
||
size_t size_name = strlen ("Name");
|
||
size_t size_objname = strlen ("Object");
|
||
size_t size_provider = strlen ("Provider");
|
||
struct probe *probe;
|
||
struct gdbarch *gdbarch = get_current_arch ();
|
||
|
||
/* Do we have a `provider:probe:objfile' style of linespec? */
|
||
provider = extract_arg (&arg);
|
||
if (provider)
|
||
{
|
||
make_cleanup (xfree, provider);
|
||
|
||
probe_name = extract_arg (&arg);
|
||
if (probe_name)
|
||
{
|
||
make_cleanup (xfree, probe_name);
|
||
|
||
objname = extract_arg (&arg);
|
||
if (objname)
|
||
make_cleanup (xfree, objname);
|
||
}
|
||
}
|
||
|
||
if (pops == NULL)
|
||
{
|
||
const struct probe_ops *po;
|
||
int ix;
|
||
|
||
/* If the probe_ops is NULL, it means the user has requested a "simple"
|
||
`info probes', i.e., she wants to print all information about all
|
||
probes. For that, we have to identify how many extra fields we will
|
||
need to add in the ui_out table.
|
||
|
||
To do that, we iterate over all probe_ops, querying each one about
|
||
its extra fields, and incrementing `ui_out_extra_fields' to reflect
|
||
that number. */
|
||
|
||
for (ix = 0; VEC_iterate (probe_ops_cp, all_probe_ops, ix, po); ++ix)
|
||
ui_out_extra_fields += get_number_extra_fields (po);
|
||
}
|
||
else
|
||
ui_out_extra_fields = get_number_extra_fields (pops);
|
||
|
||
probes = collect_probes (objname, provider, probe_name, pops);
|
||
make_cleanup (VEC_cleanup (probe_p), &probes);
|
||
make_cleanup_ui_out_table_begin_end (current_uiout,
|
||
4 + ui_out_extra_fields,
|
||
VEC_length (probe_p, probes),
|
||
"StaticProbes");
|
||
|
||
if (!VEC_empty (probe_p, probes))
|
||
qsort (VEC_address (probe_p, probes), VEC_length (probe_p, probes),
|
||
sizeof (probe_p), compare_probes);
|
||
|
||
/* What's the size of an address in our architecture? */
|
||
size_addr = gdbarch_addr_bit (gdbarch) == 64 ? 18 : 10;
|
||
|
||
/* Determining the maximum size of each field (`provider', `name' and
|
||
`objname'). */
|
||
for (i = 0; VEC_iterate (probe_p, probes, i, probe); ++i)
|
||
{
|
||
size_name = max (strlen (probe->name), size_name);
|
||
size_provider = max (strlen (probe->provider), size_provider);
|
||
size_objname = max (strlen (objfile_name (probe->objfile)), size_objname);
|
||
}
|
||
|
||
ui_out_table_header (current_uiout, size_provider, ui_left, "provider",
|
||
_("Provider"));
|
||
ui_out_table_header (current_uiout, size_name, ui_left, "name", _("Name"));
|
||
ui_out_table_header (current_uiout, size_addr, ui_left, "addr", _("Where"));
|
||
|
||
if (pops == NULL)
|
||
{
|
||
const struct probe_ops *po;
|
||
int ix;
|
||
|
||
/* We have to generate the table header for each new probe type that we
|
||
will print. */
|
||
for (ix = 0; VEC_iterate (probe_ops_cp, all_probe_ops, ix, po); ++ix)
|
||
gen_ui_out_table_header_info (probes, po);
|
||
}
|
||
else
|
||
gen_ui_out_table_header_info (probes, pops);
|
||
|
||
ui_out_table_header (current_uiout, size_objname, ui_left, "object",
|
||
_("Object"));
|
||
ui_out_table_body (current_uiout);
|
||
|
||
for (i = 0; VEC_iterate (probe_p, probes, i, probe); ++i)
|
||
{
|
||
struct cleanup *inner;
|
||
|
||
inner = make_cleanup_ui_out_tuple_begin_end (current_uiout, "probe");
|
||
|
||
ui_out_field_string (current_uiout, "provider", probe->provider);
|
||
ui_out_field_string (current_uiout, "name", probe->name);
|
||
ui_out_field_core_addr (current_uiout, "addr",
|
||
get_objfile_arch (probe->objfile),
|
||
probe->address);
|
||
|
||
if (pops == NULL)
|
||
{
|
||
const struct probe_ops *po;
|
||
int ix;
|
||
|
||
for (ix = 0; VEC_iterate (probe_ops_cp, all_probe_ops, ix, po);
|
||
++ix)
|
||
if (probe->pops == po)
|
||
print_ui_out_info (probe);
|
||
}
|
||
else
|
||
print_ui_out_info (probe);
|
||
|
||
ui_out_field_string (current_uiout, "object",
|
||
objfile_name (probe->objfile));
|
||
ui_out_text (current_uiout, "\n");
|
||
|
||
do_cleanups (inner);
|
||
}
|
||
|
||
any_found = !VEC_empty (probe_p, probes);
|
||
do_cleanups (cleanup);
|
||
|
||
if (!any_found)
|
||
ui_out_message (current_uiout, 0, _("No probes matched.\n"));
|
||
}
|
||
|
||
/* Implementation of the `info probes' command. */
|
||
|
||
static void
|
||
info_probes_command (char *arg, int from_tty)
|
||
{
|
||
info_probes_for_ops (arg, from_tty, NULL);
|
||
}
|
||
|
||
/* See comments in probe.h. */
|
||
|
||
unsigned
|
||
get_probe_argument_count (struct probe *probe)
|
||
{
|
||
const struct sym_probe_fns *probe_fns;
|
||
|
||
gdb_assert (probe->objfile != NULL);
|
||
gdb_assert (probe->objfile->sf != NULL);
|
||
|
||
probe_fns = probe->objfile->sf->sym_probe_fns;
|
||
|
||
gdb_assert (probe_fns != NULL);
|
||
|
||
return probe_fns->sym_get_probe_argument_count (probe);
|
||
}
|
||
|
||
/* See comments in probe.h. */
|
||
|
||
int
|
||
can_evaluate_probe_arguments (struct probe *probe)
|
||
{
|
||
const struct sym_probe_fns *probe_fns;
|
||
|
||
gdb_assert (probe->objfile != NULL);
|
||
gdb_assert (probe->objfile->sf != NULL);
|
||
|
||
probe_fns = probe->objfile->sf->sym_probe_fns;
|
||
|
||
gdb_assert (probe_fns != NULL);
|
||
|
||
return probe_fns->can_evaluate_probe_arguments (probe);
|
||
}
|
||
|
||
/* See comments in probe.h. */
|
||
|
||
struct value *
|
||
evaluate_probe_argument (struct probe *probe, unsigned n)
|
||
{
|
||
const struct sym_probe_fns *probe_fns;
|
||
|
||
gdb_assert (probe->objfile != NULL);
|
||
gdb_assert (probe->objfile->sf != NULL);
|
||
|
||
probe_fns = probe->objfile->sf->sym_probe_fns;
|
||
|
||
gdb_assert (probe_fns != NULL);
|
||
|
||
return probe_fns->sym_evaluate_probe_argument (probe, n);
|
||
}
|
||
|
||
/* See comments in probe.h. */
|
||
|
||
struct value *
|
||
probe_safe_evaluate_at_pc (struct frame_info *frame, unsigned n)
|
||
{
|
||
struct probe *probe;
|
||
unsigned n_args;
|
||
|
||
probe = find_probe_by_pc (get_frame_pc (frame));
|
||
if (!probe)
|
||
return NULL;
|
||
|
||
n_args = get_probe_argument_count (probe);
|
||
if (n >= n_args)
|
||
return NULL;
|
||
|
||
return evaluate_probe_argument (probe, n);
|
||
}
|
||
|
||
/* See comment in probe.h. */
|
||
|
||
const struct probe_ops *
|
||
probe_linespec_to_ops (const char **linespecp)
|
||
{
|
||
int ix;
|
||
const struct probe_ops *probe_ops;
|
||
|
||
for (ix = 0; VEC_iterate (probe_ops_cp, all_probe_ops, ix, probe_ops); ix++)
|
||
if (probe_ops->is_linespec (linespecp))
|
||
return probe_ops;
|
||
|
||
return NULL;
|
||
}
|
||
|
||
/* See comment in probe.h. */
|
||
|
||
int
|
||
probe_is_linespec_by_keyword (const char **linespecp, const char *const *keywords)
|
||
{
|
||
const char *s = *linespecp;
|
||
const char *const *csp;
|
||
|
||
for (csp = keywords; *csp; csp++)
|
||
{
|
||
const char *keyword = *csp;
|
||
size_t len = strlen (keyword);
|
||
|
||
if (strncmp (s, keyword, len) == 0 && isspace (s[len]))
|
||
{
|
||
*linespecp += len + 1;
|
||
return 1;
|
||
}
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Implementation of `is_linespec' method for `struct probe_ops'. */
|
||
|
||
static int
|
||
probe_any_is_linespec (const char **linespecp)
|
||
{
|
||
static const char *const keywords[] = { "-p", "-probe", NULL };
|
||
|
||
return probe_is_linespec_by_keyword (linespecp, keywords);
|
||
}
|
||
|
||
/* Dummy method used for `probe_ops_any'. */
|
||
|
||
static void
|
||
probe_any_get_probes (VEC (probe_p) **probesp, struct objfile *objfile)
|
||
{
|
||
/* No probes can be provided by this dummy backend. */
|
||
}
|
||
|
||
/* Operations associated with a generic probe. */
|
||
|
||
const struct probe_ops probe_ops_any =
|
||
{
|
||
probe_any_is_linespec,
|
||
probe_any_get_probes,
|
||
};
|
||
|
||
/* See comments in probe.h. */
|
||
|
||
struct cmd_list_element **
|
||
info_probes_cmdlist_get (void)
|
||
{
|
||
static struct cmd_list_element *info_probes_cmdlist;
|
||
|
||
if (info_probes_cmdlist == NULL)
|
||
add_prefix_cmd ("probes", class_info, info_probes_command,
|
||
_("\
|
||
Show available static probes.\n\
|
||
Usage: info probes [all|TYPE [ARGS]]\n\
|
||
TYPE specifies the type of the probe, and can be one of the following:\n\
|
||
- stap\n\
|
||
If you specify TYPE, there may be additional arguments needed by the\n\
|
||
subcommand.\n\
|
||
If you do not specify any argument, or specify `all', then the command\n\
|
||
will show information about all types of probes."),
|
||
&info_probes_cmdlist, "info probes ",
|
||
0/*allow-unknown*/, &infolist);
|
||
|
||
return &info_probes_cmdlist;
|
||
}
|
||
|
||
VEC (probe_ops_cp) *all_probe_ops;
|
||
|
||
void _initialize_probe (void);
|
||
|
||
void
|
||
_initialize_probe (void)
|
||
{
|
||
VEC_safe_push (probe_ops_cp, all_probe_ops, &probe_ops_any);
|
||
|
||
add_cmd ("all", class_info, info_probes_command,
|
||
_("\
|
||
Show information about all type of probes."),
|
||
info_probes_cmdlist_get ());
|
||
}
|