source: trunk/binutils/gas/config/tc-pj.c@ 3616

Last change on this file since 3616 was 610, checked in by bird, 22 years ago

This commit was generated by cvs2svn to compensate for changes in r609,
which included commits to RCS files with non-trunk default branches.

  • Property cvs2svn:cvs-rev set to 1.1.1.2
  • Property svn:eol-style set to native
  • Property svn:executable set to *
File size: 12.9 KB
Line 
1/*-
2 tc-pj.c -- Assemble code for Pico Java
3 Copyright 1999, 2000, 2001, 2002 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 2, 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
19 the Free Software Foundation, 59 Temple Place - Suite 330,
20 Boston, MA 02111-1307, USA. */
21
22/* Contributed by Steve Chamberlain of Transmeta <sac@pobox.com>. */
23
24#include "as.h"
25#include "safe-ctype.h"
26#include "opcode/pj.h"
27
28extern const pj_opc_info_t pj_opc_info[512];
29
30const char comment_chars[] = "!/";
31const char line_separator_chars[] = ";";
32const char line_comment_chars[] = "/!#";
33
34static int pending_reloc;
35static struct hash_control *opcode_hash_control;
36
37static void little
38 PARAMS ((int));
39static void big
40 PARAMS ((int));
41static char *parse_exp_save_ilp
42 PARAMS ((char *, expressionS *));
43static int c_to_r
44 PARAMS ((char));
45static void ipush_code
46 PARAMS ((pj_opc_info_t *, char *));
47static void fake_opcode
48 PARAMS ((const char *, void (*) (struct pj_opc_info_t *, char *)));
49static void alias
50 PARAMS ((const char *, const char *));
51
52static void
53little (ignore)
54 int ignore ATTRIBUTE_UNUSED;
55{
56 target_big_endian = 0;
57}
58
59static void
60big (ignore)
61 int ignore ATTRIBUTE_UNUSED;
62{
63 target_big_endian = 1;
64}
65
66const pseudo_typeS md_pseudo_table[] = {
67 {"ml", little, 0},
68 {"mb", big, 0},
69 {0, 0, 0}
70};
71
72const char FLT_CHARS[] = "rRsSfFdDxXpP";
73const char EXP_CHARS[] = "eE";
74
75void
76md_operand (op)
77 expressionS *op;
78{
79 if (strncmp (input_line_pointer, "%hi16", 5) == 0)
80 {
81 if (pending_reloc)
82 as_bad (_("confusing relocation expressions"));
83 pending_reloc = BFD_RELOC_PJ_CODE_HI16;
84 input_line_pointer += 5;
85 expression (op);
86 }
87 if (strncmp (input_line_pointer, "%lo16", 5) == 0)
88 {
89 if (pending_reloc)
90 as_bad (_("confusing relocation expressions"));
91 pending_reloc = BFD_RELOC_PJ_CODE_LO16;
92 input_line_pointer += 5;
93 expression (op);
94 }
95}
96
97/* Parse an expression and then restore the input line pointer. */
98
99static char *
100parse_exp_save_ilp (s, op)
101 char *s;
102 expressionS *op;
103{
104 char *save = input_line_pointer;
105 input_line_pointer = s;
106 expression (op);
107 s = input_line_pointer;
108 input_line_pointer = save;
109 return s;
110}
111
112/* This is called by emit_expr via TC_CONS_FIX_NEW when creating a
113 reloc for a cons. We could use the definition there, except that
114 we want to handle magic pending reloc expressions specially. */
115
116void
117pj_cons_fix_new_pj (frag, where, nbytes, exp)
118 fragS *frag;
119 int where;
120 int nbytes;
121 expressionS *exp;
122{
123 static int rv[5][2] =
124 { { 0, 0 },
125 { BFD_RELOC_8, BFD_RELOC_8 },
126 { BFD_RELOC_PJ_CODE_DIR16, BFD_RELOC_16 },
127 { 0, 0 },
128 { BFD_RELOC_PJ_CODE_DIR32, BFD_RELOC_32 }};
129
130 fix_new_exp (frag, where, nbytes, exp, 0,
131 pending_reloc ? pending_reloc
132 : rv[nbytes][(now_seg->flags & SEC_CODE) ? 0 : 1]);
133
134 pending_reloc = 0;
135}
136
137/* Turn a reloc description character from the pj-opc.h table into
138 code which BFD can handle. */
139
140static int
141c_to_r (x)
142 char x;
143{
144 switch (x)
145 {
146 case O_R8:
147 return BFD_RELOC_8_PCREL;
148 case O_U8:
149 case O_8:
150 return BFD_RELOC_8;
151 case O_R16:
152 return BFD_RELOC_PJ_CODE_REL16;
153 case O_U16:
154 case O_16:
155 return BFD_RELOC_PJ_CODE_DIR16;
156 case O_R32:
157 return BFD_RELOC_PJ_CODE_REL32;
158 case O_32:
159 return BFD_RELOC_PJ_CODE_DIR32;
160 }
161 abort ();
162 return 0;
163}
164
165/* Handler for the ipush fake opcode,
166 turns ipush <foo> into sipush lo16<foo>, sethi hi16<foo>. */
167
168static void
169ipush_code (opcode, str)
170 pj_opc_info_t *opcode ATTRIBUTE_UNUSED;
171 char *str;
172{
173 char *b = frag_more (6);
174 expressionS arg;
175
176 b[0] = 0x11;
177 b[3] = 0xed;
178 parse_exp_save_ilp (str + 1, &arg);
179 if (pending_reloc)
180 {
181 as_bad (_("can't have relocation for ipush"));
182 pending_reloc = 0;
183 }
184
185 fix_new_exp (frag_now, b - frag_now->fr_literal + 1, 2,
186 &arg, 0, BFD_RELOC_PJ_CODE_DIR16);
187 fix_new_exp (frag_now, b - frag_now->fr_literal + 4, 2,
188 &arg, 0, BFD_RELOC_PJ_CODE_HI16);
189}
190
191/* Insert names into the opcode table which are really mini macros,
192 not opcodes. The fakeness is inidicated with an opcode of -1. */
193
194static void
195fake_opcode (name, func)
196 const char *name;
197 void (*func) PARAMS ((struct pj_opc_info_t *, char *));
198{
199 pj_opc_info_t *fake = (pj_opc_info_t *) xmalloc (sizeof (pj_opc_info_t));
200
201 fake->opcode = -1;
202 fake->opcode_next = -1;
203 fake->u.func = func;
204 hash_insert (opcode_hash_control, name, (char *) fake);
205}
206
207/* Enter another entry into the opcode hash table so the same opcode
208 can have another name. */
209
210static void
211alias (new, old)
212 const char *new;
213 const char *old;
214{
215 hash_insert (opcode_hash_control, new,
216 (char *) hash_find (opcode_hash_control, old));
217}
218
219/* This function is called once, at assembler startup time. It sets
220 up the hash table with all the opcodes in it, and also initializes
221 some aliases for compatibility with other assemblers. */
222
223void
224md_begin ()
225{
226 const pj_opc_info_t *opcode;
227 opcode_hash_control = hash_new ();
228
229 /* Insert names into hash table. */
230 for (opcode = pj_opc_info; opcode->u.name; opcode++)
231 hash_insert (opcode_hash_control, opcode->u.name, (char *) opcode);
232
233 /* Insert the only fake opcode. */
234 fake_opcode ("ipush", ipush_code);
235
236 /* Add some aliases for opcode names. */
237 alias ("ifeq_s", "ifeq");
238 alias ("ifne_s", "ifne");
239 alias ("if_icmpge_s", "if_icmpge");
240 alias ("if_icmpne_s", "if_icmpne");
241 alias ("if_icmpeq_s", "if_icmpeq");
242 alias ("if_icmpgt_s", "if_icmpgt");
243 alias ("goto_s", "goto");
244
245 bfd_set_arch_mach (stdoutput, TARGET_ARCH, 0);
246}
247
248/* This is the guts of the machine-dependent assembler. STR points to
249 a machine dependent instruction. This function is supposed to emit
250 the frags/bytes it assembles to. */
251
252void
253md_assemble (str)
254 char *str;
255{
256 unsigned char *op_start;
257 unsigned char *op_end;
258
259#if 0
260 pj_operan_info operand[3];
261#endif
262 pj_opc_info_t *opcode;
263 char *output;
264 int idx = 0;
265 char pend;
266
267 int nlen = 0;
268
269 /* Drop leading whitespace. */
270 while (*str == ' ')
271 str++;
272
273 /* Find the op code end. */
274 for (op_start = op_end = (unsigned char *) (str);
275 *op_end && !is_end_of_line[*op_end] && *op_end != ' ';
276 op_end++)
277 nlen++;
278
279 pend = *op_end;
280 *op_end = 0;
281
282 if (nlen == 0)
283 as_bad (_("can't find opcode "));
284
285 opcode = (pj_opc_info_t *) hash_find (opcode_hash_control, op_start);
286 *op_end = pend;
287
288 if (opcode == NULL)
289 {
290 as_bad (_("unknown opcode %s"), op_start);
291 return;
292 }
293
294 if (opcode->opcode == -1)
295 {
296 /* It's a fake opcode. Dig out the args and pretend that was
297 what we were passed. */
298 (*opcode->u.func) (opcode, op_end);
299 }
300 else
301 {
302 int an;
303
304 output = frag_more (opcode->len);
305 output[idx++] = opcode->opcode;
306
307 if (opcode->opcode_next != -1)
308 output[idx++] = opcode->opcode_next;
309
310 for (an = 0; opcode->arg[an]; an++)
311 {
312 expressionS arg;
313
314 if (*op_end == ',' && an != 0)
315 op_end++;
316
317 if (*op_end == 0)
318 as_bad ("expected expresssion");
319
320 op_end = parse_exp_save_ilp (op_end, &arg);
321
322 fix_new_exp (frag_now,
323 output - frag_now->fr_literal + idx,
324 ASIZE (opcode->arg[an]),
325 &arg,
326 PCREL (opcode->arg[an]),
327 pending_reloc ? pending_reloc : c_to_r (opcode->arg[an]));
328
329 idx += ASIZE (opcode->arg[an]);
330 pending_reloc = 0;
331 }
332
333 while (ISSPACE (*op_end))
334 op_end++;
335
336 if (*op_end != 0)
337 as_warn ("extra stuff on line ignored");
338
339 }
340
341 if (pending_reloc)
342 as_bad ("Something forgot to clean up\n");
343
344}
345
346/* Turn a string in input_line_pointer into a floating point constant
347 of type type, and store the appropriate bytes in *LITP. The number
348 of LITTLENUMS emitted is stored in *SIZEP . An error message is
349 returned, or NULL on OK. */
350
351char *
352md_atof (type, litP, sizeP)
353 int type;
354 char *litP;
355 int *sizeP;
356{
357 int prec;
358 LITTLENUM_TYPE words[4];
359 char *t;
360 int i;
361
362 switch (type)
363 {
364 case 'f':
365 prec = 2;
366 break;
367
368 case 'd':
369 prec = 4;
370 break;
371
372 default:
373 *sizeP = 0;
374 return _("bad call to md_atof");
375 }
376
377 t = atof_ieee (input_line_pointer, type, words);
378 if (t)
379 input_line_pointer = t;
380
381 *sizeP = prec * 2;
382
383 if (!target_big_endian)
384 {
385 for (i = prec - 1; i >= 0; i--)
386 {
387 md_number_to_chars (litP, (valueT) words[i], 2);
388 litP += 2;
389 }
390 }
391 else
392 {
393 for (i = 0; i < prec; i++)
394 {
395 md_number_to_chars (litP, (valueT) words[i], 2);
396 litP += 2;
397 }
398 }
399
400 return NULL;
401}
402
403
404const char *md_shortopts = "";
405
406struct option md_longopts[] = {
407
408#define OPTION_LITTLE (OPTION_MD_BASE)
409#define OPTION_BIG (OPTION_LITTLE + 1)
410
411 {"little", no_argument, NULL, OPTION_LITTLE},
412 {"big", no_argument, NULL, OPTION_BIG},
413 {NULL, no_argument, NULL, 0}
414};
415size_t md_longopts_size = sizeof (md_longopts);
416
417int
418md_parse_option (c, arg)
419 int c;
420 char *arg ATTRIBUTE_UNUSED;
421{
422 switch (c)
423 {
424 case OPTION_LITTLE:
425 little (0);
426 break;
427 case OPTION_BIG:
428 big (0);
429 break;
430 default:
431 return 0;
432 }
433 return 1;
434}
435
436void
437md_show_usage (stream)
438 FILE *stream;
439{
440 fprintf (stream, _("\
441PJ options:\n\
442-little generate little endian code\n\
443-big generate big endian code\n"));
444}
445
446/* Apply a fixup to the object file. */
447
448void
449md_apply_fix3 (fixP, valP, seg)
450 fixS *fixP;
451 valueT * valP;
452 segT seg ATTRIBUTE_UNUSED;
453{
454 char *buf = fixP->fx_where + fixP->fx_frag->fr_literal;
455 long val = *valP;
456 long max, min;
457 int shift;
458
459 max = min = 0;
460 shift = 0;
461 switch (fixP->fx_r_type)
462 {
463 case BFD_RELOC_VTABLE_INHERIT:
464 case BFD_RELOC_VTABLE_ENTRY:
465 fixP->fx_done = 0;
466 return;
467
468 case BFD_RELOC_PJ_CODE_REL16:
469 if (val < -0x8000 || val >= 0x7fff)
470 as_bad_where (fixP->fx_file, fixP->fx_line, _("pcrel too far"));
471 buf[0] |= (val >> 8) & 0xff;
472 buf[1] = val & 0xff;
473 break;
474
475 case BFD_RELOC_PJ_CODE_HI16:
476 *buf++ = val >> 24;
477 *buf++ = val >> 16;
478 fixP->fx_addnumber = val & 0xffff;
479 break;
480
481 case BFD_RELOC_PJ_CODE_DIR16:
482 case BFD_RELOC_PJ_CODE_LO16:
483 *buf++ = val >> 8;
484 *buf++ = val >> 0;
485
486 max = 0xffff;
487 min = -0xffff;
488 break;
489
490 case BFD_RELOC_8:
491 max = 0xff;
492 min = -0xff;
493 *buf++ = val;
494 break;
495
496 case BFD_RELOC_PJ_CODE_DIR32:
497 *buf++ = val >> 24;
498 *buf++ = val >> 16;
499 *buf++ = val >> 8;
500 *buf++ = val >> 0;
501 break;
502
503 case BFD_RELOC_32:
504 if (target_big_endian)
505 {
506 *buf++ = val >> 24;
507 *buf++ = val >> 16;
508 *buf++ = val >> 8;
509 *buf++ = val >> 0;
510 }
511 else
512 {
513 *buf++ = val >> 0;
514 *buf++ = val >> 8;
515 *buf++ = val >> 16;
516 *buf++ = val >> 24;
517 }
518 break;
519
520 case BFD_RELOC_16:
521 if (target_big_endian)
522 {
523 *buf++ = val >> 8;
524 *buf++ = val >> 0;
525 }
526 else
527 {
528 *buf++ = val >> 0;
529 *buf++ = val >> 8;
530 }
531 break;
532
533 default:
534 abort ();
535 }
536
537 if (max != 0 && (val < min || val > max))
538 as_bad_where (fixP->fx_file, fixP->fx_line, _("offset out of range"));
539
540 if (fixP->fx_addsy == NULL && fixP->fx_pcrel == 0)
541 fixP->fx_done = 1;
542}
543
544/* Put number into target byte order. Always put values in an
545 executable section into big endian order. */
546
547void
548md_number_to_chars (ptr, use, nbytes)
549 char *ptr;
550 valueT use;
551 int nbytes;
552{
553 if (target_big_endian || now_seg->flags & SEC_CODE)
554 number_to_chars_bigendian (ptr, use, nbytes);
555 else
556 number_to_chars_littleendian (ptr, use, nbytes);
557}
558
559/* Translate internal representation of relocation info to BFD target
560 format. */
561
562arelent *
563tc_gen_reloc (section, fixp)
564 asection *section ATTRIBUTE_UNUSED;
565 fixS *fixp;
566{
567 arelent *rel;
568 bfd_reloc_code_real_type r_type;
569
570 rel = (arelent *) xmalloc (sizeof (arelent));
571 rel->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
572 *rel->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
573 rel->address = fixp->fx_frag->fr_address + fixp->fx_where;
574
575 r_type = fixp->fx_r_type;
576 rel->addend = fixp->fx_addnumber;
577 rel->howto = bfd_reloc_type_lookup (stdoutput, r_type);
578
579 if (rel->howto == NULL)
580 {
581 as_bad_where (fixp->fx_file, fixp->fx_line,
582 _("Cannot represent relocation type %s"),
583 bfd_get_reloc_code_name (r_type));
584 /* Set howto to a garbage value so that we can keep going. */
585 rel->howto = bfd_reloc_type_lookup (stdoutput, BFD_RELOC_32);
586 assert (rel->howto != NULL);
587 }
588
589 return rel;
590}
Note: See TracBrowser for help on using the repository browser.