forked from Imagelibrary/binutils-gdb
Convert solib_ops into an abstract base class (with abstract methods,
some of them with default implementations) and convert all the existing
solib_ops instances to solib_ops derived classes / implementations.
Prior to this patch, solib_ops is a structure holding function pointers,
of which there are only a handful of global instances (in the
`solib-*.c` files). When passing an `solib_ops *` around, it's a
pointer to one of these instances. After this patch, there are no more
global solib_ops instances. Instances are created as needed and stored
in struct program_space. These instances could eventually be made to
contain the program space-specific data, which is currently kept in
per-program space registries (I have some pending patches for that).
Prior to this patch, `gdbarch_so_ops` is a gdbarch method that returns a
pointer to the appropriate solib_ops implementation for the gdbarch.
This is replaced with the `gdbarch_make_solib_ops` method, which returns
a new instance of the appropriate solib_ops implementation for this
gdbarch. This requires introducing some factory functions for the
various solib_ops implementation, to be used as `gdbarch_make_solib_ops`
callbacks. For instance:
solib_ops_up
make_linux_ilp32_svr4_solib_ops ()
{
return std::make_unique<linux_ilp32_svr4_solib_ops> ();
}
The previous code is full of cases of tdep files copying some base
solib_ops implementation, and overriding one or more function pointer
(see ppc_linux_init_abi, for instance). I tried to convert all of this
is a class hierarchy. I like that it's now possible to get a good
static view of all the existing solib_ops variants. The hierarchy looks
like this:
solib_ops
├── aix_solib_ops
├── darwin_solib_ops
├── dsbt_solib_ops
├── frv_solib_ops
├── rocm_solib_ops
├── svr4_solib_ops
│ ├── ilp32_svr4_solib_ops
│ ├── lp64_svr4_solib_ops
│ ├── linux_ilp32_svr4_solib_ops
│ │ ├── mips_linux_ilp32_svr4_solib_ops
│ │ └── ppc_linux_ilp32_svr4_solib_ops
│ ├── linux_lp64_svr4_solib_ops
│ │ └── mips_linux_lp64_svr4_solib_ops
│ ├── mips_nbsd_ilp32_svr4_solib_ops
│ ├── mips_nbsd_lp64_svr4_solib_ops
│ ├── mips_fbsd_ilp32_svr4_solib_ops
│ └── mips_fbsd_lp64_svr4_solib_ops
└── target_solib_ops
└── windows_solib_ops
The solib-svr4 code has per-arch specialization to provide a
link_map_offsets, containing the offsets of the interesting fields in
`struct link_map` on that particular architecture. Prior to this patch,
arches would set a callback returning the appropriate link_map_offsets
by calling `set_solib_svr4_fetch_link_map_offsets`, which also happened
to set the gdbarch's so_ops to `&svr_so_ops`. I converted this to an
abstract virtual method of `struct svr4_solib_ops`, meaning that all
classes deriving from svr4_solib_ops must provide a method returning the
appropriate link_map_offsets for the architecture. I renamed
`set_solib_svr4_fetch_link_map_offsets` to `set_solib_svr4_ops`. This
function is still necessary because it also calls
set_gdbarch_iterate_over_objfiles_in_search_order, but if it was not for
that, we could get rid of it.
There is an instance of CRTP in mips-linux-tdep.c, because both
mips_linux_ilp32_svr4_solib_ops and mips_linux_lp64_svr4_solib_ops need
to derive from different SVR4 base classes (linux_ilp32_svr4_solib_ops
and linux_lp64_svr4_solib_ops), but they both want to override the
in_dynsym_resolve_code method with the same implementation.
The solib_ops::supports_namespaces method is new: the support for
namespaces was previously predicated by the presence or absence of a
find_solib_ns method. It now needs to be explicit.
There is a new progspace::release_solib_ops method, which is only needed
for rocm_solib_ops. For the moment, rocm_solib_ops replaces and wraps
the existing svr4_solib_ops instance, in order to combine the results of
the two. The plan is to have a subsequent patch to allow program spaces to have
multiple solib_ops, removing the need for release_solib_ops.
Speaking of rocm_solib_ops: it previously overrode only a few methods by
copying svr4_solib_ops and overwriting some function pointers. Now, it
needs to implement all the methods that svr4_solib_ops implements, in
order to forward the call. Otherwise, the default solib_ops method
would be called, hiding the svr4_solib_ops implementation. Again, this
can be removed once we have support for multiple solib_ops in a
program_space.
There is also a small change in how rocm_solib_ops is activated. Prior
to this patch, it's done at the end of rocm_update_solib_list. Since it
overrides the function pointer in the static svr4_solib_ops, and then
overwrites the host gdbarch, so_ops field, it's something that happens
only once. After the patch though, we need to set rocm_solib_ops in all
the program spaces that appear. We do this in
rocm_solib_target_inferior_created and in the new
rocm_solib_target_inferior_execd. After this, I will explore doing a
change where rocm_solib_ops is only set when we detect the ROCm runtime
is loaded.
Change-Id: I5896b5bcbf8bdb024d67980380feba1ffefaa4c9
Approved-By: Pedro Alves <pedro@palves.net>
222 lines
7.9 KiB
C++
222 lines
7.9 KiB
C++
/* Handle shared libraries for GDB, the GNU Debugger.
|
|
|
|
Copyright (C) 2000-2025 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/>. */
|
|
|
|
#ifndef GDB_SOLIB_SVR4_H
|
|
#define GDB_SOLIB_SVR4_H
|
|
|
|
#include "gdbarch.h"
|
|
#include "solib.h"
|
|
|
|
struct objfile;
|
|
struct link_map_offsets;
|
|
struct probe_and_action;
|
|
struct svr4_info;
|
|
struct svr4_library_list;
|
|
struct svr4_so;
|
|
|
|
/* Link map info to include in an allocated solib entry. */
|
|
|
|
struct lm_info_svr4 final : public lm_info
|
|
{
|
|
/* Amount by which addresses in the binary should be relocated to
|
|
match the inferior. The direct inferior value is L_ADDR_INFERIOR.
|
|
When prelinking is involved and the prelink base address changes,
|
|
we may need a different offset - the recomputed offset is in L_ADDR.
|
|
It is commonly the same value. It is cached as we want to warn about
|
|
the difference and compute it only once. L_ADDR is valid
|
|
iff L_ADDR_P. */
|
|
CORE_ADDR l_addr = 0, l_addr_inferior = 0;
|
|
bool l_addr_p = false;
|
|
|
|
/* The target location of lm. */
|
|
CORE_ADDR lm_addr = 0;
|
|
|
|
/* Values read in from inferior's fields of the same name. */
|
|
CORE_ADDR l_ld = 0, l_next = 0, l_prev = 0, l_name = 0;
|
|
};
|
|
|
|
using lm_info_svr4_up = std::unique_ptr<lm_info_svr4>;
|
|
|
|
/* What to do when a probe stop occurs. */
|
|
|
|
enum probe_action
|
|
{
|
|
/* Something went seriously wrong. Stop using probes and
|
|
revert to using the older interface. */
|
|
PROBES_INTERFACE_FAILED,
|
|
|
|
/* No action is required. The shared object list is still
|
|
valid. */
|
|
DO_NOTHING,
|
|
|
|
/* The shared object list should be reloaded entirely. */
|
|
FULL_RELOAD,
|
|
|
|
/* Attempt to incrementally update the shared object list. If
|
|
the update fails or is not possible, fall back to reloading
|
|
the list in full. */
|
|
UPDATE_OR_RELOAD,
|
|
};
|
|
|
|
/* solib_ops for SVR4 systems. */
|
|
|
|
struct svr4_solib_ops : public solib_ops
|
|
{
|
|
void relocate_section_addresses (solib &so, target_section *) const override;
|
|
void clear_so (const solib &so) const override;
|
|
void clear_solib (program_space *pspace) const override;
|
|
void create_inferior_hook (int from_tty) const override;
|
|
owning_intrusive_list<solib> current_sos () const override;
|
|
bool open_symbol_file_object (int from_tty) const override;
|
|
bool in_dynsym_resolve_code (CORE_ADDR pc) const override;
|
|
bool same (const solib &gdb, const solib &inferior) const override;
|
|
bool keep_data_in_core (CORE_ADDR vaddr, unsigned long size) const override;
|
|
void update_breakpoints () const override;
|
|
void handle_event () const override;
|
|
std::optional<CORE_ADDR> find_solib_addr (solib &so) const override;
|
|
bool supports_namespaces () const override { return true; }
|
|
int find_solib_ns (const solib &so) const override;
|
|
int num_active_namespaces () const override;
|
|
std::vector<const solib *> get_solibs_in_ns (int nsid) const override;
|
|
|
|
/* Return the appropriate link map offsets table for the architecture. */
|
|
virtual link_map_offsets *fetch_link_map_offsets () const = 0;
|
|
|
|
/* This needs to be public because it's accessed from an observer. */
|
|
void current_sos_direct (svr4_info *info) const;
|
|
|
|
private:
|
|
void create_probe_breakpoints (svr4_info *info, gdbarch *gdbarch,
|
|
const std::vector<probe *> *probes,
|
|
objfile *objfile) const;
|
|
bool find_and_create_probe_breakpoints (svr4_info *info, gdbarch *gdbarch,
|
|
obj_section *os,
|
|
bool with_prefix) const;
|
|
void create_event_breakpoints (svr4_info *info, gdbarch *gdbarch,
|
|
CORE_ADDR address) const;
|
|
int enable_break (svr4_info *info, int from_tty) const;
|
|
bool is_default_namespace (CORE_ADDR debug_base) const;
|
|
void free_probes_table (svr4_info *info) const;
|
|
CORE_ADDR find_r_brk (svr4_info *info) const;
|
|
CORE_ADDR find_r_ldsomap (svr4_info *info) const;
|
|
owning_intrusive_list<solib> default_sos (svr4_info *info) const;
|
|
int read_so_list (svr4_info *info, CORE_ADDR lm, CORE_ADDR prev_lm,
|
|
std::vector<svr4_so> &sos, int ignore_first) const;
|
|
lm_info_svr4_up read_lm_info (CORE_ADDR lm_addr) const;
|
|
int has_lm_dynamic_from_link_map () const;
|
|
CORE_ADDR lm_addr_check (const solib &so, bfd *abfd) const;
|
|
CORE_ADDR read_r_next (CORE_ADDR debug_base) const;
|
|
CORE_ADDR read_r_map (CORE_ADDR debug_base) const;
|
|
int parse_libraries (const char *document, svr4_library_list *list);
|
|
int current_sos_via_xfer_libraries (svr4_library_list *list,
|
|
const char *annex) const;
|
|
owning_intrusive_list<solib> collect_probes_sos (svr4_info *info) const;
|
|
owning_intrusive_list<solib> current_sos_1 (svr4_info *info) const;
|
|
owning_intrusive_list<solib> solibs_from_svr4_sos
|
|
(const std::vector<svr4_so> &sos) const;
|
|
void register_event_probe (objfile *objfile, probe *prob, CORE_ADDR address,
|
|
enum probe_action action) const;
|
|
void disable_probes_interface (svr4_info *info) const;
|
|
probe_and_action *event_probe_at (CORE_ADDR address) const;
|
|
void update_full (svr4_info *info) const;
|
|
int update_incremental (svr4_info *info, CORE_ADDR debug_base,
|
|
CORE_ADDR lm) const;
|
|
bool update_event_breakpoint (breakpoint *b) const;
|
|
CORE_ADDR find_debug_base (const solib *solib) const;
|
|
};
|
|
|
|
/* solib_ops for ILP32 SVR4 systems. */
|
|
|
|
struct ilp32_svr4_solib_ops : public svr4_solib_ops
|
|
{
|
|
link_map_offsets *fetch_link_map_offsets () const override;
|
|
};
|
|
|
|
/* Critical offsets and sizes which describe struct r_debug and
|
|
struct link_map on SVR4-like targets. All offsets and sizes are
|
|
in bytes unless otherwise specified. */
|
|
|
|
struct link_map_offsets
|
|
{
|
|
/* Offset and size of r_debug.r_version. */
|
|
int r_version_offset, r_version_size;
|
|
|
|
/* Offset of r_debug.r_map. */
|
|
int r_map_offset;
|
|
|
|
/* Offset of r_debug.r_brk. */
|
|
int r_brk_offset;
|
|
|
|
/* Offset of r_debug.r_ldsomap. */
|
|
int r_ldsomap_offset;
|
|
|
|
/* Offset of r_debug_extended.r_next. */
|
|
int r_next_offset;
|
|
|
|
/* Size of struct link_map (or equivalent), or at least enough of it
|
|
to be able to obtain the fields below. */
|
|
int link_map_size;
|
|
|
|
/* Offset to l_addr field in struct link_map. */
|
|
int l_addr_offset;
|
|
|
|
/* Offset to l_ld field in struct link_map. */
|
|
int l_ld_offset;
|
|
|
|
/* Offset to l_next field in struct link_map. */
|
|
int l_next_offset;
|
|
|
|
/* Offset to l_prev field in struct link_map. */
|
|
int l_prev_offset;
|
|
|
|
/* Offset to l_name field in struct link_map. */
|
|
int l_name_offset;
|
|
};
|
|
|
|
/* Set the gdbarch methods for SVR4 systems. */
|
|
|
|
extern void set_solib_svr4_ops (gdbarch *gdbarch,
|
|
gdbarch_make_solib_ops_ftype make_solib_ops);
|
|
|
|
/* This function is called by thread_db.c. Return the address of the
|
|
link map for the given objfile. */
|
|
extern CORE_ADDR svr4_fetch_objfile_link_map (struct objfile *objfile);
|
|
|
|
/* Return a new solib_ops for ILP32 SVR4 systems. */
|
|
|
|
extern solib_ops_up make_svr4_ilp32_solib_ops ();
|
|
|
|
/* Return a new solib_ops for LP64 SVR4 systems. */
|
|
|
|
extern solib_ops_up make_svr4_lp64_solib_ops ();
|
|
|
|
/* For the MUSL C library, given link map address LM_ADDR, return the
|
|
corresponding TLS module id, or 0 if not found. */
|
|
int musl_link_map_to_tls_module_id (CORE_ADDR lm_addr);
|
|
|
|
/* For GLIBC, given link map address LM_ADDR, return the corresponding TLS
|
|
module id, or 0 if not found. */
|
|
int glibc_link_map_to_tls_module_id (CORE_ADDR lm_addr);
|
|
|
|
/* Return program interpreter string. */
|
|
|
|
std::optional<gdb::byte_vector> svr4_find_program_interpreter ();
|
|
|
|
#endif /* GDB_SOLIB_SVR4_H */
|