318 lines
12 KiB
C
318 lines
12 KiB
C
|
/* BFD XCOFF object file private structure.
|
||
|
Copyright (C) 2001-2022 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 */
|