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

1.1       root        1: /* Subroutines for insn-output.c for Motorola 68000 family.
                      2:    Copyright (C) 1987, 1993 Free Software Foundation, Inc.
                      3: 
                      4: This file is part of GNU CC.
                      5: 
                      6: GNU CC is free software; you can redistribute it and/or modify
                      7: it under the terms of the GNU General Public License as published by
                      8: the Free Software Foundation; either version 2, or (at your option)
                      9: any later version.
                     10: 
                     11: GNU CC is distributed in the hope that it will be useful,
                     12: but WITHOUT ANY WARRANTY; without even the implied warranty of
                     13: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     14: GNU General Public License for more details.
                     15: 
                     16: You should have received a copy of the GNU General Public License
                     17: along with GNU CC; see the file COPYING.  If not, write to
                     18: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
                     19: 
                     20: 
                     21: /* Some output-actions in m68k.md need these.  */
                     22: #include <stdio.h>
                     23: #include "config.h"
                     24: #include "tree.h"
                     25: #include "rtl.h"
                     26: #include "regs.h"
                     27: #include "hard-reg-set.h"
                     28: #include "real.h"
                     29: #include "insn-config.h"
                     30: #include "conditions.h"
                     31: #include "insn-flags.h"
                     32: #include "output.h"
                     33: #include "insn-attr.h"
                     34: 
                     35: #include "machopic.h"
                     36: 
                     37: /* Needed for use_return_insn.  */
                     38: #include "flags.h"
                     39: 
                     40: #ifdef SUPPORT_SUN_FPA
                     41: 
                     42: /* Index into this array by (register number >> 3) to find the
                     43:    smallest class which contains that register.  */
                     44: enum reg_class regno_reg_class[]
                     45:   = { DATA_REGS, ADDR_REGS, FP_REGS,
                     46:       LO_FPA_REGS, LO_FPA_REGS, FPA_REGS, FPA_REGS };
                     47: 
                     48: #endif /* defined SUPPORT_SUN_FPA */
                     49: 
                     50: /* This flag is used to communicate between movhi and ASM_OUTPUT_CASE_END,
                     51:    if SGS_SWITCH_TABLE.  */
                     52: int switch_table_difference_label_flag;
                     53: 
                     54: static rtx find_addr_reg ();
                     55: rtx legitimize_pic_address ();
                     56: 
                     57: 
                     58: /* Emit a (use pic_offset_table_rtx) if we used PIC relocation in the 
                     59:    function at any time during the compilation process.  In the future 
                     60:    we should try and eliminate the USE if we can easily determine that 
                     61:    all PIC references were deleted from the current function.  That would 
                     62:    save an address register */
                     63:    
                     64: void
                     65: finalize_pic ()
                     66: {
                     67:   if (flag_pic && current_function_uses_pic_offset_table)
                     68:     emit_insn (gen_rtx (USE, VOIDmode, pic_offset_table_rtx));
                     69: }
                     70: 
                     71: 
                     72: /* This function generates the assembly code for function entry.
                     73:    STREAM is a stdio stream to output the code to.
                     74:    SIZE is an int: how many units of temporary storage to allocate.
                     75:    Refer to the array `regs_ever_live' to determine which registers
                     76:    to save; `regs_ever_live[I]' is nonzero if register number I
                     77:    is ever used in the function.  This function is responsible for
                     78:    knowing which registers should not be saved even if used.  */
                     79: 
                     80: 
                     81: /* Note that the order of the bit mask for fmovem is the opposite
                     82:    of the order for movem!  */
                     83: 
                     84: 
                     85: void
                     86: output_function_prologue (stream, size)
                     87:      FILE *stream;
                     88:      int size;
                     89: {
                     90:   register int regno;
                     91:   register int mask = 0;
                     92:   int num_saved_regs = 0;
                     93:   extern char call_used_regs[];
                     94:   int fsize = (size + 3) & -4;
                     95:   
                     96: 
                     97:   if (frame_pointer_needed)
                     98:     {
                     99:       /* Adding negative number is faster on the 68040.  */
                    100:       if (fsize < 0x8000 && !TARGET_68040)
                    101:        {
                    102: #ifdef MOTOROLA
                    103:          asm_fprintf (stream, "\tlink.w %s,%0I%d\n",
                    104:                       reg_names[FRAME_POINTER_REGNUM], -fsize);
                    105: #else
                    106:          asm_fprintf (stream, "\tlink %s,%0I%d\n",
                    107:                       reg_names[FRAME_POINTER_REGNUM], -fsize);
                    108: #endif
                    109:        }
                    110:       else if (TARGET_68020)
                    111:        {
                    112: #ifdef MOTOROLA
                    113:          asm_fprintf (stream, "\tlink.l %s,%0I%d\n",
                    114:                       reg_names[FRAME_POINTER_REGNUM], -fsize);
                    115: #else
                    116:          asm_fprintf (stream, "\tlink %s,%0I%d\n",
                    117:                       reg_names[FRAME_POINTER_REGNUM], -fsize);
                    118: #endif
                    119:        }
                    120:       else
                    121:        {
                    122: #ifdef MOTOROLA
                    123:          asm_fprintf (stream, "\tlink.w %s,%0I0\n\tadd.l %0I%d,%Rsp\n",
                    124:                       reg_names[FRAME_POINTER_REGNUM], -fsize);
                    125: #else
                    126:          asm_fprintf (stream, "\tlink %s,%0I0\n\taddl %0I%d,%Rsp\n",
                    127:                       reg_names[FRAME_POINTER_REGNUM], -fsize);
                    128: #endif
                    129:        }
                    130:     }
                    131:   else if (fsize)
                    132:     {
                    133:       /* Adding negative number is faster on the 68040.  */
                    134:       if (fsize + 4 < 0x8000)
                    135:        {
                    136: #ifdef MOTOROLA
                    137:          asm_fprintf (stream, "\tadd.w %0I%d,%Rsp\n", - (fsize + 4));
                    138: #else
                    139:          asm_fprintf (stream, "\taddw %0I%d,%Rsp\n", - (fsize + 4));
                    140: #endif
                    141:        }
                    142:       else
                    143:        {
                    144: #ifdef MOTOROLA
                    145:          asm_fprintf (stream, "\tadd.l %0I%d,%Rsp\n", - (fsize + 4));
                    146: #else
                    147:          asm_fprintf (stream, "\taddl %0I%d,%Rsp\n", - (fsize + 4));
                    148: #endif
                    149:        }
                    150:     }
                    151: #ifdef SUPPORT_SUN_FPA
                    152:   for (regno = 24; regno < 56; regno++)
                    153:     if (regs_ever_live[regno] && ! call_used_regs[regno])
                    154:       {
                    155: #ifdef MOTOROLA
                    156:        asm_fprintf (stream, "\tfpmovd %s,-(%Rsp)\n",
                    157:                     reg_names[regno]);
                    158: #else
                    159:        asm_fprintf (stream, "\tfpmoved %s,%Rsp@-\n",
                    160:                     reg_names[regno]);
                    161: #endif
                    162:       }
                    163: #endif
                    164:   for (regno = 16; regno < 24; regno++)
                    165:     if (regs_ever_live[regno] && ! call_used_regs[regno])
                    166:        mask |= 1 << (regno - 16);
                    167:   if ((mask & 0xff) != 0)
                    168:     {
                    169: #ifdef MOTOROLA
                    170:       asm_fprintf (stream, "\tfmovm %0I0x%x,-(%Rsp)\n", mask & 0xff);
                    171: #else
                    172:       asm_fprintf (stream, "\tfmovem %0I0x%x,%Rsp@-\n", mask & 0xff);
                    173: #endif
                    174:     }
                    175:   mask = 0;
                    176:   for (regno = 0; regno < 16; regno++)
                    177:     if (regs_ever_live[regno] && ! call_used_regs[regno])
                    178:       {
                    179:         mask |= 1 << (15 - regno);
                    180:         num_saved_regs++;
                    181:       }
                    182:   if (frame_pointer_needed)
                    183:     {
                    184:       mask &= ~ (1 << (15 - FRAME_POINTER_REGNUM));
                    185:       num_saved_regs--;
                    186:     }
                    187: 
                    188: #if NEED_PROBE
                    189:   fprintf (stream, "\ttstl sp@(%d)\n", NEED_PROBE - num_saved_regs * 4);
                    190: #endif
                    191: 
                    192:   if (num_saved_regs <= 2)
                    193:     {
                    194:       /* Store each separately in the same order moveml uses.
                    195:          Using two movel instructions instead of a single moveml
                    196:          is about 15% faster for the 68020 and 68030 at no expense
                    197:          in code size */
                    198: 
                    199:       int i;
                    200: 
                    201:       /* Undo the work from above. */
                    202:       for (i = 0; i< 16; i++)
                    203:         if (mask & (1 << i))
                    204:           asm_fprintf (stream,
                    205: #ifdef MOTOROLA
                    206:                       "\t%Omove.l %s,-(%Rsp)\n",
                    207: #else
                    208:                       "\tmovel %s,%Rsp@-\n",
                    209: #endif
                    210:                       reg_names[15 - i]);
                    211:     }
                    212:   else if (mask)
                    213:     {
                    214: #ifdef MOTOROLA
                    215:       asm_fprintf (stream, "\tmovm.l %0I0x%x,-(%Rsp)\n", mask);
                    216: #else
                    217:       asm_fprintf (stream, "\tmoveml %0I0x%x,%Rsp@-\n", mask);
                    218: #endif
                    219:     }
                    220:   if (flag_pic && current_function_uses_pic_offset_table)
                    221:     {
                    222: #ifdef MACHO_PIC
                    223:       assemble_name (stream, machopic_function_base_name ());
                    224:       asm_fprintf (stream, ":\n\tlea %Rpc@(-2),%s\n",
                    225:                   reg_names[PIC_OFFSET_TABLE_REGNUM]);
                    226: #else
                    227: #ifdef MOTOROLA
                    228:       asm_fprintf (stream, "\t%Olea (%Rpc, %U_GLOBAL_OFFSET_TABLE_@GOTPC), %s\n",
                    229:                   reg_names[PIC_OFFSET_TABLE_REGNUM]);
                    230: #else
                    231:       asm_fprintf (stream, "\tmovel %0I__GLOBAL_OFFSET_TABLE_, %s\n",
                    232:                   reg_names[PIC_OFFSET_TABLE_REGNUM]);
                    233:       asm_fprintf (stream, "\tlea %Rpc@(0,%s:l),%s\n",
                    234:                   reg_names[PIC_OFFSET_TABLE_REGNUM],
                    235:                   reg_names[PIC_OFFSET_TABLE_REGNUM]);
                    236: #endif
                    237: #endif
                    238:     }
                    239: }
                    240: 
                    241: /* Return true if this function's epilogue can be output as RTL.  */
                    242: 
                    243: int
                    244: use_return_insn ()
                    245: {
                    246:   int regno;
                    247: 
                    248:   if (!reload_completed || frame_pointer_needed || get_frame_size () != 0)
                    249:     return 0;
                    250:   
                    251:   /* Copied from output_function_epilogue ().  We should probably create a
                    252:      separate layout routine to perform the common work.  */
                    253:   
                    254:   for (regno = 0 ; regno < FIRST_PSEUDO_REGISTER ; regno++)
                    255:     if (regs_ever_live[regno] && ! call_used_regs[regno])
                    256:       return 0;
                    257:   
                    258:   return 1;
                    259: }
                    260: 
                    261: /* This function generates the assembly code for function exit,
                    262:    on machines that need it.  Args are same as for FUNCTION_PROLOGUE.
                    263: 
                    264:    The function epilogue should not depend on the current stack pointer!
                    265:    It should use the frame pointer only, if there is a frame pointer.
                    266:    This is mandatory because of alloca; we also take advantage of it to
                    267:    omit stack adjustments before returning.  */
                    268: 
                    269: void
                    270: output_function_epilogue (stream, size)
                    271:      FILE *stream;
                    272:      int size;
                    273: {
                    274:   register int regno;
                    275:   register int mask, fmask;
                    276:   register int nregs;
                    277:   int offset, foffset, fpoffset;
                    278:   extern char call_used_regs[];
                    279:   int fsize = (size + 3) & -4;
                    280:   int big = 0;
                    281:   rtx insn = get_last_insn ();
                    282:   
                    283:   /* If the last insn was a BARRIER, we don't have to write any code.  */
                    284:   if (GET_CODE (insn) == NOTE)
                    285:     insn = prev_nonnote_insn (insn);
                    286:   if (insn && GET_CODE (insn) == BARRIER)
                    287:     {
                    288:       /* Output just a no-op so that debuggers don't get confused
                    289:         about which function the pc is in at this address.  */
                    290:       asm_fprintf (stream, "\tnop\n");
                    291:       return;
                    292:     }
                    293: 
                    294: #ifdef FUNCTION_EXTRA_EPILOGUE
                    295:   FUNCTION_EXTRA_EPILOGUE (stream, size);
                    296: #endif
                    297:   nregs = 0;  fmask = 0; fpoffset = 0;
                    298: #ifdef SUPPORT_SUN_FPA
                    299:   for (regno = 24 ; regno < 56 ; regno++)
                    300:     if (regs_ever_live[regno] && ! call_used_regs[regno])
                    301:       nregs++;
                    302:   fpoffset = nregs * 8;
                    303: #endif
                    304:   nregs = 0;
                    305:   for (regno = 16; regno < 24; regno++)
                    306:     if (regs_ever_live[regno] && ! call_used_regs[regno])
                    307:       {
                    308:         nregs++;
                    309:        fmask |= 1 << (23 - regno);
                    310:       }
                    311:   foffset = fpoffset + nregs * 12;
                    312:   nregs = 0;  mask = 0;
                    313:   if (frame_pointer_needed)
                    314:     regs_ever_live[FRAME_POINTER_REGNUM] = 0;
                    315:   for (regno = 0; regno < 16; regno++)
                    316:     if (regs_ever_live[regno] && ! call_used_regs[regno])
                    317:       {
                    318:         nregs++;
                    319:        mask |= 1 << regno;
                    320:       }
                    321:   offset = foffset + nregs * 4;
                    322:   if (offset + fsize >= 0x8000
                    323:       && frame_pointer_needed
                    324:       && (mask || fmask || fpoffset))
                    325:     {
                    326: #ifdef MOTOROLA
                    327:       asm_fprintf (stream, "\t%Omove.l %0I%d,%Ra0\n", -fsize);
                    328: #else
                    329:       asm_fprintf (stream, "\tmovel %0I%d,%Ra0\n", -fsize);
                    330: #endif
                    331:       fsize = 0, big = 1;
                    332:     }
                    333:   if (nregs <= 2)
                    334:     {
                    335:       /* Restore each separately in the same order moveml does.
                    336:          Using two movel instructions instead of a single moveml
                    337:          is about 15% faster for the 68020 and 68030 at no expense
                    338:          in code size. */
                    339: 
                    340:       int i;
                    341: 
                    342:       /* Undo the work from above. */
                    343:       for (i = 0; i< 16; i++)
                    344:         if (mask & (1 << i))
                    345:           {
                    346:             if (big)
                    347:              {
                    348: #ifdef MOTOROLA
                    349:                asm_fprintf (stream, "\t%Omove.l -%d(%s,%Ra0.l),%s\n",
                    350:                             offset + fsize,
                    351:                             reg_names[FRAME_POINTER_REGNUM],
                    352:                             reg_names[i]);
                    353: #else
                    354:                asm_fprintf (stream, "\tmovel %s@(-%d,%Ra0:l),%s\n",
                    355:                             reg_names[FRAME_POINTER_REGNUM],
                    356:                             offset + fsize, reg_names[i]);
                    357: #endif
                    358:              }
                    359:             else if (! frame_pointer_needed)
                    360:              {
                    361: #ifdef MOTOROLA
                    362:                asm_fprintf (stream, "\t%Omove.l (%Rsp)+,%s\n",
                    363:                             reg_names[i]);
                    364: #else
                    365:                asm_fprintf (stream, "\tmovel %Rsp@+,%s\n",
                    366:                             reg_names[i]);
                    367: #endif
                    368:              }
                    369:             else
                    370:              {
                    371: #ifdef MOTOROLA
                    372:                asm_fprintf (stream, "\t%Omove.l -%d(%s),%s\n",
                    373:                             offset + fsize,
                    374:                             reg_names[FRAME_POINTER_REGNUM],
                    375:                             reg_names[i]);
                    376: #else
                    377:                asm_fprintf (stream, "\tmovel %s@(-%d),%s\n",
                    378:                             reg_names[FRAME_POINTER_REGNUM],
                    379:                             offset + fsize, reg_names[i]);
                    380: #endif
                    381:              }
                    382:             offset = offset - 4;
                    383:           }
                    384:     }
                    385:   else if (mask)
                    386:     {
                    387:       if (big)
                    388:        {
                    389: #ifdef MOTOROLA
                    390:          asm_fprintf (stream, "\tmovm.l -%d(%s,%Ra0.l),%0I0x%x\n",
                    391:                       offset + fsize,
                    392:                       reg_names[FRAME_POINTER_REGNUM],
                    393:                       mask);
                    394: #else
                    395:          asm_fprintf (stream, "\tmoveml %s@(-%d,%Ra0:l),%0I0x%x\n",
                    396:                       reg_names[FRAME_POINTER_REGNUM],
                    397:                       offset + fsize, mask);
                    398: #endif
                    399:        }
                    400:       else if (! frame_pointer_needed)
                    401:        {
                    402: #ifdef MOTOROLA
                    403:          asm_fprintf (stream, "\tmovm.l (%Rsp)+,%0I0x%x\n", mask);
                    404: #else
                    405:          asm_fprintf (stream, "\tmoveml %Rsp@+,%0I0x%x\n", mask);
                    406: #endif
                    407:        }
                    408:       else
                    409:        {
                    410: #ifdef MOTOROLA
                    411:          asm_fprintf (stream, "\tmovm.l -%d(%s),%0I0x%x\n",
                    412:                       offset + fsize,
                    413:                       reg_names[FRAME_POINTER_REGNUM],
                    414:                       mask);
                    415: #else
                    416:          asm_fprintf (stream, "\tmoveml %s@(-%d),%0I0x%x\n",
                    417:                       reg_names[FRAME_POINTER_REGNUM],
                    418:                       offset + fsize, mask);
                    419: #endif
                    420:        }
                    421:     }
                    422:   if (fmask)
                    423:     {
                    424:       if (big)
                    425:        {
                    426: #ifdef MOTOROLA
                    427:          asm_fprintf (stream, "\tfmovm -%d(%s,%Ra0.l),%0I0x%x\n",
                    428:                       foffset + fsize,
                    429:                       reg_names[FRAME_POINTER_REGNUM],
                    430:                       fmask);
                    431: #else
                    432:          asm_fprintf (stream, "\tfmovem %s@(-%d,%Ra0:l),%0I0x%x\n",
                    433:                       reg_names[FRAME_POINTER_REGNUM],
                    434:                       foffset + fsize, fmask);
                    435: #endif
                    436:        }
                    437:       else if (! frame_pointer_needed)
                    438:        {
                    439: #ifdef MOTOROLA
                    440:          asm_fprintf (stream, "\tfmovm (%Rsp)+,%0I0x%x\n", fmask);
                    441: #else
                    442:          asm_fprintf (stream, "\tfmovem %Rsp@+,%0I0x%x\n", fmask);
                    443: #endif
                    444:        }
                    445:       else
                    446:        {
                    447: #ifdef MOTOROLA
                    448:          asm_fprintf (stream, "\tfmovm -%d(%s),%0I0x%x\n",
                    449:                       foffset + fsize,
                    450:                       reg_names[FRAME_POINTER_REGNUM],
                    451:                       fmask);
                    452: #else
                    453:          asm_fprintf (stream, "\tfmovem %s@(-%d),%0I0x%x\n",
                    454:                       reg_names[FRAME_POINTER_REGNUM],
                    455:                       foffset + fsize, fmask);
                    456: #endif
                    457:        }
                    458:     }
                    459:   if (fpoffset != 0)
                    460:     for (regno = 55; regno >= 24; regno--)
                    461:       if (regs_ever_live[regno] && ! call_used_regs[regno])
                    462:         {
                    463:          if (big)
                    464:            {
                    465: #ifdef MOTOROLA
                    466:              asm_fprintf (stream, "\tfpmovd -%d(%s,%Ra0.l), %s\n",
                    467:                           fpoffset + fsize,
                    468:                           reg_names[FRAME_POINTER_REGNUM],
                    469:                           reg_names[regno]);
                    470: #else
                    471:              asm_fprintf (stream, "\tfpmoved %s@(-%d,%Ra0:l), %s\n",
                    472:                           reg_names[FRAME_POINTER_REGNUM],
                    473:                           fpoffset + fsize, reg_names[regno]);
                    474: #endif
                    475:            }
                    476:          else if (! frame_pointer_needed)
                    477:            {
                    478: #ifdef MOTOROLA
                    479:              asm_fprintf (stream, "\tfpmovd (%Rsp)+,%s\n",
                    480:                           reg_names[regno]);
                    481: #else
                    482:              asm_fprintf (stream, "\tfpmoved %Rsp@+, %s\n",
                    483:                           reg_names[regno]);
                    484: #endif
                    485:            }
                    486:          else
                    487:            {
                    488: #ifdef MOTOROLA
                    489:              asm_fprintf (stream, "\tfpmovd -%d(%s), %s\n",
                    490:                           fpoffset + fsize,
                    491:                           reg_names[FRAME_POINTER_REGNUM],
                    492:                           reg_names[regno]);
                    493: #else
                    494:              asm_fprintf (stream, "\tfpmoved %s@(-%d), %s\n",
                    495:                           reg_names[FRAME_POINTER_REGNUM],
                    496:                           fpoffset + fsize, reg_names[regno]);
                    497: #endif
                    498:            }
                    499:          fpoffset -= 8;
                    500:        }
                    501:   if (frame_pointer_needed)
                    502:     fprintf (stream, "\tunlk %s\n",
                    503:             reg_names[FRAME_POINTER_REGNUM]);
                    504:   else if (fsize)
                    505:     {
                    506:       if (fsize + 4 < 0x8000)
                    507:        {
                    508: #ifdef MOTOROLA
                    509:          asm_fprintf (stream, "\tadd.w %0I%d,%Rsp\n", fsize + 4);
                    510: #else
                    511:          asm_fprintf (stream, "\taddw %0I%d,%Rsp\n", fsize + 4);
                    512: #endif
                    513:        }
                    514:       else
                    515:        {
                    516: #ifdef MOTOROLA
                    517:          asm_fprintf (stream, "\tadd.l %0I%d,%Rsp\n", fsize + 4);
                    518: #else
                    519:          asm_fprintf (stream, "\taddl %0I%d,%Rsp\n", fsize + 4);
                    520: #endif
                    521:        }
                    522:     }
                    523:   if (current_function_pops_args)
                    524:     asm_fprintf (stream, "\trtd %0I%d\n", current_function_pops_args);
                    525:   else
                    526:     fprintf (stream, "\trts\n");
                    527: }
                    528: 
                    529: /* Similar to general_operand, but exclude stack_pointer_rtx.  */
                    530: 
                    531: int
                    532: not_sp_operand (op, mode)
                    533:      register rtx op;
                    534:      enum machine_mode mode;
                    535: {
                    536:   return op != stack_pointer_rtx && general_operand (op, mode);
                    537: }
                    538: 
                    539: /* Return TRUE if X is a valid comparison operator for the dbcc 
                    540:    instruction.  
                    541: 
                    542:    Note it rejects floating point comparison operators.
                    543:    (In the future we could use Fdbcc).
                    544: 
                    545:    It also rejects some comparisons when CC_NO_OVERFLOW is set.  */
                    546:    
                    547: int
                    548: valid_dbcc_comparison_p (x, mode)
                    549:      rtx x;
                    550:      enum machine_mode mode;
                    551: {
                    552:   /* We could add support for these in the future */
                    553:   if (cc_prev_status.flags & CC_IN_68881)
                    554:     return 0;
                    555: 
                    556:   switch (GET_CODE (x))
                    557:     {
                    558: 
                    559:       case EQ: case NE: case GTU: case LTU:
                    560:       case GEU: case LEU:
                    561:         return 1;
                    562: 
                    563:       /* Reject some when CC_NO_OVERFLOW is set.  This may be over
                    564:          conservative */
                    565:       case GT: case LT: case GE: case LE:
                    566:         return ! (cc_prev_status.flags & CC_NO_OVERFLOW);
                    567:       default:
                    568:         return 0;
                    569:     }
                    570: }
                    571: 
                    572: /* Output a dbCC; jCC sequence.  Note we do not handle the 
                    573:    floating point version of this sequence (Fdbcc).  We also
                    574:    do not handle alternative conditions when CC_NO_OVERFLOW is
                    575:    set.  It is assumed that valid_dbcc_comparison_p will kick
                    576:    those out before we get here.  */
                    577: 
                    578: output_dbcc_and_branch (operands)
                    579:      rtx *operands;
                    580: {
                    581:  
                    582:   switch (GET_CODE (operands[3]))
                    583:     {
                    584:       case EQ:
                    585: #ifdef MOTOROLA
                    586:         output_asm_insn ("dbeq %0,%l1\n\tjbeq %l2", operands);
                    587: #else
                    588:         output_asm_insn ("dbeq %0,%l1\n\tjeq %l2", operands);
                    589: #endif
                    590:         break;
                    591: 
                    592:       case NE:
                    593: #ifdef MOTOROLA
                    594:         output_asm_insn ("dbne %0,%l1\n\tjbne %l2", operands);
                    595: #else
                    596:         output_asm_insn ("dbne %0,%l1\n\tjne %l2", operands);
                    597: #endif
                    598:         break;
                    599: 
                    600:       case GT:
                    601: #ifdef MOTOROLA
                    602:         output_asm_insn ("dbgt %0,%l1\n\tjbgt %l2", operands);
                    603: #else
                    604:         output_asm_insn ("dbgt %0,%l1\n\tjgt %l2", operands);
                    605: #endif
                    606:         break;
                    607: 
                    608:       case GTU:
                    609: #ifdef MOTOROLA
                    610:         output_asm_insn ("dbhi %0,%l1\n\tjbhi %l2", operands);
                    611: #else
                    612:         output_asm_insn ("dbhi %0,%l1\n\tjhi %l2", operands);
                    613: #endif
                    614:         break;
                    615: 
                    616:       case LT:
                    617: #ifdef MOTOROLA
                    618:         output_asm_insn ("dblt %0,%l1\n\tjblt %l2", operands);
                    619: #else
                    620:         output_asm_insn ("dblt %0,%l1\n\tjlt %l2", operands);
                    621: #endif
                    622:         break;
                    623: 
                    624:       case LTU:
                    625: #ifdef MOTOROLA
                    626:         output_asm_insn ("dbcs %0,%l1\n\tjbcs %l2", operands);
                    627: #else
                    628:         output_asm_insn ("dbcs %0,%l1\n\tjcs %l2", operands);
                    629: #endif
                    630:         break;
                    631: 
                    632:       case GE:
                    633: #ifdef MOTOROLA
                    634:         output_asm_insn ("dbge %0,%l1\n\tjbge %l2", operands);
                    635: #else
                    636:         output_asm_insn ("dbge %0,%l1\n\tjge %l2", operands);
                    637: #endif
                    638:         break;
                    639: 
                    640:       case GEU:
                    641: #ifdef MOTOROLA
                    642:         output_asm_insn ("dbcc %0,%l1\n\tjbcc %l2", operands);
                    643: #else
                    644:         output_asm_insn ("dbcc %0,%l1\n\tjcc %l2", operands);
                    645: #endif
                    646:         break;
                    647: 
                    648:       case LE:
                    649: #ifdef MOTOROLA
                    650:         output_asm_insn ("dble %0,%l1\n\tjble %l2", operands);
                    651: #else
                    652:         output_asm_insn ("dble %0,%l1\n\tjle %l2", operands);
                    653: #endif
                    654:         break;
                    655: 
                    656:       case LEU:
                    657: #ifdef MOTOROLA
                    658:         output_asm_insn ("dbls %0,%l1\n\tjbls %l2", operands);
                    659: #else
                    660:         output_asm_insn ("dbls %0,%l1\n\tjls %l2", operands);
                    661: #endif
                    662:         break;
                    663: 
                    664:       default:
                    665:        abort ();
                    666:     }
                    667: 
                    668:   /* If the decrement is to be done in SImode, then we have
                    669:      to compensate for the fact that dbcc decrements in HImode. */
                    670:   switch (GET_MODE (operands[0]))
                    671:     {
                    672:       case SImode:
                    673: #ifdef MOTOROLA
                    674:         output_asm_insn ("clr%.w %0\n\tsubq%.l %#1,%0\n\tjbpl %l1", operands);
                    675: #else
                    676:         output_asm_insn ("clr%.w %0\n\tsubq%.l %#1,%0\n\tjpl %l1", operands);
                    677: #endif
                    678:         break;
                    679: 
                    680:       case HImode:
                    681:         break;
                    682: 
                    683:       default:
                    684:         abort ();
                    685:     }
                    686: }
                    687: 
                    688: char *
                    689: output_btst (operands, countop, dataop, insn, signpos)
                    690:      rtx *operands;
                    691:      rtx countop, dataop;
                    692:      rtx insn;
                    693:      int signpos;
                    694: {
                    695:   operands[0] = countop;
                    696:   operands[1] = dataop;
                    697: 
                    698:   if (GET_CODE (countop) == CONST_INT)
                    699:     {
                    700:       register int count = INTVAL (countop);
                    701:       /* If COUNT is bigger than size of storage unit in use,
                    702:         advance to the containing unit of same size.  */
                    703:       if (count > signpos)
                    704:        {
                    705:          int offset = (count & ~signpos) / 8;
                    706:          count = count & signpos;
                    707:          operands[1] = dataop = adj_offsettable_operand (dataop, offset);
                    708:        }
                    709:       if (count == signpos)
                    710:        cc_status.flags = CC_NOT_POSITIVE | CC_Z_IN_NOT_N;
                    711:       else
                    712:        cc_status.flags = CC_NOT_NEGATIVE | CC_Z_IN_NOT_N;
                    713: 
                    714:       /* These three statements used to use next_insns_test_no...
                    715:         but it appears that this should do the same job.  */
                    716:       if (count == 31
                    717:          && next_insn_tests_no_inequality (insn))
                    718:        return "tst%.l %1";
                    719:       if (count == 15
                    720:          && next_insn_tests_no_inequality (insn))
                    721:        return "tst%.w %1";
                    722:       if (count == 7
                    723:          && next_insn_tests_no_inequality (insn))
                    724:        return "tst%.b %1";
                    725: 
                    726:       cc_status.flags = CC_NOT_NEGATIVE;
                    727:     }
                    728:   return "btst %0,%1";
                    729: }
                    730: 
                    731: /* Returns 1 if OP is either a symbol reference or a sum of a symbol
                    732:    reference and a constant.  */
                    733: 
                    734: int
                    735: symbolic_operand (op, mode)
                    736:      register rtx op;
                    737:      enum machine_mode mode;
                    738: {
                    739:   switch (GET_CODE (op))
                    740:     {
                    741:     case SYMBOL_REF:
                    742:     case LABEL_REF:
                    743:       return 1;
                    744: 
                    745:     case CONST:
                    746:       op = XEXP (op, 0);
                    747:       return (GET_CODE (op) == PLUS
                    748:              && (GET_CODE (XEXP (op, 0)) == SYMBOL_REF
                    749:                  || GET_CODE (XEXP (op, 0)) == LABEL_REF)
                    750:              && GET_CODE (XEXP (op, 1)) == CONST_INT);
                    751: 
                    752: #if 0 /* Deleted, with corresponding change in m68k.h,
                    753:         so as to fit the specs.  No CONST_DOUBLE is ever symbolic.  */
                    754:     case CONST_DOUBLE:
                    755:       return GET_MODE (op) == mode;
                    756: #endif
                    757: 
                    758:     default:
                    759:       return 0;
                    760:     }
                    761: }
                    762: 
                    763: 
                    764: /* Legitimize PIC addresses.  If the address is already
                    765:    position-independent, we return ORIG.  Newly generated
                    766:    position-independent addresses go to REG.  If we need more
                    767:    than one register, we lose.  
                    768: 
                    769:    An address is legitimized by making an indirect reference
                    770:    through the Global Offset Table with the name of the symbol
                    771:    used as an offset.  
                    772: 
                    773:    The assembler and linker are responsible for placing the 
                    774:    address of the symbol in the GOT.  The function prologue
                    775:    is responsible for initializing a5 to the starting address
                    776:    of the GOT.
                    777: 
                    778:    The assembler is also responsible for translating a symbol name
                    779:    into a constant displacement from the start of the GOT.  
                    780: 
                    781:    A quick example may make things a little clearer:
                    782: 
                    783:    When not generating PIC code to store the value 12345 into _foo
                    784:    we would generate the following code:
                    785: 
                    786:        movel #12345, _foo
                    787: 
                    788:    When generating PIC two transformations are made.  First, the compiler
                    789:    loads the address of foo into a register.  So the first transformation makes:
                    790: 
                    791:        lea     _foo, a0
                    792:        movel   #12345, a0@
                    793: 
                    794:    The code in movsi will intercept the lea instruction and call this
                    795:    routine which will transform the instructions into:
                    796: 
                    797:        movel   a5@(_foo:w), a0
                    798:        movel   #12345, a0@
                    799:    
                    800: 
                    801:    That (in a nutshell) is how *all* symbol and label references are 
                    802:    handled.  */
                    803: 
                    804: rtx
                    805: legitimize_pic_address (orig, mode, reg)
                    806:      rtx orig, reg;
                    807:      enum machine_mode mode;
                    808: {
                    809:   rtx pic_ref = orig;
                    810: 
                    811: #ifdef MACHO_PIC
                    812:   return machopic_legitimize_pic_address (orig, mode, reg);
                    813: #endif
                    814: 
                    815:   /* First handle a simple SYMBOL_REF or LABEL_REF */
                    816:   if (GET_CODE (orig) == SYMBOL_REF || GET_CODE (orig) == LABEL_REF)
                    817:     {
                    818:       if (reg == 0)
                    819:        abort ();
                    820:       /* addr = pic_offset_table[foo] */
                    821:       pic_ref = gen_rtx (MEM, Pmode,
                    822:                         gen_rtx (PLUS, Pmode,
                    823:                                  pic_offset_table_rtx, orig));
                    824:       current_function_uses_pic_offset_table = 1;
                    825:       RTX_UNCHANGING_P (pic_ref) = 1;
                    826:       emit_move_insn (reg, pic_ref);
                    827:       return reg;
                    828:     }
                    829:   else if (GET_CODE (orig) == CONST)
                    830:     {
                    831:       rtx base, offset;
                    832: 
                    833:       /* Make sure this is CONST has not already been legitimized */
                    834:       if (GET_CODE (XEXP (orig, 0)) == PLUS
                    835:          && XEXP (XEXP (orig, 0), 0) == pic_offset_table_rtx)
                    836:        return orig;
                    837: 
                    838:       if (reg == 0)
                    839:        abort ();
                    840: 
                    841:       /* legitimize both operands of the PLUS */
                    842:       if (GET_CODE (XEXP (orig, 0)) == PLUS)
                    843:        {
                    844:          base = legitimize_pic_address (XEXP (XEXP (orig, 0), 0), Pmode, reg);
                    845:          orig = legitimize_pic_address (XEXP (XEXP (orig, 0), 1), Pmode,
                    846:                                         base == reg ? 0 : reg);
                    847:        }
                    848:       else abort ();
                    849: 
                    850:       if (GET_CODE (orig) == CONST_INT)
                    851:        return plus_constant_for_output (base, INTVAL (orig));
                    852:       pic_ref = gen_rtx (PLUS, Pmode, base, orig);
                    853:       /* Likewise, should we set special REG_NOTEs here?  */
                    854:     }
                    855:   return pic_ref;
                    856: }
                    857: 
                    858: 
                    859: /* Return the best assembler insn template
                    860:    for moving operands[1] into operands[0] as a fullword.  */
                    861: 
                    862: static char *
                    863: singlemove_string (operands)
                    864:      rtx *operands;
                    865: {
                    866: #ifdef SUPPORT_SUN_FPA
                    867:   if (FPA_REG_P (operands[0]) || FPA_REG_P (operands[1]))
                    868:     return "fpmoves %1,%0";
                    869: #endif
                    870:   if (DATA_REG_P (operands[0])
                    871:       && GET_CODE (operands[1]) == CONST_INT
                    872:       && INTVAL (operands[1]) < 128
                    873:       && INTVAL (operands[1]) >= -128)
                    874:     {
                    875: #if defined (MOTOROLA) && !defined (CRDS)
                    876:       return "moveq%.l %1,%0";
                    877: #else
                    878:       return "moveq %1,%0";
                    879: #endif
                    880:     }
                    881:   if (operands[1] != const0_rtx)
                    882:     return "move%.l %1,%0";
                    883:   if (! ADDRESS_REG_P (operands[0]))
                    884:     return "clr%.l %0";
                    885:   return "sub%.l %0,%0";
                    886: }
                    887: 
                    888: 
                    889: /* Output assembler code to perform a doubleword move insn
                    890:    with operands OPERANDS.  */
                    891: 
                    892: char *
                    893: output_move_double (operands)
                    894:      rtx *operands;
                    895: {
                    896:   enum
                    897:     {
                    898:       REGOP, OFFSOP, MEMOP, PUSHOP, POPOP, CNSTOP, RNDOP
                    899:     } optype0, optype1;
                    900:   rtx latehalf[2];
                    901:   rtx middlehalf[2];
                    902:   rtx xops[2];
                    903:   rtx addreg0 = 0, addreg1 = 0;
                    904:   int dest_overlapped_low = 0;
                    905:   int size = GET_MODE_SIZE (GET_MODE (operands[0]));
                    906: 
                    907:   middlehalf[0] = 0;
                    908:   middlehalf[1] = 0;
                    909: 
                    910:   /* First classify both operands.  */
                    911: 
                    912:   if (REG_P (operands[0]))
                    913:     optype0 = REGOP;
                    914:   else if (offsettable_memref_p (operands[0]))
                    915:     optype0 = OFFSOP;
                    916:   else if (GET_CODE (XEXP (operands[0], 0)) == POST_INC)
                    917:     optype0 = POPOP;
                    918:   else if (GET_CODE (XEXP (operands[0], 0)) == PRE_DEC)
                    919:     optype0 = PUSHOP;
                    920:   else if (GET_CODE (operands[0]) == MEM)
                    921:     optype0 = MEMOP;
                    922:   else
                    923:     optype0 = RNDOP;
                    924: 
                    925:   if (REG_P (operands[1]))
                    926:     optype1 = REGOP;
                    927:   else if (CONSTANT_P (operands[1]))
                    928:     optype1 = CNSTOP;
                    929:   else if (offsettable_memref_p (operands[1]))
                    930:     optype1 = OFFSOP;
                    931:   else if (GET_CODE (XEXP (operands[1], 0)) == POST_INC)
                    932:     optype1 = POPOP;
                    933:   else if (GET_CODE (XEXP (operands[1], 0)) == PRE_DEC)
                    934:     optype1 = PUSHOP;
                    935:   else if (GET_CODE (operands[1]) == MEM)
                    936:     optype1 = MEMOP;
                    937:   else
                    938:     optype1 = RNDOP;
                    939: 
                    940:   /* Check for the cases that the operand constraints are not
                    941:      supposed to allow to happen.  Abort if we get one,
                    942:      because generating code for these cases is painful.  */
                    943: 
                    944:   if (optype0 == RNDOP || optype1 == RNDOP)
                    945:     abort ();
                    946: 
                    947:   /* If one operand is decrementing and one is incrementing
                    948:      decrement the former register explicitly
                    949:      and change that operand into ordinary indexing.  */
                    950: 
                    951:   if (optype0 == PUSHOP && optype1 == POPOP)
                    952:     {
                    953:       operands[0] = XEXP (XEXP (operands[0], 0), 0);
                    954:       if (size == 12)
                    955:         output_asm_insn ("sub%.l %#12,%0", operands);
                    956:       else
                    957:         output_asm_insn ("subq%.l %#8,%0", operands);
                    958:       if (GET_MODE (operands[1]) == XFmode)
                    959:        operands[0] = gen_rtx (MEM, XFmode, operands[0]);
                    960:       else if (GET_MODE (operands[0]) == DFmode)
                    961:        operands[0] = gen_rtx (MEM, DFmode, operands[0]);
                    962:       else
                    963:        operands[0] = gen_rtx (MEM, DImode, operands[0]);
                    964:       optype0 = OFFSOP;
                    965:     }
                    966:   if (optype0 == POPOP && optype1 == PUSHOP)
                    967:     {
                    968:       operands[1] = XEXP (XEXP (operands[1], 0), 0);
                    969:       if (size == 12)
                    970:         output_asm_insn ("sub%.l %#12,%1", operands);
                    971:       else
                    972:         output_asm_insn ("subq%.l %#8,%1", operands);
                    973:       if (GET_MODE (operands[1]) == XFmode)
                    974:        operands[1] = gen_rtx (MEM, XFmode, operands[1]);
                    975:       else if (GET_MODE (operands[1]) == DFmode)
                    976:        operands[1] = gen_rtx (MEM, DFmode, operands[1]);
                    977:       else
                    978:        operands[1] = gen_rtx (MEM, DImode, operands[1]);
                    979:       optype1 = OFFSOP;
                    980:     }
                    981: 
                    982:   /* If an operand is an unoffsettable memory ref, find a register
                    983:      we can increment temporarily to make it refer to the second word.  */
                    984: 
                    985:   if (optype0 == MEMOP)
                    986:     addreg0 = find_addr_reg (XEXP (operands[0], 0));
                    987: 
                    988:   if (optype1 == MEMOP)
                    989:     addreg1 = find_addr_reg (XEXP (operands[1], 0));
                    990: 
                    991:   /* Ok, we can do one word at a time.
                    992:      Normally we do the low-numbered word first,
                    993:      but if either operand is autodecrementing then we
                    994:      do the high-numbered word first.
                    995: 
                    996:      In either case, set up in LATEHALF the operands to use
                    997:      for the high-numbered word and in some cases alter the
                    998:      operands in OPERANDS to be suitable for the low-numbered word.  */
                    999: 
                   1000:   if (size == 12)
                   1001:     {
                   1002:       if (optype0 == REGOP)
                   1003:        {
                   1004:          latehalf[0] = gen_rtx (REG, SImode, REGNO (operands[0]) + 2);
                   1005:          middlehalf[0] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1);
                   1006:        }
                   1007:       else if (optype0 == OFFSOP)
                   1008:        {
                   1009:          middlehalf[0] = adj_offsettable_operand (operands[0], 4);
                   1010:          latehalf[0] = adj_offsettable_operand (operands[0], size - 4);
                   1011:        }
                   1012:       else
                   1013:        {
                   1014:          middlehalf[0] = operands[0];
                   1015:          latehalf[0] = operands[0];
                   1016:        }
                   1017: 
                   1018:       if (optype1 == REGOP)
                   1019:        {
                   1020:          latehalf[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 2);
                   1021:          middlehalf[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1);
                   1022:        }
                   1023:       else if (optype1 == OFFSOP)
                   1024:        {
                   1025:          middlehalf[1] = adj_offsettable_operand (operands[1], 4);
                   1026:          latehalf[1] = adj_offsettable_operand (operands[1], size - 4);
                   1027:        }
                   1028:       else if (optype1 == CNSTOP)
                   1029:        {
                   1030:          if (GET_CODE (operands[1]) == CONST_DOUBLE)
                   1031:            {
                   1032:              REAL_VALUE_TYPE r;
                   1033:              long l[3];
                   1034: 
                   1035:              REAL_VALUE_FROM_CONST_DOUBLE (r, operands[1]);
                   1036:              REAL_VALUE_TO_TARGET_LONG_DOUBLE (r, l);
                   1037:              operands[1] = GEN_INT (l[0]);
                   1038:              middlehalf[1] = GEN_INT (l[1]);
                   1039:              latehalf[1] = GEN_INT (l[2]);
                   1040:            }
                   1041:          else if (CONSTANT_P (operands[1]))
                   1042:            {
                   1043:              /* actually, no non-CONST_DOUBLE constant should ever
                   1044:                 appear here.  */
                   1045:              abort ();
                   1046:              if (GET_CODE (operands[1]) == CONST_INT && INTVAL (operands[1]) < 0)
                   1047:                latehalf[1] = constm1_rtx;
                   1048:              else
                   1049:                latehalf[1] = const0_rtx;
                   1050:            }
                   1051:        }
                   1052:       else
                   1053:        {
                   1054:          middlehalf[1] = operands[1];
                   1055:          latehalf[1] = operands[1];
                   1056:        }
                   1057:     }
                   1058:   else
                   1059:     /* size is not 12: */
                   1060:     {
                   1061:       if (optype0 == REGOP)
                   1062:        latehalf[0] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1);
                   1063:       else if (optype0 == OFFSOP)
                   1064:        latehalf[0] = adj_offsettable_operand (operands[0], size - 4);
                   1065:       else
                   1066:        latehalf[0] = operands[0];
                   1067: 
                   1068:       if (optype1 == REGOP)
                   1069:        latehalf[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1);
                   1070:       else if (optype1 == OFFSOP)
                   1071:        latehalf[1] = adj_offsettable_operand (operands[1], size - 4);
                   1072:       else if (optype1 == CNSTOP)
                   1073:        split_double (operands[1], &operands[1], &latehalf[1]);
                   1074:       else
                   1075:        latehalf[1] = operands[1];
                   1076:     }
                   1077: 
                   1078:   /* If insn is effectively movd N(sp),-(sp) then we will do the
                   1079:      high word first.  We should use the adjusted operand 1 (which is N+4(sp))
                   1080:      for the low word as well, to compensate for the first decrement of sp.  */
                   1081:   if (optype0 == PUSHOP
                   1082:       && REGNO (XEXP (XEXP (operands[0], 0), 0)) == STACK_POINTER_REGNUM
                   1083:       && reg_overlap_mentioned_p (stack_pointer_rtx, operands[1]))
                   1084:     operands[1] = middlehalf[1] = latehalf[1];
                   1085: 
                   1086:   /* For (set (reg:DI N) (mem:DI ... (reg:SI N) ...)),
                   1087:      if the upper part of reg N does not appear in the MEM, arrange to
                   1088:      emit the move late-half first.  Otherwise, compute the MEM address
                   1089:      into the upper part of N and use that as a pointer to the memory
                   1090:      operand.  */
                   1091:   if (optype0 == REGOP
                   1092:       && (optype1 == OFFSOP || optype1 == MEMOP))
                   1093:     {
                   1094:       rtx testlow = gen_rtx (REG, SImode, REGNO (operands[0]));
                   1095: 
                   1096:       if (reg_overlap_mentioned_p (testlow, XEXP (operands[1], 0))
                   1097:          && reg_overlap_mentioned_p (latehalf[0], XEXP (operands[1], 0)))
                   1098:        {
                   1099:          /* If both halves of dest are used in the src memory address,
                   1100:             compute the address into latehalf of dest.
                   1101:             Note that this can't happen if the dest is two data regs.  */
                   1102: compadr:
                   1103:          xops[0] = latehalf[0];
                   1104:          xops[1] = XEXP (operands[1], 0);
                   1105:          output_asm_insn ("lea %a1,%0", xops);
                   1106:          if( GET_MODE (operands[1]) == XFmode )
                   1107:            {
                   1108:              operands[1] = gen_rtx (MEM, XFmode, latehalf[0]);
                   1109:              middlehalf[1] = adj_offsettable_operand (operands[1], size-8);
                   1110:              latehalf[1] = adj_offsettable_operand (operands[1], size-4);
                   1111:            }
                   1112:          else
                   1113:            {
                   1114:              operands[1] = gen_rtx (MEM, DImode, latehalf[0]);
                   1115:              latehalf[1] = adj_offsettable_operand (operands[1], size-4);
                   1116:            }
                   1117:        }
                   1118:       else if (size == 12
                   1119:               && reg_overlap_mentioned_p (middlehalf[0],
                   1120:                                           XEXP (operands[1], 0)))
                   1121:        {
                   1122:          /* Check for two regs used by both source and dest.
                   1123:             Note that this can't happen if the dest is all data regs.
                   1124:             It can happen if the dest is d6, d7, a0.
                   1125:             But in that case, latehalf is an addr reg, so
                   1126:             the code at compadr does ok.  */
                   1127: 
                   1128:          if (reg_overlap_mentioned_p (testlow, XEXP (operands[1], 0))
                   1129:              || reg_overlap_mentioned_p (latehalf[0], XEXP (operands[1], 0)))
                   1130:            goto compadr;
                   1131: 
                   1132:          /* JRV says this can't happen: */
                   1133:          if (addreg0 || addreg1)
                   1134:            abort ();
                   1135: 
                   1136:          /* Only the middle reg conflicts; simply put it last. */
                   1137:          output_asm_insn (singlemove_string (operands), operands);
                   1138:          output_asm_insn (singlemove_string (latehalf), latehalf);
                   1139:          output_asm_insn (singlemove_string (middlehalf), middlehalf);
                   1140:          return "";
                   1141:        }
                   1142:       else if (reg_overlap_mentioned_p (testlow, XEXP (operands[1], 0)))
                   1143:        /* If the low half of dest is mentioned in the source memory
                   1144:           address, the arrange to emit the move late half first.  */
                   1145:        dest_overlapped_low = 1;
                   1146:     }
                   1147: 
                   1148:   /* If one or both operands autodecrementing,
                   1149:      do the two words, high-numbered first.  */
                   1150: 
                   1151:   /* Likewise,  the first move would clobber the source of the second one,
                   1152:      do them in the other order.  This happens only for registers;
                   1153:      such overlap can't happen in memory unless the user explicitly
                   1154:      sets it up, and that is an undefined circumstance.  */
                   1155: 
                   1156:   if (optype0 == PUSHOP || optype1 == PUSHOP
                   1157:       || (optype0 == REGOP && optype1 == REGOP
                   1158:          && ((middlehalf[1] && REGNO (operands[0]) == REGNO (middlehalf[1]))
                   1159:              || REGNO (operands[0]) == REGNO (latehalf[1])))
                   1160:       || dest_overlapped_low)
                   1161:     {
                   1162:       /* Make any unoffsettable addresses point at high-numbered word.  */
                   1163:       if (addreg0)
                   1164:        {
                   1165:          if (size == 12)
                   1166:            output_asm_insn ("addql %#8,%0", &addreg0);
                   1167:          else
                   1168:            output_asm_insn ("addql %#4,%0", &addreg0);
                   1169:        }
                   1170:       if (addreg1)
                   1171:        {
                   1172:          if (size == 12)
                   1173:            output_asm_insn ("addql %#8,%0", &addreg1);
                   1174:          else
                   1175:            output_asm_insn ("addql %#4,%0", &addreg1);
                   1176:        }
                   1177: 
                   1178:       /* Do that word.  */
                   1179:       output_asm_insn (singlemove_string (latehalf), latehalf);
                   1180: 
                   1181:       /* Undo the adds we just did.  */
                   1182:       if (addreg0)
                   1183:        output_asm_insn ("subql %#4,%0", &addreg0);
                   1184:       if (addreg1)
                   1185:        output_asm_insn ("subql %#4,%0", &addreg1);
                   1186: 
                   1187:       if (size == 12)
                   1188:        {
                   1189:          output_asm_insn (singlemove_string (middlehalf), middlehalf);
                   1190:          if (addreg0)
                   1191:            output_asm_insn ("subql %#4,%0", &addreg0);
                   1192:          if (addreg1)
                   1193:            output_asm_insn ("subql %#4,%0", &addreg1);
                   1194:        }
                   1195: 
                   1196:       /* Do low-numbered word.  */
                   1197:       return singlemove_string (operands);
                   1198:     }
                   1199: 
                   1200:   /* Normal case: do the two words, low-numbered first.  */
                   1201: 
                   1202:   output_asm_insn (singlemove_string (operands), operands);
                   1203: 
                   1204:   /* Do the middle one of the three words for long double */
                   1205:   if (size == 12)
                   1206:     {
                   1207:       if (addreg0)
                   1208:        output_asm_insn ("addql %#4,%0", &addreg0);
                   1209:       if (addreg1)
                   1210:        output_asm_insn ("addql %#4,%0", &addreg1);
                   1211: 
                   1212:       output_asm_insn (singlemove_string (middlehalf), middlehalf);
                   1213:     }
                   1214: 
                   1215:   /* Make any unoffsettable addresses point at high-numbered word.  */
                   1216:   if (addreg0)
                   1217:     output_asm_insn ("addql %#4,%0", &addreg0);
                   1218:   if (addreg1)
                   1219:     output_asm_insn ("addql %#4,%0", &addreg1);
                   1220: 
                   1221:   /* Do that word.  */
                   1222:   output_asm_insn (singlemove_string (latehalf), latehalf);
                   1223: 
                   1224:   /* Undo the adds we just did.  */
                   1225:   if (addreg0)
                   1226:     {
                   1227:       if (size == 12)
                   1228:         output_asm_insn ("subql %#8,%0", &addreg0);
                   1229:       else
                   1230:         output_asm_insn ("subql %#4,%0", &addreg0);
                   1231:     }
                   1232:   if (addreg1)
                   1233:     {
                   1234:       if (size == 12)
                   1235:         output_asm_insn ("subql %#8,%0", &addreg1);
                   1236:       else
                   1237:         output_asm_insn ("subql %#4,%0", &addreg1);
                   1238:     }
                   1239: 
                   1240:   return "";
                   1241: }
                   1242: 
                   1243: /* Return a REG that occurs in ADDR with coefficient 1.
                   1244:    ADDR can be effectively incremented by incrementing REG.  */
                   1245: 
                   1246: static rtx
                   1247: find_addr_reg (addr)
                   1248:      rtx addr;
                   1249: {
                   1250:   while (GET_CODE (addr) == PLUS)
                   1251:     {
                   1252:       if (GET_CODE (XEXP (addr, 0)) == REG)
                   1253:        addr = XEXP (addr, 0);
                   1254:       else if (GET_CODE (XEXP (addr, 1)) == REG)
                   1255:        addr = XEXP (addr, 1);
                   1256:       else if (CONSTANT_P (XEXP (addr, 0)))
                   1257:        addr = XEXP (addr, 1);
                   1258:       else if (CONSTANT_P (XEXP (addr, 1)))
                   1259:        addr = XEXP (addr, 0);
                   1260:       else
                   1261:        abort ();
                   1262:     }
                   1263:   if (GET_CODE (addr) == REG)
                   1264:     return addr;
                   1265:   abort ();
                   1266: }
                   1267: 
                   1268: /* Store in cc_status the expressions that the condition codes will
                   1269:    describe after execution of an instruction whose pattern is EXP.
                   1270:    Do not alter them if the instruction would not alter the cc's.  */
                   1271: 
                   1272: /* On the 68000, all the insns to store in an address register fail to
                   1273:    set the cc's.  However, in some cases these instructions can make it
                   1274:    possibly invalid to use the saved cc's.  In those cases we clear out
                   1275:    some or all of the saved cc's so they won't be used.  */
                   1276: 
                   1277: notice_update_cc (exp, insn)
                   1278:      rtx exp;
                   1279:      rtx insn;
                   1280: {
                   1281:   /* If the cc is being set from the fpa and the expression is not an
                   1282:      explicit floating point test instruction (which has code to deal with
                   1283:      this), reinit the CC.  */
                   1284:   if (((cc_status.value1 && FPA_REG_P (cc_status.value1))
                   1285:        || (cc_status.value2 && FPA_REG_P (cc_status.value2)))
                   1286:       && !(GET_CODE (exp) == PARALLEL
                   1287:           && GET_CODE (XVECEXP (exp, 0, 0)) == SET
                   1288:           && XEXP (XVECEXP (exp, 0, 0), 0) == cc0_rtx))
                   1289:     {
                   1290:       CC_STATUS_INIT; 
                   1291:     }
                   1292:   else if (GET_CODE (exp) == SET)
                   1293:     {
                   1294:       if (GET_CODE (SET_SRC (exp)) == CALL)
                   1295:        {
                   1296:          CC_STATUS_INIT; 
                   1297:        }
                   1298:       else if (ADDRESS_REG_P (SET_DEST (exp)))
                   1299:        {
                   1300:          if (cc_status.value1
                   1301:              && reg_overlap_mentioned_p (SET_DEST (exp), cc_status.value1))
                   1302:            cc_status.value1 = 0;
                   1303:          if (cc_status.value2
                   1304:              && reg_overlap_mentioned_p (SET_DEST (exp), cc_status.value2))
                   1305:            cc_status.value2 = 0; 
                   1306:        }
                   1307:       else if (!FP_REG_P (SET_DEST (exp))
                   1308:               && SET_DEST (exp) != cc0_rtx
                   1309:               && (FP_REG_P (SET_SRC (exp))
                   1310:                   || GET_CODE (SET_SRC (exp)) == FIX
                   1311:                   || GET_CODE (SET_SRC (exp)) == FLOAT_TRUNCATE
                   1312:                   || GET_CODE (SET_SRC (exp)) == FLOAT_EXTEND))
                   1313:        {
                   1314:          CC_STATUS_INIT; 
                   1315:        }
                   1316:       /* A pair of move insns doesn't produce a useful overall cc.  */
                   1317:       else if (!FP_REG_P (SET_DEST (exp))
                   1318:               && !FP_REG_P (SET_SRC (exp))
                   1319:               && GET_MODE_SIZE (GET_MODE (SET_SRC (exp))) > 4
                   1320:               && (GET_CODE (SET_SRC (exp)) == REG
                   1321:                   || GET_CODE (SET_SRC (exp)) == MEM
                   1322:                   || GET_CODE (SET_SRC (exp)) == CONST_DOUBLE))
                   1323:        {
                   1324:          CC_STATUS_INIT; 
                   1325:        }
                   1326:       else if (GET_CODE (SET_SRC (exp)) == CALL)
                   1327:        {
                   1328:          CC_STATUS_INIT; 
                   1329:        }
                   1330:       else if (XEXP (exp, 0) != pc_rtx)
                   1331:        {
                   1332:          cc_status.flags = 0;
                   1333:          cc_status.value1 = XEXP (exp, 0);
                   1334:          cc_status.value2 = XEXP (exp, 1);
                   1335:        }
                   1336:     }
                   1337:   else if (GET_CODE (exp) == PARALLEL
                   1338:           && GET_CODE (XVECEXP (exp, 0, 0)) == SET)
                   1339:     {
                   1340:       if (ADDRESS_REG_P (XEXP (XVECEXP (exp, 0, 0), 0)))
                   1341:        CC_STATUS_INIT;
                   1342:       else if (XEXP (XVECEXP (exp, 0, 0), 0) != pc_rtx)
                   1343:        {
                   1344:          cc_status.flags = 0;
                   1345:          cc_status.value1 = XEXP (XVECEXP (exp, 0, 0), 0);
                   1346:          cc_status.value2 = XEXP (XVECEXP (exp, 0, 0), 1);
                   1347:        }
                   1348:     }
                   1349:   else
                   1350:     CC_STATUS_INIT;
                   1351:   if (cc_status.value2 != 0
                   1352:       && ADDRESS_REG_P (cc_status.value2)
                   1353:       && GET_MODE (cc_status.value2) == QImode)
                   1354:     CC_STATUS_INIT;
                   1355:   if (cc_status.value2 != 0
                   1356:       && !(cc_status.value1 && FPA_REG_P (cc_status.value1)))
                   1357:     switch (GET_CODE (cc_status.value2))
                   1358:       {
                   1359:       case PLUS: case MINUS: case MULT:
                   1360:       case DIV: case UDIV: case MOD: case UMOD: case NEG:
                   1361:       case ASHIFT: case LSHIFT: case ASHIFTRT: case LSHIFTRT:
                   1362:       case ROTATE: case ROTATERT:
                   1363:        if (GET_MODE (cc_status.value2) != VOIDmode)
                   1364:          cc_status.flags |= CC_NO_OVERFLOW;
                   1365:        break;
                   1366:       case ZERO_EXTEND:
                   1367:        /* (SET r1 (ZERO_EXTEND r2)) on this machine
                   1368:           ends with a move insn moving r2 in r2's mode.
                   1369:           Thus, the cc's are set for r2.
                   1370:           This can set N bit spuriously. */
                   1371:        cc_status.flags |= CC_NOT_NEGATIVE; 
                   1372:       }
                   1373:   if (cc_status.value1 && GET_CODE (cc_status.value1) == REG
                   1374:       && cc_status.value2
                   1375:       && reg_overlap_mentioned_p (cc_status.value1, cc_status.value2))
                   1376:     cc_status.value2 = 0;
                   1377:   if (((cc_status.value1 && FP_REG_P (cc_status.value1))
                   1378:        || (cc_status.value2 && FP_REG_P (cc_status.value2)))
                   1379:       && !((cc_status.value1 && FPA_REG_P (cc_status.value1))
                   1380:           || (cc_status.value2 && FPA_REG_P (cc_status.value2))))
                   1381:     cc_status.flags = CC_IN_68881;
                   1382: }
                   1383: 
                   1384: char *
                   1385: output_move_const_double (operands)
                   1386:      rtx *operands;
                   1387: {
                   1388: #ifdef SUPPORT_SUN_FPA
                   1389:   if (TARGET_FPA && FPA_REG_P (operands[0]))
                   1390:     {
                   1391:       int code = standard_sun_fpa_constant_p (operands[1]);
                   1392: 
                   1393:       if (code != 0)
                   1394:        {
                   1395:          static char buf[40];
                   1396: 
                   1397:          sprintf (buf, "fpmove%%.d %%%%%d,%%0", code & 0x1ff);
                   1398:          return buf;
                   1399:        }
                   1400:       return "fpmove%.d %1,%0";
                   1401:     }
                   1402:   else
                   1403: #endif
                   1404:     {
                   1405:       int code = standard_68881_constant_p (operands[1]);
                   1406: 
                   1407:       if (code != 0)
                   1408:        {
                   1409:          static char buf[40];
                   1410: 
                   1411:          sprintf (buf, "fmovecr %%#0x%x,%%0", code & 0xff);
                   1412:          return buf;
                   1413:        }
                   1414:       return "fmove%.d %1,%0";
                   1415:     }
                   1416: }
                   1417: 
                   1418: char *
                   1419: output_move_const_single (operands)
                   1420:      rtx *operands;
                   1421: {
                   1422: #ifdef SUPPORT_SUN_FPA
                   1423:   if (TARGET_FPA)
                   1424:     {
                   1425:       int code = standard_sun_fpa_constant_p (operands[1]);
                   1426: 
                   1427:       if (code != 0)
                   1428:        {
                   1429:          static char buf[40];
                   1430: 
                   1431:          sprintf (buf, "fpmove%%.s %%%%%d,%%0", code & 0x1ff);
                   1432:          return buf;
                   1433:        }
                   1434:       return "fpmove%.s %1,%0";
                   1435:     }
                   1436:   else
                   1437: #endif /* defined SUPPORT_SUN_FPA */
                   1438:     {
                   1439:       int code = standard_68881_constant_p (operands[1]);
                   1440: 
                   1441:       if (code != 0)
                   1442:        {
                   1443:          static char buf[40];
                   1444: 
                   1445:          sprintf (buf, "fmovecr %%#0x%x,%%0", code & 0xff);
                   1446:          return buf;
                   1447:        }
                   1448:       return "fmove%.s %f1,%0";
                   1449:     }
                   1450: }
                   1451: 
                   1452: /* Return nonzero if X, a CONST_DOUBLE, has a value that we can get
                   1453:    from the "fmovecr" instruction.
                   1454:    The value, anded with 0xff, gives the code to use in fmovecr
                   1455:    to get the desired constant.  */
                   1456: 
                   1457: /* This code has been fixed for cross-compilation. */
                   1458:   
                   1459: static int inited_68881_table = 0;
                   1460: 
                   1461: char *strings_68881[7] = {
                   1462:   "0.0",
                   1463:   "1.0",
                   1464:   "10.0",
                   1465:   "100.0",
                   1466:   "10000.0",
                   1467:   "1e8",
                   1468:   "1e16"
                   1469:   };
                   1470: 
                   1471: int codes_68881[7] = {
                   1472:   0x0f,
                   1473:   0x32,
                   1474:   0x33,
                   1475:   0x34,
                   1476:   0x35,
                   1477:   0x36,
                   1478:   0x37
                   1479:   };
                   1480: 
                   1481: REAL_VALUE_TYPE values_68881[7];
                   1482: 
                   1483: /* Set up values_68881 array by converting the decimal values
                   1484:    strings_68881 to binary.   */
                   1485: 
                   1486: void
                   1487: init_68881_table ()
                   1488: {
                   1489:   int i;
                   1490:   REAL_VALUE_TYPE r;
                   1491:   enum machine_mode mode;
                   1492: 
                   1493:   mode = DFmode;
                   1494:   for (i = 0; i < 7; i++)
                   1495:     {
                   1496:       if (i == 6)
                   1497:         mode = SFmode;
                   1498:       r = REAL_VALUE_ATOF (strings_68881[i], mode);
                   1499:       values_68881[i] = r;
                   1500:     }
                   1501:   inited_68881_table = 1;
                   1502: }
                   1503: 
                   1504: int
                   1505: standard_68881_constant_p (x)
                   1506:      rtx x;
                   1507: {
                   1508:   REAL_VALUE_TYPE r;
                   1509:   int i;
                   1510:   enum machine_mode mode;
                   1511: 
                   1512:   /* fmovecr must be emulated on the 68040, so it shouldn't be used at all. */
                   1513:   if (TARGET_68040)
                   1514:     return 0;
                   1515: 
                   1516: #ifndef REAL_ARITHMETIC
                   1517: #if HOST_FLOAT_FORMAT != TARGET_FLOAT_FORMAT
                   1518:   if (! flag_pretend_float)
                   1519:     return 0;
                   1520: #endif
                   1521: #endif
                   1522: 
                   1523:   if (! inited_68881_table)
                   1524:     init_68881_table ();
                   1525: 
                   1526:   REAL_VALUE_FROM_CONST_DOUBLE (r, x);
                   1527: 
                   1528:   for (i = 0; i < 6; i++)
                   1529:     {
                   1530:       if (REAL_VALUES_EQUAL (r, values_68881[i]))
                   1531:         return (codes_68881[i]);
                   1532:     }
                   1533:   
                   1534:   if (GET_MODE (x) == SFmode)
                   1535:     return 0;
                   1536: 
                   1537:   if (REAL_VALUES_EQUAL (r, values_68881[6]))
                   1538:     return (codes_68881[6]);
                   1539: 
                   1540:   /* larger powers of ten in the constants ram are not used
                   1541:      because they are not equal to a `double' C constant.  */
                   1542:   return 0;
                   1543: }
                   1544: 
                   1545: /* If X is a floating-point constant, return the logarithm of X base 2,
                   1546:    or 0 if X is not a power of 2.  */
                   1547: 
                   1548: int
                   1549: floating_exact_log2 (x)
                   1550:      rtx x;
                   1551: {
                   1552:   REAL_VALUE_TYPE r, r1;
                   1553:   int i;
                   1554: 
                   1555: #ifndef REAL_ARITHMETIC
                   1556: #if HOST_FLOAT_FORMAT != TARGET_FLOAT_FORMAT
                   1557:   if (! flag_pretend_float)
                   1558:     return 0;
                   1559: #endif
                   1560: #endif
                   1561: 
                   1562:   REAL_VALUE_FROM_CONST_DOUBLE (r, x);
                   1563: 
                   1564:   if (REAL_VALUES_LESS (r, dconst0))
                   1565:     return 0;
                   1566: 
                   1567:   r1 = dconst1;
                   1568:   i = 0;
                   1569:   while (REAL_VALUES_LESS (r1, r))
                   1570:     {
                   1571:       r1 = REAL_VALUE_LDEXP (dconst1, i);
                   1572:       if (REAL_VALUES_EQUAL (r1, r))
                   1573:         return i;
                   1574:       i = i + 1;
                   1575:     }
                   1576:   return 0;
                   1577: }
                   1578: 
                   1579: #ifdef SUPPORT_SUN_FPA
                   1580: /* Return nonzero if X, a CONST_DOUBLE, has a value that we can get
                   1581:    from the Sun FPA's constant RAM.
                   1582:    The value returned, anded with 0x1ff, gives the code to use in fpmove
                   1583:    to get the desired constant. */
                   1584: 
                   1585: static int inited_FPA_table = 0;
                   1586: 
                   1587: char *strings_FPA[38] = {
                   1588: /* small rationals */
                   1589:   "0.0",
                   1590:   "1.0",
                   1591:   "0.5",
                   1592:   "-1.0",
                   1593:   "2.0",
                   1594:   "3.0",
                   1595:   "4.0",
                   1596:   "8.0",
                   1597:   "0.25",
                   1598:   "0.125",
                   1599:   "10.0",
                   1600:   "-0.5",
                   1601: /* Decimal equivalents of double precision values */
                   1602:   "2.718281828459045091", /* D_E */
                   1603:   "6.283185307179586477", /* 2 pi */
                   1604:   "3.141592653589793116", /* D_PI */
                   1605:   "1.570796326794896619", /* pi/2 */
                   1606:   "1.414213562373095145", /* D_SQRT2 */
                   1607:   "0.7071067811865475244", /* 1/sqrt(2) */
                   1608:   "-1.570796326794896619", /* -pi/2 */
                   1609:   "1.442695040888963387", /* D_LOG2ofE */
                   1610:   "3.321928024887362182", /* D_LOG2of10 */
                   1611:   "0.6931471805599452862", /* D_LOGEof2 */
                   1612:   "2.302585092994045901", /* D_LOGEof10 */
                   1613:   "0.3010299956639811980", /* D_LOG10of2 */
                   1614:   "0.4342944819032518167", /* D_LOG10ofE */
                   1615: /* Decimal equivalents of single precision values */
                   1616:   "2.718281745910644531", /* S_E */
                   1617:   "6.283185307179586477", /* 2 pi */
                   1618:   "3.141592741012573242", /* S_PI */
                   1619:   "1.570796326794896619", /* pi/2 */
                   1620:   "1.414213538169860840", /* S_SQRT2 */
                   1621:   "0.7071067811865475244", /* 1/sqrt(2) */
                   1622:   "-1.570796326794896619", /* -pi/2 */
                   1623:   "1.442695021629333496", /* S_LOG2ofE */
                   1624:   "3.321928024291992188", /* S_LOG2of10 */
                   1625:   "0.6931471824645996094", /* S_LOGEof2 */
                   1626:   "2.302585124969482442", /* S_LOGEof10 */
                   1627:   "0.3010300099849700928", /* S_LOG10of2 */
                   1628:   "0.4342944920063018799", /* S_LOG10ofE */
                   1629: };
                   1630: 
                   1631: 
                   1632: int codes_FPA[38] = {
                   1633: /* small rationals */
                   1634:   0x200,
                   1635:   0xe,
                   1636:   0xf,
                   1637:   0x10,
                   1638:   0x11,
                   1639:   0xb1,
                   1640:   0x12,
                   1641:   0x13,
                   1642:   0x15,
                   1643:   0x16,
                   1644:   0x17,
                   1645:   0x2e,
                   1646: /* double precision */
                   1647:   0x8,
                   1648:   0x9,
                   1649:   0xa,
                   1650:   0xb,
                   1651:   0xc,
                   1652:   0xd,
                   1653:   0x27,
                   1654:   0x28,
                   1655:   0x29,
                   1656:   0x2a,
                   1657:   0x2b,
                   1658:   0x2c,
                   1659:   0x2d,
                   1660: /* single precision */
                   1661:   0x8,
                   1662:   0x9,
                   1663:   0xa,
                   1664:   0xb,
                   1665:   0xc,
                   1666:   0xd,
                   1667:   0x27,
                   1668:   0x28,
                   1669:   0x29,
                   1670:   0x2a,
                   1671:   0x2b,
                   1672:   0x2c,
                   1673:   0x2d
                   1674:   };
                   1675: 
                   1676: REAL_VALUE_TYPE values_FPA[38];
                   1677: 
                   1678: /* This code has been fixed for cross-compilation. */
                   1679: 
                   1680: void
                   1681: init_FPA_table ()
                   1682: {
                   1683:   enum machine_mode mode;
                   1684:   int i;
                   1685:   REAL_VALUE_TYPE r;
                   1686: 
                   1687:   mode = DFmode;
                   1688:   for (i = 0; i < 38; i++)
                   1689:     {
                   1690:       if (i == 25)
                   1691:         mode = SFmode;
                   1692:       r = REAL_VALUE_ATOF (strings_FPA[i], mode);
                   1693:       values_FPA[i] = r;
                   1694:     }
                   1695:   inited_FPA_table = 1;
                   1696: }
                   1697: 
                   1698: 
                   1699: int
                   1700: standard_sun_fpa_constant_p (x)
                   1701:      rtx x;
                   1702: {
                   1703:   REAL_VALUE_TYPE r;
                   1704:   int i;
                   1705: 
                   1706: #ifndef REAL_ARITHMETIC
                   1707: #if HOST_FLOAT_FORMAT != TARGET_FLOAT_FORMAT
                   1708:   if (! flag_pretend_float)
                   1709:     return 0;
                   1710: #endif
                   1711: #endif
                   1712: 
                   1713:   if (! inited_FPA_table)
                   1714:     init_FPA_table ();
                   1715: 
                   1716:   REAL_VALUE_FROM_CONST_DOUBLE (r, x);
                   1717: 
                   1718:   for (i=0; i<12; i++)
                   1719:     {
                   1720:       if (REAL_VALUES_EQUAL (r, values_FPA[i]))
                   1721:         return (codes_FPA[i]);
                   1722:     }
                   1723: 
                   1724:   if (GET_MODE (x) == SFmode)
                   1725:     {
                   1726:       for (i=25; i<38; i++)
                   1727:         {
                   1728:           if (REAL_VALUES_EQUAL (r, values_FPA[i]))
                   1729:             return (codes_FPA[i]);
                   1730:         }
                   1731:     }
                   1732:   else
                   1733:     {
                   1734:       for (i=12; i<25; i++)
                   1735:         {
                   1736:           if (REAL_VALUES_EQUAL (r, values_FPA[i]))
                   1737:             return (codes_FPA[i]);
                   1738:         }
                   1739:     }
                   1740:   return 0x0;
                   1741: }
                   1742: #endif /* define SUPPORT_SUN_FPA */
                   1743: 
                   1744: /* A C compound statement to output to stdio stream STREAM the
                   1745:    assembler syntax for an instruction operand X.  X is an RTL
                   1746:    expression.
                   1747: 
                   1748:    CODE is a value that can be used to specify one of several ways
                   1749:    of printing the operand.  It is used when identical operands
                   1750:    must be printed differently depending on the context.  CODE
                   1751:    comes from the `%' specification that was used to request
                   1752:    printing of the operand.  If the specification was just `%DIGIT'
                   1753:    then CODE is 0; if the specification was `%LTR DIGIT' then CODE
                   1754:    is the ASCII code for LTR.
                   1755: 
                   1756:    If X is a register, this macro should print the register's name.
                   1757:    The names can be found in an array `reg_names' whose type is
                   1758:    `char *[]'.  `reg_names' is initialized from `REGISTER_NAMES'.
                   1759: 
                   1760:    When the machine description has a specification `%PUNCT' (a `%'
                   1761:    followed by a punctuation character), this macro is called with
                   1762:    a null pointer for X and the punctuation character for CODE.
                   1763: 
                   1764:    The m68k specific codes are:
                   1765: 
                   1766:    '.' for dot needed in Motorola-style opcode names.
                   1767:    '-' for an operand pushing on the stack:
                   1768:        sp@-, -(sp) or -(%sp) depending on the style of syntax.
                   1769:    '+' for an operand pushing on the stack:
                   1770:        sp@+, (sp)+ or (%sp)+ depending on the style of syntax.
                   1771:    '@' for a reference to the top word on the stack:
                   1772:        sp@, (sp) or (%sp) depending on the style of syntax.
                   1773:    '#' for an immediate operand prefix (# in MIT and Motorola syntax
                   1774:        but & in SGS syntax).
                   1775:    '!' for the cc register (used in an `and to cc' insn).
                   1776:    '$' for the letter `s' in an op code, but only on the 68040.
                   1777:    '&' for the letter `d' in an op code, but only on the 68040.
                   1778:    '/' for register prefix needed by longlong.h.
                   1779: 
                   1780:    'b' for byte insn (no effect, on the Sun; this is for the ISI).
                   1781:    'd' to force memory addressing to be absolute, not relative.
                   1782:    'f' for float insn (print a CONST_DOUBLE as a float rather than in hex)
                   1783:    'w' for FPA insn (print a CONST_DOUBLE as a SunFPA constant rather
                   1784:        than directly).  Second part of 'y' below.
                   1785:    'x' for float insn (print a CONST_DOUBLE as a float rather than in hex),
                   1786:        or print pair of registers as rx:ry.
                   1787:    'y' for a FPA insn (print pair of registers as rx:ry).  This also outputs
                   1788:        CONST_DOUBLE's as SunFPA constant RAM registers if
                   1789:        possible, so it should not be used except for the SunFPA.
                   1790: 
                   1791:    */
                   1792: 
                   1793: void
                   1794: print_operand (file, op, letter)
                   1795:      FILE *file;               /* file to write to */
                   1796:      rtx op;                   /* operand to print */
                   1797:      int letter;               /* %<letter> or 0 */
                   1798: {
                   1799:   int i;
                   1800: 
                   1801:   if (letter == '.')
                   1802:     {
                   1803: #ifdef MOTOROLA
                   1804:       asm_fprintf (file, ".");
                   1805: #endif
                   1806:     }
                   1807:   else if (letter == '#')
                   1808:     {
                   1809:       asm_fprintf (file, "%0I");
                   1810:     }
                   1811:   else if (letter == '-')
                   1812:     {
                   1813: #ifdef MOTOROLA
                   1814:       asm_fprintf (file, "-(%Rsp)");
                   1815: #else
                   1816:       asm_fprintf (file, "%Rsp@-");
                   1817: #endif
                   1818:     }
                   1819:   else if (letter == '+')
                   1820:     {
                   1821: #ifdef MOTOROLA
                   1822:       asm_fprintf (file, "(%Rsp)+");
                   1823: #else
                   1824:       asm_fprintf (file, "%Rsp@+");
                   1825: #endif
                   1826:     }
                   1827:   else if (letter == '@')
                   1828:     {
                   1829: #ifdef MOTOROLA
                   1830:       asm_fprintf (file, "(%Rsp)");
                   1831: #else
                   1832:       asm_fprintf (file, "%Rsp@");
                   1833: #endif
                   1834:     }
                   1835:   else if (letter == '!')
                   1836:     {
                   1837:       asm_fprintf (file, "%Rfpcr");
                   1838:     }
                   1839:   else if (letter == '$')
                   1840:     {
                   1841:       if (TARGET_68040_ONLY)
                   1842:        {
                   1843:          fprintf (file, "s");
                   1844:        }
                   1845:     }
                   1846:   else if (letter == '&')
                   1847:     {
                   1848:       if (TARGET_68040_ONLY)
                   1849:        {
                   1850:          fprintf (file, "d");
                   1851:        }
                   1852:     }
                   1853:   else if (letter == '/')
                   1854:     {
                   1855:       asm_fprintf (file, "%R");
                   1856:     }
                   1857:   else if (GET_CODE (op) == REG)
                   1858:     {
                   1859:       if (REGNO (op) < 16
                   1860:          && (letter == 'y' || letter == 'x')
                   1861:          && GET_MODE (op) == DFmode)
                   1862:        {
                   1863:          fprintf (file, "%s:%s", reg_names[REGNO (op)],
                   1864:                   reg_names[REGNO (op)+1]);
                   1865:        }
                   1866:       else
                   1867:        {
                   1868:          fprintf (file, "%s", reg_names[REGNO (op)]);
                   1869:        }
                   1870:     }
                   1871:   else if (GET_CODE (op) == MEM)
                   1872:     {
                   1873:       output_address (XEXP (op, 0));
                   1874:       if (letter == 'd' && ! TARGET_68020
                   1875:          && CONSTANT_ADDRESS_P (XEXP (op, 0))
                   1876:          && !(GET_CODE (XEXP (op, 0)) == CONST_INT
                   1877:               && INTVAL (XEXP (op, 0)) < 0x8000
                   1878:               && INTVAL (XEXP (op, 0)) >= -0x8000))
                   1879:        {
                   1880:          fprintf (file, ":l");
                   1881:        }
                   1882:     }
                   1883: #ifdef SUPPORT_SUN_FPA
                   1884:   else if ((letter == 'y' || letter == 'w')
                   1885:           && GET_CODE (op) == CONST_DOUBLE
                   1886:           && (i = standard_sun_fpa_constant_p (op)))
                   1887:     {
                   1888:       fprintf (file, "%%%d", i & 0x1ff);
                   1889:     }
                   1890: #endif
                   1891:   else if (GET_CODE (op) == CONST_DOUBLE && GET_MODE (op) == SFmode)
                   1892:     {
                   1893:       REAL_VALUE_TYPE r;
                   1894:       REAL_VALUE_FROM_CONST_DOUBLE (r, op);
                   1895:       ASM_OUTPUT_FLOAT_OPERAND (letter, file, r);
                   1896:     }
                   1897:   else if (GET_CODE (op) == CONST_DOUBLE && GET_MODE (op) == XFmode)
                   1898:     {
                   1899:       REAL_VALUE_TYPE r;
                   1900:       REAL_VALUE_FROM_CONST_DOUBLE (r, op);
                   1901:       ASM_OUTPUT_LONG_DOUBLE_OPERAND (file, r);
                   1902:     }
                   1903:   else if (GET_CODE (op) == CONST_DOUBLE && GET_MODE (op) == DFmode)
                   1904:     {
                   1905:       REAL_VALUE_TYPE r;
                   1906:       REAL_VALUE_FROM_CONST_DOUBLE (r, op);
                   1907:       ASM_OUTPUT_DOUBLE_OPERAND (file, r);
                   1908:     }
                   1909:   else
                   1910:     {
                   1911:       asm_fprintf (file, "%0I"); output_addr_const (file, op);
                   1912:     }
                   1913: }
                   1914: 
                   1915: 
                   1916: /* A C compound statement to output to stdio stream STREAM the
                   1917:    assembler syntax for an instruction operand that is a memory
                   1918:    reference whose address is ADDR.  ADDR is an RTL expression.
                   1919: 
                   1920:    Note that this contains a kludge that knows that the only reason
                   1921:    we have an address (plus (label_ref...) (reg...)) when not generating
                   1922:    PIC code is in the insn before a tablejump, and we know that m68k.md
                   1923:    generates a label LInnn: on such an insn.
                   1924: 
                   1925:    It is possible for PIC to generate a (plus (label_ref...) (reg...))
                   1926:    and we handle that just like we would a (plus (symbol_ref...) (reg...)).
                   1927: 
                   1928:    Some SGS assemblers have a bug such that "Lnnn-LInnn-2.b(pc,d0.l*2)"
                   1929:    fails to assemble.  Luckily "Lnnn(pc,d0.l*2)" produces the results
                   1930:    we want.  This difference can be accommodated by using an assembler
                   1931:    define such "LDnnn" to be either "Lnnn-LInnn-2.b", "Lnnn", or any other
                   1932:    string, as necessary.  This is accomplished via the ASM_OUTPUT_CASE_END
                   1933:    macro.  See m68k/sgs.h for an example; for versions without the bug.
                   1934: 
                   1935:    They also do not like things like "pea 1.w", so we simple leave off
                   1936:    the .w on small constants. 
                   1937: 
                   1938:    This routine is responsible for distinguishing between -fpic and -fPIC 
                   1939:    style relocations in an address.  When generating -fpic code the
                   1940:    offset is output in word mode (eg movel a5@(_foo:w), a0).  When generating
                   1941:    -fPIC code the offset is output in long mode (eg movel a5@(_foo:l), a0) */
                   1942: 
                   1943: void
                   1944: print_operand_address (file, addr)
                   1945:      FILE *file;
                   1946:      rtx addr;
                   1947: {
                   1948:   register rtx reg1, reg2, breg, ireg;
                   1949:   rtx offset;
                   1950: 
                   1951:   switch (GET_CODE (addr))
                   1952:     {
                   1953:       case REG:
                   1954: #ifdef MOTOROLA
                   1955:        fprintf (file, "(%s)", reg_names[REGNO (addr)]);
                   1956: #else
                   1957:        fprintf (file, "%s@", reg_names[REGNO (addr)]);
                   1958: #endif
                   1959:        break;
                   1960:       case PRE_DEC:
                   1961: #ifdef MOTOROLA
                   1962:        fprintf (file, "-(%s)", reg_names[REGNO (XEXP (addr, 0))]);
                   1963: #else
                   1964:        fprintf (file, "%s@-", reg_names[REGNO (XEXP (addr, 0))]);
                   1965: #endif
                   1966:        break;
                   1967:       case POST_INC:
                   1968: #ifdef MOTOROLA
                   1969:        fprintf (file, "(%s)+", reg_names[REGNO (XEXP (addr, 0))]);
                   1970: #else
                   1971:        fprintf (file, "%s@+", reg_names[REGNO (XEXP (addr, 0))]);
                   1972: #endif
                   1973:        break;
                   1974:       case PLUS:
                   1975:        reg1 = reg2 = ireg = breg = offset = 0;
                   1976:        if (CONSTANT_ADDRESS_P (XEXP (addr, 0)))
                   1977:          {
                   1978:            offset = XEXP (addr, 0);
                   1979:            addr = XEXP (addr, 1);
                   1980:          }
                   1981:        else if (CONSTANT_ADDRESS_P (XEXP (addr, 1)))
                   1982:          {
                   1983:            offset = XEXP (addr, 1);
                   1984:            addr = XEXP (addr, 0);
                   1985:          }
                   1986:        if (GET_CODE (addr) != PLUS)
                   1987:          {
                   1988:            ;
                   1989:          }
                   1990:        else if (GET_CODE (XEXP (addr, 0)) == SIGN_EXTEND)
                   1991:          {
                   1992:            reg1 = XEXP (addr, 0);
                   1993:            addr = XEXP (addr, 1);
                   1994:          }
                   1995:        else if (GET_CODE (XEXP (addr, 1)) == SIGN_EXTEND)
                   1996:          {
                   1997:            reg1 = XEXP (addr, 1);
                   1998:            addr = XEXP (addr, 0);
                   1999:          }
                   2000:        else if (GET_CODE (XEXP (addr, 0)) == MULT)
                   2001:          {
                   2002:            reg1 = XEXP (addr, 0);
                   2003:            addr = XEXP (addr, 1);
                   2004:          }
                   2005:        else if (GET_CODE (XEXP (addr, 1)) == MULT)
                   2006:          {
                   2007:            reg1 = XEXP (addr, 1);
                   2008:            addr = XEXP (addr, 0);
                   2009:          }
                   2010:        else if (GET_CODE (XEXP (addr, 0)) == REG)
                   2011:          {
                   2012:            reg1 = XEXP (addr, 0);
                   2013:            addr = XEXP (addr, 1);
                   2014:          }
                   2015:        else if (GET_CODE (XEXP (addr, 1)) == REG)
                   2016:          {
                   2017:            reg1 = XEXP (addr, 1);
                   2018:            addr = XEXP (addr, 0);
                   2019:          }
                   2020:        if (GET_CODE (addr) == REG || GET_CODE (addr) == MULT
                   2021:            || GET_CODE (addr) == SIGN_EXTEND)
                   2022:          {
                   2023:            if (reg1 == 0)
                   2024:              {
                   2025:                reg1 = addr;
                   2026:              }
                   2027:            else
                   2028:              {
                   2029:                reg2 = addr;
                   2030:              }
                   2031:            addr = 0;
                   2032:          }
                   2033: #if 0  /* for OLD_INDEXING */
                   2034:        else if (GET_CODE (addr) == PLUS)
                   2035:          {
                   2036:            if (GET_CODE (XEXP (addr, 0)) == REG)
                   2037:              {
                   2038:                reg2 = XEXP (addr, 0);
                   2039:                addr = XEXP (addr, 1);
                   2040:              }
                   2041:            else if (GET_CODE (XEXP (addr, 1)) == REG)
                   2042:              {
                   2043:                reg2 = XEXP (addr, 1);
                   2044:                addr = XEXP (addr, 0);
                   2045:              }
                   2046:          }
                   2047: #endif
                   2048:        if (offset != 0)
                   2049:          {
                   2050:            if (addr != 0)
                   2051:              {
                   2052:                abort ();
                   2053:              }
                   2054:            addr = offset;
                   2055:          }
                   2056:        if ((reg1 && (GET_CODE (reg1) == SIGN_EXTEND
                   2057:                      || GET_CODE (reg1) == MULT))
                   2058:            || (reg2 != 0 && REGNO_OK_FOR_BASE_P (REGNO (reg2))))
                   2059:          {
                   2060:            breg = reg2;
                   2061:            ireg = reg1;
                   2062:          }
                   2063:        else if (reg1 != 0 && REGNO_OK_FOR_BASE_P (REGNO (reg1)))
                   2064:          {
                   2065:            breg = reg1;
                   2066:            ireg = reg2;
                   2067:          }
                   2068:        if (ireg != 0 && breg == 0 && GET_CODE (addr) == LABEL_REF
                   2069:            && ! (flag_pic && ireg == pic_offset_table_rtx))
                   2070:          {
                   2071:            int scale = 1;
                   2072:            if (GET_CODE (ireg) == MULT)
                   2073:              {
                   2074:                scale = INTVAL (XEXP (ireg, 1));
                   2075:                ireg = XEXP (ireg, 0);
                   2076:              }
                   2077:            if (GET_CODE (ireg) == SIGN_EXTEND)
                   2078:              {
                   2079: #ifdef MOTOROLA
                   2080: #ifdef SGS
                   2081:                asm_fprintf (file, "%LLD%d(%Rpc,%s.w",
                   2082:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2083:                             reg_names[REGNO (XEXP (ireg, 0))]);
                   2084: #else
                   2085:                asm_fprintf (file, "%LL%d-%LLI%d.b(%Rpc,%s.w",
                   2086:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2087:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2088:                             reg_names[REGNO (XEXP (ireg, 0))]);
                   2089: #endif
                   2090: #else
                   2091: #ifdef MACHO_PIC
                   2092:                asm_fprintf (file, "%Ra5@(%LL%d-",
                   2093:                             CODE_LABEL_NUMBER (XEXP (addr, 0)));
                   2094:                assemble_name (file, machopic_function_base_name ());
                   2095:                asm_fprintf (file, ",%s:w", reg_names[REGNO (ireg)]);
                   2096: #else
                   2097:                asm_fprintf (file, "%Rpc@(%LL%d-%LLI%d-2:b,%s:w",
                   2098:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2099:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2100:                             reg_names[REGNO (XEXP (ireg, 0))]);
                   2101: #endif
                   2102: #endif
                   2103:              }
                   2104:            else
                   2105:              {
                   2106: #ifdef MOTOROLA
                   2107: #ifdef SGS
                   2108:                asm_fprintf (file, "%LLD%d(%Rpc,%s.l",
                   2109:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2110:                             reg_names[REGNO (ireg)]);
                   2111: #else
                   2112:                asm_fprintf (file, "%LL%d-%LLI%d.b(%Rpc,%s.l",
                   2113:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2114:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2115:                             reg_names[REGNO (ireg)]);
                   2116: #endif
                   2117: #else
                   2118: #ifdef MACHO_PIC
                   2119:                asm_fprintf (file, "%Ra5@(%LL%d-",
                   2120:                             CODE_LABEL_NUMBER (XEXP (addr, 0)));
                   2121:                assemble_name (file, machopic_function_base_name ());
                   2122:                asm_fprintf (file, ",%s:l", reg_names[REGNO (ireg)]);
                   2123: #else
                   2124:                asm_fprintf (file, "%Rpc@(%LL%d-%LLI%d-2:b,%s:l",
                   2125:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2126:                             CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2127:                             reg_names[REGNO (ireg)]);
                   2128: #endif
                   2129: #endif
                   2130:              }
                   2131:            if (scale != 1)
                   2132:              {
                   2133: #ifdef MOTOROLA
                   2134:                fprintf (file, "*%d", scale);
                   2135: #else
                   2136:                fprintf (file, ":%d", scale);
                   2137: #endif
                   2138:              }
                   2139:            putc (')', file);
                   2140:            break;
                   2141:          }
                   2142:        if (breg != 0 && ireg == 0 && GET_CODE (addr) == LABEL_REF
                   2143:            && ! (flag_pic && breg == pic_offset_table_rtx))
                   2144:          {
                   2145: #ifdef MOTOROLA
                   2146: #ifdef SGS
                   2147:            asm_fprintf (file, "%LLD%d(%Rpc,%s.l",
                   2148:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2149:                         reg_names[REGNO (breg)]);
                   2150: #else
                   2151:            asm_fprintf (file, "%LL%d-%LLI%d.b(%Rpc,%s.l",
                   2152:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2153:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2154:                         reg_names[REGNO (breg)]);
                   2155: #endif
                   2156: #else
                   2157:            asm_fprintf (file, "%Rpc@(%LL%d-%LLI%d-2:b,%s:l",
                   2158:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2159:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2160:                         reg_names[REGNO (breg)]);
                   2161: #endif
                   2162:            putc (')', file);
                   2163:            break;
                   2164:          }
                   2165:        if (ireg != 0 || breg != 0)
                   2166:          {
                   2167:            int scale = 1;
                   2168:            if (breg == 0)
                   2169:              {
                   2170:                abort ();
                   2171:              }
                   2172:            if (! flag_pic && addr && GET_CODE (addr) == LABEL_REF)
                   2173:              {
                   2174:                abort ();
                   2175:              }
                   2176: #ifdef MOTOROLA
                   2177:            if (addr != 0)
                   2178:              {
                   2179:                output_addr_const (file, addr);
                   2180:                if (flag_pic && (breg == pic_offset_table_rtx))
                   2181:                  fprintf (file, "@GOT");
                   2182:              }
                   2183:            fprintf (file, "(%s", reg_names[REGNO (breg)]);
                   2184:            if (ireg != 0)
                   2185:              {
                   2186:                putc (',', file);
                   2187:              }
                   2188: #else
                   2189:            fprintf (file, "%s@(", reg_names[REGNO (breg)]);
                   2190:            if (addr != 0)
                   2191:              {
                   2192:                output_addr_const (file, addr);
                   2193: #ifndef MACHO_PIC
                   2194:                if ((flag_pic == 1) && (breg == pic_offset_table_rtx))
                   2195:                  fprintf (file, ":w");
                   2196:                if ((flag_pic == 2) && (breg == pic_offset_table_rtx))
                   2197:                  fprintf (file, ":l");
                   2198: #endif
                   2199:              }
                   2200:            if (addr != 0 && ireg != 0)
                   2201:              {
                   2202:                putc (',', file);
                   2203:              }
                   2204: #endif
                   2205:            if (ireg != 0 && GET_CODE (ireg) == MULT)
                   2206:              {
                   2207:                scale = INTVAL (XEXP (ireg, 1));
                   2208:                ireg = XEXP (ireg, 0);
                   2209:              }
                   2210:            if (ireg != 0 && GET_CODE (ireg) == SIGN_EXTEND)
                   2211:              {
                   2212: #ifdef MOTOROLA
                   2213:                fprintf (file, "%s.w", reg_names[REGNO (XEXP (ireg, 0))]);
                   2214: #else
                   2215:                fprintf (file, "%s:w", reg_names[REGNO (XEXP (ireg, 0))]);
                   2216: #endif
                   2217:              }
                   2218:            else if (ireg != 0)
                   2219:              {
                   2220: #ifdef MOTOROLA
                   2221:                fprintf (file, "%s.l", reg_names[REGNO (ireg)]);
                   2222: #else
                   2223:                fprintf (file, "%s:l", reg_names[REGNO (ireg)]);
                   2224: #endif
                   2225:              }
                   2226:            if (scale != 1)
                   2227:              {
                   2228: #ifdef MOTOROLA
                   2229:                fprintf (file, "*%d", scale);
                   2230: #else
                   2231:                fprintf (file, ":%d", scale);
                   2232: #endif
                   2233:              }
                   2234:            putc (')', file);
                   2235:            break;
                   2236:          }
                   2237:        else if (reg1 != 0 && GET_CODE (addr) == LABEL_REF
                   2238:                 && ! (flag_pic && reg1 == pic_offset_table_rtx))       
                   2239:          {
                   2240: #ifdef MOTOROLA
                   2241: #ifdef SGS
                   2242:            asm_fprintf (file, "%LLD%d(%Rpc,%s.l)",
                   2243:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2244:                         reg_names[REGNO (reg1)]);
                   2245: #else
                   2246:            asm_fprintf (file, "%LL%d-%LLI%d.b(%Rpc,%s.l)",
                   2247:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2248:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2249:                         reg_names[REGNO (reg1)]);
                   2250: #endif
                   2251: #else
                   2252:            asm_fprintf (file, "%Rpc@(%LL%d-%LLI%d-2:b,%s:l)",
                   2253:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2254:                         CODE_LABEL_NUMBER (XEXP (addr, 0)),
                   2255:                         reg_names[REGNO (reg1)]);
                   2256: #endif
                   2257:            break;
                   2258:          }
                   2259:        /* FALL-THROUGH (is this really what we want? */
                   2260:       default:
                   2261:         if (GET_CODE (addr) == CONST_INT
                   2262:            && INTVAL (addr) < 0x8000
                   2263:            && INTVAL (addr) >= -0x8000)
                   2264:          {
                   2265: #ifdef MOTOROLA
                   2266: #ifdef SGS
                   2267:            /* Many SGS assemblers croak on size specifiers for constants. */
                   2268:            fprintf (file, "%d", INTVAL (addr));
                   2269: #else
                   2270:            fprintf (file, "%d.w", INTVAL (addr));
                   2271: #endif
                   2272: #else
                   2273:            fprintf (file, "%d:w", INTVAL (addr));
                   2274: #endif
                   2275:          }
                   2276:        else
                   2277:          {
                   2278:            output_addr_const (file, addr);
                   2279:          }
                   2280:        break;
                   2281:     }
                   2282: }
                   2283: 
                   2284: /* Check for cases where a clr insns can be omitted from code using
                   2285:    strict_low_part sets.  For example, the second clrl here is not needed:
                   2286:    clrl d0; movw a0@+,d0; use d0; clrl d0; movw a0@+; use d0; ...
                   2287: 
                   2288:    MODE is the mode of this STRICT_LOW_PART set.  FIRST_INSN is the clear
                   2289:    insn we are checking for redundancy.  TARGET is the register set by the
                   2290:    clear insn.  */
                   2291: 
                   2292: int
                   2293: strict_low_part_peephole_ok (mode, first_insn, target)
                   2294:      enum machine_mode mode;
                   2295:      rtx first_insn;
                   2296:      rtx target;
                   2297: {
                   2298:   rtx p;
                   2299: 
                   2300:   p = prev_nonnote_insn (first_insn);
                   2301: 
                   2302:   while (p)
                   2303:     {
                   2304:       /* If it isn't an insn, then give up.  */
                   2305:       if (GET_CODE (p) != INSN)
                   2306:        return 0;
                   2307: 
                   2308:       if (reg_set_p (target, p))
                   2309:        {
                   2310:          rtx set = single_set (p);
                   2311:          rtx dest;
                   2312: 
                   2313:          /* If it isn't an easy to recognize insn, then give up.  */
                   2314:          if (! set)
                   2315:            return 0;
                   2316: 
                   2317:          dest = SET_DEST (set);
                   2318: 
                   2319:          /* If this sets the entire target register to zero, then our
                   2320:             first_insn is redundant.  */
                   2321:          if (rtx_equal_p (dest, target)
                   2322:              && SET_SRC (set) == const0_rtx)
                   2323:            return 1;
                   2324:          else if (GET_CODE (dest) == STRICT_LOW_PART
                   2325:                   && GET_CODE (XEXP (dest, 0)) == REG
                   2326:                   && REGNO (XEXP (dest, 0)) == REGNO (target)
                   2327:                   && (GET_MODE_SIZE (GET_MODE (XEXP (dest, 0)))
                   2328:                       <= GET_MODE_SIZE (mode)))
                   2329:            /* This is a strict low part set which modifies less than
                   2330:               we are using, so it is safe.  */
                   2331:            ;
                   2332:          else
                   2333:            return 0;
                   2334:        }
                   2335: 
                   2336:       p = prev_nonnote_insn (p);
                   2337: 
                   2338:     }
                   2339: 
                   2340:   return 0;
                   2341: }

unix.superglobalmegacorp.com

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