|
|
1.1 ! root 1: /* Subroutines for insn-output.c for SPUR. Adapted from routines for ! 2: the Motorola 68000 family. ! 3: Copyright (C) 1988, 1991 Free Software Foundation, Inc. ! 4: ! 5: This file is part of GNU CC. ! 6: ! 7: GNU CC is free software; you can redistribute it and/or modify ! 8: it under the terms of the GNU General Public License as published by ! 9: the Free Software Foundation; either version 2, or (at your option) ! 10: any later version. ! 11: ! 12: GNU CC is distributed in the hope that it will be useful, ! 13: but WITHOUT ANY WARRANTY; without even the implied warranty of ! 14: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ! 15: GNU General Public License for more details. ! 16: ! 17: You should have received a copy of the GNU General Public License ! 18: along with GNU CC; see the file COPYING. If not, write to ! 19: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ ! 20: ! 21: #include "config.h" ! 22: #include "rtl.h" ! 23: #include "regs.h" ! 24: #include "hard-reg-set.h" ! 25: #include "real.h" ! 26: #include "insn-config.h" ! 27: #include "conditions.h" ! 28: #include "insn-flags.h" ! 29: #include "output.h" ! 30: #include "insn-attr.h" ! 31: ! 32: static rtx find_addr_reg (); ! 33: ! 34: char * ! 35: output_compare (operands, opcode, exchange_opcode, ! 36: neg_opcode, neg_exchange_opcode) ! 37: rtx *operands; ! 38: char *opcode; ! 39: char *exchange_opcode; ! 40: char *neg_opcode; ! 41: char *neg_exchange_opcode; ! 42: { ! 43: static char buf[100]; ! 44: operands[2] = operands[0]; ! 45: if (GET_CODE (cc_prev_status.value1) == CONST_INT) ! 46: { ! 47: operands[1] = cc_prev_status.value1; ! 48: operands[0] = cc_prev_status.value2; ! 49: opcode = exchange_opcode, neg_opcode = neg_exchange_opcode; ! 50: } ! 51: else ! 52: { ! 53: operands[0] = cc_prev_status.value1; ! 54: operands[1] = cc_prev_status.value2; ! 55: } ! 56: if (TARGET_LONG_JUMPS) ! 57: sprintf (buf, ! 58: "cmp_br_delayed %s,%%0,%%1,1f\n\tnop\n\tjump %%l2\n\tnop\n1:", ! 59: neg_opcode); ! 60: else ! 61: sprintf (buf, "cmp_br_delayed %s,%%0,%%1,%%l2\n\tnop", opcode); ! 62: return buf; ! 63: } ! 64: ! 65: /* Return the best assembler insn template ! 66: for moving operands[1] into operands[0] as a fullword. */ ! 67: ! 68: static char * ! 69: singlemove_string (operands) ! 70: rtx *operands; ! 71: { ! 72: if (GET_CODE (operands[0]) == MEM) ! 73: return "st_32 %r1,%0"; ! 74: if (GET_CODE (operands[1]) == MEM) ! 75: return "ld_32 %0,%1\n\tnop"; ! 76: if (GET_CODE (operands[1]) == REG) ! 77: return "add_nt %0,%1,$0"; ! 78: return "add_nt %0,r0,%1"; ! 79: } ! 80: ! 81: /* Output assembler code to perform a doubleword move insn ! 82: with operands OPERANDS. */ ! 83: ! 84: char * ! 85: output_move_double (operands) ! 86: rtx *operands; ! 87: { ! 88: enum { REGOP, OFFSOP, MEMOP, PUSHOP, POPOP, CNSTOP, RNDOP } optype0, optype1; ! 89: rtx latehalf[2]; ! 90: rtx addreg0 = 0, addreg1 = 0; ! 91: ! 92: /* First classify both operands. */ ! 93: ! 94: if (REG_P (operands[0])) ! 95: optype0 = REGOP; ! 96: else if (offsettable_memref_p (operands[0])) ! 97: optype0 = OFFSOP; ! 98: else if (GET_CODE (operands[0]) == MEM) ! 99: optype0 = MEMOP; ! 100: else ! 101: optype0 = RNDOP; ! 102: ! 103: if (REG_P (operands[1])) ! 104: optype1 = REGOP; ! 105: else if (CONSTANT_P (operands[1])) ! 106: optype1 = CNSTOP; ! 107: else if (offsettable_memref_p (operands[1])) ! 108: optype1 = OFFSOP; ! 109: else if (GET_CODE (operands[1]) == MEM) ! 110: optype1 = MEMOP; ! 111: else ! 112: optype1 = RNDOP; ! 113: ! 114: /* Check for the cases that the operand constraints are not ! 115: supposed to allow to happen. Abort if we get one, ! 116: because generating code for these cases is painful. */ ! 117: ! 118: if (optype0 == RNDOP || optype1 == RNDOP) ! 119: abort (); ! 120: ! 121: /* If an operand is an unoffsettable memory ref, find a register ! 122: we can increment temporarily to make it refer to the second word. */ ! 123: ! 124: if (optype0 == MEMOP) ! 125: addreg0 = find_addr_reg (XEXP (operands[0], 0)); ! 126: ! 127: if (optype1 == MEMOP) ! 128: addreg1 = find_addr_reg (XEXP (operands[1], 0)); ! 129: ! 130: /* Ok, we can do one word at a time. ! 131: Normally we do the low-numbered word first, ! 132: but if either operand is autodecrementing then we ! 133: do the high-numbered word first. ! 134: ! 135: In either case, set up in LATEHALF the operands to use ! 136: for the high-numbered word and in some cases alter the ! 137: operands in OPERANDS to be suitable for the low-numbered word. */ ! 138: ! 139: if (optype0 == REGOP) ! 140: latehalf[0] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1); ! 141: else if (optype0 == OFFSOP) ! 142: latehalf[0] = adj_offsettable_operand (operands[0], 4); ! 143: else ! 144: latehalf[0] = operands[0]; ! 145: ! 146: if (optype1 == REGOP) ! 147: latehalf[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1); ! 148: else if (optype1 == OFFSOP) ! 149: latehalf[1] = adj_offsettable_operand (operands[1], 4); ! 150: else if (optype1 == CNSTOP) ! 151: { ! 152: if (GET_CODE (operands[1]) == CONST_DOUBLE) ! 153: { ! 154: latehalf[1] = gen_rtx (CONST_INT, VOIDmode, ! 155: CONST_DOUBLE_HIGH (operands[1])); ! 156: operands[1] = gen_rtx (CONST_INT, VOIDmode, ! 157: CONST_DOUBLE_LOW (operands[1])); ! 158: } ! 159: else if (CONSTANT_P (operands[1])) ! 160: latehalf[1] = const0_rtx; ! 161: } ! 162: else ! 163: latehalf[1] = operands[1]; ! 164: ! 165: /* If the first move would clobber the source of the second one, ! 166: do them in the other order. This happens only for registers; ! 167: such overlap can't happen in memory unless the user explicitly ! 168: sets it up, and that is an undefined circumstance. */ ! 169: ! 170: if (optype0 == REGOP && optype1 == REGOP ! 171: && REGNO (operands[0]) == REGNO (latehalf[1])) ! 172: { ! 173: /* Make any unoffsettable addresses point at high-numbered word. */ ! 174: if (addreg0) ! 175: output_asm_insn ("add_nt %0,%0,$4", &addreg0); ! 176: if (addreg1) ! 177: output_asm_insn ("add_nt %0,%0,$4", &addreg1); ! 178: ! 179: /* Do that word. */ ! 180: output_asm_insn (singlemove_string (latehalf), latehalf); ! 181: ! 182: /* Undo the adds we just did. */ ! 183: if (addreg0) ! 184: output_asm_insn ("add_nt %0,%0,$-4", &addreg0); ! 185: if (addreg1) ! 186: output_asm_insn ("add_nt %0,%0,$-4", &addreg0); ! 187: ! 188: /* Do low-numbered word. */ ! 189: return singlemove_string (operands); ! 190: } ! 191: ! 192: /* Normal case: do the two words, low-numbered first. */ ! 193: ! 194: output_asm_insn (singlemove_string (operands), operands); ! 195: ! 196: /* Make any unoffsettable addresses point at high-numbered word. */ ! 197: if (addreg0) ! 198: output_asm_insn ("add_nt %0,%0,$4", &addreg0); ! 199: if (addreg1) ! 200: output_asm_insn ("add_nt %0,%0,$4", &addreg1); ! 201: ! 202: /* Do that word. */ ! 203: output_asm_insn (singlemove_string (latehalf), latehalf); ! 204: ! 205: /* Undo the adds we just did. */ ! 206: if (addreg0) ! 207: output_asm_insn ("add_nt %0,%0,$-4", &addreg0); ! 208: if (addreg1) ! 209: output_asm_insn ("add_nt %0,%0,$-4", &addreg1); ! 210: ! 211: return ""; ! 212: } ! 213: ! 214: static char * ! 215: output_fp_move_double (operands) ! 216: rtx *operands; ! 217: { ! 218: if (FP_REG_P (operands[0])) ! 219: { ! 220: if (FP_REG_P (operands[1])) ! 221: return "fmov %0,%1"; ! 222: if (GET_CODE (operands[1]) == REG) ! 223: { ! 224: rtx xoperands[2]; ! 225: int offset = - get_frame_size () - 8; ! 226: xoperands[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1); ! 227: xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset + 4); ! 228: output_asm_insn ("st_32 %1,r25,%0", xoperands); ! 229: xoperands[1] = operands[1]; ! 230: xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset); ! 231: output_asm_insn ("st_32 %1,r25,%0", xoperands); ! 232: xoperands[1] = operands[0]; ! 233: output_asm_insn ("ld_dbl %1,r25,%0\n\tnop", xoperands); ! 234: return ""; ! 235: } ! 236: return "ld_dbl %0,%1\n\tnop"; ! 237: } ! 238: else if (FP_REG_P (operands[1])) ! 239: { ! 240: if (GET_CODE (operands[0]) == REG) ! 241: { ! 242: rtx xoperands[2]; ! 243: int offset = - get_frame_size () - 8; ! 244: xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset); ! 245: xoperands[1] = operands[1]; ! 246: output_asm_insn ("st_dbl %1,r25,%0", xoperands); ! 247: xoperands[1] = operands[0]; ! 248: output_asm_insn ("ld_32 %1,r25,%0\n\tnop", xoperands); ! 249: xoperands[1] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1); ! 250: xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset + 4); ! 251: output_asm_insn ("ld_32 %1,r25,%0\n\tnop", xoperands); ! 252: return ""; ! 253: } ! 254: return "st_dbl %1,%0"; ! 255: } ! 256: } ! 257: ! 258: /* Return a REG that occurs in ADDR with coefficient 1. ! 259: ADDR can be effectively incremented by incrementing REG. */ ! 260: ! 261: static rtx ! 262: find_addr_reg (addr) ! 263: rtx addr; ! 264: { ! 265: while (GET_CODE (addr) == PLUS) ! 266: { ! 267: if (GET_CODE (XEXP (addr, 0)) == REG) ! 268: addr = XEXP (addr, 0); ! 269: else if (GET_CODE (XEXP (addr, 1)) == REG) ! 270: addr = XEXP (addr, 1); ! 271: else if (CONSTANT_P (XEXP (addr, 0))) ! 272: addr = XEXP (addr, 1); ! 273: else if (CONSTANT_P (XEXP (addr, 1))) ! 274: addr = XEXP (addr, 0); ! 275: else ! 276: abort (); ! 277: } ! 278: if (GET_CODE (addr) == REG) ! 279: return addr; ! 280: abort (); ! 281: } ! 282: ! 283: /* Generate code to add a large integer constant to register, reg, storing ! 284: * the result in a register, target. Offset must be 27-bit signed quantity */ ! 285: ! 286: static char * ! 287: output_add_large_offset (target, reg, offset) ! 288: rtx target, reg; ! 289: int offset; ! 290: { ! 291: rtx operands[3]; ! 292: int high, n, i; ! 293: operands[0] = target, operands[1] = reg; ! 294: ! 295: for (high = offset, n = 0; ! 296: (unsigned) (high + 0x2000) >= 0x4000; ! 297: high >>= 1, n += 1) ! 298: ; ! 299: operands[2] = gen_rtx (CONST_INT, VOIDmode, high); ! 300: output_asm_insn ("add_nt r2,r0,%2", operands); ! 301: i = n; ! 302: while (i >= 3) ! 303: output_asm_insn ("sll r2,r2,$3", operands), i -= 3; ! 304: if (i == 2) ! 305: output_asm_insn ("sll r2,r2,$2", operands); ! 306: else if (i == 1) ! 307: output_asm_insn ("sll r2,r2,$1", operands); ! 308: output_asm_insn ("add_nt %0,r2,%1", operands); ! 309: if (offset - (high << n) != 0) ! 310: { ! 311: operands[2] = gen_rtx (CONST_INT, VOIDmode, offset - (high << n)); ! 312: output_asm_insn ("add_nt %0,%0,%2", operands); ! 313: } ! 314: return ""; ! 315: } ! 316: ! 317: /* Additional TESTFN for matching. Like immediate_operand, but matches big ! 318: * constants */ ! 319: ! 320: int ! 321: big_immediate_operand (op, mode) ! 322: rtx op; ! 323: enum machine_mode mode; ! 324: { ! 325: return (GET_CODE (op) == CONST_INT); ! 326: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.