2012-03-05 12:41:51 +01:00
|
|
|
/* Target-dependent code for OpenVMS IA-64.
|
|
|
|
|
2013-01-01 07:33:28 +01:00
|
|
|
Copyright (C) 2012-2013 Free Software Foundation, Inc.
|
2012-03-05 12:41:51 +01: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
|
|
|
|
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 "defs.h"
|
|
|
|
#include "frame-unwind.h"
|
|
|
|
#include "ia64-tdep.h"
|
|
|
|
#include "osabi.h"
|
|
|
|
#include "gdbtypes.h"
|
|
|
|
#include "gdbcore.h"
|
|
|
|
|
|
|
|
#ifdef HAVE_LIBUNWIND_IA64_H
|
|
|
|
|
|
|
|
/* Libunwind callback accessor function to acquire procedure unwind-info. */
|
|
|
|
|
|
|
|
static int
|
|
|
|
ia64_vms_find_proc_info_x (unw_addr_space_t as, unw_word_t ip,
|
|
|
|
unw_proc_info_t *pi,
|
|
|
|
int need_unwind_info, void *arg)
|
|
|
|
{
|
* gdbarch.sh (target_gdbarch): Remove macro.
(get_target_gdbarch): Rename to target_gdbarch.
* gdbarch.c, gdbarch.h: Rebuild.
* ada-tasks.c, aix-thread.c, amd64-linux-nat.c, arch-utils.c,
arm-tdep.c, auxv.c, breakpoint.c, bsd-uthread.c, corefile.c,
darwin-nat-info.c, dcache.c, dsrec.c, exec.c, fbsd-nat.c,
filesystem.c, gcore.c, gnu-nat.c, i386-darwin-nat.c, i386-nat.c,
ia64-vms-tdep.c, inf-ptrace.c, infcmd.c, jit.c, linux-nat.c,
linux-tdep.c, linux-thread-db.c, m32r-rom.c, memattr.c,
mep-tdep.c, microblaze-tdep.c, mips-linux-nat.c,
mips-linux-tdep.c, mips-tdep.c, monitor.c, moxie-tdep.c,
nto-procfs.c, nto-tdep.c, ppc-linux-nat.c, proc-service.c,
procfs.c, progspace.c, ravenscar-thread.c, record.c,
remote-m32r-sdi.c, remote-mips.c, remote-sim.c, remote.c,
rl78-tdep.c, rs6000-nat.c, rx-tdep.c, s390-nat.c, sol-thread.c,
solib-darwin.c, solib-dsbt.c, solib-frv.c, solib-ia64-hpux.c,
solib-irix.c, solib-pa64.c, solib-som.c, solib-spu.c,
solib-sunos.c, solib-svr4.c, solib.c, spu-linux-nat.c,
spu-multiarch.c, spu-tdep.c, symfile-mem.c, symfile.c, symtab.c,
target-descriptions.c, target.c, target.h, tracepoint.c,
windows-nat.c, windows-tdep.c, xcoffsolib.c, cli/cli-dump.c,
common/agent.c, mi/mi-interp.c, python/py-finishbreakpoint.c,
python/py-inferior.c, python/python.c: Update.
2012-11-09 20:58:03 +01:00
|
|
|
enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ());
|
2012-03-05 12:41:51 +01:00
|
|
|
unw_dyn_info_t di;
|
|
|
|
int ret;
|
|
|
|
gdb_byte buf[32];
|
|
|
|
const char *annex = core_addr_to_string (ip);
|
|
|
|
LONGEST res;
|
|
|
|
CORE_ADDR table_addr;
|
|
|
|
unsigned int info_len;
|
|
|
|
|
|
|
|
res = target_read (¤t_target, TARGET_OBJECT_OPENVMS_UIB,
|
|
|
|
annex + 2, buf, 0, sizeof (buf));
|
|
|
|
|
|
|
|
if (res != sizeof (buf))
|
|
|
|
return -UNW_ENOINFO;
|
|
|
|
|
|
|
|
pi->format = UNW_INFO_FORMAT_REMOTE_TABLE;
|
|
|
|
pi->start_ip = extract_unsigned_integer (buf + 0, 8, byte_order);
|
|
|
|
pi->end_ip = extract_unsigned_integer (buf + 8, 8, byte_order);
|
|
|
|
pi->gp = extract_unsigned_integer (buf + 24, 8, byte_order);
|
|
|
|
table_addr = extract_unsigned_integer (buf + 16, 8, byte_order);
|
|
|
|
|
|
|
|
if (table_addr == 0)
|
|
|
|
{
|
|
|
|
/* No unwind data. */
|
|
|
|
pi->unwind_info = NULL;
|
|
|
|
pi->unwind_info_size = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = target_read_memory (table_addr, buf, 8);
|
|
|
|
if (res != 0)
|
|
|
|
return -UNW_ENOINFO;
|
|
|
|
|
|
|
|
/* Check version. */
|
|
|
|
if (extract_unsigned_integer (buf + 6, 2, byte_order) != 1)
|
|
|
|
return -UNW_EBADVERSION;
|
|
|
|
info_len = extract_unsigned_integer (buf + 0, 4, byte_order);
|
|
|
|
pi->unwind_info_size = 8 * info_len;
|
|
|
|
|
|
|
|
/* Read info. */
|
|
|
|
pi->unwind_info = xmalloc (pi->unwind_info_size);
|
|
|
|
|
|
|
|
res = target_read_memory (table_addr + 8,
|
|
|
|
pi->unwind_info, pi->unwind_info_size);
|
|
|
|
if (res != 0)
|
|
|
|
{
|
|
|
|
xfree (pi->unwind_info);
|
|
|
|
pi->unwind_info = NULL;
|
|
|
|
return -UNW_ENOINFO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Handle OSSD (OS Specific Data). This extension to ia64 unwind
|
|
|
|
information by OpenVMS is currently not handled by libunwind, but
|
|
|
|
looks to be used only in very specific context, and is not generated by
|
|
|
|
GCC. */
|
|
|
|
|
|
|
|
pi->lsda = table_addr + 8 + pi->unwind_info_size;
|
|
|
|
if (extract_unsigned_integer (buf + 4, 2, byte_order) & 3)
|
|
|
|
{
|
|
|
|
pi->lsda += 8;
|
|
|
|
/* There might be an handler, but this is not used for unwinding. */
|
|
|
|
pi->handler = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Libunwind callback accessor function for cleanup. */
|
|
|
|
|
|
|
|
static void
|
|
|
|
ia64_vms_put_unwind_info (unw_addr_space_t as,
|
|
|
|
unw_proc_info_t *pip, void *arg)
|
|
|
|
{
|
|
|
|
/* Nothing required for now. */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Libunwind callback accessor function to get head of the dynamic
|
|
|
|
unwind-info registration list. */
|
|
|
|
|
|
|
|
static int
|
|
|
|
ia64_vms_get_dyn_info_list (unw_addr_space_t as,
|
|
|
|
unw_word_t *dilap, void *arg)
|
|
|
|
{
|
|
|
|
return -UNW_ENOINFO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set of libunwind callback acccessor functions. */
|
|
|
|
static unw_accessors_t ia64_vms_unw_accessors;
|
|
|
|
static unw_accessors_t ia64_vms_unw_rse_accessors;
|
|
|
|
|
2012-03-08 20:08:11 +01:00
|
|
|
/* Set of ia64-libunwind-tdep gdb callbacks and data for generic
|
|
|
|
ia64-libunwind-tdep code to use. */
|
2012-03-05 12:41:51 +01:00
|
|
|
static struct libunwind_descr ia64_vms_libunwind_descr;
|
|
|
|
|
|
|
|
#endif /* HAVE_LIBUNWIND_IA64_H */
|
|
|
|
|
|
|
|
static void
|
|
|
|
ia64_openvms_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
|
|
|
|
{
|
|
|
|
set_gdbarch_long_double_format (gdbarch, floatformats_ia64_quad);
|
|
|
|
|
|
|
|
#ifdef HAVE_LIBUNWIND_IA64_H
|
|
|
|
/* Override the default descriptor. */
|
|
|
|
ia64_vms_unw_accessors = ia64_unw_accessors;
|
|
|
|
ia64_vms_unw_accessors.find_proc_info = ia64_vms_find_proc_info_x;
|
|
|
|
ia64_vms_unw_accessors.put_unwind_info = ia64_vms_put_unwind_info;
|
|
|
|
ia64_vms_unw_accessors.get_dyn_info_list_addr = ia64_vms_get_dyn_info_list;
|
|
|
|
|
|
|
|
ia64_vms_unw_rse_accessors = ia64_unw_rse_accessors;
|
|
|
|
ia64_vms_unw_rse_accessors.find_proc_info = ia64_vms_find_proc_info_x;
|
|
|
|
ia64_vms_unw_rse_accessors.put_unwind_info = ia64_vms_put_unwind_info;
|
|
|
|
ia64_vms_unw_rse_accessors.get_dyn_info_list_addr = ia64_vms_get_dyn_info_list;
|
|
|
|
|
|
|
|
ia64_vms_libunwind_descr = ia64_libunwind_descr;
|
|
|
|
ia64_vms_libunwind_descr.accessors = &ia64_vms_unw_accessors;
|
|
|
|
ia64_vms_libunwind_descr.special_accessors = &ia64_vms_unw_rse_accessors;
|
|
|
|
|
|
|
|
libunwind_frame_set_descr (gdbarch, &ia64_vms_libunwind_descr);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Provide a prototype to silence -Wmissing-prototypes. */
|
2012-03-05 15:53:56 +01:00
|
|
|
extern initialize_file_ftype _initialize_ia64_vms_tdep;
|
2012-03-05 12:41:51 +01:00
|
|
|
|
|
|
|
void
|
2012-03-05 15:53:56 +01:00
|
|
|
_initialize_ia64_vms_tdep (void)
|
2012-03-05 12:41:51 +01:00
|
|
|
{
|
|
|
|
gdbarch_register_osabi (bfd_arch_ia64, 0, GDB_OSABI_OPENVMS,
|
|
|
|
ia64_openvms_init_abi);
|
|
|
|
}
|