1 /* tc-microblaze.c -- Assemble code for Xilinx MicroBlaze
2
3 Copyright (C) 2009-2014 Free Software Foundation, Inc.
4
5 This file is part of GAS, the GNU Assembler.
6
7 GAS is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3, or (at your option)
10 any later version.
11
12 GAS 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.
16
17 You should have received a copy of the GNU General Public License
18 along with GAS; see the file COPYING. If not, write to the Free
19 Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
20 02110-1301, USA. */
21
22 #include "as.h"
23 #include <stdio.h>
24 #include "bfd.h"
25 #include "subsegs.h"
26 #define DEFINE_TABLE
27 #include "../opcodes/microblaze-opc.h"
28 #include "../opcodes/microblaze-opcm.h"
29 #include "safe-ctype.h"
30 #include <string.h>
31 #include <dwarf2dbg.h>
32 #include "aout/stab_gnu.h"
33
34 #ifndef streq
35 #define streq(a,b) (strcmp (a, b) == 0)
36 #endif
37
38 #define OPTION_EB (OPTION_MD_BASE + 0)
39 #define OPTION_EL (OPTION_MD_BASE + 1)
40
41 void microblaze_generate_symbol (char *sym);
42 static bfd_boolean check_spl_reg (unsigned *);
43
44 /* Several places in this file insert raw instructions into the
45 object. They should generate the instruction
46 and then use these four macros to crack the instruction value into
47 the appropriate byte values. */
48 #define INST_BYTE0(x) (target_big_endian ? (((x) >> 24) & 0xFF) : ((x) & 0xFF))
49 #define INST_BYTE1(x) (target_big_endian ? (((x) >> 16) & 0xFF) : (((x) >> 8) & 0xFF))
50 #define INST_BYTE2(x) (target_big_endian ? (((x) >> 8) & 0xFF) : (((x) >> 16) & 0xFF))
51 #define INST_BYTE3(x) (target_big_endian ? ((x) & 0xFF) : (((x) >> 24) & 0xFF))
52
53 /* This array holds the chars that always start a comment. If the
54 pre-processor is disabled, these aren't very useful. */
55 const char comment_chars[] = "#";
56
57 const char line_separator_chars[] = ";";
58
59 /* This array holds the chars that only start a comment at the beginning of
60 a line. */
61 const char line_comment_chars[] = "#";
62
63 const int md_reloc_size = 8; /* Size of relocation record. */
64
65 /* Chars that can be used to separate mant
66 from exp in floating point numbers. */
67 const char EXP_CHARS[] = "eE";
68
69 /* Chars that mean this number is a floating point constant
70 As in 0f12.456
71 or 0d1.2345e12. */
72 const char FLT_CHARS[] = "rRsSfFdDxXpP";
73
74 /* INST_PC_OFFSET and INST_NO_OFFSET are 0 and 1. */
75 #define UNDEFINED_PC_OFFSET 2
76 #define DEFINED_ABS_SEGMENT 3
77 #define DEFINED_PC_OFFSET 4
78 #define DEFINED_RO_SEGMENT 5
79 #define DEFINED_RW_SEGMENT 6
80 #define LARGE_DEFINED_PC_OFFSET 7
81 #define GOT_OFFSET 8
82 #define PLT_OFFSET 9
83 #define GOTOFF_OFFSET 10
84 #define TLSGD_OFFSET 11
85 #define TLSLD_OFFSET 12
86 #define TLSDTPMOD_OFFSET 13
87 #define TLSDTPREL_OFFSET 14
88 #define TLSGOTTPREL_OFFSET 15
89 #define TLSTPREL_OFFSET 16
90
91 /* Initialize the relax table. */
92 const relax_typeS md_relax_table[] =
93 {
94 { 1, 1, 0, 0 }, /* 0: Unused. */
95 { 1, 1, 0, 0 }, /* 1: Unused. */
96 { 1, 1, 0, 0 }, /* 2: Unused. */
97 { 1, 1, 0, 0 }, /* 3: Unused. */
98 { 32767, -32768, INST_WORD_SIZE, LARGE_DEFINED_PC_OFFSET }, /* 4: DEFINED_PC_OFFSET. */
99 { 1, 1, 0, 0 }, /* 5: Unused. */
100 { 1, 1, 0, 0 }, /* 6: Unused. */
101 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 }, /* 7: LARGE_DEFINED_PC_OFFSET. */
102 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 }, /* 8: GOT_OFFSET. */
103 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 }, /* 9: PLT_OFFSET. */
104 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 }, /* 10: GOTOFF_OFFSET. */
105 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 }, /* 11: TLSGD_OFFSET. */
106 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 }, /* 12: TLSLD_OFFSET. */
107 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*1, 0 }, /* 13: TLSDTPMOD_OFFSET. */
108 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 }, /* 14: TLSDTPREL_OFFSET. */
109 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 }, /* 15: TLSGOTTPREL_OFFSET. */
110 { 0x7fffffff, 0x80000000, INST_WORD_SIZE*2, 0 } /* 16: TLSTPREL_OFFSET. */
111 };
112
113 static struct hash_control * opcode_hash_control; /* Opcode mnemonics. */
114
115 static segT sbss_segment = 0; /* Small bss section. */
116 static segT sbss2_segment = 0; /* Section not used. */
117 static segT sdata_segment = 0; /* Small data section. */
118 static segT sdata2_segment = 0; /* Small read-only section. */
119 static segT rodata_segment = 0; /* read-only section. */
120
121 /* Generate a symbol for stabs information. */
122
123 void
microblaze_generate_symbol(char * sym)124 microblaze_generate_symbol (char *sym)
125 {
126 #define MICROBLAZE_FAKE_LABEL_NAME "XL0\001"
127 static int microblaze_label_count;
128 sprintf (sym, "%sL%d", MICROBLAZE_FAKE_LABEL_NAME, microblaze_label_count);
129 ++microblaze_label_count;
130 }
131
132 /* Handle the section changing pseudo-ops. */
133
134 static void
microblaze_s_text(int ignore ATTRIBUTE_UNUSED)135 microblaze_s_text (int ignore ATTRIBUTE_UNUSED)
136 {
137 #ifdef OBJ_ELF
138 obj_elf_text (ignore);
139 #else
140 s_text (ignore);
141 #endif
142 }
143
144 static void
microblaze_s_data(int ignore ATTRIBUTE_UNUSED)145 microblaze_s_data (int ignore ATTRIBUTE_UNUSED)
146 {
147 #ifdef OBJ_ELF
148 obj_elf_change_section (".data", SHT_PROGBITS, SHF_ALLOC+SHF_WRITE, 0, 0, 0, 0);
149 #else
150 s_data (ignore);
151 #endif
152 }
153
154 /* Things in the .sdata segment are always considered to be in the small data section. */
155
156 static void
microblaze_s_sdata(int ignore ATTRIBUTE_UNUSED)157 microblaze_s_sdata (int ignore ATTRIBUTE_UNUSED)
158 {
159 #ifdef OBJ_ELF
160 obj_elf_change_section (".sdata", SHT_PROGBITS, SHF_ALLOC+SHF_WRITE, 0, 0, 0, 0);
161 #else
162 s_data (ignore);
163 #endif
164 }
165
166 /* Pseudo op to make file scope bss items. */
167
168 static void
microblaze_s_lcomm(int xxx ATTRIBUTE_UNUSED)169 microblaze_s_lcomm (int xxx ATTRIBUTE_UNUSED)
170 {
171 char *name;
172 char c;
173 char *p;
174 offsetT size;
175 symbolS *symbolP;
176 offsetT align;
177 char *pfrag;
178 int align2;
179 segT current_seg = now_seg;
180 subsegT current_subseg = now_subseg;
181
182 name = input_line_pointer;
183 c = get_symbol_end ();
184
185 /* Just after name is now '\0'. */
186 p = input_line_pointer;
187 *p = c;
188 SKIP_WHITESPACE ();
189 if (*input_line_pointer != ',')
190 {
191 as_bad (_("Expected comma after symbol-name: rest of line ignored."));
192 ignore_rest_of_line ();
193 return;
194 }
195
196 input_line_pointer++; /* skip ',' */
197 if ((size = get_absolute_expression ()) < 0)
198 {
199 as_warn (_(".COMMon length (%ld.) <0! Ignored."), (long) size);
200 ignore_rest_of_line ();
201 return;
202 }
203
204 /* The third argument to .lcomm is the alignment. */
205 if (*input_line_pointer != ',')
206 align = 8;
207 else
208 {
209 ++input_line_pointer;
210 align = get_absolute_expression ();
211 if (align <= 0)
212 {
213 as_warn (_("ignoring bad alignment"));
214 align = 8;
215 }
216 }
217
218 *p = 0;
219 symbolP = symbol_find_or_make (name);
220 *p = c;
221
222 if (S_IS_DEFINED (symbolP) && ! S_IS_COMMON (symbolP))
223 {
224 as_bad (_("Ignoring attempt to re-define symbol `%s'."),
225 S_GET_NAME (symbolP));
226 ignore_rest_of_line ();
227 return;
228 }
229
230 if (S_GET_VALUE (symbolP) && S_GET_VALUE (symbolP) != (valueT) size)
231 {
232 as_bad (_("Length of .lcomm \"%s\" is already %ld. Not changed to %ld."),
233 S_GET_NAME (symbolP),
234 (long) S_GET_VALUE (symbolP),
235 (long) size);
236
237 ignore_rest_of_line ();
238 return;
239 }
240
241 /* Allocate_bss. */
242 if (align)
243 {
244 /* Convert to a power of 2 alignment. */
245 for (align2 = 0; (align & 1) == 0; align >>= 1, ++align2);
246 if (align != 1)
247 {
248 as_bad (_("Common alignment not a power of 2"));
249 ignore_rest_of_line ();
250 return;
251 }
252 }
253 else
254 align2 = 0;
255
256 record_alignment (current_seg, align2);
257 subseg_set (current_seg, current_subseg);
258 if (align2)
259 frag_align (align2, 0, 0);
260 if (S_GET_SEGMENT (symbolP) == current_seg)
261 symbol_get_frag (symbolP)->fr_symbol = 0;
262 symbol_set_frag (symbolP, frag_now);
263 pfrag = frag_var (rs_org, 1, 1, (relax_substateT) 0, symbolP, size,
264 (char *) 0);
265 *pfrag = 0;
266 S_SET_SIZE (symbolP, size);
267 S_SET_SEGMENT (symbolP, current_seg);
268 subseg_set (current_seg, current_subseg);
269 demand_empty_rest_of_line ();
270 }
271
272 static void
microblaze_s_rdata(int localvar)273 microblaze_s_rdata (int localvar)
274 {
275 #ifdef OBJ_ELF
276 if (localvar == 0)
277 {
278 /* rodata. */
279 obj_elf_change_section (".rodata", SHT_PROGBITS, SHF_ALLOC, 0, 0, 0, 0);
280 if (rodata_segment == 0)
281 rodata_segment = subseg_new (".rodata", 0);
282 }
283 else
284 {
285 /* 1 .sdata2. */
286 obj_elf_change_section (".sdata2", SHT_PROGBITS, SHF_ALLOC, 0, 0, 0, 0);
287 }
288 #else
289 s_data (ignore);
290 #endif
291 }
292
293 static void
microblaze_s_bss(int localvar)294 microblaze_s_bss (int localvar)
295 {
296 #ifdef OBJ_ELF
297 if (localvar == 0) /* bss. */
298 obj_elf_change_section (".bss", SHT_NOBITS, SHF_ALLOC+SHF_WRITE, 0, 0, 0, 0);
299 else if (localvar == 1)
300 {
301 /* sbss. */
302 obj_elf_change_section (".sbss", SHT_NOBITS, SHF_ALLOC+SHF_WRITE, 0, 0, 0, 0);
303 if (sbss_segment == 0)
304 sbss_segment = subseg_new (".sbss", 0);
305 }
306 #else
307 s_data (ignore);
308 #endif
309 }
310
311 /* endp_p is always 1 as this func is called only for .end <funcname>
312 This func consumes the <funcname> and calls regular processing
313 s_func(1) with arg 1 (1 for end). */
314
315 static void
microblaze_s_func(int end_p ATTRIBUTE_UNUSED)316 microblaze_s_func (int end_p ATTRIBUTE_UNUSED)
317 {
318 *input_line_pointer = get_symbol_end ();
319 s_func (1);
320 }
321
322 /* Handle the .weakext pseudo-op as defined in Kane and Heinrich. */
323
324 static void
microblaze_s_weakext(int ignore ATTRIBUTE_UNUSED)325 microblaze_s_weakext (int ignore ATTRIBUTE_UNUSED)
326 {
327 char *name;
328 int c;
329 symbolS *symbolP;
330 expressionS exp;
331
332 name = input_line_pointer;
333 c = get_symbol_end ();
334 symbolP = symbol_find_or_make (name);
335 S_SET_WEAK (symbolP);
336 *input_line_pointer = c;
337
338 SKIP_WHITESPACE ();
339
340 if (!is_end_of_line[(unsigned char) *input_line_pointer])
341 {
342 if (S_IS_DEFINED (symbolP))
343 {
344 as_bad ("Ignoring attempt to redefine symbol `%s'.",
345 S_GET_NAME (symbolP));
346 ignore_rest_of_line ();
347 return;
348 }
349
350 if (*input_line_pointer == ',')
351 {
352 ++input_line_pointer;
353 SKIP_WHITESPACE ();
354 }
355
356 expression (&exp);
357 if (exp.X_op != O_symbol)
358 {
359 as_bad ("bad .weakext directive");
360 ignore_rest_of_line ();
361 return;
362 }
363 symbol_set_value_expression (symbolP, &exp);
364 }
365
366 demand_empty_rest_of_line ();
367 }
368
369 /* This table describes all the machine specific pseudo-ops the assembler
370 has to support. The fields are:
371 Pseudo-op name without dot
372 Function to call to execute this pseudo-op
373 Integer arg to pass to the function. */
374 /* If the pseudo-op is not found in this table, it searches in the obj-elf.c,
375 and then in the read.c table. */
376 const pseudo_typeS md_pseudo_table[] =
377 {
378 {"lcomm", microblaze_s_lcomm, 1},
379 {"data", microblaze_s_data, 0},
380 {"data8", cons, 1}, /* Same as byte. */
381 {"data16", cons, 2}, /* Same as hword. */
382 {"data32", cons, 4}, /* Same as word. */
383 {"ent", s_func, 0}, /* Treat ent as function entry point. */
384 {"end", microblaze_s_func, 1}, /* Treat end as function end point. */
385 {"gpword", s_rva, 4}, /* gpword label => store resolved label address in data section. */
386 {"weakext", microblaze_s_weakext, 0},
387 {"rodata", microblaze_s_rdata, 0},
388 {"sdata2", microblaze_s_rdata, 1},
389 {"sdata", microblaze_s_sdata, 0},
390 {"bss", microblaze_s_bss, 0},
391 {"sbss", microblaze_s_bss, 1},
392 {"text", microblaze_s_text, 0},
393 {"word", cons, 4},
394 {"frame", s_ignore, 0},
395 {"mask", s_ignore, 0}, /* Emitted by gcc. */
396 {NULL, NULL, 0}
397 };
398
399 /* This function is called once, at assembler startup time. This should
400 set up all the tables, etc that the MD part of the assembler needs. */
401
402 void
md_begin(void)403 md_begin (void)
404 {
405 struct op_code_struct * opcode;
406
407 opcode_hash_control = hash_new ();
408
409 /* Insert unique names into hash table. */
410 for (opcode = opcodes; opcode->name; opcode ++)
411 hash_insert (opcode_hash_control, opcode->name, (char *) opcode);
412 }
413
414 /* Try to parse a reg name. */
415
416 static char *
parse_reg(char * s,unsigned * reg)417 parse_reg (char * s, unsigned * reg)
418 {
419 unsigned tmpreg = 0;
420
421 /* Strip leading whitespace. */
422 while (ISSPACE (* s))
423 ++ s;
424
425 if (strncasecmp (s, "rpc", 3) == 0)
426 {
427 *reg = REG_PC;
428 return s + 3;
429 }
430 else if (strncasecmp (s, "rmsr", 4) == 0)
431 {
432 *reg = REG_MSR;
433 return s + 4;
434 }
435 else if (strncasecmp (s, "rear", 4) == 0)
436 {
437 *reg = REG_EAR;
438 return s + 4;
439 }
440 else if (strncasecmp (s, "resr", 4) == 0)
441 {
442 *reg = REG_ESR;
443 return s + 4;
444 }
445 else if (strncasecmp (s, "rfsr", 4) == 0)
446 {
447 *reg = REG_FSR;
448 return s + 4;
449 }
450 else if (strncasecmp (s, "rbtr", 4) == 0)
451 {
452 *reg = REG_BTR;
453 return s + 4;
454 }
455 else if (strncasecmp (s, "redr", 4) == 0)
456 {
457 *reg = REG_EDR;
458 return s + 4;
459 }
460 /* MMU registers start. */
461 else if (strncasecmp (s, "rpid", 4) == 0)
462 {
463 *reg = REG_PID;
464 return s + 4;
465 }
466 else if (strncasecmp (s, "rzpr", 4) == 0)
467 {
468 *reg = REG_ZPR;
469 return s + 4;
470 }
471 else if (strncasecmp (s, "rtlbx", 5) == 0)
472 {
473 *reg = REG_TLBX;
474 return s + 5;
475 }
476 else if (strncasecmp (s, "rtlblo", 6) == 0)
477 {
478 *reg = REG_TLBLO;
479 return s + 6;
480 }
481 else if (strncasecmp (s, "rtlbhi", 6) == 0)
482 {
483 *reg = REG_TLBHI;
484 return s + 6;
485 }
486 else if (strncasecmp (s, "rtlbsx", 6) == 0)
487 {
488 *reg = REG_TLBSX;
489 return s + 6;
490 }
491 /* MMU registers end. */
492 else if (strncasecmp (s, "rpvr", 4) == 0)
493 {
494 if (ISDIGIT (s[4]) && ISDIGIT (s[5]))
495 {
496 tmpreg = (s[4]-'0')*10 + s[5] - '0';
497 s += 6;
498 }
499
500 else if (ISDIGIT (s[4]))
501 {
502 tmpreg = s[4] - '0';
503 s += 5;
504 }
505 else
506 as_bad (_("register expected, but saw '%.6s'"), s);
507 if ((int) tmpreg >= MIN_PVR_REGNUM && tmpreg <= MAX_PVR_REGNUM)
508 *reg = REG_PVR + tmpreg;
509 else
510 {
511 as_bad (_("Invalid register number at '%.6s'"), s);
512 *reg = REG_PVR;
513 }
514 return s;
515 }
516 else if (strncasecmp (s, "rsp", 3) == 0)
517 {
518 *reg = REG_SP;
519 return s + 3;
520 }
521 else if (strncasecmp (s, "rfsl", 4) == 0)
522 {
523 if (ISDIGIT (s[4]) && ISDIGIT (s[5]))
524 {
525 tmpreg = (s[4] - '0') * 10 + s[5] - '0';
526 s += 6;
527 }
528 else if (ISDIGIT (s[4]))
529 {
530 tmpreg = s[4] - '0';
531 s += 5;
532 }
533 else
534 as_bad (_("register expected, but saw '%.6s'"), s);
535
536 if ((int) tmpreg >= MIN_REGNUM && tmpreg <= MAX_REGNUM)
537 *reg = tmpreg;
538 else
539 {
540 as_bad (_("Invalid register number at '%.6s'"), s);
541 *reg = 0;
542 }
543 return s;
544 }
545 /* Stack protection registers. */
546 else if (strncasecmp (s, "rshr", 4) == 0)
547 {
548 *reg = REG_SHR;
549 return s + 4;
550 }
551 else if (strncasecmp (s, "rslr", 4) == 0)
552 {
553 *reg = REG_SLR;
554 return s + 4;
555 }
556 else
557 {
558 if (TOLOWER (s[0]) == 'r')
559 {
560 if (ISDIGIT (s[1]) && ISDIGIT (s[2]))
561 {
562 tmpreg = (s[1] - '0') * 10 + s[2] - '0';
563 s += 3;
564 }
565 else if (ISDIGIT (s[1]))
566 {
567 tmpreg = s[1] - '0';
568 s += 2;
569 }
570 else
571 as_bad (_("register expected, but saw '%.6s'"), s);
572
573 if ((int)tmpreg >= MIN_REGNUM && tmpreg <= MAX_REGNUM)
574 *reg = tmpreg;
575 else
576 {
577 as_bad (_("Invalid register number at '%.6s'"), s);
578 *reg = 0;
579 }
580 return s;
581 }
582 }
583 as_bad (_("register expected, but saw '%.6s'"), s);
584 *reg = 0;
585 return s;
586 }
587
588 static char *
parse_exp(char * s,expressionS * e)589 parse_exp (char *s, expressionS *e)
590 {
591 char *save;
592 char *new_pointer;
593
594 /* Skip whitespace. */
595 while (ISSPACE (* s))
596 ++ s;
597
598 save = input_line_pointer;
599 input_line_pointer = s;
600
601 expression (e);
602
603 if (e->X_op == O_absent)
604 as_fatal (_("missing operand"));
605
606 new_pointer = input_line_pointer;
607 input_line_pointer = save;
608
609 return new_pointer;
610 }
611
612 /* Symbol modifiers (@GOT, @PLT, @GOTOFF). */
613 #define IMM_NONE 0
614 #define IMM_GOT 1
615 #define IMM_PLT 2
616 #define IMM_GOTOFF 3
617 #define IMM_TLSGD 4
618 #define IMM_TLSLD 5
619 #define IMM_TLSDTPMOD 6
620 #define IMM_TLSDTPREL 7
621 #define IMM_TLSTPREL 8
622 #define IMM_MAX 9
623
624 struct imm_type {
625 char *isuffix; /* Suffix String */
626 int itype; /* Suffix Type */
627 int otype; /* Offset Type */
628 };
629
630 /* These are NOT in assending order of type, GOTOFF is ahead to make
631 sure @GOTOFF does not get matched with @GOT */
632 static struct imm_type imm_types[] = {
633 { "NONE", IMM_NONE , 0 },
634 { "GOTOFF", IMM_GOTOFF , GOTOFF_OFFSET },
635 { "GOT", IMM_GOT , GOT_OFFSET },
636 { "PLT", IMM_PLT , PLT_OFFSET },
637 { "TLSGD", IMM_TLSGD , TLSGD_OFFSET },
638 { "TLSLDM", IMM_TLSLD, TLSLD_OFFSET },
639 { "TLSDTPMOD", IMM_TLSDTPMOD, TLSDTPMOD_OFFSET },
640 { "TLSDTPREL", IMM_TLSDTPREL, TLSDTPREL_OFFSET },
641 { "TLSTPREL", IMM_TLSTPREL, TLSTPREL_OFFSET }
642 };
643
644 static int
match_imm(const char * s,int * ilen)645 match_imm (const char *s, int *ilen)
646 {
647 int i;
648 int slen;
649
650 /* Check for matching suffix */
651 for (i = 1; i < IMM_MAX; i++)
652 {
653 slen = strlen (imm_types[i].isuffix);
654
655 if (strncmp (imm_types[i].isuffix, s, slen) == 0)
656 {
657 *ilen = slen;
658 return imm_types[i].itype;
659 }
660 } /* for */
661 *ilen = 0;
662 return 0;
663 }
664
665 static int
get_imm_otype(int itype)666 get_imm_otype (int itype)
667 {
668 int i, otype;
669
670 otype = 0;
671 /* Check for matching itype */
672 for (i = 1; i < IMM_MAX; i++)
673 {
674 if (imm_types[i].itype == itype)
675 {
676 otype = imm_types[i].otype;
677 break;
678 }
679 }
680 return otype;
681 }
682
683 static symbolS * GOT_symbol;
684
685 #define GOT_SYMBOL_NAME "_GLOBAL_OFFSET_TABLE_"
686
687 static char *
parse_imm(char * s,expressionS * e,int min,int max)688 parse_imm (char * s, expressionS * e, int min, int max)
689 {
690 char *new_pointer;
691 char *atp;
692 int itype, ilen;
693
694 ilen = 0;
695
696 /* Find the start of "@GOT" or "@PLT" suffix (if any) */
697 for (atp = s; *atp != '@'; atp++)
698 if (is_end_of_line[(unsigned char) *atp])
699 break;
700
701 if (*atp == '@')
702 {
703 itype = match_imm (atp + 1, &ilen);
704 if (itype != 0)
705 {
706 *atp = 0;
707 e->X_md = itype;
708 }
709 else
710 {
711 atp = NULL;
712 e->X_md = 0;
713 ilen = 0;
714 }
715 *atp = 0;
716 }
717 else
718 {
719 atp = NULL;
720 e->X_md = 0;
721 }
722
723 if (atp && !GOT_symbol)
724 {
725 GOT_symbol = symbol_find_or_make (GOT_SYMBOL_NAME);
726 }
727
728 new_pointer = parse_exp (s, e);
729
730 if (!GOT_symbol && ! strncmp (s, GOT_SYMBOL_NAME, 20))
731 {
732 GOT_symbol = symbol_find_or_make (GOT_SYMBOL_NAME);
733 }
734
735 if (e->X_op == O_absent)
736 ; /* An error message has already been emitted. */
737 else if ((e->X_op != O_constant && e->X_op != O_symbol) )
738 as_fatal (_("operand must be a constant or a label"));
739 else if ((e->X_op == O_constant) && ((int) e->X_add_number < min
740 || (int) e->X_add_number > max))
741 {
742 as_fatal (_("operand must be absolute in range %d..%d, not %d"),
743 min, max, (int) e->X_add_number);
744 }
745
746 if (atp)
747 {
748 *atp = '@'; /* restore back (needed?) */
749 if (new_pointer >= atp)
750 new_pointer += ilen + 1; /* sizeof (imm_suffix) + 1 for '@' */
751 }
752 return new_pointer;
753 }
754
755 static char *
check_got(int * got_type,int * got_len)756 check_got (int * got_type, int * got_len)
757 {
758 char *new_pointer;
759 char *atp;
760 char *past_got;
761 int first, second;
762 char *tmpbuf;
763
764 /* Find the start of "@GOT" or "@PLT" suffix (if any). */
765 for (atp = input_line_pointer; *atp != '@'; atp++)
766 if (is_end_of_line[(unsigned char) *atp])
767 return NULL;
768
769 if (strncmp (atp + 1, "GOTOFF", 5) == 0)
770 {
771 *got_len = 6;
772 *got_type = IMM_GOTOFF;
773 }
774 else if (strncmp (atp + 1, "GOT", 3) == 0)
775 {
776 *got_len = 3;
777 *got_type = IMM_GOT;
778 }
779 else if (strncmp (atp + 1, "PLT", 3) == 0)
780 {
781 *got_len = 3;
782 *got_type = IMM_PLT;
783 }
784 else
785 return NULL;
786
787 if (!GOT_symbol)
788 GOT_symbol = symbol_find_or_make (GOT_SYMBOL_NAME);
789
790 first = atp - input_line_pointer;
791
792 past_got = atp + *got_len + 1;
793 for (new_pointer = past_got; !is_end_of_line[(unsigned char) *new_pointer++];)
794 ;
795 second = new_pointer - past_got;
796 tmpbuf = xmalloc (first + second + 2); /* One extra byte for ' ' and one for NUL. */
797 memcpy (tmpbuf, input_line_pointer, first);
798 tmpbuf[first] = ' '; /* @GOTOFF is replaced with a single space. */
799 memcpy (tmpbuf + first + 1, past_got, second);
800 tmpbuf[first + second + 1] = '\0';
801
802 return tmpbuf;
803 }
804
805 extern bfd_reloc_code_real_type
parse_cons_expression_microblaze(expressionS * exp,int size)806 parse_cons_expression_microblaze (expressionS *exp, int size)
807 {
808 if (size == 4)
809 {
810 /* Handle @GOTOFF et.al. */
811 char *save, *gotfree_copy;
812 int got_len, got_type;
813
814 save = input_line_pointer;
815 gotfree_copy = check_got (& got_type, & got_len);
816 if (gotfree_copy)
817 input_line_pointer = gotfree_copy;
818
819 expression (exp);
820
821 if (gotfree_copy)
822 {
823 exp->X_md = got_type;
824 input_line_pointer = save + (input_line_pointer - gotfree_copy)
825 + got_len;
826 free (gotfree_copy);
827 }
828 }
829 else
830 expression (exp);
831 return BFD_RELOC_NONE;
832 }
833
834 /* This is the guts of the machine-dependent assembler. STR points to a
835 machine dependent instruction. This function is supposed to emit
836 the frags/bytes it assembles to. */
837
838 static char * str_microblaze_ro_anchor = "RO";
839 static char * str_microblaze_rw_anchor = "RW";
840
841 static bfd_boolean
check_spl_reg(unsigned * reg)842 check_spl_reg (unsigned * reg)
843 {
844 if ((*reg == REG_MSR) || (*reg == REG_PC)
845 || (*reg == REG_EAR) || (*reg == REG_ESR)
846 || (*reg == REG_FSR) || (*reg == REG_BTR) || (*reg == REG_EDR)
847 || (*reg == REG_PID) || (*reg == REG_ZPR)
848 || (*reg == REG_TLBX) || (*reg == REG_TLBLO)
849 || (*reg == REG_TLBHI) || (*reg == REG_TLBSX)
850 || (*reg == REG_SHR) || (*reg == REG_SLR)
851 || (*reg >= REG_PVR+MIN_PVR_REGNUM && *reg <= REG_PVR+MAX_PVR_REGNUM))
852 return TRUE;
853
854 return FALSE;
855 }
856
857 /* Here we decide which fixups can be adjusted to make them relative to
858 the beginning of the section instead of the symbol. Basically we need
859 to make sure that the dynamic relocations are done correctly, so in
860 some cases we force the original symbol to be used. */
861
862 int
tc_microblaze_fix_adjustable(struct fix * fixP)863 tc_microblaze_fix_adjustable (struct fix *fixP)
864 {
865 if (GOT_symbol && fixP->fx_subsy == GOT_symbol)
866 return 0;
867
868 if (fixP->fx_r_type == BFD_RELOC_MICROBLAZE_64_GOTOFF
869 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_32_GOTOFF
870 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_64_GOT
871 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_64_PLT
872 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_64_TLSGD
873 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_64_TLSLD
874 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_32_TLSDTPMOD
875 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_32_TLSDTPREL
876 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_64_TLSDTPREL
877 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_64_TLSGOTTPREL
878 || fixP->fx_r_type == BFD_RELOC_MICROBLAZE_64_TLSTPREL)
879 return 0;
880
881 return 1;
882 }
883
884 void
md_assemble(char * str)885 md_assemble (char * str)
886 {
887 char * op_start;
888 char * op_end;
889 struct op_code_struct * opcode, *opcode1;
890 char * output = NULL;
891 int nlen = 0;
892 int i;
893 unsigned long inst, inst1;
894 unsigned reg1;
895 unsigned reg2;
896 unsigned reg3;
897 unsigned isize;
898 unsigned int immed, temp;
899 expressionS exp;
900 char name[20];
901
902 /* Drop leading whitespace. */
903 while (ISSPACE (* str))
904 str ++;
905
906 /* Find the op code end. */
907 for (op_start = op_end = str;
908 *op_end && !is_end_of_line[(unsigned char) *op_end] && *op_end != ' ';
909 op_end++)
910 {
911 name[nlen] = op_start[nlen];
912 nlen++;
913 if (nlen == sizeof (name) - 1)
914 break;
915 }
916
917 name [nlen] = 0;
918
919 if (nlen == 0)
920 {
921 as_bad (_("can't find opcode "));
922 return;
923 }
924
925 opcode = (struct op_code_struct *) hash_find (opcode_hash_control, name);
926 if (opcode == NULL)
927 {
928 as_bad (_("unknown opcode \"%s\""), name);
929 return;
930 }
931
932 inst = opcode->bit_sequence;
933 isize = 4;
934
935 switch (opcode->inst_type)
936 {
937 case INST_TYPE_RD_R1_R2:
938 if (strcmp (op_end, ""))
939 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
940 else
941 {
942 as_fatal (_("Error in statement syntax"));
943 reg1 = 0;
944 }
945 if (strcmp (op_end, ""))
946 op_end = parse_reg (op_end + 1, ®2); /* Get r1. */
947 else
948 {
949 as_fatal (_("Error in statement syntax"));
950 reg2 = 0;
951 }
952 if (strcmp (op_end, ""))
953 op_end = parse_reg (op_end + 1, ®3); /* Get r2. */
954 else
955 {
956 as_fatal (_("Error in statement syntax"));
957 reg3 = 0;
958 }
959
960 /* Check for spl registers. */
961 if (check_spl_reg (& reg1))
962 as_fatal (_("Cannot use special register with this instruction"));
963 if (check_spl_reg (& reg2))
964 as_fatal (_("Cannot use special register with this instruction"));
965 if (check_spl_reg (& reg3))
966 as_fatal (_("Cannot use special register with this instruction"));
967
968 if (streq (name, "sub"))
969 {
970 /* sub rd, r1, r2 becomes rsub rd, r2, r1. */
971 inst |= (reg1 << RD_LOW) & RD_MASK;
972 inst |= (reg3 << RA_LOW) & RA_MASK;
973 inst |= (reg2 << RB_LOW) & RB_MASK;
974 }
975 else
976 {
977 inst |= (reg1 << RD_LOW) & RD_MASK;
978 inst |= (reg2 << RA_LOW) & RA_MASK;
979 inst |= (reg3 << RB_LOW) & RB_MASK;
980 }
981 output = frag_more (isize);
982 break;
983
984 case INST_TYPE_RD_R1_IMM:
985 if (strcmp (op_end, ""))
986 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
987 else
988 {
989 as_fatal (_("Error in statement syntax"));
990 reg1 = 0;
991 }
992 if (strcmp (op_end, ""))
993 op_end = parse_reg (op_end + 1, ®2); /* Get r1. */
994 else
995 {
996 as_fatal (_("Error in statement syntax"));
997 reg2 = 0;
998 }
999 if (strcmp (op_end, ""))
1000 op_end = parse_imm (op_end + 1, & exp, MIN_IMM, MAX_IMM);
1001 else
1002 as_fatal (_("Error in statement syntax"));
1003
1004 /* Check for spl registers. */
1005 if (check_spl_reg (& reg1))
1006 as_fatal (_("Cannot use special register with this instruction"));
1007 if (check_spl_reg (& reg2))
1008 as_fatal (_("Cannot use special register with this instruction"));
1009
1010 if (exp.X_op != O_constant)
1011 {
1012 char *opc;
1013 relax_substateT subtype;
1014
1015 if (streq (name, "lmi"))
1016 as_fatal (_("lmi pseudo instruction should not use a label in imm field"));
1017 else if (streq (name, "smi"))
1018 as_fatal (_("smi pseudo instruction should not use a label in imm field"));
1019
1020 if (reg2 == REG_ROSDP)
1021 opc = str_microblaze_ro_anchor;
1022 else if (reg2 == REG_RWSDP)
1023 opc = str_microblaze_rw_anchor;
1024 else
1025 opc = NULL;
1026 if (exp.X_md != 0)
1027 subtype = get_imm_otype(exp.X_md);
1028 else
1029 subtype = opcode->inst_offset_type;
1030
1031 output = frag_var (rs_machine_dependent,
1032 isize * 2, /* maxm of 2 words. */
1033 isize, /* minm of 1 word. */
1034 subtype, /* PC-relative or not. */
1035 exp.X_add_symbol,
1036 exp.X_add_number,
1037 opc);
1038 immed = 0;
1039 }
1040 else
1041 {
1042 output = frag_more (isize);
1043 immed = exp.X_add_number;
1044 }
1045
1046 if (streq (name, "lmi") || streq (name, "smi"))
1047 {
1048 /* Load/store 32-d consecutive registers. Used on exit/entry
1049 to subroutines to save and restore registers to stack.
1050 Generate 32-d insts. */
1051 int count;
1052
1053 count = 32 - reg1;
1054 if (streq (name, "lmi"))
1055 opcode = (struct op_code_struct *) hash_find (opcode_hash_control, "lwi");
1056 else
1057 opcode = (struct op_code_struct *) hash_find (opcode_hash_control, "swi");
1058 if (opcode == NULL)
1059 {
1060 as_bad (_("unknown opcode \"%s\""), "lwi");
1061 return;
1062 }
1063 inst = opcode->bit_sequence;
1064 inst |= (reg1 << RD_LOW) & RD_MASK;
1065 inst |= (reg2 << RA_LOW) & RA_MASK;
1066 inst |= (immed << IMM_LOW) & IMM_MASK;
1067
1068 for (i = 0; i < count - 1; i++)
1069 {
1070 output[0] = INST_BYTE0 (inst);
1071 output[1] = INST_BYTE1 (inst);
1072 output[2] = INST_BYTE2 (inst);
1073 output[3] = INST_BYTE3 (inst);
1074 output = frag_more (isize);
1075 immed = immed + 4;
1076 reg1++;
1077 inst = opcode->bit_sequence;
1078 inst |= (reg1 << RD_LOW) & RD_MASK;
1079 inst |= (reg2 << RA_LOW) & RA_MASK;
1080 inst |= (immed << IMM_LOW) & IMM_MASK;
1081 }
1082 }
1083 else
1084 {
1085 temp = immed & 0xFFFF8000;
1086 if ((temp != 0) && (temp != 0xFFFF8000))
1087 {
1088 /* Needs an immediate inst. */
1089 opcode1 = (struct op_code_struct *) hash_find (opcode_hash_control, "imm");
1090 if (opcode1 == NULL)
1091 {
1092 as_bad (_("unknown opcode \"%s\""), "imm");
1093 return;
1094 }
1095
1096 inst1 = opcode1->bit_sequence;
1097 inst1 |= ((immed & 0xFFFF0000) >> 16) & IMM_MASK;
1098 output[0] = INST_BYTE0 (inst1);
1099 output[1] = INST_BYTE1 (inst1);
1100 output[2] = INST_BYTE2 (inst1);
1101 output[3] = INST_BYTE3 (inst1);
1102 output = frag_more (isize);
1103 }
1104 inst |= (reg1 << RD_LOW) & RD_MASK;
1105 inst |= (reg2 << RA_LOW) & RA_MASK;
1106 inst |= (immed << IMM_LOW) & IMM_MASK;
1107 }
1108 break;
1109
1110 case INST_TYPE_RD_R1_IMM5:
1111 if (strcmp (op_end, ""))
1112 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1113 else
1114 {
1115 as_fatal (_("Error in statement syntax"));
1116 reg1 = 0;
1117 }
1118 if (strcmp (op_end, ""))
1119 op_end = parse_reg (op_end + 1, ®2); /* Get r1. */
1120 else
1121 {
1122 as_fatal (_("Error in statement syntax"));
1123 reg2 = 0;
1124 }
1125 if (strcmp (op_end, ""))
1126 op_end = parse_imm (op_end + 1, & exp, MIN_IMM, MAX_IMM);
1127 else
1128 as_fatal (_("Error in statement syntax"));
1129
1130 /* Check for spl registers. */
1131 if (check_spl_reg (®1))
1132 as_fatal (_("Cannot use special register with this instruction"));
1133 if (check_spl_reg (®2))
1134 as_fatal (_("Cannot use special register with this instruction"));
1135
1136 if (exp.X_op != O_constant)
1137 as_warn (_("Symbol used as immediate for shift instruction"));
1138 else
1139 {
1140 output = frag_more (isize);
1141 immed = exp.X_add_number;
1142 }
1143
1144 if (immed != (immed % 32))
1145 {
1146 as_warn (_("Shift value > 32. using <value %% 32>"));
1147 immed = immed % 32;
1148 }
1149 inst |= (reg1 << RD_LOW) & RD_MASK;
1150 inst |= (reg2 << RA_LOW) & RA_MASK;
1151 inst |= (immed << IMM_LOW) & IMM5_MASK;
1152 break;
1153
1154 case INST_TYPE_R1_R2:
1155 if (strcmp (op_end, ""))
1156 op_end = parse_reg (op_end + 1, ®1); /* Get r1. */
1157 else
1158 {
1159 as_fatal (_("Error in statement syntax"));
1160 reg1 = 0;
1161 }
1162 if (strcmp (op_end, ""))
1163 op_end = parse_reg (op_end + 1, ®2); /* Get r2. */
1164 else
1165 {
1166 as_fatal (_("Error in statement syntax"));
1167 reg2 = 0;
1168 }
1169
1170 /* Check for spl registers. */
1171 if (check_spl_reg (& reg1))
1172 as_fatal (_("Cannot use special register with this instruction"));
1173 if (check_spl_reg (& reg2))
1174 as_fatal (_("Cannot use special register with this instruction"));
1175
1176 inst |= (reg1 << RA_LOW) & RA_MASK;
1177 inst |= (reg2 << RB_LOW) & RB_MASK;
1178 output = frag_more (isize);
1179 break;
1180
1181 case INST_TYPE_RD_R1:
1182 if (strcmp (op_end, ""))
1183 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1184 else
1185 {
1186 as_fatal (_("Error in statement syntax"));
1187 reg1 = 0;
1188 }
1189 if (strcmp (op_end, ""))
1190 op_end = parse_reg (op_end + 1, ®2); /* Get r1. */
1191 else
1192 {
1193 as_fatal (_("Error in statement syntax"));
1194 reg2 =0;
1195 }
1196
1197 /* Check for spl registers. */
1198 if (check_spl_reg (®1))
1199 as_fatal (_("Cannot use special register with this instruction"));
1200 if (check_spl_reg (®2))
1201 as_fatal (_("Cannot use special register with this instruction"));
1202
1203 inst |= (reg1 << RD_LOW) & RD_MASK;
1204 inst |= (reg2 << RA_LOW) & RA_MASK;
1205 output = frag_more (isize);
1206 break;
1207
1208 case INST_TYPE_RD_RFSL:
1209 if (strcmp (op_end, ""))
1210 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1211 else
1212 {
1213 as_fatal (_("Error in statement syntax"));
1214 reg1 = 0;
1215 }
1216 if (strcmp (op_end, ""))
1217 op_end = parse_reg (op_end + 1, &immed); /* Get rfslN. */
1218 else
1219 {
1220 as_fatal (_("Error in statement syntax"));
1221 immed = 0;
1222 }
1223
1224 /* Check for spl registers. */
1225 if (check_spl_reg (®1))
1226 as_fatal (_("Cannot use special register with this instruction"));
1227
1228 inst |= (reg1 << RD_LOW) & RD_MASK;
1229 inst |= (immed << IMM_LOW) & RFSL_MASK;
1230 output = frag_more (isize);
1231 break;
1232
1233 case INST_TYPE_RD_IMM15:
1234 if (strcmp (op_end, ""))
1235 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1236 else
1237 {
1238 as_fatal (_("Error in statement syntax"));
1239 reg1 = 0;
1240 }
1241
1242 if (strcmp (op_end, ""))
1243 op_end = parse_imm (op_end + 1, & exp, MIN_IMM15, MAX_IMM15);
1244 else
1245 as_fatal (_("Error in statement syntax"));
1246
1247 /* Check for spl registers. */
1248 if (check_spl_reg (®1))
1249 as_fatal (_("Cannot use special register with this instruction"));
1250
1251 if (exp.X_op != O_constant)
1252 as_fatal (_("Symbol used as immediate value for msrset/msrclr instructions"));
1253 else
1254 {
1255 output = frag_more (isize);
1256 immed = exp.X_add_number;
1257 }
1258 inst |= (reg1 << RD_LOW) & RD_MASK;
1259 inst |= (immed << IMM_LOW) & IMM15_MASK;
1260 break;
1261
1262 case INST_TYPE_R1_RFSL:
1263 if (strcmp (op_end, ""))
1264 op_end = parse_reg (op_end + 1, ®1); /* Get r1. */
1265 else
1266 {
1267 as_fatal (_("Error in statement syntax"));
1268 reg1 = 0;
1269 }
1270 if (strcmp (op_end, ""))
1271 op_end = parse_reg (op_end + 1, &immed); /* Get rfslN. */
1272 else
1273 {
1274 as_fatal (_("Error in statement syntax"));
1275 immed = 0;
1276 }
1277
1278 /* Check for spl registers. */
1279 if (check_spl_reg (®1))
1280 as_fatal (_("Cannot use special register with this instruction"));
1281
1282 inst |= (reg1 << RA_LOW) & RA_MASK;
1283 inst |= (immed << IMM_LOW) & RFSL_MASK;
1284 output = frag_more (isize);
1285 break;
1286
1287 case INST_TYPE_RFSL:
1288 if (strcmp (op_end, ""))
1289 op_end = parse_reg (op_end + 1, &immed); /* Get rfslN. */
1290 else
1291 {
1292 as_fatal (_("Error in statement syntax"));
1293 immed = 0;
1294 }
1295 inst |= (immed << IMM_LOW) & RFSL_MASK;
1296 output = frag_more (isize);
1297 break;
1298
1299 case INST_TYPE_R1:
1300 if (strcmp (op_end, ""))
1301 op_end = parse_reg (op_end + 1, ®1); /* Get r1. */
1302 else
1303 {
1304 as_fatal (_("Error in statement syntax"));
1305 reg1 = 0;
1306 }
1307
1308 /* Check for spl registers. */
1309 if (check_spl_reg (®1))
1310 as_fatal (_("Cannot use special register with this instruction"));
1311
1312 inst |= (reg1 << RA_LOW) & RA_MASK;
1313 output = frag_more (isize);
1314 break;
1315
1316 /* For tuqula insn...:) */
1317 case INST_TYPE_RD:
1318 if (strcmp (op_end, ""))
1319 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1320 else
1321 {
1322 as_fatal (_("Error in statement syntax"));
1323 reg1 = 0;
1324 }
1325
1326 /* Check for spl registers. */
1327 if (check_spl_reg (®1))
1328 as_fatal (_("Cannot use special register with this instruction"));
1329
1330 inst |= (reg1 << RD_LOW) & RD_MASK;
1331 output = frag_more (isize);
1332 break;
1333
1334 case INST_TYPE_RD_SPECIAL:
1335 if (strcmp (op_end, ""))
1336 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1337 else
1338 {
1339 as_fatal (_("Error in statement syntax"));
1340 reg1 = 0;
1341 }
1342 if (strcmp (op_end, ""))
1343 op_end = parse_reg (op_end + 1, ®2); /* Get r1. */
1344 else
1345 {
1346 as_fatal (_("Error in statement syntax"));
1347 reg2 = 0;
1348 }
1349
1350 if (reg2 == REG_MSR)
1351 immed = opcode->immval_mask | REG_MSR_MASK;
1352 else if (reg2 == REG_PC)
1353 immed = opcode->immval_mask | REG_PC_MASK;
1354 else if (reg2 == REG_EAR)
1355 immed = opcode->immval_mask | REG_EAR_MASK;
1356 else if (reg2 == REG_ESR)
1357 immed = opcode->immval_mask | REG_ESR_MASK;
1358 else if (reg2 == REG_FSR)
1359 immed = opcode->immval_mask | REG_FSR_MASK;
1360 else if (reg2 == REG_BTR)
1361 immed = opcode->immval_mask | REG_BTR_MASK;
1362 else if (reg2 == REG_EDR)
1363 immed = opcode->immval_mask | REG_EDR_MASK;
1364 else if (reg2 == REG_PID)
1365 immed = opcode->immval_mask | REG_PID_MASK;
1366 else if (reg2 == REG_ZPR)
1367 immed = opcode->immval_mask | REG_ZPR_MASK;
1368 else if (reg2 == REG_TLBX)
1369 immed = opcode->immval_mask | REG_TLBX_MASK;
1370 else if (reg2 == REG_TLBLO)
1371 immed = opcode->immval_mask | REG_TLBLO_MASK;
1372 else if (reg2 == REG_TLBHI)
1373 immed = opcode->immval_mask | REG_TLBHI_MASK;
1374 else if (reg2 == REG_SHR)
1375 immed = opcode->immval_mask | REG_SHR_MASK;
1376 else if (reg2 == REG_SLR)
1377 immed = opcode->immval_mask | REG_SLR_MASK;
1378 else if (reg2 >= (REG_PVR+MIN_PVR_REGNUM) && reg2 <= (REG_PVR+MAX_PVR_REGNUM))
1379 immed = opcode->immval_mask | REG_PVR_MASK | reg2;
1380 else
1381 as_fatal (_("invalid value for special purpose register"));
1382 inst |= (reg1 << RD_LOW) & RD_MASK;
1383 inst |= (immed << IMM_LOW) & IMM_MASK;
1384 output = frag_more (isize);
1385 break;
1386
1387 case INST_TYPE_SPECIAL_R1:
1388 if (strcmp (op_end, ""))
1389 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1390 else
1391 {
1392 as_fatal (_("Error in statement syntax"));
1393 reg1 = 0;
1394 }
1395 if (strcmp (op_end, ""))
1396 op_end = parse_reg (op_end + 1, ®2); /* Get r1. */
1397 else
1398 {
1399 as_fatal (_("Error in statement syntax"));
1400 reg2 = 0;
1401 }
1402
1403 if (reg1 == REG_MSR)
1404 immed = opcode->immval_mask | REG_MSR_MASK;
1405 else if (reg1 == REG_PC)
1406 immed = opcode->immval_mask | REG_PC_MASK;
1407 else if (reg1 == REG_EAR)
1408 immed = opcode->immval_mask | REG_EAR_MASK;
1409 else if (reg1 == REG_ESR)
1410 immed = opcode->immval_mask | REG_ESR_MASK;
1411 else if (reg1 == REG_FSR)
1412 immed = opcode->immval_mask | REG_FSR_MASK;
1413 else if (reg1 == REG_BTR)
1414 immed = opcode->immval_mask | REG_BTR_MASK;
1415 else if (reg1 == REG_EDR)
1416 immed = opcode->immval_mask | REG_EDR_MASK;
1417 else if (reg1 == REG_PID)
1418 immed = opcode->immval_mask | REG_PID_MASK;
1419 else if (reg1 == REG_ZPR)
1420 immed = opcode->immval_mask | REG_ZPR_MASK;
1421 else if (reg1 == REG_TLBX)
1422 immed = opcode->immval_mask | REG_TLBX_MASK;
1423 else if (reg1 == REG_TLBLO)
1424 immed = opcode->immval_mask | REG_TLBLO_MASK;
1425 else if (reg1 == REG_TLBHI)
1426 immed = opcode->immval_mask | REG_TLBHI_MASK;
1427 else if (reg1 == REG_TLBSX)
1428 immed = opcode->immval_mask | REG_TLBSX_MASK;
1429 else if (reg1 == REG_SHR)
1430 immed = opcode->immval_mask | REG_SHR_MASK;
1431 else if (reg1 == REG_SLR)
1432 immed = opcode->immval_mask | REG_SLR_MASK;
1433 else
1434 as_fatal (_("invalid value for special purpose register"));
1435 inst |= (reg2 << RA_LOW) & RA_MASK;
1436 inst |= (immed << IMM_LOW) & IMM_MASK;
1437 output = frag_more (isize);
1438 break;
1439
1440 case INST_TYPE_R1_R2_SPECIAL:
1441 if (strcmp (op_end, ""))
1442 op_end = parse_reg (op_end + 1, ®1); /* Get r1. */
1443 else
1444 {
1445 as_fatal (_("Error in statement syntax"));
1446 reg1 = 0;
1447 }
1448 if (strcmp (op_end, ""))
1449 op_end = parse_reg (op_end + 1, ®2); /* Get r2. */
1450 else
1451 {
1452 as_fatal (_("Error in statement syntax"));
1453 reg2 =0;
1454 }
1455
1456 /* Check for spl registers. */
1457 if (check_spl_reg (®1))
1458 as_fatal (_("Cannot use special register with this instruction"));
1459 if (check_spl_reg (®2))
1460 as_fatal (_("Cannot use special register with this instruction"));
1461
1462 /* insn wic ra, rb => wic ra, ra, rb. */
1463 inst |= (reg1 << RA_LOW) & RA_MASK;
1464 inst |= (reg2 << RB_LOW) & RB_MASK;
1465
1466 output = frag_more (isize);
1467 break;
1468
1469 case INST_TYPE_RD_R2:
1470 if (strcmp (op_end, ""))
1471 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1472 else
1473 {
1474 as_fatal (_("Error in statement syntax"));
1475 reg1 = 0;
1476 }
1477 if (strcmp (op_end, ""))
1478 op_end = parse_reg (op_end + 1, ®2); /* Get r2. */
1479 else
1480 {
1481 as_fatal (_("Error in statement syntax"));
1482 reg2 = 0;
1483 }
1484
1485 /* Check for spl registers. */
1486 if (check_spl_reg (®1))
1487 as_fatal (_("Cannot use special register with this instruction"));
1488 if (check_spl_reg (®2))
1489 as_fatal (_("Cannot use special register with this instruction"));
1490
1491 inst |= (reg1 << RD_LOW) & RD_MASK;
1492 inst |= (reg2 << RB_LOW) & RB_MASK;
1493 output = frag_more (isize);
1494 break;
1495
1496 case INST_TYPE_R1_IMM:
1497 if (strcmp (op_end, ""))
1498 op_end = parse_reg (op_end + 1, ®1); /* Get r1. */
1499 else
1500 {
1501 as_fatal (_("Error in statement syntax"));
1502 reg1 = 0;
1503 }
1504 if (strcmp (op_end, ""))
1505 op_end = parse_imm (op_end + 1, & exp, MIN_IMM, MAX_IMM);
1506 else
1507 as_fatal (_("Error in statement syntax"));
1508
1509 /* Check for spl registers. */
1510 if (check_spl_reg (®1))
1511 as_fatal (_("Cannot use special register with this instruction"));
1512
1513 if (exp.X_op != O_constant)
1514 {
1515 char *opc = NULL;
1516 relax_substateT subtype;
1517
1518 if (exp.X_md != 0)
1519 subtype = get_imm_otype(exp.X_md);
1520 else
1521 subtype = opcode->inst_offset_type;
1522
1523 output = frag_var (rs_machine_dependent,
1524 isize * 2, /* maxm of 2 words. */
1525 isize, /* minm of 1 word. */
1526 subtype, /* PC-relative or not. */
1527 exp.X_add_symbol,
1528 exp.X_add_number,
1529 opc);
1530 immed = 0;
1531 }
1532 else
1533 {
1534 output = frag_more (isize);
1535 immed = exp.X_add_number;
1536 }
1537
1538 temp = immed & 0xFFFF8000;
1539 if ((temp != 0) && (temp != 0xFFFF8000))
1540 {
1541 /* Needs an immediate inst. */
1542 opcode1 = (struct op_code_struct *) hash_find (opcode_hash_control, "imm");
1543 if (opcode1 == NULL)
1544 {
1545 as_bad (_("unknown opcode \"%s\""), "imm");
1546 return;
1547 }
1548
1549 inst1 = opcode1->bit_sequence;
1550 inst1 |= ((immed & 0xFFFF0000) >> 16) & IMM_MASK;
1551 output[0] = INST_BYTE0 (inst1);
1552 output[1] = INST_BYTE1 (inst1);
1553 output[2] = INST_BYTE2 (inst1);
1554 output[3] = INST_BYTE3 (inst1);
1555 output = frag_more (isize);
1556 }
1557
1558 inst |= (reg1 << RA_LOW) & RA_MASK;
1559 inst |= (immed << IMM_LOW) & IMM_MASK;
1560 break;
1561
1562 case INST_TYPE_RD_IMM:
1563 if (strcmp (op_end, ""))
1564 op_end = parse_reg (op_end + 1, ®1); /* Get rd. */
1565 else
1566 {
1567 as_fatal (_("Error in statement syntax"));
1568 reg1 = 0;
1569 }
1570 if (strcmp (op_end, ""))
1571 op_end = parse_imm (op_end + 1, & exp, MIN_IMM, MAX_IMM);
1572 else
1573 as_fatal (_("Error in statement syntax"));
1574
1575 /* Check for spl registers. */
1576 if (check_spl_reg (®1))
1577 as_fatal (_("Cannot use special register with this instruction"));
1578
1579 if (exp.X_op != O_constant)
1580 {
1581 char *opc = NULL;
1582 relax_substateT subtype;
1583
1584 if (exp.X_md != 0)
1585 subtype = get_imm_otype(exp.X_md);
1586 else
1587 subtype = opcode->inst_offset_type;
1588
1589 output = frag_var (rs_machine_dependent,
1590 isize * 2, /* maxm of 2 words. */
1591 isize, /* minm of 1 word. */
1592 subtype, /* PC-relative or not. */
1593 exp.X_add_symbol,
1594 exp.X_add_number,
1595 opc);
1596 immed = 0;
1597 }
1598 else
1599 {
1600 output = frag_more (isize);
1601 immed = exp.X_add_number;
1602 }
1603
1604 temp = immed & 0xFFFF8000;
1605 if ((temp != 0) && (temp != 0xFFFF8000))
1606 {
1607 /* Needs an immediate inst. */
1608 opcode1 = (struct op_code_struct *) hash_find (opcode_hash_control, "imm");
1609 if (opcode1 == NULL)
1610 {
1611 as_bad (_("unknown opcode \"%s\""), "imm");
1612 return;
1613 }
1614
1615 inst1 = opcode1->bit_sequence;
1616 inst1 |= ((immed & 0xFFFF0000) >> 16) & IMM_MASK;
1617 output[0] = INST_BYTE0 (inst1);
1618 output[1] = INST_BYTE1 (inst1);
1619 output[2] = INST_BYTE2 (inst1);
1620 output[3] = INST_BYTE3 (inst1);
1621 output = frag_more (isize);
1622 }
1623
1624 inst |= (reg1 << RD_LOW) & RD_MASK;
1625 inst |= (immed << IMM_LOW) & IMM_MASK;
1626 break;
1627
1628 case INST_TYPE_R2:
1629 if (strcmp (op_end, ""))
1630 op_end = parse_reg (op_end + 1, ®2); /* Get r2. */
1631 else
1632 {
1633 as_fatal (_("Error in statement syntax"));
1634 reg2 = 0;
1635 }
1636
1637 /* Check for spl registers. */
1638 if (check_spl_reg (®2))
1639 as_fatal (_("Cannot use special register with this instruction"));
1640
1641 inst |= (reg2 << RB_LOW) & RB_MASK;
1642 output = frag_more (isize);
1643 break;
1644
1645 case INST_TYPE_IMM:
1646 if (streq (name, "imm"))
1647 as_fatal (_("An IMM instruction should not be present in the .s file"));
1648
1649 op_end = parse_imm (op_end + 1, & exp, MIN_IMM, MAX_IMM);
1650
1651 if (exp.X_op != O_constant)
1652 {
1653 char *opc = NULL;
1654 relax_substateT subtype;
1655
1656 if (exp.X_md != 0)
1657 subtype = get_imm_otype(exp.X_md);
1658 else
1659 subtype = opcode->inst_offset_type;
1660
1661 output = frag_var (rs_machine_dependent,
1662 isize * 2, /* maxm of 2 words. */
1663 isize, /* minm of 1 word. */
1664 subtype, /* PC-relative or not. */
1665 exp.X_add_symbol,
1666 exp.X_add_number,
1667 opc);
1668 immed = 0;
1669 }
1670 else
1671 {
1672 output = frag_more (isize);
1673 immed = exp.X_add_number;
1674 }
1675
1676
1677 temp = immed & 0xFFFF8000;
1678 if ((temp != 0) && (temp != 0xFFFF8000))
1679 {
1680 /* Needs an immediate inst. */
1681 opcode1 = (struct op_code_struct *) hash_find (opcode_hash_control, "imm");
1682 if (opcode1 == NULL)
1683 {
1684 as_bad (_("unknown opcode \"%s\""), "imm");
1685 return;
1686 }
1687
1688 inst1 = opcode1->bit_sequence;
1689 inst1 |= ((immed & 0xFFFF0000) >> 16) & IMM_MASK;
1690 output[0] = INST_BYTE0 (inst1);
1691 output[1] = INST_BYTE1 (inst1);
1692 output[2] = INST_BYTE2 (inst1);
1693 output[3] = INST_BYTE3 (inst1);
1694 output = frag_more (isize);
1695 }
1696 inst |= (immed << IMM_LOW) & IMM_MASK;
1697 break;
1698
1699 case INST_TYPE_NONE:
1700 output = frag_more (isize);
1701 break;
1702
1703 case INST_TYPE_IMM5:
1704 if (strcmp(op_end, ""))
1705 op_end = parse_imm (op_end + 1, & exp, MIN_IMM5, MAX_IMM5);
1706 else
1707 as_fatal(_("Error in statement syntax"));
1708 if (exp.X_op != O_constant) {
1709 as_warn(_("Symbol used as immediate for mbar instruction"));
1710 } else {
1711 output = frag_more (isize);
1712 immed = exp.X_add_number;
1713 }
1714 if (immed != (immed % 32)) {
1715 as_warn(_("Immediate value for mbar > 32. using <value %% 32>"));
1716 immed = immed % 32;
1717 }
1718 inst |= (immed << IMM_MBAR);
1719 break;
1720
1721 default:
1722 as_fatal (_("unimplemented opcode \"%s\""), name);
1723 }
1724
1725 /* Drop whitespace after all the operands have been parsed. */
1726 while (ISSPACE (* op_end))
1727 op_end ++;
1728
1729 /* Give warning message if the insn has more operands than required. */
1730 if (strcmp (op_end, opcode->name) && strcmp (op_end, ""))
1731 as_warn (_("ignoring operands: %s "), op_end);
1732
1733 output[0] = INST_BYTE0 (inst);
1734 output[1] = INST_BYTE1 (inst);
1735 output[2] = INST_BYTE2 (inst);
1736 output[3] = INST_BYTE3 (inst);
1737
1738 #ifdef OBJ_ELF
1739 dwarf2_emit_insn (4);
1740 #endif
1741 }
1742
1743 symbolS *
md_undefined_symbol(char * name ATTRIBUTE_UNUSED)1744 md_undefined_symbol (char * name ATTRIBUTE_UNUSED)
1745 {
1746 return NULL;
1747 }
1748
1749 /* Various routines to kill one day. */
1750 /* Equal to MAX_PRECISION in atof-ieee.c */
1751 #define MAX_LITTLENUMS 6
1752
1753 /* Turn a string in input_line_pointer into a floating point constant of type
1754 type, and store the appropriate bytes in *litP. The number of LITTLENUMS
1755 emitted is stored in *sizeP. An error message is returned, or NULL on OK.*/
1756 char *
md_atof(int type,char * litP,int * sizeP)1757 md_atof (int type, char * litP, int * sizeP)
1758 {
1759 int prec;
1760 LITTLENUM_TYPE words[MAX_LITTLENUMS];
1761 int i;
1762 char * t;
1763
1764 switch (type)
1765 {
1766 case 'f':
1767 case 'F':
1768 case 's':
1769 case 'S':
1770 prec = 2;
1771 break;
1772
1773 case 'd':
1774 case 'D':
1775 case 'r':
1776 case 'R':
1777 prec = 4;
1778 break;
1779
1780 case 'x':
1781 case 'X':
1782 prec = 6;
1783 break;
1784
1785 case 'p':
1786 case 'P':
1787 prec = 6;
1788 break;
1789
1790 default:
1791 *sizeP = 0;
1792 return _("Bad call to MD_NTOF()");
1793 }
1794
1795 t = atof_ieee (input_line_pointer, type, words);
1796
1797 if (t)
1798 input_line_pointer = t;
1799
1800 *sizeP = prec * sizeof (LITTLENUM_TYPE);
1801
1802 if (! target_big_endian)
1803 {
1804 for (i = prec - 1; i >= 0; i--)
1805 {
1806 md_number_to_chars (litP, (valueT) words[i],
1807 sizeof (LITTLENUM_TYPE));
1808 litP += sizeof (LITTLENUM_TYPE);
1809 }
1810 }
1811 else
1812 for (i = 0; i < prec; i++)
1813 {
1814 md_number_to_chars (litP, (valueT) words[i],
1815 sizeof (LITTLENUM_TYPE));
1816 litP += sizeof (LITTLENUM_TYPE);
1817 }
1818
1819 return NULL;
1820 }
1821
1822 const char * md_shortopts = "";
1823
1824 struct option md_longopts[] =
1825 {
1826 {"EB", no_argument, NULL, OPTION_EB},
1827 {"EL", no_argument, NULL, OPTION_EL},
1828 { NULL, no_argument, NULL, 0}
1829 };
1830
1831 size_t md_longopts_size = sizeof (md_longopts);
1832
1833 int md_short_jump_size;
1834
1835 void
md_create_short_jump(char * ptr ATTRIBUTE_UNUSED,addressT from_Nddr ATTRIBUTE_UNUSED,addressT to_Nddr ATTRIBUTE_UNUSED,fragS * frag ATTRIBUTE_UNUSED,symbolS * to_symbol ATTRIBUTE_UNUSED)1836 md_create_short_jump (char * ptr ATTRIBUTE_UNUSED,
1837 addressT from_Nddr ATTRIBUTE_UNUSED,
1838 addressT to_Nddr ATTRIBUTE_UNUSED,
1839 fragS * frag ATTRIBUTE_UNUSED,
1840 symbolS * to_symbol ATTRIBUTE_UNUSED)
1841 {
1842 as_fatal (_("failed sanity check: short_jump"));
1843 }
1844
1845 void
md_create_long_jump(char * ptr ATTRIBUTE_UNUSED,addressT from_Nddr ATTRIBUTE_UNUSED,addressT to_Nddr ATTRIBUTE_UNUSED,fragS * frag ATTRIBUTE_UNUSED,symbolS * to_symbol ATTRIBUTE_UNUSED)1846 md_create_long_jump (char * ptr ATTRIBUTE_UNUSED,
1847 addressT from_Nddr ATTRIBUTE_UNUSED,
1848 addressT to_Nddr ATTRIBUTE_UNUSED,
1849 fragS * frag ATTRIBUTE_UNUSED,
1850 symbolS * to_symbol ATTRIBUTE_UNUSED)
1851 {
1852 as_fatal (_("failed sanity check: long_jump"));
1853 }
1854
1855 /* Called after relaxing, change the frags so they know how big they are. */
1856
1857 void
md_convert_frag(bfd * abfd ATTRIBUTE_UNUSED,segT sec ATTRIBUTE_UNUSED,fragS * fragP)1858 md_convert_frag (bfd * abfd ATTRIBUTE_UNUSED,
1859 segT sec ATTRIBUTE_UNUSED,
1860 fragS * fragP)
1861 {
1862 fixS *fixP;
1863
1864 switch (fragP->fr_subtype)
1865 {
1866 case UNDEFINED_PC_OFFSET:
1867 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1868 fragP->fr_offset, TRUE, BFD_RELOC_64_PCREL);
1869 fragP->fr_fix += INST_WORD_SIZE * 2;
1870 fragP->fr_var = 0;
1871 break;
1872 case DEFINED_ABS_SEGMENT:
1873 if (fragP->fr_symbol == GOT_symbol)
1874 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1875 fragP->fr_offset, TRUE, BFD_RELOC_MICROBLAZE_64_GOTPC);
1876 else
1877 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1878 fragP->fr_offset, FALSE, BFD_RELOC_64);
1879 fragP->fr_fix += INST_WORD_SIZE * 2;
1880 fragP->fr_var = 0;
1881 break;
1882 case DEFINED_RO_SEGMENT:
1883 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE, fragP->fr_symbol,
1884 fragP->fr_offset, FALSE, BFD_RELOC_MICROBLAZE_32_ROSDA);
1885 fragP->fr_fix += INST_WORD_SIZE;
1886 fragP->fr_var = 0;
1887 break;
1888 case DEFINED_RW_SEGMENT:
1889 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE, fragP->fr_symbol,
1890 fragP->fr_offset, FALSE, BFD_RELOC_MICROBLAZE_32_RWSDA);
1891 fragP->fr_fix += INST_WORD_SIZE;
1892 fragP->fr_var = 0;
1893 break;
1894 case DEFINED_PC_OFFSET:
1895 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE, fragP->fr_symbol,
1896 fragP->fr_offset, TRUE, BFD_RELOC_MICROBLAZE_32_LO_PCREL);
1897 fragP->fr_fix += INST_WORD_SIZE;
1898 fragP->fr_var = 0;
1899 break;
1900 case LARGE_DEFINED_PC_OFFSET:
1901 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1902 fragP->fr_offset, TRUE, BFD_RELOC_64_PCREL);
1903 fragP->fr_fix += INST_WORD_SIZE * 2;
1904 fragP->fr_var = 0;
1905 break;
1906 case GOT_OFFSET:
1907 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1908 fragP->fr_offset, FALSE, BFD_RELOC_MICROBLAZE_64_GOT);
1909 fragP->fr_fix += INST_WORD_SIZE * 2;
1910 fragP->fr_var = 0;
1911 break;
1912 case PLT_OFFSET:
1913 fixP = fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1914 fragP->fr_offset, TRUE, BFD_RELOC_MICROBLAZE_64_PLT);
1915 /* fixP->fx_plt = 1; */
1916 (void) fixP;
1917 fragP->fr_fix += INST_WORD_SIZE * 2;
1918 fragP->fr_var = 0;
1919 break;
1920 case GOTOFF_OFFSET:
1921 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1922 fragP->fr_offset, FALSE, BFD_RELOC_MICROBLAZE_64_GOTOFF);
1923 fragP->fr_fix += INST_WORD_SIZE * 2;
1924 fragP->fr_var = 0;
1925 break;
1926 case TLSGD_OFFSET:
1927 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1928 fragP->fr_offset, FALSE, BFD_RELOC_MICROBLAZE_64_TLSGD);
1929 fragP->fr_fix += INST_WORD_SIZE * 2;
1930 fragP->fr_var = 0;
1931 break;
1932 case TLSLD_OFFSET:
1933 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1934 fragP->fr_offset, FALSE, BFD_RELOC_MICROBLAZE_64_TLSLD);
1935 fragP->fr_fix += INST_WORD_SIZE * 2;
1936 fragP->fr_var = 0;
1937 break;
1938 case TLSDTPREL_OFFSET:
1939 fix_new (fragP, fragP->fr_fix, INST_WORD_SIZE * 2, fragP->fr_symbol,
1940 fragP->fr_offset, FALSE, BFD_RELOC_MICROBLAZE_64_TLSDTPREL);
1941 fragP->fr_fix += INST_WORD_SIZE * 2;
1942 fragP->fr_var = 0;
1943 break;
1944
1945 default:
1946 abort ();
1947 }
1948 }
1949
1950 /* Applies the desired value to the specified location.
1951 Also sets up addends for 'rela' type relocations. */
1952 void
md_apply_fix(fixS * fixP,valueT * valp,segT segment)1953 md_apply_fix (fixS * fixP,
1954 valueT * valp,
1955 segT segment)
1956 {
1957 char * buf = fixP->fx_where + fixP->fx_frag->fr_literal;
1958 char * file = fixP->fx_file ? fixP->fx_file : _("unknown");
1959 const char * symname;
1960 /* Note: use offsetT because it is signed, valueT is unsigned. */
1961 offsetT val = (offsetT) * valp;
1962 int i;
1963 struct op_code_struct * opcode1;
1964 unsigned long inst1;
1965
1966 symname = fixP->fx_addsy ? S_GET_NAME (fixP->fx_addsy) : _("<unknown>");
1967
1968 /* fixP->fx_offset is supposed to be set up correctly for all
1969 symbol relocations. */
1970 if (fixP->fx_addsy == NULL)
1971 {
1972 if (!fixP->fx_pcrel)
1973 fixP->fx_offset = val; /* Absolute relocation. */
1974 else
1975 fprintf (stderr, "NULL symbol PC-relative relocation? offset = %08x, val = %08x\n",
1976 (unsigned int) fixP->fx_offset, (unsigned int) val);
1977 }
1978
1979 /* If we aren't adjusting this fixup to be against the section
1980 symbol, we need to adjust the value. */
1981 if (fixP->fx_addsy != NULL)
1982 {
1983 if (S_IS_WEAK (fixP->fx_addsy)
1984 || (symbol_used_in_reloc_p (fixP->fx_addsy)
1985 && (((bfd_get_section_flags (stdoutput,
1986 S_GET_SEGMENT (fixP->fx_addsy))
1987 & SEC_LINK_ONCE) != 0)
1988 || !strncmp (segment_name (S_GET_SEGMENT (fixP->fx_addsy)),
1989 ".gnu.linkonce",
1990 sizeof (".gnu.linkonce") - 1))))
1991 {
1992 val -= S_GET_VALUE (fixP->fx_addsy);
1993 if (val != 0 && ! fixP->fx_pcrel)
1994 {
1995 /* In this case, the bfd_install_relocation routine will
1996 incorrectly add the symbol value back in. We just want
1997 the addend to appear in the object file.
1998 FIXME: If this makes VALUE zero, we're toast. */
1999 val -= S_GET_VALUE (fixP->fx_addsy);
2000 }
2001 }
2002 }
2003
2004 /* If the fix is relative to a symbol which is not defined, or not
2005 in the same segment as the fix, we cannot resolve it here. */
2006 /* fixP->fx_addsy is NULL if valp contains the entire relocation. */
2007 if (fixP->fx_addsy != NULL
2008 && (!S_IS_DEFINED (fixP->fx_addsy)
2009 || (S_GET_SEGMENT (fixP->fx_addsy) != segment)))
2010 {
2011 fixP->fx_done = 0;
2012 #ifdef OBJ_ELF
2013 /* For ELF we can just return and let the reloc that will be generated
2014 take care of everything. For COFF we still have to insert 'val'
2015 into the insn since the addend field will be ignored. */
2016 /* return; */
2017 #endif
2018 }
2019 /* All fixups in the text section must be handled in the linker. */
2020 else if (segment->flags & SEC_CODE)
2021 fixP->fx_done = 0;
2022 else if (!fixP->fx_pcrel && fixP->fx_addsy != NULL)
2023 fixP->fx_done = 0;
2024 else
2025 fixP->fx_done = 1;
2026
2027 switch (fixP->fx_r_type)
2028 {
2029 case BFD_RELOC_MICROBLAZE_32_LO:
2030 case BFD_RELOC_MICROBLAZE_32_LO_PCREL:
2031 if (target_big_endian)
2032 {
2033 buf[2] |= ((val >> 8) & 0xff);
2034 buf[3] |= (val & 0xff);
2035 }
2036 else
2037 {
2038 buf[1] |= ((val >> 8) & 0xff);
2039 buf[0] |= (val & 0xff);
2040 }
2041 break;
2042 case BFD_RELOC_MICROBLAZE_32_ROSDA:
2043 case BFD_RELOC_MICROBLAZE_32_RWSDA:
2044 /* Don't do anything if the symbol is not defined. */
2045 if (fixP->fx_addsy == NULL || S_IS_DEFINED (fixP->fx_addsy))
2046 {
2047 if (((val & 0xFFFF8000) != 0) && ((val & 0xFFFF8000) != 0xFFFF8000))
2048 as_bad_where (file, fixP->fx_line,
2049 _("pcrel for branch to %s too far (0x%x)"),
2050 symname, (int) val);
2051 if (target_big_endian)
2052 {
2053 buf[2] |= ((val >> 8) & 0xff);
2054 buf[3] |= (val & 0xff);
2055 }
2056 else
2057 {
2058 buf[1] |= ((val >> 8) & 0xff);
2059 buf[0] |= (val & 0xff);
2060 }
2061 }
2062 break;
2063 case BFD_RELOC_32:
2064 case BFD_RELOC_RVA:
2065 case BFD_RELOC_32_PCREL:
2066 case BFD_RELOC_MICROBLAZE_32_SYM_OP_SYM:
2067 /* Don't do anything if the symbol is not defined. */
2068 if (fixP->fx_addsy == NULL || S_IS_DEFINED (fixP->fx_addsy))
2069 {
2070 if (target_big_endian)
2071 {
2072 buf[0] |= ((val >> 24) & 0xff);
2073 buf[1] |= ((val >> 16) & 0xff);
2074 buf[2] |= ((val >> 8) & 0xff);
2075 buf[3] |= (val & 0xff);
2076 }
2077 else
2078 {
2079 buf[3] |= ((val >> 24) & 0xff);
2080 buf[2] |= ((val >> 16) & 0xff);
2081 buf[1] |= ((val >> 8) & 0xff);
2082 buf[0] |= (val & 0xff);
2083 }
2084 }
2085 break;
2086 case BFD_RELOC_64_PCREL:
2087 case BFD_RELOC_64:
2088 /* Add an imm instruction. First save the current instruction. */
2089 for (i = 0; i < INST_WORD_SIZE; i++)
2090 buf[i + INST_WORD_SIZE] = buf[i];
2091
2092 /* Generate the imm instruction. */
2093 opcode1 = (struct op_code_struct *) hash_find (opcode_hash_control, "imm");
2094 if (opcode1 == NULL)
2095 {
2096 as_bad (_("unknown opcode \"%s\""), "imm");
2097 return;
2098 }
2099
2100 inst1 = opcode1->bit_sequence;
2101 if (fixP->fx_addsy == NULL || S_IS_DEFINED (fixP->fx_addsy))
2102 inst1 |= ((val & 0xFFFF0000) >> 16) & IMM_MASK;
2103
2104 buf[0] = INST_BYTE0 (inst1);
2105 buf[1] = INST_BYTE1 (inst1);
2106 buf[2] = INST_BYTE2 (inst1);
2107 buf[3] = INST_BYTE3 (inst1);
2108
2109 /* Add the value only if the symbol is defined. */
2110 if (fixP->fx_addsy == NULL || S_IS_DEFINED (fixP->fx_addsy))
2111 {
2112 if (target_big_endian)
2113 {
2114 buf[6] |= ((val >> 8) & 0xff);
2115 buf[7] |= (val & 0xff);
2116 }
2117 else
2118 {
2119 buf[5] |= ((val >> 8) & 0xff);
2120 buf[4] |= (val & 0xff);
2121 }
2122 }
2123 break;
2124
2125 case BFD_RELOC_MICROBLAZE_64_TLSDTPREL:
2126 case BFD_RELOC_MICROBLAZE_64_TLSGD:
2127 case BFD_RELOC_MICROBLAZE_64_TLSLD:
2128 S_SET_THREAD_LOCAL (fixP->fx_addsy);
2129
2130 case BFD_RELOC_MICROBLAZE_64_GOTPC:
2131 case BFD_RELOC_MICROBLAZE_64_GOT:
2132 case BFD_RELOC_MICROBLAZE_64_PLT:
2133 case BFD_RELOC_MICROBLAZE_64_GOTOFF:
2134 /* Add an imm instruction. First save the current instruction. */
2135 for (i = 0; i < INST_WORD_SIZE; i++)
2136 buf[i + INST_WORD_SIZE] = buf[i];
2137
2138 /* Generate the imm instruction. */
2139 opcode1 = (struct op_code_struct *) hash_find (opcode_hash_control, "imm");
2140 if (opcode1 == NULL)
2141 {
2142 as_bad (_("unknown opcode \"%s\""), "imm");
2143 return;
2144 }
2145
2146 inst1 = opcode1->bit_sequence;
2147
2148 /* We can fixup call to a defined non-global address
2149 within the same section only. */
2150 buf[0] = INST_BYTE0 (inst1);
2151 buf[1] = INST_BYTE1 (inst1);
2152 buf[2] = INST_BYTE2 (inst1);
2153 buf[3] = INST_BYTE3 (inst1);
2154 return;
2155
2156 default:
2157 break;
2158 }
2159
2160 if (fixP->fx_addsy == NULL)
2161 {
2162 /* This fixup has been resolved. Create a reloc in case the linker
2163 moves code around due to relaxing. */
2164 if (fixP->fx_r_type == BFD_RELOC_64_PCREL)
2165 fixP->fx_r_type = BFD_RELOC_MICROBLAZE_64_NONE;
2166 else
2167 fixP->fx_r_type = BFD_RELOC_NONE;
2168 fixP->fx_addsy = section_symbol (absolute_section);
2169 }
2170 return;
2171 }
2172
2173 void
md_operand(expressionS * expressionP)2174 md_operand (expressionS * expressionP)
2175 {
2176 /* Ignore leading hash symbol, if present. */
2177 if (*input_line_pointer == '#')
2178 {
2179 input_line_pointer ++;
2180 expression (expressionP);
2181 }
2182 }
2183
2184 /* Called just before address relaxation, return the length
2185 by which a fragment must grow to reach it's destination. */
2186
2187 int
md_estimate_size_before_relax(fragS * fragP,segT segment_type)2188 md_estimate_size_before_relax (fragS * fragP,
2189 segT segment_type)
2190 {
2191 sbss_segment = bfd_get_section_by_name (stdoutput, ".sbss");
2192 sbss2_segment = bfd_get_section_by_name (stdoutput, ".sbss2");
2193 sdata_segment = bfd_get_section_by_name (stdoutput, ".sdata");
2194 sdata2_segment = bfd_get_section_by_name (stdoutput, ".sdata2");
2195
2196 switch (fragP->fr_subtype)
2197 {
2198 case INST_PC_OFFSET:
2199 /* Used to be a PC-relative branch. */
2200 if (!fragP->fr_symbol)
2201 {
2202 /* We know the abs value: Should never happen. */
2203 as_bad (_("Absolute PC-relative value in relaxation code. Assembler error....."));
2204 abort ();
2205 }
2206 else if (S_GET_SEGMENT (fragP->fr_symbol) == segment_type &&
2207 !S_IS_WEAK (fragP->fr_symbol))
2208 {
2209 fragP->fr_subtype = DEFINED_PC_OFFSET;
2210 /* Don't know now whether we need an imm instruction. */
2211 fragP->fr_var = INST_WORD_SIZE;
2212 }
2213 else if (S_IS_DEFINED (fragP->fr_symbol)
2214 && (((S_GET_SEGMENT (fragP->fr_symbol))->flags & SEC_CODE) == 0))
2215 {
2216 /* Cannot have a PC-relative branch to a diff segment. */
2217 as_bad (_("PC relative branch to label %s which is not in the instruction space"),
2218 S_GET_NAME (fragP->fr_symbol));
2219 fragP->fr_subtype = UNDEFINED_PC_OFFSET;
2220 fragP->fr_var = INST_WORD_SIZE*2;
2221 }
2222 else
2223 {
2224 fragP->fr_subtype = UNDEFINED_PC_OFFSET;
2225 fragP->fr_var = INST_WORD_SIZE*2;
2226 }
2227 break;
2228
2229 case INST_NO_OFFSET:
2230 /* Used to be a reference to somewhere which was unknown. */
2231 if (fragP->fr_symbol)
2232 {
2233 if (fragP->fr_opcode == NULL)
2234 {
2235 /* Used as an absolute value. */
2236 fragP->fr_subtype = DEFINED_ABS_SEGMENT;
2237 /* Variable part does not change. */
2238 fragP->fr_var = INST_WORD_SIZE*2;
2239 }
2240 else if (streq (fragP->fr_opcode, str_microblaze_ro_anchor))
2241 {
2242 /* It is accessed using the small data read only anchor. */
2243 if ((S_GET_SEGMENT (fragP->fr_symbol) == bfd_com_section_ptr)
2244 || (S_GET_SEGMENT (fragP->fr_symbol) == sdata2_segment)
2245 || (S_GET_SEGMENT (fragP->fr_symbol) == sbss2_segment)
2246 || (! S_IS_DEFINED (fragP->fr_symbol)))
2247 {
2248 fragP->fr_subtype = DEFINED_RO_SEGMENT;
2249 fragP->fr_var = INST_WORD_SIZE;
2250 }
2251 else
2252 {
2253 /* Variable not in small data read only segment accessed
2254 using small data read only anchor. */
2255 char *file = fragP->fr_file ? fragP->fr_file : _("unknown");
2256
2257 as_bad_where (file, fragP->fr_line,
2258 _("Variable is accessed using small data read "
2259 "only anchor, but it is not in the small data "
2260 "read only section"));
2261 fragP->fr_subtype = DEFINED_RO_SEGMENT;
2262 fragP->fr_var = INST_WORD_SIZE;
2263 }
2264 }
2265 else if (streq (fragP->fr_opcode, str_microblaze_rw_anchor))
2266 {
2267 if ((S_GET_SEGMENT (fragP->fr_symbol) == bfd_com_section_ptr)
2268 || (S_GET_SEGMENT (fragP->fr_symbol) == sdata_segment)
2269 || (S_GET_SEGMENT (fragP->fr_symbol) == sbss_segment)
2270 || (!S_IS_DEFINED (fragP->fr_symbol)))
2271 {
2272 /* It is accessed using the small data read write anchor. */
2273 fragP->fr_subtype = DEFINED_RW_SEGMENT;
2274 fragP->fr_var = INST_WORD_SIZE;
2275 }
2276 else
2277 {
2278 char *file = fragP->fr_file ? fragP->fr_file : _("unknown");
2279
2280 as_bad_where (file, fragP->fr_line,
2281 _("Variable is accessed using small data read "
2282 "write anchor, but it is not in the small data "
2283 "read write section"));
2284 fragP->fr_subtype = DEFINED_RW_SEGMENT;
2285 fragP->fr_var = INST_WORD_SIZE;
2286 }
2287 }
2288 else
2289 {
2290 as_bad (_("Incorrect fr_opcode value in frag. Internal error....."));
2291 abort ();
2292 }
2293 }
2294 else
2295 {
2296 /* We know the abs value: Should never happen. */
2297 as_bad (_("Absolute value in relaxation code. Assembler error....."));
2298 abort ();
2299 }
2300 break;
2301
2302 case UNDEFINED_PC_OFFSET:
2303 case LARGE_DEFINED_PC_OFFSET:
2304 case DEFINED_ABS_SEGMENT:
2305 case GOT_OFFSET:
2306 case PLT_OFFSET:
2307 case GOTOFF_OFFSET:
2308 case TLSGD_OFFSET:
2309 case TLSLD_OFFSET:
2310 case TLSTPREL_OFFSET:
2311 case TLSDTPREL_OFFSET:
2312 fragP->fr_var = INST_WORD_SIZE*2;
2313 break;
2314 case DEFINED_RO_SEGMENT:
2315 case DEFINED_RW_SEGMENT:
2316 case DEFINED_PC_OFFSET:
2317 case TLSDTPMOD_OFFSET:
2318 fragP->fr_var = INST_WORD_SIZE;
2319 break;
2320 default:
2321 abort ();
2322 }
2323
2324 return fragP->fr_var;
2325 }
2326
2327 /* Put number into target byte order. */
2328
2329 void
md_number_to_chars(char * ptr,valueT use,int nbytes)2330 md_number_to_chars (char * ptr, valueT use, int nbytes)
2331 {
2332 if (target_big_endian)
2333 number_to_chars_bigendian (ptr, use, nbytes);
2334 else
2335 number_to_chars_littleendian (ptr, use, nbytes);
2336 }
2337
2338 /* Round up a section size to the appropriate boundary. */
2339
2340 valueT
md_section_align(segT segment ATTRIBUTE_UNUSED,valueT size)2341 md_section_align (segT segment ATTRIBUTE_UNUSED, valueT size)
2342 {
2343 return size; /* Byte alignment is fine. */
2344 }
2345
2346
2347 /* The location from which a PC relative jump should be calculated,
2348 given a PC relative reloc. */
2349
2350 long
md_pcrel_from_section(fixS * fixp,segT sec ATTRIBUTE_UNUSED)2351 md_pcrel_from_section (fixS * fixp, segT sec ATTRIBUTE_UNUSED)
2352 {
2353 #ifdef OBJ_ELF
2354 /* If the symbol is undefined or defined in another section
2355 we leave the add number alone for the linker to fix it later.
2356 Only account for the PC pre-bump (No PC-pre-bump on the Microblaze). */
2357
2358 if (fixp->fx_addsy != (symbolS *) NULL
2359 && (!S_IS_DEFINED (fixp->fx_addsy)
2360 || (S_GET_SEGMENT (fixp->fx_addsy) != sec)))
2361 return 0;
2362 else
2363 {
2364 /* The case where we are going to resolve things... */
2365 if (fixp->fx_r_type == BFD_RELOC_64_PCREL)
2366 return fixp->fx_where + fixp->fx_frag->fr_address + INST_WORD_SIZE;
2367 else
2368 return fixp->fx_where + fixp->fx_frag->fr_address;
2369 }
2370 #endif
2371 }
2372
2373
2374 #define F(SZ,PCREL) (((SZ) << 1) + (PCREL))
2375 #define MAP(SZ,PCREL,TYPE) case F (SZ, PCREL): code = (TYPE); break
2376
2377 arelent *
tc_gen_reloc(asection * section ATTRIBUTE_UNUSED,fixS * fixp)2378 tc_gen_reloc (asection * section ATTRIBUTE_UNUSED, fixS * fixp)
2379 {
2380 arelent * rel;
2381 bfd_reloc_code_real_type code;
2382
2383 switch (fixp->fx_r_type)
2384 {
2385 case BFD_RELOC_NONE:
2386 case BFD_RELOC_MICROBLAZE_64_NONE:
2387 case BFD_RELOC_32:
2388 case BFD_RELOC_MICROBLAZE_32_LO:
2389 case BFD_RELOC_MICROBLAZE_32_LO_PCREL:
2390 case BFD_RELOC_RVA:
2391 case BFD_RELOC_64:
2392 case BFD_RELOC_64_PCREL:
2393 case BFD_RELOC_MICROBLAZE_32_ROSDA:
2394 case BFD_RELOC_MICROBLAZE_32_RWSDA:
2395 case BFD_RELOC_MICROBLAZE_32_SYM_OP_SYM:
2396 case BFD_RELOC_MICROBLAZE_64_GOTPC:
2397 case BFD_RELOC_MICROBLAZE_64_GOT:
2398 case BFD_RELOC_MICROBLAZE_64_PLT:
2399 case BFD_RELOC_MICROBLAZE_64_GOTOFF:
2400 case BFD_RELOC_MICROBLAZE_32_GOTOFF:
2401 case BFD_RELOC_MICROBLAZE_64_TLSGD:
2402 case BFD_RELOC_MICROBLAZE_64_TLSLD:
2403 case BFD_RELOC_MICROBLAZE_32_TLSDTPMOD:
2404 case BFD_RELOC_MICROBLAZE_32_TLSDTPREL:
2405 case BFD_RELOC_MICROBLAZE_64_TLSDTPREL:
2406 case BFD_RELOC_MICROBLAZE_64_TLSGOTTPREL:
2407 case BFD_RELOC_MICROBLAZE_64_TLSTPREL:
2408 code = fixp->fx_r_type;
2409 break;
2410
2411 default:
2412 switch (F (fixp->fx_size, fixp->fx_pcrel))
2413 {
2414 MAP (1, 0, BFD_RELOC_8);
2415 MAP (2, 0, BFD_RELOC_16);
2416 MAP (4, 0, BFD_RELOC_32);
2417 MAP (1, 1, BFD_RELOC_8_PCREL);
2418 MAP (2, 1, BFD_RELOC_16_PCREL);
2419 MAP (4, 1, BFD_RELOC_32_PCREL);
2420 default:
2421 code = fixp->fx_r_type;
2422 as_bad (_("Can not do %d byte %srelocation"),
2423 fixp->fx_size,
2424 fixp->fx_pcrel ? _("pc-relative") : "");
2425 }
2426 break;
2427 }
2428
2429 rel = (arelent *) xmalloc (sizeof (arelent));
2430 rel->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
2431
2432 if (code == BFD_RELOC_MICROBLAZE_32_SYM_OP_SYM)
2433 *rel->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_subsy);
2434 else
2435 *rel->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
2436
2437 rel->address = fixp->fx_frag->fr_address + fixp->fx_where;
2438 /* Always pass the addend along! */
2439 rel->addend = fixp->fx_offset;
2440 rel->howto = bfd_reloc_type_lookup (stdoutput, code);
2441
2442 if (rel->howto == NULL)
2443 {
2444 as_bad_where (fixp->fx_file, fixp->fx_line,
2445 _("Cannot represent relocation type %s"),
2446 bfd_get_reloc_code_name (code));
2447
2448 /* Set howto to a garbage value so that we can keep going. */
2449 rel->howto = bfd_reloc_type_lookup (stdoutput, BFD_RELOC_32);
2450 gas_assert (rel->howto != NULL);
2451 }
2452 return rel;
2453 }
2454
2455 int
md_parse_option(int c,char * arg ATTRIBUTE_UNUSED)2456 md_parse_option (int c, char * arg ATTRIBUTE_UNUSED)
2457 {
2458 switch (c)
2459 {
2460 case OPTION_EB:
2461 target_big_endian = 1;
2462 break;
2463 case OPTION_EL:
2464 target_big_endian = 0;
2465 break;
2466 default:
2467 return 0;
2468 }
2469 return 1;
2470 }
2471
2472 void
md_show_usage(FILE * stream ATTRIBUTE_UNUSED)2473 md_show_usage (FILE * stream ATTRIBUTE_UNUSED)
2474 {
2475 /* fprintf(stream, _("\
2476 MicroBlaze options:\n\
2477 -noSmall Data in the comm and data sections do not go into the small data section\n")); */
2478 }
2479
2480
2481 /* Create a fixup for a cons expression. If parse_cons_expression_microblaze
2482 found a machine specific op in an expression,
2483 then we create relocs accordingly. */
2484
2485 void
cons_fix_new_microblaze(fragS * frag,int where,int size,expressionS * exp,bfd_reloc_code_real_type r)2486 cons_fix_new_microblaze (fragS * frag,
2487 int where,
2488 int size,
2489 expressionS *exp,
2490 bfd_reloc_code_real_type r)
2491 {
2492 if ((exp->X_op == O_subtract) && (exp->X_add_symbol) &&
2493 (exp->X_op_symbol) && (now_seg != absolute_section) && (size == 4)
2494 && (!S_IS_LOCAL (exp->X_op_symbol)))
2495 r = BFD_RELOC_MICROBLAZE_32_SYM_OP_SYM;
2496 else if (exp->X_md == IMM_GOTOFF && exp->X_op == O_symbol_rva)
2497 {
2498 exp->X_op = O_symbol;
2499 r = BFD_RELOC_MICROBLAZE_32_GOTOFF;
2500 }
2501 else
2502 {
2503 switch (size)
2504 {
2505 case 1:
2506 r = BFD_RELOC_8;
2507 break;
2508 case 2:
2509 r = BFD_RELOC_16;
2510 break;
2511 case 4:
2512 r = BFD_RELOC_32;
2513 break;
2514 case 8:
2515 r = BFD_RELOC_64;
2516 break;
2517 default:
2518 as_bad (_("unsupported BFD relocation size %u"), size);
2519 r = BFD_RELOC_32;
2520 break;
2521 }
2522 }
2523 fix_new_exp (frag, where, size, exp, 0, r);
2524 }
2525