| Index: gdb/ia64-vms-tdep.c
|
| diff --git a/gdb/ia64-vms-tdep.c b/gdb/ia64-vms-tdep.c
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..09238fcb3d1fe801af5afe4c06bcb9440a1e47a4
|
| --- /dev/null
|
| +++ b/gdb/ia64-vms-tdep.c
|
| @@ -0,0 +1,165 @@
|
| +/* Target-dependent code for OpenVMS IA-64.
|
| +
|
| + Copyright (C) 2012 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 "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)
|
| +{
|
| + enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch);
|
| + 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;
|
| +
|
| +/* Set of ia64-libunwind-tdep gdb callbacks and data for generic
|
| + ia64-libunwind-tdep code to use. */
|
| +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. */
|
| +extern initialize_file_ftype _initialize_ia64_vms_tdep;
|
| +
|
| +void
|
| +_initialize_ia64_vms_tdep (void)
|
| +{
|
| + gdbarch_register_osabi (bfd_arch_ia64, 0, GDB_OSABI_OPENVMS,
|
| + ia64_openvms_init_abi);
|
| +}
|
|
|