| /* Fork a Unix child process, and set up to debug it, for GDB. |
| |
| Copyright (C) 1990-2024 Free Software Foundation, Inc. |
| |
| Contributed by Cygnus Support. |
| |
| 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 "inferior.h" |
| #include "cli/cli-cmds.h" |
| #include "terminal.h" |
| #include "gdbthread.h" |
| #include "ui.h" |
| #include "gdbsupport/job-control.h" |
| #include "gdbsupport/filestuff.h" |
| #include "nat/fork-inferior.h" |
| #include "gdbsupport/common-inferior.h" |
| |
| /* The exec-wrapper, if any, that will be used when starting the |
| inferior. */ |
| |
| static std::string exec_wrapper; |
| |
| /* See gdbsupport/common-inferior.h. */ |
| |
| const char * |
| get_exec_wrapper () |
| { |
| return !exec_wrapper.empty () ? exec_wrapper.c_str () : nullptr; |
| } |
| |
| /* See nat/fork-inferior.h. */ |
| |
| void |
| gdb_flush_out_err () |
| { |
| gdb_flush (main_ui->m_gdb_stdout); |
| gdb_flush (main_ui->m_gdb_stderr); |
| } |
| |
| /* The ui structure that will be saved on 'prefork_hook' and |
| restored on 'postfork_hook'. */ |
| static struct ui *saved_ui = NULL; |
| |
| /* See nat/fork-inferior.h. */ |
| |
| void |
| prefork_hook (const char *args) |
| { |
| gdb_assert (saved_ui == NULL); |
| /* Retain a copy of our UI, since the child will replace this value |
| and if we're vforked, we have to restore it. */ |
| saved_ui = current_ui; |
| |
| /* Tell the terminal handling subsystem what tty we plan to run on; |
| it will just record the information for later. */ |
| new_tty_prefork (current_inferior ()->tty ()); |
| } |
| |
| /* See nat/fork-inferior.h. */ |
| |
| void |
| postfork_hook (pid_t pid) |
| { |
| inferior *inf = current_inferior (); |
| |
| inferior_appeared (inf, pid); |
| |
| gdb_assert (saved_ui != NULL); |
| current_ui = saved_ui; |
| saved_ui = NULL; |
| |
| new_tty_postfork (); |
| } |
| |
| /* See nat/fork-inferior.h. */ |
| |
| void |
| postfork_child_hook () |
| { |
| /* This is set to the result of setpgrp, which if vforked, will be |
| visible to you in the parent process. It's only used by humans |
| for debugging. */ |
| static int debug_setpgrp = 657473; |
| |
| /* Make sure we switch to main_ui here in order to be able to |
| use the gdb_printf/warning/error functions. */ |
| current_ui = main_ui; |
| |
| /* Create a new session for the inferior process, if necessary. |
| It will also place the inferior in a separate process group. */ |
| if (create_tty_session () <= 0) |
| { |
| /* No session was created, but we still want to run the inferior |
| in a separate process group. */ |
| debug_setpgrp = gdb_setpgid (); |
| if (debug_setpgrp == -1) |
| perror (_("setpgrp failed in child")); |
| } |
| |
| /* Ask the tty subsystem to switch to the one we specified |
| earlier (or to share the current terminal, if none was |
| specified). */ |
| new_tty (); |
| } |
| |
| /* See inferior.h. */ |
| |
| ptid_t |
| gdb_startup_inferior (pid_t pid, int num_traps) |
| { |
| inferior *inf = current_inferior (); |
| process_stratum_target *proc_target = inf->process_target (); |
| |
| scoped_restore save_starting_up |
| = make_scoped_restore (&inf->starting_up, true); |
| |
| ptid_t ptid = startup_inferior (proc_target, pid, num_traps, NULL, NULL); |
| |
| /* Mark all threads non-executing. */ |
| set_executing (proc_target, ptid, false); |
| |
| return ptid; |
| } |
| |
| /* Implement the "unset exec-wrapper" command. */ |
| |
| static void |
| unset_exec_wrapper_command (const char *args, int from_tty) |
| { |
| exec_wrapper.clear (); |
| } |
| |
| static void |
| show_startup_with_shell (struct ui_file *file, int from_tty, |
| struct cmd_list_element *c, const char *value) |
| { |
| gdb_printf (file, |
| _("Use of shell to start subprocesses is %s.\n"), |
| value); |
| } |
| |
| void _initialize_fork_child (); |
| void |
| _initialize_fork_child () |
| { |
| add_setshow_filename_cmd ("exec-wrapper", class_run, &exec_wrapper, _("\ |
| Set a wrapper for running programs.\n\ |
| The wrapper prepares the system and environment for the new program."), |
| _("\ |
| Show the wrapper for running programs."), NULL, |
| NULL, NULL, |
| &setlist, &showlist); |
| |
| add_cmd ("exec-wrapper", class_run, unset_exec_wrapper_command, |
| _("Disable use of an execution wrapper."), |
| &unsetlist); |
| |
| add_setshow_boolean_cmd ("startup-with-shell", class_support, |
| &startup_with_shell, _("\ |
| Set use of shell to start subprocesses. The default is on."), _("\ |
| Show use of shell to start subprocesses."), NULL, |
| NULL, |
| show_startup_with_shell, |
| &setlist, &showlist); |
| } |