1 /* tc-tic4x.c -- Assemble for the Texas Instruments TMS320C[34]x.
2 Copyright (C) 1997-2014 Free Software Foundation, Inc.
3
4 Contributed by Michael P. Hayes (m.hayes@elec.canterbury.ac.nz)
5
6 This file is part of GAS, the GNU Assembler.
7
8 GAS is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3, or (at your option)
11 any later version.
12
13 GAS is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with GAS; see the file COPYING. If not, write to
20 the Free Software Foundation, 51 Franklin Street - Fifth Floor,
21 Boston, MA 02110-1301, USA. */
22 /*
23 TODOs:
24 ------
25
26 o .align cannot handle fill-data-width larger than 0xFF/8-bits. It
27 should be possible to define a 32-bits pattern.
28
29 o .align: Implement a 'bu' insn if the number of nop's exceeds 4
30 within the align frag. if(fragsize>4words) insert bu fragend+1
31 first.
32
33 o .usect if has symbol on previous line not implemented
34
35 o .sym, .eos, .stag, .etag, .member not implemented
36
37 o Evaluation of constant floating point expressions (expr.c needs
38 work!)
39
40 o Support 'abc' constants (that is 0x616263). */
41
42 #include "as.h"
43 #include "safe-ctype.h"
44 #include "opcode/tic4x.h"
45 #include "subsegs.h"
46
47 /* OK, we accept a syntax similar to the other well known C30
48 assembly tools. With TIC4X_ALT_SYNTAX defined we are more
49 flexible, allowing a more Unix-like syntax: `%' in front of
50 register names, `#' in front of immediate constants, and
51 not requiring `@' in front of direct addresses. */
52
53 #define TIC4X_ALT_SYNTAX
54
55 /* Equal to MAX_PRECISION in atof-ieee.c. */
56 #define MAX_LITTLENUMS 6 /* (12 bytes) */
57
58 /* Handle of the inst mnemonic hash table. */
59 static struct hash_control *tic4x_op_hash = NULL;
60
61 /* Handle asg pseudo. */
62 static struct hash_control *tic4x_asg_hash = NULL;
63
64 static unsigned int tic4x_cpu = 0; /* Default to TMS320C40. */
65 static unsigned int tic4x_revision = 0; /* CPU revision */
66 static unsigned int tic4x_idle2 = 0; /* Idle2 support */
67 static unsigned int tic4x_lowpower = 0; /* Lowpower support */
68 static unsigned int tic4x_enhanced = 0; /* Enhanced opcode support */
69 static unsigned int tic4x_big_model = 0; /* Default to small memory model. */
70 static unsigned int tic4x_reg_args = 0; /* Default to args passed on stack. */
71 static unsigned long tic4x_oplevel = 0; /* Opcode level */
72
73 #define OPTION_CPU 'm'
74 #define OPTION_BIG (OPTION_MD_BASE + 1)
75 #define OPTION_SMALL (OPTION_MD_BASE + 2)
76 #define OPTION_MEMPARM (OPTION_MD_BASE + 3)
77 #define OPTION_REGPARM (OPTION_MD_BASE + 4)
78 #define OPTION_IDLE2 (OPTION_MD_BASE + 5)
79 #define OPTION_LOWPOWER (OPTION_MD_BASE + 6)
80 #define OPTION_ENHANCED (OPTION_MD_BASE + 7)
81 #define OPTION_REV (OPTION_MD_BASE + 8)
82
83 const char *md_shortopts = "bm:prs";
84 struct option md_longopts[] =
85 {
86 { "mcpu", required_argument, NULL, OPTION_CPU },
87 { "mdsp", required_argument, NULL, OPTION_CPU },
88 { "mbig", no_argument, NULL, OPTION_BIG },
89 { "msmall", no_argument, NULL, OPTION_SMALL },
90 { "mmemparm", no_argument, NULL, OPTION_MEMPARM },
91 { "mregparm", no_argument, NULL, OPTION_REGPARM },
92 { "midle2", no_argument, NULL, OPTION_IDLE2 },
93 { "mlowpower", no_argument, NULL, OPTION_LOWPOWER },
94 { "menhanced", no_argument, NULL, OPTION_ENHANCED },
95 { "mrev", required_argument, NULL, OPTION_REV },
96 { NULL, no_argument, NULL, 0 }
97 };
98
99 size_t md_longopts_size = sizeof (md_longopts);
100
101
102 typedef enum
103 {
104 M_UNKNOWN, M_IMMED, M_DIRECT, M_REGISTER, M_INDIRECT,
105 M_IMMED_F, M_PARALLEL, M_HI
106 }
107 tic4x_addr_mode_t;
108
109 typedef struct tic4x_operand
110 {
111 tic4x_addr_mode_t mode; /* Addressing mode. */
112 expressionS expr; /* Expression. */
113 int disp; /* Displacement for indirect addressing. */
114 int aregno; /* Aux. register number. */
115 LITTLENUM_TYPE fwords[MAX_LITTLENUMS]; /* Float immed. number. */
116 }
117 tic4x_operand_t;
118
119 typedef struct tic4x_insn
120 {
121 char name[TIC4X_NAME_MAX]; /* Mnemonic of instruction. */
122 unsigned int in_use; /* True if in_use. */
123 unsigned int parallel; /* True if parallel instruction. */
124 unsigned int nchars; /* This is always 4 for the C30. */
125 unsigned long opcode; /* Opcode number. */
126 expressionS exp; /* Expression required for relocation. */
127 int reloc; /* Relocation type required. */
128 int pcrel; /* True if relocation PC relative. */
129 char *pname; /* Name of instruction in parallel. */
130 unsigned int num_operands; /* Number of operands in total. */
131 tic4x_inst_t *inst; /* Pointer to first template. */
132 tic4x_operand_t operands[TIC4X_OPERANDS_MAX];
133 }
134 tic4x_insn_t;
135
136 static tic4x_insn_t the_insn; /* Info about our instruction. */
137 static tic4x_insn_t *insn = &the_insn;
138
139 static void tic4x_asg (int);
140 static void tic4x_bss (int);
141 static void tic4x_globl (int);
142 static void tic4x_cons (int);
143 static void tic4x_stringer (int);
144 static void tic4x_eval (int);
145 static void tic4x_newblock (int);
146 static void tic4x_sect (int);
147 static void tic4x_set (int);
148 static void tic4x_usect (int);
149 static void tic4x_version (int);
150
151
152 const pseudo_typeS
153 md_pseudo_table[] =
154 {
155 {"align", s_align_bytes, 32},
156 {"ascii", tic4x_stringer, 1},
157 {"asciz", tic4x_stringer, 0},
158 {"asg", tic4x_asg, 0},
159 {"block", s_space, 4},
160 {"byte", tic4x_cons, 1},
161 {"bss", tic4x_bss, 0},
162 {"copy", s_include, 0},
163 {"def", tic4x_globl, 0},
164 {"equ", tic4x_set, 0},
165 {"eval", tic4x_eval, 0},
166 {"global", tic4x_globl, 0},
167 {"globl", tic4x_globl, 0},
168 {"hword", tic4x_cons, 2},
169 {"ieee", float_cons, 'i'},
170 {"int", tic4x_cons, 4}, /* .int allocates 4 bytes. */
171 {"ldouble", float_cons, 'e'},
172 {"newblock", tic4x_newblock, 0},
173 {"ref", s_ignore, 0}, /* All undefined treated as external. */
174 {"set", tic4x_set, 0},
175 {"sect", tic4x_sect, 1}, /* Define named section. */
176 {"space", s_space, 4},
177 {"string", tic4x_stringer, 0},
178 {"usect", tic4x_usect, 0}, /* Reserve space in uninit. named sect. */
179 {"version", tic4x_version, 0},
180 {"word", tic4x_cons, 4}, /* .word allocates 4 bytes. */
181 {"xdef", tic4x_globl, 0},
182 {NULL, 0, 0},
183 };
184
185 int md_short_jump_size = 4;
186 int md_long_jump_size = 4;
187
188 /* This array holds the chars that always start a comment. If the
189 pre-processor is disabled, these aren't very useful. */
190 #ifdef TIC4X_ALT_SYNTAX
191 const char comment_chars[] = ";!";
192 #else
193 const char comment_chars[] = ";";
194 #endif
195
196 /* This array holds the chars that only start a comment at the beginning of
197 a line. If the line seems to have the form '# 123 filename'
198 .line and .file directives will appear in the pre-processed output.
199 Note that input_file.c hand checks for '#' at the beginning of the
200 first line of the input file. This is because the compiler outputs
201 #NO_APP at the beginning of its output.
202 Also note that comments like this one will always work. */
203 const char line_comment_chars[] = "#*";
204
205 /* We needed an unused char for line separation to work around the
206 lack of macros, using sed and such. */
207 const char line_separator_chars[] = "&";
208
209 /* Chars that can be used to separate mant from exp in floating point nums. */
210 const char EXP_CHARS[] = "eE";
211
212 /* Chars that mean this number is a floating point constant. */
213 /* As in 0f12.456 */
214 /* or 0d1.2345e12 */
215 const char FLT_CHARS[] = "fFilsS";
216
217 /* Also be aware that MAXIMUM_NUMBER_OF_CHARS_FOR_FLOAT may have to be
218 changed in read.c. Ideally it shouldn't have to know about it at
219 all, but nothing is ideal around here. */
220
221 /* Flonums returned here. */
222 extern FLONUM_TYPE generic_floating_point_number;
223
224 /* Precision in LittleNums. */
225 #define MAX_PRECISION (4) /* Its a bit overkill for us, but the code
226 requires it... */
227 #define S_PRECISION (1) /* Short float constants 16-bit. */
228 #define F_PRECISION (2) /* Float and double types 32-bit. */
229 #define E_PRECISION (4) /* Extended precision, 64-bit (real 40-bit). */
230 #define GUARD (2)
231
232 /* Turn generic_floating_point_number into a real short/float/double. */
233 static int
tic4x_gen_to_words(FLONUM_TYPE flonum,LITTLENUM_TYPE * words,int precision)234 tic4x_gen_to_words (FLONUM_TYPE flonum, LITTLENUM_TYPE *words, int precision)
235 {
236 int return_value = 0;
237 LITTLENUM_TYPE *p; /* Littlenum pointer. */
238 int mantissa_bits; /* Bits in mantissa field. */
239 int exponent_bits; /* Bits in exponent field. */
240 int exponent;
241 unsigned int sone; /* Scaled one. */
242 unsigned int sfract; /* Scaled fraction. */
243 unsigned int smant; /* Scaled mantissa. */
244 unsigned int tmp;
245 unsigned int mover; /* Mantissa overflow bits */
246 unsigned int rbit; /* Round bit. */
247 int shift; /* Shift count. */
248
249 /* NOTE: Svein Seldal <Svein@dev.seldal.com>
250 The code in this function is altered slightly to support floats
251 with 31-bits mantissas, thus the documentation below may be a
252 little bit inaccurate.
253
254 By Michael P. Hayes <m.hayes@elec.canterbury.ac.nz>
255 Here is how a generic floating point number is stored using
256 flonums (an extension of bignums) where p is a pointer to an
257 array of LITTLENUMs.
258
259 For example 2e-3 is stored with exp = -4 and
260 bits[0] = 0x0000
261 bits[1] = 0x0000
262 bits[2] = 0x4fde
263 bits[3] = 0x978d
264 bits[4] = 0x126e
265 bits[5] = 0x0083
266 with low = &bits[2], high = &bits[5], and leader = &bits[5].
267
268 This number can be written as
269 0x0083126e978d4fde.00000000 * 65536**-4 or
270 0x0.0083126e978d4fde * 65536**0 or
271 0x0.83126e978d4fde * 2**-8 = 2e-3
272
273 Note that low points to the 65536**0 littlenum (bits[2]) and
274 leader points to the most significant non-zero littlenum
275 (bits[5]).
276
277 TMS320C3X floating point numbers are a bit of a strange beast.
278 The 32-bit flavour has the 8 MSBs representing the exponent in
279 twos complement format (-128 to +127). There is then a sign bit
280 followed by 23 bits of mantissa. The mantissa is expressed in
281 twos complement format with the binary point after the most
282 significant non sign bit. The bit after the binary point is
283 suppressed since it is the complement of the sign bit. The
284 effective mantissa is thus 24 bits. Zero is represented by an
285 exponent of -128.
286
287 The 16-bit flavour has the 4 MSBs representing the exponent in
288 twos complement format (-8 to +7). There is then a sign bit
289 followed by 11 bits of mantissa. The mantissa is expressed in
290 twos complement format with the binary point after the most
291 significant non sign bit. The bit after the binary point is
292 suppressed since it is the complement of the sign bit. The
293 effective mantissa is thus 12 bits. Zero is represented by an
294 exponent of -8. For example,
295
296 number norm mant m x e s i fraction f
297 +0.500 => 1.00000000000 -1 -1 0 1 .00000000000 (1 + 0) * 2^(-1)
298 +0.999 => 1.11111111111 -1 -1 0 1 .11111111111 (1 + 0.99) * 2^(-1)
299 +1.000 => 1.00000000000 0 0 0 1 .00000000000 (1 + 0) * 2^(0)
300 +1.500 => 1.10000000000 0 0 0 1 .10000000000 (1 + 0.5) * 2^(0)
301 +1.999 => 1.11111111111 0 0 0 1 .11111111111 (1 + 0.9) * 2^(0)
302 +2.000 => 1.00000000000 1 1 0 1 .00000000000 (1 + 0) * 2^(1)
303 +4.000 => 1.00000000000 2 2 0 1 .00000000000 (1 + 0) * 2^(2)
304 -0.500 => 1.00000000000 -1 -1 1 0 .10000000000 (-2 + 0) * 2^(-2)
305 -1.000 => 1.00000000000 0 -1 1 0 .00000000000 (-2 + 0) * 2^(-1)
306 -1.500 => 1.10000000000 0 0 1 0 .10000000000 (-2 + 0.5) * 2^(0)
307 -1.999 => 1.11111111111 0 0 1 0 .00000000001 (-2 + 0.11) * 2^(0)
308 -2.000 => 1.00000000000 1 1 1 0 .00000000000 (-2 + 0) * 2^(0)
309 -4.000 => 1.00000000000 2 1 1 0 .00000000000 (-2 + 0) * 2^(1)
310
311 where e is the exponent, s is the sign bit, i is the implied bit,
312 and f is the fraction stored in the mantissa field.
313
314 num = (1 + f) * 2^x = m * 2^e if s = 0
315 num = (-2 + f) * 2^x = -m * 2^e if s = 1
316 where 0 <= f < 1.0 and 1.0 <= m < 2.0
317
318 The fraction (f) and exponent (e) fields for the TMS320C3X format
319 can be derived from the normalised mantissa (m) and exponent (x) using:
320
321 f = m - 1, e = x if s = 0
322 f = 2 - m, e = x if s = 1 and m != 1.0
323 f = 0, e = x - 1 if s = 1 and m = 1.0
324 f = 0, e = -8 if m = 0
325
326
327 OK, the other issue we have to consider is rounding since the
328 mantissa has a much higher potential precision than what we can
329 represent. To do this we add half the smallest storable fraction.
330 We then have to renormalise the number to allow for overflow.
331
332 To convert a generic flonum into a TMS320C3X floating point
333 number, here's what we try to do....
334
335 The first thing is to generate a normalised mantissa (m) where
336 1.0 <= m < 2 and to convert the exponent from base 16 to base 2.
337 We desire the binary point to be placed after the most significant
338 non zero bit. This process is done in two steps: firstly, the
339 littlenum with the most significant non zero bit is located (this
340 is done for us since leader points to this littlenum) and the
341 binary point (which is currently after the LSB of the littlenum
342 pointed to by low) is moved to before the MSB of the littlenum
343 pointed to by leader. This requires the exponent to be adjusted
344 by leader - low + 1. In the earlier example, the new exponent is
345 thus -4 + (5 - 2 + 1) = 0 (base 65536). We now need to convert
346 the exponent to base 2 by multiplying the exponent by 16 (log2
347 65536). The exponent base 2 is thus also zero.
348
349 The second step is to hunt for the most significant non zero bit
350 in the leader littlenum. We do this by left shifting a copy of
351 the leader littlenum until bit 16 is set (0x10000) and counting
352 the number of shifts, S, required. The number of shifts then has to
353 be added to correct the exponent (base 2). For our example, this
354 will require 9 shifts and thus our normalised exponent (base 2) is
355 0 + 9 = 9. Note that the worst case scenario is when the leader
356 littlenum is 1, thus requiring 16 shifts.
357
358 We now have to left shift the other littlenums by the same amount,
359 propagating the shifted bits into the more significant littlenums.
360 To save a lot of unnecessary shifting we only have to consider
361 two or three littlenums, since the greatest number of mantissa
362 bits required is 24 + 1 rounding bit. While two littlenums
363 provide 32 bits of precision, the most significant littlenum
364 may only contain a single significant bit and thus an extra
365 littlenum is required.
366
367 Denoting the number of bits in the fraction field as F, we require
368 G = F + 2 bits (one extra bit is for rounding, the other gets
369 suppressed). Say we required S shifts to find the most
370 significant bit in the leader littlenum, the number of left shifts
371 required to move this bit into bit position G - 1 is L = G + S - 17.
372 Note that this shift count may be negative for the short floating
373 point flavour (where F = 11 and thus G = 13 and potentially S < 3).
374 If L > 0 we have to shunt the next littlenum into position. Bit
375 15 (the MSB) of the next littlenum needs to get moved into position
376 L - 1 (If L > 15 we need all the bits of this littlenum and
377 some more from the next one.). We subtract 16 from L and use this
378 as the left shift count; the resultant value we or with the
379 previous result. If L > 0, we repeat this operation. */
380
381 if (precision != S_PRECISION)
382 words[1] = 0x0000;
383 if (precision == E_PRECISION)
384 words[2] = words[3] = 0x0000;
385
386 /* 0.0e0 or NaN seen. */
387 if (flonum.low > flonum.leader /* = 0.0e0 */
388 || flonum.sign == 0) /* = NaN */
389 {
390 if(flonum.sign == 0)
391 as_bad (_("Nan, using zero."));
392 words[0] = 0x8000;
393 return return_value;
394 }
395
396 if (flonum.sign == 'P')
397 {
398 /* +INF: Replace with maximum float. */
399 if (precision == S_PRECISION)
400 words[0] = 0x77ff;
401 else
402 {
403 words[0] = 0x7f7f;
404 words[1] = 0xffff;
405 }
406 if (precision == E_PRECISION)
407 {
408 words[2] = 0x7fff;
409 words[3] = 0xffff;
410 }
411 return return_value;
412 }
413 else if (flonum.sign == 'N')
414 {
415 /* -INF: Replace with maximum float. */
416 if (precision == S_PRECISION)
417 words[0] = 0x7800;
418 else
419 words[0] = 0x7f80;
420 if (precision == E_PRECISION)
421 words[2] = 0x8000;
422 return return_value;
423 }
424
425 exponent = (flonum.exponent + flonum.leader - flonum.low + 1) * 16;
426
427 if (!(tmp = *flonum.leader))
428 abort (); /* Hmmm. */
429 shift = 0; /* Find position of first sig. bit. */
430 while (tmp >>= 1)
431 shift++;
432 exponent -= (16 - shift); /* Adjust exponent. */
433
434 if (precision == S_PRECISION) /* Allow 1 rounding bit. */
435 {
436 exponent_bits = 4;
437 mantissa_bits = 11;
438 }
439 else if(precision == F_PRECISION)
440 {
441 exponent_bits = 8;
442 mantissa_bits = 23;
443 }
444 else /* E_PRECISION */
445 {
446 exponent_bits = 8;
447 mantissa_bits = 31;
448 }
449
450 shift = mantissa_bits - shift;
451
452 smant = 0;
453 mover = 0;
454 rbit = 0;
455 /* Store the mantissa data into smant and the roundbit into rbit */
456 for (p = flonum.leader; p >= flonum.low && shift > -16; p--)
457 {
458 tmp = shift >= 0 ? *p << shift : *p >> -shift;
459 rbit = shift < 0 ? ((*p >> (-shift-1)) & 0x1) : 0;
460 smant |= tmp;
461 shift -= 16;
462 }
463
464 /* OK, we've got our scaled mantissa so let's round it up */
465 if(rbit)
466 {
467 /* If the mantissa is going to overflow when added, lets store
468 the extra bit in mover. -- A special case exists when
469 mantissa_bits is 31 (E_PRECISION). Then the first test cannot
470 be trusted, as result is host-dependent, thus the second
471 test. */
472 if( smant == ((unsigned)(1<<(mantissa_bits+1))-1)
473 || smant == (unsigned)-1 ) /* This is to catch E_PRECISION cases */
474 mover=1;
475 smant++;
476 }
477
478 /* Get the scaled one value */
479 sone = (1 << (mantissa_bits));
480
481 /* The number may be unnormalised so renormalise it... */
482 if(mover)
483 {
484 smant >>= 1;
485 smant |= sone; /* Insert the bit from mover into smant */
486 exponent++;
487 }
488
489 /* The binary point is now between bit positions 11 and 10 or 23 and 22,
490 i.e., between mantissa_bits - 1 and mantissa_bits - 2 and the
491 bit at mantissa_bits - 1 should be set. */
492 if (!(sone&smant))
493 abort (); /* Ooops. */
494
495 if (flonum.sign == '+')
496 sfract = smant - sone; /* smant - 1.0. */
497 else
498 {
499 /* This seems to work. */
500 if (smant == sone)
501 {
502 exponent--;
503 sfract = 0;
504 }
505 else
506 {
507 sfract = -smant & (sone-1); /* 2.0 - smant. */
508 }
509 sfract |= sone; /* Insert sign bit. */
510 }
511
512 if (abs (exponent) >= (1 << (exponent_bits - 1)))
513 as_bad (_("Cannot represent exponent in %d bits"), exponent_bits);
514
515 /* Force exponent to fit in desired field width. */
516 exponent &= (1 << (exponent_bits)) - 1;
517
518 if (precision == E_PRECISION)
519 {
520 /* Map the float part first (100% equal format as F_PRECISION) */
521 words[0] = exponent << (mantissa_bits+1-24);
522 words[0] |= sfract >> 24;
523 words[1] = sfract >> 8;
524
525 /* Map the mantissa in the next */
526 words[2] = sfract >> 16;
527 words[3] = sfract & 0xffff;
528 }
529 else
530 {
531 /* Insert the exponent data into the word */
532 sfract |= exponent << (mantissa_bits+1);
533
534 if (precision == S_PRECISION)
535 words[0] = sfract;
536 else
537 {
538 words[0] = sfract >> 16;
539 words[1] = sfract & 0xffff;
540 }
541 }
542
543 return return_value;
544 }
545
546 /* Returns pointer past text consumed. */
547 static char *
tic4x_atof(char * str,char what_kind,LITTLENUM_TYPE * words)548 tic4x_atof (char *str, char what_kind, LITTLENUM_TYPE *words)
549 {
550 /* Extra bits for zeroed low-order bits. The 1st MAX_PRECISION are
551 zeroed, the last contain flonum bits. */
552 static LITTLENUM_TYPE bits[MAX_PRECISION + MAX_PRECISION + GUARD];
553 char *return_value;
554 /* Number of 16-bit words in the format. */
555 int precision;
556 FLONUM_TYPE save_gen_flonum;
557
558 /* We have to save the generic_floating_point_number because it
559 contains storage allocation about the array of LITTLENUMs where
560 the value is actually stored. We will allocate our own array of
561 littlenums below, but have to restore the global one on exit. */
562 save_gen_flonum = generic_floating_point_number;
563
564 return_value = str;
565 generic_floating_point_number.low = bits + MAX_PRECISION;
566 generic_floating_point_number.high = NULL;
567 generic_floating_point_number.leader = NULL;
568 generic_floating_point_number.exponent = 0;
569 generic_floating_point_number.sign = '\0';
570
571 /* Use more LittleNums than seems necessary: the highest flonum may
572 have 15 leading 0 bits, so could be useless. */
573
574 memset (bits, '\0', sizeof (LITTLENUM_TYPE) * MAX_PRECISION);
575
576 switch (what_kind)
577 {
578 case 's':
579 case 'S':
580 precision = S_PRECISION;
581 break;
582
583 case 'd':
584 case 'D':
585 case 'f':
586 case 'F':
587 precision = F_PRECISION;
588 break;
589
590 case 'E':
591 case 'e':
592 precision = E_PRECISION;
593 break;
594
595 default:
596 as_bad (_("Invalid floating point number"));
597 return (NULL);
598 }
599
600 generic_floating_point_number.high
601 = generic_floating_point_number.low + precision - 1 + GUARD;
602
603 if (atof_generic (&return_value, ".", EXP_CHARS,
604 &generic_floating_point_number))
605 {
606 as_bad (_("Invalid floating point number"));
607 return (NULL);
608 }
609
610 tic4x_gen_to_words (generic_floating_point_number,
611 words, precision);
612
613 /* Restore the generic_floating_point_number's storage alloc (and
614 everything else). */
615 generic_floating_point_number = save_gen_flonum;
616
617 return return_value;
618 }
619
620 static void
tic4x_insert_reg(char * regname,int regnum)621 tic4x_insert_reg (char *regname, int regnum)
622 {
623 char buf[32];
624 int i;
625
626 symbol_table_insert (symbol_new (regname, reg_section, (valueT) regnum,
627 &zero_address_frag));
628 for (i = 0; regname[i]; i++)
629 buf[i] = ISLOWER (regname[i]) ? TOUPPER (regname[i]) : regname[i];
630 buf[i] = '\0';
631
632 symbol_table_insert (symbol_new (buf, reg_section, (valueT) regnum,
633 &zero_address_frag));
634 }
635
636 static void
tic4x_insert_sym(char * symname,int value)637 tic4x_insert_sym (char *symname, int value)
638 {
639 symbolS *symbolP;
640
641 symbolP = symbol_new (symname, absolute_section,
642 (valueT) value, &zero_address_frag);
643 SF_SET_LOCAL (symbolP);
644 symbol_table_insert (symbolP);
645 }
646
647 static char *
tic4x_expression(char * str,expressionS * exp)648 tic4x_expression (char *str, expressionS *exp)
649 {
650 char *s;
651 char *t;
652
653 t = input_line_pointer; /* Save line pointer. */
654 input_line_pointer = str;
655 expression (exp);
656 s = input_line_pointer;
657 input_line_pointer = t; /* Restore line pointer. */
658 return s; /* Return pointer to where parsing stopped. */
659 }
660
661 static char *
tic4x_expression_abs(char * str,offsetT * value)662 tic4x_expression_abs (char *str, offsetT *value)
663 {
664 char *s;
665 char *t;
666
667 t = input_line_pointer; /* Save line pointer. */
668 input_line_pointer = str;
669 *value = get_absolute_expression ();
670 s = input_line_pointer;
671 input_line_pointer = t; /* Restore line pointer. */
672 return s;
673 }
674
675 static void
tic4x_emit_char(char c,int b)676 tic4x_emit_char (char c, int b)
677 {
678 expressionS exp;
679
680 exp.X_op = O_constant;
681 exp.X_add_number = c;
682 emit_expr (&exp, b);
683 }
684
685 static void
tic4x_seg_alloc(char * name ATTRIBUTE_UNUSED,segT seg ATTRIBUTE_UNUSED,int size,symbolS * symbolP)686 tic4x_seg_alloc (char *name ATTRIBUTE_UNUSED,
687 segT seg ATTRIBUTE_UNUSED,
688 int size,
689 symbolS *symbolP)
690 {
691 /* Note that the size is in words
692 so we multiply it by 4 to get the number of bytes to allocate. */
693
694 /* If we have symbol: .usect ".fred", size etc.,
695 the symbol needs to point to the first location reserved
696 by the pseudo op. */
697
698 if (size)
699 {
700 char *p;
701
702 p = frag_var (rs_fill, 1, 1, (relax_substateT) 0,
703 (symbolS *) symbolP,
704 size * OCTETS_PER_BYTE, (char *) 0);
705 *p = 0;
706 }
707 }
708
709 /* .asg ["]character-string["], symbol */
710 static void
tic4x_asg(int x ATTRIBUTE_UNUSED)711 tic4x_asg (int x ATTRIBUTE_UNUSED)
712 {
713 char c;
714 char *name;
715 char *str;
716 char *tmp;
717
718 SKIP_WHITESPACE ();
719 str = input_line_pointer;
720
721 /* Skip string expression. */
722 while (*input_line_pointer != ',' && *input_line_pointer)
723 input_line_pointer++;
724 if (*input_line_pointer != ',')
725 {
726 as_bad (_("Comma expected\n"));
727 return;
728 }
729 *input_line_pointer++ = '\0';
730 name = input_line_pointer;
731 c = get_symbol_end (); /* Get terminator. */
732 tmp = xmalloc (strlen (str) + 1);
733 strcpy (tmp, str);
734 str = tmp;
735 tmp = xmalloc (strlen (name) + 1);
736 strcpy (tmp, name);
737 name = tmp;
738 if (hash_find (tic4x_asg_hash, name))
739 hash_replace (tic4x_asg_hash, name, (void *) str);
740 else
741 hash_insert (tic4x_asg_hash, name, (void *) str);
742 *input_line_pointer = c;
743 demand_empty_rest_of_line ();
744 }
745
746 /* .bss symbol, size */
747 static void
tic4x_bss(int x ATTRIBUTE_UNUSED)748 tic4x_bss (int x ATTRIBUTE_UNUSED)
749 {
750 char c;
751 char *name;
752 char *p;
753 offsetT size;
754 segT current_seg;
755 subsegT current_subseg;
756 symbolS *symbolP;
757
758 current_seg = now_seg; /* Save current seg. */
759 current_subseg = now_subseg; /* Save current subseg. */
760
761 SKIP_WHITESPACE ();
762 name = input_line_pointer;
763 c = get_symbol_end (); /* Get terminator. */
764 if (c != ',')
765 {
766 as_bad (_(".bss size argument missing\n"));
767 return;
768 }
769
770 input_line_pointer =
771 tic4x_expression_abs (++input_line_pointer, &size);
772 if (size < 0)
773 {
774 as_bad (_(".bss size %ld < 0!"), (long) size);
775 return;
776 }
777 subseg_set (bss_section, 0);
778 symbolP = symbol_find_or_make (name);
779
780 if (S_GET_SEGMENT (symbolP) == bss_section)
781 symbol_get_frag (symbolP)->fr_symbol = 0;
782
783 symbol_set_frag (symbolP, frag_now);
784
785 p = frag_var (rs_org, 1, 1, (relax_substateT) 0, symbolP,
786 size * OCTETS_PER_BYTE, (char *) 0);
787 *p = 0; /* Fill char. */
788
789 S_SET_SEGMENT (symbolP, bss_section);
790
791 /* The symbol may already have been created with a preceding
792 ".globl" directive -- be careful not to step on storage class
793 in that case. Otherwise, set it to static. */
794 if (S_GET_STORAGE_CLASS (symbolP) != C_EXT)
795 S_SET_STORAGE_CLASS (symbolP, C_STAT);
796
797 subseg_set (current_seg, current_subseg); /* Restore current seg. */
798 demand_empty_rest_of_line ();
799 }
800
801 static void
tic4x_globl(int ignore ATTRIBUTE_UNUSED)802 tic4x_globl (int ignore ATTRIBUTE_UNUSED)
803 {
804 char *name;
805 int c;
806 symbolS *symbolP;
807
808 do
809 {
810 name = input_line_pointer;
811 c = get_symbol_end ();
812 symbolP = symbol_find_or_make (name);
813 *input_line_pointer = c;
814 SKIP_WHITESPACE ();
815 S_SET_STORAGE_CLASS (symbolP, C_EXT);
816 S_SET_EXTERNAL (symbolP);
817 if (c == ',')
818 {
819 input_line_pointer++;
820 SKIP_WHITESPACE ();
821 if (*input_line_pointer == '\n')
822 c = '\n';
823 }
824 }
825 while (c == ',');
826
827 demand_empty_rest_of_line ();
828 }
829
830 /* Handle .byte, .word. .int, .long */
831 static void
tic4x_cons(int bytes)832 tic4x_cons (int bytes)
833 {
834 unsigned int c;
835 do
836 {
837 SKIP_WHITESPACE ();
838 if (*input_line_pointer == '"')
839 {
840 input_line_pointer++;
841 while (is_a_char (c = next_char_of_string ()))
842 tic4x_emit_char (c, 4);
843 know (input_line_pointer[-1] == '\"');
844 }
845 else
846 {
847 expressionS exp;
848
849 input_line_pointer = tic4x_expression (input_line_pointer, &exp);
850 if (exp.X_op == O_constant)
851 {
852 switch (bytes)
853 {
854 case 1:
855 exp.X_add_number &= 255;
856 break;
857 case 2:
858 exp.X_add_number &= 65535;
859 break;
860 }
861 }
862 /* Perhaps we should disallow .byte and .hword with
863 a non constant expression that will require relocation. */
864 emit_expr (&exp, 4);
865 }
866 }
867 while (*input_line_pointer++ == ',');
868
869 input_line_pointer--; /* Put terminator back into stream. */
870 demand_empty_rest_of_line ();
871 }
872
873 /* Handle .ascii, .asciz, .string */
874 static void
tic4x_stringer(int append_zero)875 tic4x_stringer (int append_zero)
876 {
877 int bytes;
878 unsigned int c;
879
880 bytes = 0;
881 do
882 {
883 SKIP_WHITESPACE ();
884 if (*input_line_pointer == '"')
885 {
886 input_line_pointer++;
887 while (is_a_char (c = next_char_of_string ()))
888 {
889 tic4x_emit_char (c, 1);
890 bytes++;
891 }
892
893 if (append_zero)
894 {
895 tic4x_emit_char (c, 1);
896 bytes++;
897 }
898
899 know (input_line_pointer[-1] == '\"');
900 }
901 else
902 {
903 expressionS exp;
904
905 input_line_pointer = tic4x_expression (input_line_pointer, &exp);
906 if (exp.X_op != O_constant)
907 {
908 as_bad (_("Non-constant symbols not allowed\n"));
909 return;
910 }
911 exp.X_add_number &= 255; /* Limit numeber to 8-bit */
912 emit_expr (&exp, 1);
913 bytes++;
914 }
915 }
916 while (*input_line_pointer++ == ',');
917
918 /* Fill out the rest of the expression with 0's to fill up a full word */
919 if ( bytes&0x3 )
920 tic4x_emit_char (0, 4-(bytes&0x3));
921
922 input_line_pointer--; /* Put terminator back into stream. */
923 demand_empty_rest_of_line ();
924 }
925
926 /* .eval expression, symbol */
927 static void
tic4x_eval(int x ATTRIBUTE_UNUSED)928 tic4x_eval (int x ATTRIBUTE_UNUSED)
929 {
930 char c;
931 offsetT value;
932 char *name;
933
934 SKIP_WHITESPACE ();
935 input_line_pointer =
936 tic4x_expression_abs (input_line_pointer, &value);
937 if (*input_line_pointer++ != ',')
938 {
939 as_bad (_("Symbol missing\n"));
940 return;
941 }
942 name = input_line_pointer;
943 c = get_symbol_end (); /* Get terminator. */
944 tic4x_insert_sym (name, value);
945 *input_line_pointer++ = c;
946 demand_empty_rest_of_line ();
947 }
948
949 /* Reset local labels. */
950 static void
tic4x_newblock(int x ATTRIBUTE_UNUSED)951 tic4x_newblock (int x ATTRIBUTE_UNUSED)
952 {
953 dollar_label_clear ();
954 }
955
956 /* .sect "section-name" [, value] */
957 /* .sect ["]section-name[:subsection-name]["] [, value] */
958 static void
tic4x_sect(int x ATTRIBUTE_UNUSED)959 tic4x_sect (int x ATTRIBUTE_UNUSED)
960 {
961 char c;
962 char *section_name;
963 char *name;
964 segT seg;
965 offsetT num;
966
967 SKIP_WHITESPACE ();
968 if (*input_line_pointer == '"')
969 input_line_pointer++;
970 section_name = input_line_pointer;
971 c = get_symbol_end (); /* Get terminator. */
972 input_line_pointer++; /* Skip null symbol terminator. */
973 name = xmalloc (input_line_pointer - section_name + 1);
974 strcpy (name, section_name);
975
976 /* TI C from version 5.0 allows a section name to contain a
977 subsection name as well. The subsection name is separated by a
978 ':' from the section name. Currently we scan the subsection
979 name and discard it.
980 Volker Kuhlmann <v.kuhlmann@elec.canterbury.ac.nz>. */
981 if (c == ':')
982 {
983 c = get_symbol_end (); /* Get terminator. */
984 input_line_pointer++; /* Skip null symbol terminator. */
985 as_warn (_(".sect: subsection name ignored"));
986 }
987
988 /* We might still have a '"' to discard, but the character after a
989 symbol name will be overwritten with a \0 by get_symbol_end()
990 [VK]. */
991
992 if (c == ',')
993 input_line_pointer =
994 tic4x_expression_abs (input_line_pointer, &num);
995 else if (*input_line_pointer == ',')
996 {
997 input_line_pointer =
998 tic4x_expression_abs (++input_line_pointer, &num);
999 }
1000 else
1001 num = 0;
1002
1003 seg = subseg_new (name, num);
1004 if (line_label != NULL)
1005 {
1006 S_SET_SEGMENT (line_label, seg);
1007 symbol_set_frag (line_label, frag_now);
1008 }
1009
1010 if (bfd_get_section_flags (stdoutput, seg) == SEC_NO_FLAGS)
1011 {
1012 if (!bfd_set_section_flags (stdoutput, seg, SEC_DATA))
1013 as_warn (_("Error setting flags for \"%s\": %s"), name,
1014 bfd_errmsg (bfd_get_error ()));
1015 }
1016
1017 /* If the last character overwritten by get_symbol_end() was an
1018 end-of-line, we must restore it or the end of the line will not be
1019 recognised and scanning extends into the next line, stopping with
1020 an error (blame Volker Kuhlmann <v.kuhlmann@elec.canterbury.ac.nz>
1021 if this is not true). */
1022 if (is_end_of_line[(unsigned char) c])
1023 *(--input_line_pointer) = c;
1024
1025 demand_empty_rest_of_line ();
1026 }
1027
1028 /* symbol[:] .set value or .set symbol, value */
1029 static void
tic4x_set(int x ATTRIBUTE_UNUSED)1030 tic4x_set (int x ATTRIBUTE_UNUSED)
1031 {
1032 symbolS *symbolP;
1033
1034 SKIP_WHITESPACE ();
1035 if ((symbolP = line_label) == NULL)
1036 {
1037 char c;
1038 char *name;
1039
1040 name = input_line_pointer;
1041 c = get_symbol_end (); /* Get terminator. */
1042 if (c != ',')
1043 {
1044 as_bad (_(".set syntax invalid\n"));
1045 ignore_rest_of_line ();
1046 return;
1047 }
1048 ++input_line_pointer;
1049 symbolP = symbol_find_or_make (name);
1050 }
1051 else
1052 symbol_table_insert (symbolP);
1053
1054 pseudo_set (symbolP);
1055 demand_empty_rest_of_line ();
1056 }
1057
1058 /* [symbol] .usect ["]section-name["], size-in-words [, alignment-flag] */
1059 static void
tic4x_usect(int x ATTRIBUTE_UNUSED)1060 tic4x_usect (int x ATTRIBUTE_UNUSED)
1061 {
1062 char c;
1063 char *name;
1064 char *section_name;
1065 segT seg;
1066 offsetT size, alignment_flag;
1067 segT current_seg;
1068 subsegT current_subseg;
1069
1070 current_seg = now_seg; /* save current seg. */
1071 current_subseg = now_subseg; /* save current subseg. */
1072
1073 SKIP_WHITESPACE ();
1074 if (*input_line_pointer == '"')
1075 input_line_pointer++;
1076 section_name = input_line_pointer;
1077 c = get_symbol_end (); /* Get terminator. */
1078 input_line_pointer++; /* Skip null symbol terminator. */
1079 name = xmalloc (input_line_pointer - section_name + 1);
1080 strcpy (name, section_name);
1081
1082 if (c == ',')
1083 input_line_pointer =
1084 tic4x_expression_abs (input_line_pointer, &size);
1085 else if (*input_line_pointer == ',')
1086 {
1087 input_line_pointer =
1088 tic4x_expression_abs (++input_line_pointer, &size);
1089 }
1090 else
1091 size = 0;
1092
1093 /* Read a possibly present third argument (alignment flag) [VK]. */
1094 if (*input_line_pointer == ',')
1095 {
1096 input_line_pointer =
1097 tic4x_expression_abs (++input_line_pointer, &alignment_flag);
1098 }
1099 else
1100 alignment_flag = 0;
1101 if (alignment_flag)
1102 as_warn (_(".usect: non-zero alignment flag ignored"));
1103
1104 seg = subseg_new (name, 0);
1105 if (line_label != NULL)
1106 {
1107 S_SET_SEGMENT (line_label, seg);
1108 symbol_set_frag (line_label, frag_now);
1109 S_SET_VALUE (line_label, frag_now_fix ());
1110 }
1111 seg_info (seg)->bss = 1; /* Uninitialised data. */
1112 if (!bfd_set_section_flags (stdoutput, seg, SEC_ALLOC))
1113 as_warn (_("Error setting flags for \"%s\": %s"), name,
1114 bfd_errmsg (bfd_get_error ()));
1115 tic4x_seg_alloc (name, seg, size, line_label);
1116
1117 if (S_GET_STORAGE_CLASS (line_label) != C_EXT)
1118 S_SET_STORAGE_CLASS (line_label, C_STAT);
1119
1120 subseg_set (current_seg, current_subseg); /* Restore current seg. */
1121 demand_empty_rest_of_line ();
1122 }
1123
1124 /* .version cpu-version. */
1125 static void
tic4x_version(int x ATTRIBUTE_UNUSED)1126 tic4x_version (int x ATTRIBUTE_UNUSED)
1127 {
1128 offsetT temp;
1129
1130 input_line_pointer =
1131 tic4x_expression_abs (input_line_pointer, &temp);
1132 if (!IS_CPU_TIC3X (temp) && !IS_CPU_TIC4X (temp))
1133 as_bad (_("This assembler does not support processor generation %ld"),
1134 (long) temp);
1135
1136 if (tic4x_cpu && temp != (offsetT) tic4x_cpu)
1137 as_warn (_("Changing processor generation on fly not supported..."));
1138 tic4x_cpu = temp;
1139 demand_empty_rest_of_line ();
1140 }
1141
1142 static void
tic4x_init_regtable(void)1143 tic4x_init_regtable (void)
1144 {
1145 unsigned int i;
1146
1147 for (i = 0; i < tic3x_num_registers; i++)
1148 tic4x_insert_reg (tic3x_registers[i].name,
1149 tic3x_registers[i].regno);
1150
1151 if (IS_CPU_TIC4X (tic4x_cpu))
1152 {
1153 /* Add additional Tic4x registers, overriding some C3x ones. */
1154 for (i = 0; i < tic4x_num_registers; i++)
1155 tic4x_insert_reg (tic4x_registers[i].name,
1156 tic4x_registers[i].regno);
1157 }
1158 }
1159
1160 static void
tic4x_init_symbols(void)1161 tic4x_init_symbols (void)
1162 {
1163 /* The TI tools accept case insensitive versions of these symbols,
1164 we don't !
1165
1166 For TI C/Asm 5.0
1167
1168 .TMS320xx 30,31,32,40,or 44 set according to -v flag
1169 .C3X or .C3x 1 or 0 1 if -v30,-v31,or -v32
1170 .C30 1 or 0 1 if -v30
1171 .C31 1 or 0 1 if -v31
1172 .C32 1 or 0 1 if -v32
1173 .C4X or .C4x 1 or 0 1 if -v40, or -v44
1174 .C40 1 or 0 1 if -v40
1175 .C44 1 or 0 1 if -v44
1176
1177 .REGPARM 1 or 0 1 if -mr option used
1178 .BIGMODEL 1 or 0 1 if -mb option used
1179
1180 These symbols are currently supported but will be removed in a
1181 later version:
1182 .TMS320C30 1 or 0 1 if -v30,-v31,or -v32
1183 .TMS320C31 1 or 0 1 if -v31
1184 .TMS320C32 1 or 0 1 if -v32
1185 .TMS320C40 1 or 0 1 if -v40, or -v44
1186 .TMS320C44 1 or 0 1 if -v44
1187
1188 Source: TI: TMS320C3x/C4x Assembly Language Tools User's Guide,
1189 1997, SPRU035C, p. 3-17/3-18. */
1190 tic4x_insert_sym (".REGPARM", tic4x_reg_args);
1191 tic4x_insert_sym (".MEMPARM", !tic4x_reg_args);
1192 tic4x_insert_sym (".BIGMODEL", tic4x_big_model);
1193 tic4x_insert_sym (".C30INTERRUPT", 0);
1194 tic4x_insert_sym (".TMS320xx", tic4x_cpu == 0 ? 40 : tic4x_cpu);
1195 tic4x_insert_sym (".C3X", tic4x_cpu == 30 || tic4x_cpu == 31 || tic4x_cpu == 32 || tic4x_cpu == 33);
1196 tic4x_insert_sym (".C3x", tic4x_cpu == 30 || tic4x_cpu == 31 || tic4x_cpu == 32 || tic4x_cpu == 33);
1197 tic4x_insert_sym (".C4X", tic4x_cpu == 0 || tic4x_cpu == 40 || tic4x_cpu == 44);
1198 tic4x_insert_sym (".C4x", tic4x_cpu == 0 || tic4x_cpu == 40 || tic4x_cpu == 44);
1199 /* Do we need to have the following symbols also in lower case? */
1200 tic4x_insert_sym (".TMS320C30", tic4x_cpu == 30 || tic4x_cpu == 31 || tic4x_cpu == 32 || tic4x_cpu == 33);
1201 tic4x_insert_sym (".tms320C30", tic4x_cpu == 30 || tic4x_cpu == 31 || tic4x_cpu == 32 || tic4x_cpu == 33);
1202 tic4x_insert_sym (".TMS320C31", tic4x_cpu == 31);
1203 tic4x_insert_sym (".tms320C31", tic4x_cpu == 31);
1204 tic4x_insert_sym (".TMS320C32", tic4x_cpu == 32);
1205 tic4x_insert_sym (".tms320C32", tic4x_cpu == 32);
1206 tic4x_insert_sym (".TMS320C33", tic4x_cpu == 33);
1207 tic4x_insert_sym (".tms320C33", tic4x_cpu == 33);
1208 tic4x_insert_sym (".TMS320C40", tic4x_cpu == 40 || tic4x_cpu == 44 || tic4x_cpu == 0);
1209 tic4x_insert_sym (".tms320C40", tic4x_cpu == 40 || tic4x_cpu == 44 || tic4x_cpu == 0);
1210 tic4x_insert_sym (".TMS320C44", tic4x_cpu == 44);
1211 tic4x_insert_sym (".tms320C44", tic4x_cpu == 44);
1212 tic4x_insert_sym (".TMX320C40", 0); /* C40 first pass silicon ? */
1213 tic4x_insert_sym (".tmx320C40", 0);
1214 }
1215
1216 /* Insert a new instruction template into hash table. */
1217 static int
tic4x_inst_insert(const tic4x_inst_t * inst)1218 tic4x_inst_insert (const tic4x_inst_t *inst)
1219 {
1220 static char prev_name[16];
1221 const char *retval = NULL;
1222
1223 /* Only insert the first name if have several similar entries. */
1224 if (!strcmp (inst->name, prev_name) || inst->name[0] == '\0')
1225 return 1;
1226
1227 retval = hash_insert (tic4x_op_hash, inst->name, (void *) inst);
1228 if (retval != NULL)
1229 fprintf (stderr, "internal error: can't hash `%s': %s\n",
1230 inst->name, retval);
1231 else
1232 strcpy (prev_name, inst->name);
1233 return retval == NULL;
1234 }
1235
1236 /* Make a new instruction template. */
1237 static tic4x_inst_t *
tic4x_inst_make(char * name,unsigned long opcode,char * args)1238 tic4x_inst_make (char *name, unsigned long opcode, char *args)
1239 {
1240 static tic4x_inst_t *insts = NULL;
1241 static char *names = NULL;
1242 static int iindex = 0;
1243
1244 if (insts == NULL)
1245 {
1246 /* Allocate memory to store name strings. */
1247 names = (char *) xmalloc (sizeof (char) * 8192);
1248 /* Allocate memory for additional insts. */
1249 insts = (tic4x_inst_t *)
1250 xmalloc (sizeof (tic4x_inst_t) * 1024);
1251 }
1252 insts[iindex].name = names;
1253 insts[iindex].opcode = opcode;
1254 insts[iindex].opmask = 0xffffffff;
1255 insts[iindex].args = args;
1256 iindex++;
1257
1258 do
1259 *names++ = *name++;
1260 while (*name);
1261 *names++ = '\0';
1262
1263 return &insts[iindex - 1];
1264 }
1265
1266 /* Add instruction template, creating dynamic templates as required. */
1267 static int
tic4x_inst_add(const tic4x_inst_t * insts)1268 tic4x_inst_add (const tic4x_inst_t *insts)
1269 {
1270 char *s = insts->name;
1271 char *d;
1272 unsigned int i;
1273 int ok = 1;
1274 char name[16];
1275
1276 d = name;
1277
1278 /* We do not care about INSNs that is not a part of our
1279 oplevel setting. */
1280 if ((insts->oplevel & tic4x_oplevel) == 0)
1281 return ok;
1282
1283 while (1)
1284 {
1285 switch (*s)
1286 {
1287 case 'B':
1288 case 'C':
1289 /* Dynamically create all the conditional insts. */
1290 for (i = 0; i < tic4x_num_conds; i++)
1291 {
1292 tic4x_inst_t *inst;
1293 int k = 0;
1294 char *c = tic4x_conds[i].name;
1295 char *e = d;
1296
1297 while (*c)
1298 *e++ = *c++;
1299 c = s + 1;
1300 while (*c)
1301 *e++ = *c++;
1302 *e = '\0';
1303
1304 /* If instruction found then have already processed it. */
1305 if (hash_find (tic4x_op_hash, name))
1306 return 1;
1307
1308 do
1309 {
1310 inst = tic4x_inst_make (name, insts[k].opcode +
1311 (tic4x_conds[i].cond <<
1312 (*s == 'B' ? 16 : 23)),
1313 insts[k].args);
1314 if (k == 0) /* Save strcmp() with following func. */
1315 ok &= tic4x_inst_insert (inst);
1316 k++;
1317 }
1318 while (!strcmp (insts->name,
1319 insts[k].name));
1320 }
1321 return ok;
1322 break;
1323
1324 case '\0':
1325 return tic4x_inst_insert (insts);
1326 break;
1327
1328 default:
1329 *d++ = *s++;
1330 break;
1331 }
1332 }
1333 }
1334
1335 /* This function is called once, at assembler startup time. It should
1336 set up all the tables, etc., that the MD part of the assembler will
1337 need. */
1338 void
md_begin(void)1339 md_begin (void)
1340 {
1341 int ok = 1;
1342 unsigned int i;
1343
1344 /* Setup the proper opcode level according to the
1345 commandline parameters */
1346 tic4x_oplevel = OP_C3X;
1347
1348 if ( IS_CPU_TIC4X(tic4x_cpu) )
1349 tic4x_oplevel |= OP_C4X;
1350
1351 if ( ( tic4x_cpu == 31 && tic4x_revision >= 6)
1352 || (tic4x_cpu == 32 && tic4x_revision >= 2)
1353 || (tic4x_cpu == 33)
1354 || tic4x_enhanced )
1355 tic4x_oplevel |= OP_ENH;
1356
1357 if ( ( tic4x_cpu == 30 && tic4x_revision >= 7)
1358 || (tic4x_cpu == 31 && tic4x_revision >= 5)
1359 || (tic4x_cpu == 32)
1360 || tic4x_lowpower )
1361 tic4x_oplevel |= OP_LPWR;
1362
1363 if ( ( tic4x_cpu == 30 && tic4x_revision >= 7)
1364 || (tic4x_cpu == 31 && tic4x_revision >= 5)
1365 || (tic4x_cpu == 32)
1366 || (tic4x_cpu == 33)
1367 || (tic4x_cpu == 40 && tic4x_revision >= 5)
1368 || (tic4x_cpu == 44)
1369 || tic4x_idle2 )
1370 tic4x_oplevel |= OP_IDLE2;
1371
1372 /* Create hash table for mnemonics. */
1373 tic4x_op_hash = hash_new ();
1374
1375 /* Create hash table for asg pseudo. */
1376 tic4x_asg_hash = hash_new ();
1377
1378 /* Add mnemonics to hash table, expanding conditional mnemonics on fly. */
1379 for (i = 0; i < tic4x_num_insts; i++)
1380 ok &= tic4x_inst_add (tic4x_insts + i);
1381
1382 /* Create dummy inst to avoid errors accessing end of table. */
1383 tic4x_inst_make ("", 0, "");
1384
1385 if (!ok)
1386 as_fatal ("Broken assembler. No assembly attempted.");
1387
1388 /* Add registers to symbol table. */
1389 tic4x_init_regtable ();
1390
1391 /* Add predefined symbols to symbol table. */
1392 tic4x_init_symbols ();
1393 }
1394
1395 void
tic4x_end(void)1396 tic4x_end (void)
1397 {
1398 bfd_set_arch_mach (stdoutput, bfd_arch_tic4x,
1399 IS_CPU_TIC4X (tic4x_cpu) ? bfd_mach_tic4x : bfd_mach_tic3x);
1400 }
1401
1402 static int
tic4x_indirect_parse(tic4x_operand_t * operand,const tic4x_indirect_t * indirect)1403 tic4x_indirect_parse (tic4x_operand_t *operand,
1404 const tic4x_indirect_t *indirect)
1405 {
1406 char *n = indirect->name;
1407 char *s = input_line_pointer;
1408 char *b;
1409 symbolS *symbolP;
1410 char name[32];
1411
1412 operand->disp = 0;
1413 for (; *n; n++)
1414 {
1415 switch (*n)
1416 {
1417 case 'a': /* Need to match aux register. */
1418 b = name;
1419 #ifdef TIC4X_ALT_SYNTAX
1420 if (*s == '%')
1421 s++;
1422 #endif
1423 while (ISALNUM (*s))
1424 *b++ = *s++;
1425 *b++ = '\0';
1426 if (!(symbolP = symbol_find (name)))
1427 return 0;
1428
1429 if (S_GET_SEGMENT (symbolP) != reg_section)
1430 return 0;
1431
1432 operand->aregno = S_GET_VALUE (symbolP);
1433 if (operand->aregno >= REG_AR0 && operand->aregno <= REG_AR7)
1434 break;
1435
1436 as_bad (_("Auxiliary register AR0--AR7 required for indirect"));
1437 return -1;
1438
1439 case 'd': /* Need to match constant for disp. */
1440 #ifdef TIC4X_ALT_SYNTAX
1441 if (*s == '%') /* expr() will die if we don't skip this. */
1442 s++;
1443 #endif
1444 s = tic4x_expression (s, &operand->expr);
1445 if (operand->expr.X_op != O_constant)
1446 return 0;
1447 operand->disp = operand->expr.X_add_number;
1448 if (operand->disp < 0 || operand->disp > 255)
1449 {
1450 as_bad (_("Bad displacement %d (require 0--255)\n"),
1451 operand->disp);
1452 return -1;
1453 }
1454 break;
1455
1456 case 'y': /* Need to match IR0. */
1457 case 'z': /* Need to match IR1. */
1458 #ifdef TIC4X_ALT_SYNTAX
1459 if (*s == '%')
1460 s++;
1461 #endif
1462 s = tic4x_expression (s, &operand->expr);
1463 if (operand->expr.X_op != O_register)
1464 return 0;
1465 if (operand->expr.X_add_number != REG_IR0
1466 && operand->expr.X_add_number != REG_IR1)
1467 {
1468 as_bad (_("Index register IR0,IR1 required for displacement"));
1469 return -1;
1470 }
1471
1472 if (*n == 'y' && operand->expr.X_add_number == REG_IR0)
1473 break;
1474 if (*n == 'z' && operand->expr.X_add_number == REG_IR1)
1475 break;
1476 return 0;
1477
1478 case '(':
1479 if (*s != '(') /* No displacement, assume to be 1. */
1480 {
1481 operand->disp = 1;
1482 while (*n != ')')
1483 n++;
1484 }
1485 else
1486 s++;
1487 break;
1488
1489 default:
1490 if (TOLOWER (*s) != *n)
1491 return 0;
1492 s++;
1493 }
1494 }
1495 if (*s != ' ' && *s != ',' && *s != '\0')
1496 return 0;
1497 input_line_pointer = s;
1498 return 1;
1499 }
1500
1501 static char *
tic4x_operand_parse(char * s,tic4x_operand_t * operand)1502 tic4x_operand_parse (char *s, tic4x_operand_t *operand)
1503 {
1504 unsigned int i;
1505 char c;
1506 int ret;
1507 expressionS *exp = &operand->expr;
1508 char *save = input_line_pointer;
1509 char *str;
1510 char *new_pointer;
1511 struct hash_entry *entry = NULL;
1512
1513 input_line_pointer = s;
1514 SKIP_WHITESPACE ();
1515
1516 str = input_line_pointer;
1517 c = get_symbol_end (); /* Get terminator. */
1518 new_pointer = input_line_pointer;
1519 if (strlen (str) && (entry = hash_find (tic4x_asg_hash, str)) != NULL)
1520 {
1521 *input_line_pointer = c;
1522 input_line_pointer = (char *) entry;
1523 }
1524 else
1525 {
1526 *input_line_pointer = c;
1527 input_line_pointer = str;
1528 }
1529
1530 operand->mode = M_UNKNOWN;
1531 switch (*input_line_pointer)
1532 {
1533 #ifdef TIC4X_ALT_SYNTAX
1534 case '%':
1535 input_line_pointer = tic4x_expression (++input_line_pointer, exp);
1536 if (exp->X_op != O_register)
1537 as_bad (_("Expecting a register name"));
1538 operand->mode = M_REGISTER;
1539 break;
1540
1541 case '^':
1542 /* Denotes high 16 bits. */
1543 input_line_pointer = tic4x_expression (++input_line_pointer, exp);
1544 if (exp->X_op == O_constant)
1545 operand->mode = M_IMMED;
1546 else if (exp->X_op == O_big)
1547 {
1548 if (exp->X_add_number)
1549 as_bad (_("Number too large")); /* bignum required */
1550 else
1551 {
1552 tic4x_gen_to_words (generic_floating_point_number,
1553 operand->fwords, S_PRECISION);
1554 operand->mode = M_IMMED_F;
1555 }
1556 }
1557 /* Allow ori ^foo, ar0 to be equivalent to ldi .hi.foo, ar0 */
1558 /* WARNING : The TI C40 assembler cannot do this. */
1559 else if (exp->X_op == O_symbol)
1560 {
1561 operand->mode = M_HI;
1562 break;
1563 }
1564
1565 case '#':
1566 input_line_pointer = tic4x_expression (++input_line_pointer, exp);
1567 if (exp->X_op == O_constant)
1568 operand->mode = M_IMMED;
1569 else if (exp->X_op == O_big)
1570 {
1571 if (exp->X_add_number > 0)
1572 as_bad (_("Number too large")); /* bignum required. */
1573 else
1574 {
1575 tic4x_gen_to_words (generic_floating_point_number,
1576 operand->fwords, S_PRECISION);
1577 operand->mode = M_IMMED_F;
1578 }
1579 }
1580 /* Allow ori foo, ar0 to be equivalent to ldi .lo.foo, ar0 */
1581 /* WARNING : The TI C40 assembler cannot do this. */
1582 else if (exp->X_op == O_symbol)
1583 {
1584 operand->mode = M_IMMED;
1585 break;
1586 }
1587
1588 else
1589 as_bad (_("Expecting a constant value"));
1590 break;
1591 case '\\':
1592 #endif
1593 case '@':
1594 input_line_pointer = tic4x_expression (++input_line_pointer, exp);
1595 if (exp->X_op != O_constant && exp->X_op != O_symbol)
1596 as_bad (_("Bad direct addressing construct %s"), s);
1597 if (exp->X_op == O_constant)
1598 {
1599 if (exp->X_add_number < 0)
1600 as_bad (_("Direct value of %ld is not suitable"),
1601 (long) exp->X_add_number);
1602 }
1603 operand->mode = M_DIRECT;
1604 break;
1605
1606 case '*':
1607 ret = -1;
1608 for (i = 0; i < tic4x_num_indirects; i++)
1609 if ((ret = tic4x_indirect_parse (operand, &tic4x_indirects[i])))
1610 break;
1611 if (ret < 0)
1612 break;
1613 if (i < tic4x_num_indirects)
1614 {
1615 operand->mode = M_INDIRECT;
1616 /* Indirect addressing mode number. */
1617 operand->expr.X_add_number = tic4x_indirects[i].modn;
1618 /* Convert *+ARn(0) to *ARn etc. Maybe we should
1619 squeal about silly ones? */
1620 if (operand->expr.X_add_number < 0x08 && !operand->disp)
1621 operand->expr.X_add_number = 0x18;
1622 }
1623 else
1624 as_bad (_("Unknown indirect addressing mode"));
1625 break;
1626
1627 default:
1628 operand->mode = M_IMMED; /* Assume immediate. */
1629 str = input_line_pointer;
1630 input_line_pointer = tic4x_expression (input_line_pointer, exp);
1631 if (exp->X_op == O_register)
1632 {
1633 know (exp->X_add_symbol == 0);
1634 know (exp->X_op_symbol == 0);
1635 operand->mode = M_REGISTER;
1636 break;
1637 }
1638 else if (exp->X_op == O_big)
1639 {
1640 if (exp->X_add_number > 0)
1641 as_bad (_("Number too large")); /* bignum required. */
1642 else
1643 {
1644 tic4x_gen_to_words (generic_floating_point_number,
1645 operand->fwords, S_PRECISION);
1646 operand->mode = M_IMMED_F;
1647 }
1648 break;
1649 }
1650 #ifdef TIC4X_ALT_SYNTAX
1651 /* Allow ldi foo, ar0 to be equivalent to ldi @foo, ar0. */
1652 else if (exp->X_op == O_symbol)
1653 {
1654 operand->mode = M_DIRECT;
1655 break;
1656 }
1657 #endif
1658 }
1659 if (entry == NULL)
1660 new_pointer = input_line_pointer;
1661 input_line_pointer = save;
1662 return new_pointer;
1663 }
1664
1665 static int
tic4x_operands_match(tic4x_inst_t * inst,tic4x_insn_t * tinsn,int check)1666 tic4x_operands_match (tic4x_inst_t *inst, tic4x_insn_t *tinsn, int check)
1667 {
1668 const char *args = inst->args;
1669 unsigned long opcode = inst->opcode;
1670 int num_operands = tinsn->num_operands;
1671 tic4x_operand_t *operand = tinsn->operands;
1672 expressionS *exp = &operand->expr;
1673 int ret = 1;
1674 int reg;
1675
1676 /* Build the opcode, checking as we go to make sure that the
1677 operands match.
1678
1679 If an operand matches, we modify insn or opcode appropriately,
1680 and do a "continue". If an operand fails to match, we "break". */
1681
1682 tinsn->nchars = 4; /* Instructions always 4 bytes. */
1683 tinsn->reloc = NO_RELOC;
1684 tinsn->pcrel = 0;
1685
1686 if (*args == '\0')
1687 {
1688 tinsn->opcode = opcode;
1689 return num_operands == 0;
1690 }
1691
1692 for (;; ++args)
1693 {
1694 switch (*args)
1695 {
1696
1697 case '\0': /* End of args. */
1698 if (num_operands == 1)
1699 {
1700 tinsn->opcode = opcode;
1701 return ret;
1702 }
1703 break; /* Too many operands. */
1704
1705 case '#': /* This is only used for ldp. */
1706 if (operand->mode != M_DIRECT && operand->mode != M_IMMED)
1707 break;
1708 /* While this looks like a direct addressing mode, we actually
1709 use an immediate mode form of ldiu or ldpk instruction. */
1710 if (exp->X_op == O_constant)
1711 {
1712 if( ( IS_CPU_TIC4X (tic4x_cpu) && exp->X_add_number <= 65535 )
1713 || ( IS_CPU_TIC3X (tic4x_cpu) && exp->X_add_number <= 255 ) )
1714 {
1715 INSERTS (opcode, exp->X_add_number, 15, 0);
1716 continue;
1717 }
1718 else
1719 {
1720 if (!check)
1721 as_bad (_("Immediate value of %ld is too large for ldf"),
1722 (long) exp->X_add_number);
1723 ret = -1;
1724 continue;
1725 }
1726 }
1727 else if (exp->X_op == O_symbol)
1728 {
1729 tinsn->reloc = BFD_RELOC_HI16;
1730 tinsn->exp = *exp;
1731 continue;
1732 }
1733 break; /* Not direct (dp) addressing. */
1734
1735 case '@': /* direct. */
1736 if (operand->mode != M_DIRECT)
1737 break;
1738 if (exp->X_op == O_constant)
1739 {
1740 /* Store only the 16 LSBs of the number. */
1741 INSERTS (opcode, exp->X_add_number, 15, 0);
1742 continue;
1743 }
1744 else if (exp->X_op == O_symbol)
1745 {
1746 tinsn->reloc = BFD_RELOC_LO16;
1747 tinsn->exp = *exp;
1748 continue;
1749 }
1750 break; /* Not direct addressing. */
1751
1752 case 'A':
1753 if (operand->mode != M_REGISTER)
1754 break;
1755 reg = exp->X_add_number;
1756 if (reg >= REG_AR0 && reg <= REG_AR7)
1757 INSERTU (opcode, reg - REG_AR0, 24, 22);
1758 else
1759 {
1760 if (!check)
1761 as_bad (_("Destination register must be ARn"));
1762 ret = -1;
1763 }
1764 continue;
1765
1766 case 'B': /* Unsigned integer immediate. */
1767 /* Allow br label or br @label. */
1768 if (operand->mode != M_IMMED && operand->mode != M_DIRECT)
1769 break;
1770 if (exp->X_op == O_constant)
1771 {
1772 if (exp->X_add_number < (1 << 24))
1773 {
1774 INSERTU (opcode, exp->X_add_number, 23, 0);
1775 continue;
1776 }
1777 else
1778 {
1779 if (!check)
1780 as_bad (_("Immediate value of %ld is too large"),
1781 (long) exp->X_add_number);
1782 ret = -1;
1783 continue;
1784 }
1785 }
1786 if (IS_CPU_TIC4X (tic4x_cpu))
1787 {
1788 tinsn->reloc = BFD_RELOC_24_PCREL;
1789 tinsn->pcrel = 1;
1790 }
1791 else
1792 {
1793 tinsn->reloc = BFD_RELOC_24;
1794 tinsn->pcrel = 0;
1795 }
1796 tinsn->exp = *exp;
1797 continue;
1798
1799 case 'C':
1800 if (!IS_CPU_TIC4X (tic4x_cpu))
1801 break;
1802 if (operand->mode != M_INDIRECT)
1803 break;
1804 /* Require either *+ARn(disp) or *ARn. */
1805 if (operand->expr.X_add_number != 0
1806 && operand->expr.X_add_number != 0x18)
1807 {
1808 if (!check)
1809 as_bad (_("Invalid indirect addressing mode"));
1810 ret = -1;
1811 continue;
1812 }
1813 INSERTU (opcode, operand->aregno - REG_AR0, 2, 0);
1814 INSERTU (opcode, operand->disp, 7, 3);
1815 continue;
1816
1817 case 'E':
1818 if (!(operand->mode == M_REGISTER))
1819 break;
1820 INSERTU (opcode, exp->X_add_number, 7, 0);
1821 continue;
1822
1823 case 'e':
1824 if (!(operand->mode == M_REGISTER))
1825 break;
1826 reg = exp->X_add_number;
1827 if ( (reg >= REG_R0 && reg <= REG_R7)
1828 || (IS_CPU_TIC4X (tic4x_cpu) && reg >= REG_R8 && reg <= REG_R11) )
1829 INSERTU (opcode, reg, 7, 0);
1830 else
1831 {
1832 if (!check)
1833 as_bad (_("Register must be Rn"));
1834 ret = -1;
1835 }
1836 continue;
1837
1838 case 'F':
1839 if (operand->mode != M_IMMED_F
1840 && !(operand->mode == M_IMMED && exp->X_op == O_constant))
1841 break;
1842
1843 if (operand->mode != M_IMMED_F)
1844 {
1845 /* OK, we 've got something like cmpf 0, r0
1846 Why can't they stick in a bloody decimal point ?! */
1847 char string[16];
1848
1849 /* Create floating point number string. */
1850 sprintf (string, "%d.0", (int) exp->X_add_number);
1851 tic4x_atof (string, 's', operand->fwords);
1852 }
1853
1854 INSERTU (opcode, operand->fwords[0], 15, 0);
1855 continue;
1856
1857 case 'G':
1858 if (operand->mode != M_REGISTER)
1859 break;
1860 INSERTU (opcode, exp->X_add_number, 15, 8);
1861 continue;
1862
1863 case 'g':
1864 if (operand->mode != M_REGISTER)
1865 break;
1866 reg = exp->X_add_number;
1867 if ( (reg >= REG_R0 && reg <= REG_R7)
1868 || (IS_CPU_TIC4X (tic4x_cpu) && reg >= REG_R8 && reg <= REG_R11) )
1869 INSERTU (opcode, reg, 15, 8);
1870 else
1871 {
1872 if (!check)
1873 as_bad (_("Register must be Rn"));
1874 ret = -1;
1875 }
1876 continue;
1877
1878 case 'H':
1879 if (operand->mode != M_REGISTER)
1880 break;
1881 reg = exp->X_add_number;
1882 if (reg >= REG_R0 && reg <= REG_R7)
1883 INSERTU (opcode, reg - REG_R0, 18, 16);
1884 else
1885 {
1886 if (!check)
1887 as_bad (_("Register must be R0--R7"));
1888 ret = -1;
1889 }
1890 continue;
1891
1892 case 'i':
1893 if ( operand->mode == M_REGISTER
1894 && tic4x_oplevel & OP_ENH )
1895 {
1896 reg = exp->X_add_number;
1897 INSERTU (opcode, reg, 4, 0);
1898 INSERTU (opcode, 7, 7, 5);
1899 continue;
1900 }
1901 /* Fallthrough */
1902
1903 case 'I':
1904 if (operand->mode != M_INDIRECT)
1905 break;
1906 if (operand->disp != 0 && operand->disp != 1)
1907 {
1908 if (IS_CPU_TIC4X (tic4x_cpu))
1909 break;
1910 if (!check)
1911 as_bad (_("Invalid indirect addressing mode displacement %d"),
1912 operand->disp);
1913 ret = -1;
1914 continue;
1915 }
1916 INSERTU (opcode, operand->aregno - REG_AR0, 2, 0);
1917 INSERTU (opcode, operand->expr.X_add_number, 7, 3);
1918 continue;
1919
1920 case 'j':
1921 if ( operand->mode == M_REGISTER
1922 && tic4x_oplevel & OP_ENH )
1923 {
1924 reg = exp->X_add_number;
1925 INSERTU (opcode, reg, 12, 8);
1926 INSERTU (opcode, 7, 15, 13);
1927 continue;
1928 }
1929 /* Fallthrough */
1930
1931 case 'J':
1932 if (operand->mode != M_INDIRECT)
1933 break;
1934 if (operand->disp != 0 && operand->disp != 1)
1935 {
1936 if (IS_CPU_TIC4X (tic4x_cpu))
1937 break;
1938 if (!check)
1939 as_bad (_("Invalid indirect addressing mode displacement %d"),
1940 operand->disp);
1941 ret = -1;
1942 continue;
1943 }
1944 INSERTU (opcode, operand->aregno - REG_AR0, 10, 8);
1945 INSERTU (opcode, operand->expr.X_add_number, 15, 11);
1946 continue;
1947
1948 case 'K':
1949 if (operand->mode != M_REGISTER)
1950 break;
1951 reg = exp->X_add_number;
1952 if (reg >= REG_R0 && reg <= REG_R7)
1953 INSERTU (opcode, reg - REG_R0, 21, 19);
1954 else
1955 {
1956 if (!check)
1957 as_bad (_("Register must be R0--R7"));
1958 ret = -1;
1959 }
1960 continue;
1961
1962 case 'L':
1963 if (operand->mode != M_REGISTER)
1964 break;
1965 reg = exp->X_add_number;
1966 if (reg >= REG_R0 && reg <= REG_R7)
1967 INSERTU (opcode, reg - REG_R0, 24, 22);
1968 else
1969 {
1970 if (!check)
1971 as_bad (_("Register must be R0--R7"));
1972 ret = -1;
1973 }
1974 continue;
1975
1976 case 'M':
1977 if (operand->mode != M_REGISTER)
1978 break;
1979 reg = exp->X_add_number;
1980 if (reg == REG_R2 || reg == REG_R3)
1981 INSERTU (opcode, reg - REG_R2, 22, 22);
1982 else
1983 {
1984 if (!check)
1985 as_bad (_("Destination register must be R2 or R3"));
1986 ret = -1;
1987 }
1988 continue;
1989
1990 case 'N':
1991 if (operand->mode != M_REGISTER)
1992 break;
1993 reg = exp->X_add_number;
1994 if (reg == REG_R0 || reg == REG_R1)
1995 INSERTU (opcode, reg - REG_R0, 23, 23);
1996 else
1997 {
1998 if (!check)
1999 as_bad (_("Destination register must be R0 or R1"));
2000 ret = -1;
2001 }
2002 continue;
2003
2004 case 'O':
2005 if (!IS_CPU_TIC4X (tic4x_cpu))
2006 break;
2007 if (operand->mode != M_INDIRECT)
2008 break;
2009 /* Require either *+ARn(disp) or *ARn. */
2010 if (operand->expr.X_add_number != 0
2011 && operand->expr.X_add_number != 0x18)
2012 {
2013 if (!check)
2014 as_bad (_("Invalid indirect addressing mode"));
2015 ret = -1;
2016 continue;
2017 }
2018 INSERTU (opcode, operand->aregno - REG_AR0, 10, 8);
2019 INSERTU (opcode, operand->disp, 15, 11);
2020 continue;
2021
2022 case 'P': /* PC relative displacement. */
2023 /* Allow br label or br @label. */
2024 if (operand->mode != M_IMMED && operand->mode != M_DIRECT)
2025 break;
2026 if (exp->X_op == O_constant)
2027 {
2028 if (exp->X_add_number >= -32768 && exp->X_add_number <= 32767)
2029 {
2030 INSERTS (opcode, exp->X_add_number, 15, 0);
2031 continue;
2032 }
2033 else
2034 {
2035 if (!check)
2036 as_bad (_("Displacement value of %ld is too large"),
2037 (long) exp->X_add_number);
2038 ret = -1;
2039 continue;
2040 }
2041 }
2042 tinsn->reloc = BFD_RELOC_16_PCREL;
2043 tinsn->pcrel = 1;
2044 tinsn->exp = *exp;
2045 continue;
2046
2047 case 'Q':
2048 if (operand->mode != M_REGISTER)
2049 break;
2050 reg = exp->X_add_number;
2051 INSERTU (opcode, reg, 15, 0);
2052 continue;
2053
2054 case 'q':
2055 if (operand->mode != M_REGISTER)
2056 break;
2057 reg = exp->X_add_number;
2058 if ( (reg >= REG_R0 && reg <= REG_R7)
2059 || (IS_CPU_TIC4X (tic4x_cpu) && reg >= REG_R8 && reg <= REG_R11) )
2060 INSERTU (opcode, reg, 15, 0);
2061 else
2062 {
2063 if (!check)
2064 as_bad (_("Register must be Rn"));
2065 ret = -1;
2066 }
2067 continue;
2068
2069 case 'R':
2070 if (operand->mode != M_REGISTER)
2071 break;
2072 reg = exp->X_add_number;
2073 INSERTU (opcode, reg, 20, 16);
2074 continue;
2075
2076 case 'r':
2077 if (operand->mode != M_REGISTER)
2078 break;
2079 reg = exp->X_add_number;
2080 if ( (reg >= REG_R0 && reg <= REG_R7)
2081 || (IS_CPU_TIC4X (tic4x_cpu) && reg >= REG_R8 && reg <= REG_R11) )
2082 INSERTU (opcode, reg, 20, 16);
2083 else
2084 {
2085 if (!check)
2086 as_bad (_("Register must be Rn"));
2087 ret = -1;
2088 }
2089 continue;
2090
2091 case 'S': /* Short immediate int. */
2092 if (operand->mode != M_IMMED && operand->mode != M_HI)
2093 break;
2094 if (exp->X_op == O_big)
2095 {
2096 if (!check)
2097 as_bad (_("Floating point number not valid in expression"));
2098 ret = -1;
2099 continue;
2100 }
2101 if (exp->X_op == O_constant)
2102 {
2103 if (exp->X_add_number >= -32768 && exp->X_add_number <= 65535)
2104 {
2105 INSERTS (opcode, exp->X_add_number, 15, 0);
2106 continue;
2107 }
2108 else
2109 {
2110 if (!check)
2111 as_bad (_("Signed immediate value %ld too large"),
2112 (long) exp->X_add_number);
2113 ret = -1;
2114 continue;
2115 }
2116 }
2117 else if (exp->X_op == O_symbol)
2118 {
2119 if (operand->mode == M_HI)
2120 {
2121 tinsn->reloc = BFD_RELOC_HI16;
2122 }
2123 else
2124 {
2125 tinsn->reloc = BFD_RELOC_LO16;
2126 }
2127 tinsn->exp = *exp;
2128 continue;
2129 }
2130 /* Handle cases like ldi foo - $, ar0 where foo
2131 is a forward reference. Perhaps we should check
2132 for X_op == O_symbol and disallow things like
2133 ldi foo, ar0. */
2134 tinsn->reloc = BFD_RELOC_16;
2135 tinsn->exp = *exp;
2136 continue;
2137
2138 case 'T': /* 5-bit immediate value for tic4x stik. */
2139 if (!IS_CPU_TIC4X (tic4x_cpu))
2140 break;
2141 if (operand->mode != M_IMMED)
2142 break;
2143 if (exp->X_op == O_constant)
2144 {
2145 if (exp->X_add_number < 16 && exp->X_add_number >= -16)
2146 {
2147 INSERTS (opcode, exp->X_add_number, 20, 16);
2148 continue;
2149 }
2150 else
2151 {
2152 if (!check)
2153 as_bad (_("Immediate value of %ld is too large"),
2154 (long) exp->X_add_number);
2155 ret = -1;
2156 continue;
2157 }
2158 }
2159 break; /* No relocations allowed. */
2160
2161 case 'U': /* Unsigned integer immediate. */
2162 if (operand->mode != M_IMMED && operand->mode != M_HI)
2163 break;
2164 if (exp->X_op == O_constant)
2165 {
2166 if (exp->X_add_number < (1 << 16) && exp->X_add_number >= 0)
2167 {
2168 INSERTU (opcode, exp->X_add_number, 15, 0);
2169 continue;
2170 }
2171 else
2172 {
2173 if (!check)
2174 as_bad (_("Unsigned immediate value %ld too large"),
2175 (long) exp->X_add_number);
2176 ret = -1;
2177 continue;
2178 }
2179 }
2180 else if (exp->X_op == O_symbol)
2181 {
2182 if (operand->mode == M_HI)
2183 tinsn->reloc = BFD_RELOC_HI16;
2184 else
2185 tinsn->reloc = BFD_RELOC_LO16;
2186
2187 tinsn->exp = *exp;
2188 continue;
2189 }
2190 tinsn->reloc = BFD_RELOC_16;
2191 tinsn->exp = *exp;
2192 continue;
2193
2194 case 'V': /* Trap numbers (immediate field). */
2195 if (operand->mode != M_IMMED)
2196 break;
2197 if (exp->X_op == O_constant)
2198 {
2199 if (exp->X_add_number < 512 && IS_CPU_TIC4X (tic4x_cpu))
2200 {
2201 INSERTU (opcode, exp->X_add_number, 8, 0);
2202 continue;
2203 }
2204 else if (exp->X_add_number < 32 && IS_CPU_TIC3X (tic4x_cpu))
2205 {
2206 INSERTU (opcode, exp->X_add_number | 0x20, 4, 0);
2207 continue;
2208 }
2209 else
2210 {
2211 if (!check)
2212 as_bad (_("Immediate value of %ld is too large"),
2213 (long) exp->X_add_number);
2214 ret = -1;
2215 continue;
2216 }
2217 }
2218 break; /* No relocations allowed. */
2219
2220 case 'W': /* Short immediate int (0--7). */
2221 if (!IS_CPU_TIC4X (tic4x_cpu))
2222 break;
2223 if (operand->mode != M_IMMED)
2224 break;
2225 if (exp->X_op == O_big)
2226 {
2227 if (!check)
2228 as_bad (_("Floating point number not valid in expression"));
2229 ret = -1;
2230 continue;
2231 }
2232 if (exp->X_op == O_constant)
2233 {
2234 if (exp->X_add_number >= -256 && exp->X_add_number <= 127)
2235 {
2236 INSERTS (opcode, exp->X_add_number, 7, 0);
2237 continue;
2238 }
2239 else
2240 {
2241 if (!check)
2242 as_bad (_("Immediate value %ld too large"),
2243 (long) exp->X_add_number);
2244 ret = -1;
2245 continue;
2246 }
2247 }
2248 tinsn->reloc = BFD_RELOC_16;
2249 tinsn->exp = *exp;
2250 continue;
2251
2252 case 'X': /* Expansion register for tic4x. */
2253 if (operand->mode != M_REGISTER)
2254 break;
2255 reg = exp->X_add_number;
2256 if (reg >= REG_IVTP && reg <= REG_TVTP)
2257 INSERTU (opcode, reg - REG_IVTP, 4, 0);
2258 else
2259 {
2260 if (!check)
2261 as_bad (_("Register must be ivtp or tvtp"));
2262 ret = -1;
2263 }
2264 continue;
2265
2266 case 'Y': /* Address register for tic4x lda. */
2267 if (operand->mode != M_REGISTER)
2268 break;
2269 reg = exp->X_add_number;
2270 if (reg >= REG_AR0 && reg <= REG_SP)
2271 INSERTU (opcode, reg, 20, 16);
2272 else
2273 {
2274 if (!check)
2275 as_bad (_("Register must be address register"));
2276 ret = -1;
2277 }
2278 continue;
2279
2280 case 'Z': /* Expansion register for tic4x. */
2281 if (operand->mode != M_REGISTER)
2282 break;
2283 reg = exp->X_add_number;
2284 if (reg >= REG_IVTP && reg <= REG_TVTP)
2285 INSERTU (opcode, reg - REG_IVTP, 20, 16);
2286 else
2287 {
2288 if (!check)
2289 as_bad (_("Register must be ivtp or tvtp"));
2290 ret = -1;
2291 }
2292 continue;
2293
2294 case '*':
2295 if (operand->mode != M_INDIRECT)
2296 break;
2297 INSERTS (opcode, operand->disp, 7, 0);
2298 INSERTU (opcode, operand->aregno - REG_AR0, 10, 8);
2299 INSERTU (opcode, operand->expr.X_add_number, 15, 11);
2300 continue;
2301
2302 case '|': /* treat as `,' if have ldi_ldi form. */
2303 if (tinsn->parallel)
2304 {
2305 if (--num_operands < 0)
2306 break; /* Too few operands. */
2307 operand++;
2308 if (operand->mode != M_PARALLEL)
2309 break;
2310 }
2311 /* Fall through. */
2312
2313 case ',': /* Another operand. */
2314 if (--num_operands < 0)
2315 break; /* Too few operands. */
2316 operand++;
2317 exp = &operand->expr;
2318 continue;
2319
2320 case ';': /* Another optional operand. */
2321 if (num_operands == 1 || operand[1].mode == M_PARALLEL)
2322 continue;
2323 if (--num_operands < 0)
2324 break; /* Too few operands. */
2325 operand++;
2326 exp = &operand->expr;
2327 continue;
2328
2329 default:
2330 BAD_CASE (*args);
2331 }
2332 return 0;
2333 }
2334 }
2335
2336 static void
tic4x_insn_check(tic4x_insn_t * tinsn)2337 tic4x_insn_check (tic4x_insn_t *tinsn)
2338 {
2339
2340 if (!strcmp (tinsn->name, "lda"))
2341 {
2342 if (tinsn->num_operands < 2 || tinsn->num_operands > 2)
2343 as_fatal ("Illegal internal LDA insn definition");
2344
2345 if (tinsn->operands[0].mode == M_REGISTER
2346 && tinsn->operands[1].mode == M_REGISTER
2347 && tinsn->operands[0].expr.X_add_number == tinsn->operands[1].expr.X_add_number )
2348 as_bad (_("Source and destination register should not be equal"));
2349 }
2350 else if (!strcmp (tinsn->name, "ldi_ldi")
2351 || !strcmp (tinsn->name, "ldi1_ldi2")
2352 || !strcmp (tinsn->name, "ldi2_ldi1")
2353 || !strcmp (tinsn->name, "ldf_ldf")
2354 || !strcmp (tinsn->name, "ldf1_ldf2")
2355 || !strcmp (tinsn->name, "ldf2_ldf1") )
2356 {
2357 if (tinsn->num_operands < 4 && tinsn->num_operands > 5 )
2358 as_fatal ("Illegal internal %s insn definition", tinsn->name);
2359
2360 if (tinsn->operands[1].mode == M_REGISTER
2361 && tinsn->operands[tinsn->num_operands-1].mode == M_REGISTER
2362 && tinsn->operands[1].expr.X_add_number == tinsn->operands[tinsn->num_operands-1].expr.X_add_number )
2363 as_warn (_("Equal parallell destination registers, one result will be discarded"));
2364 }
2365 }
2366
2367 static void
tic4x_insn_output(tic4x_insn_t * tinsn)2368 tic4x_insn_output (tic4x_insn_t *tinsn)
2369 {
2370 char *dst;
2371
2372 /* Grab another fragment for opcode. */
2373 dst = frag_more (tinsn->nchars);
2374
2375 /* Put out opcode word as a series of bytes in little endian order. */
2376 md_number_to_chars (dst, tinsn->opcode, tinsn->nchars);
2377
2378 /* Put out the symbol-dependent stuff. */
2379 if (tinsn->reloc != NO_RELOC)
2380 {
2381 /* Where is the offset into the fragment for this instruction. */
2382 fix_new_exp (frag_now,
2383 dst - frag_now->fr_literal, /* where */
2384 tinsn->nchars, /* size */
2385 &tinsn->exp,
2386 tinsn->pcrel,
2387 tinsn->reloc);
2388 }
2389 }
2390
2391 /* Parse the operands. */
2392 static int
tic4x_operands_parse(char * s,tic4x_operand_t * operands,int num_operands)2393 tic4x_operands_parse (char *s, tic4x_operand_t *operands, int num_operands)
2394 {
2395 if (!*s)
2396 return num_operands;
2397
2398 do
2399 s = tic4x_operand_parse (s, &operands[num_operands++]);
2400 while (num_operands < TIC4X_OPERANDS_MAX && *s++ == ',');
2401
2402 if (num_operands > TIC4X_OPERANDS_MAX)
2403 {
2404 as_bad (_("Too many operands scanned"));
2405 return -1;
2406 }
2407 return num_operands;
2408 }
2409
2410 /* Assemble a single instruction. Its label has already been handled
2411 by the generic front end. We just parse mnemonic and operands, and
2412 produce the bytes of data and relocation. */
2413 void
md_assemble(char * str)2414 md_assemble (char *str)
2415 {
2416 int ok = 0;
2417 char *s;
2418 int i;
2419 int parsed = 0;
2420 tic4x_inst_t *inst; /* Instruction template. */
2421 tic4x_inst_t *first_inst;
2422
2423 /* Scan for parallel operators */
2424 if (str)
2425 {
2426 s = str;
2427 while (*s && *s != '|')
2428 s++;
2429
2430 if (*s && s[1]=='|')
2431 {
2432 if(insn->parallel)
2433 {
2434 as_bad (_("Parallel opcode cannot contain more than two instructions"));
2435 insn->parallel = 0;
2436 insn->in_use = 0;
2437 return;
2438 }
2439
2440 /* Lets take care of the first part of the parallel insn */
2441 *s++ = 0;
2442 md_assemble(str);
2443 insn->parallel = 1;
2444 str = ++s;
2445 /* .. and let the second run though here */
2446 }
2447 }
2448
2449 if (str && insn->parallel)
2450 {
2451 /* Find mnemonic (second part of parallel instruction). */
2452 s = str;
2453 /* Skip past instruction mnemonic. */
2454 while (*s && *s != ' ')
2455 s++;
2456 if (*s) /* Null terminate for hash_find. */
2457 *s++ = '\0'; /* and skip past null. */
2458 strcat (insn->name, "_");
2459 strncat (insn->name, str, TIC4X_NAME_MAX - 1 - strlen (insn->name));
2460
2461 insn->operands[insn->num_operands++].mode = M_PARALLEL;
2462
2463 if ((i = tic4x_operands_parse
2464 (s, insn->operands, insn->num_operands)) < 0)
2465 {
2466 insn->parallel = 0;
2467 insn->in_use = 0;
2468 return;
2469 }
2470 insn->num_operands = i;
2471 parsed = 1;
2472 }
2473
2474 if (insn->in_use)
2475 {
2476 if ((insn->inst = (struct tic4x_inst *)
2477 hash_find (tic4x_op_hash, insn->name)) == NULL)
2478 {
2479 as_bad (_("Unknown opcode `%s'."), insn->name);
2480 insn->parallel = 0;
2481 insn->in_use = 0;
2482 return;
2483 }
2484
2485 inst = insn->inst;
2486 first_inst = NULL;
2487 do
2488 {
2489 ok = tic4x_operands_match (inst, insn, 1);
2490 if (ok < 0)
2491 {
2492 if (!first_inst)
2493 first_inst = inst;
2494 ok = 0;
2495 }
2496 } while (!ok && !strcmp (inst->name, inst[1].name) && inst++);
2497
2498 if (ok > 0)
2499 {
2500 tic4x_insn_check (insn);
2501 tic4x_insn_output (insn);
2502 }
2503 else if (!ok)
2504 {
2505 if (first_inst)
2506 tic4x_operands_match (first_inst, insn, 0);
2507 as_bad (_("Invalid operands for %s"), insn->name);
2508 }
2509 else
2510 as_bad (_("Invalid instruction %s"), insn->name);
2511 }
2512
2513 if (str && !parsed)
2514 {
2515 /* Find mnemonic. */
2516 s = str;
2517 while (*s && *s != ' ') /* Skip past instruction mnemonic. */
2518 s++;
2519 if (*s) /* Null terminate for hash_find. */
2520 *s++ = '\0'; /* and skip past null. */
2521 strncpy (insn->name, str, TIC4X_NAME_MAX - 3);
2522
2523 if ((i = tic4x_operands_parse (s, insn->operands, 0)) < 0)
2524 {
2525 insn->inst = NULL; /* Flag that error occurred. */
2526 insn->parallel = 0;
2527 insn->in_use = 0;
2528 return;
2529 }
2530 insn->num_operands = i;
2531 insn->in_use = 1;
2532 }
2533 else
2534 insn->in_use = 0;
2535 insn->parallel = 0;
2536 }
2537
2538 void
tic4x_cleanup(void)2539 tic4x_cleanup (void)
2540 {
2541 if (insn->in_use)
2542 md_assemble (NULL);
2543 }
2544
2545 /* Turn a string in input_line_pointer into a floating point constant
2546 of type type, and store the appropriate bytes in *litP. The number
2547 of chars emitted is stored in *sizeP. An error message is
2548 returned, or NULL on OK. */
2549
2550 char *
md_atof(int type,char * litP,int * sizeP)2551 md_atof (int type, char *litP, int *sizeP)
2552 {
2553 int prec;
2554 int ieee;
2555 LITTLENUM_TYPE words[MAX_LITTLENUMS];
2556 LITTLENUM_TYPE *wordP;
2557 char *t;
2558
2559 switch (type)
2560 {
2561 case 's': /* .single */
2562 case 'S':
2563 ieee = 0;
2564 prec = 1;
2565 break;
2566
2567 case 'd': /* .double */
2568 case 'D':
2569 case 'f': /* .float */
2570 case 'F':
2571 ieee = 0;
2572 prec = 2; /* 1 32-bit word */
2573 break;
2574
2575 case 'i': /* .ieee */
2576 case 'I':
2577 prec = 2;
2578 ieee = 1;
2579 type = 'f'; /* Rewrite type to be usable by atof_ieee(). */
2580 break;
2581
2582 case 'e': /* .ldouble */
2583 case 'E':
2584 prec = 4; /* 2 32-bit words */
2585 ieee = 0;
2586 break;
2587
2588 default:
2589 *sizeP = 0;
2590 return _("Unrecognized or unsupported floating point constant");
2591 }
2592
2593 if (ieee)
2594 t = atof_ieee (input_line_pointer, type, words);
2595 else
2596 t = tic4x_atof (input_line_pointer, type, words);
2597 if (t)
2598 input_line_pointer = t;
2599 *sizeP = prec * sizeof (LITTLENUM_TYPE);
2600
2601 /* This loops outputs the LITTLENUMs in REVERSE order; in accord with
2602 little endian byte order. */
2603 /* SES: However it is required to put the words (32-bits) out in the
2604 correct order, hence we write 2 and 2 littlenums in little endian
2605 order, while we keep the original order on successive words. */
2606 for (wordP = words; wordP<(words+prec) ; wordP+=2)
2607 {
2608 if (wordP < (words + prec - 1)) /* Dump wordP[1] (if we have one). */
2609 {
2610 md_number_to_chars (litP, (valueT) (wordP[1]),
2611 sizeof (LITTLENUM_TYPE));
2612 litP += sizeof (LITTLENUM_TYPE);
2613 }
2614
2615 /* Dump wordP[0] */
2616 md_number_to_chars (litP, (valueT) (wordP[0]),
2617 sizeof (LITTLENUM_TYPE));
2618 litP += sizeof (LITTLENUM_TYPE);
2619 }
2620 return NULL;
2621 }
2622
2623 void
md_apply_fix(fixS * fixP,valueT * value,segT seg ATTRIBUTE_UNUSED)2624 md_apply_fix (fixS *fixP, valueT *value, segT seg ATTRIBUTE_UNUSED)
2625 {
2626 char *buf = fixP->fx_where + fixP->fx_frag->fr_literal;
2627 valueT val = *value;
2628
2629 switch (fixP->fx_r_type)
2630 {
2631 case BFD_RELOC_HI16:
2632 val >>= 16;
2633 break;
2634
2635 case BFD_RELOC_LO16:
2636 val &= 0xffff;
2637 break;
2638 default:
2639 break;
2640 }
2641
2642 switch (fixP->fx_r_type)
2643 {
2644 case BFD_RELOC_32:
2645 buf[3] = val >> 24;
2646 case BFD_RELOC_24:
2647 case BFD_RELOC_24_PCREL:
2648 buf[2] = val >> 16;
2649 case BFD_RELOC_16:
2650 case BFD_RELOC_16_PCREL:
2651 case BFD_RELOC_LO16:
2652 case BFD_RELOC_HI16:
2653 buf[1] = val >> 8;
2654 buf[0] = val;
2655 break;
2656
2657 case NO_RELOC:
2658 default:
2659 as_bad (_("Bad relocation type: 0x%02x"), fixP->fx_r_type);
2660 break;
2661 }
2662
2663 if (fixP->fx_addsy == NULL && fixP->fx_pcrel == 0) fixP->fx_done = 1;
2664 }
2665
2666 /* Should never be called for tic4x. */
2667 void
md_convert_frag(bfd * headers ATTRIBUTE_UNUSED,segT sec ATTRIBUTE_UNUSED,fragS * fragP ATTRIBUTE_UNUSED)2668 md_convert_frag (bfd *headers ATTRIBUTE_UNUSED,
2669 segT sec ATTRIBUTE_UNUSED,
2670 fragS *fragP ATTRIBUTE_UNUSED)
2671 {
2672 as_fatal ("md_convert_frag");
2673 }
2674
2675 /* Should never be called for tic4x. */
2676 void
md_create_short_jump(char * ptr ATTRIBUTE_UNUSED,addressT from_addr ATTRIBUTE_UNUSED,addressT to_addr ATTRIBUTE_UNUSED,fragS * frag ATTRIBUTE_UNUSED,symbolS * to_symbol ATTRIBUTE_UNUSED)2677 md_create_short_jump (char *ptr ATTRIBUTE_UNUSED,
2678 addressT from_addr ATTRIBUTE_UNUSED,
2679 addressT to_addr ATTRIBUTE_UNUSED,
2680 fragS *frag ATTRIBUTE_UNUSED,
2681 symbolS *to_symbol ATTRIBUTE_UNUSED)
2682 {
2683 as_fatal ("md_create_short_jmp\n");
2684 }
2685
2686 /* Should never be called for tic4x. */
2687 void
md_create_long_jump(char * ptr ATTRIBUTE_UNUSED,addressT from_addr ATTRIBUTE_UNUSED,addressT to_addr ATTRIBUTE_UNUSED,fragS * frag ATTRIBUTE_UNUSED,symbolS * to_symbol ATTRIBUTE_UNUSED)2688 md_create_long_jump (char *ptr ATTRIBUTE_UNUSED,
2689 addressT from_addr ATTRIBUTE_UNUSED,
2690 addressT to_addr ATTRIBUTE_UNUSED,
2691 fragS *frag ATTRIBUTE_UNUSED,
2692 symbolS *to_symbol ATTRIBUTE_UNUSED)
2693 {
2694 as_fatal ("md_create_long_jump\n");
2695 }
2696
2697 /* Should never be called for tic4x. */
2698 int
md_estimate_size_before_relax(fragS * fragP ATTRIBUTE_UNUSED,segT segtype ATTRIBUTE_UNUSED)2699 md_estimate_size_before_relax (fragS *fragP ATTRIBUTE_UNUSED,
2700 segT segtype ATTRIBUTE_UNUSED)
2701 {
2702 as_fatal ("md_estimate_size_before_relax\n");
2703 return 0;
2704 }
2705
2706
2707 int
md_parse_option(int c,char * arg)2708 md_parse_option (int c, char *arg)
2709 {
2710 switch (c)
2711 {
2712 case OPTION_CPU: /* cpu brand */
2713 if (TOLOWER (*arg) == 'c')
2714 arg++;
2715 tic4x_cpu = atoi (arg);
2716 if (!IS_CPU_TIC3X (tic4x_cpu) && !IS_CPU_TIC4X (tic4x_cpu))
2717 as_warn (_("Unsupported processor generation %d"), tic4x_cpu);
2718 break;
2719
2720 case OPTION_REV: /* cpu revision */
2721 tic4x_revision = atoi (arg);
2722 break;
2723
2724 case 'b':
2725 as_warn (_("Option -b is depreciated, please use -mbig"));
2726 case OPTION_BIG: /* big model */
2727 tic4x_big_model = 1;
2728 break;
2729
2730 case 'p':
2731 as_warn (_("Option -p is depreciated, please use -mmemparm"));
2732 case OPTION_MEMPARM: /* push args */
2733 tic4x_reg_args = 0;
2734 break;
2735
2736 case 'r':
2737 as_warn (_("Option -r is depreciated, please use -mregparm"));
2738 case OPTION_REGPARM: /* register args */
2739 tic4x_reg_args = 1;
2740 break;
2741
2742 case 's':
2743 as_warn (_("Option -s is depreciated, please use -msmall"));
2744 case OPTION_SMALL: /* small model */
2745 tic4x_big_model = 0;
2746 break;
2747
2748 case OPTION_IDLE2:
2749 tic4x_idle2 = 1;
2750 break;
2751
2752 case OPTION_LOWPOWER:
2753 tic4x_lowpower = 1;
2754 break;
2755
2756 case OPTION_ENHANCED:
2757 tic4x_enhanced = 1;
2758 break;
2759
2760 default:
2761 return 0;
2762 }
2763
2764 return 1;
2765 }
2766
2767 void
md_show_usage(FILE * stream)2768 md_show_usage (FILE *stream)
2769 {
2770 fprintf (stream,
2771 _("\nTIC4X options:\n"
2772 " -mcpu=CPU -mCPU select architecture variant. CPU can be:\n"
2773 " 30 - TMS320C30\n"
2774 " 31 - TMS320C31, TMS320LC31\n"
2775 " 32 - TMS320C32\n"
2776 " 33 - TMS320VC33\n"
2777 " 40 - TMS320C40\n"
2778 " 44 - TMS320C44\n"
2779 " -mrev=REV set cpu hardware revision (integer numbers).\n"
2780 " Combinations of -mcpu and -mrev will enable/disable\n"
2781 " the appropriate options (-midle2, -mlowpower and\n"
2782 " -menhanced) according to the selected type\n"
2783 " -mbig select big memory model\n"
2784 " -msmall select small memory model (default)\n"
2785 " -mregparm select register parameters (default)\n"
2786 " -mmemparm select memory parameters\n"
2787 " -midle2 enable IDLE2 support\n"
2788 " -mlowpower enable LOPOWER and MAXSPEED support\n"
2789 " -menhanced enable enhanced opcode support\n"));
2790 }
2791
2792 /* This is called when a line is unrecognized. This is used to handle
2793 definitions of TI C3x tools style local labels $n where n is a single
2794 decimal digit. */
2795 int
tic4x_unrecognized_line(int c)2796 tic4x_unrecognized_line (int c)
2797 {
2798 int lab;
2799 char *s;
2800
2801 if (c != '$' || ! ISDIGIT (input_line_pointer[0]))
2802 return 0;
2803
2804 s = input_line_pointer;
2805
2806 /* Let's allow multiple digit local labels. */
2807 lab = 0;
2808 while (ISDIGIT (*s))
2809 {
2810 lab = lab * 10 + *s - '0';
2811 s++;
2812 }
2813
2814 if (dollar_label_defined (lab))
2815 {
2816 as_bad (_("Label \"$%d\" redefined"), lab);
2817 return 0;
2818 }
2819
2820 define_dollar_label (lab);
2821 colon (dollar_label_name (lab, 0));
2822 input_line_pointer = s + 1;
2823
2824 return 1;
2825 }
2826
2827 /* Handle local labels peculiar to us referred to in an expression. */
2828 symbolS *
md_undefined_symbol(char * name)2829 md_undefined_symbol (char *name)
2830 {
2831 /* Look for local labels of the form $n. */
2832 if (name[0] == '$' && ISDIGIT (name[1]))
2833 {
2834 symbolS *symbolP;
2835 char *s = name + 1;
2836 int lab = 0;
2837
2838 while (ISDIGIT ((unsigned char) *s))
2839 {
2840 lab = lab * 10 + *s - '0';
2841 s++;
2842 }
2843 if (dollar_label_defined (lab))
2844 {
2845 name = dollar_label_name (lab, 0);
2846 symbolP = symbol_find (name);
2847 }
2848 else
2849 {
2850 name = dollar_label_name (lab, 1);
2851 symbolP = symbol_find_or_make (name);
2852 }
2853
2854 return symbolP;
2855 }
2856 return NULL;
2857 }
2858
2859 /* Parse an operand that is machine-specific. */
2860 void
md_operand(expressionS * expressionP ATTRIBUTE_UNUSED)2861 md_operand (expressionS *expressionP ATTRIBUTE_UNUSED)
2862 {
2863 }
2864
2865 /* Round up a section size to the appropriate boundary---do we need this? */
2866 valueT
md_section_align(segT segment ATTRIBUTE_UNUSED,valueT size)2867 md_section_align (segT segment ATTRIBUTE_UNUSED, valueT size)
2868 {
2869 return size; /* Byte (i.e., 32-bit) alignment is fine? */
2870 }
2871
2872 static int
tic4x_pc_offset(unsigned int op)2873 tic4x_pc_offset (unsigned int op)
2874 {
2875 /* Determine the PC offset for a C[34]x instruction.
2876 This could be simplified using some boolean algebra
2877 but at the expense of readability. */
2878 switch (op >> 24)
2879 {
2880 case 0x60: /* br */
2881 case 0x62: /* call (C4x) */
2882 case 0x64: /* rptb (C4x) */
2883 return 1;
2884 case 0x61: /* brd */
2885 case 0x63: /* laj */
2886 case 0x65: /* rptbd (C4x) */
2887 return 3;
2888 case 0x66: /* swi */
2889 case 0x67:
2890 return 0;
2891 default:
2892 break;
2893 }
2894
2895 switch ((op & 0xffe00000) >> 20)
2896 {
2897 case 0x6a0: /* bB */
2898 case 0x720: /* callB */
2899 case 0x740: /* trapB */
2900 return 1;
2901
2902 case 0x6a2: /* bBd */
2903 case 0x6a6: /* bBat */
2904 case 0x6aa: /* bBaf */
2905 case 0x722: /* lajB */
2906 case 0x748: /* latB */
2907 case 0x798: /* rptbd */
2908 return 3;
2909
2910 default:
2911 break;
2912 }
2913
2914 switch ((op & 0xfe200000) >> 20)
2915 {
2916 case 0x6e0: /* dbB */
2917 return 1;
2918
2919 case 0x6e2: /* dbBd */
2920 return 3;
2921
2922 default:
2923 break;
2924 }
2925
2926 return 0;
2927 }
2928
2929 /* Exactly what point is a PC-relative offset relative TO?
2930 With the C3x we have the following:
2931 DBcond, Bcond disp + PC + 1 => PC
2932 DBcondD, BcondD disp + PC + 3 => PC
2933 */
2934 long
md_pcrel_from(fixS * fixP)2935 md_pcrel_from (fixS *fixP)
2936 {
2937 unsigned char *buf;
2938 unsigned int op;
2939
2940 buf = (unsigned char *) fixP->fx_frag->fr_literal + fixP->fx_where;
2941 op = (buf[3] << 24) | (buf[2] << 16) | (buf[1] << 8) | buf[0];
2942
2943 return ((fixP->fx_where + fixP->fx_frag->fr_address) >> 2) +
2944 tic4x_pc_offset (op);
2945 }
2946
2947 /* Fill the alignment area with NOP's on .text, unless fill-data
2948 was specified. */
2949 int
tic4x_do_align(int alignment,const char * fill,int len,int max)2950 tic4x_do_align (int alignment,
2951 const char *fill,
2952 int len,
2953 int max)
2954 {
2955 /* Because we are talking lwords, not bytes, adjust alignment to do words */
2956 alignment += 2;
2957
2958 if (alignment != 0 && !need_pass_2)
2959 {
2960 if (fill == NULL)
2961 {
2962 if (subseg_text_p (now_seg))
2963 {
2964 char nop[4];
2965
2966 md_number_to_chars (nop, TIC_NOP_OPCODE, 4);
2967 frag_align_pattern (alignment, nop, sizeof (nop), max);
2968 }
2969 else
2970 frag_align (alignment, 0, max);
2971 }
2972 else if (len <= 1)
2973 frag_align (alignment, *fill, max);
2974 else
2975 frag_align_pattern (alignment, fill, len, max);
2976 }
2977
2978 /* Return 1 to skip the default alignment function */
2979 return 1;
2980 }
2981
2982 /* Look for and remove parallel instruction operator ||. */
2983 void
tic4x_start_line(void)2984 tic4x_start_line (void)
2985 {
2986 char *s = input_line_pointer;
2987
2988 SKIP_WHITESPACE ();
2989
2990 /* If parallel instruction prefix found at start of line, skip it. */
2991 if (*input_line_pointer == '|' && input_line_pointer[1] == '|')
2992 {
2993 if (insn->in_use)
2994 {
2995 insn->parallel = 1;
2996 input_line_pointer ++;
2997 *input_line_pointer = ' ';
2998 /* So line counters get bumped. */
2999 input_line_pointer[-1] = '\n';
3000 }
3001 }
3002 else
3003 {
3004 /* Write out the previous insn here */
3005 if (insn->in_use)
3006 md_assemble (NULL);
3007 input_line_pointer = s;
3008 }
3009 }
3010
3011 arelent *
tc_gen_reloc(asection * seg ATTRIBUTE_UNUSED,fixS * fixP)3012 tc_gen_reloc (asection *seg ATTRIBUTE_UNUSED, fixS *fixP)
3013 {
3014 arelent *reloc;
3015
3016 reloc = (arelent *) xmalloc (sizeof (arelent));
3017
3018 reloc->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
3019 *reloc->sym_ptr_ptr = symbol_get_bfdsym (fixP->fx_addsy);
3020 reloc->address = fixP->fx_frag->fr_address + fixP->fx_where;
3021 reloc->address /= OCTETS_PER_BYTE;
3022 reloc->howto = bfd_reloc_type_lookup (stdoutput, fixP->fx_r_type);
3023 if (reloc->howto == (reloc_howto_type *) NULL)
3024 {
3025 as_bad_where (fixP->fx_file, fixP->fx_line,
3026 _("Reloc %d not supported by object file format"),
3027 (int) fixP->fx_r_type);
3028 return NULL;
3029 }
3030
3031 if (fixP->fx_r_type == BFD_RELOC_HI16)
3032 reloc->addend = fixP->fx_offset;
3033 else
3034 reloc->addend = fixP->fx_addnumber;
3035
3036 return reloc;
3037 }
3038