Annotation of GNUtools/cc/config/pa/pa.c, revision 1.1.1.1

1.1       root        1: /* Subroutines for insn-output.c for HPPA.
                      2:    Copyright (C) 1992, 1993 Free Software Foundation, Inc.
                      3:    Contributed by Tim Moore ([email protected]), based on sparc.c
                      4: 
                      5: This file is part of GNU CC.
                      6: 
                      7: GNU CC is free software; you can redistribute it and/or modify
                      8: it under the terms of the GNU General Public License as published by
                      9: the Free Software Foundation; either version 2, or (at your option)
                     10: any later version.
                     11: 
                     12: GNU CC is distributed in the hope that it will be useful,
                     13: but WITHOUT ANY WARRANTY; without even the implied warranty of
                     14: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     15: GNU General Public License for more details.
                     16: 
                     17: You should have received a copy of the GNU General Public License
                     18: along with GNU CC; see the file COPYING.  If not, write to
                     19: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
                     20: 
                     21: #include <stdio.h>
                     22: #include "config.h"
                     23: #include "rtl.h"
                     24: #include "regs.h"
                     25: #include "hard-reg-set.h"
                     26: #include "real.h"
                     27: #include "insn-config.h"
                     28: #include "conditions.h"
                     29: #include "insn-flags.h"
                     30: #include "output.h"
                     31: #include "insn-attr.h"
                     32: #include "flags.h"
                     33: #include "tree.h"
                     34: #include "c-tree.h"
                     35: #include "expr.h"
                     36: #include "obstack.h"
                     37: #include "machopic.h"
                     38: /* Save the operands last given to a compare for use when we
                     39:    generate a scc or bcc insn.  */
                     40: 
                     41: rtx hppa_compare_op0, hppa_compare_op1;
                     42: enum cmp_type hppa_branch_type;
                     43: 
                     44: rtx hppa_save_pic_table_rtx;
                     45: 
                     46: /* Set by the FUNCTION_PROFILER macro. */
                     47: int hp_profile_labelno;
                     48: extern int profile_label_no;
                     49: 
                     50: /* Counts for the number of callee-saved general and floating point
                     51:    registers which were saved by the current function's prologue.  */
                     52: static int gr_saved, fr_saved;
                     53: 
                     54: static rtx find_addr_reg ();
                     55: 
                     56: /* Return non-zero only if OP is a register of mode MODE,
                     57:    or CONST0_RTX.  */
                     58: int
                     59: reg_or_0_operand (op, mode)
                     60:      rtx op;
                     61:      enum machine_mode mode;
                     62: {
                     63:   return (op == CONST0_RTX (mode) || register_operand (op, mode));
                     64: }
                     65: 
                     66: /* Return non-zero if OP is suitable for use in a call to a named
                     67:    function.
                     68: 
                     69:    (???) For 2.5 try to eliminate either call_operand_address or 
                     70:    function_label_operand, they perform very similar functions.  */
                     71: int
                     72: call_operand_address (op, mode)
                     73:      rtx op;
                     74:      enum machine_mode mode;
                     75: {
                     76:   return (CONSTANT_P (op) && ! TARGET_LONG_CALLS);
                     77: }
                     78: 
                     79: /* Return 1 if X contains a symbolic expression.  We know these 
                     80:    expressions will have one of a few well defined forms, so 
                     81:    we need only check those forms.  */
                     82: int
                     83: symbolic_expression_p (x)
                     84:      register rtx x;
                     85: {
                     86: 
                     87:   /* Strip off any HIGH. */ 
                     88:   if (GET_CODE (x) == HIGH)
                     89:     x = XEXP (x, 0);
                     90: 
                     91:   return (symbolic_operand (x, VOIDmode));
                     92: }
                     93: 
                     94: int
                     95: symbolic_operand (op, mode)
                     96:      register rtx op;
                     97:      enum machine_mode mode;
                     98: {
                     99:   switch (GET_CODE (op))
                    100:     {
                    101:     case SYMBOL_REF:
                    102:     case LABEL_REF:
                    103:       return 1;
                    104:     case CONST:
                    105:       op = XEXP (op, 0);
                    106:       return ((GET_CODE (XEXP (op, 0)) == SYMBOL_REF
                    107:               || GET_CODE (XEXP (op, 0)) == LABEL_REF)
                    108:              && GET_CODE (XEXP (op, 1)) == CONST_INT);
                    109:     default:
                    110:       return 0;
                    111:     }
                    112: }
                    113: 
                    114: /* Return truth value of statement that OP is a symbolic memory
                    115:    operand of mode MODE.  */
                    116: 
                    117: int
                    118: symbolic_memory_operand (op, mode)
                    119:      rtx op;
                    120:      enum machine_mode mode;
                    121: {
                    122:   if (GET_CODE (op) == SUBREG)
                    123:     op = SUBREG_REG (op);
                    124:   if (GET_CODE (op) != MEM)
                    125:     return 0;
                    126:   op = XEXP (op, 0);
                    127:   return (GET_CODE (op) == SYMBOL_REF || GET_CODE (op) == CONST
                    128:          || GET_CODE (op) == HIGH || GET_CODE (op) == LABEL_REF);
                    129: }
                    130: 
                    131: /* Return 1 if the operand is either a register or a memory operand that is
                    132:    not symbolic.  */
                    133: 
                    134: int
                    135: reg_or_nonsymb_mem_operand (op, mode)
                    136:     register rtx op;
                    137:     enum machine_mode mode;
                    138: {
                    139:   if (register_operand (op, mode))
                    140:     return 1;
                    141: 
                    142:   if (memory_operand (op, mode) && ! symbolic_memory_operand (op, mode))
                    143:     return 1;
                    144: 
                    145:   return 0;
                    146: }
                    147: 
                    148: /* Return 1 if the operand is either a register, zero,  or a memory operand 
                    149:    that is not symbolic.  */
                    150: 
                    151: int
                    152: reg_or_0_or_nonsymb_mem_operand (op, mode)
                    153:     register rtx op;
                    154:     enum machine_mode mode;
                    155: {
                    156:   if (register_operand (op, mode))
                    157:     return 1;
                    158: 
                    159:   if (op == CONST0_RTX (mode))
                    160:     return 1;
                    161: 
                    162:   if (memory_operand (op, mode) && ! symbolic_memory_operand (op, mode))
                    163:     return 1;
                    164: 
                    165:   return 0;
                    166: }
                    167: 
                    168: /* Accept any constant that can be moved in one instructions into a 
                    169:    general register.  */
                    170: int 
                    171: cint_ok_for_move (intval)
                    172:      int intval;
                    173: {
                    174:   /* OK if ldo, ldil, or zdepi, can be used.  */
                    175:   return (VAL_14_BITS_P (intval) || (intval & RIGHT_BITS_MASK) == 0
                    176:          || zdepi_cint_p (intval));
                    177: }
                    178: 
                    179: /* Accept anything that can be moved in one instruction into a general
                    180:    register.  */
                    181: int
                    182: move_operand (op, mode)
                    183:      rtx op;
                    184:      enum machine_mode mode;
                    185: {
                    186:   if (register_operand (op, mode))
                    187:     return 1;
                    188: 
                    189:   if (GET_CODE (op) == CONST_INT)
                    190:     return cint_ok_for_move (INTVAL (op));
                    191: 
                    192:   if (GET_MODE (op) != mode)
                    193:     return 0;
                    194:   if (GET_CODE (op) == SUBREG)
                    195:     op = SUBREG_REG (op);
                    196:   if (GET_CODE (op) != MEM)
                    197:     return 0;
                    198: 
                    199:   op = XEXP (op, 0);
                    200:   if (GET_CODE (op) == LO_SUM)
                    201:     return (register_operand (XEXP (op, 0), Pmode)
                    202:            && CONSTANT_P (XEXP (op, 1)));
                    203:   return memory_address_p (mode, op);
                    204: }
                    205: 
                    206: /* Accept REG and any CONST_INT that can be moved in one instruction into a
                    207:    general register.  */
                    208: int
                    209: reg_or_cint_move_operand (op, mode)
                    210:      rtx op;
                    211:      enum machine_mode mode;
                    212: {
                    213:   if (register_operand (op, mode))
                    214:     return 1;
                    215: 
                    216:   if (GET_CODE (op) == CONST_INT)
                    217:     return cint_ok_for_move (INTVAL (op));
                    218: 
                    219:   return 0;
                    220: }
                    221: 
                    222: int
                    223: pic_operand (op, mode)
                    224:      rtx op;
                    225:      enum machine_mode mode;
                    226: {
                    227:   return flag_pic && GET_CODE (op) == LABEL_REF;
                    228: }
                    229: 
                    230: int
                    231: fp_reg_operand (op, mode)
                    232:      rtx op;
                    233:      enum machine_mode mode;
                    234: {
                    235:   return reg_renumber && FP_REG_P (op);
                    236: }
                    237: 
                    238: 
                    239: extern int current_function_uses_pic_offset_table;
                    240: extern rtx force_reg (), validize_mem ();
                    241: 
                    242: /* The rtx for the global offset table which is a special form
                    243:    that *is* a position independent symbolic constant.  */
                    244: rtx pic_pc_rtx;
                    245: 
                    246: /* Ensure that we are not using patterns that are not OK with PIC.  */
                    247: 
                    248: int
                    249: check_pic (i)
                    250:      int i;
                    251: {
                    252:   extern rtx recog_operand[];
                    253:   switch (flag_pic)
                    254:     {
                    255:     case 1:
                    256:       if (GET_CODE (recog_operand[i]) == SYMBOL_REF
                    257:          || (GET_CODE (recog_operand[i]) == CONST
                    258:              && ! rtx_equal_p (pic_pc_rtx, recog_operand[i])))
                    259:        abort ();
                    260:     case 2:
                    261:     default:
                    262:       return 1;
                    263:     }
                    264: }
                    265: 
                    266: /* Return truth value of whether OP can be used as an operand in a
                    267:    three operand arithmetic insn that accepts registers of mode MODE
                    268:    or 14-bit signed integers.  */
                    269: int
                    270: arith_operand (op, mode)
                    271:      rtx op;
                    272:      enum machine_mode mode;
                    273: {
                    274:   return (register_operand (op, mode)
                    275:          || (GET_CODE (op) == CONST_INT && INT_14_BITS (op)));
                    276: }
                    277: 
                    278: /* Return truth value of whether OP can be used as an operand in a
                    279:    three operand arithmetic insn that accepts registers of mode MODE
                    280:    or 11-bit signed integers.  */
                    281: int
                    282: arith11_operand (op, mode)
                    283:      rtx op;
                    284:      enum machine_mode mode;
                    285: {
                    286:   return (register_operand (op, mode)
                    287:          || (GET_CODE (op) == CONST_INT && INT_11_BITS (op)));
                    288: }
                    289: 
                    290: /* A constant integer suitable for use in a PRE_MODIFY memory 
                    291:    reference.  */
                    292: int
                    293: pre_cint_operand (op, mode)
                    294:      rtx op;
                    295:      enum machine_mode mode;
                    296: {
                    297:   return (GET_CODE (op) == CONST_INT
                    298:          && INTVAL (op) >= -0x2000 && INTVAL (op) < 0x10);
                    299: }
                    300: 
                    301: /* A constant integer suitable for use in a POST_MODIFY memory 
                    302:    reference.  */
                    303: int
                    304: post_cint_operand (op, mode)
                    305:      rtx op;
                    306:      enum machine_mode mode;
                    307: {
                    308:   return (GET_CODE (op) == CONST_INT
                    309:          && INTVAL (op) < 0x2000 && INTVAL (op) >= -0x10);
                    310: }
                    311: 
                    312: int
                    313: arith_double_operand (op, mode)
                    314:      rtx op;
                    315:      enum machine_mode mode;
                    316: {
                    317:   return (register_operand (op, mode)
                    318:          || (GET_CODE (op) == CONST_DOUBLE
                    319:              && GET_MODE (op) == mode
                    320:              && VAL_14_BITS_P (CONST_DOUBLE_LOW (op))
                    321:              && (CONST_DOUBLE_HIGH (op) >= 0
                    322:                  == ((CONST_DOUBLE_LOW (op) & 0x1000) == 0))));
                    323: }
                    324: 
                    325: /* Return truth value of whether OP is a integer which fits the
                    326:    range constraining immediate operands in three-address insns.  */
                    327: 
                    328: int
                    329: int5_operand (op, mode)
                    330:      rtx op;
                    331:      enum machine_mode mode;
                    332: {
                    333:   return (GET_CODE (op) == CONST_INT && INT_5_BITS (op));
                    334: }
                    335: 
                    336: int
                    337: uint5_operand (op, mode)
                    338:      rtx op;
                    339:      enum machine_mode mode;
                    340: {
                    341:   return (GET_CODE (op) == CONST_INT && INT_U5_BITS (op));
                    342: }
                    343: 
                    344:   
                    345: int
                    346: int11_operand (op, mode)
                    347:      rtx op;
                    348:      enum machine_mode mode;
                    349: {
                    350:     return (GET_CODE (op) == CONST_INT && INT_11_BITS (op));
                    351: }
                    352: 
                    353: int
                    354: arith5_operand (op, mode)
                    355:      rtx op;
                    356:      enum machine_mode mode;
                    357: {
                    358:   return register_operand (op, mode) || int5_operand (op, mode);
                    359: }
                    360: 
                    361: /* True iff zdepi can be used to generate this CONST_INT.  */
                    362: int
                    363: zdepi_cint_p (x)
                    364:      unsigned x;
                    365: {
                    366:   unsigned lsb_mask, t;
                    367: 
                    368:   /* This might not be obvious, but it's at least fast.
                    369:      This function is critcal; we don't have the time loops would take.  */
                    370:   lsb_mask = x & -x;
                    371:   t = ((x >> 4) + lsb_mask) & ~(lsb_mask - 1);
                    372:   /* Return true iff t is a power of two.  */
                    373:   return ((t & (t - 1)) == 0);
                    374: }
                    375: 
                    376: /* True iff depi or extru can be used to compute (reg & mask).  */
                    377: int
                    378: and_mask_p (mask)
                    379:      unsigned mask;
                    380: {
                    381:   mask = ~mask;
                    382:   mask += mask & -mask;
                    383:   return (mask & (mask - 1)) == 0;
                    384: }
                    385: 
                    386: /* True iff depi or extru can be used to compute (reg & OP).  */
                    387: int
                    388: and_operand (op, mode)
                    389:      rtx op;
                    390:      enum machine_mode mode;
                    391: {
                    392:   return (register_operand (op, mode)
                    393:          || (GET_CODE (op) == CONST_INT && and_mask_p (INTVAL (op))));
                    394: }
                    395: 
                    396: /* True iff depi can be used to compute (reg | MASK).  */
                    397: int
                    398: ior_mask_p (mask)
                    399:      unsigned mask;
                    400: {
                    401:   mask += mask & -mask;
                    402:   return (mask & (mask - 1)) == 0;
                    403: }
                    404: 
                    405: /* True iff depi can be used to compute (reg | OP).  */
                    406: int
                    407: ior_operand (op, mode)
                    408:      rtx op;
                    409:      enum machine_mode mode;
                    410: {
                    411:   return (GET_CODE (op) == CONST_INT && ior_mask_p (INTVAL (op)));
                    412: }
                    413: 
                    414: int
                    415: lhs_lshift_operand (op, mode)
                    416:      rtx op;
                    417:      enum machine_mode mode;
                    418: {
                    419:   return register_operand (op, mode) || lhs_lshift_cint_operand (op, mode);
                    420: }
                    421: 
                    422: /* True iff OP is a CONST_INT of the forms 0...0xxxx or 0...01...1xxxx.
                    423:    Such values can be the left hand side x in (x << r), using the zvdepi
                    424:    instruction.  */
                    425: int
                    426: lhs_lshift_cint_operand (op, mode)
                    427:      rtx op;
                    428:      enum machine_mode mode;
                    429: {
                    430:   unsigned x;
                    431:   if (GET_CODE (op) != CONST_INT)
                    432:     return 0;
                    433:   x = INTVAL (op) >> 4;
                    434:   return (x & (x + 1)) == 0;
                    435: }
                    436: 
                    437: int
                    438: arith32_operand (op, mode)
                    439:      rtx op;
                    440:      enum machine_mode mode;
                    441: {
                    442:   return register_operand (op, mode) || GET_CODE (op) == CONST_INT;
                    443: }
                    444: 
                    445: int
                    446: pc_or_label_operand (op, mode)
                    447:      rtx op;
                    448:      enum machine_mode mode;
                    449: {
                    450:   return (GET_CODE (op) == PC || GET_CODE (op) == LABEL_REF);
                    451: }
                    452: 
                    453: int
                    454: code_label_operand (op, mode)
                    455:      rtx op;
                    456:      enum machine_mode mode;
                    457: {
                    458:   return (GET_CODE (op) == CODE_LABEL);
                    459: }
                    460: 
                    461: /* Legitimize PIC addresses.  If the address is already
                    462:    position-independent, we return ORIG.  Newly generated
                    463:    position-independent addresses go to REG.  If we need more
                    464:    than one register, we lose.  */
                    465: 
                    466: rtx
                    467: legitimize_pic_address (orig, mode, reg)
                    468:      rtx orig, reg;
                    469:      enum machine_mode mode;
                    470: {
                    471:   rtx pic_ref = orig;
                    472: 
                    473: #ifdef MACHO_PIC
                    474:   return machopic_legitimize_pic_address (orig, mode, reg);
                    475: #endif
                    476: 
                    477:   if (GET_CODE (orig) == SYMBOL_REF)
                    478:     {
                    479:       if (reg == 0)
                    480:        abort ();
                    481: 
                    482:       if (flag_pic == 2)
                    483:        {
                    484:          emit_insn (gen_rtx (SET, VOIDmode, reg,
                    485:                              gen_rtx (HIGH, Pmode, orig)));
                    486:          emit_insn (gen_rtx (SET, VOIDmode, reg,
                    487:                              gen_rtx (LO_SUM, Pmode, reg, orig)));
                    488:          orig = reg;
                    489:        }
                    490:       pic_ref = gen_rtx (MEM, Pmode,
                    491:                         gen_rtx (PLUS, Pmode,
                    492:                                  pic_offset_table_rtx, orig));
                    493:       current_function_uses_pic_offset_table = 1;
                    494:       RTX_UNCHANGING_P (pic_ref) = 1;
                    495:       emit_move_insn (reg, pic_ref);
                    496:       return reg;
                    497:     }
                    498:   else if (GET_CODE (orig) == CONST)
                    499:     {
                    500:       rtx base;
                    501: 
                    502:       if (GET_CODE (XEXP (orig, 0)) == PLUS
                    503:          && XEXP (XEXP (orig, 0), 0) == pic_offset_table_rtx)
                    504:        return orig;
                    505: 
                    506:       if (reg == 0)
                    507:        abort ();
                    508: 
                    509:       if (GET_CODE (XEXP (orig, 0)) == PLUS)
                    510:        {
                    511:          base = legitimize_pic_address (XEXP (XEXP (orig, 0), 0), Pmode, reg);
                    512:          orig = legitimize_pic_address (XEXP (XEXP (orig, 0), 1), Pmode,
                    513:                                         base == reg ? 0 : reg);
                    514:        }
                    515:       else abort ();
                    516:       if (GET_CODE (orig) == CONST_INT)
                    517:        {
                    518:          if (INT_14_BITS (orig))
                    519:            return plus_constant_for_output (base, INTVAL (orig));
                    520:          orig = force_reg (Pmode, orig);
                    521:        }
                    522:       pic_ref = gen_rtx (PLUS, Pmode, base, orig);
                    523:       /* Likewise, should we set special REG_NOTEs here?  */
                    524:     }
                    525:   return pic_ref;
                    526: }
                    527: 
                    528: /* Set up PIC-specific rtl.  This should not cause any insns
                    529:    to be emitted.  */
                    530: 
                    531: void
                    532: initialize_pic ()
                    533: {
                    534: }
                    535: 
                    536: /* Emit special PIC prologues and epilogues.  */
                    537: 
                    538: void
                    539: finalize_pic ()
                    540: {
                    541:   rtx insn = get_insns ();
                    542: 
                    543:   while (GET_CODE (insn) == NOTE)
                    544:     if (NOTE_LINE_NUMBER (insn) == NOTE_INSN_FUNCTION_BEG)
                    545:       break;
                    546:     else
                    547:       insn = NEXT_INSN (insn);
                    548:       
                    549: 
                    550: #ifdef NeXT_ASM
                    551:   if (current_function_uses_pic_offset_table)
                    552:     {
                    553:       rtx label = gen_label_rtx ();
                    554:       rtx target = gen_rtx (REG, Pmode, PIC_OFFSET_TABLE_REGNUM);
                    555:       XSTR (label, 4) = machopic_function_base_name ();
                    556:       insn = emit_jump_insn_after (gen_branch_and_link (label, target), insn);
                    557:       insn = emit_label_after (label, insn);
                    558:       LABEL_PRESERVE_P (label) = 1;
                    559:       insn = emit_insn_after (gen_andsi3 (target, target, 
                    560:                                          gen_rtx (CONST_INT, SImode, ~3UL)),
                    561:                              insn);
                    562:     }
                    563: #endif
                    564: 
                    565:   if (hppa_save_pic_table_rtx)
                    566:     {
                    567:       emit_insn_after (gen_rtx (SET, VOIDmode,
                    568:                                hppa_save_pic_table_rtx,
                    569:                                gen_rtx (REG, Pmode, PIC_OFFSET_TABLE_REGNUM)),
                    570:                       insn);
                    571:       /* Need to emit this whether or not we obey regdecls,
                    572:         since setjmp/longjmp can cause life info to screw up.  */
                    573:       hppa_save_pic_table_rtx = 0;
                    574:     }
                    575:   
                    576:   emit_insn (gen_rtx (USE, VOIDmode, pic_offset_table_rtx));
                    577: }
                    578: 
                    579: /* Try machine-dependent ways of modifying an illegitimate address
                    580:    to be legitimate.  If we find one, return the new, valid address.
                    581:    This macro is used in only one place: `memory_address' in explow.c.
                    582: 
                    583:    OLDX is the address as it was before break_out_memory_refs was called.
                    584:    In some cases it is useful to look at this to decide what needs to be done.
                    585: 
                    586:    MODE and WIN are passed so that this macro can use
                    587:    GO_IF_LEGITIMATE_ADDRESS.
                    588: 
                    589:    It is always safe for this macro to do nothing.  It exists to recognize
                    590:    opportunities to optimize the output. 
                    591: 
                    592:    For the PA, transform:
                    593: 
                    594:        memory(X + <large int>)
                    595: 
                    596:    into:
                    597: 
                    598:        if (<large int> & mask) >= 16
                    599:          Y = (<large int> & ~mask) + mask + 1  Round up.
                    600:        else
                    601:          Y = (<large int> & ~mask)             Round down.
                    602:        Z = X + Y
                    603:        memory (Z + (<large int> - Y));
                    604: 
                    605:    This is for CSE to find several similar references, and only use one Z. 
                    606: 
                    607:    X can either be a SYMBOL_REF or REG, but because combine can not
                    608:    perform a 4->2 combination we do nothing for SYMBOL_REF + D where
                    609:    D will not fit in 14 bits.
                    610: 
                    611:    MODE_FLOAT references allow displacements which fit in 5 bits, so use
                    612:    0x1f as the mask.  
                    613: 
                    614:    MODE_INT references allow displacements which fit in 14 bits, so use
                    615:    0x3fff as the mask. 
                    616: 
                    617:    This relies on the fact that most mode MODE_FLOAT references will use FP
                    618:    registers and most mode MODE_INT references will use integer registers.
                    619:    (In the rare case of an FP register used in an integer MODE, we depend
                    620:    on secondary reloads to clean things up.)
                    621: 
                    622: 
                    623:    It is also beneficial to handle (plus (mult (X) (Y)) (Z)) in a special
                    624:    manner if Y is 2, 4, or 8.  (allows more shadd insns and shifted indexed
                    625:    adressing modes to be used).
                    626: 
                    627:    Put X and Z into registers.  Then put the entire expression into
                    628:    a register.  */
                    629: 
                    630: rtx
                    631: hppa_legitimize_address (x, oldx, mode)
                    632:      rtx x, oldx;
                    633:      enum machine_mode mode;
                    634: {
                    635:   
                    636:   rtx orig = x;
                    637: 
                    638:   /* Strip off CONST. */
                    639:   if (GET_CODE (x) == CONST)
                    640:     x = XEXP (x, 0);
                    641: 
                    642:   if (GET_CODE (x) == PLUS
                    643:       && GET_CODE (XEXP (x, 1)) == CONST_INT
                    644:       && (GET_CODE (XEXP (x, 0)) == SYMBOL_REF
                    645:          || GET_CODE (XEXP (x, 0)) == REG))
                    646:     {
                    647:       rtx int_part, ptr_reg;
                    648:       int newoffset;
                    649:       int offset = INTVAL (XEXP (x, 1));
                    650:       int mask = GET_MODE_CLASS (mode) == MODE_FLOAT ? 0x1f : 0x3fff;
                    651: 
                    652:       /* Choose which way to round the offset.  Round up if we 
                    653:         are >= halfway to the next boundary.  */
                    654:       if ((offset & mask) >= ((mask + 1) / 2))
                    655:        newoffset = (offset & ~ mask) + mask + 1;
                    656:       else
                    657:        newoffset = (offset & ~ mask);
                    658: 
                    659:       /* If the newoffset will not fit in 14 bits (ldo), then
                    660:         handling this would take 4 or 5 instructions (2 to load
                    661:         the SYMBOL_REF + 1 or 2 to load the newoffset + 1 to
                    662:         add the new offset and the SYMBOL_REF.)  Combine can
                    663:         not handle 4->2 or 5->2 combinations, so do not create
                    664:         them.  */
                    665:       if (! VAL_14_BITS_P (newoffset)
                    666:          && GET_CODE (XEXP (x, 0)) == SYMBOL_REF)
                    667:        {
                    668:          rtx const_part = gen_rtx (CONST, VOIDmode,
                    669:                                    gen_rtx (PLUS, Pmode,
                    670:                                             XEXP (x, 0),
                    671:                                             GEN_INT (newoffset)));
                    672:          rtx tmp_reg
                    673:            = force_reg (Pmode,
                    674:                         gen_rtx (HIGH, Pmode, const_part));
                    675:          ptr_reg
                    676:            = force_reg (Pmode,
                    677:                         gen_rtx (LO_SUM, Pmode,
                    678:                                  tmp_reg, const_part));
                    679:        }
                    680:       else
                    681:        {
                    682:          if (! VAL_14_BITS_P (newoffset))
                    683:            int_part = force_reg (Pmode, GEN_INT (newoffset));
                    684:          else
                    685:            int_part = GEN_INT (newoffset);
                    686: 
                    687:          ptr_reg = force_reg (Pmode,
                    688:                               gen_rtx (PLUS, Pmode,
                    689:                                        force_reg (Pmode, XEXP (x, 0)),
                    690:                                        int_part));
                    691:        }
                    692:       return plus_constant (ptr_reg, offset - newoffset);
                    693:     }
                    694: 
                    695:   /* Try to arrange things so that indexing modes can be used, but
                    696:      only do so if indexing is safe.  
                    697: 
                    698:      Indexing is safe when the second operand for the outer PLUS
                    699:      is a REG, SUBREG, SYMBOL_REF or the like.  
                    700: 
                    701:      For 2.5, indexing is also safe for (plus (symbol_ref) (const_int)) 
                    702:      if the integer is > 0.  */
                    703:   if (GET_CODE (x) == PLUS && GET_CODE (XEXP (x, 0)) == MULT
                    704:       && GET_CODE (XEXP (XEXP (x, 0), 1)) == CONST_INT
                    705:       && shadd_constant_p (INTVAL (XEXP (XEXP (x, 0), 1)))
                    706:       && (GET_RTX_CLASS (GET_CODE (XEXP (x, 1))) == 'o'
                    707:          || GET_CODE (XEXP (x, 1)) == SUBREG)
                    708:       && GET_CODE (XEXP (x, 1)) != CONST)
                    709:     {
                    710:       int val = INTVAL (XEXP (XEXP (x, 0), 1));
                    711:       rtx reg1, reg2;
                    712:       reg1 = force_reg (Pmode, force_operand (XEXP (x, 1), 0));
                    713:       reg2 = force_reg (Pmode,
                    714:                        force_operand (XEXP (XEXP (x, 0), 0), 0));
                    715:       return force_reg (Pmode,
                    716:                        gen_rtx (PLUS, Pmode,
                    717:                                 gen_rtx (MULT, Pmode, reg2,
                    718:                                          GEN_INT (val)),
                    719:                                 reg1));
                    720:     }
                    721: 
                    722:   /* Uh-oh.  We might have an address for x[n-100000].  This needs 
                    723:      special handling.  */
                    724: 
                    725:   if (GET_CODE (x) == PLUS && GET_CODE (XEXP (x, 0)) == MULT
                    726:       && GET_CODE (XEXP (XEXP (x, 0), 1)) == CONST_INT
                    727:       && shadd_constant_p (INTVAL (XEXP (XEXP (x, 0), 1))))
                    728:     {
                    729:       /* Ugly.  We modify things here so that the address offset specified
                    730:         by the index expression is computed first, then added to x to form
                    731:         the entire address.
                    732: 
                    733:         For 2.5, it might be profitable to set things up so that we
                    734:         compute the raw (unscaled) index first, then use scaled indexing
                    735:         to access memory, or better yet have the MI parts of the compiler
                    736:         handle this.  */
                    737: 
                    738:       rtx regx1, regy1, regy2, y;
                    739: 
                    740:       /* Strip off any CONST.  */
                    741:       y = XEXP (x, 1);
                    742:       if (GET_CODE (y) == CONST)
                    743:        y = XEXP (y, 0);
                    744: 
                    745:       if (GET_CODE (y) == PLUS || GET_CODE (y) == MINUS)
                    746:        {
                    747:          regx1 = force_reg (Pmode, force_operand (XEXP (x, 0), 0));
                    748:          regy1 = force_reg (Pmode, force_operand (XEXP (y, 0), 0));
                    749:          regy2 = force_reg (Pmode, force_operand (XEXP (y, 1), 0));
                    750:          regx1 = force_reg (Pmode, gen_rtx (GET_CODE (y), Pmode, regx1, regy2));
                    751:          return force_reg (Pmode, gen_rtx (PLUS, Pmode, regx1, regy1));
                    752:        }
                    753:     }
                    754: 
                    755:   if (flag_pic) 
                    756:     return legitimize_pic_address (x, mode, gen_reg_rtx (Pmode));
                    757: 
                    758:   return orig;
                    759: }
                    760: 
                    761: /* For the HPPA, REG and REG+CONST is cost 0
                    762:    and addresses involving symbolic constants are cost 2.
                    763: 
                    764:    PIC addresses are very expensive.
                    765: 
                    766:    It is no coincidence that this has the same structure
                    767:    as GO_IF_LEGITIMATE_ADDRESS.  */
                    768: int
                    769: hppa_address_cost (X)
                    770:      rtx X;
                    771: {
                    772:   if (GET_CODE (X) == PLUS)
                    773:       return 1;
                    774:   else if (GET_CODE (X) == LO_SUM)
                    775:     return 1;
                    776:   else if (GET_CODE (X) == HIGH)
                    777:     return 2;
                    778:   return 4;
                    779: }
                    780: 
                    781: /* Emit insns to move operands[1] into operands[0].
                    782: 
                    783:    Return 1 if we have written out everything that needs to be done to
                    784:    do the move.  Otherwise, return 0 and the caller will emit the move
                    785:    normally.  */
                    786: 
                    787: int
                    788: emit_move_sequence (operands, mode, scratch_reg)
                    789:      rtx *operands;
                    790:      enum machine_mode mode;
                    791:      rtx scratch_reg;
                    792: {
                    793:   register rtx operand0 = operands[0];
                    794:   register rtx operand1 = operands[1];
                    795: 
                    796:   /* Handle secondary reloads for loads/stores of FP registers from
                    797:      REG+D addresses where D does not fit in 5 bits.  */
                    798:   if (fp_reg_operand (operand0, mode)
                    799:       && GET_CODE (operand1) == MEM
                    800:       /* Using DFmode forces only short displacements be be
                    801:         recognized as valid in reg+d addressing modes.  */
                    802:       && ! memory_address_p (DFmode, XEXP (operand1, 0))
                    803:       && scratch_reg)
                    804:     {
                    805:       emit_move_insn (scratch_reg, XEXP (operand1 , 0));
                    806:       emit_insn (gen_rtx (SET, VOIDmode, operand0, gen_rtx (MEM, mode,
                    807:                                                            scratch_reg)));
                    808:       return 1;
                    809:     }
                    810:   else if (fp_reg_operand (operand1, mode)
                    811:           && GET_CODE (operand0) == MEM
                    812:           /* Using DFmode forces only short displacements be be
                    813:              recognized as valid in reg+d addressing modes.  */
                    814:           && ! memory_address_p (DFmode, XEXP (operand0, 0))
                    815:           && scratch_reg)
                    816:     {
                    817:       emit_move_insn (scratch_reg, XEXP (operand0 , 0));
                    818:       emit_insn (gen_rtx (SET, VOIDmode, gen_rtx (MEM, mode,  scratch_reg),
                    819:                          operand1));
                    820:       return 1;
                    821:     }
                    822:   /* Handle secondary reloads for loads of FP registers from constant
                    823:      expressions by forcing the constant into memory.
                    824: 
                    825:      use scratch_reg to hold the address of the memory location. 
                    826: 
                    827:      ??? The proper fix is to change PREFERRED_RELOAD_CLASS to return 
                    828:      NO_REGS when presented with a const_int and an register class 
                    829:      containing only FP registers.  Doing so unfortunately creates
                    830:      more problems than it solves.   Fix this for 2.5.  */
                    831:   else if (fp_reg_operand (operand0, mode)
                    832:           && CONSTANT_P (operand1)
                    833:           && scratch_reg)
                    834:     {
                    835:       rtx xoperands[2];
                    836: 
                    837:       /* Force the constant into memory and put the address of the
                    838:         memory location into scratch_reg.  */
                    839:       xoperands[0] = scratch_reg;
                    840:       xoperands[1] = XEXP (force_const_mem (mode, operand1), 0);
                    841:       emit_move_sequence (xoperands, Pmode, 0);
                    842: 
                    843:       /* Now load the destination register.  */
                    844:       emit_insn (gen_rtx (SET, mode, operand0,
                    845:                          gen_rtx (MEM, mode, scratch_reg)));
                    846:       return 1;
                    847:     }
                    848:   /* Handle secondary reloads for SAR.  These occur when trying to load
                    849:      the SAR from memory or from a FP register.  */
                    850:   else if (GET_CODE (operand0) == REG
                    851:           && REGNO_REG_CLASS (REGNO (operand0)) == SHIFT_REGS
                    852:           && (GET_CODE (operand1) == MEM
                    853:               || (GET_CODE (operand1) == REG
                    854:                   && FP_REG_CLASS_P (REGNO_REG_CLASS (REGNO (operand1)))))
                    855:           && scratch_reg)
                    856:     {
                    857:       emit_move_insn (scratch_reg, operand1);
                    858:       emit_move_insn (operand0, scratch_reg);
                    859:       return 1;
                    860:     }
                    861:   /* Handle most common case: storing into a register.  */
                    862:   else if (register_operand (operand0, mode))
                    863:     {
                    864:       if (register_operand (operand1, mode)
                    865:          || (GET_CODE (operand1) == CONST_INT && INT_14_BITS (operand1))
                    866:          || (operand1 == CONST0_RTX (mode))
                    867:          || (GET_CODE (operand1) == HIGH
                    868:              && !symbolic_operand (XEXP (operand1, 0)))
                    869:          /* Only `general_operands' can come here, so MEM is ok.  */
                    870:          || GET_CODE (operand1) == MEM)
                    871:        {
                    872:          /* Run this case quickly.  */
                    873:          emit_insn (gen_rtx (SET, VOIDmode, operand0, operand1));
                    874:          return 1;
                    875:        }
                    876:     }
                    877:   else if (GET_CODE (operand0) == MEM)
                    878:     {
                    879:       if (register_operand (operand1, mode) || operand1 == CONST0_RTX (mode))
                    880:        {
                    881:          /* Run this case quickly.  */
                    882:          emit_insn (gen_rtx (SET, VOIDmode, operand0, operand1));
                    883:          return 1;
                    884:        }
                    885:       if (! (reload_in_progress || reload_completed))
                    886:        {
                    887:          operands[0] = validize_mem (operand0);
                    888:          operands[1] = operand1 = force_reg (mode, operand1);
                    889:        }
                    890:     }
                    891: 
                    892:   /* Simplify the source if we need to.  */
                    893:   if ((GET_CODE (operand1) != HIGH && immediate_operand (operand1, mode))
                    894:       || (GET_CODE (operand1) == HIGH
                    895:          && symbolic_operand (XEXP (operand1, 0), mode)))
                    896:     {
                    897:       int ishighonly = 0;
                    898: 
                    899:       if (GET_CODE (operand1) == HIGH)
                    900:        {
                    901:          ishighonly = 1;
                    902:          operand1 = XEXP (operand1, 0);
                    903:        }
                    904:       if (symbolic_operand (operand1, mode))
                    905:        {
                    906:          if (flag_pic)
                    907:            {
                    908:              rtx temp;
                    909: 
                    910:              if (reload_in_progress || reload_completed)
                    911:                temp = operand0;
                    912:              else
                    913:                temp = gen_reg_rtx (Pmode);
                    914:                
                    915:              operands[1] = legitimize_pic_address (operand1, mode, temp);
                    916:               emit_insn (gen_rtx (SET, VOIDmode, operand0, operands[1]));
                    917:            }
                    918:          /* On the HPPA, references to data space are supposed to */
                    919:          /* use dp, register 27, but showing it in the RTL inhibits various
                    920:             cse and loop optimizations.  */
                    921:          else 
                    922:            {
                    923:              rtx temp, set;
                    924: 
                    925:              if (reload_in_progress || reload_completed) 
                    926:                temp = scratch_reg ? scratch_reg : operand0;
                    927:              else
                    928:                temp = gen_reg_rtx (mode);
                    929: 
                    930:              if (ishighonly)
                    931:                set = gen_rtx (SET, mode, operand0, temp);
                    932:              else
                    933:                set = gen_rtx (SET, VOIDmode,
                    934:                               operand0,
                    935:                               gen_rtx (LO_SUM, mode, temp, operand1));
                    936:                                 
                    937:              emit_insn (gen_rtx (SET, VOIDmode,
                    938:                                  temp,
                    939:                                  gen_rtx (HIGH, mode, operand1)));
                    940:              if (function_label_operand (operand1, mode))
                    941:                {
                    942:                  rtx temp;
                    943: 
                    944:                  if (reload_in_progress || reload_completed)
                    945:                    temp = scratch_reg;
                    946:                  else
                    947:                    temp = gen_reg_rtx (mode);
                    948: 
                    949:                  if (!temp)
                    950:                    abort ();
                    951:                  emit_insn (gen_rtx (PARALLEL, VOIDmode,
                    952:                                      gen_rtvec (2,
                    953:                                                 set,
                    954:                                                 gen_rtx (CLOBBER, VOIDmode,
                    955:                                                          temp))));
                    956:                }
                    957:              else
                    958:                emit_insn (set);
                    959:              return 1;
                    960:            }
                    961:          return 1;
                    962:        }
                    963:       else if (GET_CODE (operand1) != CONST_INT
                    964:               || ! cint_ok_for_move (INTVAL (operand1)))
                    965:        {
                    966:          rtx temp;
                    967: 
                    968:          if (reload_in_progress || reload_completed)
                    969:            temp = operand0;
                    970:          else
                    971:            temp = gen_reg_rtx (mode);
                    972: 
                    973:          emit_insn (gen_rtx (SET, VOIDmode, temp,
                    974:                              gen_rtx (HIGH, mode, operand1)));
                    975:          operands[1] = gen_rtx (LO_SUM, mode, temp, operand1);
                    976:        }
                    977:     }
                    978:   /* Now have insn-emit do whatever it normally does.  */
                    979:   return 0;
                    980: }
                    981: 
                    982: /* Does operand (which is a symbolic_operand) live in text space? If
                    983:    so SYMBOL_REF_FLAG, which is set by ENCODE_SECTION_INFO, will be true.  */
                    984: 
                    985: int
                    986: read_only_operand (operand)
                    987:      rtx operand;
                    988: {
                    989:   if (GET_CODE (operand) == CONST)
                    990:     operand = XEXP (XEXP (operand, 0), 0);
                    991:   if (GET_CODE (operand) == SYMBOL_REF)
                    992:     return SYMBOL_REF_FLAG (operand) || CONSTANT_POOL_ADDRESS_P (operand);
                    993:   return 1;
                    994: }
                    995:      
                    996: 
                    997: /* Return the best assembler insn template
                    998:    for moving operands[1] into operands[0] as a fullword.   */
                    999: char *
                   1000: singlemove_string (operands)
                   1001:      rtx *operands;
                   1002: {
                   1003:   if (GET_CODE (operands[0]) == MEM)
                   1004:     return "stw %r1,%0";
                   1005:   else if (GET_CODE (operands[1]) == MEM)
                   1006:     return "ldw %1,%0";
                   1007:   else if (GET_CODE (operands[1]) == CONST_DOUBLE
                   1008:           && GET_MODE (operands[1]) == SFmode)
                   1009:     {
                   1010:       int i;
                   1011:       union real_extract u;
                   1012:       union float_extract { float f; int i; } v;
                   1013: 
                   1014:       bcopy (&CONST_DOUBLE_LOW (operands[1]), &u, sizeof u);
                   1015:       v.f = REAL_VALUE_TRUNCATE (SFmode, u.d);
                   1016:       i = v.i;
                   1017: 
                   1018:       operands[1] = gen_rtx (CONST_INT, VOIDmode, i);
                   1019: 
                   1020:       /* See if we can handle this constant in a single instruction.  */
                   1021:       if (cint_ok_for_move (INTVAL (operands[1])))
                   1022:        {
                   1023:           int intval = INTVAL (operands[1]);
                   1024: 
                   1025:           if (intval == 0)
                   1026:             return "copy 0,%0";
                   1027:           else if (VAL_14_BITS_P (intval))
                   1028:             return "ldi %1,%0";
                   1029:           else if ((intval & RIGHT_BITS_MASK) == 0)
                   1030:             return "ldil L%'%1,%0";
                   1031:           else if (zdepi_cint_p (intval))
                   1032:             return "zdepi %Z1,%0";
                   1033:        }
                   1034:       else
                   1035:        return "ldil L%'%1,%0\n\tldo R%'%1(%0),%0";
                   1036:     }
                   1037: 
                   1038:   else if (GET_CODE (operands[1]) == CONST_INT)
                   1039:     {
                   1040:       /* See if we can handle this in a single instruction.  */
                   1041:       if (cint_ok_for_move (INTVAL (operands[1])))
                   1042:        {
                   1043:           int intval = INTVAL (operands[1]);
                   1044: 
                   1045:           if (intval == 0)
                   1046:             return "copy 0,%0";
                   1047:           else if (VAL_14_BITS_P (intval))
                   1048:             return "ldi %1,%0";
                   1049:           else if ((intval & RIGHT_BITS_MASK) == 0)
                   1050:             return "ldil L%'%1,%0";
                   1051:           else if (zdepi_cint_p (intval))
                   1052:             return "zdepi %Z1,%0";
                   1053:        }
                   1054:       else
                   1055:        return "ldil L%'%1,%0\n\tldo R%'%1(%0),%0";
                   1056:     }
                   1057:   return "copy %1,%0";
                   1058: }
                   1059: 
                   1060: 
                   1061: /* Compute position (in OP[1]) and width (in OP[2])
                   1062:    useful for copying IMM to a register using the zdepi
                   1063:    instructions.  Store the immediate value to insert in OP[0].  */
                   1064: void
                   1065: compute_zdepi_operands (imm, op)
                   1066:      unsigned imm;
                   1067:      unsigned *op;
                   1068: {
                   1069:   int lsb, len;
                   1070: 
                   1071:   /* Find the least significant set bit in IMM.  */
                   1072:   for (lsb = 0; lsb < 32; lsb++)
                   1073:     {
                   1074:       if ((imm & 1) != 0)
                   1075:         break;
                   1076:       imm >>= 1;
                   1077:     }
                   1078: 
                   1079:   /* Choose variants based on *sign* of the 5-bit field.  */
                   1080:   if ((imm & 0x10) == 0)
                   1081:     len = (lsb <= 28) ? 4 : 32 - lsb;
                   1082:   else
                   1083:     {
                   1084:       /* Find the width of the bitstring in IMM.  */
                   1085:       for (len = 5; len < 32; len++)
                   1086:        {
                   1087:          if ((imm & (1 << len)) == 0)
                   1088:            break;
                   1089:        }
                   1090: 
                   1091:       /* Sign extend IMM as a 5-bit value.  */
                   1092:       imm = (imm & 0xf) - 0x10;
                   1093:     }
                   1094: 
                   1095:   op[0] = imm;
                   1096:   op[1] = 31 - lsb;
                   1097:   op[2] = len;
                   1098: }
                   1099: 
                   1100: /* Output assembler code to perform a doubleword move insn
                   1101:    with operands OPERANDS.  */
                   1102: 
                   1103: char *
                   1104: output_move_double (operands)
                   1105:      rtx *operands;
                   1106: {
                   1107:   enum { REGOP, OFFSOP, MEMOP, CNSTOP, RNDOP } optype0, optype1;
                   1108:   rtx latehalf[2];
                   1109:   rtx addreg0 = 0, addreg1 = 0;
                   1110: 
                   1111:   /* First classify both operands.  */
                   1112: 
                   1113:   if (REG_P (operands[0]))
                   1114:     optype0 = REGOP;
                   1115:   else if (offsettable_memref_p (operands[0]))
                   1116:     optype0 = OFFSOP;
                   1117:   else if (GET_CODE (operands[0]) == MEM)
                   1118:     optype0 = MEMOP;
                   1119:   else
                   1120:     optype0 = RNDOP;
                   1121: 
                   1122:   if (REG_P (operands[1]))
                   1123:     optype1 = REGOP;
                   1124:   else if (CONSTANT_P (operands[1]))
                   1125:     optype1 = CNSTOP;
                   1126:   else if (offsettable_memref_p (operands[1]))
                   1127:     optype1 = OFFSOP;
                   1128:   else if (GET_CODE (operands[1]) == MEM)
                   1129:     optype1 = MEMOP;
                   1130:   else
                   1131:     optype1 = RNDOP;
                   1132: 
                   1133:   /* Check for the cases that the operand constraints are not
                   1134:      supposed to allow to happen.  Abort if we get one,
                   1135:      because generating code for these cases is painful.  */
                   1136: 
                   1137:   if (optype0 != REGOP && optype1 != REGOP)
                   1138:     abort ();
                   1139: 
                   1140:    /* Handle auto decrementing and incrementing loads and stores
                   1141:      specifically, since the structure of the function doesn't work
                   1142:      for them without major modification.  Do it better when we learn
                   1143:      this port about the general inc/dec addressing of PA.
                   1144:      (This was written by tege.  Chide him if it doesn't work.)  */
                   1145: 
                   1146:   if (optype0 == MEMOP)
                   1147:     {
                   1148:       /* We have to output the address syntax ourselves, since print_operand
                   1149:         doesn't deal with the addresses we want to use.  Fix this later.  */
                   1150: 
                   1151:       rtx addr = XEXP (operands[0], 0);
                   1152:       if (GET_CODE (addr) == POST_INC || GET_CODE (addr) == POST_DEC)
                   1153:        {
                   1154:          rtx high_reg = gen_rtx (SUBREG, SImode, operands[1], 0);
                   1155: 
                   1156:          operands[0] = XEXP (addr, 0);
                   1157:          if (GET_CODE (operands[1]) != REG || GET_CODE (operands[0]) != REG)
                   1158:            abort ();
                   1159: 
                   1160:          if (!reg_overlap_mentioned_p (high_reg, addr))
                   1161:            {
                   1162:              /* No overlap between high target register and address
                   1163:                 register.  (We do this in a non-obvious way to
                   1164:                 save a register file writeback)  */
                   1165:              if (GET_CODE (addr) == POST_INC)
                   1166:                return "stws,ma %1,8(0,%0)\n\tstw %R1,-4(0,%0)";
                   1167:              return "stws,ma %1,-8(0,%0)\n\tstw %R1,12(0,%0)";
                   1168:            }
                   1169:          else
                   1170:            abort();
                   1171:        }
                   1172:       else if (GET_CODE (addr) == PRE_INC || GET_CODE (addr) == PRE_DEC)
                   1173:        {
                   1174:          rtx high_reg = gen_rtx (SUBREG, SImode, operands[1], 0);
                   1175: 
                   1176:          operands[0] = XEXP (addr, 0);
                   1177:          if (GET_CODE (operands[1]) != REG || GET_CODE (operands[0]) != REG)
                   1178:            abort ();
                   1179: 
                   1180:          if (!reg_overlap_mentioned_p (high_reg, addr))
                   1181:            {
                   1182:              /* No overlap between high target register and address
                   1183:                 register.  (We do this in a non-obvious way to
                   1184:                 save a register file writeback)  */
                   1185:              if (GET_CODE (addr) == PRE_INC)
                   1186:                return "stws,mb %1,8(0,%0)\n\tstw %R1,4(0,%0)";
                   1187:              return "stws,mb %1,-8(0,%0)\n\tstw %R1,4(0,%0)";
                   1188:            }
                   1189:          else
                   1190:            abort();
                   1191:        }
                   1192:     }
                   1193:   if (optype1 == MEMOP)
                   1194:     {
                   1195:       /* We have to output the address syntax ourselves, since print_operand
                   1196:         doesn't deal with the addresses we want to use.  Fix this later.  */
                   1197: 
                   1198:       rtx addr = XEXP (operands[1], 0);
                   1199:       if (GET_CODE (addr) == POST_INC || GET_CODE (addr) == POST_DEC)
                   1200:        {
                   1201:          rtx high_reg = gen_rtx (SUBREG, SImode, operands[0], 0);
                   1202: 
                   1203:          operands[1] = XEXP (addr, 0);
                   1204:          if (GET_CODE (operands[0]) != REG || GET_CODE (operands[1]) != REG)
                   1205:            abort ();
                   1206: 
                   1207:          if (!reg_overlap_mentioned_p (high_reg, addr))
                   1208:            {
                   1209:              /* No overlap between high target register and address
                   1210:                 register.  (We do this in a non-obvious way to
                   1211:                 save a register file writeback)  */
                   1212:              if (GET_CODE (addr) == POST_INC)
                   1213:                return "ldws,ma 8(0,%1),%0\n\tldw -4(0,%1),%R0";
                   1214:              return "ldws,ma -8(0,%1),%0\n\tldw 12(0,%1),%R0";
                   1215:            }
                   1216:          else
                   1217:            {
                   1218:              /* This is an undefined situation.  We should load into the
                   1219:                 address register *and* update that register.  Probably
                   1220:                 we don't need to handle this at all.  */
                   1221:              if (GET_CODE (addr) == POST_INC)
                   1222:                return "ldw 4(0,%1),%R0\n\tldws,ma 8(0,%1),%0";
                   1223:              return "ldw 4(0,%1),%R0\n\tldws,ma -8(0,%1),%0";
                   1224:            }
                   1225:        }
                   1226:       else if (GET_CODE (addr) == PRE_INC || GET_CODE (addr) == PRE_DEC)
                   1227:        {
                   1228:          rtx high_reg = gen_rtx (SUBREG, SImode, operands[0], 0);
                   1229: 
                   1230:          operands[1] = XEXP (addr, 0);
                   1231:          if (GET_CODE (operands[0]) != REG || GET_CODE (operands[1]) != REG)
                   1232:            abort ();
                   1233: 
                   1234:          if (!reg_overlap_mentioned_p (high_reg, addr))
                   1235:            {
                   1236:              /* No overlap between high target register and address
                   1237:                 register.  (We do this in a non-obvious way to
                   1238:                 save a register file writeback)  */
                   1239:              if (GET_CODE (addr) == PRE_INC)
                   1240:                return "ldws,mb 8(0,%1),%0\n\tldw 4(0,%1),%R0";
                   1241:              return "ldws,mb -8(0,%1),%0\n\tldw 4(0,%1),%R0";
                   1242:            }
                   1243:          else
                   1244:            {
                   1245:              /* This is an undefined situation.  We should load into the
                   1246:                 address register *and* update that register.  Probably
                   1247:                 we don't need to handle this at all.  */
                   1248:              if (GET_CODE (addr) == PRE_INC)
                   1249:                return "ldw 12(0,%1),%R0\n\tldws,mb 8(0,%1),%0";
                   1250:              return "ldw -4(0,%1),%R0\n\tldws,mb -8(0,%1),%0";
                   1251:            }
                   1252:        }
                   1253:     }
                   1254: 
                   1255:   /* If an operand is an unoffsettable memory ref, find a register
                   1256:      we can increment temporarily to make it refer to the second word.  */
                   1257: 
                   1258:   if (optype0 == MEMOP)
                   1259:     addreg0 = find_addr_reg (XEXP (operands[0], 0));
                   1260: 
                   1261:   if (optype1 == MEMOP)
                   1262:     addreg1 = find_addr_reg (XEXP (operands[1], 0));
                   1263: 
                   1264:   /* Ok, we can do one word at a time.
                   1265:      Normally we do the low-numbered word first.
                   1266: 
                   1267:      In either case, set up in LATEHALF the operands to use
                   1268:      for the high-numbered word and in some cases alter the
                   1269:      operands in OPERANDS to be suitable for the low-numbered word.  */
                   1270: 
                   1271:   if (optype0 == REGOP)
                   1272:     latehalf[0] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1);
                   1273:   else if (optype0 == OFFSOP)
                   1274:     latehalf[0] = adj_offsettable_operand (operands[0], 4);
                   1275:   else
                   1276:     latehalf[0] = operands[0];
                   1277: 
                   1278:   if (optype1 == REGOP)
                   1279:     latehalf[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1);
                   1280:   else if (optype1 == OFFSOP)
                   1281:     latehalf[1] = adj_offsettable_operand (operands[1], 4);
                   1282:   else if (optype1 == CNSTOP)
                   1283:     split_double (operands[1], &operands[1], &latehalf[1]);
                   1284:   else
                   1285:     latehalf[1] = operands[1];
                   1286: 
                   1287:   /* If the first move would clobber the source of the second one,
                   1288:      do them in the other order.
                   1289: 
                   1290:      RMS says "This happens only for registers;
                   1291:      such overlap can't happen in memory unless the user explicitly
                   1292:      sets it up, and that is an undefined circumstance."
                   1293: 
                   1294:      but it happens on the HP-PA when loading parameter registers,
                   1295:      so I am going to define that circumstance, and make it work
                   1296:      as expected.  */
                   1297: 
                   1298:   if (optype0 == REGOP && (optype1 == MEMOP || optype1 == OFFSOP)
                   1299:           && reg_overlap_mentioned_p (operands[0], XEXP (operands[1], 0)))
                   1300:     {
                   1301:       /* XXX THIS PROBABLY DOESN'T WORK.  */
                   1302:       /* Do the late half first.  */
                   1303:       if (addreg1)
                   1304:        output_asm_insn ("ldo 4(%0),%0", &addreg1);
                   1305:       output_asm_insn (singlemove_string (latehalf), latehalf);
                   1306:       if (addreg1)
                   1307:        output_asm_insn ("ldo -4(%0),%0", &addreg1);
                   1308:       /* Then clobber.  */
                   1309:       return singlemove_string (operands);
                   1310:     }
                   1311: 
                   1312:   if (optype0 == REGOP && optype1 == REGOP
                   1313:       && REGNO (operands[0]) == REGNO (operands[1]) + 1)
                   1314:     {
                   1315:       output_asm_insn (singlemove_string (latehalf), latehalf);
                   1316:       return singlemove_string (operands);
                   1317:     }
                   1318: 
                   1319:   /* Normal case: do the two words, low-numbered first.  */
                   1320: 
                   1321:   output_asm_insn (singlemove_string (operands), operands);
                   1322: 
                   1323:   /* Make any unoffsettable addresses point at high-numbered word.  */
                   1324:   if (addreg0)
                   1325:     output_asm_insn ("ldo 4(%0),%0", &addreg0);
                   1326:   if (addreg1)
                   1327:     output_asm_insn ("ldo 4(%0),%0", &addreg1);
                   1328: 
                   1329:   /* Do that word.  */
                   1330:   output_asm_insn (singlemove_string (latehalf), latehalf);
                   1331: 
                   1332:   /* Undo the adds we just did.  */
                   1333:   if (addreg0)
                   1334:     output_asm_insn ("ldo -4(%0),%0", &addreg0);
                   1335:   if (addreg1)
                   1336:     output_asm_insn ("ldo -4(%0),%0", &addreg1);
                   1337: 
                   1338:   return "";
                   1339: }
                   1340: 
                   1341: char *
                   1342: output_fp_move_double (operands)
                   1343:      rtx *operands;
                   1344: {
                   1345:   if (FP_REG_P (operands[0]))
                   1346:     {
                   1347:       if (FP_REG_P (operands[1]) 
                   1348:          || operands[1] == CONST0_RTX (GET_MODE (operands[0])))
                   1349:        output_asm_insn ("fcpy,dbl %r1,%0", operands);
                   1350:       else 
                   1351:        output_asm_insn ("fldds%F1 %1,%0", operands);
                   1352:     }
                   1353:   else if (FP_REG_P (operands[1]))
                   1354:     {
                   1355:       output_asm_insn ("fstds%F0 %1,%0", operands);
                   1356:     }
                   1357:   else if (operands[1] == CONST0_RTX (GET_MODE (operands[0])))
                   1358:     {
                   1359:       if (GET_CODE (operands[0]) == REG)
                   1360:        {
                   1361:          rtx xoperands[2];
                   1362:          xoperands[1] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1);
                   1363:          xoperands[0] = operands[0];
                   1364:          output_asm_insn ("copy %%r0,%0\n\tcopy %%r0,%1", xoperands);
                   1365:        }
                   1366:       /* This is a pain.  You have to be prepared to deal with an 
                   1367:         arbritary address here including pre/post increment/decrement.
                   1368: 
                   1369:         so avoid this in the MD.  */
                   1370:       else
                   1371:        abort ();
                   1372:     }
                   1373:   else abort ();
                   1374:   return "";
                   1375: }
                   1376: 
                   1377: /* Return a REG that occurs in ADDR with coefficient 1.
                   1378:    ADDR can be effectively incremented by incrementing REG.  */
                   1379: 
                   1380: static rtx
                   1381: find_addr_reg (addr)
                   1382:      rtx addr;
                   1383: {
                   1384:   while (GET_CODE (addr) == PLUS)
                   1385:     {
                   1386:       if (GET_CODE (XEXP (addr, 0)) == REG)
                   1387:        addr = XEXP (addr, 0);
                   1388:       else if (GET_CODE (XEXP (addr, 1)) == REG)
                   1389:        addr = XEXP (addr, 1);
                   1390:       else if (CONSTANT_P (XEXP (addr, 0)))
                   1391:        addr = XEXP (addr, 1);
                   1392:       else if (CONSTANT_P (XEXP (addr, 1)))
                   1393:        addr = XEXP (addr, 0);
                   1394:       else
                   1395:        abort ();
                   1396:     }
                   1397:   if (GET_CODE (addr) == REG)
                   1398:     return addr;
                   1399:   abort ();
                   1400: }
                   1401: 
                   1402: /* Emit code to perform a block move.
                   1403: 
                   1404:    Restriction: If the length argument is non-constant, alignment
                   1405:    must be 4.
                   1406: 
                   1407:    OPERANDS[0] is the destination pointer as a REG, clobbered.
                   1408:    OPERANDS[1] is the source pointer as a REG, clobbered.
                   1409:    if SIZE_IS_CONSTANT
                   1410:      OPERANDS[2] is a register for temporary storage.
                   1411:      OPERANDS[4] is the size as a CONST_INT
                   1412:    else
                   1413:      OPERANDS[2] is a REG which will contain the size, clobbered.
                   1414:    OPERANDS[3] is a register for temporary storage.
                   1415:    OPERANDS[5] is the alignment safe to use, as a CONST_INT.  */
                   1416: 
                   1417: char *
                   1418: output_block_move (operands, size_is_constant)
                   1419:      rtx *operands;
                   1420:      int size_is_constant;
                   1421: {
                   1422:   int align = INTVAL (operands[5]);
                   1423:   unsigned long n_bytes;
                   1424: 
                   1425:   /* We can't move more than four bytes at a time because the PA
                   1426:      has no longer integer move insns.  (Could use fp mem ops?)  */
                   1427:   if (align > 4)
                   1428:     align = 4;
                   1429: 
                   1430:   if (size_is_constant)
                   1431:     {
                   1432:       unsigned long offset;
                   1433:       rtx temp;
                   1434: 
                   1435:       n_bytes = INTVAL (operands[4]);
                   1436:       if (n_bytes == 0)
                   1437:        return "";
                   1438: 
                   1439:       if (align >= 4)
                   1440:        {
                   1441:          /* Don't unroll too large blocks.  */
                   1442:          if (n_bytes > 64)
                   1443:            goto copy_with_loop;
                   1444: 
                   1445:          /* Read and store using two registers, and hide latency
                   1446:             by deferring the stores until three instructions after
                   1447:             the corresponding load.  The last load insn will read
                   1448:             the entire word were the last bytes are, possibly past
                   1449:             the end of the source block, but since loads are aligned,
                   1450:             this is harmless.  */
                   1451: 
                   1452:          output_asm_insn ("ldws,ma 4(0,%1),%2", operands);
                   1453: 
                   1454:          for (offset = 4; offset < n_bytes; offset += 4)
                   1455:            {
                   1456:              output_asm_insn ("ldws,ma 4(0,%1),%3", operands);
                   1457:              output_asm_insn ("stws,ma %2,4(0,%0)", operands);
                   1458: 
                   1459:              temp = operands[2];
                   1460:              operands[2] = operands[3];
                   1461:              operands[3] = temp;
                   1462:            }
                   1463:          if (n_bytes % 4 == 0)
                   1464:            /* Store the last word.  */
                   1465:            output_asm_insn ("stw %2,0(0,%0)", operands);
                   1466:          else
                   1467:            {
                   1468:              /* Store the last, partial word.  */
                   1469:              operands[4] = gen_rtx (CONST_INT, VOIDmode, n_bytes % 4);
                   1470:              output_asm_insn ("stbys,e %2,%4(0,%0)", operands);
                   1471:            }
                   1472:          return "";
                   1473:        }
                   1474: 
                   1475:       if (align >= 2 && n_bytes >= 2)
                   1476:        {
                   1477:          output_asm_insn ("ldhs,ma 2(0,%1),%2", operands);
                   1478: 
                   1479:          for (offset = 2; offset + 2 <= n_bytes; offset += 2)
                   1480:            {
                   1481:              output_asm_insn ("ldhs,ma 2(0,%1),%3", operands);
                   1482:              output_asm_insn ("sths,ma %2,2(0,%0)", operands);
                   1483: 
                   1484:              temp = operands[2];
                   1485:              operands[2] = operands[3];
                   1486:              operands[3] = temp;
                   1487:            }
                   1488:          if (n_bytes % 2 != 0)
                   1489:            output_asm_insn ("ldb 0(0,%1),%3", operands);
                   1490: 
                   1491:          output_asm_insn ("sths,ma %2,2(0,%0)", operands);
                   1492: 
                   1493:          if (n_bytes % 2 != 0)
                   1494:            output_asm_insn ("stb %3,0(0,%0)", operands);
                   1495: 
                   1496:          return "";
                   1497:        }
                   1498: 
                   1499:       output_asm_insn ("ldbs,ma 1(0,%1),%2", operands);
                   1500: 
                   1501:       for (offset = 1; offset + 1 <= n_bytes; offset += 1)
                   1502:        {
                   1503:          output_asm_insn ("ldbs,ma 1(0,%1),%3", operands);
                   1504:          output_asm_insn ("stbs,ma %2,1(0,%0)", operands);
                   1505: 
                   1506:          temp = operands[2];
                   1507:          operands[2] = operands[3];
                   1508:          operands[3] = temp;
                   1509:        }
                   1510:       output_asm_insn ("stb %2,0(0,%0)", operands);
                   1511: 
                   1512:       return "";
                   1513:     }
                   1514: 
                   1515:   if (align != 4)
                   1516:     abort();
                   1517:      
                   1518:  copy_with_loop:
                   1519: 
                   1520:   if (size_is_constant)
                   1521:     {
                   1522:       /* Size is compile-time determined, and also not
                   1523:         very small (such small cases are handled above).  */
                   1524:       operands[4] = gen_rtx (CONST_INT, VOIDmode, n_bytes - 4);
                   1525:       output_asm_insn ("ldo %4(0),%2", operands);
                   1526:     }
                   1527:   else
                   1528:     {
                   1529:       /* Decrement counter by 4, and if it becomes negative, jump past the
                   1530:         word copying loop.  */
                   1531:       output_asm_insn ("addib,<,n -4,%2,.+16", operands);
                   1532:     }
                   1533: 
                   1534:   /* Copying loop.  Note that the first load is in the annulled delay slot
                   1535:      of addib.  Is it OK on PA to have a load in a delay slot, i.e. is a
                   1536:      possible page fault stopped in time?  */
                   1537:   output_asm_insn ("ldws,ma 4(0,%1),%3", operands);
                   1538:   output_asm_insn ("addib,>= -4,%2,.-4", operands);
                   1539:   output_asm_insn ("stws,ma %3,4(0,%0)", operands);
                   1540: 
                   1541:   /* The counter is negative, >= -4.  The remaining number of bytes are
                   1542:      determined by the two least significant bits.  */
                   1543: 
                   1544:   if (size_is_constant)
                   1545:     {
                   1546:       if (n_bytes % 4 != 0)
                   1547:        {
                   1548:          /* Read the entire word of the source block tail.  */
                   1549:          output_asm_insn ("ldw 0(0,%1),%3", operands);
                   1550:          operands[4] = gen_rtx (CONST_INT, VOIDmode, n_bytes % 4);
                   1551:          output_asm_insn ("stbys,e %3,%4(0,%0)", operands);
                   1552:        }
                   1553:     }
                   1554:   else
                   1555:     {
                   1556:       /* Add 4 to counter.  If it becomes zero, we're done.  */
                   1557:       output_asm_insn ("addib,=,n 4,%2,.+16", operands);
                   1558: 
                   1559:       /* Read the entire word of the source block tail.  (Also this
                   1560:         load is in an annulled delay slot.)  */
                   1561:       output_asm_insn ("ldw 0(0,%1),%3", operands);
                   1562: 
                   1563:       /* Make %0 point at the first byte after the destination block.  */
                   1564:       output_asm_insn ("add %2,%0,%0", operands);
                   1565:       /* Store the leftmost bytes, up to, but not including, the address
                   1566:         in %0.  */
                   1567:       output_asm_insn ("stbys,e %3,0(0,%0)", operands);
                   1568:     }
                   1569:   return "";
                   1570: }
                   1571: 
                   1572: /* Count the number of insns necessary to handle this block move.
                   1573: 
                   1574:    Basic structure is the same as emit_block_move, except that we
                   1575:    count insns rather than emit them.  */
                   1576: 
                   1577: int
                   1578: compute_movstrsi_length (insn)
                   1579:      rtx insn;
                   1580: {
                   1581:   rtx pat = PATTERN (insn);
                   1582:   int size_is_constant;
                   1583:   int align = INTVAL (XEXP (XVECEXP (pat, 0, 6), 0));
                   1584:   unsigned long n_bytes;
                   1585:   int insn_count = 0;
                   1586: 
                   1587:   if (GET_CODE (XEXP (XVECEXP (pat, 0, 5), 0)) == CONST_INT)
                   1588:     {
                   1589:       size_is_constant = 1;
                   1590:       n_bytes = INTVAL (XEXP (XVECEXP (pat, 0, 5), 0));
                   1591:     }
                   1592:   else
                   1593:     {
                   1594:       size_is_constant = 0;
                   1595:       n_bytes = 0;
                   1596:     }
                   1597: 
                   1598:   /* We can't move more than four bytes at a time because the PA
                   1599:      has no longer integer move insns.  (Could use fp mem ops?)  */
                   1600:   if (align > 4)
                   1601:     align = 4;
                   1602: 
                   1603:   if (size_is_constant)
                   1604:     {
                   1605:       unsigned long offset;
                   1606: 
                   1607:       if (n_bytes == 0)
                   1608:        return 0;
                   1609: 
                   1610:       if (align >= 4)
                   1611:        {
                   1612:          /* Don't unroll too large blocks.  */
                   1613:          if (n_bytes > 64)
                   1614:            goto copy_with_loop;
                   1615: 
                   1616:          /* first load */
                   1617:          insn_count = 1;
                   1618: 
                   1619:          /* Count the unrolled insns.  */
                   1620:          for (offset = 4; offset < n_bytes; offset += 4)
                   1621:            insn_count += 2;
                   1622: 
                   1623:          /* Count last store or partial store.  */
                   1624:          insn_count += 1;
                   1625:          return insn_count * 4;
                   1626:        }
                   1627: 
                   1628:       if (align >= 2 && n_bytes >= 2)
                   1629:        {
                   1630:          /* initial load.  */
                   1631:          insn_count = 1;
                   1632: 
                   1633:          /* Unrolled loop.  */
                   1634:          for (offset = 2; offset + 2 <= n_bytes; offset += 2)
                   1635:            insn_count += 2;
                   1636: 
                   1637:          /* ??? odd load/store */
                   1638:          if (n_bytes % 2 != 0)
                   1639:            insn_count += 2;
                   1640: 
                   1641:          /* ??? final store from loop.  */
                   1642:          insn_count += 1;
                   1643: 
                   1644:          return insn_count * 4;
                   1645:        }
                   1646: 
                   1647:       /* First load.  */
                   1648:       insn_count = 1;
                   1649: 
                   1650:       /* The unrolled loop.  */
                   1651:       for (offset = 1; offset + 1 <= n_bytes; offset += 1)
                   1652:        insn_count += 2;
                   1653: 
                   1654:       /* Final store.  */
                   1655:       insn_count += 1;
                   1656: 
                   1657:       return insn_count * 4;
                   1658:     }
                   1659: 
                   1660:   if (align != 4)
                   1661:     abort();
                   1662:      
                   1663:  copy_with_loop:
                   1664: 
                   1665:   /* setup for constant and non-constant case.  */
                   1666:   insn_count = 1;
                   1667: 
                   1668:   /* The copying loop.  */
                   1669:   insn_count += 3;
                   1670: 
                   1671:   /* The counter is negative, >= -4.  The remaining number of bytes are
                   1672:      determined by the two least significant bits.  */
                   1673: 
                   1674:   if (size_is_constant)
                   1675:     {
                   1676:       if (n_bytes % 4 != 0)
                   1677:        insn_count += 2;
                   1678:     }
                   1679:   else
                   1680:     insn_count += 4;
                   1681:   return insn_count * 4;
                   1682: }
                   1683: 
                   1684: 
                   1685: char *
                   1686: output_and (operands)
                   1687:      rtx *operands;
                   1688: {
                   1689:   if (GET_CODE (operands[2]) == CONST_INT && INTVAL (operands[2]) != 0)
                   1690:     {
                   1691:       unsigned mask = INTVAL (operands[2]);
                   1692:       int ls0, ls1, ms0, p, len;
                   1693: 
                   1694:       for (ls0 = 0; ls0 < 32; ls0++)
                   1695:        if ((mask & (1 << ls0)) == 0)
                   1696:          break;
                   1697: 
                   1698:       for (ls1 = ls0; ls1 < 32; ls1++)
                   1699:        if ((mask & (1 << ls1)) != 0)
                   1700:          break;
                   1701: 
                   1702:       for (ms0 = ls1; ms0 < 32; ms0++)
                   1703:        if ((mask & (1 << ms0)) == 0)
                   1704:          break;
                   1705: 
                   1706:       if (ms0 != 32)
                   1707:        abort();
                   1708: 
                   1709:       if (ls1 == 32)
                   1710:        {
                   1711:          len = ls0;
                   1712: 
                   1713:          if (len == 0)
                   1714:            abort ();
                   1715: 
                   1716:          operands[2] = gen_rtx (CONST_INT, VOIDmode, len);
                   1717:          return "extru %1,31,%2,%0";
                   1718:        }
                   1719:       else
                   1720:        {
                   1721:          /* We could use this `depi' for the case above as well, but `depi'
                   1722:             requires one more register file access than an `extru'.  */
                   1723: 
                   1724:          p = 31 - ls0;
                   1725:          len = ls1 - ls0;
                   1726: 
                   1727:          operands[2] = gen_rtx (CONST_INT, VOIDmode, p);
                   1728:          operands[3] = gen_rtx (CONST_INT, VOIDmode, len);
                   1729:          return "depi 0,%2,%3,%0";
                   1730:        }
                   1731:     }
                   1732:   else
                   1733:     return "and %1,%2,%0";
                   1734: }
                   1735: 
                   1736: char *
                   1737: output_ior (operands)
                   1738:      rtx *operands;
                   1739: {
                   1740:   unsigned mask = INTVAL (operands[2]);
                   1741:   int bs0, bs1, p, len;
                   1742:  
                   1743:   if (INTVAL (operands[2]) == 0)
                   1744:     return "copy %1,%0";
                   1745: 
                   1746:   for (bs0 = 0; bs0 < 32; bs0++)
                   1747:     if ((mask & (1 << bs0)) != 0)
                   1748:       break;
                   1749: 
                   1750:   for (bs1 = bs0; bs1 < 32; bs1++)
                   1751:     if ((mask & (1 << bs1)) == 0)
                   1752:       break;
                   1753: 
                   1754:   if (bs1 != 32 && ((unsigned) 1 << bs1) <= mask)
                   1755:     abort();
                   1756: 
                   1757:   p = 31 - bs0;
                   1758:   len = bs1 - bs0;
                   1759: 
                   1760:   operands[2] = gen_rtx (CONST_INT, VOIDmode, p);
                   1761:   operands[3] = gen_rtx (CONST_INT, VOIDmode, len);
                   1762:   return "depi -1,%2,%3,%0";
                   1763: }
                   1764: 
                   1765: /* Output an ascii string.  */
                   1766: void
                   1767: output_ascii (file, p, size)
                   1768:      FILE *file;
                   1769:      unsigned char *p;
                   1770:      int size;
                   1771: {
                   1772:   int i;
                   1773:   int chars_output;
                   1774:   unsigned char partial_output[16];    /* Max space 4 chars can occupy.   */
                   1775: 
                   1776:   /* The HP assembler can only take strings of 256 characters at one
                   1777:      time.  This is a limitation on input line length, *not* the
                   1778:      length of the string.  Sigh.  Even worse, it seems that the
                   1779:      restriction is in number of input characters (see \xnn &
                   1780:      \whatever).  So we have to do this very carefully.  */
                   1781: 
                   1782:   fprintf (file, "\t.%s \"", STRING_SECTION_NAME);
                   1783: 
                   1784:   chars_output = 0;
                   1785:   for (i = 0; i < size; i += 4)
                   1786:     {
                   1787:       int co = 0;
                   1788:       int io = 0;
                   1789:       for (io = 0, co = 0; io < MIN (4, size - i); io++)
                   1790:        {
                   1791:          register unsigned int c = p[i + io];
                   1792: 
                   1793:          if (c == '\"' || c == '\\')
                   1794:            partial_output[co++] = '\\';
                   1795: 
                   1796:          if (c >= ' ' && c < 0177)
                   1797:            partial_output[co++] = c;
                   1798: 
                   1799:          else if (TARGET_GAS)
                   1800:            {
                   1801:              /* output octal numbers for gas */
                   1802:              partial_output[co++] = '\\';
                   1803:              partial_output[co++] = (c / 64) % 8 - 0 + '0';
                   1804:              partial_output[co++] = (c / 8) % 8 - 0 + '0';
                   1805:              partial_output[co++] = c % 8 - 0 + '0';
                   1806:            }
                   1807:          else
                   1808:            {
                   1809:              unsigned int hexd;
                   1810:              partial_output[co++] = '\\';
                   1811:              partial_output[co++] = 'x';
                   1812:              hexd =  c  / 16 - 0 + '0';
                   1813:              if (hexd > '9')
                   1814:                hexd -= '9' - 'a' + 1;
                   1815:              partial_output[co++] = hexd;
                   1816:              hexd =  c % 16 - 0 + '0';
                   1817:              if (hexd > '9')
                   1818:                hexd -= '9' - 'a' + 1;
                   1819:              partial_output[co++] = hexd;
                   1820:            }
                   1821:        }
                   1822:       if (chars_output + co > 243)
                   1823:        {
                   1824:          fprintf (file, "\"\n\t.%s \"", STRING_SECTION_NAME);
                   1825:          chars_output = 0;
                   1826:        }
                   1827:       fwrite (partial_output, 1, co, file);
                   1828:       chars_output += co;
                   1829:       co = 0;
                   1830:     }
                   1831:   fprintf (file, "\"\n");
                   1832: }
                   1833: 
                   1834: /* You may have trouble believing this, but this is the HP-PA stack
                   1835:    layout.  Wow.
                   1836: 
                   1837:    Offset              Contents
                   1838: 
                   1839:    Variable arguments  (optional; any number may be allocated)
                   1840: 
                   1841:    SP-(4*(N+9))                arg word N
                   1842:        :                   :
                   1843:       SP-56            arg word 5
                   1844:       SP-52            arg word 4
                   1845: 
                   1846:    Fixed arguments     (must be allocated; may remain unused)
                   1847: 
                   1848:       SP-48            arg word 3
                   1849:       SP-44            arg word 2
                   1850:       SP-40            arg word 1
                   1851:       SP-36            arg word 0
                   1852: 
                   1853:    Frame Marker
                   1854: 
                   1855:       SP-32            External Data Pointer (DP)
                   1856:       SP-28            External sr4
                   1857:       SP-24            External/stub RP (RP')
                   1858:       SP-20            Current RP
                   1859:       SP-16            Static Link
                   1860:       SP-12            Clean up
                   1861:       SP-8             Calling Stub RP (RP'')
                   1862:       SP-4             Previous SP
                   1863: 
                   1864:    Top of Frame
                   1865: 
                   1866:       SP-0             Stack Pointer (points to next available address)
                   1867: 
                   1868: */
                   1869: 
                   1870: /* This function saves registers as follows.  Registers marked with ' are
                   1871:    this function's registers (as opposed to the previous function's).
                   1872:    If a frame_pointer isn't needed, r4 is saved as a general register;
                   1873:    the space for the frame pointer is still allocated, though, to keep
                   1874:    things simple.
                   1875: 
                   1876: 
                   1877:    Top of Frame
                   1878: 
                   1879:        SP (FP')                Previous FP
                   1880:        SP + 4          Alignment filler (sigh)
                   1881:        SP + 8          Space for locals reserved here.
                   1882:        .
                   1883:        .
                   1884:        .
                   1885:        SP + n          All call saved register used.
                   1886:        .
                   1887:        .
                   1888:        .
                   1889:        SP + o          All call saved fp registers used.
                   1890:        .
                   1891:        .
                   1892:        .
                   1893:        SP + p (SP')    points to next available address.
                   1894:        
                   1895: */
                   1896: 
                   1897: /* Emit RTL to store REG at the memory location specified by BASE+DISP.
                   1898:    Handle case where DISP > 8k by using the add_high_const pattern.
                   1899: 
                   1900:    Note in DISP > 8k case, we will leave the high part of the address
                   1901:    in %r1.  There is code in expand_hppa_{prologue,epilogue} that knows this.*/
                   1902: static void
                   1903: store_reg (reg, disp, base)
                   1904:      int reg, disp, base;
                   1905: {
                   1906:   if (VAL_14_BITS_P (disp))
                   1907:     {
                   1908:       emit_move_insn (gen_rtx (MEM, SImode, 
                   1909:                               gen_rtx (PLUS, SImode, 
                   1910:                                        gen_rtx (REG, SImode, base),
                   1911:                                        GEN_INT (disp))),
                   1912:                      gen_rtx (REG, SImode, reg));
                   1913:     }
                   1914:   else
                   1915:     {
                   1916:       emit_insn (gen_add_high_const (gen_rtx (REG, SImode, 1), 
                   1917:                                     gen_rtx (REG, SImode, base), 
                   1918:                                     GEN_INT (disp)));
                   1919:       emit_move_insn (gen_rtx (MEM, SImode,
                   1920:                               gen_rtx (LO_SUM, SImode, 
                   1921:                                        gen_rtx (REG, SImode, 1),
                   1922:                                        GEN_INT (disp))),
                   1923:                      gen_rtx (REG, SImode, reg));
                   1924:     }
                   1925: }
                   1926: 
                   1927: /* Emit RTL to load REG from the memory location specified by BASE+DISP.
                   1928:    Handle case where DISP > 8k by using the add_high_const pattern.
                   1929: 
                   1930:    Note in DISP > 8k case, we will leave the high part of the address
                   1931:    in %r1.  There is code in expand_hppa_{prologue,epilogue} that knows this.*/
                   1932: static void
                   1933: load_reg (reg, disp, base)
                   1934:      int reg, disp, base;
                   1935: {
                   1936:   if (VAL_14_BITS_P (disp))
                   1937:     {
                   1938:       emit_move_insn (gen_rtx (REG, SImode, reg),
                   1939:                      gen_rtx (MEM, SImode, 
                   1940:                               gen_rtx (PLUS, SImode, 
                   1941:                                        gen_rtx (REG, SImode, base),
                   1942:                                        GEN_INT (disp))));
                   1943:                      
                   1944:     }
                   1945:   else
                   1946:     {
                   1947:       emit_insn (gen_add_high_const (gen_rtx (REG, SImode, 1), 
                   1948:                                     gen_rtx (REG, SImode, base),
                   1949:                                     GEN_INT (disp)));
                   1950:       emit_move_insn (gen_rtx (REG, SImode, reg),
                   1951:                      gen_rtx (MEM, SImode,
                   1952:                               gen_rtx (LO_SUM, SImode, 
                   1953:                                        gen_rtx (REG, SImode, 1), 
                   1954:                                        GEN_INT (disp))));
                   1955:     }
                   1956: }
                   1957: 
                   1958: /* Emit RTL to set REG to the value specified by BASE+DISP.
                   1959:    Handle case where DISP > 8k by using the add_high_const pattern.
                   1960: 
                   1961:    Note in DISP > 8k case, we will leave the high part of the address
                   1962:    in %r1.  There is code in expand_hppa_{prologue,epilogue} that knows this.*/
                   1963: static void
                   1964: set_reg_plus_d(reg, base, disp)
                   1965:      int reg, base, disp;
                   1966: {
                   1967:   if (VAL_14_BITS_P (disp))
                   1968:     {
                   1969:       emit_move_insn (gen_rtx (REG, SImode, reg),
                   1970:                      gen_rtx (PLUS, SImode, 
                   1971:                               gen_rtx (REG, SImode, base),
                   1972:                               GEN_INT (disp)));
                   1973:       
                   1974:     }
                   1975:   else
                   1976:     {
                   1977:       emit_insn (gen_add_high_const (gen_rtx (REG, SImode, 1), 
                   1978:                                     gen_rtx (REG, SImode, base),
                   1979:                                     GEN_INT (disp)));
                   1980:       emit_move_insn (gen_rtx (REG, SImode, reg),
                   1981:                      gen_rtx (LO_SUM, SImode, 
                   1982:                                        gen_rtx (REG, SImode, 1),
                   1983:                                        GEN_INT (disp)));
                   1984:     }
                   1985: }
                   1986: 
                   1987: /* Global variables set by FUNCTION_PROLOGUE.  */
                   1988: /* Size of frame.  Need to know this to emit return insns from
                   1989:    leaf procedures.  */
                   1990: static int actual_fsize;
                   1991: static int local_fsize, save_fregs;
                   1992: 
                   1993: int
                   1994: compute_frame_size (size, fregs_live)
                   1995:      int size;
                   1996:      int *fregs_live;
                   1997: {
                   1998:   extern int current_function_outgoing_args_size;
                   1999:   int i, fsize;
                   2000: 
                   2001:   /* 8 is space for frame pointer + filler. If any frame is allocated 
                   2002:      we need to add this in because of STARTING_FRAME_OFFSET. */
                   2003:   fsize = size + (size || frame_pointer_needed ? 8 : 0);
                   2004: 
                   2005:   /* fp is stored in a special place. */
                   2006:   if (frame_pointer_needed)
                   2007:     {
                   2008:       for (i = 18; i >= 5; i--)
                   2009:        if (regs_ever_live[i])
                   2010:          fsize += 4;
                   2011: 
                   2012:       if (regs_ever_live[3])
                   2013:        fsize += 4;
                   2014:     }
                   2015:   else
                   2016:     {
                   2017:       for (i = 18; i >= 3; i--)
                   2018:        if (regs_ever_live[i])
                   2019:          fsize += 4;
                   2020:     }
                   2021:   fsize = (fsize + 7) & ~7;
                   2022: 
                   2023:   if (!TARGET_SNAKE)
                   2024:     {
                   2025:       for (i = 43; i >= 40; i--)
                   2026:        if (regs_ever_live[i])
                   2027:          {
                   2028:            fsize += 8;
                   2029:            if (fregs_live)
                   2030:              *fregs_live = 1;
                   2031:          }
                   2032:     }
                   2033:   else
                   2034:     {
                   2035:       for (i = 78; i >= 60; i -= 2)
                   2036:        if (regs_ever_live[i] || regs_ever_live[i + 1])
                   2037:          {
                   2038:            fsize += 8;
                   2039:            if (fregs_live)
                   2040:              *fregs_live = 1;
                   2041:          }
                   2042:     }
                   2043:   fsize += current_function_outgoing_args_size;
                   2044:   if (! leaf_function_p () || fsize)
                   2045:     fsize += 32;
                   2046:   return (fsize + 63) & ~63;
                   2047: }
                   2048:      
                   2049: rtx hp_profile_label_rtx;
                   2050: static char hp_profile_label_name[8];
                   2051: void
                   2052: output_function_prologue (file, size)
                   2053:      FILE *file;
                   2054:      int size;
                   2055: {
                   2056: 
                   2057:   /* hppa_expand_prologue does the dirty work now.  We just need
                   2058:      to output the assembler directives which denote the start
                   2059:      of a function.  */
                   2060:   fprintf (file, "\t.PROC\n\t.CALLINFO FRAME=%d", actual_fsize);
                   2061:   if (regs_ever_live[2] || profile_flag)
                   2062:     fprintf (file, ",CALLS,SAVE_RP");
                   2063:   else
                   2064:     fprintf (file, ",NO_CALLS");
                   2065: 
                   2066:   if (frame_pointer_needed)
                   2067:     fprintf (file, ",SAVE_SP");
                   2068: 
                   2069:   /* Pass on information about the number of callee register saves
                   2070:      performed in the prologue.
                   2071: 
                   2072:      The compiler is supposed to pass the highest register number
                   2073:      saved, the assembler then has to adjust that number before 
                   2074:      entering it into the unwind descriptor (to account for any
                   2075:      caller saved registers with lower register numbers than the 
                   2076:      first callee saved register).  */
                   2077:   if (gr_saved)
                   2078:     fprintf (file, ",ENTRY_GR=%d", gr_saved + 2);
                   2079: 
                   2080:   if (fr_saved)
                   2081:     fprintf (file, ",ENTRY_FR=%d", fr_saved + 11);
                   2082: 
                   2083:   fprintf (file, "\n\t.ENTRY\n");
                   2084: 
                   2085:   /* Horrid hack.  emit_function_prologue will modify this RTL in
                   2086:      place to get the expected results.  */
                   2087:   if (profile_flag)
                   2088:     sprintf(hp_profile_label_name, "LP$%04d", hp_profile_labelno);
                   2089: }
                   2090: 
                   2091: void
                   2092: hppa_expand_prologue()
                   2093: {
                   2094: 
                   2095:   extern char call_used_regs[];
                   2096:   int size = get_frame_size ();
                   2097:   int merge_sp_adjust_with_store = 0;
                   2098:   int i, offset;
                   2099:   rtx tmpreg, size_rtx;
                   2100: 
                   2101: 
                   2102:   gr_saved = 0;
                   2103:   fr_saved = 0;
                   2104:   save_fregs = 0;
                   2105:   local_fsize =  size + (size || frame_pointer_needed ? 8 : 0);
                   2106:   actual_fsize = compute_frame_size (size, &save_fregs);
                   2107: 
                   2108:   /* Compute a few things we will use often.  */
                   2109:   tmpreg = gen_rtx (REG, SImode, 1);
                   2110:   size_rtx = GEN_INT (actual_fsize);
                   2111: 
                   2112:   /* Save RP first.  The calling conventions manual states RP will 
                   2113:      always be stored into the caller's frame at sp-20.  */
                   2114:   if (regs_ever_live[2] || profile_flag)
                   2115:     store_reg (2, -20, STACK_POINTER_REGNUM);  
                   2116:     
                   2117:   /* Allocate the local frame and set up the frame pointer if needed.  */
                   2118:   if (actual_fsize)
                   2119:     if (frame_pointer_needed)
                   2120:       {
                   2121:        /* Copy the old frame pointer temporarily into %r1.  Set up the
                   2122:           new stack pointer, then store away the saved old frame pointer
                   2123:           into the stack at sp+actual_fsize and at the same time update
                   2124:           the stack pointer by actual_fsize bytes.  Two versions, first
                   2125:           handles small (<8k) frames.  The second handles large (>8k)
                   2126:           frames.  */
                   2127:        emit_move_insn (tmpreg, frame_pointer_rtx);
                   2128:        emit_move_insn (frame_pointer_rtx, stack_pointer_rtx);
                   2129:        if (VAL_14_BITS_P (actual_fsize))
                   2130:          emit_insn (gen_post_stwm (stack_pointer_rtx,
                   2131:                                    stack_pointer_rtx,
                   2132:                                    size_rtx, tmpreg));
                   2133:        else
                   2134:          {
                   2135:            store_reg (1, 0, FRAME_POINTER_REGNUM);
                   2136:            set_reg_plus_d (STACK_POINTER_REGNUM,
                   2137:                            STACK_POINTER_REGNUM,
                   2138:                            actual_fsize);
                   2139:          }
                   2140:       }
                   2141:     /* no frame pointer needed.  */
                   2142:     else
                   2143:       {
                   2144:        /* In some cases we can perform the first callee register save
                   2145:           and allocating the stack frame at the same time.   If so, just
                   2146:           make a note of it and defer allocating the frame until saving
                   2147:           the callee registers.  */
                   2148:        if (VAL_14_BITS_P (-actual_fsize) 
                   2149:            && local_fsize == 0 
                   2150:            && ! profile_flag
                   2151:            && ! flag_pic)
                   2152:          merge_sp_adjust_with_store = 1;
                   2153:        /* Can not optimize.  Adjust the stack frame by actual_fsize bytes.  */
                   2154:        else if (actual_fsize != 0)
                   2155:          set_reg_plus_d (STACK_POINTER_REGNUM,
                   2156:                          STACK_POINTER_REGNUM,
                   2157:                          actual_fsize);
                   2158:       }
                   2159:   /* The hppa calling conventions say that that %r19, the pic offset
                   2160:      register, is saved at sp - 32 (in this function's frame)  when
                   2161:      generating PIC code.  */
                   2162:   if (flag_pic)
                   2163:     {
                   2164: #ifndef NeXT_ASM
                   2165:       store_reg (PIC_OFFSET_TABLE_REGNUM, -32, STACK_POINTER_REGNUM);  
                   2166: #endif
                   2167:     }
                   2168: 
                   2169:       
                   2170:   /* Profiling code.
                   2171: 
                   2172:      Instead of taking one argument, the counter label, as most normal
                   2173:      mcounts do, _mcount appears to behave differently on the HPPA.  It
                   2174:      takes the return address of the caller, the address of this routine,      
                   2175:      and the address of the label.  Also, it isn't magic, so 
                   2176:      argument registre hsave to be preserved.  */
                   2177:   if (profile_flag)
                   2178:     {
                   2179:       int pc_offset, i, arg_offset, basereg, offsetadj;
                   2180: 
                   2181:       pc_offset = 4 + (frame_pointer_needed
                   2182:                       ? (VAL_14_BITS_P (actual_fsize) ? 12 : 20)
                   2183:                       : (VAL_14_BITS_P (actual_fsize) ? 4 : 8));
                   2184: 
                   2185:       /* When the function has a frame pointer, use it as the base
                   2186:         register for saving/restore registers.  Else use the stack
                   2187:         pointer.  Adjust the offset according to the frame size if
                   2188:         this function does not have a frame pointer.  */
                   2189: 
                   2190:       basereg = frame_pointer_needed ? FRAME_POINTER_REGNUM
                   2191:                                     : STACK_POINTER_REGNUM;
                   2192:       offsetadj = frame_pointer_needed ? 0 : actual_fsize;
                   2193: 
                   2194:       /* Horrid hack.  emit_function_prologue will modify this RTL in
                   2195:         place to get the expected results.   sprintf here is just to
                   2196:         put something in the name.  */
                   2197:       sprintf(hp_profile_label_name, "LP%d", profile_label_no);
                   2198:       hp_profile_label_rtx = gen_rtx (SYMBOL_REF, SImode,
                   2199:                                      hp_profile_label_name);
                   2200:       if (current_function_returns_struct)
                   2201:        store_reg (STRUCT_VALUE_REGNUM, - 12 - offsetadj, basereg);
                   2202: 
                   2203:       for (i = 26, arg_offset = -36 - offsetadj; i >= 23; i--, arg_offset -= 4)
                   2204:        if (regs_ever_live [i])
                   2205:          {
                   2206:            store_reg (i, arg_offset, basereg);
                   2207:            /* Deal with arg_offset not fitting in 14 bits.  */
                   2208:            pc_offset += VAL_14_BITS_P (arg_offset) ? 4 : 8;
                   2209:          }
                   2210: 
                   2211:       emit_move_insn (gen_rtx (REG, SImode, 26), gen_rtx (REG, SImode, 2));
                   2212: #ifndef NeXT_ASM
                   2213:       emit_move_insn (tmpreg, gen_rtx (HIGH, SImode, hp_profile_label_rtx));
                   2214:       emit_move_insn (gen_rtx (REG, SImode, 24),
                   2215:                      gen_rtx (LO_SUM, SImode, tmpreg, hp_profile_label_rtx));
                   2216: #endif
                   2217:       /* %r25 is set from within the output pattern.  */
                   2218:       emit_insn (gen_call_profiler (GEN_INT (- pc_offset - 20)));
                   2219: 
                   2220:       /* Restore argument registers.  */
                   2221:       for (i = 26, arg_offset = -36 - offsetadj; i >= 23; i--, arg_offset -= 4)
                   2222:        if (regs_ever_live [i])
                   2223:          load_reg (i, arg_offset, basereg);
                   2224: 
                   2225:       if (current_function_returns_struct)
                   2226:        load_reg (STRUCT_VALUE_REGNUM, -12 - offsetadj, basereg);
                   2227: 
                   2228:     }
                   2229: 
                   2230:   /* Normal register save. 
                   2231: 
                   2232:      Do not save the frame pointer in the frame_pointer_needed case.  It
                   2233:      was done earlier.  */
                   2234:   if (frame_pointer_needed)
                   2235:     {
                   2236:       for (i = 18, offset = local_fsize; i >= 3; i--)
                   2237:        if (regs_ever_live[i] && ! call_used_regs[i]
                   2238:            && i != FRAME_POINTER_REGNUM)
                   2239:          {
                   2240:            store_reg (i, offset, FRAME_POINTER_REGNUM);  
                   2241:            offset += 4;
                   2242:            gr_saved++;
                   2243:          }
                   2244:       /* Account for %r4 which is saved in a special place.  */
                   2245:       gr_saved++;
                   2246:     }
                   2247:   /* No frame pointer needed.  */
                   2248:   else
                   2249:     {
                   2250:       for (i = 18, offset = local_fsize - actual_fsize; i >= 3; i--)
                   2251:        if (regs_ever_live[i] && ! call_used_regs[i])
                   2252:          {
                   2253:            /* If merge_sp_adjust_with_store is nonzero, then we can 
                   2254:               optimize the first GR save.  */
                   2255:            if (merge_sp_adjust_with_store)
                   2256:              {
                   2257:                merge_sp_adjust_with_store = 0;
                   2258:                emit_insn (gen_post_stwm (stack_pointer_rtx,
                   2259:                                          stack_pointer_rtx,
                   2260:                                          GEN_INT (-offset),
                   2261:                                          gen_rtx (REG, SImode, i)));
                   2262:              }
                   2263:            else
                   2264:              store_reg (i, offset, STACK_POINTER_REGNUM);
                   2265:            offset += 4;
                   2266:            gr_saved++;
                   2267:          }
                   2268: 
                   2269:       /* If we wanted to merge the SP adjustment with a GR save, but we never
                   2270:         did any GR saves, then just emit the adjustment here.  */
                   2271:       if (merge_sp_adjust_with_store)
                   2272:        set_reg_plus_d (STACK_POINTER_REGNUM,
                   2273:                        STACK_POINTER_REGNUM,
                   2274:                        actual_fsize);
                   2275:     }
                   2276:       
                   2277: 
                   2278:   /* Align pointer properly (doubleword boundary).  */
                   2279:   offset = (offset + 7) & ~7;
                   2280: 
                   2281:   /* Floating point register store.  */
                   2282:   if (save_fregs)
                   2283:     {
                   2284: 
                   2285:       /* First get the frame or stack pointer to the start of the FP register
                   2286:         save area.  */
                   2287:       if (frame_pointer_needed)
                   2288:        set_reg_plus_d (1, FRAME_POINTER_REGNUM, offset);
                   2289:       else
                   2290:        set_reg_plus_d (1, STACK_POINTER_REGNUM, offset);
                   2291: 
                   2292:       /* Now actually save the FP registers.  */
                   2293:       if (! TARGET_SNAKE)
                   2294:        {
                   2295:          for (i = 43; i >= 40; i--)
                   2296:            if (regs_ever_live[i])
                   2297:              {
                   2298:                emit_move_insn (gen_rtx (MEM, DFmode, 
                   2299:                                         gen_rtx (POST_INC, DFmode, tmpreg)),
                   2300:                                gen_rtx (REG, DFmode, i));
                   2301:                fr_saved++;
                   2302:              }
                   2303:        }
                   2304:       else
                   2305:        {
                   2306:          for (i = 78; i >= 60; i -= 2)
                   2307:            if (regs_ever_live[i] || regs_ever_live[i + 1])
                   2308:              {
                   2309:                emit_move_insn (gen_rtx (MEM, DFmode, 
                   2310:                                         gen_rtx (POST_INC, DFmode, tmpreg)),
                   2311:                                gen_rtx (REG, DFmode, i));
                   2312:                fr_saved++;
                   2313:              }
                   2314:        }
                   2315:     }
                   2316: }
                   2317: 
                   2318: 
                   2319: void
                   2320: output_function_epilogue (file, size)
                   2321:      FILE *file;
                   2322:      int size;
                   2323: {
                   2324: 
                   2325:   rtx insn = get_last_insn ();
                   2326: 
                   2327:   /* hppa_expand_epilogue does the dirty work now.  We just need
                   2328:      to output the assembler directives which denote the end
                   2329:      of a function.
                   2330: 
                   2331:      To make debuggers happy, emit a nop if the epilogue was completely
                   2332:      eliminated due to a volatile call as the last insn in the
                   2333:      current function.  That way the return address (in %r2) will 
                   2334:      always point to a valid instruction in the current function.  */
                   2335: 
                   2336:   /* Get the last real insn.  */
                   2337:   if (GET_CODE (insn) == NOTE)
                   2338:     insn = prev_real_insn (insn);
                   2339: 
                   2340:   /* If it is a sequence, then look inside.  */
                   2341:   if (insn && GET_CODE (insn) == INSN && GET_CODE (PATTERN (insn)) == SEQUENCE)
                   2342:     insn = XVECEXP (PATTERN (insn), 0, 0);
                   2343: 
                   2344:   /* If insn is a CALL_INSN, then it must be a call to a volatile 
                   2345:      function (otherwise there would be epilogue insns).  */
                   2346:   if (insn && GET_CODE (insn) == CALL_INSN)
                   2347:     fprintf (file, "\tnop\n");
                   2348:   
                   2349:   fprintf (file, "\t.EXIT\n\t.PROCEND\n");
                   2350: }
                   2351: 
                   2352: void
                   2353: hppa_expand_epilogue ()
                   2354: {
                   2355:   rtx tmpreg; 
                   2356:   int offset,i;
                   2357:   int merge_sp_adjust_with_load  = 0;
                   2358: 
                   2359:   /* We will use this often.  */
                   2360:   tmpreg = gen_rtx (REG, SImode, 1);
                   2361: 
                   2362:   /* Try to restore RP early to avoid load/use interlocks when
                   2363:      RP gets used in the return (bv) instruction.  This appears to still
                   2364:      be necessary even when we schedule the prologue and epilogue. */
                   2365:   if (frame_pointer_needed
                   2366:       && (regs_ever_live [2] || profile_flag))
                   2367:     load_reg (2, -20, FRAME_POINTER_REGNUM);
                   2368: 
                   2369:   /* No frame pointer, and stack is smaller than 8k.  */
                   2370:   else if (! frame_pointer_needed
                   2371:           && VAL_14_BITS_P (actual_fsize + 20)
                   2372:           && (regs_ever_live[2] || profile_flag))
                   2373:     load_reg (2, - (actual_fsize + 20), STACK_POINTER_REGNUM);
                   2374: 
                   2375:   /* General register restores.  */
                   2376:   if (frame_pointer_needed)
                   2377:     {
                   2378:       for (i = 18, offset = local_fsize; i >= 3; i--)
                   2379:        if (regs_ever_live[i] && ! call_used_regs[i]
                   2380:            && i != FRAME_POINTER_REGNUM)
                   2381:          {
                   2382:            load_reg (i, offset, FRAME_POINTER_REGNUM);
                   2383:            offset += 4;
                   2384:          }
                   2385:     }
                   2386:   else
                   2387:     {
                   2388:       for (i = 18, offset = local_fsize - actual_fsize; i >= 3; i--)
                   2389:        if (regs_ever_live[i] && ! call_used_regs[i])
                   2390:          {
                   2391:            /* Only for the first load.
                   2392:               merge_sp_adjust_with_load holds the register load
                   2393:               with which we will merge the sp adjustment.  */
                   2394:            if (VAL_14_BITS_P (actual_fsize + 20)
                   2395:                && local_fsize == 0
                   2396:                && ! merge_sp_adjust_with_load)
                   2397:              merge_sp_adjust_with_load = i;
                   2398:            else
                   2399:              load_reg (i, offset, STACK_POINTER_REGNUM);
                   2400:            offset += 4;
                   2401:          }
                   2402:     }
                   2403: 
                   2404:   /* Align pointer properly (doubleword boundary).  */
                   2405:   offset = (offset + 7) & ~7;
                   2406: 
                   2407:   /* FP register restores.  */
                   2408:   if (save_fregs)
                   2409:     {
                   2410:       /* Adjust the register to index off of.  */
                   2411:       if (frame_pointer_needed)
                   2412:        set_reg_plus_d (1, FRAME_POINTER_REGNUM, offset);
                   2413:       else
                   2414:        set_reg_plus_d (1, STACK_POINTER_REGNUM, offset);
                   2415: 
                   2416:       /* Actually do the restores now.  */
                   2417:       if (! TARGET_SNAKE)
                   2418:        {
                   2419:          for (i = 43; i >= 40; i--)
                   2420:            if (regs_ever_live[i])
                   2421:              emit_move_insn (gen_rtx (REG, DFmode, i),
                   2422:                              gen_rtx (MEM, DFmode, 
                   2423:                                       gen_rtx (POST_INC, DFmode, tmpreg)));
                   2424:              
                   2425:        }
                   2426:       else
                   2427:        {
                   2428:          for (i = 78; i >= 60; i -= 2)
                   2429:            if (regs_ever_live[i] || regs_ever_live[i + 1])
                   2430:              emit_move_insn (gen_rtx (REG, DFmode, i),
                   2431:                              gen_rtx (MEM, DFmode, 
                   2432:                                       gen_rtx (POST_INC, DFmode, tmpreg)));
                   2433:        }
                   2434:     }
                   2435: 
                   2436:   /* No frame pointer, but we have a stack greater than 8k.  We restore
                   2437:      %r2 very late in this case.  (All other cases are restored as early
                   2438:      as possible.)  */
                   2439:   if (! frame_pointer_needed
                   2440:       && ! VAL_14_BITS_P (actual_fsize + 20)
                   2441:       && (regs_ever_live[2] || profile_flag))
                   2442:     {
                   2443:       set_reg_plus_d (STACK_POINTER_REGNUM,
                   2444:                      STACK_POINTER_REGNUM,
                   2445:                      - actual_fsize);
                   2446: #ifdef NeXT_ASM
                   2447:      /* The calculation previously used is incorrect with the NeXT
                   2448:       * assembler.  We substitute a load_reg from the stack, which
                   2449:       * unfortunately will prevent an instruction from being moved into
                   2450:       * the delay slot of the return.
                   2451:       * We could use the strategy of doing the load_reg from r1 to get rid
                   2452:       * of the extra instruction, but the calculation would be significantly
                   2453:       * more difficult (and subject to change).
                   2454:       */
                   2455:       load_reg (2, -20, STACK_POINTER_REGNUM);
                   2456: #else
                   2457:       /* Uses value left over in %r1 by set_reg_plus_d.  */
                   2458:       load_reg (2, - (actual_fsize + 20 + ((- actual_fsize) & ~0x7ff)), 1);
                   2459: #endif
                   2460:     }
                   2461: 
                   2462:   /* Reset stack pointer (and possibly frame pointer).  The stack */
                   2463:   /* pointer is initially set to fp + 64 to avoid a race condition.
                   2464:      ??? What race condition?!?  */
                   2465:   else if (frame_pointer_needed)
                   2466:     {
                   2467:       /* Emit a blockage insn here to keep these insns from being moved
                   2468:         to the beginning of the prologue or into the main instruction
                   2469:         stream, doing so avoids some very obscure problems.  */
                   2470:       emit_insn (gen_blockage ());
                   2471:       set_reg_plus_d (STACK_POINTER_REGNUM, FRAME_POINTER_REGNUM, 64);
                   2472:       emit_insn (gen_pre_ldwm (stack_pointer_rtx, stack_pointer_rtx,
                   2473:                               GEN_INT (-64), frame_pointer_rtx));
                   2474:     }
                   2475:   /* If we were deferring a callee register restore, do it now.  */
                   2476:   else if (! frame_pointer_needed  && merge_sp_adjust_with_load)
                   2477:     emit_insn (gen_pre_ldwm (stack_pointer_rtx,
                   2478:                             stack_pointer_rtx,
                   2479:                             GEN_INT (- actual_fsize),
                   2480:                             gen_rtx (REG, SImode, 
                   2481:                             merge_sp_adjust_with_load)));
                   2482:   else if (actual_fsize != 0)
                   2483:     set_reg_plus_d (STACK_POINTER_REGNUM,
                   2484:                    STACK_POINTER_REGNUM,
                   2485:                    - actual_fsize);
                   2486: }
                   2487: 
                   2488: /* This is only valid once reload has completed because it depends on
                   2489:    knowing exactly how much (if any) frame there is and...
                   2490: 
                   2491:    It's only valid if there is no frame marker to de-allocate and...
                   2492: 
                   2493:    It's only valid if %r2 hasn't been saved into the caller's frame
                   2494:    (we're not profiling and %r2 isn't live anywhere).  */
                   2495: int
                   2496: hppa_can_use_return_insn_p ()
                   2497: {
                   2498:   return (reload_completed
                   2499:          && (compute_frame_size (get_frame_size (), 0) ? 0 : 1)
                   2500:          && ! profile_flag
                   2501:          && ! regs_ever_live[2]
                   2502:          && ! frame_pointer_needed);
                   2503: }
                   2504: 
                   2505: void
                   2506: emit_bcond_fp (code, operand0)
                   2507:      enum rtx_code code;
                   2508:      rtx operand0;
                   2509: {
                   2510:   emit_jump_insn (gen_rtx (SET, VOIDmode, pc_rtx,
                   2511:                           gen_rtx (IF_THEN_ELSE, VOIDmode,
                   2512:                                    gen_rtx (code, VOIDmode, 
                   2513:                                             gen_rtx (REG, CCFPmode, 0),
                   2514:                                             const0_rtx),
                   2515:                                    gen_rtx (LABEL_REF, VOIDmode, operand0),
                   2516:                                    pc_rtx)));
                   2517: 
                   2518: }
                   2519: 
                   2520: rtx
                   2521: gen_cmp_fp (code, operand0, operand1)
                   2522:      enum rtx_code code;
                   2523:      rtx operand0, operand1;
                   2524: {
                   2525:   return gen_rtx (SET, VOIDmode, gen_rtx (REG, CCFPmode, 0),
                   2526:                  gen_rtx (code, CCFPmode, operand0, operand1));
                   2527: }
                   2528: 
                   2529: /* Adjust the cost of a scheduling dependency.  Return the new cost of
                   2530:    a dependency LINK or INSN on DEP_INSN.  COST is the current cost.  */
                   2531: 
                   2532: int
                   2533: pa_adjust_cost (insn, link, dep_insn, cost)
                   2534:      rtx insn;
                   2535:      rtx link;
                   2536:      rtx dep_insn;
                   2537:      int cost;
                   2538: {
                   2539:   if (! recog_memoized (insn))
                   2540:     return 0;
                   2541: 
                   2542:   if (REG_NOTE_KIND (link) == 0)
                   2543:     {
                   2544:       /* Data dependency; DEP_INSN writes a register that INSN reads some
                   2545:         cycles later.  */
                   2546: 
                   2547:       if (get_attr_type (insn) == TYPE_FPSTORE)
                   2548:        {
                   2549:          rtx pat = PATTERN (insn);
                   2550:          rtx dep_pat = PATTERN (dep_insn);
                   2551:          if (GET_CODE (pat) == PARALLEL)
                   2552:            {
                   2553:              /* This happens for the fstXs,mb patterns.  */
                   2554:              pat = XVECEXP (pat, 0, 0);
                   2555:            }
                   2556:          if (GET_CODE (pat) != SET || GET_CODE (dep_pat) != SET)
                   2557:            /* If this happens, we have to extend this to schedule
                   2558:               optimally.  Return 0 for now.  */
                   2559:          return 0;
                   2560: 
                   2561:          if (rtx_equal_p (SET_DEST (dep_pat), SET_SRC (pat)))
                   2562:            {
                   2563:              if (! recog_memoized (dep_insn))
                   2564:                return 0;
                   2565:              /* DEP_INSN is writing its result to the register
                   2566:                 being stored in the fpstore INSN.  */
                   2567:              switch (get_attr_type (dep_insn))
                   2568:                {
                   2569:                case TYPE_FPLOAD:
                   2570:                  /* This cost 3 cycles, not 2 as the md says.  */
                   2571:                  return cost + 1;
                   2572: 
                   2573:                case TYPE_FPALU:
                   2574:                case TYPE_FPMUL:
                   2575:                case TYPE_FPDIVSGL:
                   2576:                case TYPE_FPDIVDBL:
                   2577:                case TYPE_FPSQRTSGL:
                   2578:                case TYPE_FPSQRTDBL:
                   2579:                  /* In these important cases, we save one cycle compared to
                   2580:                     when flop instruction feed each other.  */
                   2581:                  return cost - 1;
                   2582: 
                   2583:                default:
                   2584:                  return cost;
                   2585:                }
                   2586:            }
                   2587:        }
                   2588: 
                   2589:       /* For other data dependencies, the default cost specified in the
                   2590:         md is correct.  */
                   2591:       return cost;
                   2592:     }
                   2593:   else if (REG_NOTE_KIND (link) == REG_DEP_ANTI)
                   2594:     {
                   2595:       /* Anti dependency; DEP_INSN reads a register that INSN writes some
                   2596:         cycles later.  */
                   2597: 
                   2598:       if (get_attr_type (insn) == TYPE_FPLOAD)
                   2599:        {
                   2600:          rtx pat = PATTERN (insn);
                   2601:          rtx dep_pat = PATTERN (dep_insn);
                   2602:          if (GET_CODE (pat) == PARALLEL)
                   2603:            {
                   2604:              /* This happens for the fldXs,mb patterns.  */
                   2605:              pat = XVECEXP (pat, 0, 0);
                   2606:            }
                   2607:          if (GET_CODE (pat) != SET || GET_CODE (dep_pat) != SET)
                   2608:            /* If this happens, we have to extend this to schedule
                   2609:               optimally.  Return 0 for now.  */
                   2610:          return 0;
                   2611: 
                   2612:          if (reg_mentioned_p (SET_DEST (pat), SET_SRC (dep_pat)))
                   2613:            {
                   2614:              if (! recog_memoized (dep_insn))
                   2615:                return 0;
                   2616:              switch (get_attr_type (dep_insn))
                   2617:                {
                   2618:                case TYPE_FPALU:
                   2619:                case TYPE_FPMUL:
                   2620:                case TYPE_FPDIVSGL:
                   2621:                case TYPE_FPDIVDBL:
                   2622:                case TYPE_FPSQRTSGL:
                   2623:                case TYPE_FPSQRTDBL:
                   2624:                  /* A fpload can't be issued until one cycle before a
                   2625:                     preceeding arithmetic operation has finished, if
                   2626:                     the target of the fpload is any of the sources
                   2627:                     (or destination) of the arithmetic operation.  */
                   2628:                  return cost - 1;
                   2629: 
                   2630:                default:
                   2631:                  return 0;
                   2632:                }
                   2633:            }
                   2634:        }
                   2635: 
                   2636:       /* For other anti dependencies, the cost is 0.  */
                   2637:       return 0;
                   2638:     }
                   2639: 
                   2640:   /* For output dependencies, the cost is often one too high.  */
                   2641:   return cost - 1;
                   2642: }
                   2643: 
                   2644: /* Return any length adjustment needed by INSN which already has its length
                   2645:    computed as LENGTH.   Return zero if no adjustment is necessary. 
                   2646: 
                   2647:    For the PA: function calls, millicode calls, and backwards short
                   2648:    conditional branches with unfilled delay slots need an adjustment by +1 
                   2649:    (to account for the NOP which will be inserted into the instruction stream).
                   2650: 
                   2651:    Also compute the length of an inline block move here as it is too
                   2652:    complicated to express as a length attribute in pa.md.  */
                   2653: int
                   2654: pa_adjust_insn_length (insn, length)
                   2655:     rtx insn;
                   2656:     int length;
                   2657: {
                   2658:   rtx pat = PATTERN (insn);
                   2659: 
                   2660:   /* Call insns which are *not* indirect and have unfilled delay slots.  */
                   2661:   if (GET_CODE (insn) == CALL_INSN)
                   2662:     {
                   2663: 
                   2664:       if (GET_CODE (XVECEXP (pat, 0, 0)) == CALL
                   2665:          && GET_CODE (XEXP (XEXP (XVECEXP (pat, 0, 0), 0), 0)) == SYMBOL_REF)
                   2666:        return 4;
                   2667:       else if (GET_CODE (XVECEXP (pat, 0, 0)) == SET
                   2668:               && GET_CODE (XEXP (XEXP (XEXP (XVECEXP (pat, 0, 0), 1), 0), 0))
                   2669:                  == SYMBOL_REF)
                   2670:        return 4;
                   2671:       else
                   2672:        return 0;
                   2673:     }
                   2674:   /* Millicode insn with an unfilled delay slot.  */
                   2675:   else if (GET_CODE (insn) == INSN
                   2676:           && GET_CODE (pat) != SEQUENCE
                   2677:           && GET_CODE (pat) != USE
                   2678:           && GET_CODE (pat) != CLOBBER
                   2679:           && get_attr_type (insn) == TYPE_MILLI)
                   2680:     return 4;
                   2681:   /* Block move pattern.  */
                   2682:   else if (GET_CODE (insn) == INSN
                   2683:           && GET_CODE (pat) == PARALLEL
                   2684:           && GET_CODE (XEXP (XVECEXP (pat, 0, 0), 0)) == MEM
                   2685:           && GET_CODE (XEXP (XVECEXP (pat, 0, 0), 1)) == MEM
                   2686:           && GET_MODE (XEXP (XVECEXP (pat, 0, 0), 0)) == BLKmode
                   2687:           && GET_MODE (XEXP (XVECEXP (pat, 0, 0), 1)) == BLKmode)
                   2688:     return compute_movstrsi_length (insn) - 4;
                   2689:   /* Conditional branch with an unfilled delay slot.  */
                   2690:   else if (GET_CODE (insn) == JUMP_INSN && ! simplejump_p (insn))
                   2691:     {
                   2692:       /* Adjust a short backwards conditional with an unfilled delay slot.  */
                   2693:       if (GET_CODE (pat) == SET
                   2694:          && length == 4
                   2695:          && ! forward_branch_p (insn))
                   2696:        return 4;
                   2697:       /* Adjust dbra insn with short backwards conditional branch with
                   2698:         unfilled delay slot -- only for case where counter is in a 
                   2699:         general register register. */
                   2700:       else if (GET_CODE (pat) == PARALLEL
                   2701:               && GET_CODE (XVECEXP (pat, 0, 1)) == SET
                   2702:               && GET_CODE (XEXP (XVECEXP (pat, 0, 1), 0)) == REG
                   2703:               && ! FP_REG_P (XEXP (XVECEXP (pat, 0, 1), 0)) 
                   2704:               && length == 4
                   2705:               && ! forward_branch_p (insn))
                   2706:        return 4;
                   2707:       else
                   2708:        return 0;
                   2709:     }
                   2710:   else
                   2711:     return 0;
                   2712: }
                   2713: 
                   2714: /* Print operand X (an rtx) in assembler syntax to file FILE.
                   2715:    CODE is a letter or dot (`z' in `%z0') or 0 if no letter was specified.
                   2716:    For `%' followed by punctuation, CODE is the punctuation and X is null.  */
                   2717: 
                   2718: void
                   2719: print_operand (file, x, code)
                   2720:      FILE *file;
                   2721:      rtx x;
                   2722:      int code;
                   2723: {
                   2724:   switch (code)
                   2725:     {
                   2726:     case '\'':
                   2727:       fputc (PA_QUOTE, file);
                   2728:       return;
                   2729:       break;
                   2730: 
                   2731:     case '#':
                   2732:       /* Output a 'nop' if there's nothing for the delay slot.  */
                   2733:       if (dbr_sequence_length () == 0)
                   2734:        fputs ("\n\tnop", file);
                   2735:       return;
                   2736:     case '*':
                   2737:       /* Output an nullification completer if there's nothing for the */
                   2738:       /* delay slot or nullification is requested.  */ 
                   2739:       if (dbr_sequence_length () == 0 ||
                   2740:          (final_sequence &&
                   2741:           INSN_ANNULLED_BRANCH_P (XVECEXP (final_sequence, 0, 0))))
                   2742:         fputs (",n", file);
                   2743:       return;
                   2744:     case 'R':
                   2745:       /* Print out the second register name of a register pair.
                   2746:         I.e., R (6) => 7.  */
                   2747:       fputs (reg_names[REGNO (x)+1], file);
                   2748:       return;
                   2749:     case 'r':
                   2750:       /* A register or zero. */
                   2751:       if (x == const0_rtx
                   2752:          || (x == CONST0_RTX (DFmode))
                   2753:          || (x == CONST0_RTX (SFmode)))
                   2754:        {
                   2755:          fputs ("0", file);
                   2756:          return;
                   2757:        }
                   2758:       else
                   2759:        break;
                   2760:     case 'C':                  /* Plain (C)ondition */
                   2761:     case 'X':
                   2762:       switch (GET_CODE (x))
                   2763:        {       
                   2764:        case EQ:
                   2765:          fprintf (file, "=");  break;
                   2766:        case NE:
                   2767:          fprintf (file, "<>");  break;
                   2768:        case GT:
                   2769:          fprintf (file, ">");  break;
                   2770:        case GE:
                   2771:          fprintf (file, ">=");  break;
                   2772:        case GEU:
                   2773:          fprintf (file, ">>=");  break;
                   2774:        case GTU:
                   2775:          fprintf (file, ">>");  break;
                   2776:        case LT:
                   2777:          fprintf (file, "<");  break;
                   2778:        case LE:
                   2779:          fprintf (file, "<=");  break;
                   2780:        case LEU:
                   2781:          fprintf (file, "<<=");  break;
                   2782:        case LTU:
                   2783:          fprintf (file, "<<");  break;
                   2784:        default:
                   2785:          printf ("Can't grok '%c' operator:\n", code);
                   2786:          debug_rtx (x);
                   2787:          abort ();
                   2788:        }
                   2789:       return;
                   2790:     case 'N':                  /* Condition, (N)egated */
                   2791:       switch (GET_CODE (x))
                   2792:        {
                   2793:        case EQ:
                   2794:          fprintf (file, "<>");  break;
                   2795:        case NE:
                   2796:          fprintf (file, "=");  break;
                   2797:        case GT:
                   2798:          fprintf (file, "<=");  break;
                   2799:        case GE:
                   2800:          fprintf (file, "<");  break;
                   2801:        case GEU:
                   2802:          fprintf (file, "<<");  break;
                   2803:        case GTU:
                   2804:          fprintf (file, "<<=");  break;
                   2805:        case LT:
                   2806:          fprintf (file, ">=");  break;
                   2807:        case LE:
                   2808:          fprintf (file, ">");  break;
                   2809:        case LEU:
                   2810:          fprintf (file, ">>");  break;
                   2811:        case LTU:
                   2812:          fprintf (file, ">>=");  break;
                   2813:        default:
                   2814:          printf ("Can't grok '%c' operator:\n", code);
                   2815:          debug_rtx (x);
                   2816:          abort ();
                   2817:        }
                   2818:       return;
                   2819:     /* For floating point comparisons.  Need special conditions to deal
                   2820:        with NaNs properly.  */
                   2821:     case 'Y':
                   2822:       switch (GET_CODE (x))
                   2823:        {
                   2824:        case EQ:
                   2825:          fprintf (file, "!=");  break;
                   2826:        case NE:
                   2827:          fprintf (file, "=");  break;
                   2828:        case GT:
                   2829:          fprintf (file, "!>");  break;
                   2830:        case GE:
                   2831:          fprintf (file, "!>=");  break;
                   2832:        case LT:
                   2833:          fprintf (file, "!<");  break;
                   2834:        case LE:
                   2835:          fprintf (file, "!<=");  break;
                   2836:        default:
                   2837:          printf ("Can't grok '%c' operator:\n", code);
                   2838:          debug_rtx (x);
                   2839:          abort ();
                   2840:        }
                   2841:       return;
                   2842:     case 'S':                  /* Condition, operands are (S)wapped.  */
                   2843:       switch (GET_CODE (x))
                   2844:        {
                   2845:        case EQ:
                   2846:          fprintf (file, "=");  break;
                   2847:        case NE:
                   2848:          fprintf (file, "<>");  break;
                   2849:        case GT:
                   2850:          fprintf (file, "<");  break;
                   2851:        case GE:
                   2852:          fprintf (file, "<=");  break;
                   2853:        case GEU:
                   2854:          fprintf (file, "<<=");  break;
                   2855:        case GTU:
                   2856:          fprintf (file, "<<");  break;
                   2857:        case LT:
                   2858:          fprintf (file, ">");  break;
                   2859:        case LE:
                   2860:          fprintf (file, ">=");  break;
                   2861:        case LEU:
                   2862:          fprintf (file, ">>=");  break;
                   2863:        case LTU:
                   2864:          fprintf (file, ">>");  break;
                   2865:        default:
                   2866:          printf ("Can't grok '%c' operator:\n", code);
                   2867:          debug_rtx (x);
                   2868:          abort ();
                   2869:        }         
                   2870:       return;
                   2871:     case 'B':                  /* Condition, (B)oth swapped and negate.  */
                   2872:       switch (GET_CODE (x))
                   2873:        {
                   2874:        case EQ:
                   2875:          fprintf (file, "<>");  break;
                   2876:        case NE:
                   2877:          fprintf (file, "=");  break;
                   2878:        case GT:
                   2879:          fprintf (file, ">=");  break;
                   2880:        case GE:
                   2881:          fprintf (file, ">");  break;
                   2882:        case GEU:
                   2883:          fprintf (file, ">>");  break;
                   2884:        case GTU:
                   2885:          fprintf (file, ">>=");  break;
                   2886:        case LT:
                   2887:          fprintf (file, "<=");  break;
                   2888:        case LE:
                   2889:          fprintf (file, "<");  break;
                   2890:        case LEU:
                   2891:          fprintf (file, "<<");  break;
                   2892:        case LTU:
                   2893:          fprintf (file, "<<=");  break;
                   2894:        default:
                   2895:          printf ("Can't grok '%c' operator:\n", code);
                   2896:          debug_rtx (x);
                   2897:          abort ();
                   2898:        }         
                   2899:       return;
                   2900:     case 'k':
                   2901:       if (GET_CODE (x) == CONST_INT)
                   2902:        {
                   2903:          fprintf (file, "%d", ~INTVAL (x));
                   2904:          return;
                   2905:        }
                   2906:       abort();
                   2907:     case 'L':
                   2908:       if (GET_CODE (x) == CONST_INT)
                   2909:        {
                   2910:          fprintf (file, "%d", 32 - (INTVAL (x) & 31));
                   2911:          return;
                   2912:        }
                   2913:       abort();
                   2914:     case 'O':
                   2915:       if (GET_CODE (x) == CONST_INT && exact_log2 (INTVAL (x)) >= 0)
                   2916:        {
                   2917:          fprintf (file, "%d", exact_log2 (INTVAL (x)));
                   2918:          return;
                   2919:        }
                   2920:       abort();
                   2921:     case 'P':
                   2922:       if (GET_CODE (x) == CONST_INT)
                   2923:        {
                   2924:          fprintf (file, "%d", 31 - (INTVAL (x) & 31));
                   2925:          return;
                   2926:        }
                   2927:       abort();
                   2928:     case 'I':
                   2929:       if (GET_CODE (x) == CONST_INT)
                   2930:        fputs ("i", file);
                   2931:       return;
                   2932:     case 'M':
                   2933:       switch (GET_CODE (XEXP (x, 0)))
                   2934:        {
                   2935:        case PRE_DEC:
                   2936:        case PRE_INC:
                   2937:          fprintf (file, "s,mb");
                   2938:          break;
                   2939:        case POST_DEC:
                   2940:        case POST_INC:
                   2941:          fprintf (file, "s,ma");
                   2942:          break;
                   2943:        default:
                   2944:          break;
                   2945:        }
                   2946:       return;
                   2947:     case 'F':
                   2948:       switch (GET_CODE (XEXP (x, 0)))
                   2949:        {
                   2950:        case PRE_DEC:
                   2951:        case PRE_INC:
                   2952:          fprintf (file, ",mb");
                   2953:          break;
                   2954:        case POST_DEC:
                   2955:        case POST_INC:
                   2956:          fprintf (file, ",ma");
                   2957:          break;
                   2958:        default:
                   2959:          break;
                   2960:        }
                   2961:       return;
                   2962:     case 'G':
                   2963:       output_global_address (file, x);
                   2964:       return;
                   2965:     case 0:                    /* Don't do anything special */
                   2966:       break;
                   2967:     case 'Z':
                   2968:       {
                   2969:        unsigned op[3];
                   2970:        compute_zdepi_operands (INTVAL (x), op);
                   2971:        fprintf (file, "%d,%d,%d", op[0], op[1], op[2]);
                   2972:        return;
                   2973:       }
                   2974:     default:
                   2975:       abort ();
                   2976:     }
                   2977:   if (GET_CODE (x) == REG)
                   2978:     fprintf (file, "%s", reg_names [REGNO (x)]);
                   2979:   else if (GET_CODE (x) == MEM)
                   2980:     {
                   2981:       int size = GET_MODE_SIZE (GET_MODE (x));
                   2982:       rtx base = XEXP (XEXP (x, 0), 0);
                   2983:       switch (GET_CODE (XEXP (x, 0)))
                   2984:        {
                   2985:        case PRE_DEC:
                   2986:        case POST_DEC:
                   2987:          fprintf (file, "-%d(0,%s)", size, reg_names [REGNO (base)]);
                   2988:          break;
                   2989:        case PRE_INC:
                   2990:        case POST_INC:
                   2991:          fprintf (file, "%d(0,%s)", size, reg_names [REGNO (base)]);
                   2992:          break;
                   2993:        default:
                   2994:          output_address (XEXP (x, 0));
                   2995:          break;
                   2996:        }
                   2997:     }
                   2998:   else if (GET_CODE (x) == CONST_DOUBLE && GET_MODE (x) == SFmode)
                   2999:     {
                   3000:       union { double d; int i[2]; } u;
                   3001:       union { float f; int i; } u1;
                   3002:       u.i[0] = XINT (x, 0); u.i[1] = XINT (x, 1);
                   3003:       u1.f = u.d;
                   3004:       if (code == 'f')
                   3005:        fprintf (file, "0r%.9g", u1.f);
                   3006:       else
                   3007:        fprintf (file, "0x%x", u1.i);
                   3008:     }
                   3009:   else if (GET_CODE (x) == CONST_DOUBLE && GET_MODE (x) != DImode)
                   3010:     {
                   3011:       union { double d; int i[2]; } u;
                   3012:       u.i[0] = XINT (x, 0); u.i[1] = XINT (x, 1);
                   3013:       fprintf (file, "0r%.20g", u.d);
                   3014:     }
                   3015:   else
                   3016:     output_addr_const (file, x);
                   3017: }
                   3018: 
                   3019: /* output a SYMBOL_REF or a CONST expression involving a SYMBOL_REF. */
                   3020: 
                   3021: void
                   3022: output_global_address (file, x)
                   3023:      FILE *file;
                   3024:      rtx x;
                   3025: {
                   3026: 
                   3027:   /* Imagine  (high (const (plus ...))).  */
                   3028:   if (GET_CODE (x) == HIGH)
                   3029:     x = XEXP (x, 0);
                   3030: 
                   3031:   if (GET_CODE (x) == SYMBOL_REF && read_only_operand (x))
                   3032:     assemble_name (file, XSTR (x, 0));
                   3033:   else if (GET_CODE (x) == SYMBOL_REF)
                   3034:     {
                   3035:       assemble_name (file, XSTR (x, 0));
                   3036: #ifndef NeXT_ASM
                   3037:       fprintf (file, "-$global$");
                   3038: #endif
                   3039:     }
                   3040:   else if (GET_CODE (x) == CONST)
                   3041:     {
                   3042:       char *sep = "";
                   3043:       int offset = 0;          /* assembler wants -$global$ at end */
                   3044:       rtx base;
                   3045:          
                   3046:       if (GET_CODE (XEXP (XEXP (x, 0), 0)) == SYMBOL_REF)
                   3047:        {
                   3048:          base = XEXP (XEXP (x, 0), 0);
                   3049:          output_addr_const (file, base);
                   3050:        }
                   3051:       else if (GET_CODE (XEXP (XEXP (x, 0), 0)) == CONST_INT)
                   3052:        offset = INTVAL (XEXP (XEXP (x, 0), 0));
                   3053:       else abort ();
                   3054: 
                   3055: #ifdef NeXT_ASM
                   3056:       if (GET_CODE (XEXP (x, 0)) == MINUS)
                   3057:        fprintf (file, "-");
                   3058: #endif
                   3059: 
                   3060:       if (GET_CODE (XEXP (XEXP (x, 0), 1)) == SYMBOL_REF)
                   3061:        {
                   3062:          base = XEXP (XEXP (x, 0), 1);
                   3063:          output_addr_const (file, base);
                   3064:        }
                   3065:       else if (GET_CODE (XEXP (XEXP (x, 0), 1)) == CONST_INT)
                   3066:        offset = INTVAL (XEXP (XEXP (x, 0),1));
                   3067:       else abort ();
                   3068: 
                   3069:       if (GET_CODE (XEXP (x, 0)) == PLUS)
                   3070:        {
                   3071:          if (offset < 0)
                   3072:            {
                   3073:              offset = -offset;
                   3074:              sep = "-";
                   3075:            }
                   3076:          else
                   3077:            sep = "+";
                   3078:        }
                   3079:       else if (GET_CODE (XEXP (x, 0)) == MINUS
                   3080:               && (GET_CODE (XEXP (XEXP (x, 0), 0)) == SYMBOL_REF))
                   3081:        sep = "-";
                   3082:       else abort ();
                   3083: 
                   3084: #ifndef NeXT_ASM
                   3085:       if (!read_only_operand (base))
                   3086:        fprintf (file, "-$global$");
                   3087: #endif
                   3088:       if (offset) 
                   3089:        {
                   3090:          fprintf (file,"%s%d", sep, offset);
                   3091:        }
                   3092:     }
                   3093:   else
                   3094:     output_addr_const (file, x);
                   3095: }
                   3096: 
                   3097: /* HP's millicode routines mean something special to the assembler.
                   3098:    Keep track of which ones we have used.  */
                   3099: 
                   3100: enum millicodes { remI, remU, divI, divU, mulI, mulU, end1000 };
                   3101: static char imported[(int)end1000];
                   3102: static char *milli_names[] = {"remI", "remU", "divI", "divU", "mulI", "mulU"};
                   3103: static char import_string[] = ".IMPORT $$....,MILLICODE";
                   3104: #define MILLI_START 10
                   3105: 
                   3106: static void
                   3107: import_milli (code)
                   3108:      enum millicodes code;
                   3109: {
                   3110: #ifndef NeXT_ASM
                   3111:   char str[sizeof (import_string)];
                   3112:   
                   3113:   if (!imported[(int)code])
                   3114:     {
                   3115:       imported[(int)code] = 1;
                   3116:       strcpy (str, import_string);
                   3117:       strncpy (str + MILLI_START, milli_names[(int)code], 4);
                   3118:       output_asm_insn (str, 0);
                   3119:     }
                   3120: #endif
                   3121: }
                   3122: 
                   3123: /* The register constraints have put the operands and return value in 
                   3124:    the proper registers. */
                   3125: 
                   3126: char *
                   3127: output_mul_insn (unsignedp, insn)
                   3128:      int unsignedp;
                   3129:      rtx insn;
                   3130: {
                   3131: 
                   3132:   if (unsignedp)
                   3133:     {
                   3134:       import_milli (mulU);
                   3135:       return output_call (insn, gen_rtx (SYMBOL_REF, SImode, "$$mulU"),
                   3136:                          gen_rtx (REG, SImode, 31));
                   3137:     }
                   3138:   else
                   3139:     {
                   3140:       import_milli (mulI);
                   3141:       return output_call (insn, gen_rtx (SYMBOL_REF, SImode, "$$mulI"),
                   3142:                          gen_rtx (REG, SImode, 31));
                   3143:     }
                   3144: }
                   3145: 
                   3146: /* If operands isn't NULL, then it's a CONST_INT with which we can do
                   3147:    something */
                   3148: 
                   3149: 
                   3150: /* Emit the rtl for doing a division by a constant. */
                   3151: 
                   3152:  /* Do magic division millicodes exist for this value? */
                   3153: 
                   3154: static int magic_milli[]= {0, 0, 0, 1, 0, 1, 1, 1, 0, 1, 1, 0, 1, 0,
                   3155:                             1, 1};
                   3156: 
                   3157: /* We'll use an array to keep track of the magic millicodes and 
                   3158:    whether or not we've used them already. [n][0] is signed, [n][1] is
                   3159:    unsigned. */
                   3160: 
                   3161: static int div_milli[16][2];
                   3162: 
                   3163: int
                   3164: div_operand (op, mode)
                   3165:      rtx op;
                   3166:      enum machine_mode mode;
                   3167: {
                   3168:   return (mode == SImode
                   3169:          && ((GET_CODE (op) == REG && REGNO (op) == 25)
                   3170:              || (GET_CODE (op) == CONST_INT && INTVAL (op) > 0
                   3171:                  && INTVAL (op) < 16 && magic_milli[INTVAL (op)])));
                   3172: }
                   3173: 
                   3174: int
                   3175: emit_hpdiv_const (operands, unsignedp)
                   3176:      rtx *operands;
                   3177:      int unsignedp;
                   3178: {
                   3179:   if (GET_CODE (operands[2]) == CONST_INT
                   3180:       && INTVAL (operands[2]) > 0
                   3181:       && INTVAL (operands[2]) < 16
                   3182:       && magic_milli[INTVAL (operands[2])])
                   3183:     {
                   3184:       emit_move_insn ( gen_rtx (REG, SImode, 26), operands[1]);
                   3185:       emit
                   3186:        (gen_rtx
                   3187:         (PARALLEL, VOIDmode,
                   3188:          gen_rtvec (6, gen_rtx (SET, VOIDmode, gen_rtx (REG, SImode, 29),
                   3189:                                 gen_rtx (unsignedp ? UDIV : DIV, SImode,
                   3190:                                          gen_rtx (REG, SImode, 26),
                   3191:                                          operands[2])),
                   3192:                     gen_rtx (CLOBBER, VOIDmode, operands[3]),
                   3193:                     gen_rtx (CLOBBER, VOIDmode, gen_rtx (REG, SImode, 26)),
                   3194:                     gen_rtx (CLOBBER, VOIDmode, gen_rtx (REG, SImode, 25)),
                   3195:                     gen_rtx (CLOBBER, VOIDmode, gen_rtx (REG, SImode, 19)),
                   3196:                     gen_rtx (CLOBBER, VOIDmode, gen_rtx (REG, SImode, 31)))));
                   3197:       emit_move_insn (operands[0], gen_rtx (REG, SImode, 29));
                   3198:       return 1;
                   3199:     }
                   3200:   return 0;
                   3201: }
                   3202: 
                   3203: char *
                   3204: output_div_insn (operands, unsignedp, insn)
                   3205:      rtx *operands;
                   3206:      int unsignedp;
                   3207:      rtx insn;
                   3208: {
                   3209:   int divisor;
                   3210:   
                   3211:   /* If the divisor is a constant, try to use one of the special 
                   3212:      opcodes .*/
                   3213:   if (GET_CODE (operands[0]) == CONST_INT)
                   3214:     {
                   3215:       static char buf[100];
                   3216:       divisor = INTVAL (operands[0]);
                   3217:       if (!div_milli[divisor][unsignedp])
                   3218:        {
                   3219:          div_milli[divisor][unsignedp] = 1;
                   3220: #ifndef NeXT_ASM
                   3221:          if (unsignedp)
                   3222:            output_asm_insn (".IMPORT $$divU_%0,MILLICODE", operands);
                   3223:          else
                   3224:            output_asm_insn (".IMPORT $$divI_%0,MILLICODE", operands);
                   3225: #endif
                   3226:        }
                   3227:       if (unsignedp)
                   3228:        {
                   3229:          sprintf (buf, "$$divU_%d", INTVAL (operands[0]));
                   3230:          return output_call (insn, gen_rtx (SYMBOL_REF, SImode, buf),
                   3231:                              gen_rtx (REG, SImode, 31));
                   3232:        }
                   3233:       else
                   3234:        {
                   3235:          sprintf (buf, "$$divI_%d", INTVAL (operands[0]));
                   3236:          return output_call (insn, gen_rtx (SYMBOL_REF, SImode, buf),
                   3237:                              gen_rtx (REG, SImode, 31));
                   3238:        }
                   3239:     }
                   3240:   /* Divisor isn't a special constant. */
                   3241:   else
                   3242:     {
                   3243:       if (unsignedp)
                   3244:        {
                   3245:          import_milli (divU);
                   3246:          return output_call (insn, gen_rtx (SYMBOL_REF, SImode, "$$divU"),
                   3247:                              gen_rtx (REG, SImode, 31));
                   3248:        }
                   3249:       else
                   3250:        {
                   3251:          import_milli (divI);
                   3252:          return output_call (insn, gen_rtx (SYMBOL_REF, SImode, "$$divI"),
                   3253:                              gen_rtx (REG, SImode, 31));
                   3254:        }
                   3255:     }
                   3256: }
                   3257: 
                   3258: /* Output a $$rem millicode to do mod. */
                   3259: 
                   3260: char *
                   3261: output_mod_insn (unsignedp, insn)
                   3262:      int unsignedp;
                   3263:      rtx insn;
                   3264: {
                   3265:   if (unsignedp)
                   3266:     {
                   3267:       import_milli (remU);
                   3268:       return output_call (insn, gen_rtx (SYMBOL_REF, SImode, "$$remU"),
                   3269:                          gen_rtx (REG, SImode, 31));
                   3270:     }
                   3271:   else
                   3272:     {
                   3273:       import_milli (remI);
                   3274:       return output_call (insn, gen_rtx (SYMBOL_REF, SImode, "$$remI"),
                   3275:                          gen_rtx (REG, SImode, 31));
                   3276:     }
                   3277: }
                   3278: 
                   3279: void
                   3280: output_arg_descriptor (insn)
                   3281:      rtx insn;
                   3282: {
                   3283:   char *arg_regs[4];
                   3284:   enum machine_mode arg_mode;
                   3285:   rtx prev_insn;
                   3286:   int i, output_flag = 0;
                   3287:   int regno;
                   3288:   
                   3289:   for (i = 0; i < 4; i++)
                   3290:     arg_regs[i] = 0;
                   3291: 
                   3292:   for (prev_insn = PREV_INSN (insn); GET_CODE (prev_insn) == INSN;
                   3293:        prev_insn = PREV_INSN (prev_insn))
                   3294:     {
                   3295:       /* Terminate search for arguments if a non-USE insn is encountered
                   3296:         or a USE insn which does not specify an argument, STATIC_CHAIN,
                   3297:         or STRUCT_VALUE register.  */
                   3298:       if (!(GET_CODE (PATTERN (prev_insn)) == USE
                   3299:            && GET_CODE (XEXP (PATTERN (prev_insn), 0)) == REG
                   3300:            && (FUNCTION_ARG_REGNO_P (REGNO (XEXP (PATTERN (prev_insn), 0)))
                   3301:                || REGNO (XEXP (PATTERN (prev_insn), 0)) == STATIC_CHAIN_REGNUM
                   3302:                || REGNO (XEXP (PATTERN (prev_insn), 0))
                   3303:                == STRUCT_VALUE_REGNUM)))
                   3304:        break;
                   3305: 
                   3306:       /* If this is a USE for the STATIC_CHAIN or STRUCT_VALUE register,
                   3307:         then skip it and continue the loop since those are not encoded
                   3308:         in the argument relocation bits.  */
                   3309:       if (REGNO (XEXP (PATTERN (prev_insn), 0)) == STATIC_CHAIN_REGNUM
                   3310:          || REGNO (XEXP (PATTERN (prev_insn), 0)) == STRUCT_VALUE_REGNUM)
                   3311:        continue;
                   3312: 
                   3313:       arg_mode = GET_MODE (XEXP (PATTERN (prev_insn), 0));
                   3314:       regno = REGNO (XEXP (PATTERN (prev_insn), 0));
                   3315:       if (regno >= 23 && regno <= 26)
                   3316:        {
                   3317:          arg_regs[26 - regno] = "GR";
                   3318:          if (arg_mode == DImode)
                   3319:            arg_regs[25 - regno] = "GR";
                   3320:        }
                   3321:       else if (!TARGET_SNAKE)  /* fp args */
                   3322:        {
                   3323:          if (arg_mode == SFmode)
                   3324:            arg_regs[regno - 32] = "FR";
                   3325:          else
                   3326:            {
                   3327: #ifndef HP_FP_ARG_DESCRIPTOR_REVERSED
                   3328:              arg_regs[regno - 33] = "FR";
                   3329:              arg_regs[regno - 32] = "FU";
                   3330: #else
                   3331:              arg_regs[regno - 33] = "FU";
                   3332:              arg_regs[regno - 32] = "FR";
                   3333: #endif
                   3334:            }
                   3335:        }
                   3336:       else
                   3337:        {
                   3338:          if (arg_mode == SFmode)
                   3339:            arg_regs[(regno - 44) / 2] = "FR";
                   3340:          else
                   3341:            {
                   3342: #ifndef HP_FP_ARG_DESCRIPTOR_REVERSED
                   3343:              arg_regs[(regno - 46) / 2] = "FR";
                   3344:              arg_regs[(regno - 46) / 2 + 1] = "FU";
                   3345: #else
                   3346:              arg_regs[(regno - 46) / 2] = "FU";
                   3347:              arg_regs[(regno - 46) / 2 + 1] = "FR";
                   3348: #endif
                   3349:            }
                   3350:        }
                   3351:     }
                   3352: #ifndef NeXT_ASM
                   3353:   fputs ("\t.CALL ", asm_out_file);
                   3354:   for (i = 0; i < 4; i++)
                   3355:     {
                   3356:       if (arg_regs[i])
                   3357:        {
                   3358:          if (output_flag++)
                   3359:            fputc (',', asm_out_file);
                   3360:          fprintf (asm_out_file, "ARGW%d=%s", i, arg_regs[i]);
                   3361:        }
                   3362:     }
                   3363:   fputc ('\n', asm_out_file);
                   3364: #endif
                   3365: }
                   3366: 
                   3367: /* Memory loads/stores to/from the shift need to go through
                   3368:    the general registers.  */
                   3369: 
                   3370: enum reg_class
                   3371: secondary_reload_class (class, mode, in)
                   3372:      enum reg_class class;
                   3373:      enum machine_mode mode;
                   3374:      rtx in;
                   3375: {
                   3376:   int regno = true_regnum (in);
                   3377: 
                   3378:   if (function_label_operand (in, mode)
                   3379:       || ((regno >= FIRST_PSEUDO_REGISTER || regno == -1)
                   3380:          && GET_MODE_CLASS (mode) == MODE_INT
                   3381:          && FP_REG_CLASS_P (class))
                   3382:       || (class == SHIFT_REGS && (regno <= 0 || regno >= 32)))
                   3383:     return GENERAL_REGS;
                   3384: 
                   3385:   if (GET_CODE (in) == HIGH)
                   3386:     in = XEXP (in, 0);
                   3387: 
                   3388:   if (class != R1_REGS && symbolic_operand (in, VOIDmode))
                   3389:     return R1_REGS;
                   3390: 
                   3391:   return NO_REGS;
                   3392: }
                   3393: 
                   3394: enum direction
                   3395: function_arg_padding (mode, type)
                   3396:      enum machine_mode mode;
                   3397:      tree type;
                   3398: {
                   3399:   int size;
                   3400: 
                   3401:   if (mode == BLKmode)
                   3402:     {
                   3403:       if (type && TREE_CODE (TYPE_SIZE (type)) == INTEGER_CST)
                   3404:        size = int_size_in_bytes (type) * BITS_PER_UNIT;
                   3405:       else
                   3406:        return upward;          /* Don't know if this is right, but */
                   3407:                                /* same as old definition. */
                   3408:     }
                   3409:   else
                   3410:     size = GET_MODE_BITSIZE (mode);
                   3411:   if (size < PARM_BOUNDARY)
                   3412:     return downward;
                   3413:   else if (size % PARM_BOUNDARY)
                   3414:     return upward;
                   3415:   else
                   3416:     return none;
                   3417: }
                   3418: 
                   3419: 
                   3420: /* Do what is necessary for `va_start'.  The argument is ignored;
                   3421:    We look at the current function to determine if stdargs or varargs
                   3422:    is used and fill in an initial va_list.  A pointer to this constructor
                   3423:    is returned.  */
                   3424: 
                   3425: struct rtx_def *
                   3426: hppa_builtin_saveregs (arglist)
                   3427:      tree arglist;
                   3428: {
                   3429:   rtx offset;
                   3430:   tree fntype = TREE_TYPE (current_function_decl);
                   3431:   int argadj = ((!(TYPE_ARG_TYPES (fntype) != 0
                   3432:                   && (TREE_VALUE (tree_last (TYPE_ARG_TYPES (fntype)))
                   3433:                       != void_type_node)))
                   3434:                ? UNITS_PER_WORD : 0);
                   3435: 
                   3436:   if (argadj)
                   3437:     offset = plus_constant (current_function_arg_offset_rtx, argadj);
                   3438:   else
                   3439:     offset = current_function_arg_offset_rtx;
                   3440: 
                   3441:   /* Store general registers on the stack. */
                   3442:   move_block_from_reg (23,
                   3443:                       gen_rtx (MEM, BLKmode,
                   3444:                                plus_constant
                   3445:                                (current_function_internal_arg_pointer, -16)),
                   3446:                       4, 4 * UNITS_PER_WORD); 
                   3447:   return copy_to_reg (expand_binop (Pmode, add_optab,
                   3448:                                    current_function_internal_arg_pointer,
                   3449:                                    offset, 0, 0, OPTAB_LIB_WIDEN));
                   3450: }
                   3451: 
                   3452: /* This routine handles all the normal conditional branch sequences we 
                   3453:    might need to generate.  It handles compare immediate vs compare 
                   3454:    register, nullification of delay slots, varying length branches, 
                   3455:    negated branches, and all combinations of the above.  It returns the
                   3456:    output appropriate to emit the branch corresponding to all given 
                   3457:    parameters.  */
                   3458: 
                   3459: char *
                   3460: output_cbranch (operands, nullify, length, negated, insn)
                   3461:   rtx *operands;
                   3462:   int nullify, length, negated;
                   3463:   rtx insn;
                   3464: {
                   3465:   static char buf[100];
                   3466:   int useskip = 0;
                   3467: 
                   3468:   /* A conditional branch to the following instruction (eg the delay slot) is
                   3469:      asking for a disaster.  This can happen when not optimizing.
                   3470: 
                   3471:      In such cases it is safe to emit nothing.  */
                   3472: 
                   3473:   if (JUMP_LABEL (insn) == next_nonnote_insn (insn))
                   3474:     return "";
                   3475:   
                   3476:   /* If this is a long branch with its delay slot unfilled, set `nullify'
                   3477:      as it can nullify the delay slot and save a nop.  */
                   3478:   if (length == 8 && dbr_sequence_length () == 0)
                   3479:     nullify = 1;
                   3480: 
                   3481:   /* If this is a short forward conditional branch which did not get
                   3482:      its delay slot filled, the delay slot can still be nullified.  */
                   3483:   if (! nullify && length == 4 && dbr_sequence_length () == 0)
                   3484:     nullify = forward_branch_p (insn);
                   3485: 
                   3486:   /* A forward branch over a single nullified insn can be done with a 
                   3487:      comclr instruction.  This avoids a single cycle penalty due to
                   3488:      mis-predicted branch if we fall through (branch not taken).  */
                   3489:   if (length == 4
                   3490:       && next_real_insn (insn) != 0
                   3491:       && get_attr_length (next_real_insn (insn)) == 4
                   3492:       && JUMP_LABEL (insn) == next_nonnote_insn (next_real_insn (insn))
                   3493:       && nullify)
                   3494:     useskip = 1;
                   3495: 
                   3496:   switch (length)
                   3497:     {
                   3498:       /* All short conditional branches except backwards with an unfilled
                   3499:         delay slot.  */
                   3500:       case 4:
                   3501:        if (useskip)
                   3502:          strcpy (buf, "com%I2clr,");
                   3503:        else
                   3504:          strcpy (buf, "com%I2b,");
                   3505:        if (negated)
                   3506:          strcat (buf, "%B3");
                   3507:        else
                   3508:          strcat (buf, "%S3");
                   3509:        if (useskip)
                   3510:          strcat (buf, " %2,%1,%%r0");
                   3511:        else if (nullify)
                   3512:          strcat (buf, ",n %2,%1,%0");
                   3513:        else 
                   3514:          strcat (buf, " %2,%1,%0");
                   3515:        break;
                   3516: 
                   3517:      /* All long conditionals.  Note an short backward branch with an 
                   3518:        unfilled delay slot is treated just like a long backward branch
                   3519:        with an unfilled delay slot.  */
                   3520:       case 8:
                   3521:        /* Handle weird backwards branch with a filled delay slot
                   3522:           with is nullified.  */
                   3523:        if (dbr_sequence_length () != 0
                   3524:            && ! forward_branch_p (insn)
                   3525:            && nullify)
                   3526:          {
                   3527:            strcpy (buf, "com%I2b,");
                   3528:            if (negated)
                   3529:              strcat (buf, "%S3");
                   3530:            else
                   3531:              strcat (buf, "%B3");
                   3532:            strcat (buf, ",n %2,%1,.+12\n\tbl %0,%%r0");
                   3533:          }
                   3534:        else
                   3535:          {
                   3536:            strcpy (buf, "com%I2clr,");
                   3537:            if (negated)
                   3538:              strcat (buf, "%S3");
                   3539:            else
                   3540:              strcat (buf, "%B3");
                   3541:            if (nullify)
                   3542:              strcat (buf, " %2,%1,%%r0\n\tbl,n %0,%%r0");
                   3543:            else
                   3544:              strcat (buf, " %2,%1,%%r0\n\tbl %0,%%r0");
                   3545:          }
                   3546:        break;
                   3547: 
                   3548:       default:
                   3549:        abort();
                   3550:     }
                   3551:   return buf;
                   3552: }
                   3553: 
                   3554: /* This routine handles all the branch-on-bit conditional branch sequences we 
                   3555:    might need to generate.  It handles nullification of delay slots,
                   3556:    varying length branches, negated branches and all combinations of the
                   3557:    above.  it returns the appropriate output template to emit the branch.  */
                   3558: 
                   3559: char *
                   3560: output_bb (operands, nullify, length, negated, insn, which)
                   3561:   rtx *operands;
                   3562:   int nullify, length, negated;
                   3563:   rtx insn;
                   3564:   int which;
                   3565: {
                   3566:   static char buf[100];
                   3567:   int useskip = 0;
                   3568: 
                   3569:   /* A conditional branch to the following instruction (eg the delay slot) is
                   3570:      asking for a disaster.  I do not think this can happen as this pattern
                   3571:      is only used when optimizing; jump optimization should eliminate the 
                   3572:      jump.  But be prepared just in case.  */
                   3573:      
                   3574:   if (JUMP_LABEL (insn) == next_nonnote_insn (insn))
                   3575:     return "";
                   3576:   
                   3577:   /* If this is a long branch with its delay slot unfilled, set `nullify'
                   3578:      as it can nullify the delay slot and save a nop.  */
                   3579:   if (length == 8 && dbr_sequence_length () == 0)
                   3580:     nullify = 1;
                   3581: 
                   3582:   /* If this is a short forward conditional branch which did not get
                   3583:      its delay slot filled, the delay slot can still be nullified.  */
                   3584:   if (! nullify && length == 4 && dbr_sequence_length () == 0)
                   3585:     nullify = forward_branch_p (insn);
                   3586: 
                   3587:   /* A forward branch over a single nullified insn can be done with a 
                   3588:      extrs instruction.  This avoids a single cycle penalty due to
                   3589:      mis-predicted branch if we fall through (branch not taken).  */
                   3590: 
                   3591:   if (length == 4
                   3592:       && next_real_insn (insn) != 0
                   3593:       && get_attr_length (next_real_insn (insn)) == 4
                   3594:       && JUMP_LABEL (insn) == next_nonnote_insn (next_real_insn (insn))
                   3595:       && nullify)
                   3596:     useskip = 1;
                   3597: 
                   3598:   switch (length)
                   3599:     {
                   3600: 
                   3601:       /* All short conditional branches except backwards with an unfilled
                   3602:         delay slot.  */
                   3603:       case 4:
                   3604:        if (useskip)
                   3605:          strcpy (buf, "extrs,");
                   3606:        else 
                   3607:          strcpy (buf, "bb,");
                   3608:        if ((which == 0 && negated)
                   3609:             || (which == 1 && ! negated))
                   3610:          strcat (buf, ">=");
                   3611:        else
                   3612:          strcat (buf, "<");
                   3613:        if (useskip)
                   3614:          strcat (buf, " %0,%1,1,0");
                   3615:        else if (nullify && negated)
                   3616:          strcat (buf, ",n %0,%1,%3");
                   3617:        else if (nullify && ! negated)
                   3618:          strcat (buf, ",n %0,%1,%2");
                   3619:        else if (! nullify && negated)
                   3620:          strcat (buf, "%0,%1,%3");
                   3621:        else if (! nullify && ! negated)
                   3622:          strcat (buf, " %0,%1,%2");
                   3623:        break;
                   3624: 
                   3625:      /* All long conditionals.  Note an short backward branch with an 
                   3626:        unfilled delay slot is treated just like a long backward branch
                   3627:        with an unfilled delay slot.  */
                   3628:       case 8:
                   3629:        /* Handle weird backwards branch with a filled delay slot
                   3630:           with is nullified.  */
                   3631:        if (dbr_sequence_length () != 0
                   3632:            && ! forward_branch_p (insn)
                   3633:            && nullify)
                   3634:          {
                   3635:            strcpy (buf, "bb,");
                   3636:            if ((which == 0 && negated)
                   3637:                || (which == 1 && ! negated))
                   3638:              strcat (buf, "<");
                   3639:            else
                   3640:              strcat (buf, ">=");
                   3641:            if (negated)
                   3642:              strcat (buf, " %0,%1,.+12\n\tbl %3,%%r0");
                   3643:            else
                   3644:              strcat (buf, " %0,%1,.+12\n\tbl %2,%%r0");
                   3645:          }
                   3646:        else
                   3647:          {
                   3648:            strcpy (buf, "extrs,");
                   3649:            if ((which == 0 && negated)
                   3650:                || (which == 1 && ! negated))
                   3651:              strcat (buf, "<");
                   3652:            else
                   3653:              strcat (buf, ">=");
                   3654:            if (nullify && negated)
                   3655:              strcat (buf, " %0,%1,1,0\n\tbl,n %3,%%r0");
                   3656:            else if (nullify && ! negated)
                   3657:              strcat (buf, " %0,%1,1,0\n\tbl,n %2,%%r0");
                   3658:            else if (negated)
                   3659:              strcat (buf, " %0,%1,1,0\n\tbl %3,%%r0");
                   3660:            else 
                   3661:              strcat (buf, " %0,%1,1,0\n\tbl %2,%%r0");
                   3662:          }
                   3663:        break;
                   3664: 
                   3665:       default:
                   3666:        abort();
                   3667:     }
                   3668:   return buf;
                   3669: }
                   3670: 
                   3671: /* Return the output template for emitting a dbra type insn.
                   3672: 
                   3673:    Note it may perform some output operations on its own before
                   3674:    returning the final output string.  */
                   3675: char *
                   3676: output_dbra (operands, insn, which_alternative)
                   3677:      rtx *operands;
                   3678:      rtx insn;
                   3679:      int which_alternative;
                   3680: {
                   3681: 
                   3682:   /* A conditional branch to the following instruction (eg the delay slot) is
                   3683:      asking for a disaster.  Be prepared!  */
                   3684: 
                   3685:   if (JUMP_LABEL (insn) == next_nonnote_insn (insn))
                   3686:     {
                   3687:       if (which_alternative == 0)
                   3688:        return "ldo %1(%0),%0";
                   3689:       else if (which_alternative == 1)
                   3690:        {
                   3691:          output_asm_insn ("fstws %0,-16(0,%%r30)",operands);
                   3692:          output_asm_insn ("ldw -16(0,%%r30),%4",operands);
                   3693:          output_asm_insn ("ldo %1(%4),%4\n\tstw %4,-16(0,%%r30)", operands);
                   3694:          return "fldws -16(0,%%r30),%0";
                   3695:        }
                   3696:       else
                   3697:        {
                   3698:          output_asm_insn ("ldw %0,%4", operands);
                   3699:          return "ldo %1(%4),%4\n\tstw %4,%0";
                   3700:        }
                   3701:     }
                   3702: 
                   3703:   if (which_alternative == 0)
                   3704:     {
                   3705:       int nullify = INSN_ANNULLED_BRANCH_P (insn);
                   3706:       int length = get_attr_length (insn);
                   3707: 
                   3708:       /* If this is a long branch with its delay slot unfilled, set `nullify'
                   3709:         as it can nullify the delay slot and save a nop.  */
                   3710:       if (length == 8 && dbr_sequence_length () == 0)
                   3711:        nullify = 1;
                   3712: 
                   3713:       /* If this is a short forward conditional branch which did not get
                   3714:         its delay slot filled, the delay slot can still be nullified.  */
                   3715:       if (! nullify && length == 4 && dbr_sequence_length () == 0)
                   3716:        nullify = forward_branch_p (insn);
                   3717: 
                   3718:       /* Handle short versions first.  */
                   3719:       if (length == 4 && nullify)
                   3720:        return "addib,%C2,n %1,%0,%3";
                   3721:       else if (length == 4 && ! nullify)
                   3722:        return "addib,%C2 %1,%0,%3";
                   3723:       else if (length == 8)
                   3724:        {
                   3725:          /* Handle weird backwards branch with a fulled delay slot 
                   3726:             which is nullified.  */
                   3727:          if (dbr_sequence_length () != 0
                   3728:              && ! forward_branch_p (insn)
                   3729:              && nullify)
                   3730:            return "addib,%N2,n %1,%0,.+12\n\tbl %3,0";
                   3731:          
                   3732:          /* Handle normal cases.  */  
                   3733:          if (nullify)
                   3734:            return "addi,%N2 %1,%0,%0\n\tbl,n %3,0";
                   3735:          else
                   3736:            return "addi,%N2 %1,%0,%0\n\tbl %3,0";
                   3737:        }
                   3738:       else
                   3739:        abort();
                   3740:     }
                   3741:   /* Deal with gross reload from FP register case.  */
                   3742:   else if (which_alternative == 1)
                   3743:     {
                   3744:       /* Move loop counter from FP register to MEM then into a GR,
                   3745:         increment the GR, store the GR into MEM, and finally reload
                   3746:         the FP register from MEM from within the branch's delay slot.  */ 
                   3747:       output_asm_insn ("fstws %0,-16(0,%%r30)\n\tldw -16(0,%%r30),%4",operands);
                   3748:       output_asm_insn ("ldo %1(%4),%4\n\tstw %4,-16(0,%%r30)", operands);
                   3749:       if (get_attr_length (insn) == 24)
                   3750:        return "comb,%S2 0,%4,%3\n\tfldws -16(0,%%r30),%0";
                   3751:       else
                   3752:        return "comclr,%B2 0,%4,0\n\tbl %3,0\n\tfldws -16(0,%%r30),%0";
                   3753:     }
                   3754:   /* Deal with gross reload from memory case.  */
                   3755:   else
                   3756:     {
                   3757:       /* Reload loop counter from memory, the store back to memory
                   3758:         happens in the branch's delay slot.   */
                   3759:       output_asm_insn ("ldw %0,%4", operands);
                   3760:       if (get_attr_length (insn) == 12)
                   3761:        return "addib,%C2 %1,%4,%3\n\tstw %4,%0";
                   3762:       else
                   3763:        return "addi,%N2 %1,%4,%4\n\tbl %3,0\n\tstw %4,%0";
                   3764:     }
                   3765: }
                   3766: 
                   3767: /* Return the output template for emitting a dbra type insn.
                   3768: 
                   3769:    Note it may perform some output operations on its own before
                   3770:    returning the final output string.  */
                   3771: char *
                   3772: output_movb (operands, insn, which_alternative, reverse_comparison)
                   3773:      rtx *operands;
                   3774:      rtx insn;
                   3775:      int which_alternative;
                   3776:      int reverse_comparison;
                   3777: {
                   3778: 
                   3779:   /* A conditional branch to the following instruction (eg the delay slot) is
                   3780:      asking for a disaster.  Be prepared!  */
                   3781: 
                   3782:   if (JUMP_LABEL (insn) == next_nonnote_insn (insn))
                   3783:     {
                   3784:       if (which_alternative == 0)
                   3785:        return "copy %1,%0";
                   3786:       else if (which_alternative == 1)
                   3787:        {
                   3788:          output_asm_insn ("stw %1,-16(0,%%r30)",operands);
                   3789:          return "fldws -16(0,%%r30),%0";
                   3790:        }
                   3791:       else
                   3792:        return "stw %1,%0";
                   3793:     }
                   3794: 
                   3795:   /* Support the second variant.  */
                   3796:   if (reverse_comparison)
                   3797:     PUT_CODE (operands[2], reverse_condition (GET_CODE (operands[2])));
                   3798: 
                   3799:   if (which_alternative == 0)
                   3800:     {
                   3801:       int nullify = INSN_ANNULLED_BRANCH_P (insn);
                   3802:       int length = get_attr_length (insn);
                   3803: 
                   3804:       /* If this is a long branch with its delay slot unfilled, set `nullify'
                   3805:         as it can nullify the delay slot and save a nop.  */
                   3806:       if (length == 8 && dbr_sequence_length () == 0)
                   3807:        nullify = 1;
                   3808: 
                   3809:       /* If this is a short forward conditional branch which did not get
                   3810:         its delay slot filled, the delay slot can still be nullified.  */
                   3811:       if (! nullify && length == 4 && dbr_sequence_length () == 0)
                   3812:        nullify = forward_branch_p (insn);
                   3813: 
                   3814:       /* Handle short versions first.  */
                   3815:       if (length == 4 && nullify)
                   3816:        return "movb,%C2,n %1,%0,%3";
                   3817:       else if (length == 4 && ! nullify)
                   3818:        return "movb,%C2 %1,%0,%3";
                   3819:       else if (length == 8)
                   3820:        {
                   3821:          /* Handle weird backwards branch with a fulled delay slot 
                   3822:             which is nullified.  */
                   3823:          if (dbr_sequence_length () != 0
                   3824:              && ! forward_branch_p (insn)
                   3825:              && nullify)
                   3826:            return "movb,%N2,n %1,%0,.+12\n\ttbl %3,0";
                   3827:          
                   3828:          /* Handle normal cases.  */  
                   3829:          if (nullify)
                   3830:            return "or,%N2 %1,%%r0,%0\n\tbl,n %3,0";
                   3831:          else
                   3832:            return "or,%N2 %1,%%r0,%0\n\tbl %3,0";
                   3833:        }
                   3834:       else
                   3835:        abort();
                   3836:     }
                   3837:   /* Deal with gross reload from FP register case.  */
                   3838:   else if (which_alternative == 1)
                   3839:     {
                   3840:       /* Move loop counter from FP register to MEM then into a GR,
                   3841:         increment the GR, store the GR into MEM, and finally reload
                   3842:         the FP register from MEM from within the branch's delay slot.  */ 
                   3843:       output_asm_insn ("stw %1,-16(0,%%r30)",operands);
                   3844:       if (get_attr_length (insn) == 12)
                   3845:        return "comb,%S2 0,%1,%3\n\tfldws -16(0,%%r30),%0";
                   3846:       else
                   3847:        return "comclr,%B2 0,%1,0\n\tbl %3,0\n\tfldws -16(0,%%r30),%0";
                   3848:     }
                   3849:   /* Deal with gross reload from memory case.  */
                   3850:   else
                   3851:     {
                   3852:       /* Reload loop counter from memory, the store back to memory
                   3853:         happens in the branch's delay slot.   */
                   3854:       if (get_attr_length (insn) == 8)
                   3855:        return "comb,%S2 0,%1,%3\n\tstw %1,%0";
                   3856:       else
                   3857:        return "comclr,%B2 0,%1,0\n\tbl %3,0\n\tstw %1,%0";
                   3858:     }
                   3859: }
                   3860: 
                   3861: 
                   3862: /* Output a local call insn not emitting insns for the delay slot. 
                   3863:    If -mstub-calls is enabled, we emit the pseudo insn "jbsr", to
                   3864:    a locally generated stub, and make that stub do the actual call.
                   3865:    
                   3866:    CALL_DEST is the routine we are calling.
                   3867: 
                   3868:    RETURN_POINTER is the register which will hold the return address.
                   3869:    %r2 for most calls, %r31 for millicode calls.  */
                   3870: #ifdef NeXT_ASM
                   3871: void add_compiler_stub PROTO((tree, tree, int));
                   3872: void output_compiler_stub PROTO ((void));
                   3873: int no_previous_def PROTO((tree));
                   3874: tree get_prev_label PROTO((tree));
                   3875: 
                   3876: #endif
                   3877: 
                   3878: void
                   3879: output_call_insn (insn, call_dest, return_pointer)
                   3880:      rtx insn;
                   3881:      rtx call_dest;
                   3882:      rtx return_pointer;
                   3883: 
                   3884: {
                   3885:   rtx xoperands[2];
                   3886: 
                   3887:   xoperands[0] = call_dest;
                   3888:   xoperands[1] = return_pointer;
                   3889: 
                   3890: #ifdef NeXT_ASM
                   3891:   if(GET_CODE (call_dest) == SYMBOL_REF)
                   3892:     {
                   3893:       rtx prev_insn, label_rtx;
                   3894:       int line_number;
                   3895:       static char buf[256];
                   3896:       static char temp_buf[256];
                   3897:       char *label_buf;
                   3898:       tree labelname;
                   3899:       tree funname = get_identifier (XSTR (call_dest, 0));
                   3900:       
                   3901:       if (!flag_pic && !strncmp (IDENTIFIER_POINTER (funname), "$$", 2))
                   3902:        {
                   3903:            {
                   3904:              strcpy (temp_buf, "ldil L%'");
                   3905:              strcat (temp_buf, IDENTIFIER_POINTER (funname));
                   3906:              strcat (temp_buf, ",%%r31\n\tble R%'");
                   3907:              strcat (temp_buf, IDENTIFIER_POINTER (funname));
                   3908:              strcat (temp_buf, "(4,%%r31)");
                   3909:              output_asm_insn (temp_buf, 0);
                   3910:            }
                   3911:          return;
                   3912:        }
                   3913:       else if (!strncmp (IDENTIFIER_POINTER (funname), "$$", 2))
                   3914:        {
                   3915:          strcpy (temp_buf, "*");
                   3916:          strcat (temp_buf, XSTR (call_dest, 0));
                   3917:          funname = get_identifier (temp_buf);
                   3918:        }
                   3919:       if (no_previous_def (funname))
                   3920:        {
                   3921:          label_rtx = gen_label_rtx ();
                   3922:          
                   3923:          ASM_GENERATE_INTERNAL_LABEL (temp_buf, "L", CODE_LABEL_NUMBER(label_rtx));
                   3924:          
                   3925:          if (temp_buf[0] == '*')
                   3926:            label_buf = temp_buf + 1;
                   3927:          else
                   3928:            label_buf = temp_buf;
                   3929: 
                   3930:          while (insn && GET_CODE (insn) != NOTE)
                   3931:            insn = PREV_INSN (insn);
                   3932:       
                   3933:          if (insn)
                   3934:            line_number = NOTE_LINE_NUMBER (insn);
                   3935:       
                   3936:          labelname =get_identifier (label_buf);
                   3937:          add_compiler_stub(labelname, funname, line_number);
                   3938:        }
                   3939:       else
                   3940:        {
                   3941:          labelname = get_prev_label (funname);
                   3942:        }
                   3943: 
                   3944:       strcpy(buf, "jbsr %0,%r1,");
                   3945:       strcat(buf, IDENTIFIER_POINTER (labelname));
                   3946: 
                   3947:       output_asm_insn (buf, xoperands);
                   3948:     }
                   3949:   else
                   3950: #endif
                   3951:     output_asm_insn ("bl %0,%r1", xoperands);
                   3952: }
                   3953: 
                   3954: /* INSN is either a function call or a millicode call.  It may have an
                   3955:    unconditional jump in its delay slot.  
                   3956: 
                   3957:    CALL_DEST is the routine we are calling.
                   3958: 
                   3959:    RETURN_POINTER is the register which will hold the return address.
                   3960:    %r2 for most calls, %r31 for millicode calls.  */
                   3961: char *
                   3962: output_call (insn, call_dest, return_pointer)
                   3963:   rtx insn;
                   3964:   rtx call_dest;
                   3965:   rtx return_pointer;
                   3966: 
                   3967: {
                   3968:   int distance;
                   3969:   rtx xoperands[4];
                   3970:   rtx seq_insn;
                   3971: 
                   3972:   /* Handle common case -- empty delay slot or no jump in the delay slot.  */
                   3973:   if (dbr_sequence_length () == 0
                   3974:       || (dbr_sequence_length () != 0 
                   3975:          && GET_CODE (NEXT_INSN (insn)) != JUMP_INSN))
                   3976:     {
                   3977:       output_call_insn (insn, call_dest, return_pointer);
                   3978:       if (dbr_sequence_length () == 0)
                   3979:        output_asm_insn ("nop", xoperands);
                   3980:       return "";
                   3981:     }
                   3982:     
                   3983:   /* This call has an unconditional jump in its delay slot.  */
                   3984: 
                   3985:   /* Use the containing sequence insn's address.  */
                   3986:   seq_insn = NEXT_INSN (PREV_INSN (XVECEXP (final_sequence, 0, 0)));
                   3987: 
                   3988:   distance = insn_addresses[INSN_UID (JUMP_LABEL (NEXT_INSN (insn)))] 
                   3989:               - insn_addresses[INSN_UID (seq_insn)] - 8;
                   3990: 
                   3991:   /* If the branch was too far away, emit a normal call followed
                   3992:      by a nop, followed by the unconditional branch.
                   3993: 
                   3994:      If the branch is close, then adjust %r2 from within the 
                   3995:      call's delay slot.  */
                   3996: 
                   3997:   xoperands[0] = call_dest;
                   3998:   xoperands[1] = XEXP (PATTERN (NEXT_INSN (insn)), 1);
                   3999:   xoperands[2] = return_pointer;
                   4000:   if (! VAL_14_BITS_P (distance))
                   4001:     {
                   4002:       output_call_insn (insn, call_dest, return_pointer);
                   4003:       output_asm_insn ("nop\n\tbl,n %1,%%r0", xoperands);
                   4004:     }
                   4005:   else
                   4006:     {
                   4007:       xoperands[3] = gen_label_rtx ();
                   4008:       ASM_OUTPUT_INTERNAL_LABEL (asm_out_file, "L", 
                   4009:                                 CODE_LABEL_NUMBER (xoperands[3]));
                   4010:       output_call_insn (insn, call_dest, return_pointer);
                   4011:       output_asm_insn ("ldo %1-%3-8(%r2),%r2", xoperands);
                   4012:     }
                   4013: 
                   4014:   /* Delete the jump.  */
                   4015:   PUT_CODE (NEXT_INSN (insn), NOTE);
                   4016:   NOTE_LINE_NUMBER (NEXT_INSN (insn)) = NOTE_INSN_DELETED;
                   4017:   NOTE_SOURCE_FILE (NEXT_INSN (insn)) = 0;
                   4018:   return "";
                   4019: }
                   4020: 
                   4021: extern struct obstack *saveable_obstack;
                   4022: 
                   4023: /* In HPUX 8.0's shared library scheme, special relocations are needed
                   4024:    for function labels if they might be passed to a function 
                   4025:    in a shared library (because shared libraries don't live in code
                   4026:    space), and special magic is needed to construct their address. */
                   4027: 
                   4028: void
                   4029: hppa_encode_label (sym)
                   4030:      rtx sym;
                   4031: {
                   4032:   char *str = XSTR (sym, 0);
                   4033:   int len = strlen (str);
                   4034:   char *newstr = obstack_alloc (saveable_obstack, len + 2) ;
                   4035: 
                   4036:   if (str[0] == '*')
                   4037:     *newstr++ = *str++;
                   4038:   strcpy (newstr + 1, str);
                   4039:   *newstr = '@';
                   4040:   XSTR (sym,0) = newstr;
                   4041: }
                   4042:   
                   4043: int
                   4044: function_label_operand  (op, mode)
                   4045:      rtx op;
                   4046:      enum machine_mode mode;
                   4047: {
                   4048:   return GET_CODE (op) == SYMBOL_REF && FUNCTION_NAME_P (XSTR (op, 0));
                   4049: }
                   4050: 
                   4051: /* Returns 1 if the 6 operands specified in OPERANDS are suitable for
                   4052:    use in fmpyadd instructions.  */
                   4053: int
                   4054: fmpyaddoperands(operands)
                   4055:      rtx *operands;
                   4056: {
                   4057:   enum machine_mode mode = GET_MODE (operands[0]);
                   4058: 
                   4059:   /* All modes must be the same.  */
                   4060:   if (! (mode == GET_MODE (operands[1])
                   4061:         && mode == GET_MODE (operands[2])
                   4062:         && mode == GET_MODE (operands[3])
                   4063:         && mode == GET_MODE (operands[4])
                   4064:         && mode == GET_MODE (operands[5])))
                   4065:     return 0;
                   4066: 
                   4067:   /* Both DFmode and SFmode should work.  But using SFmode makes the
                   4068:      assembler complain.  Just turn it off for now.  */
                   4069:   if (mode != DFmode)
                   4070:     return 0;
                   4071: 
                   4072:   /* Only 2 real operands to the addition.  One of the input operands must
                   4073:      be the same as the output operand.  */
                   4074:   if (! rtx_equal_p (operands[3], operands[4])
                   4075:       && ! rtx_equal_p (operands[3], operands[5]))
                   4076:     return 0;
                   4077: 
                   4078:   /* Inout operand of add can not conflict with any operands from multiply.  */
                   4079:   if (rtx_equal_p (operands[3], operands[0])
                   4080:      || rtx_equal_p (operands[3], operands[1])
                   4081:      || rtx_equal_p (operands[3], operands[2]))
                   4082:     return 0;
                   4083: 
                   4084:   /* multiply can not feed into addition operands.  */
                   4085:   if (rtx_equal_p (operands[4], operands[0])
                   4086:       || rtx_equal_p (operands[5], operands[0]))
                   4087:     return 0;
                   4088: 
                   4089:   /* Passed.  Operands are suitable for fmpyadd.  */
                   4090:   return 1;
                   4091: }
                   4092: 
                   4093: /* Returns 1 if the 6 operands specified in OPERANDS are suitable for
                   4094:    use in fmpysub instructions.  */
                   4095: int
                   4096: fmpysuboperands(operands)
                   4097:      rtx *operands;
                   4098: {
                   4099:   enum machine_mode mode = GET_MODE (operands[0]);
                   4100: 
                   4101:   /* All modes must be the same.  */
                   4102:   if (! (mode == GET_MODE (operands[1])
                   4103:         && mode == GET_MODE (operands[2])
                   4104:         && mode == GET_MODE (operands[3])
                   4105:         && mode == GET_MODE (operands[4])
                   4106:         && mode == GET_MODE (operands[5])))
                   4107:     return 0;
                   4108: 
                   4109:   /* Both DFmode and SFmode should work.  But using SFmode makes the
                   4110:      assembler complain.  Just turn it off for now.  */
                   4111:   if (mode != DFmode)
                   4112:     return 0;
                   4113: 
                   4114:   /* Only 2 real operands to the subtraction.  Subtraction is not a commutative
                   4115:      operation, so operands[4] must be the same as operand[3].  */
                   4116:   if (! rtx_equal_p (operands[3], operands[4]))
                   4117:     return 0;
                   4118: 
                   4119:   /* multiply can not feed into subtraction.  */
                   4120:   if (rtx_equal_p (operands[5], operands[0]))
                   4121:     return 0;
                   4122: 
                   4123:   /* Inout operand of sub can not conflict with any operands from multiply.  */
                   4124:   if (rtx_equal_p (operands[3], operands[0])
                   4125:      || rtx_equal_p (operands[3], operands[1])
                   4126:      || rtx_equal_p (operands[3], operands[2]))
                   4127:     return 0;
                   4128: 
                   4129:   /* Passed.  Operands are suitable for fmpysub.  */
                   4130:   return 1;
                   4131: }
                   4132: 
                   4133: int
                   4134: plus_xor_ior_operator (op, mode)
                   4135:      rtx op;
                   4136:      enum machine_mode mode;
                   4137: {
                   4138:   return (GET_CODE (op) == PLUS || GET_CODE (op) == XOR
                   4139:          || GET_CODE (op) == IOR);
                   4140: }
                   4141: 
                   4142: /* Return 1 if the given constant is 2, 4, or 8.  These are the valid
                   4143:    constants for shadd instructions.  */
                   4144: int
                   4145: shadd_constant_p (val)
                   4146:      int val;
                   4147: {
                   4148:   if (val == 2 || val == 4 || val == 8)
                   4149:     return 1;
                   4150:   else
                   4151:     return 0;
                   4152: }
                   4153: 
                   4154: /* Return 1 if OP is a CONST_INT with the value 2, 4, or 8.  These are
                   4155:    the valid constant for shadd instructions.  */
                   4156: int
                   4157: shadd_operand (op, mode)
                   4158:      rtx op;
                   4159:      enum machine_mode mode;
                   4160: {
                   4161:   return (GET_CODE (op) == CONST_INT && shadd_constant_p (INTVAL (op)));
                   4162: }
                   4163: 
                   4164: /* Return 1 if INSN branches forward.  Should be using insn_addresses
                   4165:    to avoid walking through all the insns... */
                   4166: int
                   4167: forward_branch_p (insn)
                   4168:      rtx insn;
                   4169: {
                   4170:   rtx label = JUMP_LABEL (insn);
                   4171: 
                   4172:   while (insn)
                   4173:     {
                   4174:       if (insn == label)
                   4175:        break;
                   4176:       else
                   4177:        insn = NEXT_INSN (insn);
                   4178:     }
                   4179: 
                   4180:   return (insn == label);
                   4181: }
                   4182: 
                   4183: /* Return 1 if OP is an equality comparison, else return 0.  */
                   4184: int
                   4185: eq_neq_comparison_operator (op, mode)
                   4186:      rtx op;
                   4187:      enum machine_mode mode;
                   4188: {
                   4189:   return (GET_CODE (op) == EQ || GET_CODE (op) == NE);
                   4190: }
                   4191: 
                   4192: /* Return 1 if OP is an operator suitable for use in a movb instruction.  */
                   4193: int
                   4194: movb_comparison_operator (op, mode)
                   4195:      rtx op;
                   4196:      enum machine_mode mode;
                   4197: {
                   4198:   return (GET_CODE (op) == EQ || GET_CODE (op) == NE
                   4199:          || GET_CODE (op) == LT || GET_CODE (op) == GE);
                   4200: }
                   4201: 
                   4202: /* Return 1 if INSN is in the delay slot of a call instruction.  */
                   4203: int
                   4204: jump_in_call_delay (insn)
                   4205:      rtx insn;
                   4206: {
                   4207: 
                   4208:   if (GET_CODE (insn) != JUMP_INSN)
                   4209:     return 0;
                   4210: 
                   4211:   if (PREV_INSN (insn)
                   4212:       && PREV_INSN (PREV_INSN (insn))
                   4213:       && GET_CODE (next_active_insn (PREV_INSN (PREV_INSN (insn)))) == INSN)
                   4214:     {
                   4215:       rtx test_insn = next_active_insn (PREV_INSN (PREV_INSN (insn)));
                   4216: 
                   4217:       return (GET_CODE (PATTERN (test_insn)) == SEQUENCE
                   4218:              && XVECEXP (PATTERN (test_insn), 0, 1) == insn);
                   4219: 
                   4220:     }
                   4221:   else
                   4222:     return 0;
                   4223: }
                   4224: 

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.