2004-02-18 00:23:23 +08:00
|
|
|
/* Target-dependent code for the ia64.
|
|
|
|
|
2022-01-01 22:56:03 +08:00
|
|
|
Copyright (C) 2004-2022 Free Software Foundation, Inc.
|
2004-02-18 00:23:23 +08:00
|
|
|
|
|
|
|
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
|
2007-08-24 02:08:50 +08:00
|
|
|
the Free Software Foundation; either version 3 of the License, or
|
2004-02-18 00:23:23 +08:00
|
|
|
(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
|
2007-08-24 02:08:50 +08:00
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
2004-02-18 00:23:23 +08:00
|
|
|
|
|
|
|
#ifndef IA64_TDEP_H
|
|
|
|
#define IA64_TDEP_H
|
|
|
|
|
gdb: fix gdbarch_tdep ODR violation
I would like to be able to use non-trivial types in gdbarch_tdep types.
This is not possible at the moment (in theory), because of the one
definition rule.
To allow it, rename all gdbarch_tdep types to <arch>_gdbarch_tdep, and
make them inherit from a gdbarch_tdep base class. The inheritance is
necessary to be able to pass pointers to all these <arch>_gdbarch_tdep
objects to gdbarch_alloc, which takes a pointer to gdbarch_tdep.
These objects are never deleted through a base class pointer, so I
didn't include a virtual destructor. In the future, if gdbarch objects
deletable, I could imagine that the gdbarch_tdep objects could become
owned by the gdbarch objects, and then it would become useful to have a
virtual destructor (so that the gdbarch object can delete the owned
gdbarch_tdep object). But that's not necessary right now.
It turns out that RISC-V already has a gdbarch_tdep that is
non-default-constructible, so that provides a good motivation for this
change.
Most changes are fairly straightforward, mostly needing to add some
casts all over the place. There is however the xtensa architecture,
doing its own little weird thing to define its gdbarch_tdep. I did my
best to adapt it, but I can't test those changes.
Change-Id: Ic001903f91ddd106bd6ca09a79dabe8df2d69f3b
2021-11-16 00:29:39 +08:00
|
|
|
#include "gdbarch.h"
|
|
|
|
|
2012-03-05 19:41:51 +08:00
|
|
|
#ifdef HAVE_LIBUNWIND_IA64_H
|
|
|
|
#include "libunwind-ia64.h"
|
2012-03-09 03:08:11 +08:00
|
|
|
#include "ia64-libunwind-tdep.h"
|
2012-03-05 19:41:51 +08:00
|
|
|
#endif
|
|
|
|
|
2005-01-08 07:12:06 +08:00
|
|
|
/* Register numbers of various important registers. */
|
|
|
|
|
|
|
|
/* General registers; there are 128 of these 64 bit wide registers.
|
|
|
|
The first 32 are static and the last 96 are stacked. */
|
|
|
|
#define IA64_GR0_REGNUM 0
|
|
|
|
#define IA64_GR1_REGNUM (IA64_GR0_REGNUM + 1)
|
|
|
|
#define IA64_GR2_REGNUM (IA64_GR0_REGNUM + 2)
|
|
|
|
#define IA64_GR3_REGNUM (IA64_GR0_REGNUM + 3)
|
|
|
|
#define IA64_GR4_REGNUM (IA64_GR0_REGNUM + 4)
|
|
|
|
#define IA64_GR5_REGNUM (IA64_GR0_REGNUM + 5)
|
|
|
|
#define IA64_GR6_REGNUM (IA64_GR0_REGNUM + 6)
|
|
|
|
#define IA64_GR7_REGNUM (IA64_GR0_REGNUM + 7)
|
|
|
|
#define IA64_GR8_REGNUM (IA64_GR0_REGNUM + 8)
|
|
|
|
#define IA64_GR9_REGNUM (IA64_GR0_REGNUM + 9)
|
|
|
|
#define IA64_GR10_REGNUM (IA64_GR0_REGNUM + 10)
|
|
|
|
#define IA64_GR11_REGNUM (IA64_GR0_REGNUM + 11)
|
|
|
|
#define IA64_GR12_REGNUM (IA64_GR0_REGNUM + 12)
|
|
|
|
#define IA64_GR31_REGNUM (IA64_GR0_REGNUM + 31)
|
|
|
|
#define IA64_GR32_REGNUM (IA64_GR0_REGNUM + 32)
|
|
|
|
#define IA64_GR127_REGNUM (IA64_GR0_REGNUM + 127)
|
|
|
|
|
|
|
|
/* Floating point registers; 128 82-bit wide registers. */
|
|
|
|
#define IA64_FR0_REGNUM 128
|
|
|
|
#define IA64_FR1_REGNUM (IA64_FR0_REGNUM + 1)
|
|
|
|
#define IA64_FR2_REGNUM (IA64_FR0_REGNUM + 2)
|
|
|
|
#define IA64_FR8_REGNUM (IA64_FR0_REGNUM + 8)
|
|
|
|
#define IA64_FR9_REGNUM (IA64_FR0_REGNUM + 9)
|
|
|
|
#define IA64_FR10_REGNUM (IA64_FR0_REGNUM + 10)
|
|
|
|
#define IA64_FR11_REGNUM (IA64_FR0_REGNUM + 11)
|
|
|
|
#define IA64_FR12_REGNUM (IA64_FR0_REGNUM + 12)
|
|
|
|
#define IA64_FR13_REGNUM (IA64_FR0_REGNUM + 13)
|
|
|
|
#define IA64_FR14_REGNUM (IA64_FR0_REGNUM + 14)
|
|
|
|
#define IA64_FR15_REGNUM (IA64_FR0_REGNUM + 15)
|
|
|
|
#define IA64_FR16_REGNUM (IA64_FR0_REGNUM + 16)
|
|
|
|
#define IA64_FR31_REGNUM (IA64_FR0_REGNUM + 31)
|
|
|
|
#define IA64_FR32_REGNUM (IA64_FR0_REGNUM + 32)
|
|
|
|
#define IA64_FR127_REGNUM (IA64_FR0_REGNUM + 127)
|
|
|
|
|
|
|
|
/* Predicate registers; There are 64 of these one bit registers. It'd
|
|
|
|
be more convenient (implementation-wise) to use a single 64 bit
|
|
|
|
word with all of these register in them. Note that there's also a
|
|
|
|
IA64_PR_REGNUM below which contains all the bits and is used for
|
|
|
|
communicating the actual values to the target. */
|
|
|
|
#define IA64_PR0_REGNUM 256
|
|
|
|
#define IA64_PR1_REGNUM (IA64_PR0_REGNUM + 1)
|
|
|
|
#define IA64_PR2_REGNUM (IA64_PR0_REGNUM + 2)
|
|
|
|
#define IA64_PR3_REGNUM (IA64_PR0_REGNUM + 3)
|
|
|
|
#define IA64_PR4_REGNUM (IA64_PR0_REGNUM + 4)
|
|
|
|
#define IA64_PR5_REGNUM (IA64_PR0_REGNUM + 5)
|
|
|
|
#define IA64_PR6_REGNUM (IA64_PR0_REGNUM + 6)
|
|
|
|
#define IA64_PR7_REGNUM (IA64_PR0_REGNUM + 7)
|
|
|
|
#define IA64_PR8_REGNUM (IA64_PR0_REGNUM + 8)
|
|
|
|
#define IA64_PR9_REGNUM (IA64_PR0_REGNUM + 9)
|
|
|
|
#define IA64_PR10_REGNUM (IA64_PR0_REGNUM + 10)
|
|
|
|
#define IA64_PR11_REGNUM (IA64_PR0_REGNUM + 11)
|
|
|
|
#define IA64_PR12_REGNUM (IA64_PR0_REGNUM + 12)
|
|
|
|
#define IA64_PR13_REGNUM (IA64_PR0_REGNUM + 13)
|
|
|
|
#define IA64_PR14_REGNUM (IA64_PR0_REGNUM + 14)
|
|
|
|
#define IA64_PR15_REGNUM (IA64_PR0_REGNUM + 15)
|
|
|
|
#define IA64_PR16_REGNUM (IA64_PR0_REGNUM + 16)
|
|
|
|
#define IA64_PR17_REGNUM (IA64_PR0_REGNUM + 17)
|
|
|
|
#define IA64_PR18_REGNUM (IA64_PR0_REGNUM + 18)
|
|
|
|
#define IA64_PR19_REGNUM (IA64_PR0_REGNUM + 19)
|
|
|
|
#define IA64_PR20_REGNUM (IA64_PR0_REGNUM + 20)
|
|
|
|
#define IA64_PR21_REGNUM (IA64_PR0_REGNUM + 21)
|
|
|
|
#define IA64_PR22_REGNUM (IA64_PR0_REGNUM + 22)
|
|
|
|
#define IA64_PR23_REGNUM (IA64_PR0_REGNUM + 23)
|
|
|
|
#define IA64_PR24_REGNUM (IA64_PR0_REGNUM + 24)
|
|
|
|
#define IA64_PR25_REGNUM (IA64_PR0_REGNUM + 25)
|
|
|
|
#define IA64_PR26_REGNUM (IA64_PR0_REGNUM + 26)
|
|
|
|
#define IA64_PR27_REGNUM (IA64_PR0_REGNUM + 27)
|
|
|
|
#define IA64_PR28_REGNUM (IA64_PR0_REGNUM + 28)
|
|
|
|
#define IA64_PR29_REGNUM (IA64_PR0_REGNUM + 29)
|
|
|
|
#define IA64_PR30_REGNUM (IA64_PR0_REGNUM + 30)
|
|
|
|
#define IA64_PR31_REGNUM (IA64_PR0_REGNUM + 31)
|
|
|
|
#define IA64_PR32_REGNUM (IA64_PR0_REGNUM + 32)
|
|
|
|
#define IA64_PR33_REGNUM (IA64_PR0_REGNUM + 33)
|
|
|
|
#define IA64_PR34_REGNUM (IA64_PR0_REGNUM + 34)
|
|
|
|
#define IA64_PR35_REGNUM (IA64_PR0_REGNUM + 35)
|
|
|
|
#define IA64_PR36_REGNUM (IA64_PR0_REGNUM + 36)
|
|
|
|
#define IA64_PR37_REGNUM (IA64_PR0_REGNUM + 37)
|
|
|
|
#define IA64_PR38_REGNUM (IA64_PR0_REGNUM + 38)
|
|
|
|
#define IA64_PR39_REGNUM (IA64_PR0_REGNUM + 39)
|
|
|
|
#define IA64_PR40_REGNUM (IA64_PR0_REGNUM + 40)
|
|
|
|
#define IA64_PR41_REGNUM (IA64_PR0_REGNUM + 41)
|
|
|
|
#define IA64_PR42_REGNUM (IA64_PR0_REGNUM + 42)
|
|
|
|
#define IA64_PR43_REGNUM (IA64_PR0_REGNUM + 43)
|
|
|
|
#define IA64_PR44_REGNUM (IA64_PR0_REGNUM + 44)
|
|
|
|
#define IA64_PR45_REGNUM (IA64_PR0_REGNUM + 45)
|
|
|
|
#define IA64_PR46_REGNUM (IA64_PR0_REGNUM + 46)
|
|
|
|
#define IA64_PR47_REGNUM (IA64_PR0_REGNUM + 47)
|
|
|
|
#define IA64_PR48_REGNUM (IA64_PR0_REGNUM + 48)
|
|
|
|
#define IA64_PR49_REGNUM (IA64_PR0_REGNUM + 49)
|
|
|
|
#define IA64_PR50_REGNUM (IA64_PR0_REGNUM + 50)
|
|
|
|
#define IA64_PR51_REGNUM (IA64_PR0_REGNUM + 51)
|
|
|
|
#define IA64_PR52_REGNUM (IA64_PR0_REGNUM + 52)
|
|
|
|
#define IA64_PR53_REGNUM (IA64_PR0_REGNUM + 53)
|
|
|
|
#define IA64_PR54_REGNUM (IA64_PR0_REGNUM + 54)
|
|
|
|
#define IA64_PR55_REGNUM (IA64_PR0_REGNUM + 55)
|
|
|
|
#define IA64_PR56_REGNUM (IA64_PR0_REGNUM + 56)
|
|
|
|
#define IA64_PR57_REGNUM (IA64_PR0_REGNUM + 57)
|
|
|
|
#define IA64_PR58_REGNUM (IA64_PR0_REGNUM + 58)
|
|
|
|
#define IA64_PR59_REGNUM (IA64_PR0_REGNUM + 59)
|
|
|
|
#define IA64_PR60_REGNUM (IA64_PR0_REGNUM + 60)
|
|
|
|
#define IA64_PR61_REGNUM (IA64_PR0_REGNUM + 61)
|
|
|
|
#define IA64_PR62_REGNUM (IA64_PR0_REGNUM + 62)
|
|
|
|
#define IA64_PR63_REGNUM (IA64_PR0_REGNUM + 63)
|
|
|
|
|
|
|
|
/* Branch registers: 8 64-bit registers for holding branch targets. */
|
|
|
|
#define IA64_BR0_REGNUM 320
|
|
|
|
#define IA64_BR1_REGNUM (IA64_BR0_REGNUM + 1)
|
|
|
|
#define IA64_BR2_REGNUM (IA64_BR0_REGNUM + 2)
|
|
|
|
#define IA64_BR3_REGNUM (IA64_BR0_REGNUM + 3)
|
|
|
|
#define IA64_BR4_REGNUM (IA64_BR0_REGNUM + 4)
|
|
|
|
#define IA64_BR5_REGNUM (IA64_BR0_REGNUM + 5)
|
|
|
|
#define IA64_BR6_REGNUM (IA64_BR0_REGNUM + 6)
|
|
|
|
#define IA64_BR7_REGNUM (IA64_BR0_REGNUM + 7)
|
|
|
|
|
|
|
|
/* Virtual frame pointer; this matches IA64_FRAME_POINTER_REGNUM in
|
|
|
|
gcc/config/ia64/ia64.h. */
|
|
|
|
#define IA64_VFP_REGNUM 328
|
|
|
|
|
|
|
|
/* Virtual return address pointer; this matches
|
|
|
|
IA64_RETURN_ADDRESS_POINTER_REGNUM in gcc/config/ia64/ia64.h. */
|
|
|
|
#define IA64_VRAP_REGNUM 329
|
|
|
|
|
|
|
|
/* Predicate registers: There are 64 of these 1-bit registers. We
|
|
|
|
define a single register which is used to communicate these values
|
|
|
|
to/from the target. We will somehow contrive to make it appear
|
|
|
|
that IA64_PR0_REGNUM thru IA64_PR63_REGNUM hold the actual values. */
|
|
|
|
#define IA64_PR_REGNUM 330
|
|
|
|
|
|
|
|
/* Instruction pointer: 64 bits wide. */
|
|
|
|
#define IA64_IP_REGNUM 331
|
|
|
|
|
|
|
|
/* Process Status Register. */
|
|
|
|
#define IA64_PSR_REGNUM 332
|
|
|
|
|
|
|
|
/* Current Frame Marker (raw form may be the cr.ifs). */
|
|
|
|
#define IA64_CFM_REGNUM 333
|
|
|
|
|
|
|
|
/* Application registers; 128 64-bit wide registers possible, but some
|
|
|
|
of them are reserved. */
|
|
|
|
#define IA64_AR0_REGNUM 334
|
|
|
|
#define IA64_KR0_REGNUM (IA64_AR0_REGNUM + 0)
|
|
|
|
#define IA64_KR7_REGNUM (IA64_KR0_REGNUM + 7)
|
|
|
|
|
|
|
|
#define IA64_RSC_REGNUM (IA64_AR0_REGNUM + 16)
|
|
|
|
#define IA64_BSP_REGNUM (IA64_AR0_REGNUM + 17)
|
|
|
|
#define IA64_BSPSTORE_REGNUM (IA64_AR0_REGNUM + 18)
|
|
|
|
#define IA64_RNAT_REGNUM (IA64_AR0_REGNUM + 19)
|
|
|
|
#define IA64_FCR_REGNUM (IA64_AR0_REGNUM + 21)
|
|
|
|
#define IA64_EFLAG_REGNUM (IA64_AR0_REGNUM + 24)
|
|
|
|
#define IA64_CSD_REGNUM (IA64_AR0_REGNUM + 25)
|
|
|
|
#define IA64_SSD_REGNUM (IA64_AR0_REGNUM + 26)
|
|
|
|
#define IA64_CFLG_REGNUM (IA64_AR0_REGNUM + 27)
|
|
|
|
#define IA64_FSR_REGNUM (IA64_AR0_REGNUM + 28)
|
|
|
|
#define IA64_FIR_REGNUM (IA64_AR0_REGNUM + 29)
|
|
|
|
#define IA64_FDR_REGNUM (IA64_AR0_REGNUM + 30)
|
|
|
|
#define IA64_CCV_REGNUM (IA64_AR0_REGNUM + 32)
|
|
|
|
#define IA64_UNAT_REGNUM (IA64_AR0_REGNUM + 36)
|
|
|
|
#define IA64_FPSR_REGNUM (IA64_AR0_REGNUM + 40)
|
|
|
|
#define IA64_ITC_REGNUM (IA64_AR0_REGNUM + 44)
|
|
|
|
#define IA64_PFS_REGNUM (IA64_AR0_REGNUM + 64)
|
|
|
|
#define IA64_LC_REGNUM (IA64_AR0_REGNUM + 65)
|
|
|
|
#define IA64_EC_REGNUM (IA64_AR0_REGNUM + 66)
|
|
|
|
|
|
|
|
/* NAT (Not A Thing) Bits for the general registers; there are 128 of
|
|
|
|
these. */
|
|
|
|
#define IA64_NAT0_REGNUM 462
|
|
|
|
#define IA64_NAT31_REGNUM (IA64_NAT0_REGNUM + 31)
|
|
|
|
#define IA64_NAT32_REGNUM (IA64_NAT0_REGNUM + 32)
|
|
|
|
#define IA64_NAT127_REGNUM (IA64_NAT0_REGNUM + 127)
|
|
|
|
|
2011-01-14 00:24:27 +08:00
|
|
|
struct frame_info;
|
[ia64-hpux] inferior function call support
We have two stacks to deal with on ia64, when making a function call.
The first is the usual stack frame, and the second is the register
stack frame. On ia64-linux, the register frame is setup by adjusting
the BSP register. Unfortunately for us, the HP-UX kernel does not allow
the debugger to change the value of the BSP.
To work around that limitation, the method I am using here is to push
some assembly code on the stack. This assembly code contains, among
other things, a call to the alloc insn, which sets up our frame for us.
An extensive comment in ia64-hpux-tdep.c explains the entire procedure.
Despite this approach, most of the code in ia64-tdep.c which sets up
the function call is still applicable - and only a few things need
to be done differently: For instance, instead of changing the BSP,
we do nothing. We store the parameters at a different location, etc.
So this patch also adjusts the inf-call code in ia64-tdep.c to make it
a little more extensible: I create a new ia64_infcall_ops structure
which allows an ABI to define how the few things that need to be
differentiated.
Another element that turned out to be necessary but is more of a detail
is that the computation of the linkage pointer needs to be handled
specially for symbols inside shared libraries. This is especially
visible when calling malloc, which happens everytime memory needs to
be allocated in inferior memory... The special treatment included
again the necessity to use some routines only available on the host.
So another target object TARGET_OBJECT_HPUX_SOLIB_GOT was created for
that purpose.
gdb/ChangeLog:
* ia64-tdep.h (struct regcache): Forward declare.
(struct ia64_infcall_ops): New struct type.
(struct gdbarch_tdep): New fields "find_global_pointer_from_solib"
and "infcall_ops".
* ia64-tdep.c (ia64_find_global_pointer_from_dynamic_section):
Renames ia64_find_global_pointer.
(ia64_find_global_pointer, ia64_allocate_new_rse_frame)
(ia64_store_argument_in_slot, ia64_set_function_addr: New function.
(ia64_push_dummy_call): Adjust to use the new tdep ia64_infocall_ops
methods.
(ia64_infcall_ops): New static global constant.
(ia64_gdbarch_init): Set tdep->infcall_ops.
* ia64-hpux-nat.c (ia64_hpux_xfer_solib_got): New function.
(ia64_hpux_xfer_partial): Add TARGET_OBJECT_HPUX_SOLIB_GOT handing.
* ia64-hpux-tdep.c: Include "regcache.h", "gdbcore.h" and "inferior.h".
(ia64_hpux_dummy_code): New static global constant.
(ia64_hpux_push_dummy_code, ia64_hpux_allocate_new_rse_frame)
(ia64_hpux_store_argument_in_slot, ia64_hpux_set_function_addr)
(ia64_hpux_dummy_id, ia64_hpux_find_global_pointer_from_solib):
New function.
(ia64_hpux_infcall_ops): New static global constant.
(ia64_hpux_init_abi): Install gdbarch and tdep methods needed
for inferior function calls to work properly on ia64-hpux.
2011-01-14 00:24:42 +08:00
|
|
|
struct regcache;
|
|
|
|
|
|
|
|
/* A struction containing pointers to all the target-dependent operations
|
|
|
|
performed to setup an inferior function call. */
|
|
|
|
|
|
|
|
struct ia64_infcall_ops
|
|
|
|
{
|
|
|
|
/* Allocate a new register stack frame starting after the output
|
|
|
|
region of the current frame. The new frame will contain SOF
|
|
|
|
registers, all in the output region. This is one way of protecting
|
|
|
|
the stacked registers of the current frame.
|
|
|
|
|
|
|
|
Should do nothing if this operation is not permitted by the OS. */
|
|
|
|
void (*allocate_new_rse_frame) (struct regcache *regcache, ULONGEST bsp,
|
2021-11-19 05:35:31 +08:00
|
|
|
int sof);
|
[ia64-hpux] inferior function call support
We have two stacks to deal with on ia64, when making a function call.
The first is the usual stack frame, and the second is the register
stack frame. On ia64-linux, the register frame is setup by adjusting
the BSP register. Unfortunately for us, the HP-UX kernel does not allow
the debugger to change the value of the BSP.
To work around that limitation, the method I am using here is to push
some assembly code on the stack. This assembly code contains, among
other things, a call to the alloc insn, which sets up our frame for us.
An extensive comment in ia64-hpux-tdep.c explains the entire procedure.
Despite this approach, most of the code in ia64-tdep.c which sets up
the function call is still applicable - and only a few things need
to be done differently: For instance, instead of changing the BSP,
we do nothing. We store the parameters at a different location, etc.
So this patch also adjusts the inf-call code in ia64-tdep.c to make it
a little more extensible: I create a new ia64_infcall_ops structure
which allows an ABI to define how the few things that need to be
differentiated.
Another element that turned out to be necessary but is more of a detail
is that the computation of the linkage pointer needs to be handled
specially for symbols inside shared libraries. This is especially
visible when calling malloc, which happens everytime memory needs to
be allocated in inferior memory... The special treatment included
again the necessity to use some routines only available on the host.
So another target object TARGET_OBJECT_HPUX_SOLIB_GOT was created for
that purpose.
gdb/ChangeLog:
* ia64-tdep.h (struct regcache): Forward declare.
(struct ia64_infcall_ops): New struct type.
(struct gdbarch_tdep): New fields "find_global_pointer_from_solib"
and "infcall_ops".
* ia64-tdep.c (ia64_find_global_pointer_from_dynamic_section):
Renames ia64_find_global_pointer.
(ia64_find_global_pointer, ia64_allocate_new_rse_frame)
(ia64_store_argument_in_slot, ia64_set_function_addr: New function.
(ia64_push_dummy_call): Adjust to use the new tdep ia64_infocall_ops
methods.
(ia64_infcall_ops): New static global constant.
(ia64_gdbarch_init): Set tdep->infcall_ops.
* ia64-hpux-nat.c (ia64_hpux_xfer_solib_got): New function.
(ia64_hpux_xfer_partial): Add TARGET_OBJECT_HPUX_SOLIB_GOT handing.
* ia64-hpux-tdep.c: Include "regcache.h", "gdbcore.h" and "inferior.h".
(ia64_hpux_dummy_code): New static global constant.
(ia64_hpux_push_dummy_code, ia64_hpux_allocate_new_rse_frame)
(ia64_hpux_store_argument_in_slot, ia64_hpux_set_function_addr)
(ia64_hpux_dummy_id, ia64_hpux_find_global_pointer_from_solib):
New function.
(ia64_hpux_infcall_ops): New static global constant.
(ia64_hpux_init_abi): Install gdbarch and tdep methods needed
for inferior function calls to work properly on ia64-hpux.
2011-01-14 00:24:42 +08:00
|
|
|
|
|
|
|
/* Store the argument stored in BUF into the appropriate location
|
|
|
|
given the BSP and the SLOTNUM. */
|
|
|
|
void (*store_argument_in_slot) (struct regcache *regcache, CORE_ADDR bsp,
|
2021-11-19 05:35:31 +08:00
|
|
|
int slotnum, gdb_byte *buf);
|
[ia64-hpux] inferior function call support
We have two stacks to deal with on ia64, when making a function call.
The first is the usual stack frame, and the second is the register
stack frame. On ia64-linux, the register frame is setup by adjusting
the BSP register. Unfortunately for us, the HP-UX kernel does not allow
the debugger to change the value of the BSP.
To work around that limitation, the method I am using here is to push
some assembly code on the stack. This assembly code contains, among
other things, a call to the alloc insn, which sets up our frame for us.
An extensive comment in ia64-hpux-tdep.c explains the entire procedure.
Despite this approach, most of the code in ia64-tdep.c which sets up
the function call is still applicable - and only a few things need
to be done differently: For instance, instead of changing the BSP,
we do nothing. We store the parameters at a different location, etc.
So this patch also adjusts the inf-call code in ia64-tdep.c to make it
a little more extensible: I create a new ia64_infcall_ops structure
which allows an ABI to define how the few things that need to be
differentiated.
Another element that turned out to be necessary but is more of a detail
is that the computation of the linkage pointer needs to be handled
specially for symbols inside shared libraries. This is especially
visible when calling malloc, which happens everytime memory needs to
be allocated in inferior memory... The special treatment included
again the necessity to use some routines only available on the host.
So another target object TARGET_OBJECT_HPUX_SOLIB_GOT was created for
that purpose.
gdb/ChangeLog:
* ia64-tdep.h (struct regcache): Forward declare.
(struct ia64_infcall_ops): New struct type.
(struct gdbarch_tdep): New fields "find_global_pointer_from_solib"
and "infcall_ops".
* ia64-tdep.c (ia64_find_global_pointer_from_dynamic_section):
Renames ia64_find_global_pointer.
(ia64_find_global_pointer, ia64_allocate_new_rse_frame)
(ia64_store_argument_in_slot, ia64_set_function_addr: New function.
(ia64_push_dummy_call): Adjust to use the new tdep ia64_infocall_ops
methods.
(ia64_infcall_ops): New static global constant.
(ia64_gdbarch_init): Set tdep->infcall_ops.
* ia64-hpux-nat.c (ia64_hpux_xfer_solib_got): New function.
(ia64_hpux_xfer_partial): Add TARGET_OBJECT_HPUX_SOLIB_GOT handing.
* ia64-hpux-tdep.c: Include "regcache.h", "gdbcore.h" and "inferior.h".
(ia64_hpux_dummy_code): New static global constant.
(ia64_hpux_push_dummy_code, ia64_hpux_allocate_new_rse_frame)
(ia64_hpux_store_argument_in_slot, ia64_hpux_set_function_addr)
(ia64_hpux_dummy_id, ia64_hpux_find_global_pointer_from_solib):
New function.
(ia64_hpux_infcall_ops): New static global constant.
(ia64_hpux_init_abi): Install gdbarch and tdep methods needed
for inferior function calls to work properly on ia64-hpux.
2011-01-14 00:24:42 +08:00
|
|
|
|
|
|
|
/* For targets where we cannot call the function directly, store
|
|
|
|
the address of the function we want to call at the location
|
|
|
|
expected by the calling sequence. */
|
2021-11-19 05:35:31 +08:00
|
|
|
void (*set_function_addr) (struct regcache *regcache, CORE_ADDR func_addr);
|
[ia64-hpux] inferior function call support
We have two stacks to deal with on ia64, when making a function call.
The first is the usual stack frame, and the second is the register
stack frame. On ia64-linux, the register frame is setup by adjusting
the BSP register. Unfortunately for us, the HP-UX kernel does not allow
the debugger to change the value of the BSP.
To work around that limitation, the method I am using here is to push
some assembly code on the stack. This assembly code contains, among
other things, a call to the alloc insn, which sets up our frame for us.
An extensive comment in ia64-hpux-tdep.c explains the entire procedure.
Despite this approach, most of the code in ia64-tdep.c which sets up
the function call is still applicable - and only a few things need
to be done differently: For instance, instead of changing the BSP,
we do nothing. We store the parameters at a different location, etc.
So this patch also adjusts the inf-call code in ia64-tdep.c to make it
a little more extensible: I create a new ia64_infcall_ops structure
which allows an ABI to define how the few things that need to be
differentiated.
Another element that turned out to be necessary but is more of a detail
is that the computation of the linkage pointer needs to be handled
specially for symbols inside shared libraries. This is especially
visible when calling malloc, which happens everytime memory needs to
be allocated in inferior memory... The special treatment included
again the necessity to use some routines only available on the host.
So another target object TARGET_OBJECT_HPUX_SOLIB_GOT was created for
that purpose.
gdb/ChangeLog:
* ia64-tdep.h (struct regcache): Forward declare.
(struct ia64_infcall_ops): New struct type.
(struct gdbarch_tdep): New fields "find_global_pointer_from_solib"
and "infcall_ops".
* ia64-tdep.c (ia64_find_global_pointer_from_dynamic_section):
Renames ia64_find_global_pointer.
(ia64_find_global_pointer, ia64_allocate_new_rse_frame)
(ia64_store_argument_in_slot, ia64_set_function_addr: New function.
(ia64_push_dummy_call): Adjust to use the new tdep ia64_infocall_ops
methods.
(ia64_infcall_ops): New static global constant.
(ia64_gdbarch_init): Set tdep->infcall_ops.
* ia64-hpux-nat.c (ia64_hpux_xfer_solib_got): New function.
(ia64_hpux_xfer_partial): Add TARGET_OBJECT_HPUX_SOLIB_GOT handing.
* ia64-hpux-tdep.c: Include "regcache.h", "gdbcore.h" and "inferior.h".
(ia64_hpux_dummy_code): New static global constant.
(ia64_hpux_push_dummy_code, ia64_hpux_allocate_new_rse_frame)
(ia64_hpux_store_argument_in_slot, ia64_hpux_set_function_addr)
(ia64_hpux_dummy_id, ia64_hpux_find_global_pointer_from_solib):
New function.
(ia64_hpux_infcall_ops): New static global constant.
(ia64_hpux_init_abi): Install gdbarch and tdep methods needed
for inferior function calls to work properly on ia64-hpux.
2011-01-14 00:24:42 +08:00
|
|
|
};
|
2011-01-14 00:24:27 +08:00
|
|
|
|
2022-07-25 19:07:11 +08:00
|
|
|
struct ia64_gdbarch_tdep : gdbarch_tdep_base
|
2005-01-06 06:51:47 +08:00
|
|
|
{
|
gdb: fix gdbarch_tdep ODR violation
I would like to be able to use non-trivial types in gdbarch_tdep types.
This is not possible at the moment (in theory), because of the one
definition rule.
To allow it, rename all gdbarch_tdep types to <arch>_gdbarch_tdep, and
make them inherit from a gdbarch_tdep base class. The inheritance is
necessary to be able to pass pointers to all these <arch>_gdbarch_tdep
objects to gdbarch_alloc, which takes a pointer to gdbarch_tdep.
These objects are never deleted through a base class pointer, so I
didn't include a virtual destructor. In the future, if gdbarch objects
deletable, I could imagine that the gdbarch_tdep objects could become
owned by the gdbarch objects, and then it would become useful to have a
virtual destructor (so that the gdbarch object can delete the owned
gdbarch_tdep object). But that's not necessary right now.
It turns out that RISC-V already has a gdbarch_tdep that is
non-default-constructible, so that provides a good motivation for this
change.
Most changes are fairly straightforward, mostly needing to add some
casts all over the place. There is however the xtensa architecture,
doing its own little weird thing to define its gdbarch_tdep. I did my
best to adapt it, but I can't test those changes.
Change-Id: Ic001903f91ddd106bd6ca09a79dabe8df2d69f3b
2021-11-16 00:29:39 +08:00
|
|
|
CORE_ADDR (*sigcontext_register_address) (struct gdbarch *, CORE_ADDR, int)
|
|
|
|
= nullptr;
|
|
|
|
int (*pc_in_sigtramp) (CORE_ADDR) = nullptr;
|
* gdbtypes.h (builtin_type_ieee_single, builtin_type_ieee_double,
builtin_type_i387_ext, builtin_type_m68881_ext, builtin_type_arm_ext,
builtin_type_ia64_spill, builtin_type_ia64_quad): Remove.
(init_float_type, init_complex_type): Add prototypes.
* gdbtypes.c (builtin_type_ieee_single, builtin_type_ieee_double,
builtin_type_i387_ext, builtin_type_m68881_ext, builtin_type_arm_ext,
builtin_type_ia64_spill, builtin_type_ia64_quad): Remove.
(_initialize_gdbtypes): Do not initialize them.
(build_flt): Rename to ...
(init_float_type): ... this. Make global.
(build_complex): Rename to ...
(init_complex_type): ... this. Make global. Remove BIT argument.
(gdbtypes_post_init): Update calls.
* ada-lang.c (ada_language_arch_info): Use init_float_type.
* jv-lang.c (build_java_types): Likewise.
* m2-lang.c (build_m2_types): Likewise.
* f-lang.c (build_fortran_types): Use init_float_type and
init_complex_type.
* target-descriptions.c (tdesc_gdb_type): Call init_float_type instead
of using builtin_type_ieee_single, builtin_type_ieee_double, or
builtin_type_arm_ext.
* ia64-tdep.h (struct gdbarch_tdep): Add ia64_ext_type member.
* ia64-tdep.c (builtin_type_ia64_ext): Remove.
(_initialize_ia64_tdep): Do not initialize it.
(floatformat_valid, floatformat_ia64_ext, floatformats_ia64_ext):
Move up.
(ia64_ext_type): New function.
(ia64_register_reggroup_p, ia64_convert_register_p,
ia64_register_to_value, ia64_value_to_register,
ia64_extract_return_value, ia64_store_return_value): Use ia64_ext_type
instead of builtin_type_ia64_ext.
* i386-tdep.h (struct gdbarch_tdep): Add i387_ext_type member.
(i387_ext_type): Add prototype.
* i386-tdep.c (i387_ext_type): New function.
(i386_extract_return_value, i386_store_return_value,
i386_register_type): Use it instead of builtin_type_i387_ext.
* amd64-tdep.c (amd64_register_type): Likewise.
* i387-tdep.c (print_i387_value, i387_register_to_value,
i387_value_to_register): Likewise.
(print_i387_value, print_i387_ext): Add GDBARCH argument.
(print_i387_ext, i387_print_float_info): Pass to subroutines.
* m68k-tdep.h (struct gdbarch_tdep): Add m68881_ext_type member.
* m68k-tdep.c (m68881_ext_type): New function.
(m68k_register_type, m68k_convert_register_p): Use it instead
of builtin_type_m68881_ext.
* arm-tdep.h (struct gdbarch_tdep): Add arm_ext_type member.
* arm-tdep.c (arm_ext_type): New function.
(arm_register_type): Use it instead of builtin_type_arm_ext.
* alpha-tdep.c (alpha_register_type): Use builtin types
instead of builtin_type_ieee_double.
* mips-tdep.c (mips_float_register_type, mips_double_register_type):
Remove.
(mips_register_type): Use builtin types instead of
builtin_type_ieee_single and builtin_type_ieee_double.
(mips_print_fp_register): Use builtin types instead of
mips_float_register_type and mips_double_register_type.
* hppa-tdep.c (hppa32_register_type, hppa64_register_type):
Use builtin types instead of builtin_type_ieee_single and
builtin_type_ieee_double.
2009-07-02 20:48:54 +08:00
|
|
|
|
2011-01-14 00:24:27 +08:00
|
|
|
/* Return the total size of THIS_FRAME's register frame.
|
|
|
|
CFM is THIS_FRAME's cfm register value.
|
|
|
|
|
|
|
|
Normally, the size of the register frame is always obtained by
|
|
|
|
extracting the lowest 7 bits ("cfm & 0x7f"). */
|
gdb: fix gdbarch_tdep ODR violation
I would like to be able to use non-trivial types in gdbarch_tdep types.
This is not possible at the moment (in theory), because of the one
definition rule.
To allow it, rename all gdbarch_tdep types to <arch>_gdbarch_tdep, and
make them inherit from a gdbarch_tdep base class. The inheritance is
necessary to be able to pass pointers to all these <arch>_gdbarch_tdep
objects to gdbarch_alloc, which takes a pointer to gdbarch_tdep.
These objects are never deleted through a base class pointer, so I
didn't include a virtual destructor. In the future, if gdbarch objects
deletable, I could imagine that the gdbarch_tdep objects could become
owned by the gdbarch objects, and then it would become useful to have a
virtual destructor (so that the gdbarch object can delete the owned
gdbarch_tdep object). But that's not necessary right now.
It turns out that RISC-V already has a gdbarch_tdep that is
non-default-constructible, so that provides a good motivation for this
change.
Most changes are fairly straightforward, mostly needing to add some
casts all over the place. There is however the xtensa architecture,
doing its own little weird thing to define its gdbarch_tdep. I did my
best to adapt it, but I can't test those changes.
Change-Id: Ic001903f91ddd106bd6ca09a79dabe8df2d69f3b
2021-11-16 00:29:39 +08:00
|
|
|
int (*size_of_register_frame) (struct frame_info *this_frame, ULONGEST cfm)
|
|
|
|
= nullptr;
|
2011-01-14 00:24:27 +08:00
|
|
|
|
[ia64-hpux] inferior function call support
We have two stacks to deal with on ia64, when making a function call.
The first is the usual stack frame, and the second is the register
stack frame. On ia64-linux, the register frame is setup by adjusting
the BSP register. Unfortunately for us, the HP-UX kernel does not allow
the debugger to change the value of the BSP.
To work around that limitation, the method I am using here is to push
some assembly code on the stack. This assembly code contains, among
other things, a call to the alloc insn, which sets up our frame for us.
An extensive comment in ia64-hpux-tdep.c explains the entire procedure.
Despite this approach, most of the code in ia64-tdep.c which sets up
the function call is still applicable - and only a few things need
to be done differently: For instance, instead of changing the BSP,
we do nothing. We store the parameters at a different location, etc.
So this patch also adjusts the inf-call code in ia64-tdep.c to make it
a little more extensible: I create a new ia64_infcall_ops structure
which allows an ABI to define how the few things that need to be
differentiated.
Another element that turned out to be necessary but is more of a detail
is that the computation of the linkage pointer needs to be handled
specially for symbols inside shared libraries. This is especially
visible when calling malloc, which happens everytime memory needs to
be allocated in inferior memory... The special treatment included
again the necessity to use some routines only available on the host.
So another target object TARGET_OBJECT_HPUX_SOLIB_GOT was created for
that purpose.
gdb/ChangeLog:
* ia64-tdep.h (struct regcache): Forward declare.
(struct ia64_infcall_ops): New struct type.
(struct gdbarch_tdep): New fields "find_global_pointer_from_solib"
and "infcall_ops".
* ia64-tdep.c (ia64_find_global_pointer_from_dynamic_section):
Renames ia64_find_global_pointer.
(ia64_find_global_pointer, ia64_allocate_new_rse_frame)
(ia64_store_argument_in_slot, ia64_set_function_addr: New function.
(ia64_push_dummy_call): Adjust to use the new tdep ia64_infocall_ops
methods.
(ia64_infcall_ops): New static global constant.
(ia64_gdbarch_init): Set tdep->infcall_ops.
* ia64-hpux-nat.c (ia64_hpux_xfer_solib_got): New function.
(ia64_hpux_xfer_partial): Add TARGET_OBJECT_HPUX_SOLIB_GOT handing.
* ia64-hpux-tdep.c: Include "regcache.h", "gdbcore.h" and "inferior.h".
(ia64_hpux_dummy_code): New static global constant.
(ia64_hpux_push_dummy_code, ia64_hpux_allocate_new_rse_frame)
(ia64_hpux_store_argument_in_slot, ia64_hpux_set_function_addr)
(ia64_hpux_dummy_id, ia64_hpux_find_global_pointer_from_solib):
New function.
(ia64_hpux_infcall_ops): New static global constant.
(ia64_hpux_init_abi): Install gdbarch and tdep methods needed
for inferior function calls to work properly on ia64-hpux.
2011-01-14 00:24:42 +08:00
|
|
|
/* Determine the function address FADDR belongs to a shared library.
|
|
|
|
If it does, then return the associated global pointer. If no shared
|
|
|
|
library was found to contain that function, then return zero.
|
|
|
|
|
|
|
|
This pointer may be NULL. */
|
|
|
|
CORE_ADDR (*find_global_pointer_from_solib) (struct gdbarch *gdbarch,
|
gdb: fix gdbarch_tdep ODR violation
I would like to be able to use non-trivial types in gdbarch_tdep types.
This is not possible at the moment (in theory), because of the one
definition rule.
To allow it, rename all gdbarch_tdep types to <arch>_gdbarch_tdep, and
make them inherit from a gdbarch_tdep base class. The inheritance is
necessary to be able to pass pointers to all these <arch>_gdbarch_tdep
objects to gdbarch_alloc, which takes a pointer to gdbarch_tdep.
These objects are never deleted through a base class pointer, so I
didn't include a virtual destructor. In the future, if gdbarch objects
deletable, I could imagine that the gdbarch_tdep objects could become
owned by the gdbarch objects, and then it would become useful to have a
virtual destructor (so that the gdbarch object can delete the owned
gdbarch_tdep object). But that's not necessary right now.
It turns out that RISC-V already has a gdbarch_tdep that is
non-default-constructible, so that provides a good motivation for this
change.
Most changes are fairly straightforward, mostly needing to add some
casts all over the place. There is however the xtensa architecture,
doing its own little weird thing to define its gdbarch_tdep. I did my
best to adapt it, but I can't test those changes.
Change-Id: Ic001903f91ddd106bd6ca09a79dabe8df2d69f3b
2021-11-16 00:29:39 +08:00
|
|
|
CORE_ADDR faddr) = nullptr;
|
[ia64-hpux] inferior function call support
We have two stacks to deal with on ia64, when making a function call.
The first is the usual stack frame, and the second is the register
stack frame. On ia64-linux, the register frame is setup by adjusting
the BSP register. Unfortunately for us, the HP-UX kernel does not allow
the debugger to change the value of the BSP.
To work around that limitation, the method I am using here is to push
some assembly code on the stack. This assembly code contains, among
other things, a call to the alloc insn, which sets up our frame for us.
An extensive comment in ia64-hpux-tdep.c explains the entire procedure.
Despite this approach, most of the code in ia64-tdep.c which sets up
the function call is still applicable - and only a few things need
to be done differently: For instance, instead of changing the BSP,
we do nothing. We store the parameters at a different location, etc.
So this patch also adjusts the inf-call code in ia64-tdep.c to make it
a little more extensible: I create a new ia64_infcall_ops structure
which allows an ABI to define how the few things that need to be
differentiated.
Another element that turned out to be necessary but is more of a detail
is that the computation of the linkage pointer needs to be handled
specially for symbols inside shared libraries. This is especially
visible when calling malloc, which happens everytime memory needs to
be allocated in inferior memory... The special treatment included
again the necessity to use some routines only available on the host.
So another target object TARGET_OBJECT_HPUX_SOLIB_GOT was created for
that purpose.
gdb/ChangeLog:
* ia64-tdep.h (struct regcache): Forward declare.
(struct ia64_infcall_ops): New struct type.
(struct gdbarch_tdep): New fields "find_global_pointer_from_solib"
and "infcall_ops".
* ia64-tdep.c (ia64_find_global_pointer_from_dynamic_section):
Renames ia64_find_global_pointer.
(ia64_find_global_pointer, ia64_allocate_new_rse_frame)
(ia64_store_argument_in_slot, ia64_set_function_addr: New function.
(ia64_push_dummy_call): Adjust to use the new tdep ia64_infocall_ops
methods.
(ia64_infcall_ops): New static global constant.
(ia64_gdbarch_init): Set tdep->infcall_ops.
* ia64-hpux-nat.c (ia64_hpux_xfer_solib_got): New function.
(ia64_hpux_xfer_partial): Add TARGET_OBJECT_HPUX_SOLIB_GOT handing.
* ia64-hpux-tdep.c: Include "regcache.h", "gdbcore.h" and "inferior.h".
(ia64_hpux_dummy_code): New static global constant.
(ia64_hpux_push_dummy_code, ia64_hpux_allocate_new_rse_frame)
(ia64_hpux_store_argument_in_slot, ia64_hpux_set_function_addr)
(ia64_hpux_dummy_id, ia64_hpux_find_global_pointer_from_solib):
New function.
(ia64_hpux_infcall_ops): New static global constant.
(ia64_hpux_init_abi): Install gdbarch and tdep methods needed
for inferior function calls to work properly on ia64-hpux.
2011-01-14 00:24:42 +08:00
|
|
|
|
* gdbtypes.h (builtin_type_ieee_single, builtin_type_ieee_double,
builtin_type_i387_ext, builtin_type_m68881_ext, builtin_type_arm_ext,
builtin_type_ia64_spill, builtin_type_ia64_quad): Remove.
(init_float_type, init_complex_type): Add prototypes.
* gdbtypes.c (builtin_type_ieee_single, builtin_type_ieee_double,
builtin_type_i387_ext, builtin_type_m68881_ext, builtin_type_arm_ext,
builtin_type_ia64_spill, builtin_type_ia64_quad): Remove.
(_initialize_gdbtypes): Do not initialize them.
(build_flt): Rename to ...
(init_float_type): ... this. Make global.
(build_complex): Rename to ...
(init_complex_type): ... this. Make global. Remove BIT argument.
(gdbtypes_post_init): Update calls.
* ada-lang.c (ada_language_arch_info): Use init_float_type.
* jv-lang.c (build_java_types): Likewise.
* m2-lang.c (build_m2_types): Likewise.
* f-lang.c (build_fortran_types): Use init_float_type and
init_complex_type.
* target-descriptions.c (tdesc_gdb_type): Call init_float_type instead
of using builtin_type_ieee_single, builtin_type_ieee_double, or
builtin_type_arm_ext.
* ia64-tdep.h (struct gdbarch_tdep): Add ia64_ext_type member.
* ia64-tdep.c (builtin_type_ia64_ext): Remove.
(_initialize_ia64_tdep): Do not initialize it.
(floatformat_valid, floatformat_ia64_ext, floatformats_ia64_ext):
Move up.
(ia64_ext_type): New function.
(ia64_register_reggroup_p, ia64_convert_register_p,
ia64_register_to_value, ia64_value_to_register,
ia64_extract_return_value, ia64_store_return_value): Use ia64_ext_type
instead of builtin_type_ia64_ext.
* i386-tdep.h (struct gdbarch_tdep): Add i387_ext_type member.
(i387_ext_type): Add prototype.
* i386-tdep.c (i387_ext_type): New function.
(i386_extract_return_value, i386_store_return_value,
i386_register_type): Use it instead of builtin_type_i387_ext.
* amd64-tdep.c (amd64_register_type): Likewise.
* i387-tdep.c (print_i387_value, i387_register_to_value,
i387_value_to_register): Likewise.
(print_i387_value, print_i387_ext): Add GDBARCH argument.
(print_i387_ext, i387_print_float_info): Pass to subroutines.
* m68k-tdep.h (struct gdbarch_tdep): Add m68881_ext_type member.
* m68k-tdep.c (m68881_ext_type): New function.
(m68k_register_type, m68k_convert_register_p): Use it instead
of builtin_type_m68881_ext.
* arm-tdep.h (struct gdbarch_tdep): Add arm_ext_type member.
* arm-tdep.c (arm_ext_type): New function.
(arm_register_type): Use it instead of builtin_type_arm_ext.
* alpha-tdep.c (alpha_register_type): Use builtin types
instead of builtin_type_ieee_double.
* mips-tdep.c (mips_float_register_type, mips_double_register_type):
Remove.
(mips_register_type): Use builtin types instead of
builtin_type_ieee_single and builtin_type_ieee_double.
(mips_print_fp_register): Use builtin types instead of
mips_float_register_type and mips_double_register_type.
* hppa-tdep.c (hppa32_register_type, hppa64_register_type):
Use builtin types instead of builtin_type_ieee_single and
builtin_type_ieee_double.
2009-07-02 20:48:54 +08:00
|
|
|
/* ISA-specific data types. */
|
gdb: fix gdbarch_tdep ODR violation
I would like to be able to use non-trivial types in gdbarch_tdep types.
This is not possible at the moment (in theory), because of the one
definition rule.
To allow it, rename all gdbarch_tdep types to <arch>_gdbarch_tdep, and
make them inherit from a gdbarch_tdep base class. The inheritance is
necessary to be able to pass pointers to all these <arch>_gdbarch_tdep
objects to gdbarch_alloc, which takes a pointer to gdbarch_tdep.
These objects are never deleted through a base class pointer, so I
didn't include a virtual destructor. In the future, if gdbarch objects
deletable, I could imagine that the gdbarch_tdep objects could become
owned by the gdbarch objects, and then it would become useful to have a
virtual destructor (so that the gdbarch object can delete the owned
gdbarch_tdep object). But that's not necessary right now.
It turns out that RISC-V already has a gdbarch_tdep that is
non-default-constructible, so that provides a good motivation for this
change.
Most changes are fairly straightforward, mostly needing to add some
casts all over the place. There is however the xtensa architecture,
doing its own little weird thing to define its gdbarch_tdep. I did my
best to adapt it, but I can't test those changes.
Change-Id: Ic001903f91ddd106bd6ca09a79dabe8df2d69f3b
2021-11-16 00:29:39 +08:00
|
|
|
struct type *ia64_ext_type = nullptr;
|
[ia64-hpux] inferior function call support
We have two stacks to deal with on ia64, when making a function call.
The first is the usual stack frame, and the second is the register
stack frame. On ia64-linux, the register frame is setup by adjusting
the BSP register. Unfortunately for us, the HP-UX kernel does not allow
the debugger to change the value of the BSP.
To work around that limitation, the method I am using here is to push
some assembly code on the stack. This assembly code contains, among
other things, a call to the alloc insn, which sets up our frame for us.
An extensive comment in ia64-hpux-tdep.c explains the entire procedure.
Despite this approach, most of the code in ia64-tdep.c which sets up
the function call is still applicable - and only a few things need
to be done differently: For instance, instead of changing the BSP,
we do nothing. We store the parameters at a different location, etc.
So this patch also adjusts the inf-call code in ia64-tdep.c to make it
a little more extensible: I create a new ia64_infcall_ops structure
which allows an ABI to define how the few things that need to be
differentiated.
Another element that turned out to be necessary but is more of a detail
is that the computation of the linkage pointer needs to be handled
specially for symbols inside shared libraries. This is especially
visible when calling malloc, which happens everytime memory needs to
be allocated in inferior memory... The special treatment included
again the necessity to use some routines only available on the host.
So another target object TARGET_OBJECT_HPUX_SOLIB_GOT was created for
that purpose.
gdb/ChangeLog:
* ia64-tdep.h (struct regcache): Forward declare.
(struct ia64_infcall_ops): New struct type.
(struct gdbarch_tdep): New fields "find_global_pointer_from_solib"
and "infcall_ops".
* ia64-tdep.c (ia64_find_global_pointer_from_dynamic_section):
Renames ia64_find_global_pointer.
(ia64_find_global_pointer, ia64_allocate_new_rse_frame)
(ia64_store_argument_in_slot, ia64_set_function_addr: New function.
(ia64_push_dummy_call): Adjust to use the new tdep ia64_infocall_ops
methods.
(ia64_infcall_ops): New static global constant.
(ia64_gdbarch_init): Set tdep->infcall_ops.
* ia64-hpux-nat.c (ia64_hpux_xfer_solib_got): New function.
(ia64_hpux_xfer_partial): Add TARGET_OBJECT_HPUX_SOLIB_GOT handing.
* ia64-hpux-tdep.c: Include "regcache.h", "gdbcore.h" and "inferior.h".
(ia64_hpux_dummy_code): New static global constant.
(ia64_hpux_push_dummy_code, ia64_hpux_allocate_new_rse_frame)
(ia64_hpux_store_argument_in_slot, ia64_hpux_set_function_addr)
(ia64_hpux_dummy_id, ia64_hpux_find_global_pointer_from_solib):
New function.
(ia64_hpux_infcall_ops): New static global constant.
(ia64_hpux_init_abi): Install gdbarch and tdep methods needed
for inferior function calls to work properly on ia64-hpux.
2011-01-14 00:24:42 +08:00
|
|
|
|
2021-11-19 05:35:31 +08:00
|
|
|
struct ia64_infcall_ops infcall_ops {};
|
2005-01-06 06:51:47 +08:00
|
|
|
};
|
|
|
|
|
2007-06-16 06:44:56 +08:00
|
|
|
extern void ia64_write_pc (struct regcache *, CORE_ADDR);
|
2004-02-18 00:23:23 +08:00
|
|
|
|
2012-03-05 19:41:51 +08:00
|
|
|
#ifdef HAVE_LIBUNWIND_IA64_H
|
|
|
|
extern unw_accessors_t ia64_unw_accessors;
|
|
|
|
extern unw_accessors_t ia64_unw_rse_accessors;
|
|
|
|
extern struct libunwind_descr ia64_libunwind_descr;
|
|
|
|
#endif
|
|
|
|
|
2005-01-08 07:12:06 +08:00
|
|
|
#endif /* ia64-tdep.h */
|