diff options
Diffstat (limited to 'bfd/elf-hppa.h')
-rw-r--r-- | bfd/elf-hppa.h | 540 |
1 files changed, 540 insertions, 0 deletions
diff --git a/bfd/elf-hppa.h b/bfd/elf-hppa.h index 44d4f51..c90078d 100644 --- a/bfd/elf-hppa.h +++ b/bfd/elf-hppa.h @@ -19,15 +19,50 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #define ELF_HOWTO_TABLE_SIZE R_PARISC_UNIMPLEMENTED + 1 +/* This file is included by multiple PA ELF BFD backends with different + sizes. + + Most of the routines are written to be size independent, but sometimes + external constraints require 32 or 64 bit specific code. We remap + the definitions/functions as necessary here. */ #if ARCH_SIZE == 64 #define ELF_R_TYPE(X) ELF64_R_TYPE(X) +#define ELF_R_SYM(X) ELF64_R_SYM(X) #define _bfd_elf_hppa_gen_reloc_type _bfd_elf64_hppa_gen_reloc_type +#define elf_hppa_relocate_section elf64_hppa_relocate_section +#define bfd_elf_bfd_final_link bfd_elf64_bfd_final_link +#define elf_hppa_final_link elf64_hppa_final_link #endif #if ARCH_SIZE == 32 #define ELF_R_TYPE(X) ELF32_R_TYPE(X) +#define ELF_R_SYM(X) ELF32_R_SYM(X) #define _bfd_elf_hppa_gen_reloc_type _bfd_elf32_hppa_gen_reloc_type +#define elf_hppa_relocate_section elf32_hppa_relocate_section +#define bfd_elf_bfd_final_link bfd_elf32_bfd_final_link +#define elf_hppa_final_link elf32_hppa_final_link #endif +static boolean +elf_hppa_relocate_section + PARAMS ((bfd *, struct bfd_link_info *, bfd *, asection *, + bfd_byte *, Elf_Internal_Rela *, Elf_Internal_Sym *, asection **)); + +static bfd_reloc_status_type elf_hppa_final_link_relocate + PARAMS ((reloc_howto_type *, bfd *, bfd *, asection *, + bfd_byte *, bfd_vma, bfd_vma, bfd_vma, struct bfd_link_info *, + asection *, const char *, int)); + +static unsigned long elf_hppa_relocate_insn + PARAMS ((bfd *, asection *, unsigned long, unsigned long, long, + long, unsigned long, unsigned long, unsigned long)); + +static boolean elf_hppa_add_symbol_hook + PARAMS ((bfd *, struct bfd_link_info *, const Elf_Internal_Sym, + const char **, flagword *, asection **, bfd_vma *)); + +static boolean elf_hppa_final_link + PARAMS ((bfd *, struct bfd_link_info *)); + /* ELF/PA relocation howto entries. */ static reloc_howto_type elf_hppa_howto_table[ELF_HOWTO_TABLE_SIZE] = @@ -667,3 +702,508 @@ elf_hppa_fake_sections (abfd, hdr, sec) return true; } +/* Hook called by the linker routine which adds symbols from an object + file. HP's libraries define symbols with HP specific section + indices, which we have to handle. */ + +static boolean +elf_hppa_add_symbol_hook (abfd, info, sym, namep, flagsp, secp, valp) + bfd *abfd; + struct bfd_link_info *info ATTRIBUTE_UNUSED; + const Elf_Internal_Sym *sym; + const char **namep ATTRIBUTE_UNUSED; + flagword *flagsp ATTRIBUTE_UNUSED; + asection **secp; + bfd_vma *valp; +{ + int index = sym->st_shndx; + + switch (index) + { + case SHN_PARISC_ANSI_COMMON: + *secp = bfd_make_section_old_way (abfd, ".PARISC.ansi.common"); + (*secp)->flags |= SEC_IS_COMMON; + *valp = sym->st_size; + break; + + case SHN_PARISC_HUGE_COMMON: + *secp = bfd_make_section_old_way (abfd, ".PARISC.huge.common"); + (*secp)->flags |= SEC_IS_COMMON; + *valp = sym->st_size; + break; + } + + return true; +} + +/* Called after we have seen all the input files/sections, but before + final symbol resolution and section placement has been determined. + + We use this hook to (possibly) provide a value for __gp, then we + fall back to the generic ELF final link routine. */ + +static boolean +elf_hppa_final_link (abfd, info) + bfd *abfd; + struct bfd_link_info *info; +{ + /* Make sure we've got ourselves a suitable __gp value. */ + if (!info->relocateable) + { + bfd_vma min_short_vma = (bfd_vma -1), max_short_vma = 0; + struct elf_link_hash_entry *gp; + bfd_vma gp_val = 0; + asection *os; + + /* Find the min and max vma of all short sections. */ + for (os = abfd->sections; os ; os = os->next) + { + bfd_vma lo, hi; + + if ((os->flags & SEC_ALLOC) == 0) + continue; + + lo = os->vma; + hi = os->vma + os->_raw_size; + if (hi < lo) + hi = (bfd_vma) -1; + + /* This would be cleaner if we marked sections with an attribute + indicating they are short sections. */ + if (strcmp (os->name, ".sbss") == 0 + || strcmp (os->name, ".sdata") == 0) + { + if (min_short_vma > lo) + min_short_vma = lo; + if (max_short_vma < hi) + max_short_vma = hi; + } + } + + /* See if the user wants to force a value. */ + gp = elf_link_hash_lookup (elf_hash_table (info), "__gp", false, + false, false); + + if (gp + && (gp->root.type == bfd_link_hash_defined + || gp->root.type == bfd_link_hash_defweak)) + { + asection *gp_sec = gp->root.u.def.section; + gp_val = (gp->root.u.def.value + + gp_sec->output_section->vma + + gp_sec->output_offset); + } + else if (max_short_vma != 0) + { + /* Pick a sensible value. */ + gp_val = min_short_vma; + + /* If we don't cover all the short data, adjust. */ + if (max_short_vma - gp_val >= 0x2000) + gp_val = min_short_vma + 0x2000; + + /* If we're addressing stuff past the end, adjust back. */ + if (gp_val > max_vma) + gp_val = max_vma - 0x2000 + 8; + + /* If there was no __gp symbol, create one. */ + if (!gp) + gp = elf_link_hash_lookup (elf_hash_table (info), "__gp", true, + true, false); + + /* We now know the value for the global pointer, figure out which + section to shove it into and mark it as defined in the hash + table. */ + for (os = abfd->sections; os ; os = os->next) + { + bfd_vma low = os->output_offset + os->output_section->vma; + bfd_vma high = low + os->_raw_size; + + if (gp_val >= low && gp_val <= high) + { + gp->root.type = bfd_link_hash_defined; + gp->root.u.def.section = os; + gp->root.u.def.value = gp_val - low; + break; + } + } + } + + /* Validate whether all short sections are within + range of the chosen GP. */ + + if (max_short_vma != 0) + { + if (max_short_vma - min_short_vma >= 0x4000) + { + (*_bfd_error_handler) + (_("%s: short data segment overflowed (0x%lx >= 0x4000)"), + bfd_get_filename (abfd), + (unsigned long)(max_short_vma - min_short_vma)); + return false; + } + else if ((gp_val > min_short_vma + && gp_val - min_short_vma > 0x2000) + || (gp_val < max_short_vma + && max_short_vma - gp_val >= 0x2000)) + { + (*_bfd_error_handler) + (_("%s: __gp does not cover short data segment"), + bfd_get_filename (abfd)); + return false; + } + } + + _bfd_set_gp_value (abfd, gp_val); + } + + /* Invoke the regular ELF backend linker to do all the work. */ + return bfd_elf_bfd_final_link (abfd, info); +} + +/* Relocate an HPPA ELF section. */ + +static boolean +elf_hppa_relocate_section (output_bfd, info, input_bfd, input_section, + contents, relocs, local_syms, local_sections) + bfd *output_bfd; + struct bfd_link_info *info; + bfd *input_bfd; + asection *input_section; + bfd_byte *contents; + Elf_Internal_Rela *relocs; + Elf_Internal_Sym *local_syms; + asection **local_sections; +{ + Elf_Internal_Shdr *symtab_hdr; + Elf_Internal_Rela *rel; + Elf_Internal_Rela *relend; + + symtab_hdr = &elf_tdata (input_bfd)->symtab_hdr; + + rel = relocs; + relend = relocs + input_section->reloc_count; + for (; rel < relend; rel++) + { + int r_type; + reloc_howto_type *howto; + unsigned long r_symndx; + struct elf_link_hash_entry *h; + Elf_Internal_Sym *sym; + asection *sym_sec; + bfd_vma relocation; + bfd_reloc_status_type r; + const char *sym_name; + + r_type = ELF_R_TYPE (rel->r_info); + if (r_type < 0 || r_type >= (int) R_PARISC_UNIMPLEMENTED) + { + bfd_set_error (bfd_error_bad_value); + return false; + } + howto = elf_hppa_howto_table + r_type; + + r_symndx = ELF_R_SYM (rel->r_info); + + if (info->relocateable) + { + /* This is a relocateable link. We don't have to change + anything, unless the reloc is against a section symbol, + in which case we have to adjust according to where the + section symbol winds up in the output section. */ + if (r_symndx < symtab_hdr->sh_info) + { + sym = local_syms + r_symndx; + if (ELF_ST_TYPE (sym->st_info) == STT_SECTION) + { + sym_sec = local_sections[r_symndx]; + rel->r_addend += sym_sec->output_offset; + } + } + + continue; + } + + /* This is a final link. */ + h = NULL; + sym = NULL; + sym_sec = NULL; + if (r_symndx < symtab_hdr->sh_info) + { + sym = local_syms + r_symndx; + sym_sec = local_sections[r_symndx]; + relocation = ((ELF_ST_TYPE (sym->st_info) == STT_SECTION + ? 0 : sym->st_value) + + sym_sec->output_offset + + sym_sec->output_section->vma); + } + else + { + long indx; + + indx = r_symndx - symtab_hdr->sh_info; + h = elf_sym_hashes (input_bfd)[indx]; + while (h->root.type == bfd_link_hash_indirect + || h->root.type == bfd_link_hash_warning) + h = (struct elf_link_hash_entry *) h->root.u.i.link; + if (h->root.type == bfd_link_hash_defined + || h->root.type == bfd_link_hash_defweak) + { + sym_sec = h->root.u.def.section; + relocation = (h->root.u.def.value + + sym_sec->output_offset + + sym_sec->output_section->vma); + } + else if (h->root.type == bfd_link_hash_undefweak) + relocation = 0; + else + { + if (!((*info->callbacks->undefined_symbol) + (info, h->root.root.string, input_bfd, + input_section, rel->r_offset))) + return false; + break; + } + } + + if (h != NULL) + sym_name = h->root.root.string; + else + { + sym_name = bfd_elf_string_from_elf_section (input_bfd, + symtab_hdr->sh_link, + sym->st_name); + if (sym_name == NULL) + return false; + if (*sym_name == '\0') + sym_name = bfd_section_name (input_bfd, sym_sec); + } + + r = elf_hppa_final_link_relocate (howto, input_bfd, output_bfd, + input_section, contents, + rel->r_offset, relocation, + rel->r_addend, info, sym_sec, + sym_name, h == NULL); + + if (r != bfd_reloc_ok) + { + switch (r) + { + default: + abort (); + case bfd_reloc_overflow: + { + if (!((*info->callbacks->reloc_overflow) + (info, sym_name, howto->name, (bfd_vma) 0, + input_bfd, input_section, rel->r_offset))) + return false; + } + break; + } + } + } + return true; +} + + +/* Actually perform a relocation as part of a final link. */ + +static bfd_reloc_status_type +elf_hppa_final_link_relocate (howto, input_bfd, output_bfd, + input_section, contents, offset, value, + addend, info, sym_sec, sym_name, is_local) + reloc_howto_type *howto; + bfd *input_bfd; + bfd *output_bfd ATTRIBUTE_UNUSED; + asection *input_section; + bfd_byte *contents; + bfd_vma offset; + bfd_vma value; + bfd_vma addend; + struct bfd_link_info *info; + asection *sym_sec; + const char *sym_name; + int is_local; +{ + unsigned long insn; + unsigned long r_type = howto->type; + unsigned long r_format = howto->bitsize; + unsigned long r_field = e_fsel; + bfd_byte *hit_data = contents + offset; + boolean r_pcrel = howto->pc_relative; + + insn = bfd_get_32 (input_bfd, hit_data); + +/* For reference here a quick summary of the relocations found in the + HPUX 11.00 PA64 .o and .a files, but not yet implemented. This is mostly + a guide to help prioritize what relocation support is worked on first. + The list will be deleted eventually. + + 27210 R_PARISC_SEGREL32 + 8458 R_PARISC_DLTREL14R + 8284 R_PARISC_DLTIND21L + 8218 R_PARISC_DLTIND14DR + 6675 R_PARISC_FPTR64 + 6561 R_PARISC_DLTREL21L + 3974 R_PARISC_DIR64 + 3715 R_PARISC_DLTREL14DR + 1584 R_PARISC_LTOFF_FPTR14DR + 1565 R_PARISC_LTOFF_FPTR21L + 1120 R_PARISC_PCREL64 + 1096 R_PARISC_LTOFF_TP14DR + 982 R_PARISC_LTOFF_TP21L + 791 R_PARISC_GPREL64 + 772 R_PARISC_PLTOFF14DR + 386 R_PARISC_PLTOFF21L + 77 R_PARISC_DLTREL14WR + 6 R_PARISC_LTOFF64 + 5 R_PARISC_SEGREL64 + 1 R_PARISC_DLTIND14R + 1 R_PARISC_PCREL21L + 1 R_PARISC_PCREL14R */ + + switch (r_type) + { + case R_PARISC_NONE: + break; + + case R_PARISC_PCREL22F: + case R_PARISC_PCREL17F: + { + bfd_vma location; + r_field = e_fsel; + + /* Find out where we are and where we're going. */ + location = (offset + + input_section->output_offset + + input_section->output_section->vma); + + insn = elf_hppa_relocate_insn (input_bfd, input_section, insn, + offset, value, addend, r_format, + r_field, r_pcrel); + break; + } + + /* Something we don't know how to handle. */ + default: + /* ?!? This is temporary as we flesh out basic linker support, once + the basic support is functional we will return the not_supported + error conditional appropriately. */ +#if 0 + return bfd_reloc_not_supported; +#else + return bfd_reloc_ok; +#endif + } + + /* Update the instruction word. */ + bfd_put_32 (input_bfd, insn, hit_data); + return (bfd_reloc_ok); +} + +/* Relocate the given INSN given the various input parameters. */ + +static unsigned long +elf_hppa_relocate_insn (abfd, input_sect, insn, address, sym_value, + r_addend, r_format, r_field, pcrel) + bfd *abfd; + asection *input_sect; + unsigned long insn; + unsigned long address; + long sym_value; + long r_addend; + unsigned long r_format; + unsigned long r_field; + unsigned long pcrel; +{ + unsigned char opcode = get_opcode (insn); + long constant_value; + + switch (opcode) + { + /* This is any 17 or 22bit PC-relative branch. In PA2.0 syntax it + corresponds to the "B" instruction. */ + case BL: + /* Turn SYM_VALUE into a proper PC relative address. */ + sym_value -= (address + input_sect->output_offset + + input_sect->output_section->vma); + + /* Adjust for any field selectors. */ + sym_value = hppa_field_adjust (sym_value, -8, r_field); + + /* All PC relative branches are implicitly shifted by 2 places. */ + sym_value >>= 2; + + /* Now determine if this is a 17 or 22 bit branch and take + appropriate action. */ + if (((insn >> 13) & 0x7) == 0x4 + || ((insn >> 13) & 0x7) == 0x5) + { + unsigned int w3, w2, w1, w; + + /* These are 22 bit branches. Mask off bits we do not care + about. */ + sym_value &= 0x3fffff; + + /* Now extract the W1, W2, W3 and W fields from the value. */ + dis_assemble_22 (sym_value, &w3, &w1, &w2, &w); + + /* Mask out bits for the value in the instruction. */ + insn &= 0xfc00e002; + + /* Insert the bits for the W1, W2 and W fields into the + instruction. */ + insn |= (w3 << 21) | (w2 << 2) | (w1 << 16) | w; + return insn; + } + else + { + unsigned int w2, w1, w; + /* These are 17 bit branches. Mask off bits we do not care + about. */ + sym_value &= 0x1ffff; + + /* Now extract the W1, W2 and W fields from the value. */ + dis_assemble_17 (sym_value, &w1, &w2, &w); + + /* Mask out bits for the value in the instruction. */ + insn &= 0xffe0e002; + + /* Insert the bits for the W1, W2 and W fields into the + instruction. */ + insn |= (w2 << 2) | (w1 << 16) | w; + return insn; + } + + /* This corresponds to any 17 bit absolute branch. */ + case BE: + case BLE: + { + unsigned int w2, w1, w; + + /* Adjust for any field selectors. */ + sym_value = hppa_field_adjust (sym_value, 0, r_field); + + /* All absolute branches are implicitly shifted by 2 places. */ + sym_value >>= 2; + + /* These are 17 bit branches. Mask off bits we do not care + about. */ + sym_value &= 0x1ffff; + + /* Now extract the W1, W2 and W fields from the value. */ + dis_assemble_17 (sym_value, &w1, &w2, &w); + + /* Mask out bits for the value in the instruction. */ + insn &= 0xffe0e002; + + /* Insert the bits for the W1, W2 and W fields into the + instruction. */ + insn |= (w2 << 2) | (w1 << 16) | w; + return insn; + } + + default: + return insn; + } +} |