gdb/cli/cli-cmds.c - gdb
Global variables defined
Functions defined
Source code
- #include "defs.h"
- #include "arch-utils.h"
- #include "dyn-string.h"
- #include "readline/readline.h"
- #include "readline/tilde.h"
- #include "completer.h"
- #include "target.h"
- #include "gdb_wait.h"
- #include "gdb_regex.h"
- #include "gdb_vfork.h"
- #include "linespec.h"
- #include "expression.h"
- #include "frame.h"
- #include "value.h"
- #include "language.h"
- #include "filenames.h"
- #include "objfiles.h"
- #include "source.h"
- #include "disasm.h"
- #include "tracepoint.h"
- #include "filestuff.h"
- #include "ui-out.h"
- #include "top.h"
- #include "cli/cli-decode.h"
- #include "cli/cli-script.h"
- #include "cli/cli-setshow.h"
- #include "cli/cli-cmds.h"
- #include "cli/cli-utils.h"
- #include "extension.h"
- #ifdef TUI
- #include "tui/tui.h"
- #endif
- #include <fcntl.h>
- static void complete_command (char *, int);
- static void echo_command (char *, int);
- static void pwd_command (char *, int);
- static void show_version (char *, int);
- static void help_command (char *, int);
- static void show_command (char *, int);
- static void info_command (char *, int);
- static void show_debug (char *, int);
- static void set_debug (char *, int);
- static void show_user (char *, int);
- static void make_command (char *, int);
- static void shell_escape (char *, int);
- static void edit_command (char *, int);
- static void list_command (char *, int);
- static void ambiguous_line_spec (struct symtabs_and_lines *);
- static void filter_sals (struct symtabs_and_lines *);
- unsigned int max_user_call_depth;
- struct cmd_list_element *cmdlist;
- struct cmd_list_element *infolist;
- struct cmd_list_element *enablelist;
- struct cmd_list_element *disablelist;
- struct cmd_list_element *stoplist;
- struct cmd_list_element *deletelist;
- struct cmd_list_element *detachlist;
- struct cmd_list_element *killlist;
- struct cmd_list_element *setlist;
- struct cmd_list_element *unsetlist;
- struct cmd_list_element *showlist;
- struct cmd_list_element *sethistlist;
- struct cmd_list_element *showhistlist;
- struct cmd_list_element *unsethistlist;
- struct cmd_list_element *maintenancelist;
- struct cmd_list_element *maintenanceinfolist;
- struct cmd_list_element *maintenanceprintlist;
- struct cmd_list_element *setprintlist;
- struct cmd_list_element *showprintlist;
- struct cmd_list_element *setdebuglist;
- struct cmd_list_element *showdebuglist;
- struct cmd_list_element *setchecklist;
- struct cmd_list_element *showchecklist;
- int source_verbose = 0;
- int trace_commands = 0;
- static const char script_ext_off[] = "off";
- static const char script_ext_soft[] = "soft";
- static const char script_ext_strict[] = "strict";
- static const char *const script_ext_enums[] = {
- script_ext_off,
- script_ext_soft,
- script_ext_strict,
- NULL
- };
- static const char *script_ext_mode = script_ext_soft;
- void
- error_no_arg (const char *why)
- {
- error (_("Argument required (%s)."), why);
- }
- static void
- info_command (char *arg, int from_tty)
- {
- printf_unfiltered (_("\"info\" must be followed by "
- "the name of an info command.\n"));
- help_list (infolist, "info ", all_commands, gdb_stdout);
- }
- static void
- show_command (char *arg, int from_tty)
- {
- cmd_show_list (showlist, from_tty, "");
- }
- static void
- help_command (char *command, int from_tty)
- {
- help_cmd (command, gdb_stdout);
- }
- static void
- complete_command (char *arg, int from_tty)
- {
- int argpoint;
- char *point, *arg_prefix;
- VEC (char_ptr) *completions;
- dont_repeat ();
- if (arg == NULL)
- arg = "";
- argpoint = strlen (arg);
-
- point = arg + argpoint;
- while (point > arg)
- {
- if (strchr (rl_completer_word_break_characters, point[-1]) != 0)
- break;
- point--;
- }
- arg_prefix = alloca (point - arg + 1);
- memcpy (arg_prefix, arg, point - arg);
- arg_prefix[point - arg] = 0;
- completions = complete_line (point, arg, argpoint);
- if (completions)
- {
- int ix, size = VEC_length (char_ptr, completions);
- char *item, *prev = NULL;
- qsort (VEC_address (char_ptr, completions), size,
- sizeof (char *), compare_strings);
-
- for (ix = 0; VEC_iterate (char_ptr, completions, ix, item); ++ix)
- {
- if (prev == NULL || strcmp (item, prev) != 0)
- {
- printf_unfiltered ("%s%s\n", arg_prefix, item);
- xfree (prev);
- prev = item;
- }
- else
- xfree (item);
- }
- xfree (prev);
- VEC_free (char_ptr, completions);
- }
- }
- int
- is_complete_command (struct cmd_list_element *c)
- {
- return cmd_cfunc_eq (c, complete_command);
- }
- static void
- show_version (char *args, int from_tty)
- {
- print_gdb_version (gdb_stdout);
- printf_filtered ("\n");
- }
- static void
- show_configuration (char *args, int from_tty)
- {
- print_gdb_configuration (gdb_stdout);
- }
- void
- quit_command (char *args, int from_tty)
- {
- if (!quit_confirm ())
- error (_("Not confirmed."));
- query_if_trace_running (from_tty);
- quit_force (args, from_tty);
- }
- static void
- pwd_command (char *args, int from_tty)
- {
- if (args)
- error (_("The \"pwd\" command does not take an argument: %s"), args);
- if (! getcwd (gdb_dirbuf, sizeof (gdb_dirbuf)))
- error (_("Error finding name of working directory: %s"),
- safe_strerror (errno));
- if (strcmp (gdb_dirbuf, current_directory) != 0)
- printf_unfiltered (_("Working directory %s\n (canonically %s).\n"),
- current_directory, gdb_dirbuf);
- else
- printf_unfiltered (_("Working directory %s.\n"), current_directory);
- }
- void
- cd_command (char *dir, int from_tty)
- {
- int len;
-
- int found_real_path;
- char *p;
- struct cleanup *cleanup;
-
- dont_repeat ();
- if (dir == 0)
- dir = "~";
- dir = tilde_expand (dir);
- cleanup = make_cleanup (xfree, dir);
- if (chdir (dir) < 0)
- perror_with_name (dir);
- #ifdef HAVE_DOS_BASED_FILE_SYSTEM
-
- dir = getcwd (gdb_dirbuf, sizeof (gdb_dirbuf));
- #endif
- len = strlen (dir);
- if (IS_DIR_SEPARATOR (dir[len - 1]))
- {
-
- if (!(len == 1)
- #ifdef HAVE_DOS_BASED_FILE_SYSTEM
- && !(len == 3 && dir[1] == ':')
- #endif
- )
- len--;
- }
- dir = savestring (dir, len);
- if (IS_ABSOLUTE_PATH (dir))
- current_directory = dir;
- else
- {
- if (IS_DIR_SEPARATOR (current_directory[strlen (current_directory) - 1]))
- current_directory = concat (current_directory, dir, (char *)NULL);
- else
- current_directory = concat (current_directory, SLASH_STRING,
- dir, (char *)NULL);
- xfree (dir);
- }
-
- found_real_path = 0;
- for (p = current_directory; *p;)
- {
- if (IS_DIR_SEPARATOR (p[0]) && p[1] == '.'
- && (p[2] == 0 || IS_DIR_SEPARATOR (p[2])))
- memmove (p, p + 2, strlen (p + 2) + 1);
- else if (IS_DIR_SEPARATOR (p[0]) && p[1] == '.' && p[2] == '.'
- && (p[3] == 0 || IS_DIR_SEPARATOR (p[3])))
- {
- if (found_real_path)
- {
-
- char *q = p;
- while (q != current_directory && !IS_DIR_SEPARATOR (q[-1]))
- --q;
- if (q == current_directory)
-
- ++p;
- else
- {
- memmove (q - 1, p + 3, strlen (p + 3) + 1);
- p = q - 1;
- }
- }
- else
-
- p += 3;
- }
- else
- {
- found_real_path = 1;
- ++p;
- }
- }
- forget_cached_source_info ();
- if (from_tty)
- pwd_command ((char *) 0, 1);
- do_cleanups (cleanup);
- }
- static void
- show_script_ext_mode (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
- {
- fprintf_filtered (file,
- _("Script filename extension recognition is \"%s\".\n"),
- value);
- }
- int
- find_and_open_script (const char *script_file, int search_path,
- FILE **streamp, char **full_pathp)
- {
- char *file;
- int fd;
- struct cleanup *old_cleanups;
- int search_flags = OPF_TRY_CWD_FIRST | OPF_RETURN_REALPATH;
- file = tilde_expand (script_file);
- old_cleanups = make_cleanup (xfree, file);
- if (search_path)
- search_flags |= OPF_SEARCH_IN_PATH;
-
- fd = openp (source_path, search_flags,
- file, O_RDONLY, full_pathp);
- if (fd == -1)
- {
- int save_errno = errno;
- do_cleanups (old_cleanups);
- errno = save_errno;
- return 0;
- }
- do_cleanups (old_cleanups);
- *streamp = fdopen (fd, FOPEN_RT);
- if (*streamp == NULL)
- {
- int save_errno = errno;
- close (fd);
- if (full_pathp)
- xfree (*full_pathp);
- errno = save_errno;
- return 0;
- }
- return 1;
- }
- static void
- source_script_from_stream (FILE *stream, const char *file)
- {
- if (script_ext_mode != script_ext_off)
- {
- const struct extension_language_defn *extlang
- = get_ext_lang_of_file (file);
- if (extlang != NULL)
- {
- if (ext_lang_present_p (extlang))
- {
- script_sourcer_func *sourcer
- = ext_lang_script_sourcer (extlang);
- gdb_assert (sourcer != NULL);
- sourcer (extlang, stream, file);
- return;
- }
- else if (script_ext_mode == script_ext_soft)
- {
-
- }
- else
- throw_ext_lang_unsupported (extlang);
- }
- }
- script_from_file (stream, file);
- }
- static void
- source_script_with_search (const char *file, int from_tty, int search_path)
- {
- FILE *stream;
- char *full_path;
- struct cleanup *old_cleanups;
- if (file == NULL || *file == 0)
- error (_("source command requires file name of file to source."));
- if (!find_and_open_script (file, search_path, &stream, &full_path))
- {
-
- if (from_tty)
- perror_with_name (file);
- else
- {
- perror_warning_with_name (file);
- return;
- }
- }
- old_cleanups = make_cleanup (xfree, full_path);
- make_cleanup_fclose (stream);
-
- source_script_from_stream (stream, search_path ? full_path : file);
- do_cleanups (old_cleanups);
- }
- void
- source_script (const char *file, int from_tty)
- {
- source_script_with_search (file, from_tty, 0);
- }
- static void
- source_verbose_cleanup (void *old_value)
- {
- source_verbose = *(int *)old_value;
- xfree (old_value);
- }
- static void
- source_command (char *args, int from_tty)
- {
- struct cleanup *old_cleanups;
- char *file = args;
- int *old_source_verbose = xmalloc (sizeof(int));
- int search_path = 0;
- *old_source_verbose = source_verbose;
- old_cleanups = make_cleanup (source_verbose_cleanup,
- old_source_verbose);
-
- if (args)
- {
- while (args[0] != '\0')
- {
-
- args = skip_spaces (args);
- if (args[0] != '-')
- break;
- if (args[1] == 'v' && isspace (args[2]))
- {
- source_verbose = 1;
-
- args = &args[3];
- }
- else if (args[1] == 's' && isspace (args[2]))
- {
- search_path = 1;
-
- args = &args[3];
- }
- else
- break;
- }
- file = skip_spaces (args);
- }
- source_script_with_search (file, from_tty, search_path);
- do_cleanups (old_cleanups);
- }
- static void
- echo_command (char *text, int from_tty)
- {
- const char *p = text;
- int c;
- if (text)
- while ((c = *p++) != '\0')
- {
- if (c == '\\')
- {
-
- if (*p == 0)
- return;
- c = parse_escape (get_current_arch (), &p);
- if (c >= 0)
- printf_filtered ("%c", c);
- }
- else
- printf_filtered ("%c", c);
- }
-
- wrap_here ("");
- gdb_flush (gdb_stdout);
- }
- static void
- shell_escape (char *arg, int from_tty)
- {
- #if defined(CANT_FORK) || \
- (!defined(HAVE_WORKING_VFORK) && !defined(HAVE_WORKING_FORK))
-
- int rc = system (arg ? arg : "");
- if (!arg)
- arg = "inferior shell";
- if (rc == -1)
- {
- fprintf_unfiltered (gdb_stderr, "Cannot execute %s: %s\n", arg,
- safe_strerror (errno));
- gdb_flush (gdb_stderr);
- }
- else if (rc)
- {
- fprintf_unfiltered (gdb_stderr, "%s exited with status %d\n", arg, rc);
- gdb_flush (gdb_stderr);
- }
- #ifdef GLOBAL_CURDIR
-
- chdir (current_directory);
- #endif
- #else
- int status, pid;
- if ((pid = vfork ()) == 0)
- {
- const char *p, *user_shell;
- close_most_fds ();
- if ((user_shell = (char *) getenv ("SHELL")) == NULL)
- user_shell = "/bin/sh";
-
- p = lbasename (user_shell);
- if (!arg)
- execl (user_shell, p, (char *) 0);
- else
- execl (user_shell, p, "-c", arg, (char *) 0);
- fprintf_unfiltered (gdb_stderr, "Cannot execute %s: %s\n", user_shell,
- safe_strerror (errno));
- gdb_flush (gdb_stderr);
- _exit (0177);
- }
- if (pid != -1)
- waitpid (pid, &status, 0);
- else
- error (_("Fork failed"));
- #endif
- }
- static void
- edit_command (char *arg, int from_tty)
- {
- struct symtabs_and_lines sals;
- struct symtab_and_line sal;
- struct symbol *sym;
- char *arg1;
- char *editor;
- char *p;
- const char *fn;
-
- if (arg == 0)
- {
- set_default_source_symtab_and_line ();
- sal = get_current_source_symtab_and_line ();
- }
-
- if (arg == 0)
- {
- if (sal.symtab == 0)
- error (_("No default source file yet."));
- sal.line += get_lines_to_list () / 2;
- }
- else
- {
-
- arg1 = arg;
- sals = decode_line_1 (&arg1, DECODE_LINE_LIST_MODE, 0, 0);
- filter_sals (&sals);
- if (! sals.nelts)
- {
-
- return;
- }
- if (sals.nelts > 1)
- {
- ambiguous_line_spec (&sals);
- xfree (sals.sals);
- return;
- }
- sal = sals.sals[0];
- xfree (sals.sals);
- if (*arg1)
- error (_("Junk at end of line specification."));
-
- if (*arg == '*')
- {
- struct gdbarch *gdbarch;
- if (sal.symtab == 0)
- error (_("No source file for address %s."),
- paddress (get_current_arch (), sal.pc));
- gdbarch = get_objfile_arch (SYMTAB_OBJFILE (sal.symtab));
- sym = find_pc_function (sal.pc);
- if (sym)
- printf_filtered ("%s is in %s (%s:%d).\n",
- paddress (gdbarch, sal.pc),
- SYMBOL_PRINT_NAME (sym),
- symtab_to_filename_for_display (sal.symtab),
- sal.line);
- else
- printf_filtered ("%s is at %s:%d.\n",
- paddress (gdbarch, sal.pc),
- symtab_to_filename_for_display (sal.symtab),
- sal.line);
- }
-
- if (sal.symtab == 0)
- error (_("No line number known for %s."), arg);
- }
- if ((editor = (char *) getenv ("EDITOR")) == NULL)
- editor = "/bin/ex";
- fn = symtab_to_fullname (sal.symtab);
-
- p = xstrprintf ("%s +%d \"%s\"", editor, sal.line, fn);
- shell_escape (p, from_tty);
- xfree (p);
- }
- static void
- list_command (char *arg, int from_tty)
- {
- struct symtabs_and_lines sals, sals_end;
- struct symtab_and_line sal = { 0 };
- struct symtab_and_line sal_end = { 0 };
- struct symtab_and_line cursal = { 0 };
- struct symbol *sym;
- char *arg1;
- int no_end = 1;
- int dummy_end = 0;
- int dummy_beg = 0;
- int linenum_beg = 0;
- char *p;
-
- if (arg == 0 || arg[0] == '+' || arg[0] == '-')
- {
- set_default_source_symtab_and_line ();
- cursal = get_current_source_symtab_and_line ();
-
- if (get_first_line_listed () == 0)
- {
- int first;
- first = max (cursal.line - get_lines_to_list () / 2, 1);
-
- if (arg != NULL && arg[0] == '-'
- && get_lines_to_list () == 1 && first > 1)
- first -= 1;
- print_source_lines (cursal.symtab, first,
- first + get_lines_to_list (), 0);
- return;
- }
- }
-
- if (arg == 0 || strcmp (arg, "+") == 0)
- {
- print_source_lines (cursal.symtab, cursal.line,
- cursal.line + get_lines_to_list (), 0);
- return;
- }
-
- if (strcmp (arg, "-") == 0)
- {
- print_source_lines (cursal.symtab,
- max (get_first_line_listed ()
- - get_lines_to_list (), 1),
- get_first_line_listed (), 0);
- return;
- }
-
- if (!have_full_symbols () && !have_partial_symbols ())
- error (_("No symbol table is loaded. Use the \"file\" command."));
- arg1 = arg;
- if (*arg1 == ',')
- dummy_beg = 1;
- else
- {
- sals = decode_line_1 (&arg1, DECODE_LINE_LIST_MODE, 0, 0);
- filter_sals (&sals);
- if (!sals.nelts)
- return;
- if (sals.nelts > 1)
- {
- ambiguous_line_spec (&sals);
- xfree (sals.sals);
- return;
- }
- sal = sals.sals[0];
- xfree (sals.sals);
- }
-
- for (p = arg; p != arg1 && *p >= '0' && *p <= '9'; p++);
- linenum_beg = (p == arg1);
- while (*arg1 == ' ' || *arg1 == '\t')
- arg1++;
- if (*arg1 == ',')
- {
- no_end = 0;
- arg1++;
- while (*arg1 == ' ' || *arg1 == '\t')
- arg1++;
- if (*arg1 == 0)
- dummy_end = 1;
- else
- {
- if (dummy_beg)
- sals_end = decode_line_1 (&arg1, DECODE_LINE_LIST_MODE, 0, 0);
- else
- sals_end = decode_line_1 (&arg1, DECODE_LINE_LIST_MODE,
- sal.symtab, sal.line);
- filter_sals (&sals_end);
- if (sals_end.nelts == 0)
- return;
- if (sals_end.nelts > 1)
- {
- ambiguous_line_spec (&sals_end);
- xfree (sals_end.sals);
- return;
- }
- sal_end = sals_end.sals[0];
- xfree (sals_end.sals);
- }
- }
- if (*arg1)
- error (_("Junk at end of line specification."));
- if (!no_end && !dummy_beg && !dummy_end
- && sal.symtab != sal_end.symtab)
- error (_("Specified start and end are in different files."));
- if (dummy_beg && dummy_end)
- error (_("Two empty args do not say what lines to list."));
-
- if (*arg == '*')
- {
- struct gdbarch *gdbarch;
- if (sal.symtab == 0)
- error (_("No source file for address %s."),
- paddress (get_current_arch (), sal.pc));
- gdbarch = get_objfile_arch (SYMTAB_OBJFILE (sal.symtab));
- sym = find_pc_function (sal.pc);
- if (sym)
- printf_filtered ("%s is in %s (%s:%d).\n",
- paddress (gdbarch, sal.pc),
- SYMBOL_PRINT_NAME (sym),
- symtab_to_filename_for_display (sal.symtab), sal.line);
- else
- printf_filtered ("%s is at %s:%d.\n",
- paddress (gdbarch, sal.pc),
- symtab_to_filename_for_display (sal.symtab), sal.line);
- }
-
- if (!linenum_beg && sal.symtab == 0)
- error (_("No line number known for %s."), arg);
-
- if (from_tty)
- *arg = 0;
- if (dummy_beg && sal_end.symtab == 0)
- error (_("No default source file yet. Do \"help list\"."));
- if (dummy_beg)
- print_source_lines (sal_end.symtab,
- max (sal_end.line - (get_lines_to_list () - 1), 1),
- sal_end.line + 1, 0);
- else if (sal.symtab == 0)
- error (_("No default source file yet. Do \"help list\"."));
- else if (no_end)
- {
- int first_line = sal.line - get_lines_to_list () / 2;
- if (first_line < 1) first_line = 1;
- print_source_lines (sal.symtab,
- first_line,
- first_line + get_lines_to_list (),
- 0);
- }
- else
- print_source_lines (sal.symtab, sal.line,
- (dummy_end
- ? sal.line + get_lines_to_list ()
- : sal_end.line + 1),
- 0);
- }
- static void
- print_disassembly (struct gdbarch *gdbarch, const char *name,
- CORE_ADDR low, CORE_ADDR high, int flags)
- {
- #if defined(TUI)
- if (!tui_is_window_visible (DISASSEM_WIN))
- #endif
- {
- printf_filtered ("Dump of assembler code ");
- if (name != NULL)
- printf_filtered ("for function %s:\n", name);
- else
- printf_filtered ("from %s to %s:\n",
- paddress (gdbarch, low), paddress (gdbarch, high));
-
- gdb_disassembly (gdbarch, current_uiout, 0, flags, -1, low, high);
- printf_filtered ("End of assembler dump.\n");
- gdb_flush (gdb_stdout);
- }
- #if defined(TUI)
- else
- {
- tui_show_assembly (gdbarch, low);
- }
- #endif
- }
- static void
- disassemble_current_function (int flags)
- {
- struct frame_info *frame;
- struct gdbarch *gdbarch;
- CORE_ADDR low, high, pc;
- const char *name;
- frame = get_selected_frame (_("No frame selected."));
- gdbarch = get_frame_arch (frame);
- pc = get_frame_address_in_block (frame);
- if (find_pc_partial_function (pc, &name, &low, &high) == 0)
- error (_("No function contains program counter for selected frame."));
- #if defined(TUI)
-
- if (tui_active)
- FIXME
- low = tui_get_low_disassembly_address (gdbarch, low, pc);
- #endif
- low += gdbarch_deprecated_function_start_offset (gdbarch);
- print_disassembly (gdbarch, name, low, high, flags);
- }
- static void
- disassemble_command (char *arg, int from_tty)
- {
- struct gdbarch *gdbarch = get_current_arch ();
- CORE_ADDR low, high;
- const char *name;
- CORE_ADDR pc;
- int flags;
- const char *p;
- p = arg;
- name = NULL;
- flags = 0;
- if (p && *p == '/')
- {
- ++p;
- if (*p == '\0')
- error (_("Missing modifier."));
- while (*p && ! isspace (*p))
- {
- switch (*p++)
- {
- case 'm':
- flags |= DISASSEMBLY_SOURCE;
- break;
- case 'r':
- flags |= DISASSEMBLY_RAW_INSN;
- break;
- default:
- error (_("Invalid disassembly modifier."));
- }
- }
- p = skip_spaces_const (p);
- }
- if (! p || ! *p)
- {
- flags |= DISASSEMBLY_OMIT_FNAME;
- disassemble_current_function (flags);
- return;
- }
- pc = value_as_address (parse_to_comma_and_eval (&p));
- if (p[0] == ',')
- ++p;
- if (p[0] == '\0')
- {
-
- if (find_pc_partial_function (pc, &name, &low, &high) == 0)
- error (_("No function contains specified address."));
- #if defined(TUI)
-
- if (tui_active)
- FIXME
- low = tui_get_low_disassembly_address (gdbarch, low, pc);
- #endif
- low += gdbarch_deprecated_function_start_offset (gdbarch);
- flags |= DISASSEMBLY_OMIT_FNAME;
- }
- else
- {
-
- int incl_flag = 0;
- low = pc;
- p = skip_spaces_const (p);
- if (p[0] == '+')
- {
- ++p;
- incl_flag = 1;
- }
- high = parse_and_eval_address (p);
- if (incl_flag)
- high += low;
- }
- print_disassembly (gdbarch, name, low, high, flags);
- }
- static void
- make_command (char *arg, int from_tty)
- {
- char *p;
- if (arg == 0)
- p = "make";
- else
- {
- p = xmalloc (sizeof ("make ") + strlen (arg));
- strcpy (p, "make ");
- strcpy (p + sizeof ("make ") - 1, arg);
- }
- shell_escape (p, from_tty);
- }
- static void
- show_user (char *args, int from_tty)
- {
- struct cmd_list_element *c;
- extern struct cmd_list_element *cmdlist;
- if (args)
- {
- const char *comname = args;
- c = lookup_cmd (&comname, cmdlist, "", 0, 1);
- if (!cli_user_command_p (c))
- error (_("Not a user command."));
- show_user_1 (c, "", args, gdb_stdout);
- }
- else
- {
- for (c = cmdlist; c; c = c->next)
- {
- if (cli_user_command_p (c) || c->prefixlist != NULL)
- show_user_1 (c, "", c->name, gdb_stdout);
- }
- }
- }
- static void
- apropos_command (char *searchstr, int from_tty)
- {
- regex_t pattern;
- int code;
- if (searchstr == NULL)
- error (_("REGEXP string is empty"));
- code = regcomp (&pattern, searchstr, REG_ICASE);
- if (code == 0)
- {
- struct cleanup *cleanups;
- cleanups = make_regfree_cleanup (&pattern);
- apropos_cmd (gdb_stdout, cmdlist, &pattern, "");
- do_cleanups (cleanups);
- }
- else
- {
- char *err = get_regcomp_error (code, &pattern);
- make_cleanup (xfree, err);
- error (_("Error in regular expression: %s"), err);
- }
- }
- static dyn_string_t
- argv_to_dyn_string (char **argv, int n)
- {
- int i;
- dyn_string_t result = dyn_string_new (10);
- gdb_assert (argv != NULL);
- gdb_assert (n >= 0 && n <= countargv (argv));
- for (i = 0; i < n; ++i)
- {
- if (i > 0)
- dyn_string_append_char (result, ' ');
- dyn_string_append_cstr (result, argv[i]);
- }
- return result;
- }
- static int
- valid_command_p (const char *command)
- {
- struct cmd_list_element *c;
- c = lookup_cmd_1 (& command, cmdlist, NULL, 1);
- if (c == NULL || c == (struct cmd_list_element *) -1)
- return FALSE;
-
- while (*command == ' ' || *command == '\t')
- ++command;
- return *command == '\0';
- }
- static void
- alias_command (char *args, int from_tty)
- {
- int i, alias_argc, command_argc;
- int abbrev_flag = 0;
- char *args2, *equals, *alias, *command;
- char **alias_argv, **command_argv;
- dyn_string_t alias_dyn_string, command_dyn_string;
- struct cleanup *cleanup;
- static const char usage[] = N_("Usage: alias [-a] [--] ALIAS = COMMAND");
- if (args == NULL || strchr (args, '=') == NULL)
- error (_(usage));
- args2 = xstrdup (args);
- cleanup = make_cleanup (xfree, args2);
- equals = strchr (args2, '=');
- *equals = '\0';
- alias_argv = gdb_buildargv (args2);
- make_cleanup_freeargv (alias_argv);
- command_argv = gdb_buildargv (equals + 1);
- make_cleanup_freeargv (command_argv);
- for (i = 0; alias_argv[i] != NULL; )
- {
- if (strcmp (alias_argv[i], "-a") == 0)
- {
- ++alias_argv;
- abbrev_flag = 1;
- }
- else if (strcmp (alias_argv[i], "--") == 0)
- {
- ++alias_argv;
- break;
- }
- else
- break;
- }
- if (alias_argv[0] == NULL || command_argv[0] == NULL
- || *alias_argv[0] == '\0' || *command_argv[0] == '\0')
- error (_(usage));
- for (i = 0; alias_argv[i] != NULL; ++i)
- {
- if (! valid_user_defined_cmd_name_p (alias_argv[i]))
- {
- if (i == 0)
- error (_("Invalid command name: %s"), alias_argv[i]);
- else
- error (_("Invalid command element name: %s"), alias_argv[i]);
- }
- }
- alias_argc = countargv (alias_argv);
- command_argc = countargv (command_argv);
-
- command_dyn_string = argv_to_dyn_string (command_argv, command_argc);
- make_cleanup_dyn_string_delete (command_dyn_string);
- command = dyn_string_buf (command_dyn_string);
- if (! valid_command_p (command))
- error (_("Invalid command to alias to: %s"), command);
-
- alias_dyn_string = argv_to_dyn_string (alias_argv, alias_argc);
- make_cleanup_dyn_string_delete (alias_dyn_string);
- alias = dyn_string_buf (alias_dyn_string);
- if (valid_command_p (alias))
- error (_("Alias already exists: %s"), alias);
-
- if (alias_argc == 1)
- {
-
- add_com_alias (xstrdup (alias_argv[0]), command, class_alias,
- abbrev_flag);
- }
- else
- {
- dyn_string_t alias_prefix_dyn_string, command_prefix_dyn_string;
- const char *alias_prefix, *command_prefix;
- struct cmd_list_element *c_alias, *c_command;
- if (alias_argc != command_argc)
- error (_("Mismatched command length between ALIAS and COMMAND."));
-
- alias_prefix_dyn_string =
- argv_to_dyn_string (alias_argv, alias_argc - 1);
- make_cleanup_dyn_string_delete (alias_prefix_dyn_string);
- command_prefix_dyn_string =
- argv_to_dyn_string (alias_argv, command_argc - 1);
- make_cleanup_dyn_string_delete (command_prefix_dyn_string);
- alias_prefix = dyn_string_buf (alias_prefix_dyn_string);
- command_prefix = dyn_string_buf (command_prefix_dyn_string);
- c_command = lookup_cmd_1 (& command_prefix, cmdlist, NULL, 1);
-
- gdb_assert (c_command != NULL
- && c_command != (struct cmd_list_element *) -1);
- gdb_assert (c_command->prefixlist != NULL);
- c_alias = lookup_cmd_1 (& alias_prefix, cmdlist, NULL, 1);
- if (c_alias != c_command)
- error (_("ALIAS and COMMAND prefixes do not match."));
-
- add_alias_cmd (xstrdup (alias_argv[alias_argc - 1]),
- command_argv[command_argc - 1],
- class_alias, abbrev_flag, c_command->prefixlist);
- }
- do_cleanups (cleanup);
- }
- static void
- ambiguous_line_spec (struct symtabs_and_lines *sals)
- {
- int i;
- for (i = 0; i < sals->nelts; ++i)
- printf_filtered (_("file: \"%s\", line number: %d\n"),
- symtab_to_filename_for_display (sals->sals[i].symtab),
- sals->sals[i].line);
- }
- static int
- compare_symtabs (const void *a, const void *b)
- {
- const struct symtab_and_line *sala = a;
- const struct symtab_and_line *salb = b;
- const char *dira = SYMTAB_DIRNAME (sala->symtab);
- const char *dirb = SYMTAB_DIRNAME (salb->symtab);
- int r;
- if (dira == NULL)
- {
- if (dirb != NULL)
- return -1;
- }
- else if (dirb == NULL)
- {
- if (dira != NULL)
- return 1;
- }
- else
- {
- r = filename_cmp (dira, dirb);
- if (r)
- return r;
- }
- r = filename_cmp (sala->symtab->filename, salb->symtab->filename);
- if (r)
- return r;
- if (sala->line < salb->line)
- return -1;
- return sala->line == salb->line ? 0 : 1;
- }
- static void
- filter_sals (struct symtabs_and_lines *sals)
- {
- int i, out, prev;
- out = 0;
- for (i = 0; i < sals->nelts; ++i)
- {
- if (sals->sals[i].pspace == current_program_space
- && sals->sals[i].symtab != NULL)
- {
- sals->sals[out] = sals->sals[i];
- ++out;
- }
- }
- sals->nelts = out;
- qsort (sals->sals, sals->nelts, sizeof (struct symtab_and_line),
- compare_symtabs);
- out = 1;
- prev = 0;
- for (i = 1; i < sals->nelts; ++i)
- {
- if (compare_symtabs (&sals->sals[prev], &sals->sals[i]))
- {
-
- sals->sals[out] = sals->sals[i];
- prev = out;
- ++out;
- }
- }
- if (sals->nelts == 0)
- {
- xfree (sals->sals);
- sals->sals = NULL;
- }
- else
- sals->nelts = out;
- }
- static void
- set_debug (char *arg, int from_tty)
- {
- printf_unfiltered (_("\"set debug\" must be followed by "
- "the name of a debug subcommand.\n"));
- help_list (setdebuglist, "set debug ", all_commands, gdb_stdout);
- }
- static void
- show_debug (char *args, int from_tty)
- {
- cmd_show_list (showdebuglist, from_tty, "");
- }
- void
- init_cmd_lists (void)
- {
- max_user_call_depth = 1024;
- }
- static void
- show_info_verbose (struct ui_file *file, int from_tty,
- struct cmd_list_element *c,
- const char *value)
- {
- if (info_verbose)
- fprintf_filtered (file,
- _("Verbose printing of informational messages is %s.\n"),
- value);
- else
- fprintf_filtered (file, _("Verbosity is %s.\n"), value);
- }
- static void
- show_history_expansion_p (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
- {
- fprintf_filtered (file, _("History expansion on command input is %s.\n"),
- value);
- }
- static void
- show_remote_debug (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
- {
- fprintf_filtered (file, _("Debugging of remote protocol is %s.\n"),
- value);
- }
- static void
- show_remote_timeout (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
- {
- fprintf_filtered (file,
- _("Timeout limit to wait for target to respond is %s.\n"),
- value);
- }
- static void
- show_max_user_call_depth (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
- {
- fprintf_filtered (file,
- _("The max call depth for user-defined commands is %s.\n"),
- value);
- }
- initialize_file_ftype _initialize_cli_cmds;
- void
- _initialize_cli_cmds (void)
- {
- struct cmd_list_element *c;
-
- add_cmd ("internals", class_maintenance, NULL, _("\
- Maintenance commands.\n\
- Some gdb commands are provided just for use by gdb maintainers.\n\
- These commands are subject to frequent change, and may not be as\n\
- well documented as user commands."),
- &cmdlist);
- add_cmd ("obscure", class_obscure, NULL, _("Obscure features."), &cmdlist);
- add_cmd ("aliases", class_alias, NULL,
- _("Aliases of other commands."), &cmdlist);
- add_cmd ("user-defined", class_user, NULL, _("\
- User-defined commands.\n\
- The commands in this class are those defined by the user.\n\
- Use the \"define\" command to define a command."), &cmdlist);
- add_cmd ("support", class_support, NULL, _("Support facilities."), &cmdlist);
- if (!dbx_commands)
- add_cmd ("status", class_info, NULL, _("Status inquiries."), &cmdlist);
- add_cmd ("files", class_files, NULL, _("Specifying and examining files."),
- &cmdlist);
- add_cmd ("breakpoints", class_breakpoint, NULL,
- _("Making program stop at certain points."), &cmdlist);
- add_cmd ("data", class_vars, NULL, _("Examining data."), &cmdlist);
- add_cmd ("stack", class_stack, NULL, _("\
- Examining the stack.\n\
- The stack is made up of stack frames. Gdb assigns numbers to stack frames\n\
- counting from zero for the innermost (currently executing) frame.\n\n\
- At any time gdb identifies one frame as the \"selected\" frame.\n\
- Variable lookups are done with respect to the selected frame.\n\
- When the program being debugged stops, gdb selects the innermost frame.\n\
- The commands below can be used to select other frames by number or address."),
- &cmdlist);
- add_cmd ("running", class_run, NULL, _("Running the program."), &cmdlist);
-
- add_com ("pwd", class_files, pwd_command, _("\
- Print working directory. This is used for your program as well."));
- c = add_cmd ("cd", class_files, cd_command, _("\
- Set working directory to DIR for debugger and program being debugged.\n\
- The change does not take effect for the program being debugged\n\
- until the next time it is started."), &cmdlist);
- set_cmd_completer (c, filename_completer);
- add_com ("echo", class_support, echo_command, _("\
- Print a constant string. Give string as argument.\n\
- C escape sequences may be used in the argument.\n\
- No newline is added at the end of the argument;\n\
- use \"\\n\" if you want a newline to be printed.\n\
- Since leading and trailing whitespace are ignored in command arguments,\n\
- if you want to print some you must use \"\\\" before leading whitespace\n\
- to be printed or after trailing whitespace."));
- add_setshow_enum_cmd ("script-extension", class_support,
- script_ext_enums, &script_ext_mode, _("\
- Set mode for script filename extension recognition."), _("\
- Show mode for script filename extension recognition."), _("\
- off == no filename extension recognition (all sourced files are GDB scripts)\n\
- soft == evaluate script according to filename extension, fallback to GDB script"
- "\n\
- strict == evaluate script according to filename extension, error if not supported"
- ),
- NULL,
- show_script_ext_mode,
- &setlist, &showlist);
- add_com ("quit", class_support, quit_command, _("\
- Exit gdb.\n\
- Usage: quit [EXPR]\n\
- The optional expression EXPR, if present, is evaluated and the result\n\
- used as GDB's exit code. The default is zero."));
- c = add_com ("help", class_support, help_command,
- _("Print list of commands."));
- set_cmd_completer (c, command_completer);
- add_com_alias ("q", "quit", class_support, 1);
- add_com_alias ("h", "help", class_support, 1);
- add_setshow_boolean_cmd ("verbose", class_support, &info_verbose, _("\
- Set verbosity."), _("\
- Show verbosity."), NULL,
- set_verbose,
- show_info_verbose,
- &setlist, &showlist);
- add_prefix_cmd ("history", class_support, set_history,
- _("Generic command for setting command history parameters."),
- &sethistlist, "set history ", 0, &setlist);
- add_prefix_cmd ("history", class_support, show_history,
- _("Generic command for showing command history parameters."),
- &showhistlist, "show history ", 0, &showlist);
- add_setshow_boolean_cmd ("expansion", no_class, &history_expansion_p, _("\
- Set history expansion on command input."), _("\
- Show history expansion on command input."), _("\
- Without an argument, history expansion is enabled."),
- NULL,
- show_history_expansion_p,
- &sethistlist, &showhistlist);
- add_prefix_cmd ("info", class_info, info_command, _("\
- Generic command for showing things about the program being debugged."),
- &infolist, "info ", 0, &cmdlist);
- add_com_alias ("i", "info", class_info, 1);
- add_com_alias ("inf", "info", class_info, 1);
- add_com ("complete", class_obscure, complete_command,
- _("List the completions for the rest of the line as a command."));
- add_prefix_cmd ("show", class_info, show_command, _("\
- Generic command for showing things about the debugger."),
- &showlist, "show ", 0, &cmdlist);
-
- add_info ("set", show_command, _("Show all GDB settings."));
- add_cmd ("commands", no_set_class, show_commands, _("\
- Show the history of commands you typed.\n\
- You can supply a command number to start with, or a `+' to start after\n\
- the previous command number shown."),
- &showlist);
- add_cmd ("version", no_set_class, show_version,
- _("Show what version of GDB this is."), &showlist);
- add_cmd ("configuration", no_set_class, show_configuration,
- _("Show how GDB was configured at build time."), &showlist);
- add_setshow_zinteger_cmd ("remote", no_class, &remote_debug, _("\
- Set debugging of remote protocol."), _("\
- Show debugging of remote protocol."), _("\
- When enabled, each packet sent or received with the remote target\n\
- is displayed."),
- NULL,
- show_remote_debug,
- &setdebuglist, &showdebuglist);
- add_setshow_zuinteger_unlimited_cmd ("remotetimeout", no_class,
- &remote_timeout, _("\
- Set timeout limit to wait for target to respond."), _("\
- Show timeout limit to wait for target to respond."), _("\
- This value is used to set the time limit for gdb to wait for a response\n\
- from the target."),
- NULL,
- show_remote_timeout,
- &setlist, &showlist);
- add_prefix_cmd ("debug", no_class, set_debug,
- _("Generic command for setting gdb debugging flags"),
- &setdebuglist, "set debug ", 0, &setlist);
- add_prefix_cmd ("debug", no_class, show_debug,
- _("Generic command for showing gdb debugging flags"),
- &showdebuglist, "show debug ", 0, &showlist);
- c = add_com ("shell", class_support, shell_escape, _("\
- Execute the rest of the line as a shell command.\n\
- With no arguments, run an inferior shell."));
- set_cmd_completer (c, filename_completer);
- c = add_com ("edit", class_files, edit_command, _("\
- Edit specified file or function.\n\
- With no argument, edits file containing most recent line listed.\n\
- Editing targets can be specified in these ways:\n\
- FILE:LINENUM, to edit at that line in that file,\n\
- FUNCTION, to edit at the beginning of that function,\n\
- FILE:FUNCTION, to distinguish among like-named static functions.\n\
- *ADDRESS, to edit at the line containing that address.\n\
- Uses EDITOR environment variable contents as editor (or ex as default)."));
- c->completer = location_completer;
- add_com ("list", class_files, list_command, _("\
- List specified function or line.\n\
- With no argument, lists ten more lines after or around previous listing.\n\
- \"list -\" lists the ten lines before a previous ten-line listing.\n\
- One argument specifies a line, and ten lines are listed around that line.\n\
- Two arguments with comma between specify starting and ending lines to list.\n\
- Lines can be specified in these ways:\n\
- LINENUM, to list around that line in current file,\n\
- FILE:LINENUM, to list around that line in that file,\n\
- FUNCTION, to list around beginning of that function,\n\
- FILE:FUNCTION, to distinguish among like-named static functions.\n\
- *ADDRESS, to list around the line containing that address.\n\
- With two args if one is empty it stands for ten lines away from \
- the other arg."));
- if (!xdb_commands)
- add_com_alias ("l", "list", class_files, 1);
- else
- add_com_alias ("v", "list", class_files, 1);
- if (dbx_commands)
- add_com_alias ("file", "list", class_files, 1);
- c = add_com ("disassemble", class_vars, disassemble_command, _("\
- Disassemble a specified section of memory.\n\
- Default is the function surrounding the pc of the selected frame.\n\
- With a /m modifier, source lines are included (if available).\n\
- With a /r modifier, raw instructions in hex are included.\n\
- With a single argument, the function surrounding that address is dumped.\n\
- Two arguments (separated by a comma) are taken as a range of memory to dump,\n\
- in the form of \"start,end\", or \"start,+length\".\n\
- \n\
- Note that the address is interpreted as an expression, not as a location\n\
- like in the \"break\" command.\n\
- So, for example, if you want to disassemble function bar in file foo.c\n\
- you must type \"disassemble 'foo.c'::bar\" and not \"disassemble foo.c:bar\"."));
- set_cmd_completer (c, location_completer);
- if (xdb_commands)
- add_com_alias ("va", "disassemble", class_xdb, 0);
- add_com_alias ("!", "shell", class_support, 0);
- c = add_com ("make", class_support, make_command, _("\
- Run the ``make'' program using the rest of the line as arguments."));
- set_cmd_completer (c, filename_completer);
- add_cmd ("user", no_class, show_user, _("\
- Show definitions of non-python/scheme user defined commands.\n\
- Argument is the name of the user defined command.\n\
- With no argument, show definitions of all user defined commands."), &showlist);
- add_com ("apropos", class_support, apropos_command,
- _("Search for commands matching a REGEXP"));
- add_setshow_uinteger_cmd ("max-user-call-depth", no_class,
- &max_user_call_depth, _("\
- Set the max call depth for non-python/scheme user-defined commands."), _("\
- Show the max call depth for non-python/scheme user-defined commands."), NULL,
- NULL,
- show_max_user_call_depth,
- &setlist, &showlist);
- add_setshow_boolean_cmd ("trace-commands", no_class, &trace_commands, _("\
- Set tracing of GDB CLI commands."), _("\
- Show state of GDB CLI command tracing."), _("\
- When 'on', each command is displayed as it is executed."),
- NULL,
- NULL,
- &setlist, &showlist);
- c = add_com ("alias", class_support, alias_command, _("\
- Define a new command that is an alias of an existing command.\n\
- Usage: alias [-a] [--] ALIAS = COMMAND\n\
- ALIAS is the name of the alias command to create.\n\
- COMMAND is the command being aliased to.\n\
- If \"-a\" is specified, the command is an abbreviation,\n\
- and will not appear in help command list output.\n\
- \n\
- Examples:\n\
- Make \"spe\" an alias of \"set print elements\":\n\
- alias spe = set print elements\n\
- Make \"elms\" an alias of \"elements\" in the \"set print\" command:\n\
- alias -a set print elms = set print elements"));
- }
- void
- init_cli_cmds (void)
- {
- struct cmd_list_element *c;
- char *source_help_text;
- source_help_text = xstrprintf (_("\
- Read commands from a file named FILE.\n\
- \n\
- Usage: source [-s] [-v] FILE\n\
- -s: search for the script in the source search path,\n\
- even if FILE contains directories.\n\
- -v: each command in FILE is echoed as it is executed.\n\
- \n\
- Note that the file \"%s\" is read automatically in this way\n\
- when GDB is started."), gdbinit);
- c = add_cmd ("source", class_support, source_command,
- source_help_text, &cmdlist);
- set_cmd_completer (c, filename_completer);
- }