|  | /* Target-dependent code for OpenVMS IA-64. | 
|  |  | 
|  | Copyright (C) 2012-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-unwind.h" | 
|  | #include "ia64-tdep.h" | 
|  | #include "osabi.h" | 
|  | #include "gdbtypes.h" | 
|  | #include "gdbcore.h" | 
|  | #include "gdbarch.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) | 
|  | { | 
|  | bfd_endian byte_order = gdbarch_byte_order (current_inferior ()->arch ()); | 
|  | 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 (current_inferior ()->top_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, | 
|  | (gdb_byte *) 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 accessor 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_ieee_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 | 
|  | } | 
|  |  | 
|  | INIT_GDB_FILE (ia64_vms_tdep) | 
|  | { | 
|  | gdbarch_register_osabi (bfd_arch_ia64, 0, GDB_OSABI_OPENVMS, | 
|  | ia64_openvms_init_abi); | 
|  | } |