| /* BFD XCOFF object file private structure. | 
 |    Copyright (C) 2001-2023 Free Software Foundation, Inc. | 
 |    Written by Tom Rix, Redhat. | 
 |  | 
 |    This file is part of BFD, the Binary File Descriptor library. | 
 |  | 
 |    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, write to the Free Software | 
 |    Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, | 
 |    MA 02110-1301, USA.  */ | 
 |  | 
 | #ifndef LIBXCOFF_H | 
 | #define LIBXCOFF_H | 
 |  | 
 | /* This is the backend information kept for XCOFF files.  This | 
 |    structure is constant for a particular backend.  The first element | 
 |    is the COFF backend data structure, so that XCOFF targets can use | 
 |    the generic COFF code.  */ | 
 |  | 
 | struct xcoff_backend_data_rec | 
 | { | 
 |   /* COFF backend information.  */ | 
 |   bfd_coff_backend_data coff; | 
 |  | 
 |   /* Magic number.  */ | 
 |   unsigned short _xcoff_magic_number; | 
 |  | 
 |   /* Architecture and machine for coff_set_arch_mach_hook.  */ | 
 |   enum bfd_architecture _xcoff_architecture; | 
 |   long _xcoff_machine; | 
 |  | 
 |   /* Function pointers to xcoff specific swap routines.  */ | 
 |   void (* _xcoff_swap_ldhdr_in) (bfd *, const void *, struct internal_ldhdr *); | 
 |   void (* _xcoff_swap_ldhdr_out)(bfd *, const struct internal_ldhdr *, void *); | 
 |   void (* _xcoff_swap_ldsym_in) (bfd *, const void *, struct internal_ldsym *); | 
 |   void (* _xcoff_swap_ldsym_out)(bfd *, const struct internal_ldsym *, void *); | 
 |   void (* _xcoff_swap_ldrel_in) (bfd *, const void *, struct internal_ldrel *); | 
 |   void (* _xcoff_swap_ldrel_out)(bfd *, const struct internal_ldrel *, void *); | 
 |  | 
 |   /* Size of the external struct.  */ | 
 |   unsigned int _xcoff_ldhdrsz; | 
 |   unsigned int _xcoff_ldsymsz; | 
 |   unsigned int _xcoff_ldrelsz; | 
 |  | 
 |   /* Size an entry in a descriptor section.  */ | 
 |   unsigned int _xcoff_function_descriptor_size; | 
 |  | 
 |   /* Size of the small aout file header.  */ | 
 |   unsigned int _xcoff_small_aout_header_size; | 
 |  | 
 |   /* Loader version | 
 |      1 : XCOFF32 | 
 |      2 : XCOFF64.  */ | 
 |   unsigned long _xcoff_ldhdr_version; | 
 |  | 
 |   bool (* _xcoff_put_symbol_name) | 
 |     (struct bfd_link_info *, struct bfd_strtab_hash *, | 
 |      struct internal_syment *, const char *); | 
 |  | 
 |   bool (* _xcoff_put_ldsymbol_name) | 
 |     (bfd *, struct xcoff_loader_info *, struct internal_ldsym *, | 
 |      const char *); | 
 |  | 
 |   reloc_howto_type *_xcoff_dynamic_reloc; | 
 |  | 
 |   asection * (* _xcoff_create_csect_from_smclas) | 
 |     (bfd *, union internal_auxent *, const char *); | 
 |  | 
 |   /* Line number and relocation overflow. | 
 |      XCOFF32 overflows to another section when the line number or the | 
 |      relocation count exceeds 0xffff.  XCOFF64 does not overflow.  */ | 
 |   bool (*_xcoff_is_lineno_count_overflow) (bfd *, bfd_vma); | 
 |   bool (*_xcoff_is_reloc_count_overflow)  (bfd *, bfd_vma); | 
 |  | 
 |   /* Loader section symbol and relocation table offset | 
 |      XCOFF32 is after the .loader header | 
 |      XCOFF64 is offset in .loader header.  */ | 
 |   bfd_vma (*_xcoff_loader_symbol_offset) (bfd *, struct internal_ldhdr *); | 
 |   bfd_vma (*_xcoff_loader_reloc_offset)  (bfd *, struct internal_ldhdr *); | 
 |  | 
 |   /* Global linkage.  The first word of global linkage code must be be | 
 |      modified by filling in the correct TOC offset.  */ | 
 |   const unsigned long *_xcoff_glink_code; | 
 |  | 
 |   /* Size of the global link code in bytes of the xcoff_glink_code table.  */ | 
 |   unsigned long _xcoff_glink_size; | 
 |  | 
 |   /* rtinit.  */ | 
 |   unsigned int _xcoff_rtinit_size; | 
 |   bool (*_xcoff_generate_rtinit) | 
 |     (bfd *, const char *, const char *, bool); | 
 |  | 
 |   /* Stubs code generation. | 
 |      The code part is an array which might need to be modified by | 
 |      some relocations. | 
 |      The size is in bytes.  */ | 
 |   const unsigned long *_xcoff_stub_indirect_call_code; | 
 |   unsigned long _xcoff_stub_indirect_call_size; | 
 |   const unsigned long *_xcoff_stub_shared_call_code; | 
 |   unsigned long _xcoff_stub_shared_call_size; | 
 | }; | 
 |  | 
 | /* Look up an entry in an XCOFF link hash table.  */ | 
 | #define xcoff_link_hash_lookup(table, string, create, copy, follow) \ | 
 |   ((struct xcoff_link_hash_entry *) \ | 
 |    bfd_link_hash_lookup (&(table)->root, (string), (create), (copy),\ | 
 | 			 (follow))) | 
 |  | 
 | /* Traverse an XCOFF link hash table.  */ | 
 | #define xcoff_link_hash_traverse(table, func, info)			\ | 
 |   (bfd_link_hash_traverse						\ | 
 |    (&(table)->root,							\ | 
 |     (bool (*) (struct bfd_link_hash_entry *, void *)) (func),		\ | 
 |     (info))) | 
 |  | 
 | /* Get the XCOFF link hash table from the info structure.  This is | 
 |    just a cast.  */ | 
 | #define xcoff_hash_table(p) ((struct xcoff_link_hash_table *) ((p)->hash)) | 
 |  | 
 |  | 
 | #define xcoff_backend(abfd) \ | 
 |   ((struct xcoff_backend_data_rec *) (abfd)->xvec->backend_data) | 
 |  | 
 | #define bfd_xcoff_magic_number(a) ((xcoff_backend (a)->_xcoff_magic_number)) | 
 | #define bfd_xcoff_architecture(a) ((xcoff_backend (a)->_xcoff_architecture)) | 
 | #define bfd_xcoff_machine(a)      ((xcoff_backend (a)->_xcoff_machine)) | 
 |  | 
 | #define bfd_xcoff_swap_ldhdr_in(a, b, c) \ | 
 |   ((xcoff_backend (a)->_xcoff_swap_ldhdr_in) ((a), (b), (c))) | 
 |  | 
 | #define bfd_xcoff_swap_ldhdr_out(a, b, c) \ | 
 |   ((xcoff_backend (a)->_xcoff_swap_ldhdr_out) ((a), (b), (c))) | 
 |  | 
 | #define bfd_xcoff_swap_ldsym_in(a, b, c) \ | 
 |   ((xcoff_backend (a)->_xcoff_swap_ldsym_in) ((a), (b), (c))) | 
 |  | 
 | #define bfd_xcoff_swap_ldsym_out(a, b, c) \ | 
 |   ((xcoff_backend (a)->_xcoff_swap_ldsym_out) ((a), (b), (c))) | 
 |  | 
 | #define bfd_xcoff_swap_ldrel_in(a, b, c) \ | 
 |   ((xcoff_backend (a)->_xcoff_swap_ldrel_in) ((a), (b), (c))) | 
 |  | 
 | #define bfd_xcoff_swap_ldrel_out(a, b, c) \ | 
 |   ((xcoff_backend (a)->_xcoff_swap_ldrel_out) ((a), (b), (c))) | 
 |  | 
 | #define bfd_xcoff_ldhdrsz(a) ((xcoff_backend (a)->_xcoff_ldhdrsz)) | 
 | #define bfd_xcoff_ldsymsz(a) ((xcoff_backend (a)->_xcoff_ldsymsz)) | 
 | #define bfd_xcoff_ldrelsz(a) ((xcoff_backend (a)->_xcoff_ldrelsz)) | 
 | #define bfd_xcoff_function_descriptor_size(a) \ | 
 |   ((xcoff_backend (a)->_xcoff_function_descriptor_size)) | 
 | #define bfd_xcoff_small_aout_header_size(a) \ | 
 |   ((xcoff_backend (a)->_xcoff_small_aout_header_size)) | 
 |  | 
 | #define bfd_xcoff_ldhdr_version(a) ((xcoff_backend (a)->_xcoff_ldhdr_version)) | 
 |  | 
 | #define bfd_xcoff_put_symbol_name(a, b, c, d, e) \ | 
 |   ((xcoff_backend (a)->_xcoff_put_symbol_name) ((b), (c), (d), (e))) | 
 |  | 
 | #define bfd_xcoff_put_ldsymbol_name(a, b, c, d) \ | 
 |   ((xcoff_backend (a)->_xcoff_put_ldsymbol_name) ((a), (b), (c), (d))) | 
 |  | 
 | /* Get the XCOFF hash table entries for a BFD.  */ | 
 | #define obj_xcoff_sym_hashes(bfd) \ | 
 |   ((struct xcoff_link_hash_entry **) obj_coff_sym_hashes (bfd)) | 
 |  | 
 | #define bfd_xcoff_dynamic_reloc_howto(a) \ | 
 |    ((xcoff_backend (a)->_xcoff_dynamic_reloc)) | 
 |  | 
 | #define bfd_xcoff_create_csect_from_smclas(a, b, c) \ | 
 |    ((xcoff_backend (a)->_xcoff_create_csect_from_smclas((a), (b), (c)))) | 
 |  | 
 | #define bfd_xcoff_is_lineno_count_overflow(a, b) \ | 
 |    ((xcoff_backend (a)->_xcoff_is_lineno_count_overflow((a), (b)))) | 
 |  | 
 | #define bfd_xcoff_is_reloc_count_overflow(a, b) \ | 
 |    ((xcoff_backend (a)->_xcoff_is_reloc_count_overflow((a), (b)))) | 
 |  | 
 | #define bfd_xcoff_loader_symbol_offset(a, b) \ | 
 |  ((xcoff_backend (a)->_xcoff_loader_symbol_offset((a), (b)))) | 
 |  | 
 | #define bfd_xcoff_loader_reloc_offset(a, b) \ | 
 |  ((xcoff_backend (a)->_xcoff_loader_reloc_offset((a), (b)))) | 
 |  | 
 | #define bfd_xcoff_glink_code(a, b)   ((xcoff_backend (a)->_xcoff_glink_code[(b)])) | 
 | #define bfd_xcoff_glink_code_size(a) ((xcoff_backend (a)->_xcoff_glink_size)) | 
 |  | 
 | #define bfd_xcoff_stub_indirect_call_code(a, b)   ((xcoff_backend (a)->_xcoff_stub_indirect_call_code[(b)])) | 
 | #define bfd_xcoff_stub_indirect_call_size(a) ((xcoff_backend (a)->_xcoff_stub_indirect_call_size)) | 
 | #define bfd_xcoff_stub_shared_call_code(a, b)   ((xcoff_backend (a)->_xcoff_stub_shared_call_code[(b)])) | 
 | #define bfd_xcoff_stub_shared_call_size(a) ((xcoff_backend (a)->_xcoff_stub_shared_call_size)) | 
 |  | 
 | /* Check for the magic number U803XTOCMAGIC or U64_TOCMAGIC for 64 bit | 
 |    targets.  */ | 
 | #define bfd_xcoff_is_xcoff64(a) \ | 
 |   (   (0x01EF == (bfd_xcoff_magic_number (a))) \ | 
 |    || (0x01F7 == (bfd_xcoff_magic_number (a)))) | 
 |  | 
 | /* Check for the magic number U802TOMAGIC for 32 bit targets.  */ | 
 | #define bfd_xcoff_is_xcoff32(a) (0x01DF == (bfd_xcoff_magic_number (a))) | 
 |  | 
 | #define bfd_xcoff_rtinit_size(a)	      ((xcoff_backend (a)->_xcoff_rtinit_size)) | 
 | #define bfd_xcoff_generate_rtinit(a, b, c, d) ((xcoff_backend (a)->_xcoff_generate_rtinit ((a), (b), (c), (d)))) | 
 |  | 
 | /* Accessor macros for tdata.  */ | 
 | #define bfd_xcoff_text_align_power(a) ((xcoff_data (a)->text_align_power)) | 
 | #define bfd_xcoff_data_align_power(a) ((xcoff_data (a)->data_align_power)) | 
 |  | 
 | /* xcoff*_ppc_relocate_section macros  */ | 
 | #define XCOFF_MAX_CALCULATE_RELOCATION (0x32) | 
 | #define XCOFF_MAX_COMPLAIN_OVERFLOW (4) | 
 | /* N_ONES produces N one bits, without overflowing machine arithmetic.  */ | 
 | #ifdef N_ONES | 
 | #undef N_ONES | 
 | #endif | 
 | #define N_ONES(n) (((((bfd_vma) 1 << ((n) - 1)) - 1) << 1) | 1) | 
 |  | 
 | typedef bool xcoff_reloc_function (bfd *, asection *, bfd *, | 
 | 				   struct internal_reloc *, | 
 | 				   struct internal_syment *, | 
 | 				   struct reloc_howto_struct *, | 
 | 				   bfd_vma, bfd_vma, | 
 | 				   bfd_vma *, bfd_byte *, | 
 | 				   struct bfd_link_info *); | 
 |  | 
 | typedef bool xcoff_complain_function (bfd *, bfd_vma, bfd_vma, | 
 | 					     struct reloc_howto_struct *); | 
 |  | 
 | extern xcoff_reloc_function *const xcoff_calculate_relocation[]; | 
 | extern xcoff_complain_function *const xcoff_complain_overflow[]; | 
 |  | 
 | #define XCOFF_NO_LONG_SECTION_NAMES  (false), bfd_coff_set_long_section_names_disallowed | 
 |  | 
 | /* Relocation functions */ | 
 | extern xcoff_reloc_function xcoff_reloc_type_noop; | 
 | extern xcoff_reloc_function xcoff_reloc_type_fail; | 
 | extern xcoff_reloc_function xcoff_reloc_type_pos; | 
 | extern xcoff_reloc_function xcoff_reloc_type_neg; | 
 | extern xcoff_reloc_function xcoff_reloc_type_rel; | 
 | extern xcoff_reloc_function xcoff_reloc_type_toc; | 
 | extern xcoff_reloc_function xcoff_reloc_type_ba; | 
 | extern xcoff_reloc_function xcoff_reloc_type_crel; | 
 | extern xcoff_reloc_function xcoff_reloc_type_tls; | 
 |  | 
 | /* Structure to describe dwarf sections. | 
 |    Useful to convert from XCOFF section name to flag and vice-versa. | 
 |    Also mark if section has a length field at the beginning.  */ | 
 | struct xcoff_dwsect_name { | 
 |   /* A XCOFF dwarf section is identified by its name.  */ | 
 |   unsigned int flag; | 
 |  | 
 |   /* Corresponding XCOFF section name.  */ | 
 |   const char *xcoff_name; | 
 |  | 
 |   /* Corresponding DWARF section name.  */ | 
 |   const char *dwarf_name; | 
 |  | 
 |   /* True if size must be prepended.  */ | 
 |   bool def_size; | 
 | }; | 
 |  | 
 | /* Number of entries in the array.  The number is known and public so that user | 
 |    can 'extend' this array by index.  */ | 
 | #define XCOFF_DWSECT_NBR_NAMES	11 | 
 |  | 
 | /* The dwarf sections array.  */ | 
 | extern const struct xcoff_dwsect_name | 
 |   xcoff_dwsect_names[XCOFF_DWSECT_NBR_NAMES]; | 
 |  | 
 | /* Structure and functions needed by backend in order to handle | 
 |    stubs created in xcofflink.c.  */ | 
 |  | 
 | enum xcoff_stub_type | 
 |   { | 
 |     xcoff_stub_none, | 
 |     xcoff_stub_indirect_call, | 
 |     xcoff_stub_shared_call | 
 |   }; | 
 |  | 
 | struct xcoff_stub_hash_entry | 
 | { | 
 |   /* Base hash table entry structure.  */ | 
 |   struct bfd_hash_entry root; | 
 |  | 
 |   enum xcoff_stub_type stub_type; | 
 |  | 
 |   /* The hash table entry of the stub's csect.  */ | 
 |   struct xcoff_link_hash_entry *hcsect; | 
 |  | 
 |   /* Offset in the stub's csect.  */ | 
 |   bfd_vma stub_offset; | 
 |  | 
 |   /* The target's section.  */ | 
 |   asection *target_section; | 
 |  | 
 |   /* The target's hash table entry.  */ | 
 |   struct xcoff_link_hash_entry *htarget; | 
 | }; | 
 |  | 
 |  | 
 | extern enum xcoff_stub_type bfd_xcoff_type_of_stub | 
 |   (asection *, const struct internal_reloc *, bfd_vma, | 
 |    struct xcoff_link_hash_entry *); | 
 |  | 
 | extern struct xcoff_stub_hash_entry *bfd_xcoff_get_stub_entry | 
 |   (asection *, struct xcoff_link_hash_entry *, struct bfd_link_info *); | 
 |  | 
 | #endif /* LIBXCOFF_H */ |