/* dw2gencfi.h - Support for generating Dwarf2 CFI information. Copyright (C) 2003-2025 Free Software Foundation, Inc. Contributed by Michal Ludvig This file is part of GAS, the GNU Assembler. GAS 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, or (at your option) any later version. GAS 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 GAS; see the file COPYING. If not, write to the Free Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */ #ifndef DW2GENCFI_H #define DW2GENCFI_H #include "dwarf2.h" struct symbol; struct fde_entry; extern int all_cfi_sections; extern const pseudo_typeS cfi_pseudo_table[]; #ifndef tc_cfi_frame_initial_instructions #define tc_cfi_frame_initial_instructions() ((void)0) #endif #ifndef tc_cfi_startproc # define tc_cfi_startproc() ((void)0) #endif #ifndef tc_cfi_endproc # define tc_cfi_endproc(fde) ((void) (fde)) #endif /* Parse CFI assembler directive .cfi_sections. This is an external function because SCFI functionality also uses the same implementation. */ extern void dot_cfi_sections (int); /* cfi_finish() is called at the end of file. It will complain if the last CFI wasn't properly closed by .cfi_endproc. */ extern void cfi_finish (void); /* Entry points for backends to add unwind information. */ extern void cfi_new_fde (struct symbol *); extern void cfi_end_fde (struct symbol *); extern void cfi_set_last_fde (struct fde_entry *fde); extern void cfi_set_return_column (unsigned); extern void cfi_set_sections (void); extern void cfi_add_advance_loc (struct symbol *); extern void cfi_add_label (const char *); extern void cfi_add_CFA_offset (unsigned, offsetT); extern void cfi_add_CFA_val_offset (unsigned, offsetT); extern void cfi_add_CFA_def_cfa (unsigned, offsetT); extern void cfi_add_CFA_register (unsigned, unsigned); extern void cfi_add_CFA_def_cfa_register (unsigned); extern void cfi_add_CFA_def_cfa_offset (offsetT); extern void cfi_add_CFA_restore (unsigned); extern void cfi_add_CFA_undefined (unsigned); extern void cfi_add_CFA_same_value (unsigned); extern void cfi_add_CFA_remember_state (void); extern void cfi_add_CFA_restore_state (void); /* Structures for md_cfi_end. */ #if defined (TE_PE) || defined (TE_PEP) #define SUPPORT_FRAME_LINKONCE 1 #else #define SUPPORT_FRAME_LINKONCE 0 #endif #ifdef tc_cfi_reloc_for_encoding #define SUPPORT_COMPACT_EH 1 #else #define SUPPORT_COMPACT_EH 0 #endif #ifndef TARGET_MULTIPLE_EH_FRAME_SECTIONS #define TARGET_MULTIPLE_EH_FRAME_SECTIONS 0 #endif #define MULTIPLE_FRAME_SECTIONS (SUPPORT_FRAME_LINKONCE || SUPPORT_COMPACT_EH \ || TARGET_MULTIPLE_EH_FRAME_SECTIONS) struct cfi_insn_data { struct cfi_insn_data *next; #if MULTIPLE_FRAME_SECTIONS segT cur_seg; #endif int insn; union { struct { unsigned reg; offsetT offset; } ri; struct { unsigned reg1; unsigned reg2; } rr; unsigned r; offsetT i; struct { symbolS *lab1; symbolS *lab2; } ll; struct cfi_escape_data *esc; struct { unsigned reg, encoding; expressionS exp; } ea; const char *sym_name; } u; }; /* An enumeration describing the Compact EH header format. The least significant bit is used to distinguish the entries. Inline Compact: Function offset [0] Four chars of unwind data. Out-of-line Compact: Function offset [1] Compact unwind data offset [0] Legacy: Function offset [1] Unwind data offset [1] The header type is initialized to EH_COMPACT_UNKNOWN until the format is discovered by encountering a .fde_data entry. Failure to find a .fde_data entry will cause an EH_COMPACT_LEGACY header to be generated. */ enum { EH_COMPACT_UNKNOWN, EH_COMPACT_LEGACY, EH_COMPACT_INLINE, EH_COMPACT_OUTLINE, EH_COMPACT_OUTLINE_DONE, /* Outline if .cfi_inline_lsda used, otherwise legacy FDE. */ EH_COMPACT_HAS_LSDA }; /* Stack of old CFI data, for save/restore. */ struct cfa_save_data { struct cfa_save_data *next; offsetT cfa_offset; }; /* Current open FDE entry. */ struct frch_cfi_data { struct fde_entry *cur_fde_data; symbolS *last_address; offsetT cur_cfa_offset; struct cfa_save_data *cfa_save_stack; }; struct fde_entry { struct fde_entry *next; #if MULTIPLE_FRAME_SECTIONS segT cur_seg; #endif symbolS *start_address; symbolS *end_address; struct cfi_insn_data *data; struct cfi_insn_data **last; unsigned char per_encoding; unsigned char lsda_encoding; int personality_id; expressionS personality; expressionS lsda; unsigned int return_column; unsigned int signal_frame; #if MULTIPLE_FRAME_SECTIONS int handled; #endif int eh_header_type; /* Compact unwinding opcodes, not including the PR byte or LSDA. */ int eh_data_size; bfd_byte *eh_data; /* For out of line tables and FDEs. */ symbolS *eh_loc; int sections; #ifdef tc_fde_entry_extras tc_fde_entry_extras #endif }; /* The list of all FDEs that have been collected. */ extern struct fde_entry *all_fde_data; /* Fake CFI type; outside the byte range of any real CFI insn. */ #define CFI_adjust_cfa_offset 0x100 #define CFI_return_column 0x101 #define CFI_rel_offset 0x102 #define CFI_escape 0x103 #define CFI_signal_frame 0x104 #define CFI_val_encoded_addr 0x105 #define CFI_label 0x106 /* By default emit .eh_frame only, not .debug_frame. */ #define CFI_EMIT_eh_frame (1 << 0) #define CFI_EMIT_debug_frame (1 << 1) #define CFI_EMIT_target (1 << 2) #define CFI_EMIT_eh_frame_compact (1 << 3) #define CFI_EMIT_sframe (1 << 4) #endif /* DW2GENCFI_H */