00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025 #include "config.h"
00026 #include "system.h"
00027 #include <math.h>
00028 #include "rtl.h"
00029 #include "regs.h"
00030 #include "hard-reg-set.h"
00031 #include "real.h"
00032 #include "insn-config.h"
00033 #include "conditions.h"
00034 #include "output.h"
00035 #include "insn-attr.h"
00036 #include "flags.h"
00037 #include "tree.h"
00038 #include "expr.h"
00039 #include "except.h"
00040 #include "function.h"
00041 #include "recog.h"
00042 #include "toplev.h"
00043 #include "cpplib.h"
00044 #include "c-pragma.h"
00045 #include "c-lex.h"
00046 #include "tm_p.h"
00047 #include "target.h"
00048 #include "target-def.h"
00049
00050 static void i960_output_function_prologue PARAMS ((FILE *, HOST_WIDE_INT));
00051 static void i960_output_function_epilogue PARAMS ((FILE *, HOST_WIDE_INT));
00052
00053
00054
00055
00056 rtx i960_compare_op0, i960_compare_op1;
00057
00058
00059
00060
00061 int i960_maxbitalignment;
00062 int i960_last_maxbitalignment;
00063
00064
00065
00066
00067 enum insn_types i960_last_insn_type;
00068
00069
00070
00071 static int i960_leaf_ret_reg;
00072
00073
00074
00075 static int tail_call_ok;
00076
00077
00078
00079
00080
00081 char epilogue_string[1000];
00082
00083
00084
00085 static int ret_label = 0;
00086
00087
00088
00089
00090 #define VARARGS_STDARG_FUNCTION(FNDECL) \
00091 ((TYPE_ARG_TYPES (TREE_TYPE (FNDECL)) != 0 \
00092 && (TREE_VALUE (tree_last (TYPE_ARG_TYPES (TREE_TYPE (FNDECL)))) != void_type_node)) \
00093 || current_function_varargs)
00094
00095
00096 #undef TARGET_ASM_ALIGNED_SI_OP
00097 #define TARGET_ASM_ALIGNED_SI_OP "\t.word\t"
00098
00099 #undef TARGET_ASM_FUNCTION_PROLOGUE
00100 #define TARGET_ASM_FUNCTION_PROLOGUE i960_output_function_prologue
00101 #undef TARGET_ASM_FUNCTION_EPILOGUE
00102 #define TARGET_ASM_FUNCTION_EPILOGUE i960_output_function_epilogue
00103
00104 struct gcc_target targetm = TARGET_INITIALIZER;
00105
00106
00107
00108 void
00109 i960_initialize ()
00110 {
00111 if (TARGET_IC_COMPAT2_0)
00112 {
00113 i960_maxbitalignment = 8;
00114 i960_last_maxbitalignment = 128;
00115 }
00116 else
00117 {
00118 i960_maxbitalignment = 128;
00119 i960_last_maxbitalignment = 8;
00120 }
00121 }
00122
00123
00124
00125 int
00126 fpmove_src_operand (op, mode)
00127 rtx op;
00128 enum machine_mode mode;
00129 {
00130 return (GET_CODE (op) == CONST_DOUBLE || general_operand (op, mode));
00131 }
00132
00133 #if 0
00134
00135
00136 int
00137 reg_or_zero_operand (op, mode)
00138 rtx op;
00139 enum machine_mode mode;
00140 {
00141 return register_operand (op, mode) || op == const0_rtx;
00142 }
00143 #endif
00144
00145
00146
00147
00148 int
00149 arith_operand (op, mode)
00150 rtx op;
00151 enum machine_mode mode;
00152 {
00153 return (register_operand (op, mode) || literal (op, mode));
00154 }
00155
00156
00157
00158
00159 int
00160 logic_operand (op, mode)
00161 rtx op;
00162 enum machine_mode mode;
00163 {
00164 return (register_operand (op, mode)
00165 || (GET_CODE (op) == CONST_INT
00166 && INTVAL(op) >= -32 && INTVAL(op) < 32));
00167 }
00168
00169
00170
00171 int
00172 fp_arith_operand (op, mode)
00173 rtx op;
00174 enum machine_mode mode;
00175 {
00176 return (register_operand (op, mode) || fp_literal (op, mode));
00177 }
00178
00179
00180
00181 int
00182 signed_arith_operand (op, mode)
00183 rtx op;
00184 enum machine_mode mode;
00185 {
00186 return (register_operand (op, mode) || signed_literal (op, mode));
00187 }
00188
00189
00190
00191
00192 int
00193 literal (op, mode)
00194 rtx op;
00195 enum machine_mode mode ATTRIBUTE_UNUSED;
00196 {
00197 return ((GET_CODE (op) == CONST_INT) && INTVAL(op) >= 0 && INTVAL(op) < 32);
00198 }
00199
00200
00201
00202 int
00203 fp_literal_one (op, mode)
00204 rtx op;
00205 enum machine_mode mode;
00206 {
00207 return (TARGET_NUMERICS && mode == GET_MODE (op) && op == CONST1_RTX (mode));
00208 }
00209
00210
00211
00212 int
00213 fp_literal_zero (op, mode)
00214 rtx op;
00215 enum machine_mode mode;
00216 {
00217 return (TARGET_NUMERICS && mode == GET_MODE (op) && op == CONST0_RTX (mode));
00218 }
00219
00220
00221
00222 int
00223 fp_literal(op, mode)
00224 rtx op;
00225 enum machine_mode mode;
00226 {
00227 return fp_literal_zero (op, mode) || fp_literal_one (op, mode);
00228 }
00229
00230
00231
00232 int
00233 signed_literal(op, mode)
00234 rtx op;
00235 enum machine_mode mode ATTRIBUTE_UNUSED;
00236 {
00237 return ((GET_CODE (op) == CONST_INT) && INTVAL(op) > -32 && INTVAL(op) < 32);
00238 }
00239
00240
00241
00242
00243 int
00244 symbolic_memory_operand (op, mode)
00245 rtx op;
00246 enum machine_mode mode ATTRIBUTE_UNUSED;
00247 {
00248 if (GET_CODE (op) == SUBREG)
00249 op = SUBREG_REG (op);
00250 if (GET_CODE (op) != MEM)
00251 return 0;
00252 op = XEXP (op, 0);
00253 return (GET_CODE (op) == SYMBOL_REF || GET_CODE (op) == CONST
00254 || GET_CODE (op) == HIGH || GET_CODE (op) == LABEL_REF);
00255 }
00256
00257
00258
00259 int
00260 eq_or_neq (op, mode)
00261 rtx op;
00262 enum machine_mode mode ATTRIBUTE_UNUSED;
00263 {
00264 return (GET_CODE (op) == EQ || GET_CODE (op) == NE);
00265 }
00266
00267
00268
00269 int
00270 arith32_operand (op, mode)
00271 rtx op;
00272 enum machine_mode mode;
00273 {
00274 if (register_operand (op, mode))
00275 return 1;
00276 return (CONSTANT_P (op));
00277 }
00278
00279
00280
00281 int
00282 power2_operand (op,mode)
00283 rtx op;
00284 enum machine_mode mode ATTRIBUTE_UNUSED;
00285 {
00286 if (GET_CODE (op) != CONST_INT)
00287 return 0;
00288
00289 return exact_log2 (INTVAL (op)) >= 0;
00290 }
00291
00292
00293
00294
00295 int
00296 cmplpower2_operand (op, mode)
00297 rtx op;
00298 enum machine_mode mode ATTRIBUTE_UNUSED;
00299 {
00300 if (GET_CODE (op) != CONST_INT)
00301 return 0;
00302
00303 return exact_log2 (~ INTVAL (op)) >= 0;
00304 }
00305
00306
00307
00308
00309 int
00310 bitpos (val)
00311 unsigned int val;
00312 {
00313 register int i;
00314
00315 for (i = 0; val != 0; i++, val >>= 1)
00316 {
00317 if (val & 1)
00318 {
00319 if (val != 1)
00320 return -1;
00321 return i;
00322 }
00323 }
00324 return -1;
00325 }
00326
00327
00328
00329
00330
00331
00332 int
00333 is_mask (val)
00334 unsigned int val;
00335 {
00336 register int start, end = 0, i;
00337
00338 start = -1;
00339 for (i = 0; val != 0; val >>= 1, i++)
00340 {
00341 if (val & 1)
00342 {
00343 if (start < 0)
00344 start = i;
00345
00346 end = i;
00347 continue;
00348 }
00349
00350 if (start < 0)
00351 continue;
00352
00353
00354
00355
00356 if (val)
00357 return 0;
00358 }
00359
00360
00361 return end - start + 1;
00362 }
00363
00364
00365
00366
00367
00368
00369
00370 int
00371 bitstr (val, s, e)
00372 unsigned int val;
00373 int *s, *e;
00374 {
00375 register int start, end, i;
00376
00377 start = -1;
00378 end = -1;
00379 for (i = 0; val != 0; val >>= 1, i++)
00380 {
00381 if (val & 1)
00382 {
00383 if (start < 0)
00384 start = i;
00385
00386 end = i;
00387 continue;
00388 }
00389
00390
00391 if (start < 0)
00392 continue;
00393
00394
00395
00396
00397 if (val)
00398 {
00399 start = -1;
00400 end = -1;
00401 break;
00402 }
00403 }
00404
00405
00406 *s = start;
00407 *e = end;
00408 return ((start < 0) ? 0 : end - start + 1);
00409 }
00410
00411
00412
00413 enum machine_mode
00414 select_cc_mode (op, x)
00415 RTX_CODE op;
00416 rtx x ATTRIBUTE_UNUSED;
00417 {
00418 if (op == GTU || op == LTU || op == GEU || op == LEU)
00419 return CC_UNSmode;
00420 return CCmode;
00421 }
00422
00423
00424
00425
00426 rtx
00427 gen_compare_reg (code, x, y)
00428 enum rtx_code code;
00429 rtx x, y;
00430 {
00431 rtx cc_reg;
00432 enum machine_mode ccmode = SELECT_CC_MODE (code, x, y);
00433 enum machine_mode mode
00434 = GET_MODE (x) == VOIDmode ? GET_MODE (y) : GET_MODE (x);
00435
00436 if (mode == SImode)
00437 {
00438 if (! arith_operand (x, mode))
00439 x = force_reg (SImode, x);
00440 if (! arith_operand (y, mode))
00441 y = force_reg (SImode, y);
00442 }
00443
00444 cc_reg = gen_rtx_REG (ccmode, 36);
00445 emit_insn (gen_rtx_SET (VOIDmode, cc_reg,
00446 gen_rtx_COMPARE (ccmode, x, y)));
00447
00448 return cc_reg;
00449 }
00450
00451
00452
00453
00454
00455
00456
00457 int
00458 i960_address_cost (x)
00459 rtx x;
00460 {
00461 #if 0
00462
00463 if (GET_CODE (x) == REG)
00464 return 1;
00465 #endif
00466
00467 if (GET_CODE (x) == CONST_INT
00468 && INTVAL (x) >= 0
00469 && INTVAL (x) < 4096)
00470 return 0;
00471
00472 if (GET_CODE (x) == PLUS)
00473 {
00474 rtx base = XEXP (x, 0);
00475 rtx offset = XEXP (x, 1);
00476
00477 if (GET_CODE (base) == SUBREG)
00478 base = SUBREG_REG (base);
00479 if (GET_CODE (offset) == SUBREG)
00480 offset = SUBREG_REG (offset);
00481
00482 if (GET_CODE (base) == REG)
00483 {
00484 if (GET_CODE (offset) == REG)
00485 return 2;
00486 if (GET_CODE (offset) == CONST_INT)
00487 {
00488 if ((unsigned)INTVAL (offset) < 2047)
00489 return 2;
00490 return 4;
00491 }
00492 if (CONSTANT_P (offset))
00493 return 4;
00494 }
00495 if (GET_CODE (base) == PLUS || GET_CODE (base) == MULT)
00496 return 6;
00497
00498
00499
00500 return 12;
00501 }
00502 if (GET_CODE (x) == MULT)
00503 return 6;
00504
00505
00506 return 4;
00507 }
00508
00509
00510
00511
00512
00513
00514
00515 int
00516 emit_move_sequence (operands, mode)
00517 rtx *operands;
00518 enum machine_mode mode;
00519 {
00520
00521
00522 if (GET_CODE (operands[0]) == MEM && GET_CODE (operands[1]) != REG
00523 && (operands[1] != const0_rtx || current_function_args_size
00524 || current_function_varargs || current_function_stdarg
00525 || rtx_equal_function_value_matters))
00526
00527 operands[1] = force_reg (mode, operands[1]);
00528
00529
00530
00531
00532
00533
00534
00535
00536
00537 if (GET_MODE_SIZE (mode) > UNITS_PER_WORD
00538 && (GET_CODE (operands[0]) == MEM
00539 || (GET_CODE (operands[0]) == REG
00540 && REGNO (operands[0]) >= FIRST_PSEUDO_REGISTER))
00541 && GET_CODE (operands[1]) == REG
00542 && REGNO (operands[1]) < FIRST_PSEUDO_REGISTER
00543 && ! HARD_REGNO_MODE_OK (REGNO (operands[1]), mode))
00544 {
00545 emit_insn (gen_rtx_PARALLEL
00546 (VOIDmode,
00547 gen_rtvec (2,
00548 gen_rtx_SET (VOIDmode, operands[0], operands[1]),
00549 gen_rtx_CLOBBER (VOIDmode,
00550 gen_rtx_SCRATCH (Pmode)))));
00551 return 1;
00552 }
00553
00554 return 0;
00555 }
00556
00557
00558
00559 const char *
00560 i960_output_move_double (dst, src)
00561 rtx dst, src;
00562 {
00563 rtx operands[5];
00564
00565 if (GET_CODE (dst) == REG
00566 && GET_CODE (src) == REG)
00567 {
00568 if ((REGNO (src) & 1)
00569 || (REGNO (dst) & 1))
00570 {
00571
00572
00573
00574 if (REGNO (src) + 1 == REGNO (dst))
00575 return "mov %D1,%D0\n\tmov %1,%0";
00576 else
00577 return "mov %1,%0\n\tmov %D1,%D0";
00578 }
00579 else
00580 return "movl %1,%0";
00581 }
00582 else if (GET_CODE (dst) == REG
00583 && GET_CODE (src) == CONST_INT
00584 && CONST_OK_FOR_LETTER_P (INTVAL (src), 'I'))
00585 {
00586 if (REGNO (dst) & 1)
00587 return "mov %1,%0\n\tmov 0,%D0";
00588 else
00589 return "movl %1,%0";
00590 }
00591 else if (GET_CODE (dst) == REG
00592 && GET_CODE (src) == MEM)
00593 {
00594 if (REGNO (dst) & 1)
00595 {
00596
00597
00598
00599 operands[0] = dst;
00600 operands[1] = src;
00601 operands[2] = gen_rtx_REG (Pmode, REGNO (dst) + 1);
00602 operands[3] = gen_rtx_MEM (word_mode, operands[2]);
00603 operands[4] = adjust_address (operands[3], word_mode,
00604 UNITS_PER_WORD);
00605 output_asm_insn
00606 ("lda %1,%2\n\tld %3,%0\n\tld %4,%D0", operands);
00607 return "";
00608 }
00609 else
00610 return "ldl %1,%0";
00611 }
00612 else if (GET_CODE (dst) == MEM
00613 && GET_CODE (src) == REG)
00614 {
00615 if (REGNO (src) & 1)
00616 {
00617 operands[0] = dst;
00618 operands[1] = adjust_address (dst, word_mode, UNITS_PER_WORD);
00619 if (! memory_address_p (word_mode, XEXP (operands[1], 0)))
00620 abort ();
00621 operands[2] = src;
00622 output_asm_insn ("st %2,%0\n\tst %D2,%1", operands);
00623 return "";
00624 }
00625 return "stl %1,%0";
00626 }
00627 else
00628 abort ();
00629 }
00630
00631
00632
00633 const char *
00634 i960_output_move_double_zero (dst)
00635 rtx dst;
00636 {
00637 rtx operands[2];
00638
00639 operands[0] = dst;
00640 {
00641 operands[1] = adjust_address (dst, word_mode, 4);
00642 output_asm_insn ("st g14,%0\n\tst g14,%1", operands);
00643 }
00644 return "";
00645 }
00646
00647
00648
00649 const char *
00650 i960_output_move_quad (dst, src)
00651 rtx dst, src;
00652 {
00653 rtx operands[7];
00654
00655 if (GET_CODE (dst) == REG
00656 && GET_CODE (src) == REG)
00657 {
00658 if ((REGNO (src) & 3)
00659 || (REGNO (dst) & 3))
00660 {
00661
00662
00663
00664
00665 if (REGNO (dst) <= REGNO (src) + 3
00666 && REGNO (dst) >= REGNO (src))
00667 return "mov %F1,%F0\n\tmov %E1,%E0\n\tmov %D1,%D0\n\tmov %1,%0";
00668 else
00669 return "mov %1,%0\n\tmov %D1,%D0\n\tmov %E1,%E0\n\tmov %F1,%F0";
00670 }
00671 else
00672 return "movq %1,%0";
00673 }
00674 else if (GET_CODE (dst) == REG
00675 && GET_CODE (src) == CONST_INT
00676 && CONST_OK_FOR_LETTER_P (INTVAL (src), 'I'))
00677 {
00678 if (REGNO (dst) & 3)
00679 return "mov %1,%0\n\tmov 0,%D0\n\tmov 0,%E0\n\tmov 0,%F0";
00680 else
00681 return "movq %1,%0";
00682 }
00683 else if (GET_CODE (dst) == REG
00684 && GET_CODE (src) == MEM)
00685 {
00686 if (REGNO (dst) & 3)
00687 {
00688
00689
00690
00691 operands[0] = dst;
00692 operands[1] = src;
00693 operands[2] = gen_rtx_REG (Pmode, REGNO (dst) + 3);
00694 operands[3] = gen_rtx_MEM (word_mode, operands[2]);
00695 operands[4]
00696 = adjust_address (operands[3], word_mode, UNITS_PER_WORD);
00697 operands[5]
00698 = adjust_address (operands[4], word_mode, UNITS_PER_WORD);
00699 operands[6]
00700 = adjust_address (operands[5], word_mode, UNITS_PER_WORD);
00701 output_asm_insn ("lda %1,%2\n\tld %3,%0\n\tld %4,%D0\n\tld %5,%E0\n\tld %6,%F0", operands);
00702 return "";
00703 }
00704 else
00705 return "ldq %1,%0";
00706 }
00707 else if (GET_CODE (dst) == MEM
00708 && GET_CODE (src) == REG)
00709 {
00710 if (REGNO (src) & 3)
00711 {
00712 operands[0] = dst;
00713 operands[1] = adjust_address (dst, word_mode, UNITS_PER_WORD);
00714 operands[2] = adjust_address (dst, word_mode, 2 * UNITS_PER_WORD);
00715 operands[3] = adjust_address (dst, word_mode, 3 * UNITS_PER_WORD);
00716 if (! memory_address_p (word_mode, XEXP (operands[3], 0)))
00717 abort ();
00718 operands[4] = src;
00719 output_asm_insn ("st %4,%0\n\tst %D4,%1\n\tst %E4,%2\n\tst %F4,%3", operands);
00720 return "";
00721 }
00722 return "stq %1,%0";
00723 }
00724 else
00725 abort ();
00726 }
00727
00728
00729
00730 const char *
00731 i960_output_move_quad_zero (dst)
00732 rtx dst;
00733 {
00734 rtx operands[4];
00735
00736 operands[0] = dst;
00737 {
00738 operands[1] = adjust_address (dst, word_mode, 4);
00739 operands[2] = adjust_address (dst, word_mode, 8);
00740 operands[3] = adjust_address (dst, word_mode, 12);
00741 output_asm_insn ("st g14,%0\n\tst g14,%1\n\tst g14,%2\n\tst g14,%3", operands);
00742 }
00743 return "";
00744 }
00745
00746
00747
00748
00749
00750 const char *
00751 i960_output_ldconst (dst, src)
00752 register rtx dst, src;
00753 {
00754 register int rsrc1;
00755 register unsigned rsrc2;
00756 enum machine_mode mode = GET_MODE (dst);
00757 rtx operands[4];
00758
00759 operands[0] = operands[2] = dst;
00760 operands[1] = operands[3] = src;
00761
00762
00763
00764
00765 if (GET_CODE (src) != CONST_INT && GET_CODE (src) != CONST_DOUBLE)
00766 {
00767 output_asm_insn ("ldconst %1,%0", operands);
00768 return "";
00769 }
00770 else if (mode == XFmode)
00771 {
00772 REAL_VALUE_TYPE d;
00773 long value_long[3];
00774 int i;
00775
00776 if (fp_literal_zero (src, XFmode))
00777 return "movt 0,%0";
00778
00779 REAL_VALUE_FROM_CONST_DOUBLE (d, src);
00780 REAL_VALUE_TO_TARGET_LONG_DOUBLE (d, value_long);
00781
00782 output_asm_insn ("# ldconst %1,%0",operands);
00783
00784 for (i = 0; i < 3; i++)
00785 {
00786 operands[0] = gen_rtx_REG (SImode, REGNO (dst) + i);
00787 operands[1] = GEN_INT (value_long[i]);
00788 output_asm_insn (i960_output_ldconst (operands[0], operands[1]),
00789 operands);
00790 }
00791
00792 return "";
00793 }
00794 else if (mode == DFmode)
00795 {
00796 rtx first, second;
00797
00798 if (fp_literal_zero (src, DFmode))
00799 return "movl 0,%0";
00800
00801 split_double (src, &first, &second);
00802
00803 output_asm_insn ("# ldconst %1,%0",operands);
00804
00805 operands[0] = gen_rtx_REG (SImode, REGNO (dst));
00806 operands[1] = first;
00807 output_asm_insn (i960_output_ldconst (operands[0], operands[1]),
00808 operands);
00809 operands[0] = gen_rtx_REG (SImode, REGNO (dst) + 1);
00810 operands[1] = second;
00811 output_asm_insn (i960_output_ldconst (operands[0], operands[1]),
00812 operands);
00813 return "";
00814 }
00815 else if (mode == SFmode)
00816 {
00817 REAL_VALUE_TYPE d;
00818 long value;
00819
00820 REAL_VALUE_FROM_CONST_DOUBLE (d, src);
00821 REAL_VALUE_TO_TARGET_SINGLE (d, value);
00822
00823 output_asm_insn ("# ldconst %1,%0",operands);
00824 operands[0] = gen_rtx_REG (SImode, REGNO (dst));
00825 operands[1] = GEN_INT (value);
00826 output_asm_insn (i960_output_ldconst (operands[0], operands[1]),
00827 operands);
00828 return "";
00829 }
00830 else if (mode == TImode)
00831 {
00832
00833 abort ();
00834
00835
00836 rsrc1 = INTVAL (src);
00837 if (rsrc1 >= 0 && rsrc1 < 32)
00838 return "movq %1,%0";
00839 else
00840 output_asm_insn ("movq\t0,%0\t# ldconstq %1,%0",operands);
00841
00842 }
00843 else if (mode == DImode)
00844 {
00845 rtx upperhalf, lowerhalf, xoperands[2];
00846
00847 if (GET_CODE (src) == CONST_DOUBLE || GET_CODE (src) == CONST_INT)
00848 split_double (src, &lowerhalf, &upperhalf);
00849
00850 else
00851 abort ();
00852
00853
00854
00855 rsrc1 = INTVAL (lowerhalf);
00856 if (upperhalf == const0_rtx && rsrc1 >= 0 && rsrc1 < 32)
00857 return "movl %1,%0";
00858
00859
00860 xoperands[0] = gen_rtx_REG (SImode, REGNO (dst) + 1);
00861 xoperands[1] = upperhalf;
00862 output_asm_insn (i960_output_ldconst (xoperands[0], xoperands[1]),
00863 xoperands);
00864
00865 }
00866 else
00867 {
00868 rsrc1 = INTVAL (src);
00869 if (mode == QImode)
00870 {
00871 if (rsrc1 > 0xff)
00872 rsrc1 &= 0xff;
00873 }
00874 else if (mode == HImode)
00875 {
00876 if (rsrc1 > 0xffff)
00877 rsrc1 &= 0xffff;
00878 }
00879 }
00880
00881 if (rsrc1 >= 0)
00882 {
00883
00884 if (rsrc1 < 32)
00885 {
00886 if (i960_last_insn_type == I_TYPE_REG && TARGET_C_SERIES)
00887 return "lda %1,%0";
00888 return "mov %1,%0";
00889 }
00890
00891
00892 if (rsrc1 < 63)
00893 {
00894 if (i960_last_insn_type == I_TYPE_REG && TARGET_C_SERIES)
00895 return "lda %1,%0";
00896 operands[1] = GEN_INT (rsrc1 - 31);
00897 output_asm_insn ("addo\t31,%1,%0\t# ldconst %3,%0", operands);
00898 return "";
00899 }
00900 }
00901 else if (rsrc1 < 0)
00902 {
00903
00904 if (rsrc1 >= -31)
00905 {
00906
00907 operands[1] = GEN_INT (- rsrc1);
00908 output_asm_insn ("subo\t%1,0,%0\t# ldconst %3,%0", operands);
00909 return "";
00910 }
00911
00912
00913 if (rsrc1 == -32)
00914 {
00915 operands[1] = GEN_INT (~rsrc1);
00916 output_asm_insn ("not\t%1,%0 # ldconst %3,%0", operands);
00917 return "";
00918 }
00919 }
00920
00921
00922 if (bitpos (rsrc1) >= 0)
00923 {
00924 operands[1] = GEN_INT (bitpos (rsrc1));
00925 output_asm_insn ("setbit\t%1,0,%0\t# ldconst %3,%0", operands);
00926 return "";
00927 }
00928
00929
00930 if (is_mask (rsrc1))
00931 {
00932 int s, e;
00933
00934 if (bitstr (rsrc1, &s, &e) < 6)
00935 {
00936 rsrc2 = ((unsigned int) rsrc1) >> s;
00937 operands[1] = GEN_INT (rsrc2);
00938 operands[2] = GEN_INT (s);
00939 output_asm_insn ("shlo\t%2,%1,%0\t# ldconst %3,%0", operands);
00940 return "";
00941 }
00942 }
00943
00944
00945
00946
00947
00948
00949
00950 output_asm_insn ("ldconst %1,%0", operands);
00951 return "";
00952 }
00953
00954
00955
00956
00957
00958
00959
00960
00961
00962
00963
00964 int
00965 i960_bypass (insn, op1, op2, cmpbr_flag)
00966 register rtx insn, op1, op2;
00967 int cmpbr_flag;
00968 {
00969 register rtx prev_insn, prev_dest;
00970
00971 if (TARGET_C_SERIES)
00972 return 0;
00973
00974
00975 if (! REG_P (op1))
00976 return 0;
00977
00978
00979 if (cmpbr_flag && ! REG_P (op2))
00980 return 0;
00981
00982 prev_insn = prev_real_insn (insn);
00983
00984 if (prev_insn && GET_CODE (prev_insn) == INSN
00985 && GET_CODE (PATTERN (prev_insn)) == SET)
00986 {
00987 prev_dest = SET_DEST (PATTERN (prev_insn));
00988 if ((GET_CODE (prev_dest) == REG && REGNO (prev_dest) == REGNO (op1))
00989 || (GET_CODE (prev_dest) == SUBREG
00990 && GET_CODE (SUBREG_REG (prev_dest)) == REG
00991 && REGNO (SUBREG_REG (prev_dest)) == REGNO (op1)))
00992 return 1;
00993 }
00994 return 0;
00995 }
00996
00997
00998
00999
01000
01001 void
01002 i960_function_name_declare (file, name, fndecl)
01003 FILE *file;
01004 const char *name;
01005 tree fndecl;
01006 {
01007 register int i, j;
01008 int leaf_proc_ok;
01009 rtx insn;
01010
01011
01012
01013 ret_label++;
01014
01015
01016
01017
01018 if (TARGET_TAILCALL)
01019 tail_call_ok = 1;
01020 else
01021 tail_call_ok = 0;
01022
01023 if (TARGET_LEAFPROC)
01024 leaf_proc_ok = 1;
01025 else
01026 leaf_proc_ok = 0;
01027
01028
01029
01030
01031 if (current_function_args_size != 0 || VARARGS_STDARG_FUNCTION (fndecl))
01032 {
01033 tail_call_ok = 0;
01034 leaf_proc_ok = 0;
01035 }
01036
01037
01038
01039 if (aggregate_value_p (DECL_RESULT (fndecl)))
01040 {
01041 tail_call_ok = 0;
01042 leaf_proc_ok = 0;
01043 }
01044
01045
01046
01047
01048 if (get_frame_size () != 0)
01049 leaf_proc_ok = 0;
01050
01051
01052
01053
01054
01055 if (tail_call_ok)
01056 for (insn = get_insns (); insn; insn = NEXT_INSN (insn))
01057 if (GET_CODE (insn) == INSN
01058 && reg_mentioned_p (frame_pointer_rtx, insn))
01059 {
01060 tail_call_ok = 0;
01061 break;
01062 }
01063
01064
01065
01066 if (leaf_proc_ok)
01067 for (insn = get_insns (); insn; insn = NEXT_INSN (insn))
01068 if (GET_CODE (insn) == CALL_INSN)
01069 {
01070 leaf_proc_ok = 0;
01071 break;
01072 }
01073
01074
01075
01076
01077 if (leaf_proc_ok)
01078 for (i = 0, j = 0; i < FIRST_PSEUDO_REGISTER; i++)
01079 if (regs_ever_live[i]
01080 && ((! call_used_regs[i]) || (i > 7 && i < 12)))
01081 {
01082
01083 if (i < 16 && i > 7 && i != 13)
01084 leaf_proc_ok = 0;
01085
01086 else if (i < 32)
01087 leaf_proc_ok = 0;
01088 }
01089
01090
01091
01092
01093 i960_leaf_ret_reg = -1;
01094
01095 if (optimize && leaf_proc_ok)
01096 {
01097 for (i960_leaf_ret_reg = -1, i = 0; i < 8; i++)
01098 if (regs_ever_live[i] == 0)
01099 {
01100 i960_leaf_ret_reg = i;
01101 regs_ever_live[i] = 1;
01102 break;
01103 }
01104 }
01105
01106
01107
01108
01109 fprintf (file, "\t# Function '%s'\n", (name[0] == '*' ? &name[1] : name));
01110 fprintf (file, "\t# Registers used: ");
01111
01112 for (i = 0, j = 0; i < FIRST_PSEUDO_REGISTER; i++)
01113 {
01114 if (regs_ever_live[i])
01115 {
01116 fprintf (file, "%s%s ", reg_names[i], call_used_regs[i] ? "" : "*");
01117
01118 if (i > 15 && j == 0)
01119 {
01120 fprintf (file,"\n\t#\t\t ");
01121 j++;
01122 }
01123 }
01124 }
01125
01126 fprintf (file, "\n");
01127
01128 if (i960_leaf_ret_reg >= 0)
01129 {
01130
01131
01132 if (TREE_PUBLIC (fndecl))
01133 fprintf (file,"\t.globl\t%s.lf\n", (name[0] == '*' ? &name[1] : name));
01134
01135 fprintf (file, "\t.leafproc\t");
01136 assemble_name (file, name);
01137 fprintf (file, ",%s.lf\n", (name[0] == '*' ? &name[1] : name));
01138 ASM_OUTPUT_LABEL (file, name);
01139 fprintf (file, "\tlda Li960R%d,g14\n", ret_label);
01140 fprintf (file, "%s.lf:\n", (name[0] == '*' ? &name[1] : name));
01141 fprintf (file, "\tmov g14,g%d\n", i960_leaf_ret_reg);
01142
01143 if (TARGET_C_SERIES)
01144 {
01145 fprintf (file, "\tlda 0,g14\n");
01146 i960_last_insn_type = I_TYPE_MEM;
01147 }
01148 else
01149 {
01150 fprintf (file, "\tmov 0,g14\n");
01151 i960_last_insn_type = I_TYPE_REG;
01152 }
01153 }
01154 else
01155 {
01156 ASM_OUTPUT_LABEL (file, name);
01157 i960_last_insn_type = I_TYPE_CTRL;
01158 }
01159 }
01160
01161
01162
01163 int
01164 compute_frame_size (size)
01165 int size;
01166 {
01167 int actual_fsize;
01168 int outgoing_args_size = current_function_outgoing_args_size;
01169
01170
01171
01172 actual_fsize = (size + 15) & -16;
01173 actual_fsize += (outgoing_args_size + 15) & -16;
01174
01175 return actual_fsize;
01176 }
01177
01178
01179
01180
01181 struct reg_group
01182 {
01183 char start_reg;
01184 char length;
01185 };
01186
01187 static int i960_form_reg_groups PARAMS ((int, int, int *, int, struct reg_group *));
01188 static int i960_reg_group_compare PARAMS ((const void *, const void *));
01189 static int i960_split_reg_group PARAMS ((struct reg_group *, int, int));
01190 static void i960_arg_size_and_align PARAMS ((enum machine_mode, tree, int *, int *));
01191
01192
01193
01194
01195
01196 static int
01197 i960_form_reg_groups (start_reg, finish_reg, regs, state, reg_groups)
01198 int start_reg;
01199 int finish_reg;
01200 int *regs;
01201 int state;
01202 struct reg_group *reg_groups;
01203 {
01204 int i;
01205 int nw = 0;
01206
01207 for (i = start_reg; i < finish_reg; )
01208 {
01209 if (regs [i] != state)
01210 {
01211 i++;
01212 continue;
01213 }
01214 else if (i % 2 != 0 || regs [i + 1] != state)
01215 reg_groups [nw].length = 1;
01216 else if (i % 4 != 0 || regs [i + 2] != state)
01217 reg_groups [nw].length = 2;
01218 else if (regs [i + 3] != state)
01219 reg_groups [nw].length = 3;
01220 else
01221 reg_groups [nw].length = 4;
01222 reg_groups [nw].start_reg = i;
01223 i += reg_groups [nw].length;
01224 nw++;
01225 }
01226 return nw;
01227 }
01228
01229
01230 static int
01231 i960_reg_group_compare (group1, group2)
01232 const void *group1;
01233 const void *group2;
01234 {
01235 const struct reg_group *w1 = group1;
01236 const struct reg_group *w2 = group2;
01237
01238 if (w1->length > w2->length)
01239 return -1;
01240 else if (w1->length < w2->length)
01241 return 1;
01242 else
01243 return 0;
01244 }
01245
01246
01247
01248
01249 static int
01250 i960_split_reg_group (reg_groups, nw, subgroup_length)
01251 struct reg_group *reg_groups;
01252 int nw;
01253 int subgroup_length;
01254 {
01255 if (subgroup_length < reg_groups->length - subgroup_length)
01256
01257
01258
01259
01260 subgroup_length = reg_groups->length - subgroup_length;
01261
01262
01263
01264 reg_groups[nw].length = reg_groups->length - subgroup_length;
01265 reg_groups[nw].start_reg = reg_groups->start_reg + subgroup_length;
01266 nw++;
01267 reg_groups->length = subgroup_length;
01268 qsort (reg_groups, nw, sizeof (struct reg_group), i960_reg_group_compare);
01269 return nw;
01270 }
01271
01272
01273
01274 static void
01275 i960_output_function_prologue (file, size)
01276 FILE *file;
01277 HOST_WIDE_INT size;
01278 {
01279 register int i, j, nr;
01280 int n_saved_regs = 0;
01281 int n_remaining_saved_regs;
01282 HOST_WIDE_INT lvar_size;
01283 HOST_WIDE_INT actual_fsize, offset;
01284 int gnw, lnw;
01285 struct reg_group *g, *l;
01286 char tmpstr[1000];
01287
01288
01289 int regs[FIRST_PSEUDO_REGISTER];
01290
01291 struct reg_group global_reg_groups [16];
01292
01293 struct reg_group local_reg_groups [16];
01294
01295
01296 for (i = 0; i < FIRST_PSEUDO_REGISTER; i++)
01297 if (regs_ever_live[i]
01298 && ((! call_used_regs[i]) || (i > 7 && i < 12))
01299
01300 && ! (i == STATIC_CHAIN_REGNUM && current_function_needs_context))
01301 {
01302 regs[i] = -1;
01303
01304 if (i < 16)
01305 n_saved_regs++;
01306 }
01307 else
01308 regs[i] = 0;
01309
01310 n_remaining_saved_regs = n_saved_regs;
01311
01312 epilogue_string[0] = '\0';
01313
01314 if (current_function_profile)
01315 {
01316
01317
01318
01319 for (j = 7; j >= 0 && ! regs_ever_live[j]; j--)
01320 ;
01321
01322 for (i = 20; i <= j + 20; i++)
01323 regs[i] = -1;
01324 }
01325
01326 gnw = i960_form_reg_groups (0, 16, regs, -1, global_reg_groups);
01327 lnw = i960_form_reg_groups (19, 32, regs, 0, local_reg_groups);
01328 qsort (global_reg_groups, gnw, sizeof (struct reg_group),
01329 i960_reg_group_compare);
01330 qsort (local_reg_groups, lnw, sizeof (struct reg_group),
01331 i960_reg_group_compare);
01332 for (g = global_reg_groups, l = local_reg_groups; lnw != 0 && gnw != 0;)
01333 {
01334 if (g->length == l->length)
01335 {
01336 fprintf (file, "\tmov%s %s,%s\n",
01337 ((g->length == 4) ? "q" :
01338 (g->length == 3) ? "t" :
01339 (g->length == 2) ? "l" : ""),
01340 reg_names[(unsigned char) g->start_reg],
01341 reg_names[(unsigned char) l->start_reg]);
01342 sprintf (tmpstr, "\tmov%s %s,%s\n",
01343 ((g->length == 4) ? "q" :
01344 (g->length == 3) ? "t" :
01345 (g->length == 2) ? "l" : ""),
01346 reg_names[(unsigned char) l->start_reg],
01347 reg_names[(unsigned char) g->start_reg]);
01348 strcat (epilogue_string, tmpstr);
01349 n_remaining_saved_regs -= g->length;
01350 for (i = 0; i < g->length; i++)
01351 {
01352 regs [i + g->start_reg] = 1;
01353 regs [i + l->start_reg] = -1;
01354 regs_ever_live [i + l->start_reg] = 1;
01355 }
01356 g++;
01357 l++;
01358 gnw--;
01359 lnw--;
01360 }
01361 else if (g->length > l->length)
01362 gnw = i960_split_reg_group (g, gnw, l->length);
01363 else
01364 lnw = i960_split_reg_group (l, lnw, g->length);
01365 }
01366
01367 actual_fsize = compute_frame_size (size) + 4 * n_remaining_saved_regs;
01368 #if 0
01369
01370
01371
01372
01373
01374
01375
01376
01377 actual_fsize = (actual_fsize + 15) & ~0xF;
01378 #endif
01379
01380
01381 if (current_function_limit_stack)
01382 {
01383 rtx min_stack = stack_limit_rtx;
01384 if (actual_fsize != 0)
01385 min_stack = plus_constant (stack_limit_rtx, -actual_fsize);
01386
01387
01388
01389 if (legitimate_address_p (Pmode, min_stack, 1)
01390 && !arith_operand (min_stack, Pmode))
01391 {
01392 rtx tmp = gen_rtx_MEM (Pmode, min_stack);
01393 fputs ("\tlda\t", file);
01394 i960_print_operand (file, tmp, 0);
01395 fputs (",r4\n", file);
01396 min_stack = gen_rtx_REG (Pmode, 20);
01397 }
01398 if (arith_operand (min_stack, Pmode))
01399 {
01400 fputs ("\tcmpo\tsp,", file);
01401 i960_print_operand (file, min_stack, 0);
01402 fputs ("\n\tfaultge.f\n", file);
01403 }
01404 else
01405 warning ("stack limit expression is not supported");
01406 }
01407
01408
01409 if (actual_fsize > 0)
01410 {
01411 if (actual_fsize < 32)
01412 fprintf (file, "\taddo %d,sp,sp\n", actual_fsize);
01413 else
01414 fprintf (file, "\tlda\t%d(sp),sp\n", actual_fsize);
01415 }
01416
01417
01418
01419 lvar_size = actual_fsize - compute_frame_size (0) - n_remaining_saved_regs * 4;
01420 offset = STARTING_FRAME_OFFSET + lvar_size;
01421
01422
01423
01424 for (i = 0, j = n_remaining_saved_regs; j > 0 && i < 16; i++)
01425 {
01426 if (regs[i] != -1)
01427 continue;
01428
01429 nr = 1;
01430
01431 if (i <= 14 && i % 2 == 0 && regs[i+1] == -1 && offset % 2 == 0)
01432 nr = 2;
01433
01434 if (nr == 2 && i <= 12 && i % 4 == 0 && regs[i+2] == -1
01435 && offset % 4 == 0)
01436 nr = 3;
01437
01438 if (nr == 3 && regs[i+3] == -1)
01439 nr = 4;
01440
01441 fprintf (file,"\tst%s %s,%d(fp)\n",
01442 ((nr == 4) ? "q" :
01443 (nr == 3) ? "t" :
01444 (nr == 2) ? "l" : ""),
01445 reg_names[i], offset);
01446 sprintf (tmpstr,"\tld%s %d(fp),%s\n",
01447 ((nr == 4) ? "q" :
01448 (nr == 3) ? "t" :
01449 (nr == 2) ? "l" : ""),
01450 offset, reg_names[i]);
01451 strcat (epilogue_string, tmpstr);
01452 i += nr-1;
01453 j -= nr;
01454 offset += nr * 4;
01455 }
01456
01457 if (actual_fsize == 0)
01458 return;
01459
01460 fprintf (file, "\t#Prologue stats:\n");
01461 fprintf (file, "\t# Total Frame Size: %d bytes\n", actual_fsize);
01462
01463 if (lvar_size)
01464 fprintf (file, "\t# Local Variable Size: %d bytes\n", lvar_size);
01465 if (n_saved_regs)
01466 fprintf (file, "\t# Register Save Size: %d regs, %d bytes\n",
01467 n_saved_regs, n_saved_regs * 4);
01468 fprintf (file, "\t#End Prologue#\n");
01469 }
01470
01471
01472
01473 void
01474 output_function_profiler (file, labelno)
01475 FILE *file;
01476 int labelno;
01477 {
01478
01479 int last_parm_reg;
01480 int i, j, increment;
01481 int varargs_stdarg_function
01482 = VARARGS_STDARG_FUNCTION (current_function_decl);
01483
01484
01485
01486
01487
01488
01489
01490
01491
01492
01493 for (last_parm_reg = 7;
01494 last_parm_reg >= 0 && ! regs_ever_live[last_parm_reg];
01495 last_parm_reg--)
01496 ;
01497
01498
01499
01500 for (i = 0, j = 4; i <= last_parm_reg; i += increment, j += increment)
01501 {
01502 if (i % 4 == 0 && (last_parm_reg - i) >= 3)
01503 increment = 4;
01504 else if (i % 4 == 0 && (last_parm_reg - i) >= 2)
01505 increment = 3;
01506 else if (i % 2 == 0 && (last_parm_reg - i) >= 1)
01507 increment = 2;
01508 else
01509 increment = 1;
01510
01511 fprintf (file, "\tmov%s g%d,r%d\n",
01512 (increment == 4 ? "q" : increment == 3 ? "t"
01513 : increment == 2 ? "l": ""), i, j);
01514 }
01515
01516
01517
01518
01519 if (current_function_args_size != 0 || varargs_stdarg_function)
01520 fprintf (file, "\tmov g14,r3\n\tmov 0,g14\n");
01521
01522
01523
01524 fprintf (file, "\tlda\tLP%d,g0\n\tcallx\tmcount\n", labelno);
01525
01526
01527
01528 if (current_function_args_size != 0 || varargs_stdarg_function)
01529 fprintf (file, "\tmov r3,g14\n");
01530
01531
01532
01533 for (i = 0, j = 4; i <= last_parm_reg; i += increment, j += increment)
01534 {
01535 if (i % 4 == 0 && (last_parm_reg - i) >= 3)
01536 increment = 4;
01537 else if (i % 4 == 0 && (last_parm_reg - i) >= 2)
01538 increment = 3;
01539 else if (i % 2 == 0 && (last_parm_reg - i) >= 1)
01540 increment = 2;
01541 else
01542 increment = 1;
01543
01544 fprintf (file, "\tmov%s r%d,g%d\n",
01545 (increment == 4 ? "q" : increment == 3 ? "t"
01546 : increment == 2 ? "l": ""), j, i);
01547 }
01548 }
01549
01550
01551
01552 static void
01553 i960_output_function_epilogue (file, size)
01554 FILE *file;
01555 HOST_WIDE_INT size ATTRIBUTE_UNUSED;
01556 {
01557 if (i960_leaf_ret_reg >= 0)
01558 {
01559 fprintf (file, "Li960R%d: ret\n", ret_label);
01560 return;
01561 }
01562
01563 if (*epilogue_string == 0)
01564 {
01565 register rtx tmp;
01566
01567
01568
01569 tmp = get_last_insn ();
01570 while (tmp)
01571 {
01572 if (GET_CODE (tmp) == BARRIER)
01573 return;
01574 if (GET_CODE (tmp) == CODE_LABEL)
01575 break;
01576 if (GET_CODE (tmp) == JUMP_INSN)
01577 {
01578 if (GET_CODE (PATTERN (tmp)) == RETURN)
01579 return;
01580 break;
01581 }
01582 if (GET_CODE (tmp) == NOTE)
01583 {
01584 tmp = PREV_INSN (tmp);
01585 continue;
01586 }
01587 break;
01588 }
01589 fprintf (file, "Li960R%d: ret\n", ret_label);
01590 return;
01591 }
01592
01593 fprintf (file, "Li960R%d:\n", ret_label);
01594
01595 fprintf (file, "\t#EPILOGUE#\n");
01596
01597
01598
01599
01600 if (epilogue_string[0] != '\0')
01601 fprintf (file, "%s", epilogue_string);
01602
01603
01604
01605
01606 if (VARARGS_STDARG_FUNCTION (current_function_decl))
01607 fprintf (file, "\tmov 0,g14\n");
01608
01609 fprintf (file, "\tret\n");
01610 fprintf (file, "\t#End Epilogue#\n");
01611 }
01612
01613
01614
01615 const char *
01616 i960_output_call_insn (target, argsize_rtx, arg_pointer, insn)
01617 register rtx target, argsize_rtx, arg_pointer, insn;
01618 {
01619 int argsize = INTVAL (argsize_rtx);
01620 rtx nexti = next_real_insn (insn);
01621 rtx operands[2];
01622 int varargs_stdarg_function
01623 = VARARGS_STDARG_FUNCTION (current_function_decl);
01624
01625 operands[0] = target;
01626 operands[1] = arg_pointer;
01627
01628 if (current_function_args_size != 0 || varargs_stdarg_function)
01629 output_asm_insn ("mov g14,r3", operands);
01630
01631 if (argsize > 48)
01632 output_asm_insn ("lda %a1,g14", operands);
01633 else if (current_function_args_size != 0 || varargs_stdarg_function)
01634 output_asm_insn ("mov 0,g14", operands);
01635
01636
01637
01638
01639
01640
01641
01642 if (optimize && (*epilogue_string == 0) && argsize == 0 && tail_call_ok
01643 && (nexti == 0 || GET_CODE (PATTERN (nexti)) == RETURN))
01644 {
01645
01646 if (nexti && no_labels_between_p (insn, nexti))
01647 delete_insn (nexti);
01648 output_asm_insn ("bx %0", operands);
01649 return "# notreached";
01650 }
01651
01652 output_asm_insn ("callx %0", operands);
01653
01654
01655
01656
01657 if (current_function_args_size != 0 || varargs_stdarg_function)
01658 output_asm_insn ("mov r3,g14", operands);
01659 else if (argsize > 48)
01660 output_asm_insn ("mov 0,g14", operands);
01661
01662 return "";
01663 }
01664
01665
01666
01667 const char *
01668 i960_output_ret_insn (insn)
01669 register rtx insn;
01670 {
01671 static char lbuf[20];
01672
01673 if (*epilogue_string != 0)
01674 {
01675 if (! TARGET_CODE_ALIGN && next_real_insn (insn) == 0)
01676 return "";
01677
01678 sprintf (lbuf, "b Li960R%d", ret_label);
01679 return lbuf;
01680 }
01681
01682
01683
01684
01685 if (VARARGS_STDARG_FUNCTION (current_function_decl))
01686 output_asm_insn ("mov 0,g14", 0);
01687
01688 if (i960_leaf_ret_reg >= 0)
01689 {
01690 sprintf (lbuf, "bx (%s)", reg_names[i960_leaf_ret_reg]);
01691 return lbuf;
01692 }
01693 return "ret";
01694 }
01695
01696
01697
01698 void
01699 i960_print_operand (file, x, code)
01700 FILE *file;
01701 rtx x;
01702 int code;
01703 {
01704 enum rtx_code rtxcode = x ? GET_CODE (x) : NIL;
01705
01706 if (rtxcode == REG)
01707 {
01708 switch (code)
01709 {
01710 case 'D':
01711
01712 fprintf (file, "%s", reg_names[REGNO (x)+1]);
01713 break;
01714
01715 case 'E':
01716
01717 fprintf (file, "%s", reg_names[REGNO (x)+2]);
01718 break;
01719
01720 case 'F':
01721
01722 fprintf (file, "%s", reg_names[REGNO (x)+3]);
01723 break;
01724
01725 case 0:
01726 fprintf (file, "%s", reg_names[REGNO (x)]);
01727 break;
01728
01729 default:
01730 abort ();
01731 }
01732 return;
01733 }
01734 else if (rtxcode == MEM)
01735 {
01736 output_address (XEXP (x, 0));
01737 return;
01738 }
01739 else if (rtxcode == CONST_INT)
01740 {
01741 HOST_WIDE_INT val = INTVAL (x);
01742 if (code == 'C')
01743 val = ~val;
01744 if (val > 9999 || val < -999)
01745 fprintf (file, "0x%x", val);
01746 else
01747 fprintf (file, "%d", val);
01748 return;
01749 }
01750 else if (rtxcode == CONST_DOUBLE)
01751 {
01752 REAL_VALUE_TYPE d;
01753 char dstr[30];
01754
01755 if (x == CONST0_RTX (GET_MODE (x)))
01756 {
01757 fprintf (file, "0f0.0");
01758 return;
01759 }
01760 else if (x == CONST1_RTX (GET_MODE (x)))
01761 {
01762 fprintf (file, "0f1.0");
01763 return;
01764 }
01765
01766 REAL_VALUE_FROM_CONST_DOUBLE (d, x);
01767 REAL_VALUE_TO_DECIMAL (d, "%#g", dstr);
01768 fprintf (file, "0f%s", dstr);
01769 return;
01770 }
01771
01772 switch(code)
01773 {
01774 case 'B':
01775
01776 if (TARGET_ASM_COMPAT)
01777 fputs ("j", file);
01778 else
01779 fputs ("b", file);
01780 break;
01781
01782 case 'S':
01783
01784 if ((rtxcode == EQ) || (rtxcode == NE) || (rtxcode == GTU)
01785 || (rtxcode == LTU) || (rtxcode == GEU) || (rtxcode == LEU))
01786 fputs ("o", file);
01787 else if ((rtxcode == GT) || (rtxcode == LT)
01788 || (rtxcode == GE) || (rtxcode == LE))
01789 fputs ("i", file);
01790 else
01791 abort();
01792 break;
01793
01794 case 'I':
01795
01796 rtxcode = reverse_condition (rtxcode);
01797 goto normal;
01798
01799 case 'X':
01800
01801 rtxcode = reverse_condition (rtxcode);
01802
01803
01804 case 'R':
01805
01806 rtxcode = swap_condition (rtxcode);
01807
01808
01809 case 'C':
01810
01811 normal:
01812 if (rtxcode == EQ) { fputs ("e", file); return; }
01813 else if (rtxcode == NE) { fputs ("ne", file); return; }
01814 else if (rtxcode == GT) { fputs ("g", file); return; }
01815 else if (rtxcode == GTU) { fputs ("g", file); return; }
01816 else if (rtxcode == LT) { fputs ("l", file); return; }
01817 else if (rtxcode == LTU) { fputs ("l", file); return; }
01818 else if (rtxcode == GE) { fputs ("ge", file); return; }
01819 else if (rtxcode == GEU) { fputs ("ge", file); return; }
01820 else if (rtxcode == LE) { fputs ("le", file); return; }
01821 else if (rtxcode == LEU) { fputs ("le", file); return; }
01822 else abort ();
01823 break;
01824
01825 case '+':
01826
01827 if (TARGET_BRANCH_PREDICT)
01828 {
01829 x = find_reg_note (current_output_insn, REG_BR_PROB, 0);
01830 if (x)
01831 {
01832 int pred_val = INTVAL (XEXP (x, 0));
01833 fputs ((pred_val < REG_BR_PROB_BASE / 2 ? ".f" : ".t"), file);
01834 }
01835 }
01836 break;
01837
01838 case 0:
01839 output_addr_const (file, x);
01840 break;
01841
01842 default:
01843 abort ();
01844 }
01845
01846 return;
01847 }
01848
01849
01850
01851
01852
01853
01854 void
01855 i960_print_operand_addr (file, addr)
01856 FILE *file;
01857 register rtx addr;
01858 {
01859 rtx breg, ireg;
01860 rtx scale, offset;
01861
01862 ireg = 0;
01863 breg = 0;
01864 offset = 0;
01865 scale = const1_rtx;
01866
01867 if (GET_CODE (addr) == REG)
01868 breg = addr;
01869 else if (CONSTANT_P (addr))
01870 offset = addr;
01871 else if (GET_CODE (addr) == PLUS)
01872 {
01873 rtx op0, op1;
01874
01875 op0 = XEXP (addr, 0);
01876 op1 = XEXP (addr, 1);
01877
01878 if (GET_CODE (op0) == REG)
01879 {
01880 breg = op0;
01881 if (GET_CODE (op1) == REG)
01882 ireg = op1;
01883 else if (CONSTANT_P (op1))
01884 offset = op1;
01885 else
01886 abort ();
01887 }
01888 else if (GET_CODE (op0) == PLUS)
01889 {
01890 if (GET_CODE (XEXP (op0, 0)) == MULT)
01891 {
01892 ireg = XEXP (XEXP (op0, 0), 0);
01893 scale = XEXP (XEXP (op0, 0), 1);
01894 if (GET_CODE (XEXP (op0, 1)) == REG)
01895 {
01896 breg = XEXP (op0, 1);
01897 offset = op1;
01898 }
01899 else
01900 abort ();
01901 }
01902 else if (GET_CODE (XEXP (op0, 0)) == REG)
01903 {
01904 breg = XEXP (op0, 0);
01905 if (GET_CODE (XEXP (op0, 1)) == REG)
01906 {
01907 ireg = XEXP (op0, 1);
01908 offset = op1;
01909 }
01910 else
01911 abort ();
01912 }
01913 else
01914 abort ();
01915 }
01916 else if (GET_CODE (op0) == MULT)
01917 {
01918 ireg = XEXP (op0, 0);
01919 scale = XEXP (op0, 1);
01920 if (GET_CODE (op1) == REG)
01921 breg = op1;
01922 else if (CONSTANT_P (op1))
01923 offset = op1;
01924 else
01925 abort ();
01926 }
01927 else
01928 abort ();
01929 }
01930 else if (GET_CODE (addr) == MULT)
01931 {
01932 ireg = XEXP (addr, 0);
01933 scale = XEXP (addr, 1);
01934 }
01935 else
01936 abort ();
01937
01938 if (offset)
01939 output_addr_const (file, offset);
01940 if (breg)
01941 fprintf (file, "(%s)", reg_names[REGNO (breg)]);
01942 if (ireg)
01943 fprintf (file, "[%s*%d]", reg_names[REGNO (ireg)], INTVAL (scale));
01944 }
01945
01946
01947
01948
01949
01950
01951
01952
01953
01954
01955
01956
01957
01958
01959
01960
01961
01962
01963
01964
01965
01966
01967
01968
01969
01970
01971
01972
01973
01974 #define RTX_OK_FOR_BASE_P(X, STRICT) \
01975 ((GET_CODE (X) == REG \
01976 && (STRICT ? REG_OK_FOR_BASE_P_STRICT (X) : REG_OK_FOR_BASE_P (X))) \
01977 || (GET_CODE (X) == SUBREG \
01978 && GET_CODE (SUBREG_REG (X)) == REG \
01979 && (STRICT ? REG_OK_FOR_BASE_P_STRICT (SUBREG_REG (X)) \
01980 : REG_OK_FOR_BASE_P (SUBREG_REG (X)))))
01981
01982 #define RTX_OK_FOR_INDEX_P(X, STRICT) \
01983 ((GET_CODE (X) == REG \
01984 && (STRICT ? REG_OK_FOR_INDEX_P_STRICT (X) : REG_OK_FOR_INDEX_P (X)))\
01985 || (GET_CODE (X) == SUBREG \
01986 && GET_CODE (SUBREG_REG (X)) == REG \
01987 && (STRICT ? REG_OK_FOR_INDEX_P_STRICT (SUBREG_REG (X)) \
01988 : REG_OK_FOR_INDEX_P (SUBREG_REG (X)))))
01989
01990 int
01991 legitimate_address_p (mode, addr, strict)
01992 enum machine_mode mode ATTRIBUTE_UNUSED;
01993 register rtx addr;
01994 int strict;
01995 {
01996 if (RTX_OK_FOR_BASE_P (addr, strict))
01997 return 1;
01998 else if (CONSTANT_P (addr))
01999 return 1;
02000 else if (GET_CODE (addr) == PLUS)
02001 {
02002 rtx op0, op1;
02003
02004 if (! TARGET_COMPLEX_ADDR && ! reload_completed)
02005 return 0;
02006
02007 op0 = XEXP (addr, 0);
02008 op1 = XEXP (addr, 1);
02009
02010 if (RTX_OK_FOR_BASE_P (op0, strict))
02011 {
02012 if (RTX_OK_FOR_INDEX_P (op1, strict))
02013 return 1;
02014 else if (CONSTANT_P (op1))
02015 return 1;
02016 else
02017 return 0;
02018 }
02019 else if (GET_CODE (op0) == PLUS)
02020 {
02021 if (GET_CODE (XEXP (op0, 0)) == MULT)
02022 {
02023 if (! (RTX_OK_FOR_INDEX_P (XEXP (XEXP (op0, 0), 0), strict)
02024 && SCALE_TERM_P (XEXP (XEXP (op0, 0), 1))))
02025 return 0;
02026
02027 if (RTX_OK_FOR_BASE_P (XEXP (op0, 1), strict)
02028 && CONSTANT_P (op1))
02029 return 1;
02030 else
02031 return 0;
02032 }
02033 else if (RTX_OK_FOR_BASE_P (XEXP (op0, 0), strict))
02034 {
02035 if (RTX_OK_FOR_INDEX_P (XEXP (op0, 1), strict)
02036 && CONSTANT_P (op1))
02037 return 1;
02038 else
02039 return 0;
02040 }
02041 else
02042 return 0;
02043 }
02044 else if (GET_CODE (op0) == MULT)
02045 {
02046 if (! (RTX_OK_FOR_INDEX_P (XEXP (op0, 0), strict)
02047 && SCALE_TERM_P (XEXP (op0, 1))))
02048 return 0;
02049
02050 if (RTX_OK_FOR_BASE_P (op1, strict))
02051 return 1;
02052 else if (CONSTANT_P (op1))
02053 return 1;
02054 else
02055 return 0;
02056 }
02057 else
02058 return 0;
02059 }
02060 else if (GET_CODE (addr) == MULT)
02061 {
02062 if (! TARGET_COMPLEX_ADDR && ! reload_completed)
02063 return 0;
02064
02065 return (RTX_OK_FOR_INDEX_P (XEXP (addr, 0), strict)
02066 && SCALE_TERM_P (XEXP (addr, 1)));
02067 }
02068 else
02069 return 0;
02070 }
02071
02072
02073
02074
02075
02076
02077
02078
02079 rtx
02080 legitimize_address (x, oldx, mode)
02081 register rtx x;
02082 register rtx oldx ATTRIBUTE_UNUSED;
02083 enum machine_mode mode ATTRIBUTE_UNUSED;
02084 {
02085 if (GET_CODE (x) == SYMBOL_REF)
02086 {
02087 abort ();
02088 x = copy_to_reg (x);
02089 }
02090
02091 if (! TARGET_COMPLEX_ADDR && ! reload_completed)
02092 return x;
02093
02094
02095
02096
02097
02098 if (GET_CODE (x) == PLUS && GET_CODE (XEXP (x, 0)) == MULT
02099 && GET_CODE (XEXP (x, 1)) == PLUS)
02100 x = gen_rtx_PLUS (Pmode,
02101 gen_rtx_PLUS (Pmode, XEXP (x, 0), XEXP (XEXP (x, 1), 0)),
02102 XEXP (XEXP (x, 1), 1));
02103
02104
02105
02106 else if (GET_CODE (x) == PLUS && GET_CODE (XEXP (x, 0)) == PLUS
02107 && GET_CODE (XEXP (XEXP (x, 0), 0)) == MULT
02108 && GET_CODE (XEXP (XEXP (x, 0), 1)) == PLUS
02109 && CONSTANT_P (XEXP (x, 1)))
02110 {
02111 rtx constant, other;
02112
02113 if (GET_CODE (XEXP (x, 1)) == CONST_INT)
02114 {
02115 constant = XEXP (x, 1);
02116 other = XEXP (XEXP (XEXP (x, 0), 1), 1);
02117 }
02118 else if (GET_CODE (XEXP (XEXP (XEXP (x, 0), 1), 1)) == CONST_INT)
02119 {
02120 constant = XEXP (XEXP (XEXP (x, 0), 1), 1);
02121 other = XEXP (x, 1);
02122 }
02123 else
02124 constant = 0, other = 0;
02125
02126 if (constant)
02127 x = gen_rtx_PLUS (Pmode,
02128 gen_rtx_PLUS (Pmode, XEXP (XEXP (x, 0), 0),
02129 XEXP (XEXP (XEXP (x, 0), 1), 0)),
02130 plus_constant (other, INTVAL (constant)));
02131 }
02132
02133 return x;
02134 }
02135
02136 #if 0
02137
02138
02139
02140 int
02141 i960_alignment (size, align)
02142 int size;
02143 int align;
02144 {
02145 int i;
02146
02147 if (! TARGET_STRICT_ALIGN)
02148 if (TARGET_IC_COMPAT2_0 || align >= 4)
02149 {
02150 i = i960_object_bytes_bitalign (size) / BITS_PER_UNIT;
02151 if (i > align)
02152 align = i;
02153 }
02154
02155 return align;
02156 }
02157 #endif
02158
02159
02160 int
02161 hard_regno_mode_ok (regno, mode)
02162 int regno;
02163 enum machine_mode mode;
02164 {
02165 if (regno < 32)
02166 {
02167 switch (mode)
02168 {
02169 case CCmode: case CC_UNSmode: case CC_CHKmode:
02170 return 0;
02171
02172 case DImode: case DFmode:
02173 return (regno & 1) == 0;
02174
02175 case TImode: case XFmode:
02176 return (regno & 3) == 0;
02177
02178 default:
02179 return 1;
02180 }
02181 }
02182 else if (regno >= 32 && regno < 36)
02183 {
02184 switch (mode)
02185 {
02186 case SFmode: case DFmode: case XFmode:
02187 case SCmode: case DCmode:
02188 return 1;
02189
02190 default:
02191 return 0;
02192 }
02193 }
02194 else if (regno == 36)
02195 {
02196 switch (mode)
02197 {
02198 case CCmode: case CC_UNSmode: case CC_CHKmode:
02199 return 1;
02200
02201 default:
02202 return 0;
02203 }
02204 }
02205 else if (regno == 37)
02206 return 0;
02207
02208 abort ();
02209 }
02210
02211
02212
02213
02214
02215
02216 int
02217 i960_expr_alignment (x, size)
02218 rtx x;
02219 int size;
02220 {
02221 int align = 1;
02222
02223 if (x == 0)
02224 return 1;
02225
02226 switch (GET_CODE(x))
02227 {
02228 case CONST_INT:
02229 align = INTVAL(x);
02230
02231 if ((align & 0xf) == 0)
02232 align = 16;
02233 else if ((align & 0x7) == 0)
02234 align = 8;
02235 else if ((align & 0x3) == 0)
02236 align = 4;
02237 else if ((align & 0x1) == 0)
02238 align = 2;
02239 else
02240 align = 1;
02241 break;
02242
02243 case PLUS:
02244 align = MIN (i960_expr_alignment (XEXP (x, 0), size),
02245 i960_expr_alignment (XEXP (x, 1), size));
02246 break;
02247
02248 case SYMBOL_REF:
02249
02250
02251 align = i960_object_bytes_bitalign (size) / BITS_PER_UNIT;
02252 break;
02253
02254 case REG:
02255 if (REGNO (x) == FRAME_POINTER_REGNUM)
02256 align = 16;
02257 break;
02258
02259 case ASHIFT:
02260 align = i960_expr_alignment (XEXP (x, 0), size);
02261
02262 if (GET_CODE (XEXP (x, 1)) == CONST_INT)
02263 {
02264 align = align << INTVAL (XEXP (x, 1));
02265 align = MIN (align, 16);
02266 }
02267 break;
02268
02269 case MULT:
02270 align = (i960_expr_alignment (XEXP (x, 0), size) *
02271 i960_expr_alignment (XEXP (x, 1), size));
02272
02273 align = MIN (align, 16);
02274 break;
02275 default:
02276 break;
02277 }
02278
02279 return align;
02280 }
02281
02282
02283
02284
02285
02286 int
02287 i960_improve_align (base, offset, size)
02288 rtx base;
02289 rtx offset;
02290 int size;
02291 {
02292 int i, j;
02293
02294
02295
02296
02297 i = MIN (i960_expr_alignment (base, 4),
02298 i960_expr_alignment (offset, 4));
02299
02300 i = MAX (i, 4);
02301
02302
02303
02304
02305 if (! TARGET_STRICT_ALIGN)
02306 if ((j = (i960_object_bytes_bitalign (size) / BITS_PER_UNIT)) > i)
02307 i = j;
02308
02309 return (i >= size);
02310 }
02311
02312
02313
02314
02315 int
02316 i960_si_ti (base, offset)
02317 rtx base;
02318 rtx offset;
02319 {
02320 return i960_improve_align (base, offset, 16);
02321 }
02322
02323
02324
02325
02326 int
02327 i960_si_di (base, offset)
02328 rtx base;
02329 rtx offset;
02330 {
02331 return i960_improve_align (base, offset, 8);
02332 }
02333
02334
02335
02336
02337 static void
02338 i960_arg_size_and_align (mode, type, size_out, align_out)
02339 enum machine_mode mode;
02340 tree type;
02341 int *size_out;
02342 int *align_out;
02343 {
02344 int size, align;
02345
02346
02347
02348
02349
02350
02351 if (mode == BLKmode)
02352 size = (int_size_in_bytes (type) + UNITS_PER_WORD - 1) / UNITS_PER_WORD;
02353 else if (mode == VOIDmode)
02354 {
02355
02356 if (type == 0 || TYPE_MODE (type) != VOIDmode)
02357 abort ();
02358 size = 1;
02359 }
02360 else
02361 size = (GET_MODE_SIZE (mode) + UNITS_PER_WORD - 1) / UNITS_PER_WORD;
02362
02363 if (type == 0)
02364 {
02365
02366
02367 if (size == 3)
02368 align = 4;
02369 else
02370 align = size;
02371 }
02372 else if (TYPE_ALIGN (type) >= BITS_PER_WORD)
02373 align = TYPE_ALIGN (type) / BITS_PER_WORD;
02374 else
02375 align = 1;
02376
02377 *size_out = size;
02378 *align_out = align;
02379 }
02380
02381
02382
02383
02384
02385
02386
02387
02388
02389
02390
02391
02392 void
02393 i960_function_arg_advance (cum, mode, type, named)
02394 CUMULATIVE_ARGS *cum;
02395 enum machine_mode mode;
02396 tree type;
02397 int named ATTRIBUTE_UNUSED;
02398 {
02399 int size, align;
02400
02401 i960_arg_size_and_align (mode, type, &size, &align);
02402
02403 if (size > 4 || cum->ca_nstackparms != 0
02404 || (size + ROUND_PARM (cum->ca_nregparms, align)) > NPARM_REGS
02405 || MUST_PASS_IN_STACK (mode, type))
02406 {
02407
02408
02409 cum->ca_nregparms = NPARM_REGS;
02410 cum->ca_nstackparms = ROUND_PARM (cum->ca_nstackparms, align) + size;
02411 }
02412 else
02413 cum->ca_nregparms = ROUND_PARM (cum->ca_nregparms, align) + size;
02414 }
02415
02416
02417
02418
02419 rtx
02420 i960_function_arg (cum, mode, type, named)
02421 CUMULATIVE_ARGS *cum;
02422 enum machine_mode mode;
02423 tree type;
02424 int named ATTRIBUTE_UNUSED;
02425 {
02426 rtx ret;
02427 int size, align;
02428
02429 if (mode == VOIDmode)
02430 return 0;
02431
02432 i960_arg_size_and_align (mode, type, &size, &align);
02433
02434 if (size > 4 || cum->ca_nstackparms != 0
02435 || (size + ROUND_PARM (cum->ca_nregparms, align)) > NPARM_REGS
02436 || MUST_PASS_IN_STACK (mode, type))
02437 {
02438 cum->ca_nstackparms = ROUND_PARM (cum->ca_nstackparms, align);
02439 ret = 0;
02440 }
02441 else
02442 {
02443 cum->ca_nregparms = ROUND_PARM (cum->ca_nregparms, align);
02444 ret = gen_rtx_REG (mode, cum->ca_nregparms);
02445 }
02446
02447 return ret;
02448 }
02449
02450
02451
02452 int
02453 i960_object_bytes_bitalign (n)
02454 int n;
02455 {
02456 if (n > 8) n = 128;
02457 else if (n > 4) n = 64;
02458 else if (n > 2) n = 32;
02459 else if (n > 1) n = 16;
02460 else n = 8;
02461
02462 return n;
02463 }
02464
02465
02466
02467
02468
02469 int
02470 i960_round_align (align, tsize)
02471 int align;
02472 tree tsize;
02473 {
02474 int new_align;
02475
02476 if (! tsize || TREE_CODE (tsize) != INTEGER_CST)
02477 return align;
02478
02479 new_align = i960_object_bytes_bitalign (TREE_INT_CST_LOW (tsize)
02480 / BITS_PER_UNIT);
02481
02482 if (new_align > i960_maxbitalignment)
02483 new_align = i960_maxbitalignment;
02484
02485 if (align < new_align)
02486 align = new_align;
02487
02488 return align;
02489 }
02490
02491
02492
02493
02494
02495 void
02496 i960_setup_incoming_varargs (cum, mode, type, pretend_size, no_rtl)
02497 CUMULATIVE_ARGS *cum;
02498 enum machine_mode mode ATTRIBUTE_UNUSED;
02499 tree type ATTRIBUTE_UNUSED;
02500 int *pretend_size ATTRIBUTE_UNUSED;
02501 int no_rtl;
02502 {
02503
02504 int first_reg = cum->ca_nregparms;
02505
02506
02507
02508
02509
02510
02511
02512
02513
02514
02515
02516 if (cum->ca_nstackparms == 0 && first_reg < NPARM_REGS && !no_rtl)
02517 {
02518 rtx label = gen_label_rtx ();
02519 rtx regblock;
02520
02521
02522
02523
02524
02525
02526 emit_insn (gen_cmpsi (arg_pointer_rtx, const0_rtx));
02527 emit_jump_insn (gen_bne (label));
02528 emit_insn (gen_rtx_SET (VOIDmode, arg_pointer_rtx,
02529 stack_pointer_rtx));
02530 emit_insn (gen_rtx_SET (VOIDmode, stack_pointer_rtx,
02531 memory_address (SImode,
02532 plus_constant (stack_pointer_rtx,
02533 48))));
02534 emit_label (label);
02535
02536
02537
02538 regblock = gen_rtx_MEM (BLKmode,
02539 plus_constant (arg_pointer_rtx, first_reg * 4));
02540 set_mem_alias_set (regblock, get_varargs_alias_set ());
02541 set_mem_align (regblock, BITS_PER_WORD);
02542 move_block_from_reg (first_reg, regblock,
02543 NPARM_REGS - first_reg,
02544 (NPARM_REGS - first_reg) * UNITS_PER_WORD);
02545 }
02546 }
02547
02548
02549
02550 tree
02551 i960_build_va_list ()
02552 {
02553 return build_array_type (unsigned_type_node,
02554 build_index_type (size_one_node));
02555 }
02556
02557
02558
02559 void
02560 i960_va_start (stdarg_p, valist, nextarg)
02561 int stdarg_p ATTRIBUTE_UNUSED;
02562 tree valist;
02563 rtx nextarg ATTRIBUTE_UNUSED;
02564 {
02565 tree s, t, base, num;
02566
02567
02568
02569 base = build1 (INDIRECT_REF, unsigned_type_node, valist);
02570 num = build1 (INDIRECT_REF, unsigned_type_node,
02571 build (PLUS_EXPR, unsigned_type_node, valist,
02572 TYPE_SIZE_UNIT (TREE_TYPE (valist))));
02573
02574 s = make_tree (unsigned_type_node, arg_pointer_rtx);
02575 t = build (MODIFY_EXPR, unsigned_type_node, base, s);
02576 TREE_SIDE_EFFECTS (t) = 1;
02577 expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL);
02578
02579 s = build_int_2 ((current_function_args_info.ca_nregparms
02580 + current_function_args_info.ca_nstackparms) * 4, 0);
02581 t = build (MODIFY_EXPR, unsigned_type_node, num, s);
02582 TREE_SIDE_EFFECTS (t) = 1;
02583 expand_expr (t, const0_rtx, VOIDmode, EXPAND_NORMAL);
02584 }
02585
02586
02587
02588 rtx
02589 i960_va_arg (valist, type)
02590 tree valist, type;
02591 {
02592 HOST_WIDE_INT siz, ali;
02593 tree base, num, pad, next, this, t1, t2, int48;
02594 rtx addr_rtx;
02595
02596
02597
02598 base = build1 (INDIRECT_REF, unsigned_type_node, valist);
02599 num = build1 (INDIRECT_REF, unsigned_type_node,
02600 build (PLUS_EXPR, unsigned_type_node, valist,
02601 TYPE_SIZE_UNIT (TREE_TYPE (valist))));
02602
02603
02604 siz = (int_size_in_bytes (type) + UNITS_PER_WORD - 1) & -UNITS_PER_WORD;
02605
02606
02607 ali = TYPE_ALIGN (type);
02608 if (ali < BITS_PER_WORD)
02609 ali = BITS_PER_WORD;
02610 ali /= BITS_PER_UNIT;
02611
02612
02613 pad = fold (build (PLUS_EXPR, unsigned_type_node, num,
02614 build_int_2 (ali - 1, 0)));
02615 pad = fold (build (BIT_AND_EXPR, unsigned_type_node, pad,
02616 build_int_2 (-ali, -1)));
02617 pad = save_expr (pad);
02618
02619
02620 next = fold (build (PLUS_EXPR, unsigned_type_node, pad,
02621 build_int_2 (siz, 0)));
02622 next = save_expr (next);
02623
02624
02625
02626 int48 = build_int_2 (48, 0);
02627 if (siz > 16)
02628 t2 = integer_one_node;
02629 else
02630 t2 = fold (build (GT_EXPR, integer_type_node, next, int48));
02631 t1 = fold (build (LE_EXPR, integer_type_node, num, int48));
02632 t1 = fold (build (TRUTH_AND_EXPR, integer_type_node, t1, t2));
02633 this = fold (build (COND_EXPR, unsigned_type_node, t1, int48, pad));
02634
02635
02636 t1 = fold (build (PLUS_EXPR, unsigned_type_node, base, this));
02637 t1 = build1 (NOP_EXPR, ptr_type_node, t1);
02638 addr_rtx = expand_expr (t1, NULL_RTX, Pmode, EXPAND_NORMAL);
02639
02640
02641 t1 = build (MODIFY_EXPR, unsigned_type_node, num, next);
02642 TREE_SIDE_EFFECTS (t1) = 1;
02643 expand_expr (t1, const0_rtx, VOIDmode, EXPAND_NORMAL);
02644
02645 return addr_rtx;
02646 }
02647
02648
02649
02650
02651 int
02652 i960_final_reg_parm_stack_space (const_size, var_size)
02653 int const_size;
02654 tree var_size;
02655 {
02656 if (var_size || const_size > 48)
02657 return 48;
02658 else
02659 return 0;
02660 }
02661
02662
02663
02664
02665 int
02666 i960_reg_parm_stack_space (fndecl)
02667 tree fndecl;
02668 {
02669
02670
02671 if (fndecl == 0)
02672 return 0;
02673
02674
02675
02676 if (fndecl != current_function_decl)
02677 return 48;
02678
02679
02680
02681 if (current_function_args_size != 0 || VARARGS_STDARG_FUNCTION (fndecl))
02682 return 48;
02683 else
02684 return 0;
02685 }
02686
02687
02688
02689
02690
02691 enum reg_class
02692 secondary_reload_class (class, mode, in)
02693 enum reg_class class;
02694 enum machine_mode mode;
02695 rtx in;
02696 {
02697 int regno = -1;
02698
02699 if (GET_CODE (in) == REG || GET_CODE (in) == SUBREG)
02700 regno = true_regnum (in);
02701
02702
02703
02704 if (class == LOCAL_OR_GLOBAL_REGS || class == LOCAL_REGS
02705 || class == GLOBAL_REGS || (regno >= 0 && regno < 32))
02706 return NO_REGS;
02707
02708
02709 if (class == FP_REGS
02710 && ((regno >= 0 && regno < FIRST_PSEUDO_REGISTER)
02711 || in == CONST0_RTX (mode) || in == CONST1_RTX (mode)))
02712 return NO_REGS;
02713
02714 return LOCAL_OR_GLOBAL_REGS;
02715 }
02716
02717
02718
02719
02720
02721
02722 void
02723 i960_scan_opcode (p)
02724 const char *p;
02725 {
02726 switch (*p)
02727 {
02728 case 'a':
02729 case 'd':
02730 case 'e':
02731 case 'm':
02732 case 'n':
02733 case 'o':
02734 case 'r':
02735
02736
02737 case 'u':
02738 case 'x':
02739 i960_last_insn_type = I_TYPE_REG;
02740 break;
02741
02742 case 'b':
02743 if (p[1] == 'x' || p[3] == 'x')
02744 i960_last_insn_type = I_TYPE_MEM;
02745 i960_last_insn_type = I_TYPE_CTRL;
02746 break;
02747
02748 case 'f':
02749 case 't':
02750 i960_last_insn_type = I_TYPE_CTRL;
02751 break;
02752
02753 case 'c':
02754 if (p[1] == 'a')
02755 {
02756 if (p[4] == 'x')
02757 i960_last_insn_type = I_TYPE_MEM;
02758 else
02759 i960_last_insn_type = I_TYPE_CTRL;
02760 }
02761 else if (p[1] == 'm')
02762 {
02763 if (p[3] == 'd')
02764 i960_last_insn_type = I_TYPE_REG;
02765 else if (p[4] == 'b' || p[4] == 'j')
02766 i960_last_insn_type = I_TYPE_CTRL;
02767 else
02768 i960_last_insn_type = I_TYPE_REG;
02769 }
02770 else
02771 i960_last_insn_type = I_TYPE_REG;
02772 break;
02773
02774 case 'l':
02775 i960_last_insn_type = I_TYPE_MEM;
02776 break;
02777
02778 case 's':
02779 if (p[1] == 't')
02780 i960_last_insn_type = I_TYPE_MEM;
02781 else
02782 i960_last_insn_type = I_TYPE_REG;
02783 break;
02784 }
02785 }