Use std::vector on tdesc->reg_defs (gdbserver/tdesc.h)
[deliverable/binutils-gdb.git] / bfd / elfxx-ia64.c
CommitLineData
800eeca4 1/* IA-64 support for 64-bit ELF
2571583a 2 Copyright (C) 1998-2017 Free Software Foundation, Inc.
800eeca4
JW
3 Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
4
5e8d7549 5 This file is part of BFD, the Binary File Descriptor library.
800eeca4 6
5e8d7549
NC
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
cd123cb7 9 the Free Software Foundation; either version 3 of the License, or
5e8d7549 10 (at your option) any later version.
800eeca4 11
5e8d7549
NC
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
800eeca4 16
5e8d7549
NC
17 You should have received a copy of the GNU General Public License
18 along with this program; if not, write to the Free Software
cd123cb7
NC
19 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
20 MA 02110-1301, USA. */
800eeca4 21
800eeca4 22#include "sysdep.h"
3db64b00 23#include "bfd.h"
800eeca4
JW
24#include "libbfd.h"
25#include "elf-bfd.h"
26#include "opcode/ia64.h"
27#include "elf/ia64.h"
0aa92b58
JJ
28#include "objalloc.h"
29#include "hashtab.h"
6f8027ae 30#include "bfd_stdint.h"
cbe79dfe 31#include "elfxx-ia64.h"
5a260b66 32
5e8d7549 33/* THE RULES for all the stuff the linker creates --
b34976b6 34
5e8d7549
NC
35 GOT Entries created in response to LTOFF or LTOFF_FPTR
36 relocations. Dynamic relocs created for dynamic
37 symbols in an application; REL relocs for locals
38 in a shared library.
b34976b6 39
5e8d7549
NC
40 FPTR The canonical function descriptor. Created for local
41 symbols in applications. Descriptors for dynamic symbols
42 and local symbols in shared libraries are created by
43 ld.so. Thus there are no dynamic relocs against these
44 objects. The FPTR relocs for such _are_ passed through
45 to the dynamic relocation tables.
b34976b6 46
5e8d7549
NC
47 FULL_PLT Created for a PCREL21B relocation against a dynamic symbol.
48 Requires the creation of a PLTOFF entry. This does not
49 require any dynamic relocations.
b34976b6 50
5e8d7549
NC
51 PLTOFF Created by PLTOFF relocations. For local symbols, this
52 is an alternate function descriptor, and in shared libraries
53 requires two REL relocations. Note that this cannot be
54 transformed into an FPTR relocation, since it must be in
55 range of the GP. For dynamic symbols, this is a function
56 descriptor for a MIN_PLT entry, and requires one IPLT reloc.
b34976b6 57
5e8d7549 58 MIN_PLT Created by PLTOFF entries against dynamic symbols. This
4cc11e76 59 does not require dynamic relocations. */
800eeca4 60
cbe79dfe 61/* ia64-specific relocation. */
7256a114 62
800eeca4
JW
63#define NELEMS(a) ((int) (sizeof (a) / sizeof ((a)[0])))
64
800eeca4 65/* Perform a relocation. Not much to do here as all the hard work is
bbe66d08 66 done in elfNN_ia64_final_link_relocate. */
800eeca4 67static bfd_reloc_status_type
cbe79dfe
TG
68ia64_elf_reloc (bfd *abfd ATTRIBUTE_UNUSED, arelent *reloc,
69 asymbol *sym ATTRIBUTE_UNUSED,
70 PTR data ATTRIBUTE_UNUSED, asection *input_section,
71 bfd *output_bfd, char **error_message)
800eeca4
JW
72{
73 if (output_bfd)
74 {
75 reloc->address += input_section->output_offset;
76 return bfd_reloc_ok;
77 }
6e84a906
DJ
78
79 if (input_section->flags & SEC_DEBUGGING)
80 return bfd_reloc_continue;
81
cbe79dfe 82 *error_message = "Unsupported call to ia64_elf_reloc";
800eeca4
JW
83 return bfd_reloc_notsupported;
84}
85
86#define IA64_HOWTO(TYPE, NAME, SIZE, PCREL, IN) \
87 HOWTO (TYPE, 0, SIZE, 0, PCREL, 0, complain_overflow_signed, \
cbe79dfe 88 ia64_elf_reloc, NAME, FALSE, 0, -1, IN)
800eeca4
JW
89
90/* This table has to be sorted according to increasing number of the
91 TYPE field. */
92static reloc_howto_type ia64_howto_table[] =
93 {
6346d5ca 94 IA64_HOWTO (R_IA64_NONE, "NONE", 3, FALSE, TRUE),
b34976b6
AM
95
96 IA64_HOWTO (R_IA64_IMM14, "IMM14", 0, FALSE, TRUE),
97 IA64_HOWTO (R_IA64_IMM22, "IMM22", 0, FALSE, TRUE),
98 IA64_HOWTO (R_IA64_IMM64, "IMM64", 0, FALSE, TRUE),
99 IA64_HOWTO (R_IA64_DIR32MSB, "DIR32MSB", 2, FALSE, TRUE),
100 IA64_HOWTO (R_IA64_DIR32LSB, "DIR32LSB", 2, FALSE, TRUE),
101 IA64_HOWTO (R_IA64_DIR64MSB, "DIR64MSB", 4, FALSE, TRUE),
102 IA64_HOWTO (R_IA64_DIR64LSB, "DIR64LSB", 4, FALSE, TRUE),
103
104 IA64_HOWTO (R_IA64_GPREL22, "GPREL22", 0, FALSE, TRUE),
105 IA64_HOWTO (R_IA64_GPREL64I, "GPREL64I", 0, FALSE, TRUE),
106 IA64_HOWTO (R_IA64_GPREL32MSB, "GPREL32MSB", 2, FALSE, TRUE),
107 IA64_HOWTO (R_IA64_GPREL32LSB, "GPREL32LSB", 2, FALSE, TRUE),
108 IA64_HOWTO (R_IA64_GPREL64MSB, "GPREL64MSB", 4, FALSE, TRUE),
109 IA64_HOWTO (R_IA64_GPREL64LSB, "GPREL64LSB", 4, FALSE, TRUE),
110
111 IA64_HOWTO (R_IA64_LTOFF22, "LTOFF22", 0, FALSE, TRUE),
112 IA64_HOWTO (R_IA64_LTOFF64I, "LTOFF64I", 0, FALSE, TRUE),
113
114 IA64_HOWTO (R_IA64_PLTOFF22, "PLTOFF22", 0, FALSE, TRUE),
115 IA64_HOWTO (R_IA64_PLTOFF64I, "PLTOFF64I", 0, FALSE, TRUE),
116 IA64_HOWTO (R_IA64_PLTOFF64MSB, "PLTOFF64MSB", 4, FALSE, TRUE),
117 IA64_HOWTO (R_IA64_PLTOFF64LSB, "PLTOFF64LSB", 4, FALSE, TRUE),
118
119 IA64_HOWTO (R_IA64_FPTR64I, "FPTR64I", 0, FALSE, TRUE),
120 IA64_HOWTO (R_IA64_FPTR32MSB, "FPTR32MSB", 2, FALSE, TRUE),
121 IA64_HOWTO (R_IA64_FPTR32LSB, "FPTR32LSB", 2, FALSE, TRUE),
122 IA64_HOWTO (R_IA64_FPTR64MSB, "FPTR64MSB", 4, FALSE, TRUE),
123 IA64_HOWTO (R_IA64_FPTR64LSB, "FPTR64LSB", 4, FALSE, TRUE),
124
125 IA64_HOWTO (R_IA64_PCREL60B, "PCREL60B", 0, TRUE, TRUE),
126 IA64_HOWTO (R_IA64_PCREL21B, "PCREL21B", 0, TRUE, TRUE),
127 IA64_HOWTO (R_IA64_PCREL21M, "PCREL21M", 0, TRUE, TRUE),
128 IA64_HOWTO (R_IA64_PCREL21F, "PCREL21F", 0, TRUE, TRUE),
129 IA64_HOWTO (R_IA64_PCREL32MSB, "PCREL32MSB", 2, TRUE, TRUE),
130 IA64_HOWTO (R_IA64_PCREL32LSB, "PCREL32LSB", 2, TRUE, TRUE),
131 IA64_HOWTO (R_IA64_PCREL64MSB, "PCREL64MSB", 4, TRUE, TRUE),
132 IA64_HOWTO (R_IA64_PCREL64LSB, "PCREL64LSB", 4, TRUE, TRUE),
133
134 IA64_HOWTO (R_IA64_LTOFF_FPTR22, "LTOFF_FPTR22", 0, FALSE, TRUE),
135 IA64_HOWTO (R_IA64_LTOFF_FPTR64I, "LTOFF_FPTR64I", 0, FALSE, TRUE),
136 IA64_HOWTO (R_IA64_LTOFF_FPTR32MSB, "LTOFF_FPTR32MSB", 2, FALSE, TRUE),
137 IA64_HOWTO (R_IA64_LTOFF_FPTR32LSB, "LTOFF_FPTR32LSB", 2, FALSE, TRUE),
138 IA64_HOWTO (R_IA64_LTOFF_FPTR64MSB, "LTOFF_FPTR64MSB", 4, FALSE, TRUE),
139 IA64_HOWTO (R_IA64_LTOFF_FPTR64LSB, "LTOFF_FPTR64LSB", 4, FALSE, TRUE),
140
141 IA64_HOWTO (R_IA64_SEGREL32MSB, "SEGREL32MSB", 2, FALSE, TRUE),
142 IA64_HOWTO (R_IA64_SEGREL32LSB, "SEGREL32LSB", 2, FALSE, TRUE),
143 IA64_HOWTO (R_IA64_SEGREL64MSB, "SEGREL64MSB", 4, FALSE, TRUE),
144 IA64_HOWTO (R_IA64_SEGREL64LSB, "SEGREL64LSB", 4, FALSE, TRUE),
145
146 IA64_HOWTO (R_IA64_SECREL32MSB, "SECREL32MSB", 2, FALSE, TRUE),
147 IA64_HOWTO (R_IA64_SECREL32LSB, "SECREL32LSB", 2, FALSE, TRUE),
148 IA64_HOWTO (R_IA64_SECREL64MSB, "SECREL64MSB", 4, FALSE, TRUE),
149 IA64_HOWTO (R_IA64_SECREL64LSB, "SECREL64LSB", 4, FALSE, TRUE),
150
151 IA64_HOWTO (R_IA64_REL32MSB, "REL32MSB", 2, FALSE, TRUE),
152 IA64_HOWTO (R_IA64_REL32LSB, "REL32LSB", 2, FALSE, TRUE),
153 IA64_HOWTO (R_IA64_REL64MSB, "REL64MSB", 4, FALSE, TRUE),
154 IA64_HOWTO (R_IA64_REL64LSB, "REL64LSB", 4, FALSE, TRUE),
155
156 IA64_HOWTO (R_IA64_LTV32MSB, "LTV32MSB", 2, FALSE, TRUE),
157 IA64_HOWTO (R_IA64_LTV32LSB, "LTV32LSB", 2, FALSE, TRUE),
158 IA64_HOWTO (R_IA64_LTV64MSB, "LTV64MSB", 4, FALSE, TRUE),
159 IA64_HOWTO (R_IA64_LTV64LSB, "LTV64LSB", 4, FALSE, TRUE),
160
161 IA64_HOWTO (R_IA64_PCREL21BI, "PCREL21BI", 0, TRUE, TRUE),
162 IA64_HOWTO (R_IA64_PCREL22, "PCREL22", 0, TRUE, TRUE),
163 IA64_HOWTO (R_IA64_PCREL64I, "PCREL64I", 0, TRUE, TRUE),
164
165 IA64_HOWTO (R_IA64_IPLTMSB, "IPLTMSB", 4, FALSE, TRUE),
166 IA64_HOWTO (R_IA64_IPLTLSB, "IPLTLSB", 4, FALSE, TRUE),
167 IA64_HOWTO (R_IA64_COPY, "COPY", 4, FALSE, TRUE),
168 IA64_HOWTO (R_IA64_LTOFF22X, "LTOFF22X", 0, FALSE, TRUE),
169 IA64_HOWTO (R_IA64_LDXMOV, "LDXMOV", 0, FALSE, TRUE),
170
171 IA64_HOWTO (R_IA64_TPREL14, "TPREL14", 0, FALSE, FALSE),
172 IA64_HOWTO (R_IA64_TPREL22, "TPREL22", 0, FALSE, FALSE),
173 IA64_HOWTO (R_IA64_TPREL64I, "TPREL64I", 0, FALSE, FALSE),
1fc0d173
JJ
174 IA64_HOWTO (R_IA64_TPREL64MSB, "TPREL64MSB", 4, FALSE, FALSE),
175 IA64_HOWTO (R_IA64_TPREL64LSB, "TPREL64LSB", 4, FALSE, FALSE),
b34976b6
AM
176 IA64_HOWTO (R_IA64_LTOFF_TPREL22, "LTOFF_TPREL22", 0, FALSE, FALSE),
177
0ca3e455
JB
178 IA64_HOWTO (R_IA64_DTPMOD64MSB, "DTPMOD64MSB", 4, FALSE, FALSE),
179 IA64_HOWTO (R_IA64_DTPMOD64LSB, "DTPMOD64LSB", 4, FALSE, FALSE),
b34976b6
AM
180 IA64_HOWTO (R_IA64_LTOFF_DTPMOD22, "LTOFF_DTPMOD22", 0, FALSE, FALSE),
181
182 IA64_HOWTO (R_IA64_DTPREL14, "DTPREL14", 0, FALSE, FALSE),
183 IA64_HOWTO (R_IA64_DTPREL22, "DTPREL22", 0, FALSE, FALSE),
184 IA64_HOWTO (R_IA64_DTPREL64I, "DTPREL64I", 0, FALSE, FALSE),
1fc0d173
JJ
185 IA64_HOWTO (R_IA64_DTPREL32MSB, "DTPREL32MSB", 2, FALSE, FALSE),
186 IA64_HOWTO (R_IA64_DTPREL32LSB, "DTPREL32LSB", 2, FALSE, FALSE),
187 IA64_HOWTO (R_IA64_DTPREL64MSB, "DTPREL64MSB", 4, FALSE, FALSE),
188 IA64_HOWTO (R_IA64_DTPREL64LSB, "DTPREL64LSB", 4, FALSE, FALSE),
b34976b6 189 IA64_HOWTO (R_IA64_LTOFF_DTPREL22, "LTOFF_DTPREL22", 0, FALSE, FALSE),
800eeca4
JW
190 };
191
192static unsigned char elf_code_to_howto_index[R_IA64_MAX_RELOC_CODE + 1];
193
194/* Given a BFD reloc type, return the matching HOWTO structure. */
195
cbe79dfe
TG
196reloc_howto_type *
197ia64_elf_lookup_howto (unsigned int rtype)
800eeca4
JW
198{
199 static int inited = 0;
200 int i;
201
202 if (!inited)
203 {
204 inited = 1;
205
206 memset (elf_code_to_howto_index, 0xff, sizeof (elf_code_to_howto_index));
207 for (i = 0; i < NELEMS (ia64_howto_table); ++i)
208 elf_code_to_howto_index[ia64_howto_table[i].type] = i;
209 }
210
d0fb9a8d
JJ
211 if (rtype > R_IA64_MAX_RELOC_CODE)
212 return 0;
800eeca4
JW
213 i = elf_code_to_howto_index[rtype];
214 if (i >= NELEMS (ia64_howto_table))
215 return 0;
216 return ia64_howto_table + i;
217}
218
cbe79dfe
TG
219reloc_howto_type*
220ia64_elf_reloc_type_lookup (bfd *abfd ATTRIBUTE_UNUSED,
221 bfd_reloc_code_real_type bfd_code)
800eeca4
JW
222{
223 unsigned int rtype;
224
225 switch (bfd_code)
226 {
227 case BFD_RELOC_NONE: rtype = R_IA64_NONE; break;
228
229 case BFD_RELOC_IA64_IMM14: rtype = R_IA64_IMM14; break;
230 case BFD_RELOC_IA64_IMM22: rtype = R_IA64_IMM22; break;
231 case BFD_RELOC_IA64_IMM64: rtype = R_IA64_IMM64; break;
232
233 case BFD_RELOC_IA64_DIR32MSB: rtype = R_IA64_DIR32MSB; break;
234 case BFD_RELOC_IA64_DIR32LSB: rtype = R_IA64_DIR32LSB; break;
235 case BFD_RELOC_IA64_DIR64MSB: rtype = R_IA64_DIR64MSB; break;
236 case BFD_RELOC_IA64_DIR64LSB: rtype = R_IA64_DIR64LSB; break;
237
238 case BFD_RELOC_IA64_GPREL22: rtype = R_IA64_GPREL22; break;
239 case BFD_RELOC_IA64_GPREL64I: rtype = R_IA64_GPREL64I; break;
240 case BFD_RELOC_IA64_GPREL32MSB: rtype = R_IA64_GPREL32MSB; break;
241 case BFD_RELOC_IA64_GPREL32LSB: rtype = R_IA64_GPREL32LSB; break;
242 case BFD_RELOC_IA64_GPREL64MSB: rtype = R_IA64_GPREL64MSB; break;
243 case BFD_RELOC_IA64_GPREL64LSB: rtype = R_IA64_GPREL64LSB; break;
244
245 case BFD_RELOC_IA64_LTOFF22: rtype = R_IA64_LTOFF22; break;
246 case BFD_RELOC_IA64_LTOFF64I: rtype = R_IA64_LTOFF64I; break;
247
248 case BFD_RELOC_IA64_PLTOFF22: rtype = R_IA64_PLTOFF22; break;
249 case BFD_RELOC_IA64_PLTOFF64I: rtype = R_IA64_PLTOFF64I; break;
250 case BFD_RELOC_IA64_PLTOFF64MSB: rtype = R_IA64_PLTOFF64MSB; break;
251 case BFD_RELOC_IA64_PLTOFF64LSB: rtype = R_IA64_PLTOFF64LSB; break;
252 case BFD_RELOC_IA64_FPTR64I: rtype = R_IA64_FPTR64I; break;
253 case BFD_RELOC_IA64_FPTR32MSB: rtype = R_IA64_FPTR32MSB; break;
254 case BFD_RELOC_IA64_FPTR32LSB: rtype = R_IA64_FPTR32LSB; break;
255 case BFD_RELOC_IA64_FPTR64MSB: rtype = R_IA64_FPTR64MSB; break;
256 case BFD_RELOC_IA64_FPTR64LSB: rtype = R_IA64_FPTR64LSB; break;
257
258 case BFD_RELOC_IA64_PCREL21B: rtype = R_IA64_PCREL21B; break;
748abff6 259 case BFD_RELOC_IA64_PCREL21BI: rtype = R_IA64_PCREL21BI; break;
800eeca4
JW
260 case BFD_RELOC_IA64_PCREL21M: rtype = R_IA64_PCREL21M; break;
261 case BFD_RELOC_IA64_PCREL21F: rtype = R_IA64_PCREL21F; break;
748abff6
RH
262 case BFD_RELOC_IA64_PCREL22: rtype = R_IA64_PCREL22; break;
263 case BFD_RELOC_IA64_PCREL60B: rtype = R_IA64_PCREL60B; break;
264 case BFD_RELOC_IA64_PCREL64I: rtype = R_IA64_PCREL64I; break;
800eeca4
JW
265 case BFD_RELOC_IA64_PCREL32MSB: rtype = R_IA64_PCREL32MSB; break;
266 case BFD_RELOC_IA64_PCREL32LSB: rtype = R_IA64_PCREL32LSB; break;
267 case BFD_RELOC_IA64_PCREL64MSB: rtype = R_IA64_PCREL64MSB; break;
268 case BFD_RELOC_IA64_PCREL64LSB: rtype = R_IA64_PCREL64LSB; break;
269
270 case BFD_RELOC_IA64_LTOFF_FPTR22: rtype = R_IA64_LTOFF_FPTR22; break;
271 case BFD_RELOC_IA64_LTOFF_FPTR64I: rtype = R_IA64_LTOFF_FPTR64I; break;
a4bd8390
JW
272 case BFD_RELOC_IA64_LTOFF_FPTR32MSB: rtype = R_IA64_LTOFF_FPTR32MSB; break;
273 case BFD_RELOC_IA64_LTOFF_FPTR32LSB: rtype = R_IA64_LTOFF_FPTR32LSB; break;
800eeca4
JW
274 case BFD_RELOC_IA64_LTOFF_FPTR64MSB: rtype = R_IA64_LTOFF_FPTR64MSB; break;
275 case BFD_RELOC_IA64_LTOFF_FPTR64LSB: rtype = R_IA64_LTOFF_FPTR64LSB; break;
276
800eeca4
JW
277 case BFD_RELOC_IA64_SEGREL32MSB: rtype = R_IA64_SEGREL32MSB; break;
278 case BFD_RELOC_IA64_SEGREL32LSB: rtype = R_IA64_SEGREL32LSB; break;
279 case BFD_RELOC_IA64_SEGREL64MSB: rtype = R_IA64_SEGREL64MSB; break;
280 case BFD_RELOC_IA64_SEGREL64LSB: rtype = R_IA64_SEGREL64LSB; break;
281
282 case BFD_RELOC_IA64_SECREL32MSB: rtype = R_IA64_SECREL32MSB; break;
283 case BFD_RELOC_IA64_SECREL32LSB: rtype = R_IA64_SECREL32LSB; break;
284 case BFD_RELOC_IA64_SECREL64MSB: rtype = R_IA64_SECREL64MSB; break;
285 case BFD_RELOC_IA64_SECREL64LSB: rtype = R_IA64_SECREL64LSB; break;
286
287 case BFD_RELOC_IA64_REL32MSB: rtype = R_IA64_REL32MSB; break;
288 case BFD_RELOC_IA64_REL32LSB: rtype = R_IA64_REL32LSB; break;
289 case BFD_RELOC_IA64_REL64MSB: rtype = R_IA64_REL64MSB; break;
290 case BFD_RELOC_IA64_REL64LSB: rtype = R_IA64_REL64LSB; break;
291
292 case BFD_RELOC_IA64_LTV32MSB: rtype = R_IA64_LTV32MSB; break;
293 case BFD_RELOC_IA64_LTV32LSB: rtype = R_IA64_LTV32LSB; break;
294 case BFD_RELOC_IA64_LTV64MSB: rtype = R_IA64_LTV64MSB; break;
295 case BFD_RELOC_IA64_LTV64LSB: rtype = R_IA64_LTV64LSB; break;
296
297 case BFD_RELOC_IA64_IPLTMSB: rtype = R_IA64_IPLTMSB; break;
298 case BFD_RELOC_IA64_IPLTLSB: rtype = R_IA64_IPLTLSB; break;
800eeca4
JW
299 case BFD_RELOC_IA64_COPY: rtype = R_IA64_COPY; break;
300 case BFD_RELOC_IA64_LTOFF22X: rtype = R_IA64_LTOFF22X; break;
301 case BFD_RELOC_IA64_LDXMOV: rtype = R_IA64_LDXMOV; break;
302
13ae64f3 303 case BFD_RELOC_IA64_TPREL14: rtype = R_IA64_TPREL14; break;
800eeca4 304 case BFD_RELOC_IA64_TPREL22: rtype = R_IA64_TPREL22; break;
13ae64f3 305 case BFD_RELOC_IA64_TPREL64I: rtype = R_IA64_TPREL64I; break;
800eeca4
JW
306 case BFD_RELOC_IA64_TPREL64MSB: rtype = R_IA64_TPREL64MSB; break;
307 case BFD_RELOC_IA64_TPREL64LSB: rtype = R_IA64_TPREL64LSB; break;
13ae64f3
JJ
308 case BFD_RELOC_IA64_LTOFF_TPREL22: rtype = R_IA64_LTOFF_TPREL22; break;
309
310 case BFD_RELOC_IA64_DTPMOD64MSB: rtype = R_IA64_DTPMOD64MSB; break;
311 case BFD_RELOC_IA64_DTPMOD64LSB: rtype = R_IA64_DTPMOD64LSB; break;
312 case BFD_RELOC_IA64_LTOFF_DTPMOD22: rtype = R_IA64_LTOFF_DTPMOD22; break;
313
314 case BFD_RELOC_IA64_DTPREL14: rtype = R_IA64_DTPREL14; break;
315 case BFD_RELOC_IA64_DTPREL22: rtype = R_IA64_DTPREL22; break;
316 case BFD_RELOC_IA64_DTPREL64I: rtype = R_IA64_DTPREL64I; break;
317 case BFD_RELOC_IA64_DTPREL32MSB: rtype = R_IA64_DTPREL32MSB; break;
318 case BFD_RELOC_IA64_DTPREL32LSB: rtype = R_IA64_DTPREL32LSB; break;
319 case BFD_RELOC_IA64_DTPREL64MSB: rtype = R_IA64_DTPREL64MSB; break;
320 case BFD_RELOC_IA64_DTPREL64LSB: rtype = R_IA64_DTPREL64LSB; break;
321 case BFD_RELOC_IA64_LTOFF_DTPREL22: rtype = R_IA64_LTOFF_DTPREL22; break;
800eeca4
JW
322
323 default: return 0;
324 }
cbe79dfe 325 return ia64_elf_lookup_howto (rtype);
800eeca4
JW
326}
327
cbe79dfe
TG
328reloc_howto_type *
329ia64_elf_reloc_name_lookup (bfd *abfd ATTRIBUTE_UNUSED,
330 const char *r_name)
157090f7
AM
331{
332 unsigned int i;
333
334 for (i = 0;
335 i < sizeof (ia64_howto_table) / sizeof (ia64_howto_table[0]);
336 i++)
337 if (ia64_howto_table[i].name != NULL
338 && strcasecmp (ia64_howto_table[i].name, r_name) == 0)
339 return &ia64_howto_table[i];
340
341 return NULL;
342}
343
83b6bd86
L
344#define BTYPE_SHIFT 6
345#define Y_SHIFT 26
346#define X6_SHIFT 27
347#define X4_SHIFT 27
348#define X3_SHIFT 33
349#define X2_SHIFT 31
350#define X_SHIFT 33
351#define OPCODE_SHIFT 37
352
353#define OPCODE_BITS (0xfLL << OPCODE_SHIFT)
354#define X6_BITS (0x3fLL << X6_SHIFT)
355#define X4_BITS (0xfLL << X4_SHIFT)
356#define X3_BITS (0x7LL << X3_SHIFT)
357#define X2_BITS (0x3LL << X2_SHIFT)
358#define X_BITS (0x1LL << X_SHIFT)
359#define Y_BITS (0x1LL << Y_SHIFT)
360#define BTYPE_BITS (0x7LL << BTYPE_SHIFT)
361#define PREDICATE_BITS (0x3fLL)
362
363#define IS_NOP_B(i) \
364 (((i) & (OPCODE_BITS | X6_BITS)) == (2LL << OPCODE_SHIFT))
365#define IS_NOP_F(i) \
366 (((i) & (OPCODE_BITS | X_BITS | X6_BITS | Y_BITS)) \
367 == (0x1LL << X6_SHIFT))
368#define IS_NOP_I(i) \
369 (((i) & (OPCODE_BITS | X3_BITS | X6_BITS | Y_BITS)) \
370 == (0x1LL << X6_SHIFT))
371#define IS_NOP_M(i) \
372 (((i) & (OPCODE_BITS | X3_BITS | X2_BITS | X4_BITS | Y_BITS)) \
373 == (0x1LL << X4_SHIFT))
374#define IS_BR_COND(i) \
375 (((i) & (OPCODE_BITS | BTYPE_BITS)) == (0x4LL << OPCODE_SHIFT))
376#define IS_BR_CALL(i) \
377 (((i) & OPCODE_BITS) == (0x5LL << OPCODE_SHIFT))
378
cbe79dfe
TG
379bfd_boolean
380ia64_elf_relax_br (bfd_byte *contents, bfd_vma off)
83b6bd86 381{
d3ce72d0 382 unsigned int template_val, mlx;
83b6bd86
L
383 bfd_vma t0, t1, s0, s1, s2, br_code;
384 long br_slot;
385 bfd_byte *hit_addr;
386
387 hit_addr = (bfd_byte *) (contents + off);
d68cd58c 388 br_slot = (intptr_t) hit_addr & 0x3;
83b6bd86
L
389 hit_addr -= br_slot;
390 t0 = bfd_getl64 (hit_addr + 0);
391 t1 = bfd_getl64 (hit_addr + 8);
392
393 /* Check if we can turn br into brl. A label is always at the start
394 of the bundle. Even if there are predicates on NOPs, we still
395 perform this optimization. */
d3ce72d0 396 template_val = t0 & 0x1e;
83b6bd86
L
397 s0 = (t0 >> 5) & 0x1ffffffffffLL;
398 s1 = ((t0 >> 46) | (t1 << 18)) & 0x1ffffffffffLL;
399 s2 = (t1 >> 23) & 0x1ffffffffffLL;
400 switch (br_slot)
401 {
402 case 0:
403 /* Check if slot 1 and slot 2 are NOPs. Possible template is
404 BBB. We only need to check nop.b. */
405 if (!(IS_NOP_B (s1) && IS_NOP_B (s2)))
406 return FALSE;
407 br_code = s0;
408 break;
409 case 1:
410 /* Check if slot 2 is NOP. Possible templates are MBB and BBB.
411 For BBB, slot 0 also has to be nop.b. */
d3ce72d0 412 if (!((template_val == 0x12 /* MBB */
83b6bd86 413 && IS_NOP_B (s2))
d3ce72d0 414 || (template_val == 0x16 /* BBB */
83b6bd86
L
415 && IS_NOP_B (s0)
416 && IS_NOP_B (s2))))
417 return FALSE;
418 br_code = s1;
419 break;
420 case 2:
421 /* Check if slot 1 is NOP. Possible templates are MIB, MBB, BBB,
422 MMB and MFB. For BBB, slot 0 also has to be nop.b. */
d3ce72d0 423 if (!((template_val == 0x10 /* MIB */
83b6bd86 424 && IS_NOP_I (s1))
d3ce72d0 425 || (template_val == 0x12 /* MBB */
83b6bd86 426 && IS_NOP_B (s1))
d3ce72d0 427 || (template_val == 0x16 /* BBB */
83b6bd86
L
428 && IS_NOP_B (s0)
429 && IS_NOP_B (s1))
d3ce72d0 430 || (template_val == 0x18 /* MMB */
83b6bd86 431 && IS_NOP_M (s1))
d3ce72d0 432 || (template_val == 0x1c /* MFB */
83b6bd86
L
433 && IS_NOP_F (s1))))
434 return FALSE;
435 br_code = s2;
436 break;
437 default:
438 /* It should never happen. */
439 abort ();
440 }
9a2e389a 441
83b6bd86
L
442 /* We can turn br.cond/br.call into brl.cond/brl.call. */
443 if (!(IS_BR_COND (br_code) || IS_BR_CALL (br_code)))
444 return FALSE;
445
446 /* Turn br into brl by setting bit 40. */
447 br_code |= 0x1LL << 40;
448
449 /* Turn the old bundle into a MLX bundle with the same stop-bit
450 variety. */
451 if (t0 & 0x1)
452 mlx = 0x5;
453 else
454 mlx = 0x4;
455
d3ce72d0 456 if (template_val == 0x16)
83b6bd86 457 {
5e27d427
L
458 /* For BBB, we need to put nop.m in slot 0. We keep the original
459 predicate only if slot 0 isn't br. */
460 if (br_slot == 0)
461 t0 = 0LL;
462 else
463 t0 &= PREDICATE_BITS << 5;
83b6bd86
L
464 t0 |= 0x1LL << (X4_SHIFT + 5);
465 }
466 else
467 {
468 /* Keep the original instruction in slot 0. */
469 t0 &= 0x1ffffffffffLL << 5;
470 }
471
472 t0 |= mlx;
473
474 /* Put brl in slot 1. */
475 t1 = br_code << 23;
476
477 bfd_putl64 (t0, hit_addr);
478 bfd_putl64 (t1, hit_addr + 8);
479 return TRUE;
480}
481
cbe79dfe
TG
482void
483ia64_elf_relax_brl (bfd_byte *contents, bfd_vma off)
03609792 484{
d3ce72d0 485 int template_val;
03609792 486 bfd_byte *hit_addr;
fc3ab699 487 bfd_vma t0, t1, i0, i1, i2;
03609792
L
488
489 hit_addr = (bfd_byte *) (contents + off);
d68cd58c 490 hit_addr -= (intptr_t) hit_addr & 0x3;
fc3ab699
L
491 t0 = bfd_getl64 (hit_addr);
492 t1 = bfd_getl64 (hit_addr + 8);
03609792 493
7e3102a7 494 /* Keep the instruction in slot 0. */
fc3ab699
L
495 i0 = (t0 >> 5) & 0x1ffffffffffLL;
496 /* Use nop.b for slot 1. */
497 i1 = 0x4000000000LL;
7e3102a7 498 /* For slot 2, turn brl into br by masking out bit 40. */
fc3ab699 499 i2 = (t1 >> 23) & 0x0ffffffffffLL;
7e3102a7 500
fc3ab699
L
501 /* Turn a MLX bundle into a MBB bundle with the same stop-bit
502 variety. */
503 if (t0 & 0x1)
d3ce72d0 504 template_val = 0x13;
fc3ab699 505 else
d3ce72d0
NC
506 template_val = 0x12;
507 t0 = (i1 << 46) | (i0 << 5) | template_val;
fc3ab699 508 t1 = (i2 << 23) | (i1 >> 18);
7e3102a7 509
fc3ab699
L
510 bfd_putl64 (t0, hit_addr);
511 bfd_putl64 (t1, hit_addr + 8);
03609792 512}
fbbc3759 513
cbe79dfe
TG
514void
515ia64_elf_relax_ldxmov (bfd_byte *contents, bfd_vma off)
2c4c2bc0
RH
516{
517 int shift, r1, r3;
518 bfd_vma dword, insn;
519
520 switch ((int)off & 0x3)
521 {
522 case 0: shift = 5; break;
523 case 1: shift = 14; off += 3; break;
524 case 2: shift = 23; off += 6; break;
60d8b524 525 default:
2c4c2bc0
RH
526 abort ();
527 }
528
bbb268c3 529 dword = bfd_getl64 (contents + off);
2c4c2bc0
RH
530 insn = (dword >> shift) & 0x1ffffffffffLL;
531
532 r1 = (insn >> 6) & 127;
533 r3 = (insn >> 20) & 127;
534 if (r1 == r3)
535 insn = 0x8000000; /* nop */
536 else
537 insn = (insn & 0x7f01fff) | 0x10800000000LL; /* (qp) mov r1 = r3 */
538
539 dword &= ~(0x1ffffffffffLL << shift);
540 dword |= (insn << shift);
bbb268c3 541 bfd_putl64 (dword, contents + off);
2c4c2bc0 542}
800eeca4 543\f
cbe79dfe
TG
544bfd_reloc_status_type
545ia64_elf_install_value (bfd_byte *hit_addr, bfd_vma v, unsigned int r_type)
81545d45 546{
cbe79dfe
TG
547 const struct ia64_operand *op;
548 int bigendian = 0, shift = 0;
549 bfd_vma t0, t1, dword;
550 ia64_insn insn;
551 enum ia64_opnd opnd;
552 const char *err;
553 size_t size = 8;
554#ifdef BFD_HOST_U_64_BIT
555 BFD_HOST_U_64_BIT val = (BFD_HOST_U_64_BIT) v;
556#else
557 bfd_vma val = v;
558#endif
d9cf1b54 559
cbe79dfe
TG
560 opnd = IA64_OPND_NIL;
561 switch (r_type)
562 {
563 case R_IA64_NONE:
564 case R_IA64_LDXMOV:
565 return bfd_reloc_ok;
81545d45 566
cbe79dfe 567 /* Instruction relocations. */
800eeca4 568
cbe79dfe
TG
569 case R_IA64_IMM14:
570 case R_IA64_TPREL14:
571 case R_IA64_DTPREL14:
572 opnd = IA64_OPND_IMM14;
800eeca4
JW
573 break;
574
cbe79dfe
TG
575 case R_IA64_PCREL21F: opnd = IA64_OPND_TGT25; break;
576 case R_IA64_PCREL21M: opnd = IA64_OPND_TGT25b; break;
577 case R_IA64_PCREL60B: opnd = IA64_OPND_TGT64; break;
578 case R_IA64_PCREL21B:
579 case R_IA64_PCREL21BI:
580 opnd = IA64_OPND_TGT25c;
800eeca4
JW
581 break;
582
cbe79dfe
TG
583 case R_IA64_IMM22:
584 case R_IA64_GPREL22:
585 case R_IA64_LTOFF22:
586 case R_IA64_LTOFF22X:
587 case R_IA64_PLTOFF22:
588 case R_IA64_PCREL22:
589 case R_IA64_LTOFF_FPTR22:
590 case R_IA64_TPREL22:
591 case R_IA64_DTPREL22:
592 case R_IA64_LTOFF_TPREL22:
593 case R_IA64_LTOFF_DTPMOD22:
594 case R_IA64_LTOFF_DTPREL22:
595 opnd = IA64_OPND_IMM22;
596 break;
800eeca4 597
cbe79dfe
TG
598 case R_IA64_IMM64:
599 case R_IA64_GPREL64I:
600 case R_IA64_LTOFF64I:
601 case R_IA64_PLTOFF64I:
602 case R_IA64_PCREL64I:
603 case R_IA64_FPTR64I:
604 case R_IA64_LTOFF_FPTR64I:
605 case R_IA64_TPREL64I:
606 case R_IA64_DTPREL64I:
607 opnd = IA64_OPND_IMMU64;
608 break;
800eeca4 609
cbe79dfe 610 /* Data relocations. */
b59dd4a5 611
cbe79dfe
TG
612 case R_IA64_DIR32MSB:
613 case R_IA64_GPREL32MSB:
614 case R_IA64_FPTR32MSB:
615 case R_IA64_PCREL32MSB:
616 case R_IA64_LTOFF_FPTR32MSB:
617 case R_IA64_SEGREL32MSB:
618 case R_IA64_SECREL32MSB:
619 case R_IA64_LTV32MSB:
620 case R_IA64_DTPREL32MSB:
621 size = 4; bigendian = 1;
622 break;
b59dd4a5 623
cbe79dfe
TG
624 case R_IA64_DIR32LSB:
625 case R_IA64_GPREL32LSB:
626 case R_IA64_FPTR32LSB:
627 case R_IA64_PCREL32LSB:
628 case R_IA64_LTOFF_FPTR32LSB:
629 case R_IA64_SEGREL32LSB:
630 case R_IA64_SECREL32LSB:
631 case R_IA64_LTV32LSB:
632 case R_IA64_DTPREL32LSB:
633 size = 4; bigendian = 0;
b59dd4a5 634 break;
b59dd4a5 635
cbe79dfe
TG
636 case R_IA64_DIR64MSB:
637 case R_IA64_GPREL64MSB:
638 case R_IA64_PLTOFF64MSB:
639 case R_IA64_FPTR64MSB:
640 case R_IA64_PCREL64MSB:
641 case R_IA64_LTOFF_FPTR64MSB:
642 case R_IA64_SEGREL64MSB:
643 case R_IA64_SECREL64MSB:
644 case R_IA64_LTV64MSB:
645 case R_IA64_TPREL64MSB:
646 case R_IA64_DTPMOD64MSB:
647 case R_IA64_DTPREL64MSB:
648 size = 8; bigendian = 1;
649 break;
7256a114 650
cbe79dfe
TG
651 case R_IA64_DIR64LSB:
652 case R_IA64_GPREL64LSB:
653 case R_IA64_PLTOFF64LSB:
654 case R_IA64_FPTR64LSB:
655 case R_IA64_PCREL64LSB:
656 case R_IA64_LTOFF_FPTR64LSB:
657 case R_IA64_SEGREL64LSB:
658 case R_IA64_SECREL64LSB:
659 case R_IA64_LTV64LSB:
660 case R_IA64_TPREL64LSB:
661 case R_IA64_DTPMOD64LSB:
662 case R_IA64_DTPREL64LSB:
663 size = 8; bigendian = 0;
01e1a5bc
NC
664 break;
665
cbe79dfe 666 /* Unsupported / Dynamic relocations. */
01e1a5bc 667 default:
cbe79dfe 668 return bfd_reloc_notsupported;
01e1a5bc 669 }
01e1a5bc 670
cbe79dfe 671 switch (opnd)
01e1a5bc 672 {
cbe79dfe
TG
673 case IA64_OPND_IMMU64:
674 hit_addr -= (intptr_t) hit_addr & 0x3;
675 t0 = bfd_getl64 (hit_addr);
676 t1 = bfd_getl64 (hit_addr + 8);
01e1a5bc 677
cbe79dfe
TG
678 /* tmpl/s: bits 0.. 5 in t0
679 slot 0: bits 5..45 in t0
680 slot 1: bits 46..63 in t0, bits 0..22 in t1
681 slot 2: bits 23..63 in t1 */
01e1a5bc 682
cbe79dfe
TG
683 /* First, clear the bits that form the 64 bit constant. */
684 t0 &= ~(0x3ffffLL << 46);
685 t1 &= ~(0x7fffffLL
686 | (( (0x07fLL << 13) | (0x1ffLL << 27)
687 | (0x01fLL << 22) | (0x001LL << 21)
688 | (0x001LL << 36)) << 23));
01e1a5bc 689
cbe79dfe
TG
690 t0 |= ((val >> 22) & 0x03ffffLL) << 46; /* 18 lsbs of imm41 */
691 t1 |= ((val >> 40) & 0x7fffffLL) << 0; /* 23 msbs of imm41 */
692 t1 |= ( (((val >> 0) & 0x07f) << 13) /* imm7b */
693 | (((val >> 7) & 0x1ff) << 27) /* imm9d */
694 | (((val >> 16) & 0x01f) << 22) /* imm5c */
695 | (((val >> 21) & 0x001) << 21) /* ic */
696 | (((val >> 63) & 0x001) << 36)) << 23; /* i */
01e1a5bc 697
cbe79dfe
TG
698 bfd_putl64 (t0, hit_addr);
699 bfd_putl64 (t1, hit_addr + 8);
700 break;
01e1a5bc 701
cbe79dfe
TG
702 case IA64_OPND_TGT64:
703 hit_addr -= (intptr_t) hit_addr & 0x3;
704 t0 = bfd_getl64 (hit_addr);
705 t1 = bfd_getl64 (hit_addr + 8);
01e1a5bc 706
cbe79dfe
TG
707 /* tmpl/s: bits 0.. 5 in t0
708 slot 0: bits 5..45 in t0
709 slot 1: bits 46..63 in t0, bits 0..22 in t1
710 slot 2: bits 23..63 in t1 */
01e1a5bc 711
cbe79dfe
TG
712 /* First, clear the bits that form the 64 bit constant. */
713 t0 &= ~(0x3ffffLL << 46);
714 t1 &= ~(0x7fffffLL
715 | ((1LL << 36 | 0xfffffLL << 13) << 23));
01e1a5bc 716
cbe79dfe
TG
717 val >>= 4;
718 t0 |= ((val >> 20) & 0xffffLL) << 2 << 46; /* 16 lsbs of imm39 */
719 t1 |= ((val >> 36) & 0x7fffffLL) << 0; /* 23 msbs of imm39 */
720 t1 |= ((((val >> 0) & 0xfffffLL) << 13) /* imm20b */
721 | (((val >> 59) & 0x1LL) << 36)) << 23; /* i */
01e1a5bc 722
cbe79dfe
TG
723 bfd_putl64 (t0, hit_addr);
724 bfd_putl64 (t1, hit_addr + 8);
725 break;
01e1a5bc 726
cbe79dfe
TG
727 default:
728 switch ((intptr_t) hit_addr & 0x3)
01e1a5bc 729 {
cbe79dfe
TG
730 case 0: shift = 5; break;
731 case 1: shift = 14; hit_addr += 3; break;
732 case 2: shift = 23; hit_addr += 6; break;
733 case 3: return bfd_reloc_notsupported; /* shouldn't happen... */
01e1a5bc 734 }
cbe79dfe
TG
735 dword = bfd_getl64 (hit_addr);
736 insn = (dword >> shift) & 0x1ffffffffffLL;
01e1a5bc 737
cbe79dfe
TG
738 op = elf64_ia64_operands + opnd;
739 err = (*op->insert) (op, val, &insn);
740 if (err)
741 return bfd_reloc_overflow;
01e1a5bc 742
cbe79dfe
TG
743 dword &= ~(0x1ffffffffffLL << shift);
744 dword |= (insn << shift);
745 bfd_putl64 (dword, hit_addr);
746 break;
01e1a5bc 747
cbe79dfe
TG
748 case IA64_OPND_NIL:
749 /* A data relocation. */
750 if (bigendian)
751 if (size == 4)
752 bfd_putb32 (val, hit_addr);
753 else
754 bfd_putb64 (val, hit_addr);
755 else
756 if (size == 4)
757 bfd_putl32 (val, hit_addr);
758 else
759 bfd_putl64 (val, hit_addr);
760 break;
01e1a5bc
NC
761 }
762
cbe79dfe 763 return bfd_reloc_ok;
01e1a5bc 764}
This page took 1.334258 seconds and 4 git commands to generate.