gdb/probe.c - gdb
Global variables defined
Data types defined
Functions defined
Source code
- #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 "linespec.h"
- #include "gdb_regex.h"
- #include "frame.h"
- #include "arch-utils.h"
- #include <ctype.h>
- typedef struct bound_probe bound_probe_s;
- DEF_VEC_O (bound_probe_s);
- 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);
- if (probe_ops == NULL)
- error (_("'%s' is not a probe linespec"), arg_start);
- arg = (char *) cs;
- arg = skip_spaces (arg);
- if (!*arg)
- error (_("argument to `%s' missing"), arg_start);
- arg_end = skip_to_space (arg);
-
- arg = savestring (arg, arg_end - arg);
- cleanup = make_cleanup (xfree, arg);
-
- p = strchr (arg, ':');
- if (p == NULL)
- {
-
- name = arg;
- }
- else
- {
- char *hold = p + 1;
- *p = '\0';
- p = strchr (hold, ':');
- if (p == NULL)
- {
-
- provider = arg;
- name = hold;
- }
- else
- {
-
- *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 = get_probe_address (probe, objfile);
- sal->explicit_pc = 1;
- sal->section = find_pc_overlay (sal->pc);
- sal->pspace = pspace;
- sal->probe = probe;
- sal->objfile = objfile;
- }
- }
- 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;
- }
- 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;
- }
- struct bound_probe
- find_probe_by_pc (CORE_ADDR pc)
- {
- struct objfile *objfile;
- struct bound_probe result;
- result.objfile = NULL;
- result.probe = NULL;
- ALL_OBJFILES (objfile)
- {
- VEC (probe_p) *probes;
- int ix;
- struct probe *probe;
- if (!objfile->sf || !objfile->sf->sym_probe_fns
- || objfile->sect_index_text == -1)
- continue;
-
- probes = objfile->sf->sym_probe_fns->sym_get_probes (objfile);
- for (ix = 0; VEC_iterate (probe_p, probes, ix, probe); ix++)
- if (get_probe_address (probe, objfile) == pc)
- {
- result.objfile = objfile;
- result.probe = probe;
- return result;
- }
- }
- return result;
- }
- static VEC (bound_probe_s) *
- collect_probes (char *objname, char *provider, char *probe_name,
- const struct probe_ops *pops)
- {
- struct objfile *objfile;
- VEC (bound_probe_s) *result = NULL;
- struct cleanup *cleanup, *cleanup_temps;
- regex_t obj_pat, prov_pat, probe_pat;
- cleanup = make_cleanup (VEC_cleanup (bound_probe_s), &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++)
- {
- struct bound_probe bound;
- 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;
- bound.objfile = objfile;
- bound.probe = probe;
- VEC_safe_push (bound_probe_s, result, &bound);
- }
- }
- do_cleanups (cleanup_temps);
- discard_cleanups (cleanup);
- return result;
- }
- static int
- compare_probes (const void *a, const void *b)
- {
- const struct bound_probe *pa = (const struct bound_probe *) a;
- const struct bound_probe *pb = (const struct bound_probe *) b;
- int v;
- v = strcmp (pa->probe->provider, pb->probe->provider);
- if (v)
- return v;
- v = strcmp (pa->probe->name, pb->probe->name);
- if (v)
- return v;
- if (pa->probe->address < pb->probe->address)
- return -1;
- if (pa->probe->address > pb->probe->address)
- return 1;
- return strcmp (objfile_name (pa->objfile), objfile_name (pb->objfile));
- }
- static void
- gen_ui_out_table_header_info (VEC (bound_probe_s) *probes,
- const struct probe_ops *p)
- {
-
- 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 bound_probe *probe;
- int jx;
- size_t size_max = strlen (column->print_name);
- for (jx = 0; VEC_iterate (bound_probe_s, probes, jx, probe); ++jx)
- {
-
- VEC (const_char_ptr) *probe_fields = NULL;
- struct cleanup *c2;
- const char *val;
- int kx;
- if (probe->probe->pops != p)
- continue;
- c2 = make_cleanup (VEC_cleanup (const_char_ptr), &probe_fields);
- p->gen_info_probes_table_values (probe->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)
- {
-
- 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);
- }
- static void
- print_ui_out_info (struct probe *probe)
- {
- int ix;
- int j = 0;
-
- 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);
- }
- 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;
- }
- void
- info_probes_for_ops (const 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 (bound_probe_s) *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 bound_probe *probe;
- struct gdbarch *gdbarch = get_current_arch ();
-
- provider = extract_arg_const (&arg);
- if (provider)
- {
- make_cleanup (xfree, provider);
- probe_name = extract_arg_const (&arg);
- if (probe_name)
- {
- make_cleanup (xfree, probe_name);
- objname = extract_arg_const (&arg);
- if (objname)
- make_cleanup (xfree, objname);
- }
- }
- if (pops == NULL)
- {
- const struct probe_ops *po;
- int ix;
-
- 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 (bound_probe_s, probes),
- "StaticProbes");
- if (!VEC_empty (bound_probe_s, probes))
- qsort (VEC_address (bound_probe_s, probes),
- VEC_length (bound_probe_s, probes),
- sizeof (bound_probe_s), compare_probes);
-
- size_addr = gdbarch_addr_bit (gdbarch) == 64 ? 18 : 10;
-
- for (i = 0; VEC_iterate (bound_probe_s, probes, i, probe); ++i)
- {
- size_name = max (strlen (probe->probe->name), size_name);
- size_provider = max (strlen (probe->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;
-
- 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 (bound_probe_s, 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->probe->provider);
- ui_out_field_string (current_uiout, "name", probe->probe->name);
- ui_out_field_core_addr (current_uiout, "addr",
- probe->probe->arch,
- get_probe_address (probe->probe, probe->objfile));
- 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->probe->pops == po)
- print_ui_out_info (probe->probe);
- }
- else
- print_ui_out_info (probe->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 (bound_probe_s, probes);
- do_cleanups (cleanup);
- if (!any_found)
- ui_out_message (current_uiout, 0, _("No probes matched.\n"));
- }
- static void
- info_probes_command (char *arg, int from_tty)
- {
- info_probes_for_ops (arg, from_tty, NULL);
- }
- CORE_ADDR
- get_probe_address (struct probe *probe, struct objfile *objfile)
- {
- return probe->pops->get_probe_address (probe, objfile);
- }
- unsigned
- get_probe_argument_count (struct probe *probe, struct frame_info *frame)
- {
- return probe->pops->get_probe_argument_count (probe, frame);
- }
- int
- can_evaluate_probe_arguments (struct probe *probe)
- {
- return probe->pops->can_evaluate_probe_arguments (probe);
- }
- struct value *
- evaluate_probe_argument (struct probe *probe, unsigned n,
- struct frame_info *frame)
- {
- return probe->pops->evaluate_probe_argument (probe, n, frame);
- }
- struct value *
- probe_safe_evaluate_at_pc (struct frame_info *frame, unsigned n)
- {
- struct bound_probe probe;
- unsigned n_args;
- probe = find_probe_by_pc (get_frame_pc (frame));
- if (!probe.probe)
- return NULL;
- n_args = get_probe_argument_count (probe.probe, frame);
- if (n >= n_args)
- return NULL;
- return evaluate_probe_argument (probe.probe, n, frame);
- }
- 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;
- }
- 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;
- }
- 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);
- }
- static void
- probe_any_get_probes (VEC (probe_p) **probesp, struct objfile *objfile)
- {
-
- }
- const struct probe_ops probe_ops_any =
- {
- probe_any_is_linespec,
- probe_any_get_probes,
- };
- 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, &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 ());
- }