| /* SPARC-specific support for 32-bit ELF | 
 |    Copyright (C) 1993-2022 Free Software Foundation, Inc. | 
 |  | 
 |    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.  */ | 
 |  | 
 | #include "sysdep.h" | 
 | #include "bfd.h" | 
 | #include "bfdlink.h" | 
 | #include "libbfd.h" | 
 | #include "elf-bfd.h" | 
 | #include "elf/sparc.h" | 
 | #include "opcode/sparc.h" | 
 | #include "elfxx-sparc.h" | 
 | #include "elf-vxworks.h" | 
 |  | 
 | /* Support for core dump NOTE sections.  */ | 
 |  | 
 | static bool | 
 | elf32_sparc_grok_psinfo (bfd *abfd, Elf_Internal_Note *note) | 
 | { | 
 |   switch (note->descsz) | 
 |     { | 
 |     default: | 
 |       return false; | 
 |  | 
 |     case 260:			/* Solaris prpsinfo_t.  */ | 
 |       elf_tdata (abfd)->core->program | 
 | 	= _bfd_elfcore_strndup (abfd, note->descdata + 84, 16); | 
 |       elf_tdata (abfd)->core->command | 
 | 	= _bfd_elfcore_strndup (abfd, note->descdata + 100, 80); | 
 |       break; | 
 |  | 
 |     case 336:			/* Solaris psinfo_t.  */ | 
 |       elf_tdata (abfd)->core->program | 
 | 	= _bfd_elfcore_strndup (abfd, note->descdata + 88, 16); | 
 |       elf_tdata (abfd)->core->command | 
 | 	= _bfd_elfcore_strndup (abfd, note->descdata + 104, 80); | 
 |       break; | 
 |     } | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | /* Functions for dealing with the e_flags field. | 
 |  | 
 |    We don't define set_private_flags or copy_private_bfd_data because | 
 |    the only currently defined values are based on the bfd mach number, | 
 |    so we use the latter instead and defer setting e_flags until the | 
 |    file is written out.  */ | 
 |  | 
 | /* Merge backend specific data from an object file to the output | 
 |    object file when linking.  */ | 
 |  | 
 | static bool | 
 | elf32_sparc_merge_private_bfd_data (bfd *ibfd, struct bfd_link_info *info) | 
 | { | 
 |   bfd *obfd = info->output_bfd; | 
 |   bool error; | 
 |   unsigned long ibfd_mach; | 
 |   /* FIXME: This should not be static.  */ | 
 |   static unsigned long previous_ibfd_e_flags = (unsigned long) -1; | 
 |  | 
 |   if (bfd_get_flavour (ibfd) != bfd_target_elf_flavour | 
 |       || bfd_get_flavour (obfd) != bfd_target_elf_flavour) | 
 |     return true; | 
 |  | 
 |   error = false; | 
 |  | 
 |   ibfd_mach = bfd_get_mach (ibfd); | 
 |   if (bfd_mach_sparc_64bit_p (ibfd_mach)) | 
 |     { | 
 |       error = true; | 
 |       _bfd_error_handler | 
 | 	(_("%pB: compiled for a 64 bit system and target is 32 bit"), ibfd); | 
 |     } | 
 |   else if ((ibfd->flags & DYNAMIC) == 0) | 
 |     { | 
 |       if (bfd_get_mach (obfd) < ibfd_mach) | 
 | 	bfd_set_arch_mach (obfd, bfd_arch_sparc, ibfd_mach); | 
 |     } | 
 |  | 
 |   if (((elf_elfheader (ibfd)->e_flags & EF_SPARC_LEDATA) | 
 |        != previous_ibfd_e_flags) | 
 |       && previous_ibfd_e_flags != (unsigned long) -1) | 
 |     { | 
 |       _bfd_error_handler | 
 | 	(_("%pB: linking little endian files with big endian files"), ibfd); | 
 |       error = true; | 
 |     } | 
 |   previous_ibfd_e_flags = elf_elfheader (ibfd)->e_flags & EF_SPARC_LEDATA; | 
 |  | 
 |   if (error) | 
 |     { | 
 |       bfd_set_error (bfd_error_bad_value); | 
 |       return false; | 
 |     } | 
 |  | 
 |   return _bfd_sparc_elf_merge_private_bfd_data (ibfd, info); | 
 | } | 
 |  | 
 | /* The final processing done just before writing out the object file. | 
 |    We need to set the e_machine field appropriately.  */ | 
 |  | 
 | static void | 
 | sparc_final_write_processing (bfd *abfd) | 
 | { | 
 |   switch (bfd_get_mach (abfd)) | 
 |     { | 
 |     case bfd_mach_sparc: | 
 |     case bfd_mach_sparc_sparclet: | 
 |     case bfd_mach_sparc_sparclite: | 
 |       break; /* nothing to do */ | 
 |     case bfd_mach_sparc_v8plus: | 
 |       elf_elfheader (abfd)->e_machine = EM_SPARC32PLUS; | 
 |       elf_elfheader (abfd)->e_flags &=~ EF_SPARC_32PLUS_MASK; | 
 |       elf_elfheader (abfd)->e_flags |= EF_SPARC_32PLUS; | 
 |       break; | 
 |     case bfd_mach_sparc_v8plusa: | 
 |       elf_elfheader (abfd)->e_machine = EM_SPARC32PLUS; | 
 |       elf_elfheader (abfd)->e_flags &=~ EF_SPARC_32PLUS_MASK; | 
 |       elf_elfheader (abfd)->e_flags |= EF_SPARC_32PLUS | EF_SPARC_SUN_US1; | 
 |       break; | 
 |     case bfd_mach_sparc_v8plusb: | 
 |     case bfd_mach_sparc_v8plusc: | 
 |     case bfd_mach_sparc_v8plusd: | 
 |     case bfd_mach_sparc_v8pluse: | 
 |     case bfd_mach_sparc_v8plusv: | 
 |     case bfd_mach_sparc_v8plusm: | 
 |     case bfd_mach_sparc_v8plusm8: | 
 |       elf_elfheader (abfd)->e_machine = EM_SPARC32PLUS; | 
 |       elf_elfheader (abfd)->e_flags &=~ EF_SPARC_32PLUS_MASK; | 
 |       elf_elfheader (abfd)->e_flags |= EF_SPARC_32PLUS | EF_SPARC_SUN_US1 | 
 | 				       | EF_SPARC_SUN_US3; | 
 |       break; | 
 |     case bfd_mach_sparc_sparclite_le: | 
 |       elf_elfheader (abfd)->e_flags |= EF_SPARC_LEDATA; | 
 |       break; | 
 |     case 0: /* A non-sparc architecture - ignore.  */ | 
 |       break; | 
 |     default: | 
 |       _bfd_error_handler | 
 | 	(_("%pB: unhandled sparc machine value '%lu' detected during write processing"), | 
 | 	 abfd, (long) bfd_get_mach (abfd)); | 
 |       break; | 
 |     } | 
 | } | 
 |  | 
 | static bool | 
 | elf32_sparc_final_write_processing (bfd *abfd) | 
 | { | 
 |   sparc_final_write_processing (abfd); | 
 |   return _bfd_elf_final_write_processing (abfd); | 
 | } | 
 |  | 
 | /* Used to decide how to sort relocs in an optimal manner for the | 
 |    dynamic linker, before writing them out.  */ | 
 |  | 
 | static enum elf_reloc_type_class | 
 | elf32_sparc_reloc_type_class (const struct bfd_link_info *info, | 
 | 			      const asection *rel_sec ATTRIBUTE_UNUSED, | 
 | 			      const Elf_Internal_Rela *rela) | 
 | { | 
 |   bfd *abfd = info->output_bfd; | 
 |   const struct elf_backend_data *bed = get_elf_backend_data (abfd); | 
 |   struct _bfd_sparc_elf_link_hash_table *htab | 
 |     = _bfd_sparc_elf_hash_table (info); | 
 |   BFD_ASSERT (htab != NULL); | 
 |  | 
 |   if (htab->elf.dynsym != NULL | 
 |       && htab->elf.dynsym->contents != NULL) | 
 |     { | 
 |       /* Check relocation against STT_GNU_IFUNC symbol if there are | 
 | 	 dynamic symbols.  */ | 
 |       unsigned long r_symndx = htab->r_symndx (rela->r_info); | 
 |       if (r_symndx != STN_UNDEF) | 
 | 	{ | 
 | 	  Elf_Internal_Sym sym; | 
 | 	  if (!bed->s->swap_symbol_in (abfd, | 
 | 				       (htab->elf.dynsym->contents | 
 | 					+ r_symndx * bed->s->sizeof_sym), | 
 | 				       0, &sym)) | 
 | 	    abort (); | 
 |  | 
 | 	  if (ELF_ST_TYPE (sym.st_info) == STT_GNU_IFUNC) | 
 | 	    return reloc_class_ifunc; | 
 | 	} | 
 |     } | 
 |  | 
 |   switch ((int) ELF32_R_TYPE (rela->r_info)) | 
 |     { | 
 |     case R_SPARC_IRELATIVE: | 
 |       return reloc_class_ifunc; | 
 |     case R_SPARC_RELATIVE: | 
 |       return reloc_class_relative; | 
 |     case R_SPARC_JMP_SLOT: | 
 |       return reloc_class_plt; | 
 |     case R_SPARC_COPY: | 
 |       return reloc_class_copy; | 
 |     default: | 
 |       return reloc_class_normal; | 
 |     } | 
 | } | 
 |  | 
 | #define TARGET_BIG_SYM	sparc_elf32_vec | 
 | #define TARGET_BIG_NAME	"elf32-sparc" | 
 | #define ELF_ARCH	bfd_arch_sparc | 
 | #define ELF_TARGET_ID	SPARC_ELF_DATA | 
 | #define ELF_MACHINE_CODE EM_SPARC | 
 | #define ELF_MACHINE_ALT1 EM_SPARC32PLUS | 
 | #define ELF_MAXPAGESIZE 0x10000 | 
 | #define ELF_COMMONPAGESIZE 0x2000 | 
 |  | 
 | #define bfd_elf32_bfd_merge_private_bfd_data \ | 
 | 					elf32_sparc_merge_private_bfd_data | 
 | #define elf_backend_final_write_processing \ | 
 | 					elf32_sparc_final_write_processing | 
 | #define elf_backend_grok_psinfo		elf32_sparc_grok_psinfo | 
 | #define elf_backend_reloc_type_class	elf32_sparc_reloc_type_class | 
 |  | 
 | #define elf_info_to_howto		_bfd_sparc_elf_info_to_howto | 
 | #define bfd_elf32_bfd_reloc_type_lookup	_bfd_sparc_elf_reloc_type_lookup | 
 | #define bfd_elf32_bfd_reloc_name_lookup \ | 
 |   _bfd_sparc_elf_reloc_name_lookup | 
 | #define bfd_elf32_bfd_link_hash_table_create \ | 
 | 					_bfd_sparc_elf_link_hash_table_create | 
 | #define bfd_elf32_bfd_relax_section	_bfd_sparc_elf_relax_section | 
 | #define bfd_elf32_new_section_hook	_bfd_sparc_elf_new_section_hook | 
 | #define elf_backend_copy_indirect_symbol \ | 
 | 					_bfd_sparc_elf_copy_indirect_symbol | 
 | #define elf_backend_create_dynamic_sections \ | 
 | 					_bfd_sparc_elf_create_dynamic_sections | 
 | #define elf_backend_check_relocs	_bfd_sparc_elf_check_relocs | 
 | #define elf_backend_adjust_dynamic_symbol \ | 
 | 					_bfd_sparc_elf_adjust_dynamic_symbol | 
 | #define elf_backend_omit_section_dynsym	_bfd_sparc_elf_omit_section_dynsym | 
 | #define elf_backend_size_dynamic_sections \ | 
 | 					_bfd_sparc_elf_size_dynamic_sections | 
 | #define elf_backend_relocate_section	_bfd_sparc_elf_relocate_section | 
 | #define elf_backend_finish_dynamic_symbol \ | 
 | 					_bfd_sparc_elf_finish_dynamic_symbol | 
 | #define elf_backend_finish_dynamic_sections \ | 
 | 					_bfd_sparc_elf_finish_dynamic_sections | 
 | #define bfd_elf32_mkobject		_bfd_sparc_elf_mkobject | 
 | #define elf_backend_object_p		_bfd_sparc_elf_object_p | 
 | #define elf_backend_gc_mark_hook	_bfd_sparc_elf_gc_mark_hook | 
 | #define elf_backend_plt_sym_val		_bfd_sparc_elf_plt_sym_val | 
 | #define elf_backend_init_index_section	_bfd_elf_init_1_index_section | 
 | #define elf_backend_fixup_symbol	_bfd_sparc_elf_fixup_symbol | 
 |  | 
 | #define elf_backend_can_gc_sections 1 | 
 | #define elf_backend_can_refcount 1 | 
 | #define elf_backend_want_got_plt 0 | 
 | #define elf_backend_plt_readonly 0 | 
 | #define elf_backend_want_plt_sym 1 | 
 | #define elf_backend_got_header_size 4 | 
 | #define elf_backend_want_dynrelro 1 | 
 | #define elf_backend_rela_normal 1 | 
 |  | 
 | #define elf_backend_linux_prpsinfo32_ugid16	true | 
 |  | 
 | #include "elf32-target.h" | 
 |  | 
 | /* Solaris 2.  */ | 
 |  | 
 | #undef	TARGET_BIG_SYM | 
 | #define	TARGET_BIG_SYM				sparc_elf32_sol2_vec | 
 | #undef	TARGET_BIG_NAME | 
 | #define	TARGET_BIG_NAME				"elf32-sparc-sol2" | 
 |  | 
 | #undef elf32_bed | 
 | #define elf32_bed				elf32_sparc_sol2_bed | 
 |  | 
 | /* The 32-bit static TLS arena size is rounded to the nearest 8-byte | 
 |    boundary.  */ | 
 | #undef	elf_backend_static_tls_alignment | 
 | #define elf_backend_static_tls_alignment	8 | 
 |  | 
 | #undef	elf_backend_strtab_flags | 
 | #define elf_backend_strtab_flags	SHF_STRINGS | 
 |  | 
 | static bool | 
 | elf32_sparc_copy_solaris_special_section_fields (const bfd *ibfd ATTRIBUTE_UNUSED, | 
 | 						 bfd *obfd ATTRIBUTE_UNUSED, | 
 | 						 const Elf_Internal_Shdr *isection ATTRIBUTE_UNUSED, | 
 | 						 Elf_Internal_Shdr *osection ATTRIBUTE_UNUSED) | 
 | { | 
 |   /* PR 19938: FIXME: Need to add code for setting the sh_info | 
 |      and sh_link fields of Solaris specific section types.  */ | 
 |   return false; | 
 | } | 
 |  | 
 | #undef  elf_backend_copy_special_section_fields | 
 | #define elf_backend_copy_special_section_fields elf32_sparc_copy_solaris_special_section_fields | 
 |  | 
 | #include "elf32-target.h" | 
 |  | 
 | /* A final_write_processing hook that does both the SPARC- and VxWorks- | 
 |    specific handling.  */ | 
 |  | 
 | static bool | 
 | elf32_sparc_vxworks_final_write_processing (bfd *abfd) | 
 | { | 
 |   sparc_final_write_processing (abfd); | 
 |   return elf_vxworks_final_write_processing (abfd); | 
 | } | 
 |  | 
 | #undef  TARGET_BIG_SYM | 
 | #define TARGET_BIG_SYM	sparc_elf32_vxworks_vec | 
 | #undef  TARGET_BIG_NAME | 
 | #define TARGET_BIG_NAME	"elf32-sparc-vxworks" | 
 |  | 
 | #undef  ELF_MINPAGESIZE | 
 | #define ELF_MINPAGESIZE	0x1000 | 
 |  | 
 | #undef	ELF_TARGET_OS | 
 | #define	ELF_TARGET_OS	is_vxworks | 
 |  | 
 | #undef  elf_backend_want_got_plt | 
 | #define elf_backend_want_got_plt		1 | 
 | #undef  elf_backend_plt_readonly | 
 | #define elf_backend_plt_readonly		1 | 
 | #undef  elf_backend_got_header_size | 
 | #define elf_backend_got_header_size		12 | 
 | #undef  elf_backend_dtrel_excludes_plt | 
 | #define elf_backend_dtrel_excludes_plt		1 | 
 | #undef  elf_backend_add_symbol_hook | 
 | #define elf_backend_add_symbol_hook \ | 
 |   elf_vxworks_add_symbol_hook | 
 | #undef  elf_backend_link_output_symbol_hook | 
 | #define elf_backend_link_output_symbol_hook \ | 
 |   elf_vxworks_link_output_symbol_hook | 
 | #undef  elf_backend_emit_relocs | 
 | #define elf_backend_emit_relocs \ | 
 |   elf_vxworks_emit_relocs | 
 | #undef  elf_backend_final_write_processing | 
 | #define elf_backend_final_write_processing \ | 
 |   elf32_sparc_vxworks_final_write_processing | 
 | #undef  elf_backend_static_tls_alignment | 
 | #undef  elf_backend_strtab_flags | 
 | #undef  elf_backend_copy_special_section_fields | 
 |  | 
 | #undef  elf32_bed | 
 | #define elf32_bed				sparc_elf_vxworks_bed | 
 |  | 
 | #include "elf32-target.h" |