/* 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 */