Annotation of researchv10dc/cmd/gcc/output-spur.c, revision 1.1

1.1     ! root        1: /* Subroutines for insn-output.c for Motorola 68000 family.
        !             2:    Copyright (C) 1988 Free Software Foundation, Inc.
        !             3: 
        !             4: This file is part of GNU CC.
        !             5: 
        !             6: GNU CC is distributed in the hope that it will be useful,
        !             7: but WITHOUT ANY WARRANTY.  No author or distributor
        !             8: accepts responsibility to anyone for the consequences of using it
        !             9: or for whether it serves any particular purpose or works at all,
        !            10: unless he says so in writing.  Refer to the GNU CC General Public
        !            11: License for full details.
        !            12: 
        !            13: Everyone is granted permission to copy, modify and redistribute
        !            14: GNU CC, but only under the conditions described in the
        !            15: GNU CC General Public License.   A copy of this license is
        !            16: supposed to have been given to you along with GNU CC so you
        !            17: can know your rights and responsibilities.  It should be in a
        !            18: file named COPYING.  Among other things, the copyright notice
        !            19: and this notice must be preserved on all copies.  */
        !            20: 
        !            21: static rtx find_addr_reg ();
        !            22: 
        !            23: char *
        !            24: output_compare (operands, opcode, exchange_opcode)
        !            25:      rtx *operands;
        !            26:      char *opcode;
        !            27:      char *exchange_opcode;
        !            28: {
        !            29:   static char buf[40];
        !            30:   operands[2] = operands[0];
        !            31:   if (GET_CODE (cc_prev_status.value1) == CONST_INT)
        !            32:     {
        !            33:       operands[1] = cc_prev_status.value1;
        !            34:       operands[0] = cc_prev_status.value2;
        !            35:       opcode = exchange_opcode;
        !            36:     }
        !            37:   else
        !            38:     {
        !            39:       operands[0] = cc_prev_status.value1;
        !            40:       operands[1] = cc_prev_status.value2;
        !            41:     }
        !            42:   sprintf (buf, "cmp_br_delayed %s,%%0,%%1,%%l2\n\tnop", opcode);
        !            43:   return buf;
        !            44: }
        !            45: 
        !            46: /* Return the best assembler insn template
        !            47:    for moving operands[1] into operands[0] as a fullword.  */
        !            48: 
        !            49: static char *
        !            50: singlemove_string (operands)
        !            51:      rtx *operands;
        !            52: {
        !            53:   if (GET_CODE (operands[0]) == MEM)
        !            54:     return "st_32 %r1,%0";
        !            55:   if (GET_CODE (operands[1]) == MEM)
        !            56:     return "ld_32 %0,%1\n\tnop";
        !            57:   return "add_nt %0,r0,%1";
        !            58: }
        !            59: 
        !            60: /* Output assembler code to perform a doubleword move insn
        !            61:    with operands OPERANDS.  */
        !            62: 
        !            63: char *
        !            64: output_move_double (operands)
        !            65:      rtx *operands;
        !            66: {
        !            67:   enum { REGOP, OFFSOP, MEMOP, PUSHOP, POPOP, CNSTOP, RNDOP } optype0, optype1;
        !            68:   rtx latehalf[2];
        !            69:   rtx addreg0 = 0, addreg1 = 0;
        !            70: 
        !            71:   /* First classify both operands.  */
        !            72: 
        !            73:   if (REG_P (operands[0]))
        !            74:     optype0 = REGOP;
        !            75:   else if (offsetable_memref_p (operands[0]))
        !            76:     optype0 = OFFSOP;
        !            77:   else if (GET_CODE (operands[0]) == MEM)
        !            78:     optype0 = MEMOP;
        !            79:   else
        !            80:     optype0 = RNDOP;
        !            81: 
        !            82:   if (REG_P (operands[1]))
        !            83:     optype1 = REGOP;
        !            84:   else if (CONSTANT_P (operands[1])
        !            85:           || GET_CODE (operands[1]) == CONST_DOUBLE)
        !            86:     optype1 = CNSTOP;
        !            87:   else if (offsetable_memref_p (operands[1]))
        !            88:     optype1 = OFFSOP;
        !            89:   else if (GET_CODE (operands[1]) == MEM)
        !            90:     optype0 = MEMOP;
        !            91:   else
        !            92:     optype1 = RNDOP;
        !            93: 
        !            94:   /* Check for the cases that the operand constraints are not
        !            95:      supposed to allow to happen.  Abort if we get one,
        !            96:      because generating code for these cases is painful.  */
        !            97: 
        !            98:   if (optype0 == RNDOP || optype1 == RNDOP)
        !            99:     abort ();
        !           100: 
        !           101:   /* If an operand is an unoffsettable memory ref, find a register
        !           102:      we can increment temporarily to make it refer to the second word.  */
        !           103: 
        !           104:   if (optype0 == MEMOP)
        !           105:     addreg0 = find_addr_reg (operands[0]);
        !           106: 
        !           107:   if (optype1 == MEMOP)
        !           108:     addreg1 = find_addr_reg (operands[1]);
        !           109: 
        !           110:   /* Ok, we can do one word at a time.
        !           111:      Normally we do the low-numbered word first,
        !           112:      but if either operand is autodecrementing then we
        !           113:      do the high-numbered word first.
        !           114: 
        !           115:      In either case, set up in LATEHALF the operands to use
        !           116:      for the high-numbered word and in some cases alter the
        !           117:      operands in OPERANDS to be suitable for the low-numbered word.  */
        !           118: 
        !           119:   if (optype0 == REGOP)
        !           120:     latehalf[0] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1);
        !           121:   else if (optype0 == OFFSOP)
        !           122:     latehalf[0] = adj_offsetable_operand (operands[0], 4);
        !           123:   else
        !           124:     latehalf[0] = operands[0];
        !           125: 
        !           126:   if (optype1 == REGOP)
        !           127:     latehalf[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1);
        !           128:   else if (optype1 == OFFSOP)
        !           129:     latehalf[1] = adj_offsetable_operand (operands[1], 4);
        !           130:   else if (optype1 == CNSTOP)
        !           131:     {
        !           132:       if (CONSTANT_P (operands[1]))
        !           133:        latehalf[1] = const0_rtx;
        !           134:       else if (GET_CODE (operands[1]) == CONST_DOUBLE)
        !           135:        {
        !           136:          latehalf[1] = gen_rtx (CONST_INT, VOIDmode, XINT (operands[1], 1));
        !           137:          operands[1] = gen_rtx (CONST_INT, VOIDmode, XINT (operands[1], 0));
        !           138:        }
        !           139:     }
        !           140:   else
        !           141:     latehalf[1] = operands[1];
        !           142: 
        !           143:   /* If the first move would clobber the source of the second one,
        !           144:      do them in the other order.  This happens only for registers;
        !           145:      such overlap can't happen in memory unless the user explicitly
        !           146:      sets it up, and that is an undefined circumstance.  */
        !           147: 
        !           148:   if (optype0 == REGOP && optype1 == REGOP
        !           149:       && REGNO (operands[0]) == REGNO (latehalf[1]))
        !           150:     {
        !           151:       /* Make any unoffsetable addresses point at high-numbered word.  */
        !           152:       if (addreg0)
        !           153:        output_asm_insn ("add_nt %0,%0,$4", &addreg0);
        !           154:       if (addreg1)
        !           155:        output_asm_insn ("add_nt %0,%0,$4", &addreg1);
        !           156: 
        !           157:       /* Do that word.  */
        !           158:       output_asm_insn (singlemove_string (latehalf), latehalf);
        !           159: 
        !           160:       /* Undo the adds we just did.  */
        !           161:       if (addreg0)
        !           162:        output_asm_insn ("add_nt %0,%0,$-4", &addreg0);
        !           163:       if (addreg1)
        !           164:        output_asm_insn ("add_nt %0,%0,$-4", &addreg0);
        !           165: 
        !           166:       /* Do low-numbered word.  */
        !           167:       return singlemove_string (operands);
        !           168:     }
        !           169: 
        !           170:   /* Normal case: do the two words, low-numbered first.  */
        !           171: 
        !           172:   output_asm_insn (singlemove_string (operands), operands);
        !           173: 
        !           174:   /* Make any unoffsetable addresses point at high-numbered word.  */
        !           175:   if (addreg0)
        !           176:     output_asm_insn ("add_nt %0,%0,$4", &addreg0);
        !           177:   if (addreg1)
        !           178:     output_asm_insn ("add_nt %0,%0,$4", &addreg1);
        !           179: 
        !           180:   /* Do that word.  */
        !           181:   output_asm_insn (singlemove_string (latehalf), latehalf);
        !           182: 
        !           183:   /* Undo the adds we just did.  */
        !           184:   if (addreg0)
        !           185:     output_asm_insn ("add_nt %0,%0,$-4", &addreg0);
        !           186:   if (addreg1)
        !           187:     output_asm_insn ("add_nt %0,%0,$-4", &addreg1);
        !           188: 
        !           189:   return "";
        !           190: }
        !           191: 
        !           192: static char *
        !           193: output_fp_move_double (operands)
        !           194:      rtx *operands;
        !           195: {
        !           196:   if (FP_REG_P (operands[0]))
        !           197:     {
        !           198:       if (FP_REG_P (operands[1]))
        !           199:        return "fmov %0,%1";
        !           200:       if (GET_CODE (operands[1]) == REG)
        !           201:        {
        !           202:          rtx xoperands[2];
        !           203:          int offset = - get_frame_size () - 8;
        !           204:          xoperands[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1);
        !           205:          xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset + 4);
        !           206:          output_asm_insn ("st_32 %1,r25,%0", xoperands);
        !           207:          xoperands[1] = operands[1];
        !           208:          xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset);
        !           209:          output_asm_insn ("st_32 %1,r25,%0", xoperands);
        !           210:          xoperands[1] = operands[0];
        !           211:          output_asm_insn ("ld_dbl %1,r25,%0\n\tnop", xoperands);
        !           212:          return "";
        !           213:        }
        !           214:       return "ld_dbl %0,%1\n\tnop";
        !           215:     }
        !           216:   else if (FP_REG_P (operands[1]))
        !           217:     {
        !           218:       if (GET_CODE (operands[0]) == REG)
        !           219:        {
        !           220:          rtx xoperands[2];
        !           221:          int offset = - get_frame_size () - 8;
        !           222:          xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset);
        !           223:          xoperands[1] = operands[1];
        !           224:          output_asm_insn ("st_dbl %1,r25,%0", xoperands);
        !           225:          xoperands[1] = operands[0];
        !           226:          output_asm_insn ("ld_32 %1,r25,%0\n\tnop", xoperands);
        !           227:          xoperands[1] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1);
        !           228:          xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset + 4);
        !           229:          output_asm_insn ("ld_32 %1,r25,%0\n\tnop", xoperands);
        !           230:          return "";
        !           231:        }
        !           232:       return "st_dbl %1,%0";
        !           233:     }
        !           234: }
        !           235: 
        !           236: /* Return a REG that occurs in ADDR with coefficient 1.
        !           237:    ADDR can be effectively incremented by incrementing REG.  */
        !           238: 
        !           239: static rtx
        !           240: find_addr_reg (addr)
        !           241:      rtx addr;
        !           242: {
        !           243:   while (GET_CODE (addr) == PLUS)
        !           244:     {
        !           245:       if (GET_CODE (XEXP (addr, 0)) == REG)
        !           246:        addr = XEXP (addr, 0);
        !           247:       if (GET_CODE (XEXP (addr, 1)) == REG)
        !           248:        addr = XEXP (addr, 1);
        !           249:       if (CONSTANT_P (XEXP (addr, 0)))
        !           250:        addr = XEXP (addr, 1);
        !           251:       if (CONSTANT_P (XEXP (addr, 1)))
        !           252:        addr = XEXP (addr, 0);
        !           253:     }
        !           254:   if (GET_CODE (addr) == REG)
        !           255:     return addr;
        !           256:   return 0;
        !           257: }

unix.superglobalmegacorp.com

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