mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-24 12:35:55 +08:00
f9582a22db
When running test-case gdb.base/vfork-follow-parent.exp on powerpc64 (likewise on s390x), I run into: ... (gdb) PASS: gdb.base/vfork-follow-parent.exp: \ exec_file=vfork-follow-parent-exit: target-non-stop=on: non-stop=off: \ resolution_method=schedule-multiple: print unblock_parent = 1 continue^M Continuing.^M Reading symbols from vfork-follow-parent-exit...^M ^M ^M Fatal signal: Segmentation fault^M ----- Backtrace -----^M 0x1027d3e7 gdb_internal_backtrace_1^M src/gdb/bt-utils.c:122^M 0x1027d54f _Z22gdb_internal_backtracev^M src/gdb/bt-utils.c:168^M 0x1057643f handle_fatal_signal^M src/gdb/event-top.c:889^M 0x10576677 handle_sigsegv^M src/gdb/event-top.c:962^M 0x3fffa7610477 ???^M 0x103f2144 for_each_block^M src/gdb/dcache.c:199^M 0x103f235b _Z17dcache_invalidateP13dcache_struct^M src/gdb/dcache.c:251^M 0x10bde8c7 _Z24target_dcache_invalidatev^M src/gdb/target-dcache.c:50^M ... or similar. The root cause for the segmentation fault is that linux_is_uclinux gives an incorrect result: it should always return false, given that we're running on a regular linux system, but instead it returns first true, then false. In more detail, the segmentation fault happens as follows: - a program space with an address space is created - a second program space is about to be created. maybe_new_address_space is called, and because linux_is_uclinux returns true, maybe_new_address_space returns false, and no new address space is created - a second program space with the same address space is created - a program space is deleted. Because linux_is_uclinux now returns false, gdbarch_has_shared_address_space (current_inferior ()->arch ()) returns false, and the address space is deleted - when gdb uses the address space of the remaining program space, we run into the segfault, because the address space is deleted. Hardcoding linux_is_uclinux to false makes the test-case pass. We leave addressing the root cause for the following commit in this series. For now, prevent the segmentation fault by making the address space a refcounted object. This was already suggested here [1]: ... A better solution might be to have the address spaces be reference counted ... Tested on top of trunk on x86_64-linux and ppc64le-linux. Tested on top of gdb-14-branch on ppc64-linux. Co-Authored-By: Simon Marchi <simon.marchi@polymtl.ca> PR gdb/30547 Bug: https://sourceware.org/bugzilla/show_bug.cgi?id=30547 [1] https://sourceware.org/pipermail/gdb-patches/2023-October/202928.html
873 lines
28 KiB
C++
873 lines
28 KiB
C++
/* Variables that describe the inferior process running under GDB:
|
||
Where it is, why it stopped, and how to step it.
|
||
|
||
Copyright (C) 1986-2023 Free Software Foundation, Inc.
|
||
|
||
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/>. */
|
||
|
||
#if !defined (INFERIOR_H)
|
||
#define INFERIOR_H 1
|
||
|
||
#include <exception>
|
||
#include <list>
|
||
|
||
struct target_waitstatus;
|
||
class frame_info_ptr;
|
||
struct ui_file;
|
||
struct type;
|
||
struct gdbarch;
|
||
struct regcache;
|
||
struct ui_out;
|
||
struct terminal_info;
|
||
struct target_desc_info;
|
||
struct inferior;
|
||
struct thread_info;
|
||
|
||
/* For bpstat. */
|
||
#include "breakpoint.h"
|
||
|
||
/* For enum gdb_signal. */
|
||
#include "target.h"
|
||
|
||
/* For struct frame_id. */
|
||
#include "frame.h"
|
||
|
||
/* For gdb_environ. */
|
||
#include "gdbsupport/environ.h"
|
||
|
||
#include "progspace.h"
|
||
#include "registry.h"
|
||
|
||
#include "symfile-add-flags.h"
|
||
#include "gdbsupport/refcounted-object.h"
|
||
#include "gdbsupport/forward-scope-exit.h"
|
||
#include "gdbsupport/gdb_unique_ptr.h"
|
||
#include "gdbsupport/intrusive_list.h"
|
||
|
||
#include "gdbsupport/common-inferior.h"
|
||
#include "gdbthread.h"
|
||
|
||
#include "process-stratum-target.h"
|
||
#include "displaced-stepping.h"
|
||
|
||
#include <unordered_map>
|
||
|
||
struct infcall_suspend_state;
|
||
struct infcall_control_state;
|
||
|
||
extern void restore_infcall_suspend_state (struct infcall_suspend_state *);
|
||
extern void restore_infcall_control_state (struct infcall_control_state *);
|
||
|
||
/* A deleter for infcall_suspend_state that calls
|
||
restore_infcall_suspend_state. */
|
||
struct infcall_suspend_state_deleter
|
||
{
|
||
void operator() (struct infcall_suspend_state *state) const
|
||
{
|
||
try
|
||
{
|
||
restore_infcall_suspend_state (state);
|
||
}
|
||
catch (const gdb_exception_error &e)
|
||
{
|
||
/* If we are restoring the inferior state due to an exception,
|
||
some error message will be printed. So, only warn the user
|
||
when we cannot restore during normal execution. */
|
||
bool unwinding;
|
||
#if __cpp_lib_uncaught_exceptions
|
||
unwinding = std::uncaught_exceptions () > 0;
|
||
#else
|
||
unwinding = std::uncaught_exception ();
|
||
#endif
|
||
if (!unwinding)
|
||
warning (_("Failed to restore inferior state: %s"), e.what ());
|
||
}
|
||
}
|
||
};
|
||
|
||
/* A unique_ptr specialization for infcall_suspend_state. */
|
||
typedef std::unique_ptr<infcall_suspend_state, infcall_suspend_state_deleter>
|
||
infcall_suspend_state_up;
|
||
|
||
extern infcall_suspend_state_up save_infcall_suspend_state ();
|
||
|
||
/* A deleter for infcall_control_state that calls
|
||
restore_infcall_control_state. */
|
||
struct infcall_control_state_deleter
|
||
{
|
||
void operator() (struct infcall_control_state *state) const
|
||
{
|
||
restore_infcall_control_state (state);
|
||
}
|
||
};
|
||
|
||
/* A unique_ptr specialization for infcall_control_state. */
|
||
typedef std::unique_ptr<infcall_control_state, infcall_control_state_deleter>
|
||
infcall_control_state_up;
|
||
|
||
extern infcall_control_state_up save_infcall_control_state ();
|
||
|
||
extern void discard_infcall_suspend_state (struct infcall_suspend_state *);
|
||
extern void discard_infcall_control_state (struct infcall_control_state *);
|
||
|
||
extern readonly_detached_regcache *
|
||
get_infcall_suspend_state_regcache (struct infcall_suspend_state *);
|
||
|
||
extern void set_sigint_trap (void);
|
||
|
||
extern void clear_sigint_trap (void);
|
||
|
||
/* Collected pid, tid, etc. of the debugged inferior. When there's
|
||
no inferior, inferior_ptid.pid () will be 0. */
|
||
|
||
extern ptid_t inferior_ptid;
|
||
|
||
extern void generic_mourn_inferior (void);
|
||
|
||
extern CORE_ADDR unsigned_pointer_to_address (struct gdbarch *gdbarch,
|
||
struct type *type,
|
||
const gdb_byte *buf);
|
||
extern void unsigned_address_to_pointer (struct gdbarch *gdbarch,
|
||
struct type *type, gdb_byte *buf,
|
||
CORE_ADDR addr);
|
||
extern CORE_ADDR signed_pointer_to_address (struct gdbarch *gdbarch,
|
||
struct type *type,
|
||
const gdb_byte *buf);
|
||
extern void address_to_signed_pointer (struct gdbarch *gdbarch,
|
||
struct type *type, gdb_byte *buf,
|
||
CORE_ADDR addr);
|
||
|
||
extern void reopen_exec_file (void);
|
||
|
||
/* From misc files */
|
||
|
||
extern void default_print_registers_info (struct gdbarch *gdbarch,
|
||
struct ui_file *file,
|
||
frame_info_ptr frame,
|
||
int regnum, int all);
|
||
|
||
/* Default implementation of gdbarch_print_float_info. Print
|
||
the values of all floating point registers. */
|
||
|
||
extern void default_print_float_info (struct gdbarch *gdbarch,
|
||
struct ui_file *file,
|
||
frame_info_ptr frame,
|
||
const char *args);
|
||
|
||
/* Try to determine whether TTY is GDB's input terminal. Returns
|
||
TRIBOOL_UNKNOWN if we can't tell. */
|
||
|
||
extern tribool is_gdb_terminal (const char *tty);
|
||
|
||
/* Helper for sharing_input_terminal. Try to determine whether pid
|
||
PID is using the same TTY for input as GDB is. Returns
|
||
TRIBOOL_UNKNOWN if we can't tell. */
|
||
|
||
extern tribool sharing_input_terminal (int pid);
|
||
|
||
/* The type of the function that is called when SIGINT is handled. */
|
||
|
||
typedef void c_c_handler_ftype (int);
|
||
|
||
/* Install a new SIGINT handler in a host-dependent way. The previous
|
||
handler is returned. It is fine to pass SIG_IGN for FN, but not
|
||
SIG_DFL. */
|
||
|
||
extern c_c_handler_ftype *install_sigint_handler (c_c_handler_ftype *fn);
|
||
|
||
extern void child_terminal_info (struct target_ops *self, const char *, int);
|
||
|
||
extern void child_terminal_ours (struct target_ops *self);
|
||
|
||
extern void child_terminal_ours_for_output (struct target_ops *self);
|
||
|
||
extern void child_terminal_inferior (struct target_ops *self);
|
||
|
||
extern void child_terminal_save_inferior (struct target_ops *self);
|
||
|
||
extern void child_terminal_init (struct target_ops *self);
|
||
|
||
extern void child_pass_ctrlc (struct target_ops *self);
|
||
|
||
extern void child_interrupt (struct target_ops *self);
|
||
|
||
/* From fork-child.c */
|
||
|
||
/* Helper function to call STARTUP_INFERIOR with PID and NUM_TRAPS.
|
||
This function already calls set_executing. Return the ptid_t from
|
||
STARTUP_INFERIOR. */
|
||
extern ptid_t gdb_startup_inferior (pid_t pid, int num_traps);
|
||
|
||
/* From infcmd.c */
|
||
|
||
/* Initial inferior setup. Determines the exec file is not yet known,
|
||
takes any necessary post-attaching actions, fetches the target
|
||
description and syncs the shared library list. */
|
||
|
||
extern void setup_inferior (int from_tty);
|
||
|
||
extern void post_create_inferior (int from_tty);
|
||
|
||
extern void attach_command (const char *, int);
|
||
|
||
extern void registers_info (const char *, int);
|
||
|
||
extern void continue_1 (int all_threads);
|
||
|
||
extern void interrupt_target_1 (bool all_threads);
|
||
|
||
using delete_longjmp_breakpoint_cleanup
|
||
= FORWARD_SCOPE_EXIT (delete_longjmp_breakpoint);
|
||
|
||
extern void detach_command (const char *, int);
|
||
|
||
extern void notice_new_inferior (struct thread_info *, bool, int);
|
||
|
||
/* Return the value of the result of a function at the end of a 'finish'
|
||
command/BP. If the result's value cannot be retrieved, return NULL.
|
||
|
||
FUNC_SYMBOL is the symbol of the function being returned from. FUNCTION is
|
||
a value containing the address of the function. */
|
||
|
||
extern struct value *get_return_value (struct symbol *func_symbol,
|
||
struct value *function);
|
||
|
||
/* Prepare for execution command. TARGET is the target that will run
|
||
the command. BACKGROUND determines whether this is a foreground
|
||
(synchronous) or background (asynchronous) command. */
|
||
|
||
extern void prepare_execution_command (struct target_ops *target,
|
||
int background);
|
||
|
||
/* Nonzero if stopped due to completion of a stack dummy routine. */
|
||
|
||
extern enum stop_stack_kind stop_stack_dummy;
|
||
|
||
/* Nonzero if program stopped due to a random (unexpected) signal in
|
||
inferior process. */
|
||
|
||
extern int stopped_by_random_signal;
|
||
|
||
/* Print notices on inferior events (attach, detach, etc.), set with
|
||
`set print inferior-events'. */
|
||
extern bool print_inferior_events;
|
||
|
||
/* Anything but NO_STOP_QUIETLY means we expect a trap and the caller
|
||
will handle it themselves. STOP_QUIETLY is used when running in
|
||
the shell before the child program has been exec'd and when running
|
||
through shared library loading. STOP_QUIETLY_REMOTE is used when
|
||
setting up a remote connection; it is like STOP_QUIETLY_NO_SIGSTOP
|
||
except that there is no need to hide a signal. */
|
||
|
||
/* STOP_QUIETLY_NO_SIGSTOP is used to handle a tricky situation with attach.
|
||
When doing an attach, the kernel stops the debuggee with a SIGSTOP.
|
||
On newer GNU/Linux kernels (>= 2.5.61) the handling of SIGSTOP for
|
||
a ptraced process has changed. Earlier versions of the kernel
|
||
would ignore these SIGSTOPs, while now SIGSTOP is treated like any
|
||
other signal, i.e. it is not muffled.
|
||
|
||
If the gdb user does a 'continue' after the 'attach', gdb passes
|
||
the global variable stop_signal (which stores the signal from the
|
||
attach, SIGSTOP) to the ptrace(PTRACE_CONT,...) call. This is
|
||
problematic, because the kernel doesn't ignore such SIGSTOP
|
||
now. I.e. it is reported back to gdb, which in turn presents it
|
||
back to the user.
|
||
|
||
To avoid the problem, we use STOP_QUIETLY_NO_SIGSTOP, which allows
|
||
gdb to clear the value of stop_signal after the attach, so that it
|
||
is not passed back down to the kernel. */
|
||
|
||
enum stop_kind
|
||
{
|
||
NO_STOP_QUIETLY = 0,
|
||
STOP_QUIETLY,
|
||
STOP_QUIETLY_REMOTE,
|
||
STOP_QUIETLY_NO_SIGSTOP
|
||
};
|
||
|
||
|
||
|
||
/* Base class for target-specific inferior data. */
|
||
|
||
struct private_inferior
|
||
{
|
||
virtual ~private_inferior () = 0;
|
||
};
|
||
|
||
/* Inferior process specific part of `struct infcall_control_state'.
|
||
|
||
Inferior thread counterpart is `struct thread_control_state'. */
|
||
|
||
struct inferior_control_state
|
||
{
|
||
inferior_control_state ()
|
||
: stop_soon (NO_STOP_QUIETLY)
|
||
{
|
||
}
|
||
|
||
explicit inferior_control_state (enum stop_kind when)
|
||
: stop_soon (when)
|
||
{
|
||
}
|
||
|
||
/* See the definition of stop_kind above. */
|
||
enum stop_kind stop_soon;
|
||
};
|
||
|
||
/* Return a pointer to the current inferior. */
|
||
extern inferior *current_inferior ();
|
||
|
||
extern void set_current_inferior (inferior *);
|
||
|
||
/* Switch inferior (and program space) to INF, and switch to no thread
|
||
selected. */
|
||
extern void switch_to_inferior_no_thread (inferior *inf);
|
||
|
||
/* Ensure INF is the current inferior.
|
||
|
||
If the current inferior was changed, return an RAII object that will
|
||
restore the original current context. */
|
||
extern std::optional<scoped_restore_current_thread> maybe_switch_inferior
|
||
(inferior *inf);
|
||
|
||
/* Info about an inferior's target description. There's one of these
|
||
for each inferior. */
|
||
|
||
struct target_desc_info
|
||
{
|
||
/* Returns true if this target description information has been supplied by
|
||
the user. */
|
||
bool from_user_p ()
|
||
{ return !this->filename.empty (); }
|
||
|
||
/* A flag indicating that a description has already been fetched
|
||
from the target, so it should not be queried again. */
|
||
bool fetched = false;
|
||
|
||
/* The description fetched from the target, or NULL if the target
|
||
did not supply any description. Only valid when
|
||
FETCHED is set. Only the description initialization
|
||
code should access this; normally, the description should be
|
||
accessed through the gdbarch object. */
|
||
const struct target_desc *tdesc = nullptr;
|
||
|
||
/* If not empty, the filename to read a target description from, as set by
|
||
"set tdesc filename ...".
|
||
|
||
If empty, there is not filename specified by the user. */
|
||
std::string filename;
|
||
};
|
||
|
||
/* GDB represents the state of each program execution with an object
|
||
called an inferior. An inferior typically corresponds to a process
|
||
but is more general and applies also to targets that do not have a
|
||
notion of processes. Each run of an executable creates a new
|
||
inferior, as does each attachment to an existing process.
|
||
Inferiors have unique internal identifiers that are different from
|
||
target process ids. Each inferior may in turn have multiple
|
||
threads running in it.
|
||
|
||
Inferiors are intrusively refcounted objects. Unlike thread
|
||
objects, being the user-selected inferior is considered a strong
|
||
reference and is thus accounted for in the inferior object's
|
||
refcount (see set_current_inferior). When GDB needs to remember
|
||
the selected inferior to later restore it, GDB temporarily bumps
|
||
the inferior object's refcount, to prevent something deleting the
|
||
inferior object before reverting back (e.g., due to a
|
||
"remove-inferiors" command (see
|
||
scoped_restore_current_inferior). All other inferior
|
||
references are considered weak references. Inferiors are always
|
||
listed exactly once in the inferior list, so placing an inferior in
|
||
the inferior list is an implicit, not counted strong reference. */
|
||
|
||
class inferior : public refcounted_object,
|
||
public intrusive_list_node<inferior>
|
||
{
|
||
public:
|
||
explicit inferior (int pid);
|
||
~inferior ();
|
||
|
||
/* Returns true if we can delete this inferior. */
|
||
bool deletable () const { return refcount () == 0; }
|
||
|
||
/* Push T in this inferior's target stack. */
|
||
void push_target (struct target_ops *t)
|
||
{ m_target_stack.push (t); }
|
||
|
||
/* An overload that deletes the target on failure. */
|
||
void push_target (target_ops_up &&t)
|
||
{
|
||
m_target_stack.push (t.get ());
|
||
t.release ();
|
||
}
|
||
|
||
/* Unpush T from this inferior's target stack. */
|
||
int unpush_target (struct target_ops *t);
|
||
|
||
/* Returns true if T is pushed in this inferior's target stack. */
|
||
bool target_is_pushed (const target_ops *t) const
|
||
{ return m_target_stack.is_pushed (t); }
|
||
|
||
/* Find the target beneath T in this inferior's target stack. */
|
||
target_ops *find_target_beneath (const target_ops *t)
|
||
{ return m_target_stack.find_beneath (t); }
|
||
|
||
/* Return the target at the top of this inferior's target stack. */
|
||
target_ops *top_target ()
|
||
{ return m_target_stack.top (); }
|
||
|
||
/* Unpush all targets except the dummy target from m_target_stack. As
|
||
targets are removed from m_target_stack their reference count is
|
||
decremented, which may cause a target to close. */
|
||
void pop_all_targets ()
|
||
{ pop_all_targets_above (dummy_stratum); }
|
||
|
||
/* Unpush all targets above STRATUM from m_target_stack. As targets are
|
||
removed from m_target_stack their reference count is decremented,
|
||
which may cause a target to close. */
|
||
void pop_all_targets_above (enum strata stratum);
|
||
|
||
/* Unpush all targets at and above STRATUM from m_target_stack. As
|
||
targets are removed from m_target_stack their reference count is
|
||
decremented, which may cause a target to close. */
|
||
void pop_all_targets_at_and_above (enum strata stratum);
|
||
|
||
/* Return the target at process_stratum level in this inferior's
|
||
target stack. */
|
||
struct process_stratum_target *process_target ()
|
||
{ return (process_stratum_target *) m_target_stack.at (process_stratum); }
|
||
|
||
/* Return the target at STRATUM in this inferior's target stack. */
|
||
target_ops *target_at (enum strata stratum)
|
||
{ return m_target_stack.at (stratum); }
|
||
|
||
bool has_execution ()
|
||
{ return target_has_execution (this); }
|
||
|
||
/* This inferior's thread list, sorted by creation order. */
|
||
intrusive_list<thread_info> thread_list;
|
||
|
||
/* A map of ptid_t to thread_info*, for average O(1) ptid_t lookup.
|
||
Exited threads do not appear in the map. */
|
||
std::unordered_map<ptid_t, thread_info *> ptid_thread_map;
|
||
|
||
/* Returns a range adapter covering the inferior's threads,
|
||
including exited threads. Used like this:
|
||
|
||
for (thread_info *thr : inf->threads ())
|
||
{ .... }
|
||
*/
|
||
inf_threads_range threads ()
|
||
{ return inf_threads_range (this->thread_list.begin ()); }
|
||
|
||
/* Returns a range adapter covering the inferior's non-exited
|
||
threads. Used like this:
|
||
|
||
for (thread_info *thr : inf->non_exited_threads ())
|
||
{ .... }
|
||
*/
|
||
inf_non_exited_threads_range non_exited_threads ()
|
||
{ return inf_non_exited_threads_range (this->thread_list.begin ()); }
|
||
|
||
/* Like inferior::threads(), but returns a range adapter that can be
|
||
used with range-for, safely. I.e., it is safe to delete the
|
||
currently-iterated thread, like this:
|
||
|
||
for (thread_info *t : inf->threads_safe ())
|
||
if (some_condition ())
|
||
delete f;
|
||
*/
|
||
inline safe_inf_threads_range threads_safe ()
|
||
{ return safe_inf_threads_range (this->thread_list.begin ()); }
|
||
|
||
/* Find (non-exited) thread PTID of this inferior. */
|
||
thread_info *find_thread (ptid_t ptid);
|
||
|
||
/* Delete all threads in the thread list, silently. */
|
||
void clear_thread_list ();
|
||
|
||
/* Continuations-related methods. A continuation is an std::function
|
||
to be called to finish the execution of a command when running
|
||
GDB asynchronously. A continuation is executed after any thread
|
||
of this inferior stops. Continuations are used by the attach
|
||
command and the remote target when a new inferior is detected. */
|
||
void add_continuation (std::function<void ()> &&cont);
|
||
void do_all_continuations ();
|
||
|
||
/* Set/get file name for default use for standard in/out in the inferior.
|
||
|
||
On Unix systems, we try to make TERMINAL_NAME the inferior's controlling
|
||
terminal.
|
||
|
||
If TERMINAL_NAME is the empty string, then the inferior inherits GDB's
|
||
terminal (or GDBserver's if spawning a remote process). */
|
||
void set_tty (std::string terminal_name);
|
||
const std::string &tty ();
|
||
|
||
/* Set the argument string to use when running this inferior.
|
||
|
||
An empty string can be used to represent "no arguments". */
|
||
void set_args (std::string args)
|
||
{
|
||
m_args = std::move (args);
|
||
};
|
||
|
||
/* Set the argument string from some strings. */
|
||
void set_args (gdb::array_view<char * const> args);
|
||
|
||
/* Get the argument string to use when running this inferior.
|
||
|
||
No arguments is represented by an empty string. */
|
||
const std::string &args () const
|
||
{
|
||
return m_args;
|
||
}
|
||
|
||
/* Set the inferior current working directory.
|
||
|
||
If CWD is empty, unset the directory. */
|
||
void set_cwd (std::string cwd)
|
||
{
|
||
m_cwd = std::move (cwd);
|
||
}
|
||
|
||
/* Get the inferior current working directory.
|
||
|
||
Return an empty string if the current working directory is not
|
||
specified. */
|
||
const std::string &cwd () const
|
||
{
|
||
return m_cwd;
|
||
}
|
||
|
||
/* Set this inferior's arch. */
|
||
void set_arch (gdbarch *arch);
|
||
|
||
/* Get this inferior's arch. */
|
||
gdbarch *arch ()
|
||
{ return m_gdbarch; }
|
||
|
||
/* Convenient handle (GDB inferior id). Unique across all
|
||
inferiors. */
|
||
int num = 0;
|
||
|
||
/* Actual target inferior id, usually, a process id. This matches
|
||
the ptid_t.pid member of threads of this inferior. */
|
||
int pid = 0;
|
||
/* True if the PID was actually faked by GDB. */
|
||
bool fake_pid_p = false;
|
||
|
||
/* The highest thread number this inferior ever had. */
|
||
int highest_thread_num = 0;
|
||
|
||
/* State of GDB control of inferior process execution.
|
||
See `struct inferior_control_state'. */
|
||
inferior_control_state control;
|
||
|
||
/* True if this was an auto-created inferior, e.g. created from
|
||
following a fork; false, if this inferior was manually added by
|
||
the user, and we should not attempt to prune it
|
||
automatically. */
|
||
bool removable = false;
|
||
|
||
/* The address space bound to this inferior. */
|
||
address_space_ref_ptr aspace;
|
||
|
||
/* The program space bound to this inferior. */
|
||
struct program_space *pspace = NULL;
|
||
|
||
/* The terminal state as set by the last target_terminal::terminal_*
|
||
call. */
|
||
target_terminal_state terminal_state = target_terminal_state::is_ours;
|
||
|
||
/* Environment to use for running inferior,
|
||
in format described in environ.h. */
|
||
gdb_environ environment;
|
||
|
||
/* True if this child process was attached rather than forked. */
|
||
bool attach_flag = false;
|
||
|
||
/* If this inferior is a vfork child, then this is the pointer to
|
||
its vfork parent, if GDB is still attached to it. */
|
||
inferior *vfork_parent = NULL;
|
||
|
||
/* If this process is a vfork parent, this is the pointer to the
|
||
child. Since a vfork parent is left frozen by the kernel until
|
||
the child execs or exits, a process can only have one vfork child
|
||
at a given time. */
|
||
inferior *vfork_child = NULL;
|
||
|
||
/* True if this inferior should be detached when it's vfork sibling
|
||
exits or execs. */
|
||
bool pending_detach = false;
|
||
|
||
/* If non-nullptr, points to a thread that called vfork and is now waiting
|
||
for a vfork child not under our control to be done with the shared memory
|
||
region, either by exiting or execing. */
|
||
thread_info *thread_waiting_for_vfork_done = nullptr;
|
||
|
||
/* True if we're in the process of detaching from this inferior. */
|
||
bool detaching = false;
|
||
|
||
/* True if setup_inferior wasn't called for this inferior yet.
|
||
Until that is done, we must not access inferior memory or
|
||
registers, as we haven't determined the target
|
||
architecture/description. */
|
||
bool needs_setup = false;
|
||
|
||
/* True if the inferior is starting up (inside startup_inferior),
|
||
and we're nursing it along (through the shell) until it is ready
|
||
to execute its first instruction. Until that is done, we must
|
||
not access inferior memory or registers, as we haven't determined
|
||
the target architecture/description. */
|
||
bool starting_up = false;
|
||
|
||
/* True when we are reading the library list of the inferior during an
|
||
attach or handling a fork child. */
|
||
bool in_initial_library_scan = false;
|
||
|
||
/* Private data used by the process_stratum target. */
|
||
std::unique_ptr<private_inferior> priv;
|
||
|
||
/* HAS_EXIT_CODE is true if the inferior exited with an exit code.
|
||
In this case, the EXIT_CODE field is also valid. */
|
||
bool has_exit_code = false;
|
||
LONGEST exit_code = 0;
|
||
|
||
/* Default flags to pass to the symbol reading functions. These are
|
||
used whenever a new objfile is created. */
|
||
symfile_add_flags symfile_flags = 0;
|
||
|
||
/* Info about an inferior's target description (if it's fetched; the
|
||
user supplied description's filename, if any; etc.). */
|
||
target_desc_info tdesc_info;
|
||
|
||
/* Data related to displaced stepping. */
|
||
displaced_step_inferior_state displaced_step_state;
|
||
|
||
/* Per inferior data-pointers required by other GDB modules. */
|
||
registry<inferior> registry_fields;
|
||
|
||
private:
|
||
|
||
/* Unpush TARGET and assert that it worked. */
|
||
void unpush_target_and_assert (struct target_ops *target);
|
||
|
||
/* The inferior's target stack. */
|
||
target_stack m_target_stack;
|
||
|
||
/* The name of terminal device to use for I/O. */
|
||
std::string m_terminal;
|
||
|
||
/* The list of continuations. */
|
||
std::list<std::function<void ()>> m_continuations;
|
||
|
||
/* The arguments string to use when running. */
|
||
std::string m_args;
|
||
|
||
/* The current working directory that will be used when starting
|
||
this inferior. */
|
||
std::string m_cwd;
|
||
|
||
/* The architecture associated with the inferior through the
|
||
connection to the target.
|
||
|
||
The architecture vector provides some information that is really
|
||
a property of the inferior, accessed through a particular target:
|
||
ptrace operations; the layout of certain RSP packets; the
|
||
solib_ops vector; etc. To differentiate architecture accesses to
|
||
per-inferior/target properties from
|
||
per-thread/per-frame/per-objfile properties, accesses to
|
||
per-inferior/target properties should be made through
|
||
this gdbarch. */
|
||
gdbarch *m_gdbarch = nullptr;
|
||
};
|
||
|
||
/* Add an inferior to the inferior list, print a message that a new
|
||
inferior is found, and return the pointer to the new inferior.
|
||
Caller may use this pointer to initialize the private inferior
|
||
data. */
|
||
extern struct inferior *add_inferior (int pid);
|
||
|
||
/* Same as add_inferior, but don't print new inferior notifications to
|
||
the CLI. */
|
||
extern struct inferior *add_inferior_silent (int pid);
|
||
|
||
extern void delete_inferior (struct inferior *todel);
|
||
|
||
/* Delete an existing inferior list entry, due to inferior detaching. */
|
||
extern void detach_inferior (inferior *inf);
|
||
|
||
/* Notify observers and interpreters that INF has gone away. Reset the INF
|
||
object back to an default, empty, state. Clear register and frame
|
||
caches. */
|
||
extern void exit_inferior (inferior *inf);
|
||
|
||
extern void inferior_appeared (struct inferior *inf, int pid);
|
||
|
||
/* Search function to lookup an inferior of TARG by target 'pid'. */
|
||
extern struct inferior *find_inferior_pid (process_stratum_target *targ,
|
||
int pid);
|
||
|
||
/* Search function to lookup an inferior of TARG whose pid is equal to
|
||
'ptid.pid'. */
|
||
extern struct inferior *find_inferior_ptid (process_stratum_target *targ,
|
||
ptid_t ptid);
|
||
|
||
/* Search function to lookup an inferior by GDB 'num'. */
|
||
extern struct inferior *find_inferior_id (int num);
|
||
|
||
/* Find an inferior bound to PSPACE, giving preference to the current
|
||
inferior. */
|
||
extern struct inferior *
|
||
find_inferior_for_program_space (struct program_space *pspace);
|
||
|
||
/* Returns true if the inferior list is not empty. */
|
||
extern int have_inferiors (void);
|
||
|
||
/* Returns the number of live inferiors running on PROC_TARGET (real
|
||
live processes with execution). */
|
||
extern int number_of_live_inferiors (process_stratum_target *proc_target);
|
||
|
||
/* Returns true if there are any live inferiors in the inferior list
|
||
(not cores, not executables, real live processes). */
|
||
extern int have_live_inferiors (void);
|
||
|
||
/* Save/restore the current inferior. */
|
||
|
||
class scoped_restore_current_inferior
|
||
{
|
||
public:
|
||
scoped_restore_current_inferior ()
|
||
: m_saved_inf (current_inferior ())
|
||
{}
|
||
|
||
~scoped_restore_current_inferior ()
|
||
{ set_current_inferior (m_saved_inf); }
|
||
|
||
DISABLE_COPY_AND_ASSIGN (scoped_restore_current_inferior);
|
||
|
||
private:
|
||
inferior *m_saved_inf;
|
||
};
|
||
|
||
/* When reading memory from an inferior, the global inferior_ptid must
|
||
also be set. This class arranges to save and restore the necessary
|
||
state for reading or writing memory, but without invalidating the
|
||
frame cache. */
|
||
|
||
class scoped_restore_current_inferior_for_memory
|
||
{
|
||
public:
|
||
|
||
/* Save the current globals and switch to the given inferior and the
|
||
inferior's program space. inferior_ptid is set to point to the
|
||
inferior's process id (and not to any particular thread). */
|
||
explicit scoped_restore_current_inferior_for_memory (inferior *inf)
|
||
: m_save_ptid (&inferior_ptid)
|
||
{
|
||
set_current_inferior (inf);
|
||
set_current_program_space (inf->pspace);
|
||
inferior_ptid = ptid_t (inf->pid);
|
||
}
|
||
|
||
DISABLE_COPY_AND_ASSIGN (scoped_restore_current_inferior_for_memory);
|
||
|
||
private:
|
||
|
||
scoped_restore_current_inferior m_save_inferior;
|
||
scoped_restore_current_program_space m_save_progspace;
|
||
scoped_restore_tmpl<ptid_t> m_save_ptid;
|
||
};
|
||
|
||
|
||
/* Traverse all inferiors. */
|
||
|
||
extern intrusive_list<inferior> inferior_list;
|
||
|
||
/* Pull in the internals of the inferiors ranges and iterators. Must
|
||
be done after struct inferior is defined. */
|
||
#include "inferior-iter.h"
|
||
|
||
/* Return a range that can be used to walk over all inferiors
|
||
inferiors, with range-for, safely. I.e., it is safe to delete the
|
||
currently-iterated inferior. When combined with range-for, this
|
||
allow convenient patterns like this:
|
||
|
||
for (inferior *inf : all_inferiors_safe ())
|
||
if (some_condition ())
|
||
delete inf;
|
||
*/
|
||
|
||
inline all_inferiors_safe_range
|
||
all_inferiors_safe ()
|
||
{
|
||
return all_inferiors_safe_range (nullptr, inferior_list);
|
||
}
|
||
|
||
/* Returns a range representing all inferiors, suitable to use with
|
||
range-for, like this:
|
||
|
||
for (inferior *inf : all_inferiors ())
|
||
[...]
|
||
*/
|
||
|
||
inline all_inferiors_range
|
||
all_inferiors (process_stratum_target *proc_target = nullptr)
|
||
{
|
||
return all_inferiors_range (proc_target, inferior_list);
|
||
}
|
||
|
||
/* Return a range that can be used to walk over all inferiors with PID
|
||
not zero, with range-for. */
|
||
|
||
inline all_non_exited_inferiors_range
|
||
all_non_exited_inferiors (process_stratum_target *proc_target = nullptr)
|
||
{
|
||
return all_non_exited_inferiors_range (proc_target, inferior_list);
|
||
}
|
||
|
||
/* Prune away automatically added inferiors that aren't required
|
||
anymore. */
|
||
extern void prune_inferiors (void);
|
||
|
||
extern int number_of_inferiors (void);
|
||
|
||
extern struct inferior *add_inferior_with_spaces (void);
|
||
|
||
/* Print the current selected inferior. */
|
||
extern void print_selected_inferior (struct ui_out *uiout);
|
||
|
||
/* Switch to inferior NEW_INF, a new inferior, and unless
|
||
NO_CONNECTION is true, push the process_stratum_target of ORG_INF
|
||
to NEW_INF. */
|
||
|
||
extern void switch_to_inferior_and_push_target
|
||
(inferior *new_inf, bool no_connection, inferior *org_inf);
|
||
|
||
/* Return true if ID is a valid global inferior number. */
|
||
|
||
inline bool
|
||
valid_global_inferior_id (int id)
|
||
{
|
||
for (inferior *inf : all_inferiors ())
|
||
if (inf->num == id)
|
||
return true;
|
||
return false;
|
||
}
|
||
|
||
#endif /* !defined (INFERIOR_H) */
|