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

unix.superglobalmegacorp.com

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