| /* Target-dependent code for the GNU C Library (glibc). | 
 |  | 
 |    Copyright (C) 2002-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/>.  */ | 
 |  | 
 | #include "frame.h" | 
 | #include "symtab.h" | 
 | #include "symfile.h" | 
 | #include "objfiles.h" | 
 |  | 
 | #include "glibc-tdep.h" | 
 |  | 
 | /* Calling functions in shared libraries.  */ | 
 |  | 
 | /* See the comments for SKIP_SOLIB_RESOLVER at the top of infrun.c. | 
 |    This function: | 
 |    1) decides whether a PLT has sent us into the linker to resolve | 
 |       a function reference, and  | 
 |    2) if so, tells us where to set a temporary breakpoint that will | 
 |       trigger when the dynamic linker is done.  */ | 
 |  | 
 | CORE_ADDR | 
 | glibc_skip_solib_resolver (struct gdbarch *gdbarch, CORE_ADDR pc) | 
 | { | 
 |   /* The GNU dynamic linker is part of the GNU C library, and is used | 
 |      by all GNU systems (GNU/Hurd, GNU/Linux).  An unresolved PLT | 
 |      entry points to "_dl_runtime_resolve", which calls "fixup" to | 
 |      patch the PLT, and then passes control to the function. | 
 |  | 
 |      We look for the symbol `_dl_runtime_resolve', and find `fixup' in | 
 |      the same objfile.  If we are at the entry point of `fixup', then | 
 |      we set a breakpoint at the return address (at the top of the | 
 |      stack), and continue. | 
 |    | 
 |      It's kind of gross to do all these checks every time we're | 
 |      called, since they don't change once the executable has gotten | 
 |      started.  But this is only a temporary hack --- upcoming versions | 
 |      of GNU/Linux will provide a portable, efficient interface for | 
 |      debugging programs that use shared libraries.  */ | 
 |  | 
 |   bound_minimal_symbol resolver | 
 |     = lookup_minimal_symbol (current_program_space, "_dl_runtime_resolve"); | 
 |  | 
 |   if (resolver.minsym) | 
 |     { | 
 |       /* The dynamic linker began using this name in early 2005.  */ | 
 |       bound_minimal_symbol fixup | 
 | 	= lookup_minimal_symbol (current_program_space, "_dl_fixup", | 
 | 				 resolver.objfile); | 
 |  | 
 |       /* This is the name used in older versions.  */ | 
 |       if (! fixup.minsym) | 
 | 	fixup = lookup_minimal_symbol (current_program_space, "fixup", | 
 | 				       resolver.objfile); | 
 |  | 
 |       if (fixup.minsym && fixup.value_address () == pc) | 
 | 	return frame_unwind_caller_pc (get_current_frame ()); | 
 |     } | 
 |  | 
 |   return 0; | 
 | }       |