gdb/inferior.c - gdb
Global variables defined
Data types defined
Functions defined
Source code
- #include "defs.h"
- #include "exec.h"
- #include "inferior.h"
- #include "target.h"
- #include "command.h"
- #include "completer.h"
- #include "gdbcmd.h"
- #include "gdbthread.h"
- #include "ui-out.h"
- #include "observer.h"
- #include "gdbcore.h"
- #include "symfile.h"
- #include "environ.h"
- #include "cli/cli-utils.h"
- #include "continuations.h"
- #include "arch-utils.h"
- #include "target-descriptions.h"
- #include "readline/tilde.h"
- void _initialize_inferiors (void);
- DEFINE_REGISTRY (inferior, REGISTRY_ACCESS_FIELD)
- struct inferior *inferior_list = NULL;
- static int highest_inferior_num;
- static int print_inferior_events = 0;
- static struct inferior *current_inferior_ = NULL;
- struct inferior*
- current_inferior (void)
- {
- return current_inferior_;
- }
- void
- set_current_inferior (struct inferior *inf)
- {
-
- gdb_assert (inf != NULL);
- current_inferior_ = inf;
- }
- static void
- restore_inferior (void *arg)
- {
- struct inferior *saved_inferior = arg;
- set_current_inferior (saved_inferior);
- }
- struct cleanup *
- save_current_inferior (void)
- {
- struct cleanup *old_chain = make_cleanup (restore_inferior,
- current_inferior_);
- return old_chain;
- }
- static void
- free_inferior (struct inferior *inf)
- {
- discard_all_inferior_continuations (inf);
- inferior_free_data (inf);
- xfree (inf->args);
- xfree (inf->terminal);
- free_environ (inf->environment);
- target_desc_info_free (inf->tdesc_info);
- xfree (inf->private);
- xfree (inf);
- }
- void
- init_inferior_list (void)
- {
- struct inferior *inf, *infnext;
- highest_inferior_num = 0;
- if (!inferior_list)
- return;
- for (inf = inferior_list; inf; inf = infnext)
- {
- infnext = inf->next;
- free_inferior (inf);
- }
- inferior_list = NULL;
- }
- struct inferior *
- add_inferior_silent (int pid)
- {
- struct inferior *inf;
- inf = xmalloc (sizeof (*inf));
- memset (inf, 0, sizeof (*inf));
- inf->pid = pid;
- inf->control.stop_soon = NO_STOP_QUIETLY;
- inf->num = ++highest_inferior_num;
- inf->next = inferior_list;
- inferior_list = inf;
- inf->environment = make_environ ();
- init_environ (inf->environment);
- inferior_alloc_data (inf);
- observer_notify_inferior_added (inf);
- if (pid != 0)
- inferior_appeared (inf, pid);
- return inf;
- }
- struct inferior *
- add_inferior (int pid)
- {
- struct inferior *inf = add_inferior_silent (pid);
- if (print_inferior_events)
- printf_unfiltered (_("[New inferior %d]\n"), pid);
- return inf;
- }
- struct delete_thread_of_inferior_arg
- {
- int pid;
- int silent;
- };
- static int
- delete_thread_of_inferior (struct thread_info *tp, void *data)
- {
- struct delete_thread_of_inferior_arg *arg = data;
- if (ptid_get_pid (tp->ptid) == arg->pid)
- {
- if (arg->silent)
- delete_thread_silent (tp->ptid);
- else
- delete_thread (tp->ptid);
- }
- return 0;
- }
- void
- delete_inferior_1 (struct inferior *todel, int silent)
- {
- struct inferior *inf, *infprev;
- struct delete_thread_of_inferior_arg arg;
- infprev = NULL;
- for (inf = inferior_list; inf; infprev = inf, inf = inf->next)
- if (inf == todel)
- break;
- if (!inf)
- return;
- arg.pid = inf->pid;
- arg.silent = silent;
- iterate_over_threads (delete_thread_of_inferior, &arg);
- if (infprev)
- infprev->next = inf->next;
- else
- inferior_list = inf->next;
- observer_notify_inferior_removed (inf);
- free_inferior (inf);
- }
- void
- delete_inferior (int pid)
- {
- struct inferior *inf = find_inferior_pid (pid);
- delete_inferior_1 (inf, 0);
- if (print_inferior_events)
- printf_unfiltered (_("[Inferior %d exited]\n"), pid);
- }
- void
- delete_inferior_silent (int pid)
- {
- struct inferior *inf = find_inferior_pid (pid);
- delete_inferior_1 (inf, 1);
- }
- static void
- exit_inferior_1 (struct inferior *inftoex, int silent)
- {
- struct inferior *inf;
- struct delete_thread_of_inferior_arg arg;
- for (inf = inferior_list; inf; inf = inf->next)
- if (inf == inftoex)
- break;
- if (!inf)
- return;
- arg.pid = inf->pid;
- arg.silent = silent;
- iterate_over_threads (delete_thread_of_inferior, &arg);
-
- observer_notify_inferior_exit (inf);
- inf->pid = 0;
- inf->fake_pid_p = 0;
- if (inf->vfork_parent != NULL)
- {
- inf->vfork_parent->vfork_child = NULL;
- inf->vfork_parent = NULL;
- }
- if (inf->vfork_child != NULL)
- {
- inf->vfork_child->vfork_parent = NULL;
- inf->vfork_child = NULL;
- }
- inf->pending_detach = 0;
- }
- void
- exit_inferior (int pid)
- {
- struct inferior *inf = find_inferior_pid (pid);
- exit_inferior_1 (inf, 0);
- if (print_inferior_events)
- printf_unfiltered (_("[Inferior %d exited]\n"), pid);
- }
- void
- exit_inferior_silent (int pid)
- {
- struct inferior *inf = find_inferior_pid (pid);
- exit_inferior_1 (inf, 1);
- }
- void
- exit_inferior_num_silent (int num)
- {
- struct inferior *inf = find_inferior_id (num);
- exit_inferior_1 (inf, 1);
- }
- void
- detach_inferior (int pid)
- {
- struct inferior *inf = find_inferior_pid (pid);
- exit_inferior_1 (inf, 1);
- if (print_inferior_events)
- printf_unfiltered (_("[Inferior %d detached]\n"), pid);
- }
- void
- inferior_appeared (struct inferior *inf, int pid)
- {
- inf->pid = pid;
- inf->has_exit_code = 0;
- inf->exit_code = 0;
- observer_notify_inferior_appeared (inf);
- }
- void
- discard_all_inferiors (void)
- {
- struct inferior *inf;
- for (inf = inferior_list; inf; inf = inf->next)
- {
- if (inf->pid != 0)
- exit_inferior_silent (inf->pid);
- }
- }
- struct inferior *
- find_inferior_id (int num)
- {
- struct inferior *inf;
- for (inf = inferior_list; inf; inf = inf->next)
- if (inf->num == num)
- return inf;
- return NULL;
- }
- struct inferior *
- find_inferior_pid (int pid)
- {
- struct inferior *inf;
-
- gdb_assert (pid != 0);
- for (inf = inferior_list; inf; inf = inf->next)
- if (inf->pid == pid)
- return inf;
- return NULL;
- }
- struct inferior *
- find_inferior_ptid (ptid_t ptid)
- {
- return find_inferior_pid (ptid_get_pid (ptid));
- }
- struct inferior *
- find_inferior_for_program_space (struct program_space *pspace)
- {
- struct inferior *inf = current_inferior ();
- if (inf->pspace == pspace)
- return inf;
- for (inf = inferior_list; inf != NULL; inf = inf->next)
- {
- if (inf->pspace == pspace)
- return inf;
- }
- return NULL;
- }
- struct inferior *
- iterate_over_inferiors (int (*callback) (struct inferior *, void *),
- void *data)
- {
- struct inferior *inf, *infnext;
- for (inf = inferior_list; inf; inf = infnext)
- {
- infnext = inf->next;
- if ((*callback) (inf, data))
- return inf;
- }
- return NULL;
- }
- int
- valid_gdb_inferior_id (int num)
- {
- struct inferior *inf;
- for (inf = inferior_list; inf; inf = inf->next)
- if (inf->num == num)
- return 1;
- return 0;
- }
- int
- pid_to_gdb_inferior_id (int pid)
- {
- struct inferior *inf;
- for (inf = inferior_list; inf; inf = inf->next)
- if (inf->pid == pid)
- return inf->num;
- return 0;
- }
- int
- gdb_inferior_id_to_pid (int num)
- {
- struct inferior *inferior = find_inferior_id (num);
- if (inferior)
- return inferior->pid;
- else
- return -1;
- }
- int
- in_inferior_list (int pid)
- {
- struct inferior *inf;
- for (inf = inferior_list; inf; inf = inf->next)
- if (inf->pid == pid)
- return 1;
- return 0;
- }
- int
- have_inferiors (void)
- {
- struct inferior *inf;
- for (inf = inferior_list; inf; inf = inf->next)
- if (inf->pid != 0)
- return 1;
- return 0;
- }
- int
- have_live_inferiors (void)
- {
- struct inferior *inf;
- for (inf = inferior_list; inf; inf = inf->next)
- if (inf->pid != 0)
- {
- struct thread_info *tp;
- tp = any_thread_of_process (inf->pid);
- if (tp && target_has_execution_1 (tp->ptid))
- break;
- }
- return inf != NULL;
- }
- void
- prune_inferiors (void)
- {
- struct inferior *ss, **ss_link;
- struct inferior *current = current_inferior ();
- ss = inferior_list;
- ss_link = &inferior_list;
- while (ss)
- {
- if (ss == current
- || !ss->removable
- || ss->pid != 0)
- {
- ss_link = &ss->next;
- ss = *ss_link;
- continue;
- }
- *ss_link = ss->next;
- delete_inferior_1 (ss, 1);
- ss = *ss_link;
- }
- prune_program_spaces ();
- }
- int
- number_of_inferiors (void)
- {
- struct inferior *inf;
- int count = 0;
- for (inf = inferior_list; inf != NULL; inf = inf->next)
- count++;
- return count;
- }
- static char *
- inferior_pid_to_str (int pid)
- {
- if (pid != 0)
- return target_pid_to_str (pid_to_ptid (pid));
- else
- return _("<null>");
- }
- static void
- print_inferior (struct ui_out *uiout, char *requested_inferiors)
- {
- struct inferior *inf;
- struct cleanup *old_chain;
- int inf_count = 0;
-
- for (inf = inferior_list; inf; inf = inf->next)
- {
- if (!number_is_in_list (requested_inferiors, inf->num))
- continue;
- ++inf_count;
- }
- if (inf_count == 0)
- {
- ui_out_message (uiout, 0, "No inferiors.\n");
- return;
- }
- old_chain = make_cleanup_ui_out_table_begin_end (uiout, 4, inf_count,
- "inferiors");
- ui_out_table_header (uiout, 1, ui_left, "current", "");
- ui_out_table_header (uiout, 4, ui_left, "number", "Num");
- ui_out_table_header (uiout, 17, ui_left, "target-id", "Description");
- ui_out_table_header (uiout, 17, ui_left, "exec", "Executable");
- ui_out_table_body (uiout);
- for (inf = inferior_list; inf; inf = inf->next)
- {
- struct cleanup *chain2;
- if (!number_is_in_list (requested_inferiors, inf->num))
- continue;
- chain2 = make_cleanup_ui_out_tuple_begin_end (uiout, NULL);
- if (inf == current_inferior ())
- ui_out_field_string (uiout, "current", "*");
- else
- ui_out_field_skip (uiout, "current");
- ui_out_field_int (uiout, "number", inf->num);
- ui_out_field_string (uiout, "target-id",
- inferior_pid_to_str (inf->pid));
- if (inf->pspace->pspace_exec_filename != NULL)
- ui_out_field_string (uiout, "exec", inf->pspace->pspace_exec_filename);
- else
- ui_out_field_skip (uiout, "exec");
-
- if (inf->vfork_parent)
- {
- ui_out_text (uiout, _("\n\tis vfork child of inferior "));
- ui_out_field_int (uiout, "vfork-parent", inf->vfork_parent->num);
- }
- if (inf->vfork_child)
- {
- ui_out_text (uiout, _("\n\tis vfork parent of inferior "));
- ui_out_field_int (uiout, "vfork-child", inf->vfork_child->num);
- }
- ui_out_text (uiout, "\n");
- do_cleanups (chain2);
- }
- do_cleanups (old_chain);
- }
- static void
- detach_inferior_command (char *args, int from_tty)
- {
- int num, pid;
- struct thread_info *tp;
- struct get_number_or_range_state state;
- if (!args || !*args)
- error (_("Requires argument (inferior id(s) to detach)"));
- init_number_or_range (&state, args);
- while (!state.finished)
- {
- num = get_number_or_range (&state);
- if (!valid_gdb_inferior_id (num))
- {
- warning (_("Inferior ID %d not known."), num);
- continue;
- }
- pid = gdb_inferior_id_to_pid (num);
- tp = any_thread_of_process (pid);
- if (!tp)
- {
- warning (_("Inferior ID %d has no threads."), num);
- continue;
- }
- switch_to_thread (tp->ptid);
- detach_command (NULL, from_tty);
- }
- }
- static void
- kill_inferior_command (char *args, int from_tty)
- {
- int num, pid;
- struct thread_info *tp;
- struct get_number_or_range_state state;
- if (!args || !*args)
- error (_("Requires argument (inferior id(s) to kill)"));
- init_number_or_range (&state, args);
- while (!state.finished)
- {
- num = get_number_or_range (&state);
- if (!valid_gdb_inferior_id (num))
- {
- warning (_("Inferior ID %d not known."), num);
- continue;
- }
- pid = gdb_inferior_id_to_pid (num);
- tp = any_thread_of_process (pid);
- if (!tp)
- {
- warning (_("Inferior ID %d has no threads."), num);
- continue;
- }
- switch_to_thread (tp->ptid);
- target_kill ();
- }
- bfd_cache_close_all ();
- }
- static void
- inferior_command (char *args, int from_tty)
- {
- struct inferior *inf;
- int num;
- num = parse_and_eval_long (args);
- inf = find_inferior_id (num);
- if (inf == NULL)
- error (_("Inferior ID %d not known."), num);
- printf_filtered (_("[Switching to inferior %d [%s] (%s)]\n"),
- inf->num,
- inferior_pid_to_str (inf->pid),
- (inf->pspace->pspace_exec_filename != NULL
- ? inf->pspace->pspace_exec_filename
- : _("<noexec>")));
- if (inf->pid != 0)
- {
- if (inf->pid != ptid_get_pid (inferior_ptid))
- {
- struct thread_info *tp;
- tp = any_thread_of_process (inf->pid);
- if (!tp)
- error (_("Inferior has no threads."));
- switch_to_thread (tp->ptid);
- }
- printf_filtered (_("[Switching to thread %d (%s)] "),
- pid_to_thread_id (inferior_ptid),
- target_pid_to_str (inferior_ptid));
- }
- else
- {
- struct inferior *inf;
- inf = find_inferior_id (num);
- set_current_inferior (inf);
- switch_to_thread (null_ptid);
- set_current_program_space (inf->pspace);
- }
- if (inf->pid != 0 && is_running (inferior_ptid))
- ui_out_text (current_uiout, "(running)\n");
- else if (inf->pid != 0)
- {
- ui_out_text (current_uiout, "\n");
- print_stack_frame (get_selected_frame (NULL), 1, SRC_AND_LOC, 1);
- }
- }
- static void
- info_inferiors_command (char *args, int from_tty)
- {
- print_inferior (current_uiout, args);
- }
- static void
- remove_inferior_command (char *args, int from_tty)
- {
- int num;
- struct inferior *inf;
- struct get_number_or_range_state state;
- if (args == NULL || *args == '\0')
- error (_("Requires an argument (inferior id(s) to remove)"));
- init_number_or_range (&state, args);
- while (!state.finished)
- {
- num = get_number_or_range (&state);
- inf = find_inferior_id (num);
- if (inf == NULL)
- {
- warning (_("Inferior ID %d not known."), num);
- continue;
- }
- if (inf == current_inferior ())
- {
- warning (_("Can not remove current symbol inferior %d."), num);
- continue;
- }
- if (inf->pid != 0)
- {
- warning (_("Can not remove active inferior %d."), num);
- continue;
- }
- delete_inferior_1 (inf, 1);
- }
- prune_program_spaces ();
- }
- struct inferior *
- add_inferior_with_spaces (void)
- {
- struct address_space *aspace;
- struct program_space *pspace;
- struct inferior *inf;
- struct gdbarch_info info;
-
- aspace = maybe_new_address_space ();
- pspace = add_program_space (aspace);
- inf = add_inferior (0);
- inf->pspace = pspace;
- inf->aspace = pspace->aspace;
-
- gdbarch_info_init (&info);
- inf->gdbarch = gdbarch_find_by_info (info);
-
- gdb_assert (inf->gdbarch != NULL);
- return inf;
- }
- static void
- add_inferior_command (char *args, int from_tty)
- {
- int i, copies = 1;
- char *exec = NULL;
- char **argv;
- struct cleanup *old_chain = make_cleanup (null_cleanup, NULL);
- if (args)
- {
- argv = gdb_buildargv (args);
- make_cleanup_freeargv (argv);
- for (; *argv != NULL; argv++)
- {
- if (**argv == '-')
- {
- if (strcmp (*argv, "-copies") == 0)
- {
- ++argv;
- if (!*argv)
- error (_("No argument to -copies"));
- copies = parse_and_eval_long (*argv);
- }
- else if (strcmp (*argv, "-exec") == 0)
- {
- ++argv;
- if (!*argv)
- error (_("No argument to -exec"));
- exec = tilde_expand (*argv);
- make_cleanup (xfree, exec);
- }
- }
- else
- error (_("Invalid argument"));
- }
- }
- save_current_space_and_thread ();
- for (i = 0; i < copies; ++i)
- {
- struct inferior *inf = add_inferior_with_spaces ();
- printf_filtered (_("Added inferior %d\n"), inf->num);
- if (exec != NULL)
- {
-
- set_current_program_space (inf->pspace);
- set_current_inferior (inf);
- switch_to_thread (null_ptid);
- exec_file_attach (exec, from_tty);
- symbol_file_add_main (exec, from_tty);
- }
- }
- do_cleanups (old_chain);
- }
- static void
- clone_inferior_command (char *args, int from_tty)
- {
- int i, copies = 1;
- char **argv;
- struct inferior *orginf = NULL;
- struct cleanup *old_chain = make_cleanup (null_cleanup, NULL);
- if (args)
- {
- argv = gdb_buildargv (args);
- make_cleanup_freeargv (argv);
- for (; *argv != NULL; argv++)
- {
- if (**argv == '-')
- {
- if (strcmp (*argv, "-copies") == 0)
- {
- ++argv;
- if (!*argv)
- error (_("No argument to -copies"));
- copies = parse_and_eval_long (*argv);
- if (copies < 0)
- error (_("Invalid copies number"));
- }
- }
- else
- {
- if (orginf == NULL)
- {
- int num;
-
- num = parse_and_eval_long (*argv);
- orginf = find_inferior_id (num);
- if (orginf == NULL)
- error (_("Inferior ID %d not known."), num);
- continue;
- }
- else
- error (_("Invalid argument"));
- }
- }
- }
-
- if (orginf == NULL)
- orginf = current_inferior ();
- save_current_space_and_thread ();
- for (i = 0; i < copies; ++i)
- {
- struct address_space *aspace;
- struct program_space *pspace;
- struct inferior *inf;
-
- aspace = maybe_new_address_space ();
- pspace = add_program_space (aspace);
- inf = add_inferior (0);
- inf->pspace = pspace;
- inf->aspace = pspace->aspace;
- inf->gdbarch = orginf->gdbarch;
-
- if (target_desc_info_from_user_p (inf->tdesc_info))
- copy_inferior_target_desc_info (inf, orginf);
- printf_filtered (_("Added inferior %d.\n"), inf->num);
- set_current_inferior (inf);
- switch_to_thread (null_ptid);
- clone_program_space (pspace, orginf->pspace);
- }
- do_cleanups (old_chain);
- }
- static void
- show_print_inferior_events (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
- {
- fprintf_filtered (file, _("Printing of inferior events is %s.\n"), value);
- }
- void
- initialize_inferiors (void)
- {
- struct cmd_list_element *c = NULL;
-
- current_inferior_ = add_inferior (0);
- current_inferior_->pspace = current_program_space;
- current_inferior_->aspace = current_program_space->aspace;
-
- add_info ("inferiors", info_inferiors_command,
- _("IDs of specified inferiors (all inferiors if no argument)."));
- c = add_com ("add-inferior", no_class, add_inferior_command, _("\
- Add a new inferior.\n\
- Usage: add-inferior [-copies <N>] [-exec <FILENAME>]\n\
- N is the optional number of inferiors to add, default is 1.\n\
- FILENAME is the file name of the executable to use\n\
- as main program."));
- set_cmd_completer (c, filename_completer);
- add_com ("remove-inferiors", no_class, remove_inferior_command, _("\
- Remove inferior ID (or list of IDs).\n\
- Usage: remove-inferiors ID..."));
- add_com ("clone-inferior", no_class, clone_inferior_command, _("\
- Clone inferior ID.\n\
- Usage: clone-inferior [-copies <N>] [ID]\n\
- Add N copies of inferior ID. The new inferior has the same\n\
- executable loaded as the copied inferior. If -copies is not specified,\n\
- adds 1 copy. If ID is not specified, it is the current inferior\n\
- that is cloned."));
- add_cmd ("inferiors", class_run, detach_inferior_command, _("\
- Detach from inferior ID (or list of IDS)."),
- &detachlist);
- add_cmd ("inferiors", class_run, kill_inferior_command, _("\
- Kill inferior ID (or list of IDs)."),
- &killlist);
- add_cmd ("inferior", class_run, inferior_command, _("\
- Use this command to switch between inferiors.\n\
- The new inferior ID must be currently known."),
- &cmdlist);
- add_setshow_boolean_cmd ("inferior-events", no_class,
- &print_inferior_events, _("\
- Set printing of inferior events (e.g., inferior start and exit)."), _("\
- Show printing of inferior events (e.g., inferior start and exit)."), NULL,
- NULL,
- show_print_inferior_events,
- &setprintlist, &showprintlist);
- }