| /* Definitions for frame address handler, for GDB, the GNU debugger. | 
 |  | 
 |    Copyright (C) 2003-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-base.h" | 
 | #include "frame.h" | 
 | #include "gdbsupport/gdb_obstack.h" | 
 | #include "gdbarch.h" | 
 |  | 
 | /* A default frame base implementations.  If it wasn't for the old | 
 |    DEPRECATED_FRAME_LOCALS_ADDRESS and DEPRECATED_FRAME_ARGS_ADDRESS, | 
 |    these could be combined into a single function.  All architectures | 
 |    really need to override this.  */ | 
 |  | 
 | static CORE_ADDR | 
 | default_frame_base_address (const frame_info_ptr &this_frame, void **this_cache) | 
 | { | 
 |   return get_frame_base (this_frame); /* sigh! */ | 
 | } | 
 |  | 
 | static CORE_ADDR | 
 | default_frame_locals_address (const frame_info_ptr &this_frame, void **this_cache) | 
 | { | 
 |   return default_frame_base_address (this_frame, this_cache); | 
 | } | 
 |  | 
 | static CORE_ADDR | 
 | default_frame_args_address (const frame_info_ptr &this_frame, void **this_cache) | 
 | { | 
 |   return default_frame_base_address (this_frame, this_cache); | 
 | } | 
 |  | 
 | const struct frame_base default_frame_base = { | 
 |   NULL, /* No parent.  */ | 
 |   default_frame_base_address, | 
 |   default_frame_locals_address, | 
 |   default_frame_args_address | 
 | }; | 
 |  | 
 | struct frame_base_table_entry | 
 | { | 
 |   frame_base_sniffer_ftype *sniffer; | 
 |   struct frame_base_table_entry *next; | 
 | }; | 
 |  | 
 | struct frame_base_table | 
 | { | 
 |   struct frame_base_table_entry *head = nullptr; | 
 |   struct frame_base_table_entry **tail = &head; | 
 |   const struct frame_base *default_base = &default_frame_base; | 
 | }; | 
 |  | 
 | static const registry<gdbarch>::key<struct frame_base_table> frame_base_data; | 
 |  | 
 | static struct frame_base_table * | 
 | get_frame_base_table (struct gdbarch *gdbarch) | 
 | { | 
 |   struct frame_base_table *table = frame_base_data.get (gdbarch); | 
 |   if (table == nullptr) | 
 |     table = frame_base_data.emplace (gdbarch); | 
 |   return table; | 
 | } | 
 |  | 
 | void | 
 | frame_base_append_sniffer (struct gdbarch *gdbarch, | 
 | 			   frame_base_sniffer_ftype *sniffer) | 
 | { | 
 |   struct frame_base_table *table = get_frame_base_table (gdbarch); | 
 |  | 
 |   (*table->tail) | 
 |     = GDBARCH_OBSTACK_ZALLOC (gdbarch, struct frame_base_table_entry); | 
 |   (*table->tail)->sniffer = sniffer; | 
 |   table->tail = &(*table->tail)->next; | 
 | } | 
 |  | 
 | void | 
 | frame_base_set_default (struct gdbarch *gdbarch, | 
 | 			const struct frame_base *default_base) | 
 | { | 
 |   struct frame_base_table *table = get_frame_base_table (gdbarch); | 
 |  | 
 |   table->default_base = default_base; | 
 | } | 
 |  | 
 | const struct frame_base * | 
 | frame_base_find_by_frame (const frame_info_ptr &this_frame) | 
 | { | 
 |   struct gdbarch *gdbarch = get_frame_arch (this_frame); | 
 |   struct frame_base_table *table = get_frame_base_table (gdbarch); | 
 |   struct frame_base_table_entry *entry; | 
 |  | 
 |   for (entry = table->head; entry != NULL; entry = entry->next) | 
 |     { | 
 |       const struct frame_base *desc = NULL; | 
 |  | 
 |       desc = entry->sniffer (this_frame); | 
 |       if (desc != NULL) | 
 | 	return desc; | 
 |     } | 
 |   return table->default_base; | 
 | } |