2001-02-20 20:43:12 +00:00
/* Handle AIX5 shared libraries for GDB, the GNU Debugger.
2001-03-06 08:22:02 +00:00
Copyright 1990 , 1991 , 1992 , 1993 , 1994 , 1995 , 1996 , 1998 , 1999 , 2000 ,
2001
2001-02-20 20:43:12 +00:00
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 2 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 , write to the Free Software
Foundation , Inc . , 59 Temple Place - Suite 330 ,
Boston , MA 02111 - 1307 , USA . */
# include "defs.h"
# include <sys/types.h>
# include <signal.h>
# include "gdb_string.h"
# include <sys/param.h>
# include <fcntl.h>
# include <sys/procfs.h>
# include "elf/external.h"
# include "symtab.h"
# include "bfd.h"
# include "symfile.h"
# include "objfiles.h"
# include "gdbcore.h"
# include "command.h"
# include "target.h"
# include "frame.h"
# include "gdb_regex.h"
# include "inferior.h"
# include "environ.h"
# include "language.h"
# include "gdbcmd.h"
# include "solist.h"
/* Link map info to include in an allocated so_list entry */
enum maptype {
MT_READONLY = 0 ,
MT_READWRITE = 1 ,
MT_LAST = 2
} ;
struct lm_info
{
struct
{
CORE_ADDR addr ; /* base address */
CORE_ADDR size ; /* size of mapped object */
CORE_ADDR offset ; /* offset into mapped object */
long flags ; /* MA_ protection and attribute flags */
CORE_ADDR gp ; /* global pointer value */
} mapping [ MT_LAST ] ;
char * mapname ; /* name in /proc/pid/object */
char * pathname ; /* full pathname to object */
char * membername ; /* member name in archive file */
} ;
2001-02-20 23:42:23 +00:00
/* List of symbols in the dynamic linker where GDB can try to place
a breakpoint to monitor shared library events . */
2001-02-20 20:43:12 +00:00
static char * solib_break_names [ ] =
{
" _r_debug_state " ,
NULL
} ;
static void aix5_relocate_main_executable ( void ) ;
/*
LOCAL FUNCTION
bfd_lookup_symbol - - lookup the value for a specific symbol
SYNOPSIS
CORE_ADDR bfd_lookup_symbol ( bfd * abfd , char * symname )
DESCRIPTION
An expensive way to lookup the value of a single symbol for
bfd ' s that are only temporary anyway . This is used by the
shared library support to find the address of the debugger
interface structures in the shared library .
Note that 0 is specifically allowed as an error return ( no
such symbol ) .
*/
static CORE_ADDR
bfd_lookup_symbol ( bfd * abfd , char * symname )
{
2001-03-16 18:06:44 +00:00
long storage_needed ;
2001-02-20 20:43:12 +00:00
asymbol * sym ;
asymbol * * symbol_table ;
unsigned int number_of_symbols ;
unsigned int i ;
struct cleanup * back_to ;
CORE_ADDR symaddr = 0 ;
storage_needed = bfd_get_symtab_upper_bound ( abfd ) ;
if ( storage_needed > 0 )
{
symbol_table = ( asymbol * * ) xmalloc ( storage_needed ) ;
2001-03-16 03:52:12 +00:00
back_to = make_cleanup ( xfree , symbol_table ) ;
2001-02-20 20:43:12 +00:00
number_of_symbols = bfd_canonicalize_symtab ( abfd , symbol_table ) ;
for ( i = 0 ; i < number_of_symbols ; i + + )
{
sym = * symbol_table + + ;
2001-02-20 23:42:23 +00:00
if ( strcmp ( sym - > name , symname ) = = 0 )
2001-02-20 20:43:12 +00:00
{
/* Bfd symbols are section relative. */
symaddr = sym - > value + sym - > section - > vma ;
break ;
}
}
do_cleanups ( back_to ) ;
}
if ( symaddr )
return symaddr ;
2001-02-20 23:42:23 +00:00
/* Look for the symbol in the dynamic string table too. */
2001-02-20 20:43:12 +00:00
storage_needed = bfd_get_dynamic_symtab_upper_bound ( abfd ) ;
if ( storage_needed > 0 )
{
symbol_table = ( asymbol * * ) xmalloc ( storage_needed ) ;
2001-03-16 03:52:12 +00:00
back_to = make_cleanup ( xfree , symbol_table ) ;
2001-02-20 20:43:12 +00:00
number_of_symbols = bfd_canonicalize_dynamic_symtab ( abfd , symbol_table ) ;
for ( i = 0 ; i < number_of_symbols ; i + + )
{
sym = * symbol_table + + ;
2001-02-20 23:42:23 +00:00
if ( strcmp ( sym - > name , symname ) = = 0 )
2001-02-20 20:43:12 +00:00
{
/* Bfd symbols are section relative. */
symaddr = sym - > value + sym - > section - > vma ;
break ;
}
}
do_cleanups ( back_to ) ;
}
return symaddr ;
}
/* Read /proc/PID/map and build a list of shared objects such that
the pr_mflags value AND ' d with MATCH_MASK is equal to MATCH_VAL .
This gives us a convenient way to find all of the mappings that
don ' t belong to the main executable or vice versa . Here are
some of the possibilities :
- Fetch all mappings :
MATCH_MASK : 0
MATCH_VAL : 0
- Fetch all mappings except for main executable :
MATCH_MASK : MA_MAINEXEC
MATCH_VAL : 0
- Fetch only main executable :
MATCH_MASK : MA_MAINEXEC
MATCH_VAL : MA_MAINEXEC
A cleanup chain for the list allocations done by this function should
be established prior to calling build_so_list_from_mapfile ( ) . */
static struct so_list *
build_so_list_from_mapfile ( int pid , long match_mask , long match_val )
{
char * mapbuf = NULL ;
struct prmap * prmap ;
int mapbuf_size ;
struct so_list * sos = NULL ;
{
int mapbuf_allocation_size = 8192 ;
2001-03-27 06:23:25 +00:00
char * map_pathname ;
2001-02-20 20:43:12 +00:00
int map_fd ;
/* Open the map file */
2001-03-27 06:23:25 +00:00
xasprintf ( & map_pathname , " /proc/%d/map " , pid ) ;
2001-02-20 20:43:12 +00:00
map_fd = open ( map_pathname , O_RDONLY ) ;
2001-03-27 06:23:25 +00:00
free ( map_pathname ) ;
2001-02-20 20:43:12 +00:00
if ( map_fd < 0 )
return 0 ;
/* Read the entire map file in */
do
{
if ( mapbuf )
{
2001-03-16 03:52:12 +00:00
xfree ( mapbuf ) ;
2001-02-20 20:43:12 +00:00
mapbuf_allocation_size * = 2 ;
lseek ( map_fd , 0 , SEEK_SET ) ;
}
mapbuf = xmalloc ( mapbuf_allocation_size ) ;
mapbuf_size = read ( map_fd , mapbuf , mapbuf_allocation_size ) ;
if ( mapbuf_size < 0 )
{
2001-03-16 03:52:12 +00:00
xfree ( mapbuf ) ;
2001-02-20 20:43:12 +00:00
/* FIXME: This warrants an error or a warning of some sort */
return 0 ;
}
} while ( mapbuf_size = = mapbuf_allocation_size ) ;
close ( map_fd ) ;
}
for ( prmap = ( struct prmap * ) mapbuf ;
( char * ) prmap < mapbuf + mapbuf_size ;
prmap + + )
{
char * mapname , * pathname , * membername ;
struct so_list * sop ;
enum maptype maptype ;
if ( prmap - > pr_size = = 0 )
break ;
/* Skip to the next entry if there's no path associated with the
map , unless we ' re looking for the kernel text region , in which
case it ' s okay if there ' s no path . */
if ( ( prmap - > pr_pathoff = = 0 | | prmap - > pr_pathoff > = mapbuf_size )
& & ( ( match_mask & MA_KERNTEXT ) = = 0 ) )
continue ;
/* Skip to the next entry if our match conditions don't hold. */
if ( ( prmap - > pr_mflags & match_mask ) ! = match_val )
continue ;
mapname = prmap - > pr_mapname ;
if ( prmap - > pr_pathoff = = 0 )
{
pathname = " " ;
membername = " " ;
}
else
{
pathname = mapbuf + prmap - > pr_pathoff ;
membername = pathname + strlen ( pathname ) + 1 ;
}
for ( sop = sos ; sop ! = NULL ; sop = sop - > next )
if ( strcmp ( pathname , sop - > lm_info - > pathname ) = = 0
& & strcmp ( membername , sop - > lm_info - > membername ) = = 0 )
break ;
if ( sop = = NULL )
{
sop = xcalloc ( sizeof ( struct so_list ) , 1 ) ;
2001-03-16 03:52:12 +00:00
make_cleanup ( xfree , sop ) ;
2001-02-20 20:43:12 +00:00
sop - > lm_info = xcalloc ( sizeof ( struct lm_info ) , 1 ) ;
2001-03-16 03:52:12 +00:00
make_cleanup ( xfree , sop - > lm_info ) ;
2001-02-20 20:43:12 +00:00
sop - > lm_info - > mapname = xstrdup ( mapname ) ;
2001-03-16 03:52:12 +00:00
make_cleanup ( xfree , sop - > lm_info - > mapname ) ;
2001-02-20 20:43:12 +00:00
/* FIXME: Eliminate the pathname field once length restriction
is lifted on so_name and so_original_name . */
sop - > lm_info - > pathname = xstrdup ( pathname ) ;
2001-03-16 03:52:12 +00:00
make_cleanup ( xfree , sop - > lm_info - > pathname ) ;
2001-02-20 20:43:12 +00:00
sop - > lm_info - > membername = xstrdup ( membername ) ;
2001-03-16 03:52:12 +00:00
make_cleanup ( xfree , sop - > lm_info - > membername ) ;
2001-02-20 20:43:12 +00:00
strncpy ( sop - > so_name , pathname , SO_NAME_MAX_PATH_SIZE - 1 ) ;
sop - > so_name [ SO_NAME_MAX_PATH_SIZE - 1 ] = ' \0 ' ;
strcpy ( sop - > so_original_name , sop - > so_name ) ;
sop - > next = sos ;
sos = sop ;
}
maptype = ( prmap - > pr_mflags & MA_WRITE ) ? MT_READWRITE : MT_READONLY ;
sop - > lm_info - > mapping [ maptype ] . addr = ( CORE_ADDR ) prmap - > pr_vaddr ;
sop - > lm_info - > mapping [ maptype ] . size = prmap - > pr_size ;
sop - > lm_info - > mapping [ maptype ] . offset = prmap - > pr_off ;
sop - > lm_info - > mapping [ maptype ] . flags = prmap - > pr_mflags ;
sop - > lm_info - > mapping [ maptype ] . gp = ( CORE_ADDR ) prmap - > pr_gp ;
}
2001-03-16 03:52:12 +00:00
xfree ( mapbuf ) ;
2001-02-20 20:43:12 +00:00
return sos ;
}
/*
LOCAL FUNCTION
open_symbol_file_object
SYNOPSIS
void open_symbol_file_object ( void * from_tty )
DESCRIPTION
If no open symbol file , attempt to locate and open the main symbol
file .
If FROM_TTYP dereferences to a non - zero integer , allow messages to
be printed . This parameter is a pointer rather than an int because
open_symbol_file_object ( ) is called via catch_errors ( ) and
catch_errors ( ) requires a pointer argument . */
static int
open_symbol_file_object ( void * from_ttyp )
{
CORE_ADDR lm , l_name ;
char * filename ;
int errcode ;
int from_tty = * ( int * ) from_ttyp ;
struct cleanup * old_chain = make_cleanup ( null_cleanup , 0 ) ;
struct so_list * sos ;
sos = build_so_list_from_mapfile ( PIDGET ( inferior_pid ) ,
MA_MAINEXEC , MA_MAINEXEC ) ;
if ( sos = = NULL )
{
warning ( " Could not find name of main executable in map file " ) ;
return 0 ;
}
symbol_file_command ( sos - > lm_info - > pathname , from_tty ) ;
do_cleanups ( old_chain ) ;
aix5_relocate_main_executable ( ) ;
return 1 ;
}
/* LOCAL FUNCTION
aix5_current_sos - - build a list of currently loaded shared objects
SYNOPSIS
struct so_list * aix5_current_sos ( )
DESCRIPTION
Build a list of ` struct so_list ' objects describing the shared
objects currently loaded in the inferior . This list does not
include an entry for the main executable file .
Note that we only gather information directly available from the
inferior - - - we don ' t examine any of the shared library files
themselves . The declaration of ` struct so_list ' says which fields
we provide values for . */
static struct so_list *
aix5_current_sos ( void )
{
struct cleanup * old_chain = make_cleanup ( null_cleanup , 0 ) ;
struct so_list * sos ;
/* Fetch the list of mappings, excluding the main executable. */
sos = build_so_list_from_mapfile ( PIDGET ( inferior_pid ) , MA_MAINEXEC , 0 ) ;
/* Reverse the list; it looks nicer when we print it if the mappings
are in the same order as in the map file . */
if ( sos )
{
struct so_list * next = sos - > next ;
sos - > next = 0 ;
while ( next )
{
struct so_list * prev = sos ;
sos = next ;
next = next - > next ;
sos - > next = prev ;
}
}
discard_cleanups ( old_chain ) ;
return sos ;
}
/* Return 1 if PC lies in the dynamic symbol resolution code of the
2001-02-20 23:42:23 +00:00
run time loader . */
2001-02-20 20:43:12 +00:00
static CORE_ADDR interp_text_sect_low ;
static CORE_ADDR interp_text_sect_high ;
static CORE_ADDR interp_plt_sect_low ;
static CORE_ADDR interp_plt_sect_high ;
2001-02-22 03:01:27 +00:00
static int
aix5_in_dynsym_resolve_code ( CORE_ADDR pc )
2001-02-20 20:43:12 +00:00
{
return ( ( pc > = interp_text_sect_low & & pc < interp_text_sect_high )
| | ( pc > = interp_plt_sect_low & & pc < interp_plt_sect_high )
| | in_plt_section ( pc , NULL ) ) ;
}
/*
LOCAL FUNCTION
enable_break - - arrange for dynamic linker to hit breakpoint
SYNOPSIS
int enable_break ( void )
DESCRIPTION
2001-02-20 23:42:23 +00:00
The dynamic linkers has , as part of its debugger interface , support
for arranging for the inferior to hit a breakpoint after mapping in
the shared libraries . This function enables that breakpoint .
2001-02-20 20:43:12 +00:00
*/
static int
enable_break ( void )
{
int success = 0 ;
struct minimal_symbol * msymbol ;
char * * bkpt_namep ;
asection * interp_sect ;
/* First, remove all the solib event breakpoints. Their addresses
may have changed since the last time we ran the program . */
remove_solib_event_breakpoints ( ) ;
interp_text_sect_low = interp_text_sect_high = 0 ;
interp_plt_sect_low = interp_plt_sect_high = 0 ;
/* Find the .interp section; if not found, warn the user and drop
into the old breakpoint at symbol code . */
interp_sect = bfd_get_section_by_name ( exec_bfd , " .interp " ) ;
if ( interp_sect )
{
unsigned int interp_sect_size ;
char * buf ;
CORE_ADDR load_addr ;
bfd * tmp_bfd ;
CORE_ADDR sym_addr = 0 ;
/* Read the contents of the .interp section into a local buffer;
the contents specify the dynamic linker this program uses . */
interp_sect_size = bfd_section_size ( exec_bfd , interp_sect ) ;
buf = alloca ( interp_sect_size ) ;
bfd_get_section_contents ( exec_bfd , interp_sect ,
buf , 0 , interp_sect_size ) ;
/* Now we need to figure out where the dynamic linker was
loaded so that we can load its symbols and place a breakpoint
in the dynamic linker itself .
This address is stored on the stack . However , I ' ve been unable
to find any magic formula to find it for Solaris ( appears to
be trivial on GNU / Linux ) . Therefore , we have to try an alternate
mechanism to find the dynamic linker ' s base address . */
tmp_bfd = bfd_openr ( buf , gnutarget ) ;
if ( tmp_bfd = = NULL )
goto bkpt_at_symbol ;
/* Make sure the dynamic linker's really a useful object. */
if ( ! bfd_check_format ( tmp_bfd , bfd_object ) )
{
warning ( " Unable to grok dynamic linker %s as an object file " , buf ) ;
bfd_close ( tmp_bfd ) ;
goto bkpt_at_symbol ;
}
/* We find the dynamic linker's base address by examining the
current pc ( which point at the entry point for the dynamic
linker ) and subtracting the offset of the entry point . */
load_addr = read_pc ( ) - tmp_bfd - > start_address ;
/* Record the relocated start and end address of the dynamic linker
2001-02-22 03:01:27 +00:00
text and plt section for aix5_in_dynsym_resolve_code . */
2001-02-20 20:43:12 +00:00
interp_sect = bfd_get_section_by_name ( tmp_bfd , " .text " ) ;
if ( interp_sect )
{
interp_text_sect_low =
bfd_section_vma ( tmp_bfd , interp_sect ) + load_addr ;
interp_text_sect_high =
interp_text_sect_low + bfd_section_size ( tmp_bfd , interp_sect ) ;
}
interp_sect = bfd_get_section_by_name ( tmp_bfd , " .plt " ) ;
if ( interp_sect )
{
interp_plt_sect_low =
bfd_section_vma ( tmp_bfd , interp_sect ) + load_addr ;
interp_plt_sect_high =
interp_plt_sect_low + bfd_section_size ( tmp_bfd , interp_sect ) ;
}
/* Now try to set a breakpoint in the dynamic linker. */
for ( bkpt_namep = solib_break_names ; * bkpt_namep ! = NULL ; bkpt_namep + + )
{
sym_addr = bfd_lookup_symbol ( tmp_bfd , * bkpt_namep ) ;
if ( sym_addr ! = 0 )
break ;
}
/* We're done with the temporary bfd. */
bfd_close ( tmp_bfd ) ;
if ( sym_addr ! = 0 )
{
create_solib_event_breakpoint ( load_addr + sym_addr ) ;
return 1 ;
}
/* For whatever reason we couldn't set a breakpoint in the dynamic
linker . Warn and drop into the old code . */
bkpt_at_symbol :
warning ( " Unable to find dynamic linker breakpoint function. \n GDB will be unable to debug shared library initializers \n and track explicitly loaded dynamic code. " ) ;
}
/* Nothing good happened. */
success = 0 ;
return ( success ) ;
}
/*
LOCAL FUNCTION
special_symbol_handling - - additional shared library symbol handling
SYNOPSIS
void special_symbol_handling ( )
DESCRIPTION
Once the symbols from a shared object have been loaded in the usual
way , we are called to do any system specific symbol handling that
is needed .
*/
static void
aix5_special_symbol_handling ( void )
{
/* Nothing needed (yet) for AIX5. */
}
# define SECTMAPMASK (~ (CORE_ADDR) 0x03ffffff)
static void
aix5_relocate_main_executable ( void )
{
struct so_list * so ;
struct section_offsets * new_offsets ;
int i ;
int changed = 0 ;
struct cleanup * old_chain = make_cleanup ( null_cleanup , 0 ) ;
/* Fetch the mappings for the main executable from the map file. */
so = build_so_list_from_mapfile ( PIDGET ( inferior_pid ) ,
MA_MAINEXEC , MA_MAINEXEC ) ;
/* Make sure we actually have some mappings to work with. */
if ( so = = NULL )
{
warning ( " Could not find main executable in map file " ) ;
do_cleanups ( old_chain ) ;
return ;
}
/* Allocate the data structure which'll contain the new offsets to
relocate by . Initialize it so it contains the current offsets . */
new_offsets = xcalloc ( sizeof ( struct section_offsets ) ,
symfile_objfile - > num_sections ) ;
2001-03-16 03:52:12 +00:00
make_cleanup ( xfree , new_offsets ) ;
2001-02-20 20:43:12 +00:00
for ( i = 0 ; i < symfile_objfile - > num_sections ; i + + )
2001-03-03 08:24:05 +00:00
new_offsets - > offsets [ i ] = ANOFFSET ( symfile_objfile - > section_offsets , i ) ;
2001-02-20 20:43:12 +00:00
/* Iterate over the mappings in the main executable and compute
the new offset value as appropriate . */
for ( i = 0 ; i < MT_LAST ; i + + )
{
CORE_ADDR increment = 0 ;
struct obj_section * sect ;
bfd * obfd = symfile_objfile - > obfd ;
ALL_OBJFILE_OSECTIONS ( symfile_objfile , sect )
{
int flags = bfd_get_section_flags ( obfd , sect - > the_bfd_section ) ;
if ( flags & SEC_ALLOC )
{
if ( ( ( so - > lm_info - > mapping [ i ] . flags & MA_WRITE ) = = 0 )
= = ( ( flags & SEC_READONLY ) ! = 0 ) )
{
int idx = sect - > the_bfd_section - > index ;
if ( increment = = 0 )
increment = so - > lm_info - > mapping [ i ] . addr
- ( bfd_section_vma ( obfd , sect - > the_bfd_section )
& SECTMAPMASK ) ;
if ( increment ! = ANOFFSET ( new_offsets , idx ) )
{
2001-03-03 08:24:05 +00:00
new_offsets - > offsets [ idx ] = increment ;
2001-02-20 20:43:12 +00:00
changed = 1 ;
}
}
}
}
}
/* If any of the offsets have changed, then relocate the objfile. */
if ( changed )
objfile_relocate ( symfile_objfile , new_offsets ) ;
/* Free up all the space we've allocated. */
do_cleanups ( old_chain ) ;
}
/*
GLOBAL FUNCTION
aix5_solib_create_inferior_hook - - shared library startup support
SYNOPSIS
void aix5_solib_create_inferior_hook ( )
DESCRIPTION
When gdb starts up the inferior , it nurses it along ( through the
shell ) until it is ready to execute it ' s first instruction . At this
point , this function gets called via expansion of the macro
SOLIB_CREATE_INFERIOR_HOOK .
2001-02-20 23:42:23 +00:00
For AIX5 executables , this first instruction is the first
2001-02-20 20:43:12 +00:00
instruction in the dynamic linker ( for dynamically linked
executables ) or the instruction at " start " for statically linked
executables . For dynamically linked executables , the system
2001-02-20 23:42:23 +00:00
first exec ' s libc . so . N , which contains the dynamic linker ,
2001-02-20 20:43:12 +00:00
and starts it running . The dynamic linker maps in any needed
shared libraries , maps in the actual user executable , and then
jumps to " start " in the user executable .
*/
static void
aix5_solib_create_inferior_hook ( void )
{
aix5_relocate_main_executable ( ) ;
if ( ! enable_break ( ) )
{
warning ( " shared library handler failed to enable breakpoint " ) ;
return ;
}
}
static void
aix5_clear_solib ( void )
{
}
static void
aix5_free_so ( struct so_list * so )
{
2001-03-16 03:52:12 +00:00
xfree ( so - > lm_info - > mapname ) ;
xfree ( so - > lm_info - > pathname ) ;
xfree ( so - > lm_info - > membername ) ;
xfree ( so - > lm_info ) ;
2001-02-20 20:43:12 +00:00
}
static void
aix5_relocate_section_addresses ( struct so_list * so ,
struct section_table * sec )
{
int flags = bfd_get_section_flags ( sec - > bfd , sec - > the_bfd_section ) ;
if ( flags & SEC_ALLOC )
{
int idx = ( flags & SEC_READONLY ) ? MT_READONLY : MT_READWRITE ;
CORE_ADDR addr = so - > lm_info - > mapping [ idx ] . addr ;
sec - > addr + = addr ;
sec - > endaddr + = addr ;
}
}
/* Find the global pointer for the given function address ADDR. */
static CORE_ADDR
aix5_find_global_pointer ( CORE_ADDR addr )
{
struct so_list * sos , * so ;
CORE_ADDR global_pointer = 0 ;
struct cleanup * old_chain = make_cleanup ( null_cleanup , 0 ) ;
sos = build_so_list_from_mapfile ( PIDGET ( inferior_pid ) , 0 , 0 ) ;
for ( so = sos ; so ! = NULL ; so = so - > next )
{
if ( so - > lm_info - > mapping [ MT_READONLY ] . addr < = addr
& & addr < = so - > lm_info - > mapping [ MT_READONLY ] . addr
+ so - > lm_info - > mapping [ MT_READONLY ] . size )
{
global_pointer = so - > lm_info - > mapping [ MT_READWRITE ] . gp ;
break ;
}
}
do_cleanups ( old_chain ) ;
return global_pointer ;
}
/* Find the execute-only kernel region known as the gate page. This
page is where the signal trampoline lives . It may be found by
querying the map file and looking for the MA_KERNTEXT flag . */
static void
aix5_find_gate_addresses ( CORE_ADDR * start , CORE_ADDR * end )
{
struct so_list * so ;
struct cleanup * old_chain = make_cleanup ( null_cleanup , 0 ) ;
/* Fetch the mappings for the main executable from the map file. */
so = build_so_list_from_mapfile ( PIDGET ( inferior_pid ) ,
MA_KERNTEXT , MA_KERNTEXT ) ;
/* Make sure we actually have some mappings to work with. */
if ( so = = NULL )
{
warning ( " Could not find gate page in map file " ) ;
* start = 0 ;
* end = 0 ;
do_cleanups ( old_chain ) ;
return ;
}
/* There should only be on kernel mapping for the gate page and
it ' ll be in the read - only ( even though it ' s execute - only )
mapping in the lm_info struct . */
* start = so - > lm_info - > mapping [ MT_READONLY ] . addr ;
* end = * start + so - > lm_info - > mapping [ MT_READONLY ] . size ;
/* Free up all the space we've allocated. */
do_cleanups ( old_chain ) ;
}
/* From ia64-tdep.c. FIXME: If we end up using this for rs6000 too,
we ' ll need to make the names match . */
extern CORE_ADDR ( * native_find_global_pointer ) ( CORE_ADDR ) ;
/* From ia64-aix-tdep.c. Hook for finding the starting and
ending gate page addresses . The only reason that this hook
is in this file is because this is where the map file reading
code is located . */
extern void ( * aix5_find_gate_addresses_hook ) ( CORE_ADDR * , CORE_ADDR * ) ;
static struct target_so_ops aix5_so_ops ;
void
_initialize_aix5_solib ( void )
{
aix5_so_ops . relocate_section_addresses = aix5_relocate_section_addresses ;
aix5_so_ops . free_so = aix5_free_so ;
aix5_so_ops . clear_solib = aix5_clear_solib ;
aix5_so_ops . solib_create_inferior_hook = aix5_solib_create_inferior_hook ;
aix5_so_ops . special_symbol_handling = aix5_special_symbol_handling ;
aix5_so_ops . current_sos = aix5_current_sos ;
aix5_so_ops . open_symbol_file_object = open_symbol_file_object ;
2001-02-22 03:01:27 +00:00
aix5_so_ops . in_dynsym_resolve_code = aix5_in_dynsym_resolve_code ;
2001-02-20 20:43:12 +00:00
native_find_global_pointer = aix5_find_global_pointer ;
aix5_find_gate_addresses_hook = aix5_find_gate_addresses ;
/* FIXME: Don't do this here. *_gdbarch_init() should set so_ops. */
current_target_so_ops = & aix5_so_ops ;
}