gdb/macrocmd.c - gdb
Global variables defined
Functions defined
Source code
- #include "defs.h"
- #include "macrotab.h"
- #include "macroexp.h"
- #include "macroscope.h"
- #include "cli/cli-utils.h"
- #include "command.h"
- #include "gdbcmd.h"
- #include "linespec.h"
- static struct cmd_list_element *macrolist;
- static void
- macro_command (char *arg, int from_tty)
- {
- printf_unfiltered
- ("\"macro\" must be followed by the name of a macro command.\n");
- help_list (macrolist, "macro ", all_commands, gdb_stdout);
- }
- static void
- macro_inform_no_debuginfo (void)
- {
- puts_filtered ("GDB has no preprocessor macro information for that code.\n");
- }
- static void
- macro_expand_command (char *exp, int from_tty)
- {
- struct macro_scope *ms = NULL;
- char *expanded = NULL;
- struct cleanup *cleanup_chain = make_cleanup (free_current_contents, &ms);
- make_cleanup (free_current_contents, &expanded);
-
- if (! exp || ! *exp)
- error (_("You must follow the `macro expand' command with the"
- " expression you\n"
- "want to expand."));
- ms = default_macro_scope ();
- if (ms)
- {
- expanded = macro_expand (exp, standard_macro_lookup, ms);
- fputs_filtered ("expands to: ", gdb_stdout);
- fputs_filtered (expanded, gdb_stdout);
- fputs_filtered ("\n", gdb_stdout);
- }
- else
- macro_inform_no_debuginfo ();
- do_cleanups (cleanup_chain);
- return;
- }
- static void
- macro_expand_once_command (char *exp, int from_tty)
- {
- struct macro_scope *ms = NULL;
- char *expanded = NULL;
- struct cleanup *cleanup_chain = make_cleanup (free_current_contents, &ms);
- make_cleanup (free_current_contents, &expanded);
-
- if (! exp || ! *exp)
- error (_("You must follow the `macro expand-once' command with"
- " the expression\n"
- "you want to expand."));
- ms = default_macro_scope ();
- if (ms)
- {
- expanded = macro_expand_once (exp, standard_macro_lookup, ms);
- fputs_filtered ("expands to: ", gdb_stdout);
- fputs_filtered (expanded, gdb_stdout);
- fputs_filtered ("\n", gdb_stdout);
- }
- else
- macro_inform_no_debuginfo ();
- do_cleanups (cleanup_chain);
- return;
- }
- static void
- show_pp_source_pos (struct ui_file *stream,
- struct macro_source_file *file,
- int line)
- {
- char *fullname;
- fullname = macro_source_fullname (file);
- fprintf_filtered (stream, "%s:%d\n", fullname, line);
- xfree (fullname);
- while (file->included_by)
- {
- fullname = macro_source_fullname (file->included_by);
- fprintf_filtered (gdb_stdout, " included at %s:%d\n", fullname,
- file->included_at_line);
- xfree (fullname);
- file = file->included_by;
- }
- }
- static void
- print_macro_definition (const char *name,
- const struct macro_definition *d,
- struct macro_source_file *file,
- int line)
- {
- fprintf_filtered (gdb_stdout, "Defined at ");
- show_pp_source_pos (gdb_stdout, file, line);
- if (line != 0)
- fprintf_filtered (gdb_stdout, "#define %s", name);
- else
- fprintf_filtered (gdb_stdout, "-D%s", name);
- if (d->kind == macro_function_like)
- {
- int i;
- fputs_filtered ("(", gdb_stdout);
- for (i = 0; i < d->argc; i++)
- {
- fputs_filtered (d->argv[i], gdb_stdout);
- if (i + 1 < d->argc)
- fputs_filtered (", ", gdb_stdout);
- }
- fputs_filtered (")", gdb_stdout);
- }
- if (line != 0)
- fprintf_filtered (gdb_stdout, " %s\n", d->replacement);
- else
- fprintf_filtered (gdb_stdout, "=%s\n", d->replacement);
- }
- static void
- print_macro_callback (const char *name, const struct macro_definition *macro,
- struct macro_source_file *source, int line,
- void *user_data)
- {
- if (! user_data || strcmp (user_data, name) == 0)
- print_macro_definition (name, macro, source, line);
- }
- static void
- info_macro_command (char *args, int from_tty)
- {
- struct macro_scope *ms = NULL;
- struct cleanup *cleanup_chain;
- char *name;
- int show_all_macros_named = 0;
- char *arg_start = args;
- int processing_args = 1;
- while (processing_args
- && arg_start && *arg_start == '-' && *arg_start != '\0')
- {
- char *p = skip_to_space (arg_start);
- if (strncmp (arg_start, "-a", p - arg_start) == 0
- || strncmp (arg_start, "-all", p - arg_start) == 0)
- show_all_macros_named = 1;
- else if (strncmp (arg_start, "--", p - arg_start) == 0)
-
- processing_args = 0;
- else
- {
-
- *p = '\0';
- error (_("Unrecognized option '%s' to info macro command. "
- "Try \"help info macro\"."), arg_start);
- }
- arg_start = skip_spaces (p);
- }
- name = arg_start;
- if (! name || ! *name)
- error (_("You must follow the `info macro' command with the name"
- " of the macro\n"
- "whose definition you want to see."));
- ms = default_macro_scope ();
- cleanup_chain = make_cleanup (free_current_contents, &ms);
- if (! ms)
- macro_inform_no_debuginfo ();
- else if (show_all_macros_named)
- macro_for_each (ms->file->table, print_macro_callback, name);
- else
- {
- struct macro_definition *d;
- d = macro_lookup_definition (ms->file, ms->line, name);
- if (d)
- {
- int line;
- struct macro_source_file *file
- = macro_definition_location (ms->file, ms->line, name, &line);
- print_macro_definition (name, d, file, line);
- }
- else
- {
- fprintf_filtered (gdb_stdout,
- "The symbol `%s' has no definition as a C/C++"
- " preprocessor macro\n"
- "at ", name);
- show_pp_source_pos (gdb_stdout, ms->file, ms->line);
- }
- }
- do_cleanups (cleanup_chain);
- }
- static void
- info_macros_command (char *args, int from_tty)
- {
- struct macro_scope *ms = NULL;
- struct cleanup *cleanup_chain = make_cleanup (free_current_contents, &ms);
- if (args == NULL)
- ms = default_macro_scope ();
- else
- {
- struct symtabs_and_lines sals =
- decode_line_with_current_source (args, 0);
- if (sals.nelts)
- ms = sal_macro_scope (sals.sals[0]);
- }
- if (! ms || ! ms->file || ! ms->file->table)
- macro_inform_no_debuginfo ();
- else
- macro_for_each_in_scope (ms->file, ms->line, print_macro_callback, NULL);
- do_cleanups (cleanup_chain);
- }
- static void
- skip_ws (char **expp)
- {
- while (macro_is_whitespace (**expp))
- ++*expp;
- }
- static char *
- extract_identifier (char **expp, int is_parameter)
- {
- char *result;
- char *p = *expp;
- unsigned int len;
- if (is_parameter && !strncmp (p, "...", 3))
- {
-
- }
- else
- {
- if (! *p || ! macro_is_identifier_nondigit (*p))
- return NULL;
- for (++p;
- *p && (macro_is_identifier_nondigit (*p) || macro_is_digit (*p));
- ++p)
- ;
- }
- if (is_parameter && !strncmp (p, "...", 3))
- p += 3;
- len = p - *expp;
- result = (char *) xmalloc (len + 1);
- memcpy (result, *expp, len);
- result[len] = '\0';
- *expp += len;
- return result;
- }
- static void
- free_macro_definition_ptr (void *ptr)
- {
- int i;
- struct macro_definition *loc = (struct macro_definition *) ptr;
- for (i = 0; i < loc->argc; ++i)
- xfree ((char *) loc->argv[i]);
- xfree ((char *) loc->argv);
-
- }
- static void
- macro_define_command (char *exp, int from_tty)
- {
- struct macro_definition new_macro;
- char *name = NULL;
- struct cleanup *cleanup_chain;
- if (!exp)
- error (_("usage: macro define NAME[(ARGUMENT-LIST)] [REPLACEMENT-LIST]"));
- cleanup_chain = make_cleanup (free_macro_definition_ptr, &new_macro);
- make_cleanup (free_current_contents, &name);
- memset (&new_macro, 0, sizeof (struct macro_definition));
- skip_ws (&exp);
- name = extract_identifier (&exp, 0);
- if (! name)
- error (_("Invalid macro name."));
- if (*exp == '(')
- {
-
- int alloced = 5;
- char **argv = (char **) xmalloc (alloced * sizeof (char *));
- new_macro.kind = macro_function_like;
- new_macro.argc = 0;
- new_macro.argv = (const char * const *) argv;
-
- ++exp;
- skip_ws (&exp);
- while (*exp != ')')
- {
- int i;
- if (new_macro.argc == alloced)
- {
- alloced *= 2;
- argv = (char **) xrealloc (argv, alloced * sizeof (char *));
-
- new_macro.argv = (const char * const *) argv;
- }
- argv[new_macro.argc] = extract_identifier (&exp, 1);
- if (! argv[new_macro.argc])
- error (_("Macro is missing an argument."));
- ++new_macro.argc;
- for (i = new_macro.argc - 2; i >= 0; --i)
- {
- if (! strcmp (argv[i], argv[new_macro.argc - 1]))
- error (_("Two macro arguments with identical names."));
- }
- skip_ws (&exp);
- if (*exp == ',')
- {
- ++exp;
- skip_ws (&exp);
- }
- else if (*exp != ')')
- error (_("',' or ')' expected at end of macro arguments."));
- }
-
- ++exp;
- skip_ws (&exp);
- macro_define_function (macro_main (macro_user_macros), -1, name,
- new_macro.argc, (const char **) new_macro.argv,
- exp);
- }
- else
- {
- skip_ws (&exp);
- macro_define_object (macro_main (macro_user_macros), -1, name, exp);
- }
- do_cleanups (cleanup_chain);
- }
- static void
- macro_undef_command (char *exp, int from_tty)
- {
- char *name;
- if (!exp)
- error (_("usage: macro undef NAME"));
- skip_ws (&exp);
- name = extract_identifier (&exp, 0);
- if (! name)
- error (_("Invalid macro name."));
- macro_undef (macro_main (macro_user_macros), -1, name);
- xfree (name);
- }
- static void
- print_one_macro (const char *name, const struct macro_definition *macro,
- struct macro_source_file *source, int line,
- void *ignore)
- {
- fprintf_filtered (gdb_stdout, "macro define %s", name);
- if (macro->kind == macro_function_like)
- {
- int i;
- fprintf_filtered (gdb_stdout, "(");
- for (i = 0; i < macro->argc; ++i)
- fprintf_filtered (gdb_stdout, "%s%s", (i > 0) ? ", " : "",
- macro->argv[i]);
- fprintf_filtered (gdb_stdout, ")");
- }
- fprintf_filtered (gdb_stdout, " %s\n", macro->replacement);
- }
- static void
- macro_list_command (char *exp, int from_tty)
- {
- macro_for_each (macro_user_macros, print_one_macro, NULL);
- }
- extern initialize_file_ftype _initialize_macrocmd;
- void
- _initialize_macrocmd (void)
- {
-
- add_prefix_cmd ("macro", class_info, macro_command,
- _("Prefix for commands dealing with C preprocessor macros."),
- ¯olist, "macro ", 0, &cmdlist);
- add_cmd ("expand", no_class, macro_expand_command, _("\
- Fully expand any C/C++ preprocessor macro invocations in EXPRESSION.\n\
- Show the expanded expression."),
- ¯olist);
- add_alias_cmd ("exp", "expand", no_class, 1, ¯olist);
- add_cmd ("expand-once", no_class, macro_expand_once_command, _("\
- Expand C/C++ preprocessor macro invocations appearing directly in EXPRESSION.\n\
- Show the expanded expression.\n\
- \n\
- This command differs from `macro expand' in that it only expands macro\n\
- invocations that appear directly in EXPRESSION; if expanding a macro\n\
- introduces further macro invocations, those are left unexpanded.\n\
- \n\
- `macro expand-once' helps you see how a particular macro expands,\n\
- whereas `macro expand' shows you how all the macros involved in an\n\
- expression work together to yield a pre-processed expression."),
- ¯olist);
- add_alias_cmd ("exp1", "expand-once", no_class, 1, ¯olist);
- add_cmd ("macro", no_class, info_macro_command,
- _("Show the definition of MACRO, and it's source location.\n\
- Usage: info macro [-a|-all] [--] MACRO\n\
- Options: \n\
- -a, --all Output all definitions of MACRO in the current compilation\
- unit.\n\
- -- Specify the end of arguments and the beginning of the MACRO."),
- &infolist);
- add_cmd ("macros", no_class, info_macros_command,
- _("Show the definitions of all macros at LINESPEC, or the current \
- source location.\n\
- Usage: info macros [LINESPEC]"),
- &infolist);
- add_cmd ("define", no_class, macro_define_command, _("\
- Define a new C/C++ preprocessor macro.\n\
- The GDB command `macro define DEFINITION' is equivalent to placing a\n\
- preprocessor directive of the form `#define DEFINITION' such that the\n\
- definition is visible in all the inferior's source files.\n\
- For example:\n\
- (gdb) macro define PI (3.1415926)\n\
- (gdb) macro define MIN(x,y) ((x) < (y) ? (x) : (y))"),
- ¯olist);
- add_cmd ("undef", no_class, macro_undef_command, _("\
- Remove the definition of the C/C++ preprocessor macro with the given name."),
- ¯olist);
- add_cmd ("list", no_class, macro_list_command,
- _("List all the macros defined using the `macro define' command."),
- ¯olist);
- }