* elf-bfd.h (emum elf_object_id): Rename to elf_target_id. Add
[binutils.git] / bfd / elfxx-ia64.c
blobe35fcbf8fe580871be1b452354d9387d0f892b45
1 /* IA-64 support for 64-bit ELF
2 Copyright 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007,
3 2008, 2009, 2010 Free Software Foundation, Inc.
4 Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
6 This file is part of BFD, the Binary File Descriptor library.
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
21 MA 02110-1301, USA. */
23 #include "sysdep.h"
24 #include "bfd.h"
25 #include "libbfd.h"
26 #include "elf-bfd.h"
27 #include "opcode/ia64.h"
28 #include "elf/ia64.h"
29 #include "objalloc.h"
30 #include "hashtab.h"
32 #define ARCH_SIZE NN
34 #if ARCH_SIZE == 64
35 #define LOG_SECTION_ALIGN 3
36 #endif
38 #if ARCH_SIZE == 32
39 #define LOG_SECTION_ALIGN 2
40 #endif
42 /* THE RULES for all the stuff the linker creates --
44 GOT Entries created in response to LTOFF or LTOFF_FPTR
45 relocations. Dynamic relocs created for dynamic
46 symbols in an application; REL relocs for locals
47 in a shared library.
49 FPTR The canonical function descriptor. Created for local
50 symbols in applications. Descriptors for dynamic symbols
51 and local symbols in shared libraries are created by
52 ld.so. Thus there are no dynamic relocs against these
53 objects. The FPTR relocs for such _are_ passed through
54 to the dynamic relocation tables.
56 FULL_PLT Created for a PCREL21B relocation against a dynamic symbol.
57 Requires the creation of a PLTOFF entry. This does not
58 require any dynamic relocations.
60 PLTOFF Created by PLTOFF relocations. For local symbols, this
61 is an alternate function descriptor, and in shared libraries
62 requires two REL relocations. Note that this cannot be
63 transformed into an FPTR relocation, since it must be in
64 range of the GP. For dynamic symbols, this is a function
65 descriptor for a MIN_PLT entry, and requires one IPLT reloc.
67 MIN_PLT Created by PLTOFF entries against dynamic symbols. This
68 does not require dynamic relocations. */
70 #define NELEMS(a) ((int) (sizeof (a) / sizeof ((a)[0])))
72 typedef struct bfd_hash_entry *(*new_hash_entry_func)
73 (struct bfd_hash_entry *, struct bfd_hash_table *, const char *);
75 /* In dynamically (linker-) created sections, we generally need to keep track
76 of the place a symbol or expression got allocated to. This is done via hash
77 tables that store entries of the following type. */
79 struct elfNN_ia64_dyn_sym_info
81 /* The addend for which this entry is relevant. */
82 bfd_vma addend;
84 bfd_vma got_offset;
85 bfd_vma fptr_offset;
86 bfd_vma pltoff_offset;
87 bfd_vma plt_offset;
88 bfd_vma plt2_offset;
89 bfd_vma tprel_offset;
90 bfd_vma dtpmod_offset;
91 bfd_vma dtprel_offset;
93 /* The symbol table entry, if any, that this was derived from. */
94 struct elf_link_hash_entry *h;
96 /* Used to count non-got, non-plt relocations for delayed sizing
97 of relocation sections. */
98 struct elfNN_ia64_dyn_reloc_entry
100 struct elfNN_ia64_dyn_reloc_entry *next;
101 asection *srel;
102 int type;
103 int count;
105 /* Is this reloc against readonly section? */
106 bfd_boolean reltext;
107 } *reloc_entries;
109 /* TRUE when the section contents have been updated. */
110 unsigned got_done : 1;
111 unsigned fptr_done : 1;
112 unsigned pltoff_done : 1;
113 unsigned tprel_done : 1;
114 unsigned dtpmod_done : 1;
115 unsigned dtprel_done : 1;
117 /* TRUE for the different kinds of linker data we want created. */
118 unsigned want_got : 1;
119 unsigned want_gotx : 1;
120 unsigned want_fptr : 1;
121 unsigned want_ltoff_fptr : 1;
122 unsigned want_plt : 1;
123 unsigned want_plt2 : 1;
124 unsigned want_pltoff : 1;
125 unsigned want_tprel : 1;
126 unsigned want_dtpmod : 1;
127 unsigned want_dtprel : 1;
130 struct elfNN_ia64_local_hash_entry
132 int id;
133 unsigned int r_sym;
134 /* The number of elements in elfNN_ia64_dyn_sym_info array. */
135 unsigned int count;
136 /* The number of sorted elements in elfNN_ia64_dyn_sym_info array. */
137 unsigned int sorted_count;
138 /* The size of elfNN_ia64_dyn_sym_info array. */
139 unsigned int size;
140 /* The array of elfNN_ia64_dyn_sym_info. */
141 struct elfNN_ia64_dyn_sym_info *info;
143 /* TRUE if this hash entry's addends was translated for
144 SHF_MERGE optimization. */
145 unsigned sec_merge_done : 1;
148 struct elfNN_ia64_link_hash_entry
150 struct elf_link_hash_entry root;
151 /* The number of elements in elfNN_ia64_dyn_sym_info array. */
152 unsigned int count;
153 /* The number of sorted elements in elfNN_ia64_dyn_sym_info array. */
154 unsigned int sorted_count;
155 /* The size of elfNN_ia64_dyn_sym_info array. */
156 unsigned int size;
157 /* The array of elfNN_ia64_dyn_sym_info. */
158 struct elfNN_ia64_dyn_sym_info *info;
161 struct elfNN_ia64_link_hash_table
163 /* The main hash table. */
164 struct elf_link_hash_table root;
166 asection *fptr_sec; /* Function descriptor table (or NULL). */
167 asection *rel_fptr_sec; /* Dynamic relocation section for same. */
168 asection *pltoff_sec; /* Private descriptors for plt (or NULL). */
169 asection *rel_pltoff_sec; /* Dynamic relocation section for same. */
171 bfd_size_type minplt_entries; /* Number of minplt entries. */
172 unsigned reltext : 1; /* Are there relocs against readonly sections? */
173 unsigned self_dtpmod_done : 1;/* Has self DTPMOD entry been finished? */
174 bfd_vma self_dtpmod_offset; /* .got offset to self DTPMOD entry. */
175 /* There are maybe R_IA64_GPREL22 relocations, including those
176 optimized from R_IA64_LTOFF22X, against non-SHF_IA_64_SHORT
177 sections. We need to record those sections so that we can choose
178 a proper GP to cover all R_IA64_GPREL22 relocations. */
179 asection *max_short_sec; /* Maximum short output section. */
180 bfd_vma max_short_offset; /* Maximum short offset. */
181 asection *min_short_sec; /* Minimum short output section. */
182 bfd_vma min_short_offset; /* Minimum short offset. */
184 htab_t loc_hash_table;
185 void *loc_hash_memory;
188 struct elfNN_ia64_allocate_data
190 struct bfd_link_info *info;
191 bfd_size_type ofs;
192 bfd_boolean only_got;
195 #define elfNN_ia64_hash_table(p) \
196 (elf_hash_table_id ((struct elf_link_hash_table *) ((p)->hash)) \
197 == IA64_ELF_DATA ? ((struct elfNN_ia64_link_hash_table *) ((p)->hash)) : NULL)
199 static struct elfNN_ia64_dyn_sym_info * get_dyn_sym_info
200 (struct elfNN_ia64_link_hash_table *ia64_info,
201 struct elf_link_hash_entry *h,
202 bfd *abfd, const Elf_Internal_Rela *rel, bfd_boolean create);
203 static bfd_boolean elfNN_ia64_dynamic_symbol_p
204 (struct elf_link_hash_entry *h, struct bfd_link_info *info, int);
205 static bfd_reloc_status_type elfNN_ia64_install_value
206 (bfd_byte *hit_addr, bfd_vma val, unsigned int r_type);
207 static bfd_boolean elfNN_ia64_choose_gp
208 (bfd *abfd, struct bfd_link_info *info);
209 static void elfNN_ia64_relax_ldxmov
210 (bfd_byte *contents, bfd_vma off);
211 static void elfNN_ia64_dyn_sym_traverse
212 (struct elfNN_ia64_link_hash_table *ia64_info,
213 bfd_boolean (*func) (struct elfNN_ia64_dyn_sym_info *, PTR),
214 PTR info);
215 static bfd_boolean allocate_global_data_got
216 (struct elfNN_ia64_dyn_sym_info *dyn_i, PTR data);
217 static bfd_boolean allocate_global_fptr_got
218 (struct elfNN_ia64_dyn_sym_info *dyn_i, PTR data);
219 static bfd_boolean allocate_local_got
220 (struct elfNN_ia64_dyn_sym_info *dyn_i, PTR data);
221 static bfd_boolean elfNN_ia64_hpux_vec
222 (const bfd_target *vec);
223 static bfd_boolean allocate_dynrel_entries
224 (struct elfNN_ia64_dyn_sym_info *dyn_i, PTR data);
225 static asection *get_pltoff
226 (bfd *abfd, struct bfd_link_info *info,
227 struct elfNN_ia64_link_hash_table *ia64_info);
229 /* ia64-specific relocation. */
231 /* Perform a relocation. Not much to do here as all the hard work is
232 done in elfNN_ia64_final_link_relocate. */
233 static bfd_reloc_status_type
234 elfNN_ia64_reloc (bfd *abfd ATTRIBUTE_UNUSED, arelent *reloc,
235 asymbol *sym ATTRIBUTE_UNUSED,
236 PTR data ATTRIBUTE_UNUSED, asection *input_section,
237 bfd *output_bfd, char **error_message)
239 if (output_bfd)
241 reloc->address += input_section->output_offset;
242 return bfd_reloc_ok;
245 if (input_section->flags & SEC_DEBUGGING)
246 return bfd_reloc_continue;
248 *error_message = "Unsupported call to elfNN_ia64_reloc";
249 return bfd_reloc_notsupported;
252 #define IA64_HOWTO(TYPE, NAME, SIZE, PCREL, IN) \
253 HOWTO (TYPE, 0, SIZE, 0, PCREL, 0, complain_overflow_signed, \
254 elfNN_ia64_reloc, NAME, FALSE, 0, -1, IN)
256 /* This table has to be sorted according to increasing number of the
257 TYPE field. */
258 static reloc_howto_type ia64_howto_table[] =
260 IA64_HOWTO (R_IA64_NONE, "NONE", 0, FALSE, TRUE),
262 IA64_HOWTO (R_IA64_IMM14, "IMM14", 0, FALSE, TRUE),
263 IA64_HOWTO (R_IA64_IMM22, "IMM22", 0, FALSE, TRUE),
264 IA64_HOWTO (R_IA64_IMM64, "IMM64", 0, FALSE, TRUE),
265 IA64_HOWTO (R_IA64_DIR32MSB, "DIR32MSB", 2, FALSE, TRUE),
266 IA64_HOWTO (R_IA64_DIR32LSB, "DIR32LSB", 2, FALSE, TRUE),
267 IA64_HOWTO (R_IA64_DIR64MSB, "DIR64MSB", 4, FALSE, TRUE),
268 IA64_HOWTO (R_IA64_DIR64LSB, "DIR64LSB", 4, FALSE, TRUE),
270 IA64_HOWTO (R_IA64_GPREL22, "GPREL22", 0, FALSE, TRUE),
271 IA64_HOWTO (R_IA64_GPREL64I, "GPREL64I", 0, FALSE, TRUE),
272 IA64_HOWTO (R_IA64_GPREL32MSB, "GPREL32MSB", 2, FALSE, TRUE),
273 IA64_HOWTO (R_IA64_GPREL32LSB, "GPREL32LSB", 2, FALSE, TRUE),
274 IA64_HOWTO (R_IA64_GPREL64MSB, "GPREL64MSB", 4, FALSE, TRUE),
275 IA64_HOWTO (R_IA64_GPREL64LSB, "GPREL64LSB", 4, FALSE, TRUE),
277 IA64_HOWTO (R_IA64_LTOFF22, "LTOFF22", 0, FALSE, TRUE),
278 IA64_HOWTO (R_IA64_LTOFF64I, "LTOFF64I", 0, FALSE, TRUE),
280 IA64_HOWTO (R_IA64_PLTOFF22, "PLTOFF22", 0, FALSE, TRUE),
281 IA64_HOWTO (R_IA64_PLTOFF64I, "PLTOFF64I", 0, FALSE, TRUE),
282 IA64_HOWTO (R_IA64_PLTOFF64MSB, "PLTOFF64MSB", 4, FALSE, TRUE),
283 IA64_HOWTO (R_IA64_PLTOFF64LSB, "PLTOFF64LSB", 4, FALSE, TRUE),
285 IA64_HOWTO (R_IA64_FPTR64I, "FPTR64I", 0, FALSE, TRUE),
286 IA64_HOWTO (R_IA64_FPTR32MSB, "FPTR32MSB", 2, FALSE, TRUE),
287 IA64_HOWTO (R_IA64_FPTR32LSB, "FPTR32LSB", 2, FALSE, TRUE),
288 IA64_HOWTO (R_IA64_FPTR64MSB, "FPTR64MSB", 4, FALSE, TRUE),
289 IA64_HOWTO (R_IA64_FPTR64LSB, "FPTR64LSB", 4, FALSE, TRUE),
291 IA64_HOWTO (R_IA64_PCREL60B, "PCREL60B", 0, TRUE, TRUE),
292 IA64_HOWTO (R_IA64_PCREL21B, "PCREL21B", 0, TRUE, TRUE),
293 IA64_HOWTO (R_IA64_PCREL21M, "PCREL21M", 0, TRUE, TRUE),
294 IA64_HOWTO (R_IA64_PCREL21F, "PCREL21F", 0, TRUE, TRUE),
295 IA64_HOWTO (R_IA64_PCREL32MSB, "PCREL32MSB", 2, TRUE, TRUE),
296 IA64_HOWTO (R_IA64_PCREL32LSB, "PCREL32LSB", 2, TRUE, TRUE),
297 IA64_HOWTO (R_IA64_PCREL64MSB, "PCREL64MSB", 4, TRUE, TRUE),
298 IA64_HOWTO (R_IA64_PCREL64LSB, "PCREL64LSB", 4, TRUE, TRUE),
300 IA64_HOWTO (R_IA64_LTOFF_FPTR22, "LTOFF_FPTR22", 0, FALSE, TRUE),
301 IA64_HOWTO (R_IA64_LTOFF_FPTR64I, "LTOFF_FPTR64I", 0, FALSE, TRUE),
302 IA64_HOWTO (R_IA64_LTOFF_FPTR32MSB, "LTOFF_FPTR32MSB", 2, FALSE, TRUE),
303 IA64_HOWTO (R_IA64_LTOFF_FPTR32LSB, "LTOFF_FPTR32LSB", 2, FALSE, TRUE),
304 IA64_HOWTO (R_IA64_LTOFF_FPTR64MSB, "LTOFF_FPTR64MSB", 4, FALSE, TRUE),
305 IA64_HOWTO (R_IA64_LTOFF_FPTR64LSB, "LTOFF_FPTR64LSB", 4, FALSE, TRUE),
307 IA64_HOWTO (R_IA64_SEGREL32MSB, "SEGREL32MSB", 2, FALSE, TRUE),
308 IA64_HOWTO (R_IA64_SEGREL32LSB, "SEGREL32LSB", 2, FALSE, TRUE),
309 IA64_HOWTO (R_IA64_SEGREL64MSB, "SEGREL64MSB", 4, FALSE, TRUE),
310 IA64_HOWTO (R_IA64_SEGREL64LSB, "SEGREL64LSB", 4, FALSE, TRUE),
312 IA64_HOWTO (R_IA64_SECREL32MSB, "SECREL32MSB", 2, FALSE, TRUE),
313 IA64_HOWTO (R_IA64_SECREL32LSB, "SECREL32LSB", 2, FALSE, TRUE),
314 IA64_HOWTO (R_IA64_SECREL64MSB, "SECREL64MSB", 4, FALSE, TRUE),
315 IA64_HOWTO (R_IA64_SECREL64LSB, "SECREL64LSB", 4, FALSE, TRUE),
317 IA64_HOWTO (R_IA64_REL32MSB, "REL32MSB", 2, FALSE, TRUE),
318 IA64_HOWTO (R_IA64_REL32LSB, "REL32LSB", 2, FALSE, TRUE),
319 IA64_HOWTO (R_IA64_REL64MSB, "REL64MSB", 4, FALSE, TRUE),
320 IA64_HOWTO (R_IA64_REL64LSB, "REL64LSB", 4, FALSE, TRUE),
322 IA64_HOWTO (R_IA64_LTV32MSB, "LTV32MSB", 2, FALSE, TRUE),
323 IA64_HOWTO (R_IA64_LTV32LSB, "LTV32LSB", 2, FALSE, TRUE),
324 IA64_HOWTO (R_IA64_LTV64MSB, "LTV64MSB", 4, FALSE, TRUE),
325 IA64_HOWTO (R_IA64_LTV64LSB, "LTV64LSB", 4, FALSE, TRUE),
327 IA64_HOWTO (R_IA64_PCREL21BI, "PCREL21BI", 0, TRUE, TRUE),
328 IA64_HOWTO (R_IA64_PCREL22, "PCREL22", 0, TRUE, TRUE),
329 IA64_HOWTO (R_IA64_PCREL64I, "PCREL64I", 0, TRUE, TRUE),
331 IA64_HOWTO (R_IA64_IPLTMSB, "IPLTMSB", 4, FALSE, TRUE),
332 IA64_HOWTO (R_IA64_IPLTLSB, "IPLTLSB", 4, FALSE, TRUE),
333 IA64_HOWTO (R_IA64_COPY, "COPY", 4, FALSE, TRUE),
334 IA64_HOWTO (R_IA64_LTOFF22X, "LTOFF22X", 0, FALSE, TRUE),
335 IA64_HOWTO (R_IA64_LDXMOV, "LDXMOV", 0, FALSE, TRUE),
337 IA64_HOWTO (R_IA64_TPREL14, "TPREL14", 0, FALSE, FALSE),
338 IA64_HOWTO (R_IA64_TPREL22, "TPREL22", 0, FALSE, FALSE),
339 IA64_HOWTO (R_IA64_TPREL64I, "TPREL64I", 0, FALSE, FALSE),
340 IA64_HOWTO (R_IA64_TPREL64MSB, "TPREL64MSB", 4, FALSE, FALSE),
341 IA64_HOWTO (R_IA64_TPREL64LSB, "TPREL64LSB", 4, FALSE, FALSE),
342 IA64_HOWTO (R_IA64_LTOFF_TPREL22, "LTOFF_TPREL22", 0, FALSE, FALSE),
344 IA64_HOWTO (R_IA64_DTPMOD64MSB, "DTPMOD64MSB", 4, FALSE, FALSE),
345 IA64_HOWTO (R_IA64_DTPMOD64LSB, "DTPMOD64LSB", 4, FALSE, FALSE),
346 IA64_HOWTO (R_IA64_LTOFF_DTPMOD22, "LTOFF_DTPMOD22", 0, FALSE, FALSE),
348 IA64_HOWTO (R_IA64_DTPREL14, "DTPREL14", 0, FALSE, FALSE),
349 IA64_HOWTO (R_IA64_DTPREL22, "DTPREL22", 0, FALSE, FALSE),
350 IA64_HOWTO (R_IA64_DTPREL64I, "DTPREL64I", 0, FALSE, FALSE),
351 IA64_HOWTO (R_IA64_DTPREL32MSB, "DTPREL32MSB", 2, FALSE, FALSE),
352 IA64_HOWTO (R_IA64_DTPREL32LSB, "DTPREL32LSB", 2, FALSE, FALSE),
353 IA64_HOWTO (R_IA64_DTPREL64MSB, "DTPREL64MSB", 4, FALSE, FALSE),
354 IA64_HOWTO (R_IA64_DTPREL64LSB, "DTPREL64LSB", 4, FALSE, FALSE),
355 IA64_HOWTO (R_IA64_LTOFF_DTPREL22, "LTOFF_DTPREL22", 0, FALSE, FALSE),
358 static unsigned char elf_code_to_howto_index[R_IA64_MAX_RELOC_CODE + 1];
360 /* Given a BFD reloc type, return the matching HOWTO structure. */
362 static reloc_howto_type *
363 lookup_howto (unsigned int rtype)
365 static int inited = 0;
366 int i;
368 if (!inited)
370 inited = 1;
372 memset (elf_code_to_howto_index, 0xff, sizeof (elf_code_to_howto_index));
373 for (i = 0; i < NELEMS (ia64_howto_table); ++i)
374 elf_code_to_howto_index[ia64_howto_table[i].type] = i;
377 if (rtype > R_IA64_MAX_RELOC_CODE)
378 return 0;
379 i = elf_code_to_howto_index[rtype];
380 if (i >= NELEMS (ia64_howto_table))
381 return 0;
382 return ia64_howto_table + i;
385 static reloc_howto_type*
386 elfNN_ia64_reloc_type_lookup (bfd *abfd ATTRIBUTE_UNUSED,
387 bfd_reloc_code_real_type bfd_code)
389 unsigned int rtype;
391 switch (bfd_code)
393 case BFD_RELOC_NONE: rtype = R_IA64_NONE; break;
395 case BFD_RELOC_IA64_IMM14: rtype = R_IA64_IMM14; break;
396 case BFD_RELOC_IA64_IMM22: rtype = R_IA64_IMM22; break;
397 case BFD_RELOC_IA64_IMM64: rtype = R_IA64_IMM64; break;
399 case BFD_RELOC_IA64_DIR32MSB: rtype = R_IA64_DIR32MSB; break;
400 case BFD_RELOC_IA64_DIR32LSB: rtype = R_IA64_DIR32LSB; break;
401 case BFD_RELOC_IA64_DIR64MSB: rtype = R_IA64_DIR64MSB; break;
402 case BFD_RELOC_IA64_DIR64LSB: rtype = R_IA64_DIR64LSB; break;
404 case BFD_RELOC_IA64_GPREL22: rtype = R_IA64_GPREL22; break;
405 case BFD_RELOC_IA64_GPREL64I: rtype = R_IA64_GPREL64I; break;
406 case BFD_RELOC_IA64_GPREL32MSB: rtype = R_IA64_GPREL32MSB; break;
407 case BFD_RELOC_IA64_GPREL32LSB: rtype = R_IA64_GPREL32LSB; break;
408 case BFD_RELOC_IA64_GPREL64MSB: rtype = R_IA64_GPREL64MSB; break;
409 case BFD_RELOC_IA64_GPREL64LSB: rtype = R_IA64_GPREL64LSB; break;
411 case BFD_RELOC_IA64_LTOFF22: rtype = R_IA64_LTOFF22; break;
412 case BFD_RELOC_IA64_LTOFF64I: rtype = R_IA64_LTOFF64I; break;
414 case BFD_RELOC_IA64_PLTOFF22: rtype = R_IA64_PLTOFF22; break;
415 case BFD_RELOC_IA64_PLTOFF64I: rtype = R_IA64_PLTOFF64I; break;
416 case BFD_RELOC_IA64_PLTOFF64MSB: rtype = R_IA64_PLTOFF64MSB; break;
417 case BFD_RELOC_IA64_PLTOFF64LSB: rtype = R_IA64_PLTOFF64LSB; break;
418 case BFD_RELOC_IA64_FPTR64I: rtype = R_IA64_FPTR64I; break;
419 case BFD_RELOC_IA64_FPTR32MSB: rtype = R_IA64_FPTR32MSB; break;
420 case BFD_RELOC_IA64_FPTR32LSB: rtype = R_IA64_FPTR32LSB; break;
421 case BFD_RELOC_IA64_FPTR64MSB: rtype = R_IA64_FPTR64MSB; break;
422 case BFD_RELOC_IA64_FPTR64LSB: rtype = R_IA64_FPTR64LSB; break;
424 case BFD_RELOC_IA64_PCREL21B: rtype = R_IA64_PCREL21B; break;
425 case BFD_RELOC_IA64_PCREL21BI: rtype = R_IA64_PCREL21BI; break;
426 case BFD_RELOC_IA64_PCREL21M: rtype = R_IA64_PCREL21M; break;
427 case BFD_RELOC_IA64_PCREL21F: rtype = R_IA64_PCREL21F; break;
428 case BFD_RELOC_IA64_PCREL22: rtype = R_IA64_PCREL22; break;
429 case BFD_RELOC_IA64_PCREL60B: rtype = R_IA64_PCREL60B; break;
430 case BFD_RELOC_IA64_PCREL64I: rtype = R_IA64_PCREL64I; break;
431 case BFD_RELOC_IA64_PCREL32MSB: rtype = R_IA64_PCREL32MSB; break;
432 case BFD_RELOC_IA64_PCREL32LSB: rtype = R_IA64_PCREL32LSB; break;
433 case BFD_RELOC_IA64_PCREL64MSB: rtype = R_IA64_PCREL64MSB; break;
434 case BFD_RELOC_IA64_PCREL64LSB: rtype = R_IA64_PCREL64LSB; break;
436 case BFD_RELOC_IA64_LTOFF_FPTR22: rtype = R_IA64_LTOFF_FPTR22; break;
437 case BFD_RELOC_IA64_LTOFF_FPTR64I: rtype = R_IA64_LTOFF_FPTR64I; break;
438 case BFD_RELOC_IA64_LTOFF_FPTR32MSB: rtype = R_IA64_LTOFF_FPTR32MSB; break;
439 case BFD_RELOC_IA64_LTOFF_FPTR32LSB: rtype = R_IA64_LTOFF_FPTR32LSB; break;
440 case BFD_RELOC_IA64_LTOFF_FPTR64MSB: rtype = R_IA64_LTOFF_FPTR64MSB; break;
441 case BFD_RELOC_IA64_LTOFF_FPTR64LSB: rtype = R_IA64_LTOFF_FPTR64LSB; break;
443 case BFD_RELOC_IA64_SEGREL32MSB: rtype = R_IA64_SEGREL32MSB; break;
444 case BFD_RELOC_IA64_SEGREL32LSB: rtype = R_IA64_SEGREL32LSB; break;
445 case BFD_RELOC_IA64_SEGREL64MSB: rtype = R_IA64_SEGREL64MSB; break;
446 case BFD_RELOC_IA64_SEGREL64LSB: rtype = R_IA64_SEGREL64LSB; break;
448 case BFD_RELOC_IA64_SECREL32MSB: rtype = R_IA64_SECREL32MSB; break;
449 case BFD_RELOC_IA64_SECREL32LSB: rtype = R_IA64_SECREL32LSB; break;
450 case BFD_RELOC_IA64_SECREL64MSB: rtype = R_IA64_SECREL64MSB; break;
451 case BFD_RELOC_IA64_SECREL64LSB: rtype = R_IA64_SECREL64LSB; break;
453 case BFD_RELOC_IA64_REL32MSB: rtype = R_IA64_REL32MSB; break;
454 case BFD_RELOC_IA64_REL32LSB: rtype = R_IA64_REL32LSB; break;
455 case BFD_RELOC_IA64_REL64MSB: rtype = R_IA64_REL64MSB; break;
456 case BFD_RELOC_IA64_REL64LSB: rtype = R_IA64_REL64LSB; break;
458 case BFD_RELOC_IA64_LTV32MSB: rtype = R_IA64_LTV32MSB; break;
459 case BFD_RELOC_IA64_LTV32LSB: rtype = R_IA64_LTV32LSB; break;
460 case BFD_RELOC_IA64_LTV64MSB: rtype = R_IA64_LTV64MSB; break;
461 case BFD_RELOC_IA64_LTV64LSB: rtype = R_IA64_LTV64LSB; break;
463 case BFD_RELOC_IA64_IPLTMSB: rtype = R_IA64_IPLTMSB; break;
464 case BFD_RELOC_IA64_IPLTLSB: rtype = R_IA64_IPLTLSB; break;
465 case BFD_RELOC_IA64_COPY: rtype = R_IA64_COPY; break;
466 case BFD_RELOC_IA64_LTOFF22X: rtype = R_IA64_LTOFF22X; break;
467 case BFD_RELOC_IA64_LDXMOV: rtype = R_IA64_LDXMOV; break;
469 case BFD_RELOC_IA64_TPREL14: rtype = R_IA64_TPREL14; break;
470 case BFD_RELOC_IA64_TPREL22: rtype = R_IA64_TPREL22; break;
471 case BFD_RELOC_IA64_TPREL64I: rtype = R_IA64_TPREL64I; break;
472 case BFD_RELOC_IA64_TPREL64MSB: rtype = R_IA64_TPREL64MSB; break;
473 case BFD_RELOC_IA64_TPREL64LSB: rtype = R_IA64_TPREL64LSB; break;
474 case BFD_RELOC_IA64_LTOFF_TPREL22: rtype = R_IA64_LTOFF_TPREL22; break;
476 case BFD_RELOC_IA64_DTPMOD64MSB: rtype = R_IA64_DTPMOD64MSB; break;
477 case BFD_RELOC_IA64_DTPMOD64LSB: rtype = R_IA64_DTPMOD64LSB; break;
478 case BFD_RELOC_IA64_LTOFF_DTPMOD22: rtype = R_IA64_LTOFF_DTPMOD22; break;
480 case BFD_RELOC_IA64_DTPREL14: rtype = R_IA64_DTPREL14; break;
481 case BFD_RELOC_IA64_DTPREL22: rtype = R_IA64_DTPREL22; break;
482 case BFD_RELOC_IA64_DTPREL64I: rtype = R_IA64_DTPREL64I; break;
483 case BFD_RELOC_IA64_DTPREL32MSB: rtype = R_IA64_DTPREL32MSB; break;
484 case BFD_RELOC_IA64_DTPREL32LSB: rtype = R_IA64_DTPREL32LSB; break;
485 case BFD_RELOC_IA64_DTPREL64MSB: rtype = R_IA64_DTPREL64MSB; break;
486 case BFD_RELOC_IA64_DTPREL64LSB: rtype = R_IA64_DTPREL64LSB; break;
487 case BFD_RELOC_IA64_LTOFF_DTPREL22: rtype = R_IA64_LTOFF_DTPREL22; break;
489 default: return 0;
491 return lookup_howto (rtype);
494 static reloc_howto_type *
495 elfNN_ia64_reloc_name_lookup (bfd *abfd ATTRIBUTE_UNUSED,
496 const char *r_name)
498 unsigned int i;
500 for (i = 0;
501 i < sizeof (ia64_howto_table) / sizeof (ia64_howto_table[0]);
502 i++)
503 if (ia64_howto_table[i].name != NULL
504 && strcasecmp (ia64_howto_table[i].name, r_name) == 0)
505 return &ia64_howto_table[i];
507 return NULL;
510 /* Given a ELF reloc, return the matching HOWTO structure. */
512 static void
513 elfNN_ia64_info_to_howto (bfd *abfd ATTRIBUTE_UNUSED,
514 arelent *bfd_reloc,
515 Elf_Internal_Rela *elf_reloc)
517 bfd_reloc->howto
518 = lookup_howto ((unsigned int) ELFNN_R_TYPE (elf_reloc->r_info));
521 #define PLT_HEADER_SIZE (3 * 16)
522 #define PLT_MIN_ENTRY_SIZE (1 * 16)
523 #define PLT_FULL_ENTRY_SIZE (2 * 16)
524 #define PLT_RESERVED_WORDS 3
526 static const bfd_byte plt_header[PLT_HEADER_SIZE] =
528 0x0b, 0x10, 0x00, 0x1c, 0x00, 0x21, /* [MMI] mov r2=r14;; */
529 0xe0, 0x00, 0x08, 0x00, 0x48, 0x00, /* addl r14=0,r2 */
530 0x00, 0x00, 0x04, 0x00, /* nop.i 0x0;; */
531 0x0b, 0x80, 0x20, 0x1c, 0x18, 0x14, /* [MMI] ld8 r16=[r14],8;; */
532 0x10, 0x41, 0x38, 0x30, 0x28, 0x00, /* ld8 r17=[r14],8 */
533 0x00, 0x00, 0x04, 0x00, /* nop.i 0x0;; */
534 0x11, 0x08, 0x00, 0x1c, 0x18, 0x10, /* [MIB] ld8 r1=[r14] */
535 0x60, 0x88, 0x04, 0x80, 0x03, 0x00, /* mov b6=r17 */
536 0x60, 0x00, 0x80, 0x00 /* br.few b6;; */
539 static const bfd_byte plt_min_entry[PLT_MIN_ENTRY_SIZE] =
541 0x11, 0x78, 0x00, 0x00, 0x00, 0x24, /* [MIB] mov r15=0 */
542 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, /* nop.i 0x0 */
543 0x00, 0x00, 0x00, 0x40 /* br.few 0 <PLT0>;; */
546 static const bfd_byte plt_full_entry[PLT_FULL_ENTRY_SIZE] =
548 0x0b, 0x78, 0x00, 0x02, 0x00, 0x24, /* [MMI] addl r15=0,r1;; */
549 0x00, 0x41, 0x3c, 0x70, 0x29, 0xc0, /* ld8.acq r16=[r15],8*/
550 0x01, 0x08, 0x00, 0x84, /* mov r14=r1;; */
551 0x11, 0x08, 0x00, 0x1e, 0x18, 0x10, /* [MIB] ld8 r1=[r15] */
552 0x60, 0x80, 0x04, 0x80, 0x03, 0x00, /* mov b6=r16 */
553 0x60, 0x00, 0x80, 0x00 /* br.few b6;; */
556 #define ELF_DYNAMIC_INTERPRETER "/usr/lib/ld.so.1"
558 static const bfd_byte oor_brl[16] =
560 0x05, 0x00, 0x00, 0x00, 0x01, 0x00, /* [MLX] nop.m 0 */
561 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* brl.sptk.few tgt;; */
562 0x00, 0x00, 0x00, 0xc0
565 static const bfd_byte oor_ip[48] =
567 0x04, 0x00, 0x00, 0x00, 0x01, 0x00, /* [MLX] nop.m 0 */
568 0x00, 0x00, 0x00, 0x00, 0x00, 0xe0, /* movl r15=0 */
569 0x01, 0x00, 0x00, 0x60,
570 0x03, 0x00, 0x00, 0x00, 0x01, 0x00, /* [MII] nop.m 0 */
571 0x00, 0x01, 0x00, 0x60, 0x00, 0x00, /* mov r16=ip;; */
572 0xf2, 0x80, 0x00, 0x80, /* add r16=r15,r16;; */
573 0x11, 0x00, 0x00, 0x00, 0x01, 0x00, /* [MIB] nop.m 0 */
574 0x60, 0x80, 0x04, 0x80, 0x03, 0x00, /* mov b6=r16 */
575 0x60, 0x00, 0x80, 0x00 /* br b6;; */
578 static size_t oor_branch_size = sizeof (oor_brl);
580 void
581 bfd_elfNN_ia64_after_parse (int itanium)
583 oor_branch_size = itanium ? sizeof (oor_ip) : sizeof (oor_brl);
586 #define BTYPE_SHIFT 6
587 #define Y_SHIFT 26
588 #define X6_SHIFT 27
589 #define X4_SHIFT 27
590 #define X3_SHIFT 33
591 #define X2_SHIFT 31
592 #define X_SHIFT 33
593 #define OPCODE_SHIFT 37
595 #define OPCODE_BITS (0xfLL << OPCODE_SHIFT)
596 #define X6_BITS (0x3fLL << X6_SHIFT)
597 #define X4_BITS (0xfLL << X4_SHIFT)
598 #define X3_BITS (0x7LL << X3_SHIFT)
599 #define X2_BITS (0x3LL << X2_SHIFT)
600 #define X_BITS (0x1LL << X_SHIFT)
601 #define Y_BITS (0x1LL << Y_SHIFT)
602 #define BTYPE_BITS (0x7LL << BTYPE_SHIFT)
603 #define PREDICATE_BITS (0x3fLL)
605 #define IS_NOP_B(i) \
606 (((i) & (OPCODE_BITS | X6_BITS)) == (2LL << OPCODE_SHIFT))
607 #define IS_NOP_F(i) \
608 (((i) & (OPCODE_BITS | X_BITS | X6_BITS | Y_BITS)) \
609 == (0x1LL << X6_SHIFT))
610 #define IS_NOP_I(i) \
611 (((i) & (OPCODE_BITS | X3_BITS | X6_BITS | Y_BITS)) \
612 == (0x1LL << X6_SHIFT))
613 #define IS_NOP_M(i) \
614 (((i) & (OPCODE_BITS | X3_BITS | X2_BITS | X4_BITS | Y_BITS)) \
615 == (0x1LL << X4_SHIFT))
616 #define IS_BR_COND(i) \
617 (((i) & (OPCODE_BITS | BTYPE_BITS)) == (0x4LL << OPCODE_SHIFT))
618 #define IS_BR_CALL(i) \
619 (((i) & OPCODE_BITS) == (0x5LL << OPCODE_SHIFT))
621 static bfd_boolean
622 elfNN_ia64_relax_br (bfd_byte *contents, bfd_vma off)
624 unsigned int template_val, mlx;
625 bfd_vma t0, t1, s0, s1, s2, br_code;
626 long br_slot;
627 bfd_byte *hit_addr;
629 hit_addr = (bfd_byte *) (contents + off);
630 br_slot = (long) hit_addr & 0x3;
631 hit_addr -= br_slot;
632 t0 = bfd_getl64 (hit_addr + 0);
633 t1 = bfd_getl64 (hit_addr + 8);
635 /* Check if we can turn br into brl. A label is always at the start
636 of the bundle. Even if there are predicates on NOPs, we still
637 perform this optimization. */
638 template_val = t0 & 0x1e;
639 s0 = (t0 >> 5) & 0x1ffffffffffLL;
640 s1 = ((t0 >> 46) | (t1 << 18)) & 0x1ffffffffffLL;
641 s2 = (t1 >> 23) & 0x1ffffffffffLL;
642 switch (br_slot)
644 case 0:
645 /* Check if slot 1 and slot 2 are NOPs. Possible template is
646 BBB. We only need to check nop.b. */
647 if (!(IS_NOP_B (s1) && IS_NOP_B (s2)))
648 return FALSE;
649 br_code = s0;
650 break;
651 case 1:
652 /* Check if slot 2 is NOP. Possible templates are MBB and BBB.
653 For BBB, slot 0 also has to be nop.b. */
654 if (!((template_val == 0x12 /* MBB */
655 && IS_NOP_B (s2))
656 || (template_val == 0x16 /* BBB */
657 && IS_NOP_B (s0)
658 && IS_NOP_B (s2))))
659 return FALSE;
660 br_code = s1;
661 break;
662 case 2:
663 /* Check if slot 1 is NOP. Possible templates are MIB, MBB, BBB,
664 MMB and MFB. For BBB, slot 0 also has to be nop.b. */
665 if (!((template_val == 0x10 /* MIB */
666 && IS_NOP_I (s1))
667 || (template_val == 0x12 /* MBB */
668 && IS_NOP_B (s1))
669 || (template_val == 0x16 /* BBB */
670 && IS_NOP_B (s0)
671 && IS_NOP_B (s1))
672 || (template_val == 0x18 /* MMB */
673 && IS_NOP_M (s1))
674 || (template_val == 0x1c /* MFB */
675 && IS_NOP_F (s1))))
676 return FALSE;
677 br_code = s2;
678 break;
679 default:
680 /* It should never happen. */
681 abort ();
684 /* We can turn br.cond/br.call into brl.cond/brl.call. */
685 if (!(IS_BR_COND (br_code) || IS_BR_CALL (br_code)))
686 return FALSE;
688 /* Turn br into brl by setting bit 40. */
689 br_code |= 0x1LL << 40;
691 /* Turn the old bundle into a MLX bundle with the same stop-bit
692 variety. */
693 if (t0 & 0x1)
694 mlx = 0x5;
695 else
696 mlx = 0x4;
698 if (template_val == 0x16)
700 /* For BBB, we need to put nop.m in slot 0. We keep the original
701 predicate only if slot 0 isn't br. */
702 if (br_slot == 0)
703 t0 = 0LL;
704 else
705 t0 &= PREDICATE_BITS << 5;
706 t0 |= 0x1LL << (X4_SHIFT + 5);
708 else
710 /* Keep the original instruction in slot 0. */
711 t0 &= 0x1ffffffffffLL << 5;
714 t0 |= mlx;
716 /* Put brl in slot 1. */
717 t1 = br_code << 23;
719 bfd_putl64 (t0, hit_addr);
720 bfd_putl64 (t1, hit_addr + 8);
721 return TRUE;
724 static void
725 elfNN_ia64_relax_brl (bfd_byte *contents, bfd_vma off)
727 int template_val;
728 bfd_byte *hit_addr;
729 bfd_vma t0, t1, i0, i1, i2;
731 hit_addr = (bfd_byte *) (contents + off);
732 hit_addr -= (long) hit_addr & 0x3;
733 t0 = bfd_getl64 (hit_addr);
734 t1 = bfd_getl64 (hit_addr + 8);
736 /* Keep the instruction in slot 0. */
737 i0 = (t0 >> 5) & 0x1ffffffffffLL;
738 /* Use nop.b for slot 1. */
739 i1 = 0x4000000000LL;
740 /* For slot 2, turn brl into br by masking out bit 40. */
741 i2 = (t1 >> 23) & 0x0ffffffffffLL;
743 /* Turn a MLX bundle into a MBB bundle with the same stop-bit
744 variety. */
745 if (t0 & 0x1)
746 template_val = 0x13;
747 else
748 template_val = 0x12;
749 t0 = (i1 << 46) | (i0 << 5) | template_val;
750 t1 = (i2 << 23) | (i1 >> 18);
752 bfd_putl64 (t0, hit_addr);
753 bfd_putl64 (t1, hit_addr + 8);
756 /* Rename some of the generic section flags to better document how they
757 are used here. */
758 #define skip_relax_pass_0 need_finalize_relax
759 #define skip_relax_pass_1 has_gp_reloc
762 /* These functions do relaxation for IA-64 ELF. */
764 static void
765 elfNN_ia64_update_short_info (asection *sec, bfd_vma offset,
766 struct elfNN_ia64_link_hash_table *ia64_info)
768 /* Skip ABS and SHF_IA_64_SHORT sections. */
769 if (sec == bfd_abs_section_ptr
770 || (sec->flags & SEC_SMALL_DATA) != 0)
771 return;
773 if (!ia64_info->min_short_sec)
775 ia64_info->max_short_sec = sec;
776 ia64_info->max_short_offset = offset;
777 ia64_info->min_short_sec = sec;
778 ia64_info->min_short_offset = offset;
780 else if (sec == ia64_info->max_short_sec
781 && offset > ia64_info->max_short_offset)
782 ia64_info->max_short_offset = offset;
783 else if (sec == ia64_info->min_short_sec
784 && offset < ia64_info->min_short_offset)
785 ia64_info->min_short_offset = offset;
786 else if (sec->output_section->vma
787 > ia64_info->max_short_sec->vma)
789 ia64_info->max_short_sec = sec;
790 ia64_info->max_short_offset = offset;
792 else if (sec->output_section->vma
793 < ia64_info->min_short_sec->vma)
795 ia64_info->min_short_sec = sec;
796 ia64_info->min_short_offset = offset;
800 static bfd_boolean
801 elfNN_ia64_relax_section (bfd *abfd, asection *sec,
802 struct bfd_link_info *link_info,
803 bfd_boolean *again)
805 struct one_fixup
807 struct one_fixup *next;
808 asection *tsec;
809 bfd_vma toff;
810 bfd_vma trampoff;
813 Elf_Internal_Shdr *symtab_hdr;
814 Elf_Internal_Rela *internal_relocs;
815 Elf_Internal_Rela *irel, *irelend;
816 bfd_byte *contents;
817 Elf_Internal_Sym *isymbuf = NULL;
818 struct elfNN_ia64_link_hash_table *ia64_info;
819 struct one_fixup *fixups = NULL;
820 bfd_boolean changed_contents = FALSE;
821 bfd_boolean changed_relocs = FALSE;
822 bfd_boolean changed_got = FALSE;
823 bfd_boolean skip_relax_pass_0 = TRUE;
824 bfd_boolean skip_relax_pass_1 = TRUE;
825 bfd_vma gp = 0;
827 /* Assume we're not going to change any sizes, and we'll only need
828 one pass. */
829 *again = FALSE;
831 if (link_info->relocatable)
832 (*link_info->callbacks->einfo)
833 (_("%P%F: --relax and -r may not be used together\n"));
835 /* Don't even try to relax for non-ELF outputs. */
836 if (!is_elf_hash_table (link_info->hash))
837 return FALSE;
839 /* Nothing to do if there are no relocations or there is no need for
840 the current pass. */
841 if ((sec->flags & SEC_RELOC) == 0
842 || sec->reloc_count == 0
843 || (link_info->relax_pass == 0 && sec->skip_relax_pass_0)
844 || (link_info->relax_pass == 1 && sec->skip_relax_pass_1))
845 return TRUE;
847 ia64_info = elfNN_ia64_hash_table (link_info);
848 if (ia64_info == NULL)
849 return FALSE;
851 symtab_hdr = &elf_tdata (abfd)->symtab_hdr;
853 /* Load the relocations for this section. */
854 internal_relocs = (_bfd_elf_link_read_relocs
855 (abfd, sec, NULL, (Elf_Internal_Rela *) NULL,
856 link_info->keep_memory));
857 if (internal_relocs == NULL)
858 return FALSE;
860 irelend = internal_relocs + sec->reloc_count;
862 /* Get the section contents. */
863 if (elf_section_data (sec)->this_hdr.contents != NULL)
864 contents = elf_section_data (sec)->this_hdr.contents;
865 else
867 if (!bfd_malloc_and_get_section (abfd, sec, &contents))
868 goto error_return;
871 for (irel = internal_relocs; irel < irelend; irel++)
873 unsigned long r_type = ELFNN_R_TYPE (irel->r_info);
874 bfd_vma symaddr, reladdr, trampoff, toff, roff;
875 asection *tsec;
876 struct one_fixup *f;
877 bfd_size_type amt;
878 bfd_boolean is_branch;
879 struct elfNN_ia64_dyn_sym_info *dyn_i;
880 char symtype;
882 switch (r_type)
884 case R_IA64_PCREL21B:
885 case R_IA64_PCREL21BI:
886 case R_IA64_PCREL21M:
887 case R_IA64_PCREL21F:
888 /* In pass 1, all br relaxations are done. We can skip it. */
889 if (link_info->relax_pass == 1)
890 continue;
891 skip_relax_pass_0 = FALSE;
892 is_branch = TRUE;
893 break;
895 case R_IA64_PCREL60B:
896 /* We can't optimize brl to br in pass 0 since br relaxations
897 will increase the code size. Defer it to pass 1. */
898 if (link_info->relax_pass == 0)
900 skip_relax_pass_1 = FALSE;
901 continue;
903 is_branch = TRUE;
904 break;
906 case R_IA64_GPREL22:
907 /* Update max_short_sec/min_short_sec. */
909 case R_IA64_LTOFF22X:
910 case R_IA64_LDXMOV:
911 /* We can't relax ldx/mov in pass 0 since br relaxations will
912 increase the code size. Defer it to pass 1. */
913 if (link_info->relax_pass == 0)
915 skip_relax_pass_1 = FALSE;
916 continue;
918 is_branch = FALSE;
919 break;
921 default:
922 continue;
925 /* Get the value of the symbol referred to by the reloc. */
926 if (ELFNN_R_SYM (irel->r_info) < symtab_hdr->sh_info)
928 /* A local symbol. */
929 Elf_Internal_Sym *isym;
931 /* Read this BFD's local symbols. */
932 if (isymbuf == NULL)
934 isymbuf = (Elf_Internal_Sym *) symtab_hdr->contents;
935 if (isymbuf == NULL)
936 isymbuf = bfd_elf_get_elf_syms (abfd, symtab_hdr,
937 symtab_hdr->sh_info, 0,
938 NULL, NULL, NULL);
939 if (isymbuf == 0)
940 goto error_return;
943 isym = isymbuf + ELFNN_R_SYM (irel->r_info);
944 if (isym->st_shndx == SHN_UNDEF)
945 continue; /* We can't do anything with undefined symbols. */
946 else if (isym->st_shndx == SHN_ABS)
947 tsec = bfd_abs_section_ptr;
948 else if (isym->st_shndx == SHN_COMMON)
949 tsec = bfd_com_section_ptr;
950 else if (isym->st_shndx == SHN_IA_64_ANSI_COMMON)
951 tsec = bfd_com_section_ptr;
952 else
953 tsec = bfd_section_from_elf_index (abfd, isym->st_shndx);
955 toff = isym->st_value;
956 dyn_i = get_dyn_sym_info (ia64_info, NULL, abfd, irel, FALSE);
957 symtype = ELF_ST_TYPE (isym->st_info);
959 else
961 unsigned long indx;
962 struct elf_link_hash_entry *h;
964 indx = ELFNN_R_SYM (irel->r_info) - symtab_hdr->sh_info;
965 h = elf_sym_hashes (abfd)[indx];
966 BFD_ASSERT (h != NULL);
968 while (h->root.type == bfd_link_hash_indirect
969 || h->root.type == bfd_link_hash_warning)
970 h = (struct elf_link_hash_entry *) h->root.u.i.link;
972 dyn_i = get_dyn_sym_info (ia64_info, h, abfd, irel, FALSE);
974 /* For branches to dynamic symbols, we're interested instead
975 in a branch to the PLT entry. */
976 if (is_branch && dyn_i && dyn_i->want_plt2)
978 /* Internal branches shouldn't be sent to the PLT.
979 Leave this for now and we'll give an error later. */
980 if (r_type != R_IA64_PCREL21B)
981 continue;
983 tsec = ia64_info->root.splt;
984 toff = dyn_i->plt2_offset;
985 BFD_ASSERT (irel->r_addend == 0);
988 /* Can't do anything else with dynamic symbols. */
989 else if (elfNN_ia64_dynamic_symbol_p (h, link_info, r_type))
990 continue;
992 else
994 /* We can't do anything with undefined symbols. */
995 if (h->root.type == bfd_link_hash_undefined
996 || h->root.type == bfd_link_hash_undefweak)
997 continue;
999 tsec = h->root.u.def.section;
1000 toff = h->root.u.def.value;
1003 symtype = h->type;
1006 if (tsec->sec_info_type == ELF_INFO_TYPE_MERGE)
1008 /* At this stage in linking, no SEC_MERGE symbol has been
1009 adjusted, so all references to such symbols need to be
1010 passed through _bfd_merged_section_offset. (Later, in
1011 relocate_section, all SEC_MERGE symbols *except* for
1012 section symbols have been adjusted.)
1014 gas may reduce relocations against symbols in SEC_MERGE
1015 sections to a relocation against the section symbol when
1016 the original addend was zero. When the reloc is against
1017 a section symbol we should include the addend in the
1018 offset passed to _bfd_merged_section_offset, since the
1019 location of interest is the original symbol. On the
1020 other hand, an access to "sym+addend" where "sym" is not
1021 a section symbol should not include the addend; Such an
1022 access is presumed to be an offset from "sym"; The
1023 location of interest is just "sym". */
1024 if (symtype == STT_SECTION)
1025 toff += irel->r_addend;
1027 toff = _bfd_merged_section_offset (abfd, &tsec,
1028 elf_section_data (tsec)->sec_info,
1029 toff);
1031 if (symtype != STT_SECTION)
1032 toff += irel->r_addend;
1034 else
1035 toff += irel->r_addend;
1037 symaddr = tsec->output_section->vma + tsec->output_offset + toff;
1039 roff = irel->r_offset;
1041 if (is_branch)
1043 bfd_signed_vma offset;
1045 reladdr = (sec->output_section->vma
1046 + sec->output_offset
1047 + roff) & (bfd_vma) -4;
1049 /* The .plt section is aligned at 32byte and the .text section
1050 is aligned at 64byte. The .text section is right after the
1051 .plt section. After the first relaxation pass, linker may
1052 increase the gap between the .plt and .text sections up
1053 to 32byte. We assume linker will always insert 32byte
1054 between the .plt and .text sections after the the first
1055 relaxation pass. */
1056 if (tsec == ia64_info->root.splt)
1057 offset = -0x1000000 + 32;
1058 else
1059 offset = -0x1000000;
1061 /* If the branch is in range, no need to do anything. */
1062 if ((bfd_signed_vma) (symaddr - reladdr) >= offset
1063 && (bfd_signed_vma) (symaddr - reladdr) <= 0x0FFFFF0)
1065 /* If the 60-bit branch is in 21-bit range, optimize it. */
1066 if (r_type == R_IA64_PCREL60B)
1068 elfNN_ia64_relax_brl (contents, roff);
1070 irel->r_info
1071 = ELFNN_R_INFO (ELFNN_R_SYM (irel->r_info),
1072 R_IA64_PCREL21B);
1074 /* If the original relocation offset points to slot
1075 1, change it to slot 2. */
1076 if ((irel->r_offset & 3) == 1)
1077 irel->r_offset += 1;
1080 continue;
1082 else if (r_type == R_IA64_PCREL60B)
1083 continue;
1084 else if (elfNN_ia64_relax_br (contents, roff))
1086 irel->r_info
1087 = ELFNN_R_INFO (ELFNN_R_SYM (irel->r_info),
1088 R_IA64_PCREL60B);
1090 /* Make the relocation offset point to slot 1. */
1091 irel->r_offset = (irel->r_offset & ~((bfd_vma) 0x3)) + 1;
1092 continue;
1095 /* We can't put a trampoline in a .init/.fini section. Issue
1096 an error. */
1097 if (strcmp (sec->output_section->name, ".init") == 0
1098 || strcmp (sec->output_section->name, ".fini") == 0)
1100 (*_bfd_error_handler)
1101 (_("%B: Can't relax br at 0x%lx in section `%A'. Please use brl or indirect branch."),
1102 sec->owner, sec, (unsigned long) roff);
1103 bfd_set_error (bfd_error_bad_value);
1104 goto error_return;
1107 /* If the branch and target are in the same section, you've
1108 got one honking big section and we can't help you unless
1109 you are branching backwards. You'll get an error message
1110 later. */
1111 if (tsec == sec && toff > roff)
1112 continue;
1114 /* Look for an existing fixup to this address. */
1115 for (f = fixups; f ; f = f->next)
1116 if (f->tsec == tsec && f->toff == toff)
1117 break;
1119 if (f == NULL)
1121 /* Two alternatives: If it's a branch to a PLT entry, we can
1122 make a copy of the FULL_PLT entry. Otherwise, we'll have
1123 to use a `brl' insn to get where we're going. */
1125 size_t size;
1127 if (tsec == ia64_info->root.splt)
1128 size = sizeof (plt_full_entry);
1129 else
1130 size = oor_branch_size;
1132 /* Resize the current section to make room for the new branch. */
1133 trampoff = (sec->size + 15) & (bfd_vma) -16;
1135 /* If trampoline is out of range, there is nothing we
1136 can do. */
1137 offset = trampoff - (roff & (bfd_vma) -4);
1138 if (offset < -0x1000000 || offset > 0x0FFFFF0)
1139 continue;
1141 amt = trampoff + size;
1142 contents = (bfd_byte *) bfd_realloc (contents, amt);
1143 if (contents == NULL)
1144 goto error_return;
1145 sec->size = amt;
1147 if (tsec == ia64_info->root.splt)
1149 memcpy (contents + trampoff, plt_full_entry, size);
1151 /* Hijack the old relocation for use as the PLTOFF reloc. */
1152 irel->r_info = ELFNN_R_INFO (ELFNN_R_SYM (irel->r_info),
1153 R_IA64_PLTOFF22);
1154 irel->r_offset = trampoff;
1156 else
1158 if (size == sizeof (oor_ip))
1160 memcpy (contents + trampoff, oor_ip, size);
1161 irel->r_info = ELFNN_R_INFO (ELFNN_R_SYM (irel->r_info),
1162 R_IA64_PCREL64I);
1163 irel->r_addend -= 16;
1164 irel->r_offset = trampoff + 2;
1166 else
1168 memcpy (contents + trampoff, oor_brl, size);
1169 irel->r_info = ELFNN_R_INFO (ELFNN_R_SYM (irel->r_info),
1170 R_IA64_PCREL60B);
1171 irel->r_offset = trampoff + 2;
1176 /* Record the fixup so we don't do it again this section. */
1177 f = (struct one_fixup *)
1178 bfd_malloc ((bfd_size_type) sizeof (*f));
1179 f->next = fixups;
1180 f->tsec = tsec;
1181 f->toff = toff;
1182 f->trampoff = trampoff;
1183 fixups = f;
1185 else
1187 /* If trampoline is out of range, there is nothing we
1188 can do. */
1189 offset = f->trampoff - (roff & (bfd_vma) -4);
1190 if (offset < -0x1000000 || offset > 0x0FFFFF0)
1191 continue;
1193 /* Nop out the reloc, since we're finalizing things here. */
1194 irel->r_info = ELFNN_R_INFO (0, R_IA64_NONE);
1197 /* Fix up the existing branch to hit the trampoline. */
1198 if (elfNN_ia64_install_value (contents + roff, offset, r_type)
1199 != bfd_reloc_ok)
1200 goto error_return;
1202 changed_contents = TRUE;
1203 changed_relocs = TRUE;
1205 else
1207 /* Fetch the gp. */
1208 if (gp == 0)
1210 bfd *obfd = sec->output_section->owner;
1211 gp = _bfd_get_gp_value (obfd);
1212 if (gp == 0)
1214 if (!elfNN_ia64_choose_gp (obfd, link_info))
1215 goto error_return;
1216 gp = _bfd_get_gp_value (obfd);
1220 /* If the data is out of range, do nothing. */
1221 if ((bfd_signed_vma) (symaddr - gp) >= 0x200000
1222 ||(bfd_signed_vma) (symaddr - gp) < -0x200000)
1223 continue;
1225 if (r_type == R_IA64_GPREL22)
1226 elfNN_ia64_update_short_info (tsec->output_section,
1227 tsec->output_offset + toff,
1228 ia64_info);
1229 else if (r_type == R_IA64_LTOFF22X)
1231 irel->r_info = ELFNN_R_INFO (ELFNN_R_SYM (irel->r_info),
1232 R_IA64_GPREL22);
1233 changed_relocs = TRUE;
1234 if (dyn_i->want_gotx)
1236 dyn_i->want_gotx = 0;
1237 changed_got |= !dyn_i->want_got;
1240 elfNN_ia64_update_short_info (tsec->output_section,
1241 tsec->output_offset + toff,
1242 ia64_info);
1244 else
1246 elfNN_ia64_relax_ldxmov (contents, roff);
1247 irel->r_info = ELFNN_R_INFO (0, R_IA64_NONE);
1248 changed_contents = TRUE;
1249 changed_relocs = TRUE;
1254 /* ??? If we created fixups, this may push the code segment large
1255 enough that the data segment moves, which will change the GP.
1256 Reset the GP so that we re-calculate next round. We need to
1257 do this at the _beginning_ of the next round; now will not do. */
1259 /* Clean up and go home. */
1260 while (fixups)
1262 struct one_fixup *f = fixups;
1263 fixups = fixups->next;
1264 free (f);
1267 if (isymbuf != NULL
1268 && symtab_hdr->contents != (unsigned char *) isymbuf)
1270 if (! link_info->keep_memory)
1271 free (isymbuf);
1272 else
1274 /* Cache the symbols for elf_link_input_bfd. */
1275 symtab_hdr->contents = (unsigned char *) isymbuf;
1279 if (contents != NULL
1280 && elf_section_data (sec)->this_hdr.contents != contents)
1282 if (!changed_contents && !link_info->keep_memory)
1283 free (contents);
1284 else
1286 /* Cache the section contents for elf_link_input_bfd. */
1287 elf_section_data (sec)->this_hdr.contents = contents;
1291 if (elf_section_data (sec)->relocs != internal_relocs)
1293 if (!changed_relocs)
1294 free (internal_relocs);
1295 else
1296 elf_section_data (sec)->relocs = internal_relocs;
1299 if (changed_got)
1301 struct elfNN_ia64_allocate_data data;
1302 data.info = link_info;
1303 data.ofs = 0;
1304 ia64_info->self_dtpmod_offset = (bfd_vma) -1;
1306 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_global_data_got, &data);
1307 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_global_fptr_got, &data);
1308 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_local_got, &data);
1309 ia64_info->root.sgot->size = data.ofs;
1311 if (ia64_info->root.dynamic_sections_created
1312 && ia64_info->root.srelgot != NULL)
1314 /* Resize .rela.got. */
1315 ia64_info->root.srelgot->size = 0;
1316 if (link_info->shared
1317 && ia64_info->self_dtpmod_offset != (bfd_vma) -1)
1318 ia64_info->root.srelgot->size += sizeof (ElfNN_External_Rela);
1319 data.only_got = TRUE;
1320 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_dynrel_entries,
1321 &data);
1325 if (link_info->relax_pass == 0)
1327 /* Pass 0 is only needed to relax br. */
1328 sec->skip_relax_pass_0 = skip_relax_pass_0;
1329 sec->skip_relax_pass_1 = skip_relax_pass_1;
1332 *again = changed_contents || changed_relocs;
1333 return TRUE;
1335 error_return:
1336 if (isymbuf != NULL && (unsigned char *) isymbuf != symtab_hdr->contents)
1337 free (isymbuf);
1338 if (contents != NULL
1339 && elf_section_data (sec)->this_hdr.contents != contents)
1340 free (contents);
1341 if (internal_relocs != NULL
1342 && elf_section_data (sec)->relocs != internal_relocs)
1343 free (internal_relocs);
1344 return FALSE;
1346 #undef skip_relax_pass_0
1347 #undef skip_relax_pass_1
1349 static void
1350 elfNN_ia64_relax_ldxmov (bfd_byte *contents, bfd_vma off)
1352 int shift, r1, r3;
1353 bfd_vma dword, insn;
1355 switch ((int)off & 0x3)
1357 case 0: shift = 5; break;
1358 case 1: shift = 14; off += 3; break;
1359 case 2: shift = 23; off += 6; break;
1360 default:
1361 abort ();
1364 dword = bfd_getl64 (contents + off);
1365 insn = (dword >> shift) & 0x1ffffffffffLL;
1367 r1 = (insn >> 6) & 127;
1368 r3 = (insn >> 20) & 127;
1369 if (r1 == r3)
1370 insn = 0x8000000; /* nop */
1371 else
1372 insn = (insn & 0x7f01fff) | 0x10800000000LL; /* (qp) mov r1 = r3 */
1374 dword &= ~(0x1ffffffffffLL << shift);
1375 dword |= (insn << shift);
1376 bfd_putl64 (dword, contents + off);
1379 /* Return TRUE if NAME is an unwind table section name. */
1381 static inline bfd_boolean
1382 is_unwind_section_name (bfd *abfd, const char *name)
1384 if (elfNN_ia64_hpux_vec (abfd->xvec)
1385 && !strcmp (name, ELF_STRING_ia64_unwind_hdr))
1386 return FALSE;
1388 return ((CONST_STRNEQ (name, ELF_STRING_ia64_unwind)
1389 && ! CONST_STRNEQ (name, ELF_STRING_ia64_unwind_info))
1390 || CONST_STRNEQ (name, ELF_STRING_ia64_unwind_once));
1393 /* Handle an IA-64 specific section when reading an object file. This
1394 is called when bfd_section_from_shdr finds a section with an unknown
1395 type. */
1397 static bfd_boolean
1398 elfNN_ia64_section_from_shdr (bfd *abfd,
1399 Elf_Internal_Shdr *hdr,
1400 const char *name,
1401 int shindex)
1403 asection *newsect;
1405 /* There ought to be a place to keep ELF backend specific flags, but
1406 at the moment there isn't one. We just keep track of the
1407 sections by their name, instead. Fortunately, the ABI gives
1408 suggested names for all the MIPS specific sections, so we will
1409 probably get away with this. */
1410 switch (hdr->sh_type)
1412 case SHT_IA_64_UNWIND:
1413 case SHT_IA_64_HP_OPT_ANOT:
1414 break;
1416 case SHT_IA_64_EXT:
1417 if (strcmp (name, ELF_STRING_ia64_archext) != 0)
1418 return FALSE;
1419 break;
1421 default:
1422 return FALSE;
1425 if (! _bfd_elf_make_section_from_shdr (abfd, hdr, name, shindex))
1426 return FALSE;
1427 newsect = hdr->bfd_section;
1429 return TRUE;
1432 /* Convert IA-64 specific section flags to bfd internal section flags. */
1434 /* ??? There is no bfd internal flag equivalent to the SHF_IA_64_NORECOV
1435 flag. */
1437 static bfd_boolean
1438 elfNN_ia64_section_flags (flagword *flags,
1439 const Elf_Internal_Shdr *hdr)
1441 if (hdr->sh_flags & SHF_IA_64_SHORT)
1442 *flags |= SEC_SMALL_DATA;
1444 return TRUE;
1447 /* Set the correct type for an IA-64 ELF section. We do this by the
1448 section name, which is a hack, but ought to work. */
1450 static bfd_boolean
1451 elfNN_ia64_fake_sections (bfd *abfd, Elf_Internal_Shdr *hdr,
1452 asection *sec)
1454 const char *name;
1456 name = bfd_get_section_name (abfd, sec);
1458 if (is_unwind_section_name (abfd, name))
1460 /* We don't have the sections numbered at this point, so sh_info
1461 is set later, in elfNN_ia64_final_write_processing. */
1462 hdr->sh_type = SHT_IA_64_UNWIND;
1463 hdr->sh_flags |= SHF_LINK_ORDER;
1465 else if (strcmp (name, ELF_STRING_ia64_archext) == 0)
1466 hdr->sh_type = SHT_IA_64_EXT;
1467 else if (strcmp (name, ".HP.opt_annot") == 0)
1468 hdr->sh_type = SHT_IA_64_HP_OPT_ANOT;
1469 else if (strcmp (name, ".reloc") == 0)
1470 /* This is an ugly, but unfortunately necessary hack that is
1471 needed when producing EFI binaries on IA-64. It tells
1472 elf.c:elf_fake_sections() not to consider ".reloc" as a section
1473 containing ELF relocation info. We need this hack in order to
1474 be able to generate ELF binaries that can be translated into
1475 EFI applications (which are essentially COFF objects). Those
1476 files contain a COFF ".reloc" section inside an ELFNN object,
1477 which would normally cause BFD to segfault because it would
1478 attempt to interpret this section as containing relocation
1479 entries for section "oc". With this hack enabled, ".reloc"
1480 will be treated as a normal data section, which will avoid the
1481 segfault. However, you won't be able to create an ELFNN binary
1482 with a section named "oc" that needs relocations, but that's
1483 the kind of ugly side-effects you get when detecting section
1484 types based on their names... In practice, this limitation is
1485 unlikely to bite. */
1486 hdr->sh_type = SHT_PROGBITS;
1488 if (sec->flags & SEC_SMALL_DATA)
1489 hdr->sh_flags |= SHF_IA_64_SHORT;
1491 /* Some HP linkers look for the SHF_IA_64_HP_TLS flag instead of SHF_TLS. */
1493 if (elfNN_ia64_hpux_vec (abfd->xvec) && (sec->flags & SHF_TLS))
1494 hdr->sh_flags |= SHF_IA_64_HP_TLS;
1496 return TRUE;
1499 /* The final processing done just before writing out an IA-64 ELF
1500 object file. */
1502 static void
1503 elfNN_ia64_final_write_processing (bfd *abfd,
1504 bfd_boolean linker ATTRIBUTE_UNUSED)
1506 Elf_Internal_Shdr *hdr;
1507 asection *s;
1509 for (s = abfd->sections; s; s = s->next)
1511 hdr = &elf_section_data (s)->this_hdr;
1512 switch (hdr->sh_type)
1514 case SHT_IA_64_UNWIND:
1515 /* The IA-64 processor-specific ABI requires setting sh_link
1516 to the unwind section, whereas HP-UX requires sh_info to
1517 do so. For maximum compatibility, we'll set both for
1518 now... */
1519 hdr->sh_info = hdr->sh_link;
1520 break;
1524 if (! elf_flags_init (abfd))
1526 unsigned long flags = 0;
1528 if (abfd->xvec->byteorder == BFD_ENDIAN_BIG)
1529 flags |= EF_IA_64_BE;
1530 if (bfd_get_mach (abfd) == bfd_mach_ia64_elf64)
1531 flags |= EF_IA_64_ABI64;
1533 elf_elfheader(abfd)->e_flags = flags;
1534 elf_flags_init (abfd) = TRUE;
1538 /* Hook called by the linker routine which adds symbols from an object
1539 file. We use it to put .comm items in .sbss, and not .bss. */
1541 static bfd_boolean
1542 elfNN_ia64_add_symbol_hook (bfd *abfd,
1543 struct bfd_link_info *info,
1544 Elf_Internal_Sym *sym,
1545 const char **namep ATTRIBUTE_UNUSED,
1546 flagword *flagsp ATTRIBUTE_UNUSED,
1547 asection **secp,
1548 bfd_vma *valp)
1550 if (sym->st_shndx == SHN_COMMON
1551 && !info->relocatable
1552 && sym->st_size <= elf_gp_size (abfd))
1554 /* Common symbols less than or equal to -G nn bytes are
1555 automatically put into .sbss. */
1557 asection *scomm = bfd_get_section_by_name (abfd, ".scommon");
1559 if (scomm == NULL)
1561 scomm = bfd_make_section_with_flags (abfd, ".scommon",
1562 (SEC_ALLOC
1563 | SEC_IS_COMMON
1564 | SEC_LINKER_CREATED));
1565 if (scomm == NULL)
1566 return FALSE;
1569 *secp = scomm;
1570 *valp = sym->st_size;
1573 return TRUE;
1576 /* Return the number of additional phdrs we will need. */
1578 static int
1579 elfNN_ia64_additional_program_headers (bfd *abfd,
1580 struct bfd_link_info *info ATTRIBUTE_UNUSED)
1582 asection *s;
1583 int ret = 0;
1585 /* See if we need a PT_IA_64_ARCHEXT segment. */
1586 s = bfd_get_section_by_name (abfd, ELF_STRING_ia64_archext);
1587 if (s && (s->flags & SEC_LOAD))
1588 ++ret;
1590 /* Count how many PT_IA_64_UNWIND segments we need. */
1591 for (s = abfd->sections; s; s = s->next)
1592 if (is_unwind_section_name (abfd, s->name) && (s->flags & SEC_LOAD))
1593 ++ret;
1595 return ret;
1598 static bfd_boolean
1599 elfNN_ia64_modify_segment_map (bfd *abfd,
1600 struct bfd_link_info *info ATTRIBUTE_UNUSED)
1602 struct elf_segment_map *m, **pm;
1603 Elf_Internal_Shdr *hdr;
1604 asection *s;
1606 /* If we need a PT_IA_64_ARCHEXT segment, it must come before
1607 all PT_LOAD segments. */
1608 s = bfd_get_section_by_name (abfd, ELF_STRING_ia64_archext);
1609 if (s && (s->flags & SEC_LOAD))
1611 for (m = elf_tdata (abfd)->segment_map; m != NULL; m = m->next)
1612 if (m->p_type == PT_IA_64_ARCHEXT)
1613 break;
1614 if (m == NULL)
1616 m = ((struct elf_segment_map *)
1617 bfd_zalloc (abfd, (bfd_size_type) sizeof *m));
1618 if (m == NULL)
1619 return FALSE;
1621 m->p_type = PT_IA_64_ARCHEXT;
1622 m->count = 1;
1623 m->sections[0] = s;
1625 /* We want to put it after the PHDR and INTERP segments. */
1626 pm = &elf_tdata (abfd)->segment_map;
1627 while (*pm != NULL
1628 && ((*pm)->p_type == PT_PHDR
1629 || (*pm)->p_type == PT_INTERP))
1630 pm = &(*pm)->next;
1632 m->next = *pm;
1633 *pm = m;
1637 /* Install PT_IA_64_UNWIND segments, if needed. */
1638 for (s = abfd->sections; s; s = s->next)
1640 hdr = &elf_section_data (s)->this_hdr;
1641 if (hdr->sh_type != SHT_IA_64_UNWIND)
1642 continue;
1644 if (s && (s->flags & SEC_LOAD))
1646 for (m = elf_tdata (abfd)->segment_map; m != NULL; m = m->next)
1647 if (m->p_type == PT_IA_64_UNWIND)
1649 int i;
1651 /* Look through all sections in the unwind segment
1652 for a match since there may be multiple sections
1653 to a segment. */
1654 for (i = m->count - 1; i >= 0; --i)
1655 if (m->sections[i] == s)
1656 break;
1658 if (i >= 0)
1659 break;
1662 if (m == NULL)
1664 m = ((struct elf_segment_map *)
1665 bfd_zalloc (abfd, (bfd_size_type) sizeof *m));
1666 if (m == NULL)
1667 return FALSE;
1669 m->p_type = PT_IA_64_UNWIND;
1670 m->count = 1;
1671 m->sections[0] = s;
1672 m->next = NULL;
1674 /* We want to put it last. */
1675 pm = &elf_tdata (abfd)->segment_map;
1676 while (*pm != NULL)
1677 pm = &(*pm)->next;
1678 *pm = m;
1683 return TRUE;
1686 /* Turn on PF_IA_64_NORECOV if needed. This involves traversing all of
1687 the input sections for each output section in the segment and testing
1688 for SHF_IA_64_NORECOV on each. */
1690 static bfd_boolean
1691 elfNN_ia64_modify_program_headers (bfd *abfd,
1692 struct bfd_link_info *info ATTRIBUTE_UNUSED)
1694 struct elf_obj_tdata *tdata = elf_tdata (abfd);
1695 struct elf_segment_map *m;
1696 Elf_Internal_Phdr *p;
1698 for (p = tdata->phdr, m = tdata->segment_map; m != NULL; m = m->next, p++)
1699 if (m->p_type == PT_LOAD)
1701 int i;
1702 for (i = m->count - 1; i >= 0; --i)
1704 struct bfd_link_order *order = m->sections[i]->map_head.link_order;
1706 while (order != NULL)
1708 if (order->type == bfd_indirect_link_order)
1710 asection *is = order->u.indirect.section;
1711 bfd_vma flags = elf_section_data(is)->this_hdr.sh_flags;
1712 if (flags & SHF_IA_64_NORECOV)
1714 p->p_flags |= PF_IA_64_NORECOV;
1715 goto found;
1718 order = order->next;
1721 found:;
1724 return TRUE;
1727 /* According to the Tahoe assembler spec, all labels starting with a
1728 '.' are local. */
1730 static bfd_boolean
1731 elfNN_ia64_is_local_label_name (bfd *abfd ATTRIBUTE_UNUSED,
1732 const char *name)
1734 return name[0] == '.';
1737 /* Should we do dynamic things to this symbol? */
1739 static bfd_boolean
1740 elfNN_ia64_dynamic_symbol_p (struct elf_link_hash_entry *h,
1741 struct bfd_link_info *info, int r_type)
1743 bfd_boolean ignore_protected
1744 = ((r_type & 0xf8) == 0x40 /* FPTR relocs */
1745 || (r_type & 0xf8) == 0x50); /* LTOFF_FPTR relocs */
1747 return _bfd_elf_dynamic_symbol_p (h, info, ignore_protected);
1750 static struct bfd_hash_entry*
1751 elfNN_ia64_new_elf_hash_entry (struct bfd_hash_entry *entry,
1752 struct bfd_hash_table *table,
1753 const char *string)
1755 struct elfNN_ia64_link_hash_entry *ret;
1756 ret = (struct elfNN_ia64_link_hash_entry *) entry;
1758 /* Allocate the structure if it has not already been allocated by a
1759 subclass. */
1760 if (!ret)
1761 ret = bfd_hash_allocate (table, sizeof (*ret));
1763 if (!ret)
1764 return 0;
1766 /* Call the allocation method of the superclass. */
1767 ret = ((struct elfNN_ia64_link_hash_entry *)
1768 _bfd_elf_link_hash_newfunc ((struct bfd_hash_entry *) ret,
1769 table, string));
1771 ret->info = NULL;
1772 ret->count = 0;
1773 ret->sorted_count = 0;
1774 ret->size = 0;
1775 return (struct bfd_hash_entry *) ret;
1778 static void
1779 elfNN_ia64_hash_copy_indirect (struct bfd_link_info *info,
1780 struct elf_link_hash_entry *xdir,
1781 struct elf_link_hash_entry *xind)
1783 struct elfNN_ia64_link_hash_entry *dir, *ind;
1785 dir = (struct elfNN_ia64_link_hash_entry *) xdir;
1786 ind = (struct elfNN_ia64_link_hash_entry *) xind;
1788 /* Copy down any references that we may have already seen to the
1789 symbol which just became indirect. */
1791 dir->root.ref_dynamic |= ind->root.ref_dynamic;
1792 dir->root.ref_regular |= ind->root.ref_regular;
1793 dir->root.ref_regular_nonweak |= ind->root.ref_regular_nonweak;
1794 dir->root.needs_plt |= ind->root.needs_plt;
1796 if (ind->root.root.type != bfd_link_hash_indirect)
1797 return;
1799 /* Copy over the got and plt data. This would have been done
1800 by check_relocs. */
1802 if (ind->info != NULL)
1804 struct elfNN_ia64_dyn_sym_info *dyn_i;
1805 unsigned int count;
1807 if (dir->info)
1808 free (dir->info);
1810 dir->info = ind->info;
1811 dir->count = ind->count;
1812 dir->sorted_count = ind->sorted_count;
1813 dir->size = ind->size;
1815 ind->info = NULL;
1816 ind->count = 0;
1817 ind->sorted_count = 0;
1818 ind->size = 0;
1820 /* Fix up the dyn_sym_info pointers to the global symbol. */
1821 for (count = dir->count, dyn_i = dir->info;
1822 count != 0;
1823 count--, dyn_i++)
1824 dyn_i->h = &dir->root;
1827 /* Copy over the dynindx. */
1829 if (ind->root.dynindx != -1)
1831 if (dir->root.dynindx != -1)
1832 _bfd_elf_strtab_delref (elf_hash_table (info)->dynstr,
1833 dir->root.dynstr_index);
1834 dir->root.dynindx = ind->root.dynindx;
1835 dir->root.dynstr_index = ind->root.dynstr_index;
1836 ind->root.dynindx = -1;
1837 ind->root.dynstr_index = 0;
1841 static void
1842 elfNN_ia64_hash_hide_symbol (struct bfd_link_info *info,
1843 struct elf_link_hash_entry *xh,
1844 bfd_boolean force_local)
1846 struct elfNN_ia64_link_hash_entry *h;
1847 struct elfNN_ia64_dyn_sym_info *dyn_i;
1848 unsigned int count;
1850 h = (struct elfNN_ia64_link_hash_entry *)xh;
1852 _bfd_elf_link_hash_hide_symbol (info, &h->root, force_local);
1854 for (count = h->count, dyn_i = h->info;
1855 count != 0;
1856 count--, dyn_i++)
1858 dyn_i->want_plt2 = 0;
1859 dyn_i->want_plt = 0;
1863 /* Compute a hash of a local hash entry. */
1865 static hashval_t
1866 elfNN_ia64_local_htab_hash (const void *ptr)
1868 struct elfNN_ia64_local_hash_entry *entry
1869 = (struct elfNN_ia64_local_hash_entry *) ptr;
1871 return ELF_LOCAL_SYMBOL_HASH (entry->id, entry->r_sym);
1874 /* Compare local hash entries. */
1876 static int
1877 elfNN_ia64_local_htab_eq (const void *ptr1, const void *ptr2)
1879 struct elfNN_ia64_local_hash_entry *entry1
1880 = (struct elfNN_ia64_local_hash_entry *) ptr1;
1881 struct elfNN_ia64_local_hash_entry *entry2
1882 = (struct elfNN_ia64_local_hash_entry *) ptr2;
1884 return entry1->id == entry2->id && entry1->r_sym == entry2->r_sym;
1887 /* Create the derived linker hash table. The IA-64 ELF port uses this
1888 derived hash table to keep information specific to the IA-64 ElF
1889 linker (without using static variables). */
1891 static struct bfd_link_hash_table *
1892 elfNN_ia64_hash_table_create (bfd *abfd)
1894 struct elfNN_ia64_link_hash_table *ret;
1896 ret = bfd_zmalloc ((bfd_size_type) sizeof (*ret));
1897 if (!ret)
1898 return NULL;
1900 if (!_bfd_elf_link_hash_table_init (&ret->root, abfd,
1901 elfNN_ia64_new_elf_hash_entry,
1902 sizeof (struct elfNN_ia64_link_hash_entry),
1903 IA64_ELF_DATA))
1905 free (ret);
1906 return NULL;
1909 ret->loc_hash_table = htab_try_create (1024, elfNN_ia64_local_htab_hash,
1910 elfNN_ia64_local_htab_eq, NULL);
1911 ret->loc_hash_memory = objalloc_create ();
1912 if (!ret->loc_hash_table || !ret->loc_hash_memory)
1914 free (ret);
1915 return NULL;
1918 return &ret->root.root;
1921 /* Free the global elfNN_ia64_dyn_sym_info array. */
1923 static bfd_boolean
1924 elfNN_ia64_global_dyn_info_free (void **xentry,
1925 PTR unused ATTRIBUTE_UNUSED)
1927 struct elfNN_ia64_link_hash_entry *entry
1928 = (struct elfNN_ia64_link_hash_entry *) xentry;
1930 if (entry->root.root.type == bfd_link_hash_warning)
1931 entry = (struct elfNN_ia64_link_hash_entry *) entry->root.root.u.i.link;
1933 if (entry->info)
1935 free (entry->info);
1936 entry->info = NULL;
1937 entry->count = 0;
1938 entry->sorted_count = 0;
1939 entry->size = 0;
1942 return TRUE;
1945 /* Free the local elfNN_ia64_dyn_sym_info array. */
1947 static bfd_boolean
1948 elfNN_ia64_local_dyn_info_free (void **slot,
1949 PTR unused ATTRIBUTE_UNUSED)
1951 struct elfNN_ia64_local_hash_entry *entry
1952 = (struct elfNN_ia64_local_hash_entry *) *slot;
1954 if (entry->info)
1956 free (entry->info);
1957 entry->info = NULL;
1958 entry->count = 0;
1959 entry->sorted_count = 0;
1960 entry->size = 0;
1963 return TRUE;
1966 /* Destroy IA-64 linker hash table. */
1968 static void
1969 elfNN_ia64_hash_table_free (struct bfd_link_hash_table *hash)
1971 struct elfNN_ia64_link_hash_table *ia64_info
1972 = (struct elfNN_ia64_link_hash_table *) hash;
1973 if (ia64_info->loc_hash_table)
1975 htab_traverse (ia64_info->loc_hash_table,
1976 elfNN_ia64_local_dyn_info_free, NULL);
1977 htab_delete (ia64_info->loc_hash_table);
1979 if (ia64_info->loc_hash_memory)
1980 objalloc_free ((struct objalloc *) ia64_info->loc_hash_memory);
1981 elf_link_hash_traverse (&ia64_info->root,
1982 elfNN_ia64_global_dyn_info_free, NULL);
1983 _bfd_generic_link_hash_table_free (hash);
1986 /* Traverse both local and global hash tables. */
1988 struct elfNN_ia64_dyn_sym_traverse_data
1990 bfd_boolean (*func) (struct elfNN_ia64_dyn_sym_info *, PTR);
1991 PTR data;
1994 static bfd_boolean
1995 elfNN_ia64_global_dyn_sym_thunk (struct bfd_hash_entry *xentry,
1996 PTR xdata)
1998 struct elfNN_ia64_link_hash_entry *entry
1999 = (struct elfNN_ia64_link_hash_entry *) xentry;
2000 struct elfNN_ia64_dyn_sym_traverse_data *data
2001 = (struct elfNN_ia64_dyn_sym_traverse_data *) xdata;
2002 struct elfNN_ia64_dyn_sym_info *dyn_i;
2003 unsigned int count;
2005 if (entry->root.root.type == bfd_link_hash_warning)
2006 entry = (struct elfNN_ia64_link_hash_entry *) entry->root.root.u.i.link;
2008 for (count = entry->count, dyn_i = entry->info;
2009 count != 0;
2010 count--, dyn_i++)
2011 if (! (*data->func) (dyn_i, data->data))
2012 return FALSE;
2013 return TRUE;
2016 static bfd_boolean
2017 elfNN_ia64_local_dyn_sym_thunk (void **slot, PTR xdata)
2019 struct elfNN_ia64_local_hash_entry *entry
2020 = (struct elfNN_ia64_local_hash_entry *) *slot;
2021 struct elfNN_ia64_dyn_sym_traverse_data *data
2022 = (struct elfNN_ia64_dyn_sym_traverse_data *) xdata;
2023 struct elfNN_ia64_dyn_sym_info *dyn_i;
2024 unsigned int count;
2026 for (count = entry->count, dyn_i = entry->info;
2027 count != 0;
2028 count--, dyn_i++)
2029 if (! (*data->func) (dyn_i, data->data))
2030 return FALSE;
2031 return TRUE;
2034 static void
2035 elfNN_ia64_dyn_sym_traverse (struct elfNN_ia64_link_hash_table *ia64_info,
2036 bfd_boolean (*func) (struct elfNN_ia64_dyn_sym_info *, PTR),
2037 PTR data)
2039 struct elfNN_ia64_dyn_sym_traverse_data xdata;
2041 xdata.func = func;
2042 xdata.data = data;
2044 elf_link_hash_traverse (&ia64_info->root,
2045 elfNN_ia64_global_dyn_sym_thunk, &xdata);
2046 htab_traverse (ia64_info->loc_hash_table,
2047 elfNN_ia64_local_dyn_sym_thunk, &xdata);
2050 static bfd_boolean
2051 elfNN_ia64_create_dynamic_sections (bfd *abfd,
2052 struct bfd_link_info *info)
2054 struct elfNN_ia64_link_hash_table *ia64_info;
2055 asection *s;
2057 if (! _bfd_elf_create_dynamic_sections (abfd, info))
2058 return FALSE;
2060 ia64_info = elfNN_ia64_hash_table (info);
2061 if (ia64_info == NULL)
2062 return FALSE;
2065 flagword flags = bfd_get_section_flags (abfd, ia64_info->root.sgot);
2066 bfd_set_section_flags (abfd, ia64_info->root.sgot,
2067 SEC_SMALL_DATA | flags);
2068 /* The .got section is always aligned at 8 bytes. */
2069 bfd_set_section_alignment (abfd, ia64_info->root.sgot, 3);
2072 if (!get_pltoff (abfd, info, ia64_info))
2073 return FALSE;
2075 s = bfd_make_section_with_flags (abfd, ".rela.IA_64.pltoff",
2076 (SEC_ALLOC | SEC_LOAD
2077 | SEC_HAS_CONTENTS
2078 | SEC_IN_MEMORY
2079 | SEC_LINKER_CREATED
2080 | SEC_READONLY));
2081 if (s == NULL
2082 || !bfd_set_section_alignment (abfd, s, LOG_SECTION_ALIGN))
2083 return FALSE;
2084 ia64_info->rel_pltoff_sec = s;
2086 return TRUE;
2089 /* Find and/or create a hash entry for local symbol. */
2090 static struct elfNN_ia64_local_hash_entry *
2091 get_local_sym_hash (struct elfNN_ia64_link_hash_table *ia64_info,
2092 bfd *abfd, const Elf_Internal_Rela *rel,
2093 bfd_boolean create)
2095 struct elfNN_ia64_local_hash_entry e, *ret;
2096 asection *sec = abfd->sections;
2097 hashval_t h = ELF_LOCAL_SYMBOL_HASH (sec->id,
2098 ELFNN_R_SYM (rel->r_info));
2099 void **slot;
2101 e.id = sec->id;
2102 e.r_sym = ELFNN_R_SYM (rel->r_info);
2103 slot = htab_find_slot_with_hash (ia64_info->loc_hash_table, &e, h,
2104 create ? INSERT : NO_INSERT);
2106 if (!slot)
2107 return NULL;
2109 if (*slot)
2110 return (struct elfNN_ia64_local_hash_entry *) *slot;
2112 ret = (struct elfNN_ia64_local_hash_entry *)
2113 objalloc_alloc ((struct objalloc *) ia64_info->loc_hash_memory,
2114 sizeof (struct elfNN_ia64_local_hash_entry));
2115 if (ret)
2117 memset (ret, 0, sizeof (*ret));
2118 ret->id = sec->id;
2119 ret->r_sym = ELFNN_R_SYM (rel->r_info);
2120 *slot = ret;
2122 return ret;
2125 /* Used to sort elfNN_ia64_dyn_sym_info array. */
2127 static int
2128 addend_compare (const void *xp, const void *yp)
2130 const struct elfNN_ia64_dyn_sym_info *x
2131 = (const struct elfNN_ia64_dyn_sym_info *) xp;
2132 const struct elfNN_ia64_dyn_sym_info *y
2133 = (const struct elfNN_ia64_dyn_sym_info *) yp;
2135 return x->addend < y->addend ? -1 : x->addend > y->addend ? 1 : 0;
2138 /* Sort elfNN_ia64_dyn_sym_info array and remove duplicates. */
2140 static unsigned int
2141 sort_dyn_sym_info (struct elfNN_ia64_dyn_sym_info *info,
2142 unsigned int count)
2144 bfd_vma curr, prev, got_offset;
2145 unsigned int i, kept, dupes, diff, dest, src, len;
2147 qsort (info, count, sizeof (*info), addend_compare);
2149 /* Find the first duplicate. */
2150 prev = info [0].addend;
2151 got_offset = info [0].got_offset;
2152 for (i = 1; i < count; i++)
2154 curr = info [i].addend;
2155 if (curr == prev)
2157 /* For duplicates, make sure that GOT_OFFSET is valid. */
2158 if (got_offset == (bfd_vma) -1)
2159 got_offset = info [i].got_offset;
2160 break;
2162 got_offset = info [i].got_offset;
2163 prev = curr;
2166 /* We may move a block of elements to here. */
2167 dest = i++;
2169 /* Remove duplicates. */
2170 if (i < count)
2172 while (i < count)
2174 /* For duplicates, make sure that the kept one has a valid
2175 got_offset. */
2176 kept = dest - 1;
2177 if (got_offset != (bfd_vma) -1)
2178 info [kept].got_offset = got_offset;
2180 curr = info [i].addend;
2181 got_offset = info [i].got_offset;
2183 /* Move a block of elements whose first one is different from
2184 the previous. */
2185 if (curr == prev)
2187 for (src = i + 1; src < count; src++)
2189 if (info [src].addend != curr)
2190 break;
2191 /* For duplicates, make sure that GOT_OFFSET is
2192 valid. */
2193 if (got_offset == (bfd_vma) -1)
2194 got_offset = info [src].got_offset;
2197 /* Make sure that the kept one has a valid got_offset. */
2198 if (got_offset != (bfd_vma) -1)
2199 info [kept].got_offset = got_offset;
2201 else
2202 src = i;
2204 if (src >= count)
2205 break;
2207 /* Find the next duplicate. SRC will be kept. */
2208 prev = info [src].addend;
2209 got_offset = info [src].got_offset;
2210 for (dupes = src + 1; dupes < count; dupes ++)
2212 curr = info [dupes].addend;
2213 if (curr == prev)
2215 /* Make sure that got_offset is valid. */
2216 if (got_offset == (bfd_vma) -1)
2217 got_offset = info [dupes].got_offset;
2219 /* For duplicates, make sure that the kept one has
2220 a valid got_offset. */
2221 if (got_offset != (bfd_vma) -1)
2222 info [dupes - 1].got_offset = got_offset;
2223 break;
2225 got_offset = info [dupes].got_offset;
2226 prev = curr;
2229 /* How much to move. */
2230 len = dupes - src;
2231 i = dupes + 1;
2233 if (len == 1 && dupes < count)
2235 /* If we only move 1 element, we combine it with the next
2236 one. There must be at least a duplicate. Find the
2237 next different one. */
2238 for (diff = dupes + 1, src++; diff < count; diff++, src++)
2240 if (info [diff].addend != curr)
2241 break;
2242 /* Make sure that got_offset is valid. */
2243 if (got_offset == (bfd_vma) -1)
2244 got_offset = info [diff].got_offset;
2247 /* Makre sure that the last duplicated one has an valid
2248 offset. */
2249 BFD_ASSERT (curr == prev);
2250 if (got_offset != (bfd_vma) -1)
2251 info [diff - 1].got_offset = got_offset;
2253 if (diff < count)
2255 /* Find the next duplicate. Track the current valid
2256 offset. */
2257 prev = info [diff].addend;
2258 got_offset = info [diff].got_offset;
2259 for (dupes = diff + 1; dupes < count; dupes ++)
2261 curr = info [dupes].addend;
2262 if (curr == prev)
2264 /* For duplicates, make sure that GOT_OFFSET
2265 is valid. */
2266 if (got_offset == (bfd_vma) -1)
2267 got_offset = info [dupes].got_offset;
2268 break;
2270 got_offset = info [dupes].got_offset;
2271 prev = curr;
2272 diff++;
2275 len = diff - src + 1;
2276 i = diff + 1;
2280 memmove (&info [dest], &info [src], len * sizeof (*info));
2282 dest += len;
2285 count = dest;
2287 else
2289 /* When we get here, either there is no duplicate at all or
2290 the only duplicate is the last element. */
2291 if (dest < count)
2293 /* If the last element is a duplicate, make sure that the
2294 kept one has a valid got_offset. We also update count. */
2295 if (got_offset != (bfd_vma) -1)
2296 info [dest - 1].got_offset = got_offset;
2297 count = dest;
2301 return count;
2304 /* Find and/or create a descriptor for dynamic symbol info. This will
2305 vary based on global or local symbol, and the addend to the reloc.
2307 We don't sort when inserting. Also, we sort and eliminate
2308 duplicates if there is an unsorted section. Typically, this will
2309 only happen once, because we do all insertions before lookups. We
2310 then use bsearch to do a lookup. This also allows lookups to be
2311 fast. So we have fast insertion (O(log N) due to duplicate check),
2312 fast lookup (O(log N)) and one sort (O(N log N) expected time).
2313 Previously, all lookups were O(N) because of the use of the linked
2314 list and also all insertions were O(N) because of the check for
2315 duplicates. There are some complications here because the array
2316 size grows occasionally, which may add an O(N) factor, but this
2317 should be rare. Also, we free the excess array allocation, which
2318 requires a copy which is O(N), but this only happens once. */
2320 static struct elfNN_ia64_dyn_sym_info *
2321 get_dyn_sym_info (struct elfNN_ia64_link_hash_table *ia64_info,
2322 struct elf_link_hash_entry *h, bfd *abfd,
2323 const Elf_Internal_Rela *rel, bfd_boolean create)
2325 struct elfNN_ia64_dyn_sym_info **info_p, *info, *dyn_i, key;
2326 unsigned int *count_p, *sorted_count_p, *size_p;
2327 unsigned int count, sorted_count, size;
2328 bfd_vma addend = rel ? rel->r_addend : 0;
2329 bfd_size_type amt;
2331 if (h)
2333 struct elfNN_ia64_link_hash_entry *global_h;
2335 global_h = (struct elfNN_ia64_link_hash_entry *) h;
2336 info_p = &global_h->info;
2337 count_p = &global_h->count;
2338 sorted_count_p = &global_h->sorted_count;
2339 size_p = &global_h->size;
2341 else
2343 struct elfNN_ia64_local_hash_entry *loc_h;
2345 loc_h = get_local_sym_hash (ia64_info, abfd, rel, create);
2346 if (!loc_h)
2348 BFD_ASSERT (!create);
2349 return NULL;
2352 info_p = &loc_h->info;
2353 count_p = &loc_h->count;
2354 sorted_count_p = &loc_h->sorted_count;
2355 size_p = &loc_h->size;
2358 count = *count_p;
2359 sorted_count = *sorted_count_p;
2360 size = *size_p;
2361 info = *info_p;
2362 if (create)
2364 /* When we create the array, we don't check for duplicates,
2365 except in the previously sorted section if one exists, and
2366 against the last inserted entry. This allows insertions to
2367 be fast. */
2368 if (info)
2370 if (sorted_count)
2372 /* Try bsearch first on the sorted section. */
2373 key.addend = addend;
2374 dyn_i = bsearch (&key, info, sorted_count,
2375 sizeof (*info), addend_compare);
2377 if (dyn_i)
2379 return dyn_i;
2383 /* Do a quick check for the last inserted entry. */
2384 dyn_i = info + count - 1;
2385 if (dyn_i->addend == addend)
2387 return dyn_i;
2391 if (size == 0)
2393 /* It is the very first element. We create the array of size
2394 1. */
2395 size = 1;
2396 amt = size * sizeof (*info);
2397 info = bfd_malloc (amt);
2399 else if (size <= count)
2401 /* We double the array size every time when we reach the
2402 size limit. */
2403 size += size;
2404 amt = size * sizeof (*info);
2405 info = bfd_realloc (info, amt);
2407 else
2408 goto has_space;
2410 if (info == NULL)
2411 return NULL;
2412 *size_p = size;
2413 *info_p = info;
2415 has_space:
2416 /* Append the new one to the array. */
2417 dyn_i = info + count;
2418 memset (dyn_i, 0, sizeof (*dyn_i));
2419 dyn_i->got_offset = (bfd_vma) -1;
2420 dyn_i->addend = addend;
2422 /* We increment count only since the new ones are unsorted and
2423 may have duplicate. */
2424 (*count_p)++;
2426 else
2428 /* It is a lookup without insertion. Sort array if part of the
2429 array isn't sorted. */
2430 if (count != sorted_count)
2432 count = sort_dyn_sym_info (info, count);
2433 *count_p = count;
2434 *sorted_count_p = count;
2437 /* Free unused memory. */
2438 if (size != count)
2440 amt = count * sizeof (*info);
2441 info = bfd_malloc (amt);
2442 if (info != NULL)
2444 memcpy (info, *info_p, amt);
2445 free (*info_p);
2446 *size_p = count;
2447 *info_p = info;
2451 key.addend = addend;
2452 dyn_i = bsearch (&key, info, count,
2453 sizeof (*info), addend_compare);
2456 return dyn_i;
2459 static asection *
2460 get_got (bfd *abfd, struct bfd_link_info *info,
2461 struct elfNN_ia64_link_hash_table *ia64_info)
2463 asection *got;
2464 bfd *dynobj;
2466 got = ia64_info->root.sgot;
2467 if (!got)
2469 flagword flags;
2471 dynobj = ia64_info->root.dynobj;
2472 if (!dynobj)
2473 ia64_info->root.dynobj = dynobj = abfd;
2474 if (!_bfd_elf_create_got_section (dynobj, info))
2475 return 0;
2477 got = ia64_info->root.sgot;
2479 /* The .got section is always aligned at 8 bytes. */
2480 if (!bfd_set_section_alignment (abfd, got, 3))
2481 return 0;
2483 flags = bfd_get_section_flags (abfd, got);
2484 bfd_set_section_flags (abfd, got, SEC_SMALL_DATA | flags);
2487 return got;
2490 /* Create function descriptor section (.opd). This section is called .opd
2491 because it contains "official procedure descriptors". The "official"
2492 refers to the fact that these descriptors are used when taking the address
2493 of a procedure, thus ensuring a unique address for each procedure. */
2495 static asection *
2496 get_fptr (bfd *abfd, struct bfd_link_info *info,
2497 struct elfNN_ia64_link_hash_table *ia64_info)
2499 asection *fptr;
2500 bfd *dynobj;
2502 fptr = ia64_info->fptr_sec;
2503 if (!fptr)
2505 dynobj = ia64_info->root.dynobj;
2506 if (!dynobj)
2507 ia64_info->root.dynobj = dynobj = abfd;
2509 fptr = bfd_make_section_with_flags (dynobj, ".opd",
2510 (SEC_ALLOC
2511 | SEC_LOAD
2512 | SEC_HAS_CONTENTS
2513 | SEC_IN_MEMORY
2514 | (info->pie ? 0 : SEC_READONLY)
2515 | SEC_LINKER_CREATED));
2516 if (!fptr
2517 || !bfd_set_section_alignment (abfd, fptr, 4))
2519 BFD_ASSERT (0);
2520 return NULL;
2523 ia64_info->fptr_sec = fptr;
2525 if (info->pie)
2527 asection *fptr_rel;
2528 fptr_rel = bfd_make_section_with_flags (dynobj, ".rela.opd",
2529 (SEC_ALLOC | SEC_LOAD
2530 | SEC_HAS_CONTENTS
2531 | SEC_IN_MEMORY
2532 | SEC_LINKER_CREATED
2533 | SEC_READONLY));
2534 if (fptr_rel == NULL
2535 || !bfd_set_section_alignment (abfd, fptr_rel,
2536 LOG_SECTION_ALIGN))
2538 BFD_ASSERT (0);
2539 return NULL;
2542 ia64_info->rel_fptr_sec = fptr_rel;
2546 return fptr;
2549 static asection *
2550 get_pltoff (bfd *abfd, struct bfd_link_info *info ATTRIBUTE_UNUSED,
2551 struct elfNN_ia64_link_hash_table *ia64_info)
2553 asection *pltoff;
2554 bfd *dynobj;
2556 pltoff = ia64_info->pltoff_sec;
2557 if (!pltoff)
2559 dynobj = ia64_info->root.dynobj;
2560 if (!dynobj)
2561 ia64_info->root.dynobj = dynobj = abfd;
2563 pltoff = bfd_make_section_with_flags (dynobj,
2564 ELF_STRING_ia64_pltoff,
2565 (SEC_ALLOC
2566 | SEC_LOAD
2567 | SEC_HAS_CONTENTS
2568 | SEC_IN_MEMORY
2569 | SEC_SMALL_DATA
2570 | SEC_LINKER_CREATED));
2571 if (!pltoff
2572 || !bfd_set_section_alignment (abfd, pltoff, 4))
2574 BFD_ASSERT (0);
2575 return NULL;
2578 ia64_info->pltoff_sec = pltoff;
2581 return pltoff;
2584 static asection *
2585 get_reloc_section (bfd *abfd,
2586 struct elfNN_ia64_link_hash_table *ia64_info,
2587 asection *sec, bfd_boolean create)
2589 const char *srel_name;
2590 asection *srel;
2591 bfd *dynobj;
2593 srel_name = (bfd_elf_string_from_elf_section
2594 (abfd, elf_elfheader(abfd)->e_shstrndx,
2595 elf_section_data(sec)->rel_hdr.sh_name));
2596 if (srel_name == NULL)
2597 return NULL;
2599 BFD_ASSERT ((CONST_STRNEQ (srel_name, ".rela")
2600 && strcmp (bfd_get_section_name (abfd, sec),
2601 srel_name+5) == 0)
2602 || (CONST_STRNEQ (srel_name, ".rel")
2603 && strcmp (bfd_get_section_name (abfd, sec),
2604 srel_name+4) == 0));
2606 dynobj = ia64_info->root.dynobj;
2607 if (!dynobj)
2608 ia64_info->root.dynobj = dynobj = abfd;
2610 srel = bfd_get_section_by_name (dynobj, srel_name);
2611 if (srel == NULL && create)
2613 srel = bfd_make_section_with_flags (dynobj, srel_name,
2614 (SEC_ALLOC | SEC_LOAD
2615 | SEC_HAS_CONTENTS
2616 | SEC_IN_MEMORY
2617 | SEC_LINKER_CREATED
2618 | SEC_READONLY));
2619 if (srel == NULL
2620 || !bfd_set_section_alignment (dynobj, srel,
2621 LOG_SECTION_ALIGN))
2622 return NULL;
2625 return srel;
2628 static bfd_boolean
2629 count_dyn_reloc (bfd *abfd, struct elfNN_ia64_dyn_sym_info *dyn_i,
2630 asection *srel, int type, bfd_boolean reltext)
2632 struct elfNN_ia64_dyn_reloc_entry *rent;
2634 for (rent = dyn_i->reloc_entries; rent; rent = rent->next)
2635 if (rent->srel == srel && rent->type == type)
2636 break;
2638 if (!rent)
2640 rent = ((struct elfNN_ia64_dyn_reloc_entry *)
2641 bfd_alloc (abfd, (bfd_size_type) sizeof (*rent)));
2642 if (!rent)
2643 return FALSE;
2645 rent->next = dyn_i->reloc_entries;
2646 rent->srel = srel;
2647 rent->type = type;
2648 rent->count = 0;
2649 dyn_i->reloc_entries = rent;
2651 rent->reltext = reltext;
2652 rent->count++;
2654 return TRUE;
2657 static bfd_boolean
2658 elfNN_ia64_check_relocs (bfd *abfd, struct bfd_link_info *info,
2659 asection *sec,
2660 const Elf_Internal_Rela *relocs)
2662 struct elfNN_ia64_link_hash_table *ia64_info;
2663 const Elf_Internal_Rela *relend;
2664 Elf_Internal_Shdr *symtab_hdr;
2665 const Elf_Internal_Rela *rel;
2666 asection *got, *fptr, *srel, *pltoff;
2667 enum {
2668 NEED_GOT = 1,
2669 NEED_GOTX = 2,
2670 NEED_FPTR = 4,
2671 NEED_PLTOFF = 8,
2672 NEED_MIN_PLT = 16,
2673 NEED_FULL_PLT = 32,
2674 NEED_DYNREL = 64,
2675 NEED_LTOFF_FPTR = 128,
2676 NEED_TPREL = 256,
2677 NEED_DTPMOD = 512,
2678 NEED_DTPREL = 1024
2680 int need_entry;
2681 struct elf_link_hash_entry *h;
2682 unsigned long r_symndx;
2683 bfd_boolean maybe_dynamic;
2685 if (info->relocatable)
2686 return TRUE;
2688 symtab_hdr = &elf_tdata (abfd)->symtab_hdr;
2689 ia64_info = elfNN_ia64_hash_table (info);
2690 if (ia64_info == NULL)
2691 return FALSE;
2693 got = fptr = srel = pltoff = NULL;
2695 relend = relocs + sec->reloc_count;
2697 /* We scan relocations first to create dynamic relocation arrays. We
2698 modified get_dyn_sym_info to allow fast insertion and support fast
2699 lookup in the next loop. */
2700 for (rel = relocs; rel < relend; ++rel)
2702 r_symndx = ELFNN_R_SYM (rel->r_info);
2703 if (r_symndx >= symtab_hdr->sh_info)
2705 long indx = r_symndx - symtab_hdr->sh_info;
2706 h = elf_sym_hashes (abfd)[indx];
2707 while (h->root.type == bfd_link_hash_indirect
2708 || h->root.type == bfd_link_hash_warning)
2709 h = (struct elf_link_hash_entry *) h->root.u.i.link;
2711 else
2712 h = NULL;
2714 /* We can only get preliminary data on whether a symbol is
2715 locally or externally defined, as not all of the input files
2716 have yet been processed. Do something with what we know, as
2717 this may help reduce memory usage and processing time later. */
2718 maybe_dynamic = (h && ((!info->executable
2719 && (!SYMBOLIC_BIND (info, h)
2720 || info->unresolved_syms_in_shared_libs == RM_IGNORE))
2721 || !h->def_regular
2722 || h->root.type == bfd_link_hash_defweak));
2724 need_entry = 0;
2725 switch (ELFNN_R_TYPE (rel->r_info))
2727 case R_IA64_TPREL64MSB:
2728 case R_IA64_TPREL64LSB:
2729 if (info->shared || maybe_dynamic)
2730 need_entry = NEED_DYNREL;
2731 break;
2733 case R_IA64_LTOFF_TPREL22:
2734 need_entry = NEED_TPREL;
2735 if (info->shared)
2736 info->flags |= DF_STATIC_TLS;
2737 break;
2739 case R_IA64_DTPREL32MSB:
2740 case R_IA64_DTPREL32LSB:
2741 case R_IA64_DTPREL64MSB:
2742 case R_IA64_DTPREL64LSB:
2743 if (info->shared || maybe_dynamic)
2744 need_entry = NEED_DYNREL;
2745 break;
2747 case R_IA64_LTOFF_DTPREL22:
2748 need_entry = NEED_DTPREL;
2749 break;
2751 case R_IA64_DTPMOD64MSB:
2752 case R_IA64_DTPMOD64LSB:
2753 if (info->shared || maybe_dynamic)
2754 need_entry = NEED_DYNREL;
2755 break;
2757 case R_IA64_LTOFF_DTPMOD22:
2758 need_entry = NEED_DTPMOD;
2759 break;
2761 case R_IA64_LTOFF_FPTR22:
2762 case R_IA64_LTOFF_FPTR64I:
2763 case R_IA64_LTOFF_FPTR32MSB:
2764 case R_IA64_LTOFF_FPTR32LSB:
2765 case R_IA64_LTOFF_FPTR64MSB:
2766 case R_IA64_LTOFF_FPTR64LSB:
2767 need_entry = NEED_FPTR | NEED_GOT | NEED_LTOFF_FPTR;
2768 break;
2770 case R_IA64_FPTR64I:
2771 case R_IA64_FPTR32MSB:
2772 case R_IA64_FPTR32LSB:
2773 case R_IA64_FPTR64MSB:
2774 case R_IA64_FPTR64LSB:
2775 if (info->shared || h)
2776 need_entry = NEED_FPTR | NEED_DYNREL;
2777 else
2778 need_entry = NEED_FPTR;
2779 break;
2781 case R_IA64_LTOFF22:
2782 case R_IA64_LTOFF64I:
2783 need_entry = NEED_GOT;
2784 break;
2786 case R_IA64_LTOFF22X:
2787 need_entry = NEED_GOTX;
2788 break;
2790 case R_IA64_PLTOFF22:
2791 case R_IA64_PLTOFF64I:
2792 case R_IA64_PLTOFF64MSB:
2793 case R_IA64_PLTOFF64LSB:
2794 need_entry = NEED_PLTOFF;
2795 if (h)
2797 if (maybe_dynamic)
2798 need_entry |= NEED_MIN_PLT;
2800 else
2802 (*info->callbacks->warning)
2803 (info, _("@pltoff reloc against local symbol"), 0,
2804 abfd, 0, (bfd_vma) 0);
2806 break;
2808 case R_IA64_PCREL21B:
2809 case R_IA64_PCREL60B:
2810 /* Depending on where this symbol is defined, we may or may not
2811 need a full plt entry. Only skip if we know we'll not need
2812 the entry -- static or symbolic, and the symbol definition
2813 has already been seen. */
2814 if (maybe_dynamic && rel->r_addend == 0)
2815 need_entry = NEED_FULL_PLT;
2816 break;
2818 case R_IA64_IMM14:
2819 case R_IA64_IMM22:
2820 case R_IA64_IMM64:
2821 case R_IA64_DIR32MSB:
2822 case R_IA64_DIR32LSB:
2823 case R_IA64_DIR64MSB:
2824 case R_IA64_DIR64LSB:
2825 /* Shared objects will always need at least a REL relocation. */
2826 if (info->shared || maybe_dynamic)
2827 need_entry = NEED_DYNREL;
2828 break;
2830 case R_IA64_IPLTMSB:
2831 case R_IA64_IPLTLSB:
2832 /* Shared objects will always need at least a REL relocation. */
2833 if (info->shared || maybe_dynamic)
2834 need_entry = NEED_DYNREL;
2835 break;
2837 case R_IA64_PCREL22:
2838 case R_IA64_PCREL64I:
2839 case R_IA64_PCREL32MSB:
2840 case R_IA64_PCREL32LSB:
2841 case R_IA64_PCREL64MSB:
2842 case R_IA64_PCREL64LSB:
2843 if (maybe_dynamic)
2844 need_entry = NEED_DYNREL;
2845 break;
2848 if (!need_entry)
2849 continue;
2851 if ((need_entry & NEED_FPTR) != 0
2852 && rel->r_addend)
2854 (*info->callbacks->warning)
2855 (info, _("non-zero addend in @fptr reloc"), 0,
2856 abfd, 0, (bfd_vma) 0);
2859 if (get_dyn_sym_info (ia64_info, h, abfd, rel, TRUE) == NULL)
2860 return FALSE;
2863 /* Now, we only do lookup without insertion, which is very fast
2864 with the modified get_dyn_sym_info. */
2865 for (rel = relocs; rel < relend; ++rel)
2867 struct elfNN_ia64_dyn_sym_info *dyn_i;
2868 int dynrel_type = R_IA64_NONE;
2870 r_symndx = ELFNN_R_SYM (rel->r_info);
2871 if (r_symndx >= symtab_hdr->sh_info)
2873 /* We're dealing with a global symbol -- find its hash entry
2874 and mark it as being referenced. */
2875 long indx = r_symndx - symtab_hdr->sh_info;
2876 h = elf_sym_hashes (abfd)[indx];
2877 while (h->root.type == bfd_link_hash_indirect
2878 || h->root.type == bfd_link_hash_warning)
2879 h = (struct elf_link_hash_entry *) h->root.u.i.link;
2881 h->ref_regular = 1;
2883 else
2884 h = NULL;
2886 /* We can only get preliminary data on whether a symbol is
2887 locally or externally defined, as not all of the input files
2888 have yet been processed. Do something with what we know, as
2889 this may help reduce memory usage and processing time later. */
2890 maybe_dynamic = (h && ((!info->executable
2891 && (!SYMBOLIC_BIND (info, h)
2892 || info->unresolved_syms_in_shared_libs == RM_IGNORE))
2893 || !h->def_regular
2894 || h->root.type == bfd_link_hash_defweak));
2896 need_entry = 0;
2897 switch (ELFNN_R_TYPE (rel->r_info))
2899 case R_IA64_TPREL64MSB:
2900 case R_IA64_TPREL64LSB:
2901 if (info->shared || maybe_dynamic)
2902 need_entry = NEED_DYNREL;
2903 dynrel_type = R_IA64_TPREL64LSB;
2904 if (info->shared)
2905 info->flags |= DF_STATIC_TLS;
2906 break;
2908 case R_IA64_LTOFF_TPREL22:
2909 need_entry = NEED_TPREL;
2910 if (info->shared)
2911 info->flags |= DF_STATIC_TLS;
2912 break;
2914 case R_IA64_DTPREL32MSB:
2915 case R_IA64_DTPREL32LSB:
2916 case R_IA64_DTPREL64MSB:
2917 case R_IA64_DTPREL64LSB:
2918 if (info->shared || maybe_dynamic)
2919 need_entry = NEED_DYNREL;
2920 dynrel_type = R_IA64_DTPRELNNLSB;
2921 break;
2923 case R_IA64_LTOFF_DTPREL22:
2924 need_entry = NEED_DTPREL;
2925 break;
2927 case R_IA64_DTPMOD64MSB:
2928 case R_IA64_DTPMOD64LSB:
2929 if (info->shared || maybe_dynamic)
2930 need_entry = NEED_DYNREL;
2931 dynrel_type = R_IA64_DTPMOD64LSB;
2932 break;
2934 case R_IA64_LTOFF_DTPMOD22:
2935 need_entry = NEED_DTPMOD;
2936 break;
2938 case R_IA64_LTOFF_FPTR22:
2939 case R_IA64_LTOFF_FPTR64I:
2940 case R_IA64_LTOFF_FPTR32MSB:
2941 case R_IA64_LTOFF_FPTR32LSB:
2942 case R_IA64_LTOFF_FPTR64MSB:
2943 case R_IA64_LTOFF_FPTR64LSB:
2944 need_entry = NEED_FPTR | NEED_GOT | NEED_LTOFF_FPTR;
2945 break;
2947 case R_IA64_FPTR64I:
2948 case R_IA64_FPTR32MSB:
2949 case R_IA64_FPTR32LSB:
2950 case R_IA64_FPTR64MSB:
2951 case R_IA64_FPTR64LSB:
2952 if (info->shared || h)
2953 need_entry = NEED_FPTR | NEED_DYNREL;
2954 else
2955 need_entry = NEED_FPTR;
2956 dynrel_type = R_IA64_FPTRNNLSB;
2957 break;
2959 case R_IA64_LTOFF22:
2960 case R_IA64_LTOFF64I:
2961 need_entry = NEED_GOT;
2962 break;
2964 case R_IA64_LTOFF22X:
2965 need_entry = NEED_GOTX;
2966 break;
2968 case R_IA64_PLTOFF22:
2969 case R_IA64_PLTOFF64I:
2970 case R_IA64_PLTOFF64MSB:
2971 case R_IA64_PLTOFF64LSB:
2972 need_entry = NEED_PLTOFF;
2973 if (h)
2975 if (maybe_dynamic)
2976 need_entry |= NEED_MIN_PLT;
2978 break;
2980 case R_IA64_PCREL21B:
2981 case R_IA64_PCREL60B:
2982 /* Depending on where this symbol is defined, we may or may not
2983 need a full plt entry. Only skip if we know we'll not need
2984 the entry -- static or symbolic, and the symbol definition
2985 has already been seen. */
2986 if (maybe_dynamic && rel->r_addend == 0)
2987 need_entry = NEED_FULL_PLT;
2988 break;
2990 case R_IA64_IMM14:
2991 case R_IA64_IMM22:
2992 case R_IA64_IMM64:
2993 case R_IA64_DIR32MSB:
2994 case R_IA64_DIR32LSB:
2995 case R_IA64_DIR64MSB:
2996 case R_IA64_DIR64LSB:
2997 /* Shared objects will always need at least a REL relocation. */
2998 if (info->shared || maybe_dynamic)
2999 need_entry = NEED_DYNREL;
3000 dynrel_type = R_IA64_DIRNNLSB;
3001 break;
3003 case R_IA64_IPLTMSB:
3004 case R_IA64_IPLTLSB:
3005 /* Shared objects will always need at least a REL relocation. */
3006 if (info->shared || maybe_dynamic)
3007 need_entry = NEED_DYNREL;
3008 dynrel_type = R_IA64_IPLTLSB;
3009 break;
3011 case R_IA64_PCREL22:
3012 case R_IA64_PCREL64I:
3013 case R_IA64_PCREL32MSB:
3014 case R_IA64_PCREL32LSB:
3015 case R_IA64_PCREL64MSB:
3016 case R_IA64_PCREL64LSB:
3017 if (maybe_dynamic)
3018 need_entry = NEED_DYNREL;
3019 dynrel_type = R_IA64_PCRELNNLSB;
3020 break;
3023 if (!need_entry)
3024 continue;
3026 dyn_i = get_dyn_sym_info (ia64_info, h, abfd, rel, FALSE);
3028 /* Record whether or not this is a local symbol. */
3029 dyn_i->h = h;
3031 /* Create what's needed. */
3032 if (need_entry & (NEED_GOT | NEED_GOTX | NEED_TPREL
3033 | NEED_DTPMOD | NEED_DTPREL))
3035 if (!got)
3037 got = get_got (abfd, info, ia64_info);
3038 if (!got)
3039 return FALSE;
3041 if (need_entry & NEED_GOT)
3042 dyn_i->want_got = 1;
3043 if (need_entry & NEED_GOTX)
3044 dyn_i->want_gotx = 1;
3045 if (need_entry & NEED_TPREL)
3046 dyn_i->want_tprel = 1;
3047 if (need_entry & NEED_DTPMOD)
3048 dyn_i->want_dtpmod = 1;
3049 if (need_entry & NEED_DTPREL)
3050 dyn_i->want_dtprel = 1;
3052 if (need_entry & NEED_FPTR)
3054 if (!fptr)
3056 fptr = get_fptr (abfd, info, ia64_info);
3057 if (!fptr)
3058 return FALSE;
3061 /* FPTRs for shared libraries are allocated by the dynamic
3062 linker. Make sure this local symbol will appear in the
3063 dynamic symbol table. */
3064 if (!h && info->shared)
3066 if (! (bfd_elf_link_record_local_dynamic_symbol
3067 (info, abfd, (long) r_symndx)))
3068 return FALSE;
3071 dyn_i->want_fptr = 1;
3073 if (need_entry & NEED_LTOFF_FPTR)
3074 dyn_i->want_ltoff_fptr = 1;
3075 if (need_entry & (NEED_MIN_PLT | NEED_FULL_PLT))
3077 if (!ia64_info->root.dynobj)
3078 ia64_info->root.dynobj = abfd;
3079 h->needs_plt = 1;
3080 dyn_i->want_plt = 1;
3082 if (need_entry & NEED_FULL_PLT)
3083 dyn_i->want_plt2 = 1;
3084 if (need_entry & NEED_PLTOFF)
3086 /* This is needed here, in case @pltoff is used in a non-shared
3087 link. */
3088 if (!pltoff)
3090 pltoff = get_pltoff (abfd, info, ia64_info);
3091 if (!pltoff)
3092 return FALSE;
3095 dyn_i->want_pltoff = 1;
3097 if ((need_entry & NEED_DYNREL) && (sec->flags & SEC_ALLOC))
3099 if (!srel)
3101 srel = get_reloc_section (abfd, ia64_info, sec, TRUE);
3102 if (!srel)
3103 return FALSE;
3105 if (!count_dyn_reloc (abfd, dyn_i, srel, dynrel_type,
3106 (sec->flags & SEC_READONLY) != 0))
3107 return FALSE;
3111 return TRUE;
3114 /* For cleanliness, and potentially faster dynamic loading, allocate
3115 external GOT entries first. */
3117 static bfd_boolean
3118 allocate_global_data_got (struct elfNN_ia64_dyn_sym_info *dyn_i,
3119 void * data)
3121 struct elfNN_ia64_allocate_data *x = (struct elfNN_ia64_allocate_data *)data;
3123 if ((dyn_i->want_got || dyn_i->want_gotx)
3124 && ! dyn_i->want_fptr
3125 && elfNN_ia64_dynamic_symbol_p (dyn_i->h, x->info, 0))
3127 dyn_i->got_offset = x->ofs;
3128 x->ofs += 8;
3130 if (dyn_i->want_tprel)
3132 dyn_i->tprel_offset = x->ofs;
3133 x->ofs += 8;
3135 if (dyn_i->want_dtpmod)
3137 if (elfNN_ia64_dynamic_symbol_p (dyn_i->h, x->info, 0))
3139 dyn_i->dtpmod_offset = x->ofs;
3140 x->ofs += 8;
3142 else
3144 struct elfNN_ia64_link_hash_table *ia64_info;
3146 ia64_info = elfNN_ia64_hash_table (x->info);
3147 if (ia64_info == NULL)
3148 return FALSE;
3150 if (ia64_info->self_dtpmod_offset == (bfd_vma) -1)
3152 ia64_info->self_dtpmod_offset = x->ofs;
3153 x->ofs += 8;
3155 dyn_i->dtpmod_offset = ia64_info->self_dtpmod_offset;
3158 if (dyn_i->want_dtprel)
3160 dyn_i->dtprel_offset = x->ofs;
3161 x->ofs += 8;
3163 return TRUE;
3166 /* Next, allocate all the GOT entries used by LTOFF_FPTR relocs. */
3168 static bfd_boolean
3169 allocate_global_fptr_got (struct elfNN_ia64_dyn_sym_info *dyn_i,
3170 void * data)
3172 struct elfNN_ia64_allocate_data *x = (struct elfNN_ia64_allocate_data *)data;
3174 if (dyn_i->want_got
3175 && dyn_i->want_fptr
3176 && elfNN_ia64_dynamic_symbol_p (dyn_i->h, x->info, R_IA64_FPTRNNLSB))
3178 dyn_i->got_offset = x->ofs;
3179 x->ofs += 8;
3181 return TRUE;
3184 /* Lastly, allocate all the GOT entries for local data. */
3186 static bfd_boolean
3187 allocate_local_got (struct elfNN_ia64_dyn_sym_info *dyn_i,
3188 PTR data)
3190 struct elfNN_ia64_allocate_data *x = (struct elfNN_ia64_allocate_data *)data;
3192 if ((dyn_i->want_got || dyn_i->want_gotx)
3193 && !elfNN_ia64_dynamic_symbol_p (dyn_i->h, x->info, 0))
3195 dyn_i->got_offset = x->ofs;
3196 x->ofs += 8;
3198 return TRUE;
3201 /* Search for the index of a global symbol in it's defining object file. */
3203 static long
3204 global_sym_index (struct elf_link_hash_entry *h)
3206 struct elf_link_hash_entry **p;
3207 bfd *obj;
3209 BFD_ASSERT (h->root.type == bfd_link_hash_defined
3210 || h->root.type == bfd_link_hash_defweak);
3212 obj = h->root.u.def.section->owner;
3213 for (p = elf_sym_hashes (obj); *p != h; ++p)
3214 continue;
3216 return p - elf_sym_hashes (obj) + elf_tdata (obj)->symtab_hdr.sh_info;
3219 /* Allocate function descriptors. We can do these for every function
3220 in a main executable that is not exported. */
3222 static bfd_boolean
3223 allocate_fptr (struct elfNN_ia64_dyn_sym_info *dyn_i, PTR data)
3225 struct elfNN_ia64_allocate_data *x = (struct elfNN_ia64_allocate_data *)data;
3227 if (dyn_i->want_fptr)
3229 struct elf_link_hash_entry *h = dyn_i->h;
3231 if (h)
3232 while (h->root.type == bfd_link_hash_indirect
3233 || h->root.type == bfd_link_hash_warning)
3234 h = (struct elf_link_hash_entry *) h->root.u.i.link;
3236 if (!x->info->executable
3237 && (!h
3238 || ELF_ST_VISIBILITY (h->other) == STV_DEFAULT
3239 || (h->root.type != bfd_link_hash_undefweak
3240 && h->root.type != bfd_link_hash_undefined)))
3242 if (h && h->dynindx == -1)
3244 BFD_ASSERT ((h->root.type == bfd_link_hash_defined)
3245 || (h->root.type == bfd_link_hash_defweak));
3247 if (!bfd_elf_link_record_local_dynamic_symbol
3248 (x->info, h->root.u.def.section->owner,
3249 global_sym_index (h)))
3250 return FALSE;
3253 dyn_i->want_fptr = 0;
3255 else if (h == NULL || h->dynindx == -1)
3257 dyn_i->fptr_offset = x->ofs;
3258 x->ofs += 16;
3260 else
3261 dyn_i->want_fptr = 0;
3263 return TRUE;
3266 /* Allocate all the minimal PLT entries. */
3268 static bfd_boolean
3269 allocate_plt_entries (struct elfNN_ia64_dyn_sym_info *dyn_i,
3270 PTR data)
3272 struct elfNN_ia64_allocate_data *x = (struct elfNN_ia64_allocate_data *)data;
3274 if (dyn_i->want_plt)
3276 struct elf_link_hash_entry *h = dyn_i->h;
3278 if (h)
3279 while (h->root.type == bfd_link_hash_indirect
3280 || h->root.type == bfd_link_hash_warning)
3281 h = (struct elf_link_hash_entry *) h->root.u.i.link;
3283 /* ??? Versioned symbols seem to lose NEEDS_PLT. */
3284 if (elfNN_ia64_dynamic_symbol_p (h, x->info, 0))
3286 bfd_size_type offset = x->ofs;
3287 if (offset == 0)
3288 offset = PLT_HEADER_SIZE;
3289 dyn_i->plt_offset = offset;
3290 x->ofs = offset + PLT_MIN_ENTRY_SIZE;
3292 dyn_i->want_pltoff = 1;
3294 else
3296 dyn_i->want_plt = 0;
3297 dyn_i->want_plt2 = 0;
3300 return TRUE;
3303 /* Allocate all the full PLT entries. */
3305 static bfd_boolean
3306 allocate_plt2_entries (struct elfNN_ia64_dyn_sym_info *dyn_i,
3307 PTR data)
3309 struct elfNN_ia64_allocate_data *x = (struct elfNN_ia64_allocate_data *)data;
3311 if (dyn_i->want_plt2)
3313 struct elf_link_hash_entry *h = dyn_i->h;
3314 bfd_size_type ofs = x->ofs;
3316 dyn_i->plt2_offset = ofs;
3317 x->ofs = ofs + PLT_FULL_ENTRY_SIZE;
3319 while (h->root.type == bfd_link_hash_indirect
3320 || h->root.type == bfd_link_hash_warning)
3321 h = (struct elf_link_hash_entry *) h->root.u.i.link;
3322 dyn_i->h->plt.offset = ofs;
3324 return TRUE;
3327 /* Allocate all the PLTOFF entries requested by relocations and
3328 plt entries. We can't share space with allocated FPTR entries,
3329 because the latter are not necessarily addressable by the GP.
3330 ??? Relaxation might be able to determine that they are. */
3332 static bfd_boolean
3333 allocate_pltoff_entries (struct elfNN_ia64_dyn_sym_info *dyn_i,
3334 PTR data)
3336 struct elfNN_ia64_allocate_data *x = (struct elfNN_ia64_allocate_data *)data;
3338 if (dyn_i->want_pltoff)
3340 dyn_i->pltoff_offset = x->ofs;
3341 x->ofs += 16;
3343 return TRUE;
3346 /* Allocate dynamic relocations for those symbols that turned out
3347 to be dynamic. */
3349 static bfd_boolean
3350 allocate_dynrel_entries (struct elfNN_ia64_dyn_sym_info *dyn_i,
3351 PTR data)
3353 struct elfNN_ia64_allocate_data *x = (struct elfNN_ia64_allocate_data *)data;
3354 struct elfNN_ia64_link_hash_table *ia64_info;
3355 struct elfNN_ia64_dyn_reloc_entry *rent;
3356 bfd_boolean dynamic_symbol, shared, resolved_zero;
3358 ia64_info = elfNN_ia64_hash_table (x->info);
3359 if (ia64_info == NULL)
3360 return FALSE;
3362 /* Note that this can't be used in relation to FPTR relocs below. */
3363 dynamic_symbol = elfNN_ia64_dynamic_symbol_p (dyn_i->h, x->info, 0);
3365 shared = x->info->shared;
3366 resolved_zero = (dyn_i->h
3367 && ELF_ST_VISIBILITY (dyn_i->h->other)
3368 && dyn_i->h->root.type == bfd_link_hash_undefweak);
3370 /* Take care of the GOT and PLT relocations. */
3372 if ((!resolved_zero
3373 && (dynamic_symbol || shared)
3374 && (dyn_i->want_got || dyn_i->want_gotx))
3375 || (dyn_i->want_ltoff_fptr
3376 && dyn_i->h
3377 && dyn_i->h->dynindx != -1))
3379 if (!dyn_i->want_ltoff_fptr
3380 || !x->info->pie
3381 || dyn_i->h == NULL
3382 || dyn_i->h->root.type != bfd_link_hash_undefweak)
3383 ia64_info->root.srelgot->size += sizeof (ElfNN_External_Rela);
3385 if ((dynamic_symbol || shared) && dyn_i->want_tprel)
3386 ia64_info->root.srelgot->size += sizeof (ElfNN_External_Rela);
3387 if (dynamic_symbol && dyn_i->want_dtpmod)
3388 ia64_info->root.srelgot->size += sizeof (ElfNN_External_Rela);
3389 if (dynamic_symbol && dyn_i->want_dtprel)
3390 ia64_info->root.srelgot->size += sizeof (ElfNN_External_Rela);
3392 if (x->only_got)
3393 return TRUE;
3395 if (ia64_info->rel_fptr_sec && dyn_i->want_fptr)
3397 if (dyn_i->h == NULL || dyn_i->h->root.type != bfd_link_hash_undefweak)
3398 ia64_info->rel_fptr_sec->size += sizeof (ElfNN_External_Rela);
3401 if (!resolved_zero && dyn_i->want_pltoff)
3403 bfd_size_type t = 0;
3405 /* Dynamic symbols get one IPLT relocation. Local symbols in
3406 shared libraries get two REL relocations. Local symbols in
3407 main applications get nothing. */
3408 if (dynamic_symbol)
3409 t = sizeof (ElfNN_External_Rela);
3410 else if (shared)
3411 t = 2 * sizeof (ElfNN_External_Rela);
3413 ia64_info->rel_pltoff_sec->size += t;
3416 /* Take care of the normal data relocations. */
3418 for (rent = dyn_i->reloc_entries; rent; rent = rent->next)
3420 int count = rent->count;
3422 switch (rent->type)
3424 case R_IA64_FPTR32LSB:
3425 case R_IA64_FPTR64LSB:
3426 /* Allocate one iff !want_fptr and not PIE, which by this point
3427 will be true only if we're actually allocating one statically
3428 in the main executable. Position independent executables
3429 need a relative reloc. */
3430 if (dyn_i->want_fptr && !x->info->pie)
3431 continue;
3432 break;
3433 case R_IA64_PCREL32LSB:
3434 case R_IA64_PCREL64LSB:
3435 if (!dynamic_symbol)
3436 continue;
3437 break;
3438 case R_IA64_DIR32LSB:
3439 case R_IA64_DIR64LSB:
3440 if (!dynamic_symbol && !shared)
3441 continue;
3442 break;
3443 case R_IA64_IPLTLSB:
3444 if (!dynamic_symbol && !shared)
3445 continue;
3446 /* Use two REL relocations for IPLT relocations
3447 against local symbols. */
3448 if (!dynamic_symbol)
3449 count *= 2;
3450 break;
3451 case R_IA64_DTPREL32LSB:
3452 case R_IA64_TPREL64LSB:
3453 case R_IA64_DTPREL64LSB:
3454 case R_IA64_DTPMOD64LSB:
3455 break;
3456 default:
3457 abort ();
3459 if (rent->reltext)
3460 ia64_info->reltext = 1;
3461 rent->srel->size += sizeof (ElfNN_External_Rela) * count;
3464 return TRUE;
3467 static bfd_boolean
3468 elfNN_ia64_adjust_dynamic_symbol (struct bfd_link_info *info ATTRIBUTE_UNUSED,
3469 struct elf_link_hash_entry *h)
3471 /* ??? Undefined symbols with PLT entries should be re-defined
3472 to be the PLT entry. */
3474 /* If this is a weak symbol, and there is a real definition, the
3475 processor independent code will have arranged for us to see the
3476 real definition first, and we can just use the same value. */
3477 if (h->u.weakdef != NULL)
3479 BFD_ASSERT (h->u.weakdef->root.type == bfd_link_hash_defined
3480 || h->u.weakdef->root.type == bfd_link_hash_defweak);
3481 h->root.u.def.section = h->u.weakdef->root.u.def.section;
3482 h->root.u.def.value = h->u.weakdef->root.u.def.value;
3483 return TRUE;
3486 /* If this is a reference to a symbol defined by a dynamic object which
3487 is not a function, we might allocate the symbol in our .dynbss section
3488 and allocate a COPY dynamic relocation.
3490 But IA-64 code is canonically PIC, so as a rule we can avoid this sort
3491 of hackery. */
3493 return TRUE;
3496 static bfd_boolean
3497 elfNN_ia64_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
3498 struct bfd_link_info *info)
3500 struct elfNN_ia64_allocate_data data;
3501 struct elfNN_ia64_link_hash_table *ia64_info;
3502 asection *sec;
3503 bfd *dynobj;
3504 bfd_boolean relplt = FALSE;
3506 dynobj = elf_hash_table(info)->dynobj;
3507 ia64_info = elfNN_ia64_hash_table (info);
3508 if (ia64_info == NULL)
3509 return FALSE;
3510 ia64_info->self_dtpmod_offset = (bfd_vma) -1;
3511 BFD_ASSERT(dynobj != NULL);
3512 data.info = info;
3514 /* Set the contents of the .interp section to the interpreter. */
3515 if (ia64_info->root.dynamic_sections_created
3516 && info->executable)
3518 sec = bfd_get_section_by_name (dynobj, ".interp");
3519 BFD_ASSERT (sec != NULL);
3520 sec->contents = (bfd_byte *) ELF_DYNAMIC_INTERPRETER;
3521 sec->size = strlen (ELF_DYNAMIC_INTERPRETER) + 1;
3524 /* Allocate the GOT entries. */
3526 if (ia64_info->root.sgot)
3528 data.ofs = 0;
3529 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_global_data_got, &data);
3530 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_global_fptr_got, &data);
3531 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_local_got, &data);
3532 ia64_info->root.sgot->size = data.ofs;
3535 /* Allocate the FPTR entries. */
3537 if (ia64_info->fptr_sec)
3539 data.ofs = 0;
3540 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_fptr, &data);
3541 ia64_info->fptr_sec->size = data.ofs;
3544 /* Now that we've seen all of the input files, we can decide which
3545 symbols need plt entries. Allocate the minimal PLT entries first.
3546 We do this even though dynamic_sections_created may be FALSE, because
3547 this has the side-effect of clearing want_plt and want_plt2. */
3549 data.ofs = 0;
3550 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_plt_entries, &data);
3552 ia64_info->minplt_entries = 0;
3553 if (data.ofs)
3555 ia64_info->minplt_entries
3556 = (data.ofs - PLT_HEADER_SIZE) / PLT_MIN_ENTRY_SIZE;
3559 /* Align the pointer for the plt2 entries. */
3560 data.ofs = (data.ofs + 31) & (bfd_vma) -32;
3562 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_plt2_entries, &data);
3563 if (data.ofs != 0 || ia64_info->root.dynamic_sections_created)
3565 /* FIXME: we always reserve the memory for dynamic linker even if
3566 there are no PLT entries since dynamic linker may assume the
3567 reserved memory always exists. */
3569 BFD_ASSERT (ia64_info->root.dynamic_sections_created);
3571 ia64_info->root.splt->size = data.ofs;
3573 /* If we've got a .plt, we need some extra memory for the dynamic
3574 linker. We stuff these in .got.plt. */
3575 sec = bfd_get_section_by_name (dynobj, ".got.plt");
3576 sec->size = 8 * PLT_RESERVED_WORDS;
3579 /* Allocate the PLTOFF entries. */
3581 if (ia64_info->pltoff_sec)
3583 data.ofs = 0;
3584 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_pltoff_entries, &data);
3585 ia64_info->pltoff_sec->size = data.ofs;
3588 if (ia64_info->root.dynamic_sections_created)
3590 /* Allocate space for the dynamic relocations that turned out to be
3591 required. */
3593 if (info->shared && ia64_info->self_dtpmod_offset != (bfd_vma) -1)
3594 ia64_info->root.srelgot->size += sizeof (ElfNN_External_Rela);
3595 data.only_got = FALSE;
3596 elfNN_ia64_dyn_sym_traverse (ia64_info, allocate_dynrel_entries, &data);
3599 /* We have now determined the sizes of the various dynamic sections.
3600 Allocate memory for them. */
3601 for (sec = dynobj->sections; sec != NULL; sec = sec->next)
3603 bfd_boolean strip;
3605 if (!(sec->flags & SEC_LINKER_CREATED))
3606 continue;
3608 /* If we don't need this section, strip it from the output file.
3609 There were several sections primarily related to dynamic
3610 linking that must be create before the linker maps input
3611 sections to output sections. The linker does that before
3612 bfd_elf_size_dynamic_sections is called, and it is that
3613 function which decides whether anything needs to go into
3614 these sections. */
3616 strip = (sec->size == 0);
3618 if (sec == ia64_info->root.sgot)
3619 strip = FALSE;
3620 else if (sec == ia64_info->root.srelgot)
3622 if (strip)
3623 ia64_info->root.srelgot = NULL;
3624 else
3625 /* We use the reloc_count field as a counter if we need to
3626 copy relocs into the output file. */
3627 sec->reloc_count = 0;
3629 else if (sec == ia64_info->fptr_sec)
3631 if (strip)
3632 ia64_info->fptr_sec = NULL;
3634 else if (sec == ia64_info->rel_fptr_sec)
3636 if (strip)
3637 ia64_info->rel_fptr_sec = NULL;
3638 else
3639 /* We use the reloc_count field as a counter if we need to
3640 copy relocs into the output file. */
3641 sec->reloc_count = 0;
3643 else if (sec == ia64_info->root.splt)
3645 if (strip)
3646 ia64_info->root.splt = NULL;
3648 else if (sec == ia64_info->pltoff_sec)
3650 if (strip)
3651 ia64_info->pltoff_sec = NULL;
3653 else if (sec == ia64_info->rel_pltoff_sec)
3655 if (strip)
3656 ia64_info->rel_pltoff_sec = NULL;
3657 else
3659 relplt = TRUE;
3660 /* We use the reloc_count field as a counter if we need to
3661 copy relocs into the output file. */
3662 sec->reloc_count = 0;
3665 else
3667 const char *name;
3669 /* It's OK to base decisions on the section name, because none
3670 of the dynobj section names depend upon the input files. */
3671 name = bfd_get_section_name (dynobj, sec);
3673 if (strcmp (name, ".got.plt") == 0)
3674 strip = FALSE;
3675 else if (CONST_STRNEQ (name, ".rel"))
3677 if (!strip)
3679 /* We use the reloc_count field as a counter if we need to
3680 copy relocs into the output file. */
3681 sec->reloc_count = 0;
3684 else
3685 continue;
3688 if (strip)
3689 sec->flags |= SEC_EXCLUDE;
3690 else
3692 /* Allocate memory for the section contents. */
3693 sec->contents = (bfd_byte *) bfd_zalloc (dynobj, sec->size);
3694 if (sec->contents == NULL && sec->size != 0)
3695 return FALSE;
3699 if (elf_hash_table (info)->dynamic_sections_created)
3701 /* Add some entries to the .dynamic section. We fill in the values
3702 later (in finish_dynamic_sections) but we must add the entries now
3703 so that we get the correct size for the .dynamic section. */
3705 if (info->executable)
3707 /* The DT_DEBUG entry is filled in by the dynamic linker and used
3708 by the debugger. */
3709 #define add_dynamic_entry(TAG, VAL) \
3710 _bfd_elf_add_dynamic_entry (info, TAG, VAL)
3712 if (!add_dynamic_entry (DT_DEBUG, 0))
3713 return FALSE;
3716 if (!add_dynamic_entry (DT_IA_64_PLT_RESERVE, 0))
3717 return FALSE;
3718 if (!add_dynamic_entry (DT_PLTGOT, 0))
3719 return FALSE;
3721 if (relplt)
3723 if (!add_dynamic_entry (DT_PLTRELSZ, 0)
3724 || !add_dynamic_entry (DT_PLTREL, DT_RELA)
3725 || !add_dynamic_entry (DT_JMPREL, 0))
3726 return FALSE;
3729 if (!add_dynamic_entry (DT_RELA, 0)
3730 || !add_dynamic_entry (DT_RELASZ, 0)
3731 || !add_dynamic_entry (DT_RELAENT, sizeof (ElfNN_External_Rela)))
3732 return FALSE;
3734 if (ia64_info->reltext)
3736 if (!add_dynamic_entry (DT_TEXTREL, 0))
3737 return FALSE;
3738 info->flags |= DF_TEXTREL;
3742 /* ??? Perhaps force __gp local. */
3744 return TRUE;
3747 static bfd_reloc_status_type
3748 elfNN_ia64_install_value (bfd_byte *hit_addr, bfd_vma v,
3749 unsigned int r_type)
3751 const struct ia64_operand *op;
3752 int bigendian = 0, shift = 0;
3753 bfd_vma t0, t1, dword;
3754 ia64_insn insn;
3755 enum ia64_opnd opnd;
3756 const char *err;
3757 size_t size = 8;
3758 #ifdef BFD_HOST_U_64_BIT
3759 BFD_HOST_U_64_BIT val = (BFD_HOST_U_64_BIT) v;
3760 #else
3761 bfd_vma val = v;
3762 #endif
3764 opnd = IA64_OPND_NIL;
3765 switch (r_type)
3767 case R_IA64_NONE:
3768 case R_IA64_LDXMOV:
3769 return bfd_reloc_ok;
3771 /* Instruction relocations. */
3773 case R_IA64_IMM14:
3774 case R_IA64_TPREL14:
3775 case R_IA64_DTPREL14:
3776 opnd = IA64_OPND_IMM14;
3777 break;
3779 case R_IA64_PCREL21F: opnd = IA64_OPND_TGT25; break;
3780 case R_IA64_PCREL21M: opnd = IA64_OPND_TGT25b; break;
3781 case R_IA64_PCREL60B: opnd = IA64_OPND_TGT64; break;
3782 case R_IA64_PCREL21B:
3783 case R_IA64_PCREL21BI:
3784 opnd = IA64_OPND_TGT25c;
3785 break;
3787 case R_IA64_IMM22:
3788 case R_IA64_GPREL22:
3789 case R_IA64_LTOFF22:
3790 case R_IA64_LTOFF22X:
3791 case R_IA64_PLTOFF22:
3792 case R_IA64_PCREL22:
3793 case R_IA64_LTOFF_FPTR22:
3794 case R_IA64_TPREL22:
3795 case R_IA64_DTPREL22:
3796 case R_IA64_LTOFF_TPREL22:
3797 case R_IA64_LTOFF_DTPMOD22:
3798 case R_IA64_LTOFF_DTPREL22:
3799 opnd = IA64_OPND_IMM22;
3800 break;
3802 case R_IA64_IMM64:
3803 case R_IA64_GPREL64I:
3804 case R_IA64_LTOFF64I:
3805 case R_IA64_PLTOFF64I:
3806 case R_IA64_PCREL64I:
3807 case R_IA64_FPTR64I:
3808 case R_IA64_LTOFF_FPTR64I:
3809 case R_IA64_TPREL64I:
3810 case R_IA64_DTPREL64I:
3811 opnd = IA64_OPND_IMMU64;
3812 break;
3814 /* Data relocations. */
3816 case R_IA64_DIR32MSB:
3817 case R_IA64_GPREL32MSB:
3818 case R_IA64_FPTR32MSB:
3819 case R_IA64_PCREL32MSB:
3820 case R_IA64_LTOFF_FPTR32MSB:
3821 case R_IA64_SEGREL32MSB:
3822 case R_IA64_SECREL32MSB:
3823 case R_IA64_LTV32MSB:
3824 case R_IA64_DTPREL32MSB:
3825 size = 4; bigendian = 1;
3826 break;
3828 case R_IA64_DIR32LSB:
3829 case R_IA64_GPREL32LSB:
3830 case R_IA64_FPTR32LSB:
3831 case R_IA64_PCREL32LSB:
3832 case R_IA64_LTOFF_FPTR32LSB:
3833 case R_IA64_SEGREL32LSB:
3834 case R_IA64_SECREL32LSB:
3835 case R_IA64_LTV32LSB:
3836 case R_IA64_DTPREL32LSB:
3837 size = 4; bigendian = 0;
3838 break;
3840 case R_IA64_DIR64MSB:
3841 case R_IA64_GPREL64MSB:
3842 case R_IA64_PLTOFF64MSB:
3843 case R_IA64_FPTR64MSB:
3844 case R_IA64_PCREL64MSB:
3845 case R_IA64_LTOFF_FPTR64MSB:
3846 case R_IA64_SEGREL64MSB:
3847 case R_IA64_SECREL64MSB:
3848 case R_IA64_LTV64MSB:
3849 case R_IA64_TPREL64MSB:
3850 case R_IA64_DTPMOD64MSB:
3851 case R_IA64_DTPREL64MSB:
3852 size = 8; bigendian = 1;
3853 break;
3855 case R_IA64_DIR64LSB:
3856 case R_IA64_GPREL64LSB:
3857 case R_IA64_PLTOFF64LSB:
3858 case R_IA64_FPTR64LSB:
3859 case R_IA64_PCREL64LSB:
3860 case R_IA64_LTOFF_FPTR64LSB:
3861 case R_IA64_SEGREL64LSB:
3862 case R_IA64_SECREL64LSB:
3863 case R_IA64_LTV64LSB:
3864 case R_IA64_TPREL64LSB:
3865 case R_IA64_DTPMOD64LSB:
3866 case R_IA64_DTPREL64LSB:
3867 size = 8; bigendian = 0;
3868 break;
3870 /* Unsupported / Dynamic relocations. */
3871 default:
3872 return bfd_reloc_notsupported;
3875 switch (opnd)
3877 case IA64_OPND_IMMU64:
3878 hit_addr -= (long) hit_addr & 0x3;
3879 t0 = bfd_getl64 (hit_addr);
3880 t1 = bfd_getl64 (hit_addr + 8);
3882 /* tmpl/s: bits 0.. 5 in t0
3883 slot 0: bits 5..45 in t0
3884 slot 1: bits 46..63 in t0, bits 0..22 in t1
3885 slot 2: bits 23..63 in t1 */
3887 /* First, clear the bits that form the 64 bit constant. */
3888 t0 &= ~(0x3ffffLL << 46);
3889 t1 &= ~(0x7fffffLL
3890 | (( (0x07fLL << 13) | (0x1ffLL << 27)
3891 | (0x01fLL << 22) | (0x001LL << 21)
3892 | (0x001LL << 36)) << 23));
3894 t0 |= ((val >> 22) & 0x03ffffLL) << 46; /* 18 lsbs of imm41 */
3895 t1 |= ((val >> 40) & 0x7fffffLL) << 0; /* 23 msbs of imm41 */
3896 t1 |= ( (((val >> 0) & 0x07f) << 13) /* imm7b */
3897 | (((val >> 7) & 0x1ff) << 27) /* imm9d */
3898 | (((val >> 16) & 0x01f) << 22) /* imm5c */
3899 | (((val >> 21) & 0x001) << 21) /* ic */
3900 | (((val >> 63) & 0x001) << 36)) << 23; /* i */
3902 bfd_putl64 (t0, hit_addr);
3903 bfd_putl64 (t1, hit_addr + 8);
3904 break;
3906 case IA64_OPND_TGT64:
3907 hit_addr -= (long) hit_addr & 0x3;
3908 t0 = bfd_getl64 (hit_addr);
3909 t1 = bfd_getl64 (hit_addr + 8);
3911 /* tmpl/s: bits 0.. 5 in t0
3912 slot 0: bits 5..45 in t0
3913 slot 1: bits 46..63 in t0, bits 0..22 in t1
3914 slot 2: bits 23..63 in t1 */
3916 /* First, clear the bits that form the 64 bit constant. */
3917 t0 &= ~(0x3ffffLL << 46);
3918 t1 &= ~(0x7fffffLL
3919 | ((1LL << 36 | 0xfffffLL << 13) << 23));
3921 val >>= 4;
3922 t0 |= ((val >> 20) & 0xffffLL) << 2 << 46; /* 16 lsbs of imm39 */
3923 t1 |= ((val >> 36) & 0x7fffffLL) << 0; /* 23 msbs of imm39 */
3924 t1 |= ((((val >> 0) & 0xfffffLL) << 13) /* imm20b */
3925 | (((val >> 59) & 0x1LL) << 36)) << 23; /* i */
3927 bfd_putl64 (t0, hit_addr);
3928 bfd_putl64 (t1, hit_addr + 8);
3929 break;
3931 default:
3932 switch ((long) hit_addr & 0x3)
3934 case 0: shift = 5; break;
3935 case 1: shift = 14; hit_addr += 3; break;
3936 case 2: shift = 23; hit_addr += 6; break;
3937 case 3: return bfd_reloc_notsupported; /* shouldn't happen... */
3939 dword = bfd_getl64 (hit_addr);
3940 insn = (dword >> shift) & 0x1ffffffffffLL;
3942 op = elf64_ia64_operands + opnd;
3943 err = (*op->insert) (op, val, &insn);
3944 if (err)
3945 return bfd_reloc_overflow;
3947 dword &= ~(0x1ffffffffffLL << shift);
3948 dword |= (insn << shift);
3949 bfd_putl64 (dword, hit_addr);
3950 break;
3952 case IA64_OPND_NIL:
3953 /* A data relocation. */
3954 if (bigendian)
3955 if (size == 4)
3956 bfd_putb32 (val, hit_addr);
3957 else
3958 bfd_putb64 (val, hit_addr);
3959 else
3960 if (size == 4)
3961 bfd_putl32 (val, hit_addr);
3962 else
3963 bfd_putl64 (val, hit_addr);
3964 break;
3967 return bfd_reloc_ok;
3970 static void
3971 elfNN_ia64_install_dyn_reloc (bfd *abfd, struct bfd_link_info *info,
3972 asection *sec, asection *srel,
3973 bfd_vma offset, unsigned int type,
3974 long dynindx, bfd_vma addend)
3976 Elf_Internal_Rela outrel;
3977 bfd_byte *loc;
3979 BFD_ASSERT (dynindx != -1);
3980 outrel.r_info = ELFNN_R_INFO (dynindx, type);
3981 outrel.r_addend = addend;
3982 outrel.r_offset = _bfd_elf_section_offset (abfd, info, sec, offset);
3983 if (outrel.r_offset >= (bfd_vma) -2)
3985 /* Run for the hills. We shouldn't be outputting a relocation
3986 for this. So do what everyone else does and output a no-op. */
3987 outrel.r_info = ELFNN_R_INFO (0, R_IA64_NONE);
3988 outrel.r_addend = 0;
3989 outrel.r_offset = 0;
3991 else
3992 outrel.r_offset += sec->output_section->vma + sec->output_offset;
3994 loc = srel->contents;
3995 loc += srel->reloc_count++ * sizeof (ElfNN_External_Rela);
3996 bfd_elfNN_swap_reloca_out (abfd, &outrel, loc);
3997 BFD_ASSERT (sizeof (ElfNN_External_Rela) * srel->reloc_count <= srel->size);
4000 /* Store an entry for target address TARGET_ADDR in the linkage table
4001 and return the gp-relative address of the linkage table entry. */
4003 static bfd_vma
4004 set_got_entry (bfd *abfd, struct bfd_link_info *info,
4005 struct elfNN_ia64_dyn_sym_info *dyn_i,
4006 long dynindx, bfd_vma addend, bfd_vma value,
4007 unsigned int dyn_r_type)
4009 struct elfNN_ia64_link_hash_table *ia64_info;
4010 asection *got_sec;
4011 bfd_boolean done;
4012 bfd_vma got_offset;
4014 ia64_info = elfNN_ia64_hash_table (info);
4015 if (ia64_info == NULL)
4016 return 0;
4018 got_sec = ia64_info->root.sgot;
4020 switch (dyn_r_type)
4022 case R_IA64_TPREL64LSB:
4023 done = dyn_i->tprel_done;
4024 dyn_i->tprel_done = TRUE;
4025 got_offset = dyn_i->tprel_offset;
4026 break;
4027 case R_IA64_DTPMOD64LSB:
4028 if (dyn_i->dtpmod_offset != ia64_info->self_dtpmod_offset)
4030 done = dyn_i->dtpmod_done;
4031 dyn_i->dtpmod_done = TRUE;
4033 else
4035 done = ia64_info->self_dtpmod_done;
4036 ia64_info->self_dtpmod_done = TRUE;
4037 dynindx = 0;
4039 got_offset = dyn_i->dtpmod_offset;
4040 break;
4041 case R_IA64_DTPREL32LSB:
4042 case R_IA64_DTPREL64LSB:
4043 done = dyn_i->dtprel_done;
4044 dyn_i->dtprel_done = TRUE;
4045 got_offset = dyn_i->dtprel_offset;
4046 break;
4047 default:
4048 done = dyn_i->got_done;
4049 dyn_i->got_done = TRUE;
4050 got_offset = dyn_i->got_offset;
4051 break;
4054 BFD_ASSERT ((got_offset & 7) == 0);
4056 if (! done)
4058 /* Store the target address in the linkage table entry. */
4059 bfd_put_64 (abfd, value, got_sec->contents + got_offset);
4061 /* Install a dynamic relocation if needed. */
4062 if (((info->shared
4063 && (!dyn_i->h
4064 || ELF_ST_VISIBILITY (dyn_i->h->other) == STV_DEFAULT
4065 || dyn_i->h->root.type != bfd_link_hash_undefweak)
4066 && dyn_r_type != R_IA64_DTPREL32LSB
4067 && dyn_r_type != R_IA64_DTPREL64LSB)
4068 || elfNN_ia64_dynamic_symbol_p (dyn_i->h, info, dyn_r_type)
4069 || (dynindx != -1
4070 && (dyn_r_type == R_IA64_FPTR32LSB
4071 || dyn_r_type == R_IA64_FPTR64LSB)))
4072 && (!dyn_i->want_ltoff_fptr
4073 || !info->pie
4074 || !dyn_i->h
4075 || dyn_i->h->root.type != bfd_link_hash_undefweak))
4077 if (dynindx == -1
4078 && dyn_r_type != R_IA64_TPREL64LSB
4079 && dyn_r_type != R_IA64_DTPMOD64LSB
4080 && dyn_r_type != R_IA64_DTPREL32LSB
4081 && dyn_r_type != R_IA64_DTPREL64LSB)
4083 dyn_r_type = R_IA64_RELNNLSB;
4084 dynindx = 0;
4085 addend = value;
4088 if (bfd_big_endian (abfd))
4090 switch (dyn_r_type)
4092 case R_IA64_REL32LSB:
4093 dyn_r_type = R_IA64_REL32MSB;
4094 break;
4095 case R_IA64_DIR32LSB:
4096 dyn_r_type = R_IA64_DIR32MSB;
4097 break;
4098 case R_IA64_FPTR32LSB:
4099 dyn_r_type = R_IA64_FPTR32MSB;
4100 break;
4101 case R_IA64_DTPREL32LSB:
4102 dyn_r_type = R_IA64_DTPREL32MSB;
4103 break;
4104 case R_IA64_REL64LSB:
4105 dyn_r_type = R_IA64_REL64MSB;
4106 break;
4107 case R_IA64_DIR64LSB:
4108 dyn_r_type = R_IA64_DIR64MSB;
4109 break;
4110 case R_IA64_FPTR64LSB:
4111 dyn_r_type = R_IA64_FPTR64MSB;
4112 break;
4113 case R_IA64_TPREL64LSB:
4114 dyn_r_type = R_IA64_TPREL64MSB;
4115 break;
4116 case R_IA64_DTPMOD64LSB:
4117 dyn_r_type = R_IA64_DTPMOD64MSB;
4118 break;
4119 case R_IA64_DTPREL64LSB:
4120 dyn_r_type = R_IA64_DTPREL64MSB;
4121 break;
4122 default:
4123 BFD_ASSERT (FALSE);
4124 break;
4128 elfNN_ia64_install_dyn_reloc (abfd, NULL, got_sec,
4129 ia64_info->root.srelgot,
4130 got_offset, dyn_r_type,
4131 dynindx, addend);
4135 /* Return the address of the linkage table entry. */
4136 value = (got_sec->output_section->vma
4137 + got_sec->output_offset
4138 + got_offset);
4140 return value;
4143 /* Fill in a function descriptor consisting of the function's code
4144 address and its global pointer. Return the descriptor's address. */
4146 static bfd_vma
4147 set_fptr_entry (bfd *abfd, struct bfd_link_info *info,
4148 struct elfNN_ia64_dyn_sym_info *dyn_i,
4149 bfd_vma value)
4151 struct elfNN_ia64_link_hash_table *ia64_info;
4152 asection *fptr_sec;
4154 ia64_info = elfNN_ia64_hash_table (info);
4155 if (ia64_info == NULL)
4156 return 0;
4158 fptr_sec = ia64_info->fptr_sec;
4160 if (!dyn_i->fptr_done)
4162 dyn_i->fptr_done = 1;
4164 /* Fill in the function descriptor. */
4165 bfd_put_64 (abfd, value, fptr_sec->contents + dyn_i->fptr_offset);
4166 bfd_put_64 (abfd, _bfd_get_gp_value (abfd),
4167 fptr_sec->contents + dyn_i->fptr_offset + 8);
4168 if (ia64_info->rel_fptr_sec)
4170 Elf_Internal_Rela outrel;
4171 bfd_byte *loc;
4173 if (bfd_little_endian (abfd))
4174 outrel.r_info = ELFNN_R_INFO (0, R_IA64_IPLTLSB);
4175 else
4176 outrel.r_info = ELFNN_R_INFO (0, R_IA64_IPLTMSB);
4177 outrel.r_addend = value;
4178 outrel.r_offset = (fptr_sec->output_section->vma
4179 + fptr_sec->output_offset
4180 + dyn_i->fptr_offset);
4181 loc = ia64_info->rel_fptr_sec->contents;
4182 loc += ia64_info->rel_fptr_sec->reloc_count++
4183 * sizeof (ElfNN_External_Rela);
4184 bfd_elfNN_swap_reloca_out (abfd, &outrel, loc);
4188 /* Return the descriptor's address. */
4189 value = (fptr_sec->output_section->vma
4190 + fptr_sec->output_offset
4191 + dyn_i->fptr_offset);
4193 return value;
4196 /* Fill in a PLTOFF entry consisting of the function's code address
4197 and its global pointer. Return the descriptor's address. */
4199 static bfd_vma
4200 set_pltoff_entry (bfd *abfd, struct bfd_link_info *info,
4201 struct elfNN_ia64_dyn_sym_info *dyn_i,
4202 bfd_vma value, bfd_boolean is_plt)
4204 struct elfNN_ia64_link_hash_table *ia64_info;
4205 asection *pltoff_sec;
4207 ia64_info = elfNN_ia64_hash_table (info);
4208 if (ia64_info == NULL)
4209 return 0;
4211 pltoff_sec = ia64_info->pltoff_sec;
4213 /* Don't do anything if this symbol uses a real PLT entry. In
4214 that case, we'll fill this in during finish_dynamic_symbol. */
4215 if ((! dyn_i->want_plt || is_plt)
4216 && !dyn_i->pltoff_done)
4218 bfd_vma gp = _bfd_get_gp_value (abfd);
4220 /* Fill in the function descriptor. */
4221 bfd_put_64 (abfd, value, pltoff_sec->contents + dyn_i->pltoff_offset);
4222 bfd_put_64 (abfd, gp, pltoff_sec->contents + dyn_i->pltoff_offset + 8);
4224 /* Install dynamic relocations if needed. */
4225 if (!is_plt
4226 && info->shared
4227 && (!dyn_i->h
4228 || ELF_ST_VISIBILITY (dyn_i->h->other) == STV_DEFAULT
4229 || dyn_i->h->root.type != bfd_link_hash_undefweak))
4231 unsigned int dyn_r_type;
4233 if (bfd_big_endian (abfd))
4234 dyn_r_type = R_IA64_RELNNMSB;
4235 else
4236 dyn_r_type = R_IA64_RELNNLSB;
4238 elfNN_ia64_install_dyn_reloc (abfd, NULL, pltoff_sec,
4239 ia64_info->rel_pltoff_sec,
4240 dyn_i->pltoff_offset,
4241 dyn_r_type, 0, value);
4242 elfNN_ia64_install_dyn_reloc (abfd, NULL, pltoff_sec,
4243 ia64_info->rel_pltoff_sec,
4244 dyn_i->pltoff_offset + ARCH_SIZE / 8,
4245 dyn_r_type, 0, gp);
4248 dyn_i->pltoff_done = 1;
4251 /* Return the descriptor's address. */
4252 value = (pltoff_sec->output_section->vma
4253 + pltoff_sec->output_offset
4254 + dyn_i->pltoff_offset);
4256 return value;
4259 /* Return the base VMA address which should be subtracted from real addresses
4260 when resolving @tprel() relocation.
4261 Main program TLS (whose template starts at PT_TLS p_vaddr)
4262 is assigned offset round(2 * size of pointer, PT_TLS p_align). */
4264 static bfd_vma
4265 elfNN_ia64_tprel_base (struct bfd_link_info *info)
4267 asection *tls_sec = elf_hash_table (info)->tls_sec;
4268 return tls_sec->vma - align_power ((bfd_vma) ARCH_SIZE / 4,
4269 tls_sec->alignment_power);
4272 /* Return the base VMA address which should be subtracted from real addresses
4273 when resolving @dtprel() relocation.
4274 This is PT_TLS segment p_vaddr. */
4276 static bfd_vma
4277 elfNN_ia64_dtprel_base (struct bfd_link_info *info)
4279 return elf_hash_table (info)->tls_sec->vma;
4282 /* Called through qsort to sort the .IA_64.unwind section during a
4283 non-relocatable link. Set elfNN_ia64_unwind_entry_compare_bfd
4284 to the output bfd so we can do proper endianness frobbing. */
4286 static bfd *elfNN_ia64_unwind_entry_compare_bfd;
4288 static int
4289 elfNN_ia64_unwind_entry_compare (const PTR a, const PTR b)
4291 bfd_vma av, bv;
4293 av = bfd_get_64 (elfNN_ia64_unwind_entry_compare_bfd, a);
4294 bv = bfd_get_64 (elfNN_ia64_unwind_entry_compare_bfd, b);
4296 return (av < bv ? -1 : av > bv ? 1 : 0);
4299 /* Make sure we've got ourselves a nice fat __gp value. */
4300 static bfd_boolean
4301 elfNN_ia64_choose_gp (bfd *abfd, struct bfd_link_info *info)
4303 bfd_vma min_vma = (bfd_vma) -1, max_vma = 0;
4304 bfd_vma min_short_vma = min_vma, max_short_vma = 0;
4305 struct elf_link_hash_entry *gp;
4306 bfd_vma gp_val;
4307 asection *os;
4308 struct elfNN_ia64_link_hash_table *ia64_info;
4310 ia64_info = elfNN_ia64_hash_table (info);
4311 if (ia64_info == NULL)
4312 return FALSE;
4314 /* Find the min and max vma of all sections marked short. Also collect
4315 min and max vma of any type, for use in selecting a nice gp. */
4316 for (os = abfd->sections; os ; os = os->next)
4318 bfd_vma lo, hi;
4320 if ((os->flags & SEC_ALLOC) == 0)
4321 continue;
4323 lo = os->vma;
4324 hi = os->vma + (os->rawsize ? os->rawsize : os->size);
4325 if (hi < lo)
4326 hi = (bfd_vma) -1;
4328 if (min_vma > lo)
4329 min_vma = lo;
4330 if (max_vma < hi)
4331 max_vma = hi;
4332 if (os->flags & SEC_SMALL_DATA)
4334 if (min_short_vma > lo)
4335 min_short_vma = lo;
4336 if (max_short_vma < hi)
4337 max_short_vma = hi;
4341 if (ia64_info->min_short_sec)
4343 if (min_short_vma
4344 > (ia64_info->min_short_sec->vma
4345 + ia64_info->min_short_offset))
4346 min_short_vma = (ia64_info->min_short_sec->vma
4347 + ia64_info->min_short_offset);
4348 if (max_short_vma
4349 < (ia64_info->max_short_sec->vma
4350 + ia64_info->max_short_offset))
4351 max_short_vma = (ia64_info->max_short_sec->vma
4352 + ia64_info->max_short_offset);
4355 /* See if the user wants to force a value. */
4356 gp = elf_link_hash_lookup (elf_hash_table (info), "__gp", FALSE,
4357 FALSE, FALSE);
4359 if (gp
4360 && (gp->root.type == bfd_link_hash_defined
4361 || gp->root.type == bfd_link_hash_defweak))
4363 asection *gp_sec = gp->root.u.def.section;
4364 gp_val = (gp->root.u.def.value
4365 + gp_sec->output_section->vma
4366 + gp_sec->output_offset);
4368 else
4370 /* Pick a sensible value. */
4372 if (ia64_info->min_short_sec)
4374 bfd_vma short_range = max_short_vma - min_short_vma;
4376 /* If min_short_sec is set, pick one in the middle bewteen
4377 min_short_vma and max_short_vma. */
4378 if (short_range >= 0x400000)
4379 goto overflow;
4380 gp_val = min_short_vma + short_range / 2;
4382 else
4384 asection *got_sec = ia64_info->root.sgot;
4386 /* Start with just the address of the .got. */
4387 if (got_sec)
4388 gp_val = got_sec->output_section->vma;
4389 else if (max_short_vma != 0)
4390 gp_val = min_short_vma;
4391 else if (max_vma - min_vma < 0x200000)
4392 gp_val = min_vma;
4393 else
4394 gp_val = max_vma - 0x200000 + 8;
4397 /* If it is possible to address the entire image, but we
4398 don't with the choice above, adjust. */
4399 if (max_vma - min_vma < 0x400000
4400 && (max_vma - gp_val >= 0x200000
4401 || gp_val - min_vma > 0x200000))
4402 gp_val = min_vma + 0x200000;
4403 else if (max_short_vma != 0)
4405 /* If we don't cover all the short data, adjust. */
4406 if (max_short_vma - gp_val >= 0x200000)
4407 gp_val = min_short_vma + 0x200000;
4409 /* If we're addressing stuff past the end, adjust back. */
4410 if (gp_val > max_vma)
4411 gp_val = max_vma - 0x200000 + 8;
4415 /* Validate whether all SHF_IA_64_SHORT sections are within
4416 range of the chosen GP. */
4418 if (max_short_vma != 0)
4420 if (max_short_vma - min_short_vma >= 0x400000)
4422 overflow:
4423 (*_bfd_error_handler)
4424 (_("%s: short data segment overflowed (0x%lx >= 0x400000)"),
4425 bfd_get_filename (abfd),
4426 (unsigned long) (max_short_vma - min_short_vma));
4427 return FALSE;
4429 else if ((gp_val > min_short_vma
4430 && gp_val - min_short_vma > 0x200000)
4431 || (gp_val < max_short_vma
4432 && max_short_vma - gp_val >= 0x200000))
4434 (*_bfd_error_handler)
4435 (_("%s: __gp does not cover short data segment"),
4436 bfd_get_filename (abfd));
4437 return FALSE;
4441 _bfd_set_gp_value (abfd, gp_val);
4443 return TRUE;
4446 static bfd_boolean
4447 elfNN_ia64_final_link (bfd *abfd, struct bfd_link_info *info)
4449 struct elfNN_ia64_link_hash_table *ia64_info;
4450 asection *unwind_output_sec;
4452 ia64_info = elfNN_ia64_hash_table (info);
4453 if (ia64_info == NULL)
4454 return FALSE;
4456 /* Make sure we've got ourselves a nice fat __gp value. */
4457 if (!info->relocatable)
4459 bfd_vma gp_val;
4460 struct elf_link_hash_entry *gp;
4462 /* We assume after gp is set, section size will only decrease. We
4463 need to adjust gp for it. */
4464 _bfd_set_gp_value (abfd, 0);
4465 if (! elfNN_ia64_choose_gp (abfd, info))
4466 return FALSE;
4467 gp_val = _bfd_get_gp_value (abfd);
4469 gp = elf_link_hash_lookup (elf_hash_table (info), "__gp", FALSE,
4470 FALSE, FALSE);
4471 if (gp)
4473 gp->root.type = bfd_link_hash_defined;
4474 gp->root.u.def.value = gp_val;
4475 gp->root.u.def.section = bfd_abs_section_ptr;
4479 /* If we're producing a final executable, we need to sort the contents
4480 of the .IA_64.unwind section. Force this section to be relocated
4481 into memory rather than written immediately to the output file. */
4482 unwind_output_sec = NULL;
4483 if (!info->relocatable)
4485 asection *s = bfd_get_section_by_name (abfd, ELF_STRING_ia64_unwind);
4486 if (s)
4488 unwind_output_sec = s->output_section;
4489 unwind_output_sec->contents
4490 = bfd_malloc (unwind_output_sec->size);
4491 if (unwind_output_sec->contents == NULL)
4492 return FALSE;
4496 /* Invoke the regular ELF backend linker to do all the work. */
4497 if (!bfd_elf_final_link (abfd, info))
4498 return FALSE;
4500 if (unwind_output_sec)
4502 elfNN_ia64_unwind_entry_compare_bfd = abfd;
4503 qsort (unwind_output_sec->contents,
4504 (size_t) (unwind_output_sec->size / 24),
4506 elfNN_ia64_unwind_entry_compare);
4508 if (! bfd_set_section_contents (abfd, unwind_output_sec,
4509 unwind_output_sec->contents, (bfd_vma) 0,
4510 unwind_output_sec->size))
4511 return FALSE;
4514 return TRUE;
4517 static bfd_boolean
4518 elfNN_ia64_relocate_section (bfd *output_bfd,
4519 struct bfd_link_info *info,
4520 bfd *input_bfd,
4521 asection *input_section,
4522 bfd_byte *contents,
4523 Elf_Internal_Rela *relocs,
4524 Elf_Internal_Sym *local_syms,
4525 asection **local_sections)
4527 struct elfNN_ia64_link_hash_table *ia64_info;
4528 Elf_Internal_Shdr *symtab_hdr;
4529 Elf_Internal_Rela *rel;
4530 Elf_Internal_Rela *relend;
4531 asection *srel;
4532 bfd_boolean ret_val = TRUE; /* for non-fatal errors */
4533 bfd_vma gp_val;
4535 symtab_hdr = &elf_tdata (input_bfd)->symtab_hdr;
4536 ia64_info = elfNN_ia64_hash_table (info);
4537 if (ia64_info == NULL)
4538 return FALSE;
4540 /* Infect various flags from the input section to the output section. */
4541 if (info->relocatable)
4543 bfd_vma flags;
4545 flags = elf_section_data(input_section)->this_hdr.sh_flags;
4546 flags &= SHF_IA_64_NORECOV;
4548 elf_section_data(input_section->output_section)
4549 ->this_hdr.sh_flags |= flags;
4552 gp_val = _bfd_get_gp_value (output_bfd);
4553 srel = get_reloc_section (input_bfd, ia64_info, input_section, FALSE);
4555 rel = relocs;
4556 relend = relocs + input_section->reloc_count;
4557 for (; rel < relend; ++rel)
4559 struct elf_link_hash_entry *h;
4560 struct elfNN_ia64_dyn_sym_info *dyn_i;
4561 bfd_reloc_status_type r;
4562 reloc_howto_type *howto;
4563 unsigned long r_symndx;
4564 Elf_Internal_Sym *sym;
4565 unsigned int r_type;
4566 bfd_vma value;
4567 asection *sym_sec;
4568 bfd_byte *hit_addr;
4569 bfd_boolean dynamic_symbol_p;
4570 bfd_boolean undef_weak_ref;
4572 r_type = ELFNN_R_TYPE (rel->r_info);
4573 if (r_type > R_IA64_MAX_RELOC_CODE)
4575 (*_bfd_error_handler)
4576 (_("%B: unknown relocation type %d"),
4577 input_bfd, (int) r_type);
4578 bfd_set_error (bfd_error_bad_value);
4579 ret_val = FALSE;
4580 continue;
4583 howto = lookup_howto (r_type);
4584 r_symndx = ELFNN_R_SYM (rel->r_info);
4585 h = NULL;
4586 sym = NULL;
4587 sym_sec = NULL;
4588 undef_weak_ref = FALSE;
4590 if (r_symndx < symtab_hdr->sh_info)
4592 /* Reloc against local symbol. */
4593 asection *msec;
4594 sym = local_syms + r_symndx;
4595 sym_sec = local_sections[r_symndx];
4596 msec = sym_sec;
4597 value = _bfd_elf_rela_local_sym (output_bfd, sym, &msec, rel);
4598 if (!info->relocatable
4599 && (sym_sec->flags & SEC_MERGE) != 0
4600 && ELF_ST_TYPE (sym->st_info) == STT_SECTION
4601 && sym_sec->sec_info_type == ELF_INFO_TYPE_MERGE)
4603 struct elfNN_ia64_local_hash_entry *loc_h;
4605 loc_h = get_local_sym_hash (ia64_info, input_bfd, rel, FALSE);
4606 if (loc_h && ! loc_h->sec_merge_done)
4608 struct elfNN_ia64_dyn_sym_info *dynent;
4609 unsigned int count;
4611 for (count = loc_h->count, dynent = loc_h->info;
4612 count != 0;
4613 count--, dynent++)
4615 msec = sym_sec;
4616 dynent->addend =
4617 _bfd_merged_section_offset (output_bfd, &msec,
4618 elf_section_data (msec)->
4619 sec_info,
4620 sym->st_value
4621 + dynent->addend);
4622 dynent->addend -= sym->st_value;
4623 dynent->addend += msec->output_section->vma
4624 + msec->output_offset
4625 - sym_sec->output_section->vma
4626 - sym_sec->output_offset;
4629 /* We may have introduced duplicated entries. We need
4630 to remove them properly. */
4631 count = sort_dyn_sym_info (loc_h->info, loc_h->count);
4632 if (count != loc_h->count)
4634 loc_h->count = count;
4635 loc_h->sorted_count = count;
4638 loc_h->sec_merge_done = 1;
4642 else
4644 bfd_boolean unresolved_reloc;
4645 bfd_boolean warned;
4646 struct elf_link_hash_entry **sym_hashes = elf_sym_hashes (input_bfd);
4648 RELOC_FOR_GLOBAL_SYMBOL (info, input_bfd, input_section, rel,
4649 r_symndx, symtab_hdr, sym_hashes,
4650 h, sym_sec, value,
4651 unresolved_reloc, warned);
4653 if (h->root.type == bfd_link_hash_undefweak)
4654 undef_weak_ref = TRUE;
4655 else if (warned)
4656 continue;
4659 /* For relocs against symbols from removed linkonce sections,
4660 or sections discarded by a linker script, we just want the
4661 section contents zeroed. Avoid any special processing. */
4662 if (sym_sec != NULL && elf_discarded_section (sym_sec))
4664 _bfd_clear_contents (howto, input_bfd, contents + rel->r_offset);
4665 rel->r_info = 0;
4666 rel->r_addend = 0;
4667 continue;
4670 if (info->relocatable)
4671 continue;
4673 hit_addr = contents + rel->r_offset;
4674 value += rel->r_addend;
4675 dynamic_symbol_p = elfNN_ia64_dynamic_symbol_p (h, info, r_type);
4677 switch (r_type)
4679 case R_IA64_NONE:
4680 case R_IA64_LDXMOV:
4681 continue;
4683 case R_IA64_IMM14:
4684 case R_IA64_IMM22:
4685 case R_IA64_IMM64:
4686 case R_IA64_DIR32MSB:
4687 case R_IA64_DIR32LSB:
4688 case R_IA64_DIR64MSB:
4689 case R_IA64_DIR64LSB:
4690 /* Install a dynamic relocation for this reloc. */
4691 if ((dynamic_symbol_p || info->shared)
4692 && r_symndx != 0
4693 && (input_section->flags & SEC_ALLOC) != 0)
4695 unsigned int dyn_r_type;
4696 long dynindx;
4697 bfd_vma addend;
4699 BFD_ASSERT (srel != NULL);
4701 switch (r_type)
4703 case R_IA64_IMM14:
4704 case R_IA64_IMM22:
4705 case R_IA64_IMM64:
4706 /* ??? People shouldn't be doing non-pic code in
4707 shared libraries nor dynamic executables. */
4708 (*_bfd_error_handler)
4709 (_("%B: non-pic code with imm relocation against dynamic symbol `%s'"),
4710 input_bfd,
4711 h ? h->root.root.string
4712 : bfd_elf_sym_name (input_bfd, symtab_hdr, sym,
4713 sym_sec));
4714 ret_val = FALSE;
4715 continue;
4717 default:
4718 break;
4721 /* If we don't need dynamic symbol lookup, find a
4722 matching RELATIVE relocation. */
4723 dyn_r_type = r_type;
4724 if (dynamic_symbol_p)
4726 dynindx = h->dynindx;
4727 addend = rel->r_addend;
4728 value = 0;
4730 else
4732 switch (r_type)
4734 case R_IA64_DIR32MSB:
4735 dyn_r_type = R_IA64_REL32MSB;
4736 break;
4737 case R_IA64_DIR32LSB:
4738 dyn_r_type = R_IA64_REL32LSB;
4739 break;
4740 case R_IA64_DIR64MSB:
4741 dyn_r_type = R_IA64_REL64MSB;
4742 break;
4743 case R_IA64_DIR64LSB:
4744 dyn_r_type = R_IA64_REL64LSB;
4745 break;
4747 default:
4748 break;
4750 dynindx = 0;
4751 addend = value;
4754 elfNN_ia64_install_dyn_reloc (output_bfd, info, input_section,
4755 srel, rel->r_offset, dyn_r_type,
4756 dynindx, addend);
4758 /* Fall through. */
4760 case R_IA64_LTV32MSB:
4761 case R_IA64_LTV32LSB:
4762 case R_IA64_LTV64MSB:
4763 case R_IA64_LTV64LSB:
4764 r = elfNN_ia64_install_value (hit_addr, value, r_type);
4765 break;
4767 case R_IA64_GPREL22:
4768 case R_IA64_GPREL64I:
4769 case R_IA64_GPREL32MSB:
4770 case R_IA64_GPREL32LSB:
4771 case R_IA64_GPREL64MSB:
4772 case R_IA64_GPREL64LSB:
4773 if (dynamic_symbol_p)
4775 (*_bfd_error_handler)
4776 (_("%B: @gprel relocation against dynamic symbol %s"),
4777 input_bfd,
4778 h ? h->root.root.string
4779 : bfd_elf_sym_name (input_bfd, symtab_hdr, sym,
4780 sym_sec));
4781 ret_val = FALSE;
4782 continue;
4784 value -= gp_val;
4785 r = elfNN_ia64_install_value (hit_addr, value, r_type);
4786 break;
4788 case R_IA64_LTOFF22:
4789 case R_IA64_LTOFF22X:
4790 case R_IA64_LTOFF64I:
4791 dyn_i = get_dyn_sym_info (ia64_info, h, input_bfd, rel, FALSE);
4792 value = set_got_entry (input_bfd, info, dyn_i, (h ? h->dynindx : -1),
4793 rel->r_addend, value, R_IA64_DIRNNLSB);
4794 value -= gp_val;
4795 r = elfNN_ia64_install_value (hit_addr, value, r_type);
4796 break;
4798 case R_IA64_PLTOFF22:
4799 case R_IA64_PLTOFF64I:
4800 case R_IA64_PLTOFF64MSB:
4801 case R_IA64_PLTOFF64LSB:
4802 dyn_i = get_dyn_sym_info (ia64_info, h, input_bfd, rel, FALSE);
4803 value = set_pltoff_entry (output_bfd, info, dyn_i, value, FALSE);
4804 value -= gp_val;
4805 r = elfNN_ia64_install_value (hit_addr, value, r_type);
4806 break;
4808 case R_IA64_FPTR64I:
4809 case R_IA64_FPTR32MSB:
4810 case R_IA64_FPTR32LSB:
4811 case R_IA64_FPTR64MSB:
4812 case R_IA64_FPTR64LSB:
4813 dyn_i = get_dyn_sym_info (ia64_info, h, input_bfd, rel, FALSE);
4814 if (dyn_i->want_fptr)
4816 if (!undef_weak_ref)
4817 value = set_fptr_entry (output_bfd, info, dyn_i, value);
4819 if (!dyn_i->want_fptr || info->pie)
4821 long dynindx;
4822 unsigned int dyn_r_type = r_type;
4823 bfd_vma addend = rel->r_addend;
4825 /* Otherwise, we expect the dynamic linker to create
4826 the entry. */
4828 if (dyn_i->want_fptr)
4830 if (r_type == R_IA64_FPTR64I)
4832 /* We can't represent this without a dynamic symbol.
4833 Adjust the relocation to be against an output
4834 section symbol, which are always present in the
4835 dynamic symbol table. */
4836 /* ??? People shouldn't be doing non-pic code in
4837 shared libraries. Hork. */
4838 (*_bfd_error_handler)
4839 (_("%B: linking non-pic code in a position independent executable"),
4840 input_bfd);
4841 ret_val = FALSE;
4842 continue;
4844 dynindx = 0;
4845 addend = value;
4846 dyn_r_type = r_type + R_IA64_RELNNLSB - R_IA64_FPTRNNLSB;
4848 else if (h)
4850 if (h->dynindx != -1)
4851 dynindx = h->dynindx;
4852 else
4853 dynindx = (_bfd_elf_link_lookup_local_dynindx
4854 (info, h->root.u.def.section->owner,
4855 global_sym_index (h)));
4856 value = 0;
4858 else
4860 dynindx = (_bfd_elf_link_lookup_local_dynindx
4861 (info, input_bfd, (long) r_symndx));
4862 value = 0;
4865 elfNN_ia64_install_dyn_reloc (output_bfd, info, input_section,
4866 srel, rel->r_offset, dyn_r_type,
4867 dynindx, addend);
4870 r = elfNN_ia64_install_value (hit_addr, value, r_type);
4871 break;
4873 case R_IA64_LTOFF_FPTR22:
4874 case R_IA64_LTOFF_FPTR64I:
4875 case R_IA64_LTOFF_FPTR32MSB:
4876 case R_IA64_LTOFF_FPTR32LSB:
4877 case R_IA64_LTOFF_FPTR64MSB:
4878 case R_IA64_LTOFF_FPTR64LSB:
4880 long dynindx;
4882 dyn_i = get_dyn_sym_info (ia64_info, h, input_bfd, rel, FALSE);
4883 if (dyn_i->want_fptr)
4885 BFD_ASSERT (h == NULL || h->dynindx == -1);
4886 if (!undef_weak_ref)
4887 value = set_fptr_entry (output_bfd, info, dyn_i, value);
4888 dynindx = -1;
4890 else
4892 /* Otherwise, we expect the dynamic linker to create
4893 the entry. */
4894 if (h)
4896 if (h->dynindx != -1)
4897 dynindx = h->dynindx;
4898 else
4899 dynindx = (_bfd_elf_link_lookup_local_dynindx
4900 (info, h->root.u.def.section->owner,
4901 global_sym_index (h)));
4903 else
4904 dynindx = (_bfd_elf_link_lookup_local_dynindx
4905 (info, input_bfd, (long) r_symndx));
4906 value = 0;
4909 value = set_got_entry (output_bfd, info, dyn_i, dynindx,
4910 rel->r_addend, value, R_IA64_FPTRNNLSB);
4911 value -= gp_val;
4912 r = elfNN_ia64_install_value (hit_addr, value, r_type);
4914 break;
4916 case R_IA64_PCREL32MSB:
4917 case R_IA64_PCREL32LSB:
4918 case R_IA64_PCREL64MSB:
4919 case R_IA64_PCREL64LSB:
4920 /* Install a dynamic relocation for this reloc. */
4921 if (dynamic_symbol_p && r_symndx != 0)
4923 BFD_ASSERT (srel != NULL);
4925 elfNN_ia64_install_dyn_reloc (output_bfd, info, input_section,
4926 srel, rel->r_offset, r_type,
4927 h->dynindx, rel->r_addend);
4929 goto finish_pcrel;
4931 case R_IA64_PCREL21B:
4932 case R_IA64_PCREL60B:
4933 /* We should have created a PLT entry for any dynamic symbol. */
4934 dyn_i = NULL;
4935 if (h)
4936 dyn_i = get_dyn_sym_info (ia64_info, h, NULL, NULL, FALSE);
4938 if (dyn_i && dyn_i->want_plt2)
4940 /* Should have caught this earlier. */
4941 BFD_ASSERT (rel->r_addend == 0);
4943 value = (ia64_info->root.splt->output_section->vma
4944 + ia64_info->root.splt->output_offset
4945 + dyn_i->plt2_offset);
4947 else
4949 /* Since there's no PLT entry, Validate that this is
4950 locally defined. */
4951 BFD_ASSERT (undef_weak_ref || sym_sec->output_section != NULL);
4953 /* If the symbol is undef_weak, we shouldn't be trying
4954 to call it. There's every chance that we'd wind up
4955 with an out-of-range fixup here. Don't bother setting
4956 any value at all. */
4957 if (undef_weak_ref)
4958 continue;
4960 goto finish_pcrel;
4962 case R_IA64_PCREL21BI:
4963 case R_IA64_PCREL21F:
4964 case R_IA64_PCREL21M:
4965 case R_IA64_PCREL22:
4966 case R_IA64_PCREL64I:
4967 /* The PCREL21BI reloc is specifically not intended for use with
4968 dynamic relocs. PCREL21F and PCREL21M are used for speculation
4969 fixup code, and thus probably ought not be dynamic. The
4970 PCREL22 and PCREL64I relocs aren't emitted as dynamic relocs. */
4971 if (dynamic_symbol_p)
4973 const char *msg;
4975 if (r_type == R_IA64_PCREL21BI)
4976 msg = _("%B: @internal branch to dynamic symbol %s");
4977 else if (r_type == R_IA64_PCREL21F || r_type == R_IA64_PCREL21M)
4978 msg = _("%B: speculation fixup to dynamic symbol %s");
4979 else
4980 msg = _("%B: @pcrel relocation against dynamic symbol %s");
4981 (*_bfd_error_handler) (msg, input_bfd,
4982 h ? h->root.root.string
4983 : bfd_elf_sym_name (input_bfd,
4984 symtab_hdr,
4985 sym,
4986 sym_sec));
4987 ret_val = FALSE;
4988 continue;
4990 goto finish_pcrel;
4992 finish_pcrel:
4993 /* Make pc-relative. */
4994 value -= (input_section->output_section->vma
4995 + input_section->output_offset
4996 + rel->r_offset) & ~ (bfd_vma) 0x3;
4997 r = elfNN_ia64_install_value (hit_addr, value, r_type);
4998 break;
5000 case R_IA64_SEGREL32MSB:
5001 case R_IA64_SEGREL32LSB:
5002 case R_IA64_SEGREL64MSB:
5003 case R_IA64_SEGREL64LSB:
5005 /* Find the segment that contains the output_section. */
5006 Elf_Internal_Phdr *p = _bfd_elf_find_segment_containing_section
5007 (output_bfd, input_section->output_section);
5009 if (p == NULL)
5011 r = bfd_reloc_notsupported;
5013 else
5015 /* The VMA of the segment is the vaddr of the associated
5016 program header. */
5017 if (value > p->p_vaddr)
5018 value -= p->p_vaddr;
5019 else
5020 value = 0;
5021 r = elfNN_ia64_install_value (hit_addr, value, r_type);
5023 break;
5026 case R_IA64_SECREL32MSB:
5027 case R_IA64_SECREL32LSB:
5028 case R_IA64_SECREL64MSB:
5029 case R_IA64_SECREL64LSB:
5030 /* Make output-section relative to section where the symbol
5031 is defined. PR 475 */
5032 if (sym_sec)
5033 value -= sym_sec->output_section->vma;
5034 r = elfNN_ia64_install_value (hit_addr, value, r_type);
5035 break;
5037 case R_IA64_IPLTMSB:
5038 case R_IA64_IPLTLSB:
5039 /* Install a dynamic relocation for this reloc. */
5040 if ((dynamic_symbol_p || info->shared)
5041 && (input_section->flags & SEC_ALLOC) != 0)
5043 BFD_ASSERT (srel != NULL);
5045 /* If we don't need dynamic symbol lookup, install two
5046 RELATIVE relocations. */
5047 if (!dynamic_symbol_p)
5049 unsigned int dyn_r_type;
5051 if (r_type == R_IA64_IPLTMSB)
5052 dyn_r_type = R_IA64_REL64MSB;
5053 else
5054 dyn_r_type = R_IA64_REL64LSB;
5056 elfNN_ia64_install_dyn_reloc (output_bfd, info,
5057 input_section,
5058 srel, rel->r_offset,
5059 dyn_r_type, 0, value);
5060 elfNN_ia64_install_dyn_reloc (output_bfd, info,
5061 input_section,
5062 srel, rel->r_offset + 8,
5063 dyn_r_type, 0, gp_val);
5065 else
5066 elfNN_ia64_install_dyn_reloc (output_bfd, info, input_section,
5067 srel, rel->r_offset, r_type,
5068 h->dynindx, rel->r_addend);
5071 if (r_type == R_IA64_IPLTMSB)
5072 r_type = R_IA64_DIR64MSB;
5073 else
5074 r_type = R_IA64_DIR64LSB;
5075 elfNN_ia64_install_value (hit_addr, value, r_type);
5076 r = elfNN_ia64_install_value (hit_addr + 8, gp_val, r_type);
5077 break;
5079 case R_IA64_TPREL14:
5080 case R_IA64_TPREL22:
5081 case R_IA64_TPREL64I:
5082 if (elf_hash_table (info)->tls_sec == NULL)
5083 goto missing_tls_sec;
5084 value -= elfNN_ia64_tprel_base (info);
5085 r = elfNN_ia64_install_value (hit_addr, value, r_type);
5086 break;
5088 case R_IA64_DTPREL14:
5089 case R_IA64_DTPREL22:
5090 case R_IA64_DTPREL64I:
5091 case R_IA64_DTPREL32LSB:
5092 case R_IA64_DTPREL32MSB:
5093 case R_IA64_DTPREL64LSB:
5094 case R_IA64_DTPREL64MSB:
5095 if (elf_hash_table (info)->tls_sec == NULL)
5096 goto missing_tls_sec;
5097 value -= elfNN_ia64_dtprel_base (info);
5098 r = elfNN_ia64_install_value (hit_addr, value, r_type);
5099 break;
5101 case R_IA64_LTOFF_TPREL22:
5102 case R_IA64_LTOFF_DTPMOD22:
5103 case R_IA64_LTOFF_DTPREL22:
5105 int got_r_type;
5106 long dynindx = h ? h->dynindx : -1;
5107 bfd_vma r_addend = rel->r_addend;
5109 switch (r_type)
5111 default:
5112 case R_IA64_LTOFF_TPREL22:
5113 if (!dynamic_symbol_p)
5115 if (elf_hash_table (info)->tls_sec == NULL)
5116 goto missing_tls_sec;
5117 if (!info->shared)
5118 value -= elfNN_ia64_tprel_base (info);
5119 else
5121 r_addend += value - elfNN_ia64_dtprel_base (info);
5122 dynindx = 0;
5125 got_r_type = R_IA64_TPREL64LSB;
5126 break;
5127 case R_IA64_LTOFF_DTPMOD22:
5128 if (!dynamic_symbol_p && !info->shared)
5129 value = 1;
5130 got_r_type = R_IA64_DTPMOD64LSB;
5131 break;
5132 case R_IA64_LTOFF_DTPREL22:
5133 if (!dynamic_symbol_p)
5135 if (elf_hash_table (info)->tls_sec == NULL)
5136 goto missing_tls_sec;
5137 value -= elfNN_ia64_dtprel_base (info);
5139 got_r_type = R_IA64_DTPRELNNLSB;
5140 break;
5142 dyn_i = get_dyn_sym_info (ia64_info, h, input_bfd, rel, FALSE);
5143 value = set_got_entry (input_bfd, info, dyn_i, dynindx, r_addend,
5144 value, got_r_type);
5145 value -= gp_val;
5146 r = elfNN_ia64_install_value (hit_addr, value, r_type);
5148 break;
5150 default:
5151 r = bfd_reloc_notsupported;
5152 break;
5155 switch (r)
5157 case bfd_reloc_ok:
5158 break;
5160 case bfd_reloc_undefined:
5161 /* This can happen for global table relative relocs if
5162 __gp is undefined. This is a panic situation so we
5163 don't try to continue. */
5164 (*info->callbacks->undefined_symbol)
5165 (info, "__gp", input_bfd, input_section, rel->r_offset, 1);
5166 return FALSE;
5168 case bfd_reloc_notsupported:
5170 const char *name;
5172 if (h)
5173 name = h->root.root.string;
5174 else
5175 name = bfd_elf_sym_name (input_bfd, symtab_hdr, sym,
5176 sym_sec);
5177 if (!(*info->callbacks->warning) (info, _("unsupported reloc"),
5178 name, input_bfd,
5179 input_section, rel->r_offset))
5180 return FALSE;
5181 ret_val = FALSE;
5183 break;
5185 case bfd_reloc_dangerous:
5186 case bfd_reloc_outofrange:
5187 case bfd_reloc_overflow:
5188 default:
5189 missing_tls_sec:
5191 const char *name;
5193 if (h)
5194 name = h->root.root.string;
5195 else
5196 name = bfd_elf_sym_name (input_bfd, symtab_hdr, sym,
5197 sym_sec);
5199 switch (r_type)
5201 case R_IA64_TPREL14:
5202 case R_IA64_TPREL22:
5203 case R_IA64_TPREL64I:
5204 case R_IA64_DTPREL14:
5205 case R_IA64_DTPREL22:
5206 case R_IA64_DTPREL64I:
5207 case R_IA64_DTPREL32LSB:
5208 case R_IA64_DTPREL32MSB:
5209 case R_IA64_DTPREL64LSB:
5210 case R_IA64_DTPREL64MSB:
5211 case R_IA64_LTOFF_TPREL22:
5212 case R_IA64_LTOFF_DTPMOD22:
5213 case R_IA64_LTOFF_DTPREL22:
5214 (*_bfd_error_handler)
5215 (_("%B: missing TLS section for relocation %s against `%s' at 0x%lx in section `%A'."),
5216 input_bfd, input_section, howto->name, name,
5217 rel->r_offset);
5218 break;
5220 case R_IA64_PCREL21B:
5221 case R_IA64_PCREL21BI:
5222 case R_IA64_PCREL21M:
5223 case R_IA64_PCREL21F:
5224 if (is_elf_hash_table (info->hash))
5226 /* Relaxtion is always performed for ELF output.
5227 Overflow failures for those relocations mean
5228 that the section is too big to relax. */
5229 (*_bfd_error_handler)
5230 (_("%B: Can't relax br (%s) to `%s' at 0x%lx in section `%A' with size 0x%lx (> 0x1000000)."),
5231 input_bfd, input_section, howto->name, name,
5232 rel->r_offset, input_section->size);
5233 break;
5235 default:
5236 if (!(*info->callbacks->reloc_overflow) (info,
5237 &h->root,
5238 name,
5239 howto->name,
5240 (bfd_vma) 0,
5241 input_bfd,
5242 input_section,
5243 rel->r_offset))
5244 return FALSE;
5245 break;
5248 ret_val = FALSE;
5250 break;
5254 return ret_val;
5257 static bfd_boolean
5258 elfNN_ia64_finish_dynamic_symbol (bfd *output_bfd,
5259 struct bfd_link_info *info,
5260 struct elf_link_hash_entry *h,
5261 Elf_Internal_Sym *sym)
5263 struct elfNN_ia64_link_hash_table *ia64_info;
5264 struct elfNN_ia64_dyn_sym_info *dyn_i;
5266 ia64_info = elfNN_ia64_hash_table (info);
5267 if (ia64_info == NULL)
5268 return FALSE;
5270 dyn_i = get_dyn_sym_info (ia64_info, h, NULL, NULL, FALSE);
5272 /* Fill in the PLT data, if required. */
5273 if (dyn_i && dyn_i->want_plt)
5275 Elf_Internal_Rela outrel;
5276 bfd_byte *loc;
5277 asection *plt_sec;
5278 bfd_vma plt_addr, pltoff_addr, gp_val, plt_index;
5280 gp_val = _bfd_get_gp_value (output_bfd);
5282 /* Initialize the minimal PLT entry. */
5284 plt_index = (dyn_i->plt_offset - PLT_HEADER_SIZE) / PLT_MIN_ENTRY_SIZE;
5285 plt_sec = ia64_info->root.splt;
5286 loc = plt_sec->contents + dyn_i->plt_offset;
5288 memcpy (loc, plt_min_entry, PLT_MIN_ENTRY_SIZE);
5289 elfNN_ia64_install_value (loc, plt_index, R_IA64_IMM22);
5290 elfNN_ia64_install_value (loc+2, -dyn_i->plt_offset, R_IA64_PCREL21B);
5292 plt_addr = (plt_sec->output_section->vma
5293 + plt_sec->output_offset
5294 + dyn_i->plt_offset);
5295 pltoff_addr = set_pltoff_entry (output_bfd, info, dyn_i, plt_addr, TRUE);
5297 /* Initialize the FULL PLT entry, if needed. */
5298 if (dyn_i->want_plt2)
5300 loc = plt_sec->contents + dyn_i->plt2_offset;
5302 memcpy (loc, plt_full_entry, PLT_FULL_ENTRY_SIZE);
5303 elfNN_ia64_install_value (loc, pltoff_addr - gp_val, R_IA64_IMM22);
5305 /* Mark the symbol as undefined, rather than as defined in the
5306 plt section. Leave the value alone. */
5307 /* ??? We didn't redefine it in adjust_dynamic_symbol in the
5308 first place. But perhaps elflink.c did some for us. */
5309 if (!h->def_regular)
5310 sym->st_shndx = SHN_UNDEF;
5313 /* Create the dynamic relocation. */
5314 outrel.r_offset = pltoff_addr;
5315 if (bfd_little_endian (output_bfd))
5316 outrel.r_info = ELFNN_R_INFO (h->dynindx, R_IA64_IPLTLSB);
5317 else
5318 outrel.r_info = ELFNN_R_INFO (h->dynindx, R_IA64_IPLTMSB);
5319 outrel.r_addend = 0;
5321 /* This is fun. In the .IA_64.pltoff section, we've got entries
5322 that correspond both to real PLT entries, and those that
5323 happened to resolve to local symbols but need to be created
5324 to satisfy @pltoff relocations. The .rela.IA_64.pltoff
5325 relocations for the real PLT should come at the end of the
5326 section, so that they can be indexed by plt entry at runtime.
5328 We emitted all of the relocations for the non-PLT @pltoff
5329 entries during relocate_section. So we can consider the
5330 existing sec->reloc_count to be the base of the array of
5331 PLT relocations. */
5333 loc = ia64_info->rel_pltoff_sec->contents;
5334 loc += ((ia64_info->rel_pltoff_sec->reloc_count + plt_index)
5335 * sizeof (ElfNN_External_Rela));
5336 bfd_elfNN_swap_reloca_out (output_bfd, &outrel, loc);
5339 /* Mark some specially defined symbols as absolute. */
5340 if (strcmp (h->root.root.string, "_DYNAMIC") == 0
5341 || h == ia64_info->root.hgot
5342 || h == ia64_info->root.hplt)
5343 sym->st_shndx = SHN_ABS;
5345 return TRUE;
5348 static bfd_boolean
5349 elfNN_ia64_finish_dynamic_sections (bfd *abfd,
5350 struct bfd_link_info *info)
5352 struct elfNN_ia64_link_hash_table *ia64_info;
5353 bfd *dynobj;
5355 ia64_info = elfNN_ia64_hash_table (info);
5356 if (ia64_info == NULL)
5357 return FALSE;
5359 dynobj = ia64_info->root.dynobj;
5361 if (elf_hash_table (info)->dynamic_sections_created)
5363 ElfNN_External_Dyn *dyncon, *dynconend;
5364 asection *sdyn, *sgotplt;
5365 bfd_vma gp_val;
5367 sdyn = bfd_get_section_by_name (dynobj, ".dynamic");
5368 sgotplt = bfd_get_section_by_name (dynobj, ".got.plt");
5369 BFD_ASSERT (sdyn != NULL);
5370 dyncon = (ElfNN_External_Dyn *) sdyn->contents;
5371 dynconend = (ElfNN_External_Dyn *) (sdyn->contents + sdyn->size);
5373 gp_val = _bfd_get_gp_value (abfd);
5375 for (; dyncon < dynconend; dyncon++)
5377 Elf_Internal_Dyn dyn;
5379 bfd_elfNN_swap_dyn_in (dynobj, dyncon, &dyn);
5381 switch (dyn.d_tag)
5383 case DT_PLTGOT:
5384 dyn.d_un.d_ptr = gp_val;
5385 break;
5387 case DT_PLTRELSZ:
5388 dyn.d_un.d_val = (ia64_info->minplt_entries
5389 * sizeof (ElfNN_External_Rela));
5390 break;
5392 case DT_JMPREL:
5393 /* See the comment above in finish_dynamic_symbol. */
5394 dyn.d_un.d_ptr = (ia64_info->rel_pltoff_sec->output_section->vma
5395 + ia64_info->rel_pltoff_sec->output_offset
5396 + (ia64_info->rel_pltoff_sec->reloc_count
5397 * sizeof (ElfNN_External_Rela)));
5398 break;
5400 case DT_IA_64_PLT_RESERVE:
5401 dyn.d_un.d_ptr = (sgotplt->output_section->vma
5402 + sgotplt->output_offset);
5403 break;
5405 case DT_RELASZ:
5406 /* Do not have RELASZ include JMPREL. This makes things
5407 easier on ld.so. This is not what the rest of BFD set up. */
5408 dyn.d_un.d_val -= (ia64_info->minplt_entries
5409 * sizeof (ElfNN_External_Rela));
5410 break;
5413 bfd_elfNN_swap_dyn_out (abfd, &dyn, dyncon);
5416 /* Initialize the PLT0 entry. */
5417 if (ia64_info->root.splt)
5419 bfd_byte *loc = ia64_info->root.splt->contents;
5420 bfd_vma pltres;
5422 memcpy (loc, plt_header, PLT_HEADER_SIZE);
5424 pltres = (sgotplt->output_section->vma
5425 + sgotplt->output_offset
5426 - gp_val);
5428 elfNN_ia64_install_value (loc+1, pltres, R_IA64_GPREL22);
5432 return TRUE;
5435 /* ELF file flag handling: */
5437 /* Function to keep IA-64 specific file flags. */
5438 static bfd_boolean
5439 elfNN_ia64_set_private_flags (bfd *abfd, flagword flags)
5441 BFD_ASSERT (!elf_flags_init (abfd)
5442 || elf_elfheader (abfd)->e_flags == flags);
5444 elf_elfheader (abfd)->e_flags = flags;
5445 elf_flags_init (abfd) = TRUE;
5446 return TRUE;
5449 /* Merge backend specific data from an object file to the output
5450 object file when linking. */
5451 static bfd_boolean
5452 elfNN_ia64_merge_private_bfd_data (bfd *ibfd, bfd *obfd)
5454 flagword out_flags;
5455 flagword in_flags;
5456 bfd_boolean ok = TRUE;
5458 /* Don't even pretend to support mixed-format linking. */
5459 if (bfd_get_flavour (ibfd) != bfd_target_elf_flavour
5460 || bfd_get_flavour (obfd) != bfd_target_elf_flavour)
5461 return FALSE;
5463 in_flags = elf_elfheader (ibfd)->e_flags;
5464 out_flags = elf_elfheader (obfd)->e_flags;
5466 if (! elf_flags_init (obfd))
5468 elf_flags_init (obfd) = TRUE;
5469 elf_elfheader (obfd)->e_flags = in_flags;
5471 if (bfd_get_arch (obfd) == bfd_get_arch (ibfd)
5472 && bfd_get_arch_info (obfd)->the_default)
5474 return bfd_set_arch_mach (obfd, bfd_get_arch (ibfd),
5475 bfd_get_mach (ibfd));
5478 return TRUE;
5481 /* Check flag compatibility. */
5482 if (in_flags == out_flags)
5483 return TRUE;
5485 /* Output has EF_IA_64_REDUCEDFP set only if all inputs have it set. */
5486 if (!(in_flags & EF_IA_64_REDUCEDFP) && (out_flags & EF_IA_64_REDUCEDFP))
5487 elf_elfheader (obfd)->e_flags &= ~EF_IA_64_REDUCEDFP;
5489 if ((in_flags & EF_IA_64_TRAPNIL) != (out_flags & EF_IA_64_TRAPNIL))
5491 (*_bfd_error_handler)
5492 (_("%B: linking trap-on-NULL-dereference with non-trapping files"),
5493 ibfd);
5495 bfd_set_error (bfd_error_bad_value);
5496 ok = FALSE;
5498 if ((in_flags & EF_IA_64_BE) != (out_flags & EF_IA_64_BE))
5500 (*_bfd_error_handler)
5501 (_("%B: linking big-endian files with little-endian files"),
5502 ibfd);
5504 bfd_set_error (bfd_error_bad_value);
5505 ok = FALSE;
5507 if ((in_flags & EF_IA_64_ABI64) != (out_flags & EF_IA_64_ABI64))
5509 (*_bfd_error_handler)
5510 (_("%B: linking 64-bit files with 32-bit files"),
5511 ibfd);
5513 bfd_set_error (bfd_error_bad_value);
5514 ok = FALSE;
5516 if ((in_flags & EF_IA_64_CONS_GP) != (out_flags & EF_IA_64_CONS_GP))
5518 (*_bfd_error_handler)
5519 (_("%B: linking constant-gp files with non-constant-gp files"),
5520 ibfd);
5522 bfd_set_error (bfd_error_bad_value);
5523 ok = FALSE;
5525 if ((in_flags & EF_IA_64_NOFUNCDESC_CONS_GP)
5526 != (out_flags & EF_IA_64_NOFUNCDESC_CONS_GP))
5528 (*_bfd_error_handler)
5529 (_("%B: linking auto-pic files with non-auto-pic files"),
5530 ibfd);
5532 bfd_set_error (bfd_error_bad_value);
5533 ok = FALSE;
5536 return ok;
5539 static bfd_boolean
5540 elfNN_ia64_print_private_bfd_data (bfd *abfd, PTR ptr)
5542 FILE *file = (FILE *) ptr;
5543 flagword flags = elf_elfheader (abfd)->e_flags;
5545 BFD_ASSERT (abfd != NULL && ptr != NULL);
5547 fprintf (file, "private flags = %s%s%s%s%s%s%s%s\n",
5548 (flags & EF_IA_64_TRAPNIL) ? "TRAPNIL, " : "",
5549 (flags & EF_IA_64_EXT) ? "EXT, " : "",
5550 (flags & EF_IA_64_BE) ? "BE, " : "LE, ",
5551 (flags & EF_IA_64_REDUCEDFP) ? "REDUCEDFP, " : "",
5552 (flags & EF_IA_64_CONS_GP) ? "CONS_GP, " : "",
5553 (flags & EF_IA_64_NOFUNCDESC_CONS_GP) ? "NOFUNCDESC_CONS_GP, " : "",
5554 (flags & EF_IA_64_ABSOLUTE) ? "ABSOLUTE, " : "",
5555 (flags & EF_IA_64_ABI64) ? "ABI64" : "ABI32");
5557 _bfd_elf_print_private_bfd_data (abfd, ptr);
5558 return TRUE;
5561 static enum elf_reloc_type_class
5562 elfNN_ia64_reloc_type_class (const Elf_Internal_Rela *rela)
5564 switch ((int) ELFNN_R_TYPE (rela->r_info))
5566 case R_IA64_REL32MSB:
5567 case R_IA64_REL32LSB:
5568 case R_IA64_REL64MSB:
5569 case R_IA64_REL64LSB:
5570 return reloc_class_relative;
5571 case R_IA64_IPLTMSB:
5572 case R_IA64_IPLTLSB:
5573 return reloc_class_plt;
5574 case R_IA64_COPY:
5575 return reloc_class_copy;
5576 default:
5577 return reloc_class_normal;
5581 static const struct bfd_elf_special_section elfNN_ia64_special_sections[] =
5583 { STRING_COMMA_LEN (".sbss"), -1, SHT_NOBITS, SHF_ALLOC + SHF_WRITE + SHF_IA_64_SHORT },
5584 { STRING_COMMA_LEN (".sdata"), -1, SHT_PROGBITS, SHF_ALLOC + SHF_WRITE + SHF_IA_64_SHORT },
5585 { NULL, 0, 0, 0, 0 }
5588 static bfd_boolean
5589 elfNN_ia64_object_p (bfd *abfd)
5591 asection *sec;
5592 asection *group, *unwi, *unw;
5593 flagword flags;
5594 const char *name;
5595 char *unwi_name, *unw_name;
5596 bfd_size_type amt;
5598 if (abfd->flags & DYNAMIC)
5599 return TRUE;
5601 /* Flags for fake group section. */
5602 flags = (SEC_LINKER_CREATED | SEC_GROUP | SEC_LINK_ONCE
5603 | SEC_EXCLUDE);
5605 /* We add a fake section group for each .gnu.linkonce.t.* section,
5606 which isn't in a section group, and its unwind sections. */
5607 for (sec = abfd->sections; sec != NULL; sec = sec->next)
5609 if (elf_sec_group (sec) == NULL
5610 && ((sec->flags & (SEC_LINK_ONCE | SEC_CODE | SEC_GROUP))
5611 == (SEC_LINK_ONCE | SEC_CODE))
5612 && CONST_STRNEQ (sec->name, ".gnu.linkonce.t."))
5614 name = sec->name + 16;
5616 amt = strlen (name) + sizeof (".gnu.linkonce.ia64unwi.");
5617 unwi_name = bfd_alloc (abfd, amt);
5618 if (!unwi_name)
5619 return FALSE;
5621 strcpy (stpcpy (unwi_name, ".gnu.linkonce.ia64unwi."), name);
5622 unwi = bfd_get_section_by_name (abfd, unwi_name);
5624 amt = strlen (name) + sizeof (".gnu.linkonce.ia64unw.");
5625 unw_name = bfd_alloc (abfd, amt);
5626 if (!unw_name)
5627 return FALSE;
5629 strcpy (stpcpy (unw_name, ".gnu.linkonce.ia64unw."), name);
5630 unw = bfd_get_section_by_name (abfd, unw_name);
5632 /* We need to create a fake group section for it and its
5633 unwind sections. */
5634 group = bfd_make_section_anyway_with_flags (abfd, name,
5635 flags);
5636 if (group == NULL)
5637 return FALSE;
5639 /* Move the fake group section to the beginning. */
5640 bfd_section_list_remove (abfd, group);
5641 bfd_section_list_prepend (abfd, group);
5643 elf_next_in_group (group) = sec;
5645 elf_group_name (sec) = name;
5646 elf_next_in_group (sec) = sec;
5647 elf_sec_group (sec) = group;
5649 if (unwi)
5651 elf_group_name (unwi) = name;
5652 elf_next_in_group (unwi) = sec;
5653 elf_next_in_group (sec) = unwi;
5654 elf_sec_group (unwi) = group;
5657 if (unw)
5659 elf_group_name (unw) = name;
5660 if (unwi)
5662 elf_next_in_group (unw) = elf_next_in_group (unwi);
5663 elf_next_in_group (unwi) = unw;
5665 else
5667 elf_next_in_group (unw) = sec;
5668 elf_next_in_group (sec) = unw;
5670 elf_sec_group (unw) = group;
5673 /* Fake SHT_GROUP section header. */
5674 elf_section_data (group)->this_hdr.bfd_section = group;
5675 elf_section_data (group)->this_hdr.sh_type = SHT_GROUP;
5678 return TRUE;
5681 static bfd_boolean
5682 elfNN_ia64_hpux_vec (const bfd_target *vec)
5684 extern const bfd_target bfd_elfNN_ia64_hpux_big_vec;
5685 return (vec == & bfd_elfNN_ia64_hpux_big_vec);
5688 static void
5689 elfNN_hpux_post_process_headers (bfd *abfd,
5690 struct bfd_link_info *info ATTRIBUTE_UNUSED)
5692 Elf_Internal_Ehdr *i_ehdrp = elf_elfheader (abfd);
5694 i_ehdrp->e_ident[EI_OSABI] = get_elf_backend_data (abfd)->elf_osabi;
5695 i_ehdrp->e_ident[EI_ABIVERSION] = 1;
5698 static bfd_boolean
5699 elfNN_hpux_backend_section_from_bfd_section (bfd *abfd ATTRIBUTE_UNUSED,
5700 asection *sec, int *retval)
5702 if (bfd_is_com_section (sec))
5704 *retval = SHN_IA_64_ANSI_COMMON;
5705 return TRUE;
5707 return FALSE;
5710 static void
5711 elfNN_hpux_backend_symbol_processing (bfd *abfd ATTRIBUTE_UNUSED,
5712 asymbol *asym)
5714 elf_symbol_type *elfsym = (elf_symbol_type *) asym;
5716 switch (elfsym->internal_elf_sym.st_shndx)
5718 case SHN_IA_64_ANSI_COMMON:
5719 asym->section = bfd_com_section_ptr;
5720 asym->value = elfsym->internal_elf_sym.st_size;
5721 asym->flags &= ~BSF_GLOBAL;
5722 break;
5726 static bfd_boolean
5727 elfNN_vms_section_from_shdr (bfd *abfd,
5728 Elf_Internal_Shdr *hdr,
5729 const char *name,
5730 int shindex)
5732 asection *newsect;
5734 switch (hdr->sh_type)
5736 case SHT_IA_64_VMS_TRACE:
5737 case SHT_IA_64_VMS_DEBUG:
5738 case SHT_IA_64_VMS_DEBUG_STR:
5739 break;
5741 default:
5742 return elfNN_ia64_section_from_shdr (abfd, hdr, name, shindex);
5745 if (! _bfd_elf_make_section_from_shdr (abfd, hdr, name, shindex))
5746 return FALSE;
5747 newsect = hdr->bfd_section;
5749 return TRUE;
5752 static bfd_boolean
5753 elfNN_vms_object_p (bfd *abfd)
5755 Elf_Internal_Ehdr *i_ehdrp = elf_elfheader (abfd);
5756 Elf_Internal_Phdr *i_phdr = elf_tdata (abfd)->phdr;
5757 unsigned int i;
5758 unsigned int num_text = 0;
5759 unsigned int num_data = 0;
5760 unsigned int num_rodata = 0;
5761 char name[16];
5763 if (!elfNN_ia64_object_p (abfd))
5764 return FALSE;
5766 for (i = 0; i < i_ehdrp->e_phnum; i++, i_phdr++)
5768 /* Is there a section for this segment? */
5769 bfd_vma base_vma = i_phdr->p_vaddr;
5770 bfd_vma limit_vma = base_vma + i_phdr->p_filesz;
5772 if (i_phdr->p_type != PT_LOAD)
5773 continue;
5775 again:
5776 while (base_vma < limit_vma)
5778 bfd_vma next_vma = limit_vma;
5779 asection *nsec;
5780 asection *sec;
5781 flagword flags;
5782 char *nname = NULL;
5784 /* Find a section covering base_vma. */
5785 for (sec = abfd->sections; sec != NULL; sec = sec->next)
5787 if ((sec->flags & (SEC_ALLOC | SEC_LOAD)) == 0)
5788 continue;
5789 if (sec->vma <= base_vma && sec->vma + sec->size > base_vma)
5791 base_vma = sec->vma + sec->size;
5792 goto again;
5794 if (sec->vma < next_vma && sec->vma + sec->size >= base_vma)
5795 next_vma = sec->vma;
5798 /* No section covering [base_vma; next_vma). Create a fake one. */
5799 flags = SEC_ALLOC | SEC_LOAD | SEC_HAS_CONTENTS;
5800 if (i_phdr->p_flags & PF_X)
5802 flags |= SEC_CODE;
5803 if (num_text++ == 0)
5804 nname = ".text";
5805 else
5806 sprintf (name, ".text$%u", num_text);
5808 else if ((i_phdr->p_flags & (PF_R | PF_W)) == PF_R)
5810 flags |= SEC_READONLY;
5811 sprintf (name, ".rodata$%u", num_rodata++);
5813 else
5815 flags |= SEC_DATA;
5816 sprintf (name, ".data$%u", num_data++);
5819 /* Allocate name. */
5820 if (nname == NULL)
5822 size_t name_len = strlen (name) + 1;
5823 nname = bfd_alloc (abfd, name_len);
5824 if (nname == NULL)
5825 return FALSE;
5826 memcpy (nname, name, name_len);
5829 /* Create and fill new section. */
5830 nsec = bfd_make_section_anyway_with_flags (abfd, nname, flags);
5831 if (nsec == NULL)
5832 return FALSE;
5833 nsec->vma = base_vma;
5834 nsec->size = next_vma - base_vma;
5835 nsec->filepos = i_phdr->p_offset + (base_vma - i_phdr->p_vaddr);
5837 base_vma = next_vma;
5840 return TRUE;
5843 static void
5844 elfNN_vms_post_process_headers (bfd *abfd,
5845 struct bfd_link_info *info ATTRIBUTE_UNUSED)
5847 Elf_Internal_Ehdr *i_ehdrp = elf_elfheader (abfd);
5849 i_ehdrp->e_ident[EI_OSABI] = ELFOSABI_OPENVMS;
5850 i_ehdrp->e_ident[EI_ABIVERSION] = 2;
5853 static bfd_boolean
5854 elfNN_vms_section_processing (bfd *abfd ATTRIBUTE_UNUSED,
5855 Elf_Internal_Shdr *hdr)
5857 if (hdr->bfd_section != NULL)
5859 const char *name = bfd_get_section_name (abfd, hdr->bfd_section);
5861 if (strcmp (name, ".text") == 0)
5862 hdr->sh_flags |= SHF_IA_64_VMS_SHARED;
5863 else if ((strcmp (name, ".debug") == 0)
5864 || (strcmp (name, ".debug_abbrev") == 0)
5865 || (strcmp (name, ".debug_aranges") == 0)
5866 || (strcmp (name, ".debug_frame") == 0)
5867 || (strcmp (name, ".debug_info") == 0)
5868 || (strcmp (name, ".debug_loc") == 0)
5869 || (strcmp (name, ".debug_macinfo") == 0)
5870 || (strcmp (name, ".debug_pubnames") == 0)
5871 || (strcmp (name, ".debug_pubtypes") == 0))
5872 hdr->sh_type = SHT_IA_64_VMS_DEBUG;
5873 else if ((strcmp (name, ".debug_line") == 0)
5874 || (strcmp (name, ".debug_ranges") == 0))
5875 hdr->sh_type = SHT_IA_64_VMS_TRACE;
5876 else if (strcmp (name, ".debug_str") == 0)
5877 hdr->sh_type = SHT_IA_64_VMS_DEBUG_STR;
5878 else if (strcmp (name, ".vms_display_name_info") == 0)
5880 int idx, symcount;
5881 asymbol **syms;
5882 struct elf_obj_tdata *t = elf_tdata (abfd);
5883 int buf[2];
5884 int demangler_sym_idx = -1;
5886 symcount = bfd_get_symcount (abfd);
5887 syms = bfd_get_outsymbols (abfd);
5888 for (idx = 0; idx < symcount; idx++)
5890 asymbol *sym;
5891 sym = syms[idx];
5892 if ((sym->flags & (BSF_DEBUGGING | BSF_DYNAMIC))
5893 && strchr (sym->name, '@')
5894 && (strcmp (sym->section->name, BFD_ABS_SECTION_NAME) == 0))
5896 demangler_sym_idx = sym->udata.i;
5897 break;
5901 hdr->sh_type = SHT_IA_64_VMS_DISPLAY_NAME_INFO;
5902 hdr->sh_entsize = 4;
5903 hdr->sh_addralign = 0;
5904 hdr->sh_link = t->symtab_section;
5906 /* Find symtab index of demangler routine and stuff it in
5907 the second long word of section data. */
5909 if (demangler_sym_idx > -1)
5911 bfd_seek (abfd, hdr->sh_offset, SEEK_SET);
5912 bfd_bread (buf, hdr->sh_size, abfd);
5913 buf [1] = demangler_sym_idx;
5914 bfd_seek (abfd, hdr->sh_offset, SEEK_SET);
5915 bfd_bwrite (buf, hdr->sh_size, abfd);
5920 return TRUE;
5923 /* The final processing done just before writing out a VMS IA-64 ELF
5924 object file. */
5926 static void
5927 elfNN_vms_final_write_processing (bfd *abfd,
5928 bfd_boolean linker ATTRIBUTE_UNUSED)
5930 Elf_Internal_Shdr *hdr;
5931 asection *s;
5932 int unwind_info_sect_idx = 0;
5934 for (s = abfd->sections; s; s = s->next)
5936 hdr = &elf_section_data (s)->this_hdr;
5938 if (strcmp (bfd_get_section_name (abfd, hdr->bfd_section),
5939 ".IA_64.unwind_info") == 0)
5940 unwind_info_sect_idx = elf_section_data (s)->this_idx;
5942 switch (hdr->sh_type)
5944 case SHT_IA_64_UNWIND:
5945 /* VMS requires sh_info to point to the unwind info section. */
5946 hdr->sh_info = unwind_info_sect_idx;
5947 break;
5951 if (! elf_flags_init (abfd))
5953 unsigned long flags = 0;
5955 if (abfd->xvec->byteorder == BFD_ENDIAN_BIG)
5956 flags |= EF_IA_64_BE;
5957 if (bfd_get_mach (abfd) == bfd_mach_ia64_elf64)
5958 flags |= EF_IA_64_ABI64;
5960 elf_elfheader(abfd)->e_flags = flags;
5961 elf_flags_init (abfd) = TRUE;
5965 static bfd_boolean
5966 elfNN_vms_close_and_cleanup (bfd *abfd)
5968 if (bfd_get_format (abfd) == bfd_object)
5970 long isize, irsize;
5972 if (elf_shstrtab (abfd) != NULL)
5973 _bfd_elf_strtab_free (elf_shstrtab (abfd));
5975 /* Pad to 8 byte boundary for IPF/VMS. */
5976 isize = bfd_get_size (abfd);
5977 if ((irsize = isize/8*8) < isize)
5979 int ishort = (irsize + 8) - isize;
5980 bfd_seek (abfd, isize, SEEK_SET);
5981 bfd_bwrite (bfd_zmalloc (ishort), ishort, abfd);
5985 return _bfd_generic_close_and_cleanup (abfd);
5988 #define TARGET_LITTLE_SYM bfd_elfNN_ia64_little_vec
5989 #define TARGET_LITTLE_NAME "elfNN-ia64-little"
5990 #define TARGET_BIG_SYM bfd_elfNN_ia64_big_vec
5991 #define TARGET_BIG_NAME "elfNN-ia64-big"
5992 #define ELF_ARCH bfd_arch_ia64
5993 #define ELF_MACHINE_CODE EM_IA_64
5994 #define ELF_MACHINE_ALT1 1999 /* EAS2.3 */
5995 #define ELF_MACHINE_ALT2 1998 /* EAS2.2 */
5996 #define ELF_MAXPAGESIZE 0x10000 /* 64KB */
5997 #define ELF_COMMONPAGESIZE 0x4000 /* 16KB */
5999 #define elf_backend_section_from_shdr \
6000 elfNN_ia64_section_from_shdr
6001 #define elf_backend_section_flags \
6002 elfNN_ia64_section_flags
6003 #define elf_backend_fake_sections \
6004 elfNN_ia64_fake_sections
6005 #define elf_backend_final_write_processing \
6006 elfNN_ia64_final_write_processing
6007 #define elf_backend_add_symbol_hook \
6008 elfNN_ia64_add_symbol_hook
6009 #define elf_backend_additional_program_headers \
6010 elfNN_ia64_additional_program_headers
6011 #define elf_backend_modify_segment_map \
6012 elfNN_ia64_modify_segment_map
6013 #define elf_backend_modify_program_headers \
6014 elfNN_ia64_modify_program_headers
6015 #define elf_info_to_howto \
6016 elfNN_ia64_info_to_howto
6018 #define bfd_elfNN_bfd_reloc_type_lookup \
6019 elfNN_ia64_reloc_type_lookup
6020 #define bfd_elfNN_bfd_reloc_name_lookup \
6021 elfNN_ia64_reloc_name_lookup
6022 #define bfd_elfNN_bfd_is_local_label_name \
6023 elfNN_ia64_is_local_label_name
6024 #define bfd_elfNN_bfd_relax_section \
6025 elfNN_ia64_relax_section
6027 #define elf_backend_object_p \
6028 elfNN_ia64_object_p
6030 /* Stuff for the BFD linker: */
6031 #define bfd_elfNN_bfd_link_hash_table_create \
6032 elfNN_ia64_hash_table_create
6033 #define bfd_elfNN_bfd_link_hash_table_free \
6034 elfNN_ia64_hash_table_free
6035 #define elf_backend_create_dynamic_sections \
6036 elfNN_ia64_create_dynamic_sections
6037 #define elf_backend_check_relocs \
6038 elfNN_ia64_check_relocs
6039 #define elf_backend_adjust_dynamic_symbol \
6040 elfNN_ia64_adjust_dynamic_symbol
6041 #define elf_backend_size_dynamic_sections \
6042 elfNN_ia64_size_dynamic_sections
6043 #define elf_backend_omit_section_dynsym \
6044 ((bfd_boolean (*) (bfd *, struct bfd_link_info *, asection *)) bfd_true)
6045 #define elf_backend_relocate_section \
6046 elfNN_ia64_relocate_section
6047 #define elf_backend_finish_dynamic_symbol \
6048 elfNN_ia64_finish_dynamic_symbol
6049 #define elf_backend_finish_dynamic_sections \
6050 elfNN_ia64_finish_dynamic_sections
6051 #define bfd_elfNN_bfd_final_link \
6052 elfNN_ia64_final_link
6054 #define bfd_elfNN_bfd_merge_private_bfd_data \
6055 elfNN_ia64_merge_private_bfd_data
6056 #define bfd_elfNN_bfd_set_private_flags \
6057 elfNN_ia64_set_private_flags
6058 #define bfd_elfNN_bfd_print_private_bfd_data \
6059 elfNN_ia64_print_private_bfd_data
6061 #define elf_backend_plt_readonly 1
6062 #define elf_backend_want_plt_sym 0
6063 #define elf_backend_plt_alignment 5
6064 #define elf_backend_got_header_size 0
6065 #define elf_backend_want_got_plt 1
6066 #define elf_backend_may_use_rel_p 1
6067 #define elf_backend_may_use_rela_p 1
6068 #define elf_backend_default_use_rela_p 1
6069 #define elf_backend_want_dynbss 0
6070 #define elf_backend_copy_indirect_symbol elfNN_ia64_hash_copy_indirect
6071 #define elf_backend_hide_symbol elfNN_ia64_hash_hide_symbol
6072 #define elf_backend_fixup_symbol _bfd_elf_link_hash_fixup_symbol
6073 #define elf_backend_reloc_type_class elfNN_ia64_reloc_type_class
6074 #define elf_backend_rela_normal 1
6075 #define elf_backend_special_sections elfNN_ia64_special_sections
6076 #define elf_backend_default_execstack 0
6078 /* FIXME: PR 290: The Intel C compiler generates SHT_IA_64_UNWIND with
6079 SHF_LINK_ORDER. But it doesn't set the sh_link or sh_info fields.
6080 We don't want to flood users with so many error messages. We turn
6081 off the warning for now. It will be turned on later when the Intel
6082 compiler is fixed. */
6083 #define elf_backend_link_order_error_handler NULL
6085 #include "elfNN-target.h"
6087 /* HPUX-specific vectors. */
6089 #undef TARGET_LITTLE_SYM
6090 #undef TARGET_LITTLE_NAME
6091 #undef TARGET_BIG_SYM
6092 #define TARGET_BIG_SYM bfd_elfNN_ia64_hpux_big_vec
6093 #undef TARGET_BIG_NAME
6094 #define TARGET_BIG_NAME "elfNN-ia64-hpux-big"
6096 /* These are HP-UX specific functions. */
6098 #undef elf_backend_post_process_headers
6099 #define elf_backend_post_process_headers elfNN_hpux_post_process_headers
6101 #undef elf_backend_section_from_bfd_section
6102 #define elf_backend_section_from_bfd_section elfNN_hpux_backend_section_from_bfd_section
6104 #undef elf_backend_symbol_processing
6105 #define elf_backend_symbol_processing elfNN_hpux_backend_symbol_processing
6107 #undef elf_backend_want_p_paddr_set_to_zero
6108 #define elf_backend_want_p_paddr_set_to_zero 1
6110 #undef ELF_COMMONPAGESIZE
6111 #undef ELF_OSABI
6112 #define ELF_OSABI ELFOSABI_HPUX
6114 #undef elfNN_bed
6115 #define elfNN_bed elfNN_ia64_hpux_bed
6117 #include "elfNN-target.h"
6119 /* VMS-specific vectors. */
6121 #undef TARGET_LITTLE_SYM
6122 #define TARGET_LITTLE_SYM bfd_elfNN_ia64_vms_vec
6123 #undef TARGET_LITTLE_NAME
6124 #define TARGET_LITTLE_NAME "elfNN-ia64-vms"
6125 #undef TARGET_BIG_SYM
6126 #undef TARGET_BIG_NAME
6128 /* These are VMS specific functions. */
6130 #undef elf_backend_object_p
6131 #define elf_backend_object_p elfNN_vms_object_p
6133 #undef elf_backend_section_from_shdr
6134 #define elf_backend_section_from_shdr elfNN_vms_section_from_shdr
6136 #undef elf_backend_post_process_headers
6137 #define elf_backend_post_process_headers elfNN_vms_post_process_headers
6139 #undef elf_backend_section_processing
6140 #define elf_backend_section_processing elfNN_vms_section_processing
6142 #undef elf_backend_final_write_processing
6143 #define elf_backend_final_write_processing elfNN_vms_final_write_processing
6145 #undef bfd_elfNN_close_and_cleanup
6146 #define bfd_elfNN_close_and_cleanup elfNN_vms_close_and_cleanup
6148 #undef elf_backend_section_from_bfd_section
6150 #undef elf_backend_symbol_processing
6152 #undef elf_backend_want_p_paddr_set_to_zero
6154 #undef ELF_OSABI
6155 #define ELF_OSABI ELFOSABI_OPENVMS
6157 #undef ELF_MAXPAGESIZE
6158 #define ELF_MAXPAGESIZE 0x10000 /* 64KB */
6160 #undef elfNN_bed
6161 #define elfNN_bed elfNN_ia64_vms_bed
6163 #include "elfNN-target.h"