|
|
1.1 ! root 1: /* Definitions of target machine for GNU compiler. ! 2: Copyright (C) 1990, 1993 Free Software Foundation, Inc. ! 3: ! 4: Written by Robert Andersson, International Systems, Oslo, Norway. ! 5: Send bug reports, questions and improvements to [email protected]. ! 6: ! 7: For NCR Tower 32/4x0 and 32/6x0 running System V Release 3. ! 8: This file outputs assembler source suitable for the native Tower as ! 9: and with sdb debugging symbols. See tower.h for more comments. ! 10: ! 11: This file was based on m68k.h, hp320.h and 3b1.h ! 12: as of the 1.37.1 version. ! 13: ! 14: ! 15: This file is part of GNU CC. ! 16: ! 17: GNU CC is free software; you can redistribute it and/or modify ! 18: it under the terms of the GNU General Public License as published by ! 19: the Free Software Foundation; either version 2, or (at your option) ! 20: any later version. ! 21: ! 22: GNU CC is distributed in the hope that it will be useful, ! 23: but WITHOUT ANY WARRANTY; without even the implied warranty of ! 24: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ! 25: GNU General Public License for more details. ! 26: ! 27: You should have received a copy of the GNU General Public License ! 28: along with GNU CC; see the file COPYING. If not, write to ! 29: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ ! 30: ! 31: ! 32: #include "m68k/tower.h" ! 33: #undef SELECT_RTX_SECTION ! 34: ! 35: /* Use default settings for system V.3. */ ! 36: ! 37: #include "svr3.h" ! 38: ! 39: /* Names to predefine in the preprocessor for this target machine. */ ! 40: ! 41: #define CPP_PREDEFINES "-Dunix -Dtower32 -Dtower32_600 -D__motorola__ -Asystem(unix) -Asystem(svr3) -Acpu(m68k) -Amachine(m68k)" ! 42: ! 43: /* Define __HAVE_68881 in preprocessor only if -m68881 is specified. ! 44: This will control the use of inline 68881 insns in certain macros. ! 45: Also, define special define used to identify the Tower assembler. */ ! 46: ! 47: #define CPP_SPEC "-D__TOWER_ASM__ %{m68881:-D__HAVE_68881__}" ! 48: ! 49: /* We don't want local labels to start with period. ! 50: See ASM_OUTPUT_INTERNAL_LABEL. */ ! 51: #undef LOCAL_LABEL_PREFIX ! 52: #define LOCAL_LABEL_PREFIX "" ! 53: ! 54: /* These four macros control how m68k.md is expanded. */ ! 55: ! 56: #define MOTOROLA /* Use Motorola syntax rather than "MIT" */ ! 57: #define SGS /* Uses SGS assembler */ ! 58: #define SGS_CMP_ORDER /* Takes cmp operands in reverse order */ ! 59: #define SGS_NO_LI /* Suppress jump table label usage */ ! 60: ! 61: /* Turn on SDB debugging info. */ ! 62: ! 63: #define SDB_DEBUGGING_INFO ! 64: ! 65: /* This is only useful if gdb is changed, but doesn't harm anyway. */ ! 66: ! 67: #define ASM_IDENTIFY_GCC(FILE) \ ! 68: fprintf (FILE, "gcc2_compiled%%:\n") ! 69: ! 70: /* All the ASM_OUTPUT macros need to conform to the Tower as syntax. */ ! 71: ! 72: #define ASM_OUTPUT_SOURCE_FILENAME(FILE, FILENAME) \ ! 73: do { \ ! 74: fprintf (FILE, "\tfile\t"); \ ! 75: output_quoted_string (FILE, FILENAME); \ ! 76: fprintf (FILE, "\n"); \ ! 77: fprintf (FILE, "section ~init,\"x\"\n"); \ ! 78: fprintf (FILE, "section ~fini,\"x\"\n"); \ ! 79: fprintf (FILE, "section ~rodata,\"x\"\n"); \ ! 80: fprintf (FILE, "text\n"); \ ! 81: } while (0) ! 82: ! 83: #define ASM_OUTPUT_SOURCE_LINE(FILE, LINENO) \ ! 84: fprintf (FILE, "\tln\t%d\n", \ ! 85: (sdb_begin_function_line \ ! 86: ? last_linenum - sdb_begin_function_line : 1)) ! 87: ! 88: #undef ASM_OUTPUT_IDENT ! 89: #define ASM_OUTPUT_IDENT(FILE, NAME) \ ! 90: fprintf (FILE, "\tident\t\"%s\" \n", NAME) ! 91: ! 92: #define ASM_OUTPUT_ASCII(FILE,PTR,LEN) \ ! 93: { register int sp = 0, lp = 0; \ ! 94: fprintf ((FILE), "\tbyte\t"); \ ! 95: loop: \ ! 96: if ((PTR)[sp] > ' ' && ! ((PTR)[sp] & 0x80) && (PTR)[sp] != '\\') \ ! 97: { lp += 3; \ ! 98: fprintf ((FILE), "'%c", (PTR)[sp]); } \ ! 99: else \ ! 100: { lp += 5; \ ! 101: fprintf ((FILE), "0x%x", (PTR)[sp]); } \ ! 102: if (++sp < (LEN)) \ ! 103: { if (lp > 60) \ ! 104: { lp = 0; \ ! 105: fprintf ((FILE), "\n\tbyte\t"); } \ ! 106: else \ ! 107: putc (',', (FILE)); \ ! 108: goto loop; } \ ! 109: putc ('\n', (FILE)); } ! 110: ! 111: /* Translate Motorola opcodes such as `jbeq' ! 112: into SGS/Tower opcodes such as `beq.w'. ! 113: Change `move' to `mov'. ! 114: Change `cmpm' to `cmp'. ! 115: Change `divsl' to `tdivs'. ! 116: Change `divul' to `tdivu'. ! 117: Change `ftst' to `ftest'. ! 118: Change `fmove' to `fmov'. */ ! 119: ! 120: #define ASM_OUTPUT_OPCODE(FILE, PTR) \ ! 121: { if ((PTR)[0] == 'j' && (PTR)[1] == 'b') \ ! 122: { ++(PTR); \ ! 123: while (*(PTR) != ' ') \ ! 124: { putc (*(PTR), (FILE)); ++(PTR); } \ ! 125: fprintf ((FILE), ".w"); } \ ! 126: else if ((PTR)[0] == 'm' && (PTR)[1] == 'o' \ ! 127: && (PTR)[2] == 'v' && (PTR)[3] == 'e') \ ! 128: { fprintf ((FILE), "mov"); (PTR) += 4; } \ ! 129: else if ((PTR)[0] == 'c' && (PTR)[1] == 'm' \ ! 130: && (PTR)[2] == 'p' && (PTR)[3] == 'm') \ ! 131: { fprintf ((FILE), "cmp"); (PTR) += 4; } \ ! 132: else if ((PTR)[0] == 'd' && (PTR)[1] == 'i' \ ! 133: && (PTR)[2] == 'v' && (PTR)[3] == 's' \ ! 134: && (PTR)[4] == 'l') \ ! 135: { fprintf ((FILE), "tdivs"); (PTR) += 5; } \ ! 136: else if ((PTR)[0] == 'd' && (PTR)[1] == 'i' \ ! 137: && (PTR)[2] == 'v' && (PTR)[3] == 'u' \ ! 138: && (PTR)[4] == 'l') \ ! 139: { fprintf ((FILE), "tdivu"); (PTR) += 5; } \ ! 140: else if ((PTR)[0] == 'f' && (PTR)[1] == 't' \ ! 141: && (PTR)[2] == 's' && (PTR)[3] == 't') \ ! 142: { fprintf ((FILE), "ftest"); (PTR) += 4; } \ ! 143: else if ((PTR)[0] == 'f' && (PTR)[1] == 'm' \ ! 144: && (PTR)[2] == 'o' && (PTR)[3] == 'v' \ ! 145: && (PTR)[4] == 'e') \ ! 146: { fprintf ((FILE), "fmov"); (PTR) += 5; } \ ! 147: } ! 148: ! 149: ! 150: ! 151: /* Override parts of m68k.h to fit the Tower assembler. ! 152: This section needs to track changes done to m68k.h in the future. */ ! 153: ! 154: #undef TARGET_VERSION ! 155: #define TARGET_VERSION fprintf (stderr, " (68k, Motorola/SGS/Tower32 syntax)"); ! 156: ! 157: #undef FUNCTION_BLOCK_PROFILER ! 158: #define FUNCTION_BLOCK_PROFILER(FILE, LABELNO) \ ! 159: do { \ ! 160: char label1[20], label2[20]; \ ! 161: ASM_GENERATE_INTERNAL_LABEL (label1, "LPBX", 0); \ ! 162: ASM_GENERATE_INTERNAL_LABEL (label2, "LPI", LABELNO); \ ! 163: fprintf (FILE, "\ttst.l %s\n\tbne %s\n\tpea %s\n\tjsr __bb_init_func\n\taddq.l &4,%%sp\n", \ ! 164: label1, label2, label1); \ ! 165: ASM_OUTPUT_INTERNAL_LABEL (FILE, "LPI", LABELNO); \ ! 166: putc ('\n', FILE); \ ! 167: } while (0) ! 168: ! 169: #undef BLOCK_PROFILER ! 170: #define BLOCK_PROFILER(FILE, BLOCKNO) \ ! 171: do { \ ! 172: char label[20]; \ ! 173: ASM_GENERATE_INTERNAL_LABEL (label, "LPBX", 2); \ ! 174: fprintf (FILE, "\taddq.l &1,%s+%d\n", label, 4 * BLOCKNO); \ ! 175: } while (0) ! 176: ! 177: #undef FUNCTION_PROFILER ! 178: #define FUNCTION_PROFILER(FILE, LABEL_NO) \ ! 179: fprintf (FILE, "\tmov.l &LP%%%d,%%a0\n\tjsr mcount%%\n", (LABEL_NO)) ! 180: ! 181: /* The prologue is identical to the one in m68k.h except that the ! 182: assembler syntax is different. */ ! 183: ! 184: #undef FUNCTION_PROLOGUE ! 185: #define FUNCTION_PROLOGUE(FILE, SIZE) \ ! 186: { register int regno; \ ! 187: register int mask = 0; \ ! 188: extern char call_used_regs[]; \ ! 189: int fsize = ((SIZE) + 3) & -4; \ ! 190: if (frame_pointer_needed) \ ! 191: { if (TARGET_68020 || fsize < 0x8000) \ ! 192: fprintf (FILE, "\tlink %%a6,&%d\n", -fsize); \ ! 193: else \ ! 194: fprintf (FILE, "\tlink %%a6,&0\n\tsub.l &%d,%%sp\n", fsize); } \ ! 195: for (regno = 16; regno < 24; regno++) \ ! 196: if (regs_ever_live[regno] && ! call_used_regs[regno]) \ ! 197: mask |= 1 << (regno - 16); \ ! 198: if ((mask & 0xff) != 0) \ ! 199: fprintf (FILE, "\tfmovm &0x%x,-(%%sp)\n", mask & 0xff); \ ! 200: mask = 0; \ ! 201: for (regno = 0; regno < 16; regno++) \ ! 202: if (regs_ever_live[regno] && ! call_used_regs[regno]) \ ! 203: mask |= 1 << (15 - regno); \ ! 204: if (frame_pointer_needed) \ ! 205: mask &= ~ (1 << (15-FRAME_POINTER_REGNUM)); \ ! 206: if (exact_log2 (mask) >= 0) \ ! 207: fprintf (FILE, "\tmov.l %s,-(%%sp)\n", reg_names[15 - exact_log2 (mask)]); \ ! 208: else if (mask) fprintf (FILE, "\tmovm.l &0x%x,-(%%sp)\n", mask); } ! 209: ! 210: /* The epilogue is identical to the one in m68k.h except that: ! 211: a) The assembler syntax is different. ! 212: b) Pointers are returned both in %d0 and %a0. ! 213: c) FUNCTION_EXTRA_EPILOGUE is not needed. */ ! 214: ! 215: #undef FUNCTION_EPILOGUE ! 216: #define FUNCTION_EPILOGUE(FILE, SIZE) \ ! 217: { register int regno; \ ! 218: register int mask, fmask; \ ! 219: register int nregs; \ ! 220: int offset, foffset; \ ! 221: extern char call_used_regs[]; \ ! 222: int fsize = ((SIZE) + 3) & -4; \ ! 223: int big = 0; \ ! 224: nregs = 0; fmask = 0; \ ! 225: for (regno = 16; regno < 24; regno++) \ ! 226: if (regs_ever_live[regno] && ! call_used_regs[regno]) \ ! 227: { nregs++; fmask |= 1 << (23 - regno); } \ ! 228: foffset = nregs * 12; \ ! 229: nregs = 0; mask = 0; \ ! 230: if (frame_pointer_needed) regs_ever_live[FRAME_POINTER_REGNUM] = 0; \ ! 231: for (regno = 0; regno < 16; regno++) \ ! 232: if (regs_ever_live[regno] && ! call_used_regs[regno]) \ ! 233: { nregs++; mask |= 1 << regno; } \ ! 234: offset = foffset + nregs * 4; \ ! 235: if (offset + fsize >= 0x8000 \ ! 236: && frame_pointer_needed \ ! 237: && (mask || fmask)) \ ! 238: { fprintf (FILE, "\tmov.l &%d,%%a0\n", -fsize); \ ! 239: fsize = 0, big = 1; } \ ! 240: if (exact_log2 (mask) >= 0) { \ ! 241: if (big) \ ! 242: fprintf (FILE, "\tmov.l -%d(%%a6,%%a0.l),%s\n", \ ! 243: offset + fsize, reg_names[exact_log2 (mask)]); \ ! 244: else if (! frame_pointer_needed) \ ! 245: fprintf (FILE, "\tmov.l (%%sp)+,%s\n", \ ! 246: reg_names[exact_log2 (mask)]); \ ! 247: else \ ! 248: fprintf (FILE, "\tmov.l -%d(%%a6),%s\n", \ ! 249: offset + fsize, reg_names[exact_log2 (mask)]); } \ ! 250: else if (mask) { \ ! 251: if (big) \ ! 252: fprintf (FILE, "\tmovm.l -%d(%%a6,%%a0.l),&0x%x\n", \ ! 253: offset + fsize, mask); \ ! 254: else if (! frame_pointer_needed) \ ! 255: fprintf (FILE, "\tmovm.l (%%sp)+,&0x%x\n", mask); \ ! 256: else \ ! 257: fprintf (FILE, "\tmovm.l -%d(%%a6),&0x%x\n", \ ! 258: offset + fsize, mask); } \ ! 259: if (fmask) { \ ! 260: if (big) \ ! 261: fprintf (FILE, "\tfmovm -%d(%%a6,%%a0.l),&0x%x\n", \ ! 262: foffset + fsize, fmask); \ ! 263: else if (! frame_pointer_needed) \ ! 264: fprintf (FILE, "\tfmovm (%%sp)+,&0x%x\n", fmask); \ ! 265: else \ ! 266: fprintf (FILE, "\tfmovm -%d(%%a6),&0x%x\n", \ ! 267: foffset + fsize, fmask); } \ ! 268: if (current_function_returns_pointer) \ ! 269: fprintf (FILE, "\tmov.l %%d0,%%a0\n"); \ ! 270: if (frame_pointer_needed) \ ! 271: fprintf (FILE, "\tunlk %%a6\n"); \ ! 272: if (current_function_pops_args) \ ! 273: fprintf (FILE, "\trtd &%d\n", current_function_pops_args); \ ! 274: else fprintf (FILE, "\trts\n"); } ! 275: ! 276: /* This is how to output an insn to push a register on the stack. ! 277: It need not be very fast code. */ ! 278: ! 279: #undef ASM_OUTPUT_REG_PUSH ! 280: #define ASM_OUTPUT_REG_PUSH(FILE,REGNO) \ ! 281: fprintf (FILE, "\tmov.l %s,-(%%sp)\n", reg_names[REGNO]) ! 282: ! 283: /* This is how to output an insn to pop a register from the stack. ! 284: It need not be very fast code. */ ! 285: ! 286: #undef ASM_OUTPUT_REG_POP ! 287: #define ASM_OUTPUT_REG_POP(FILE,REGNO) \ ! 288: fprintf (FILE, "\tmov.l (%%sp)+,%s\n", reg_names[REGNO]) ! 289: ! 290: #undef ASM_FILE_START ! 291: #define ASM_FILE_START(FILE) \ ! 292: ( fprintf (FILE, "#NO_APP\n"), \ ! 293: output_file_directive ((FILE), main_input_filename)) ! 294: ! 295: #undef TEXT_SECTION_ASM_OP ! 296: #define TEXT_SECTION_ASM_OP "text" ! 297: ! 298: #undef DATA_SECTION_ASM_OP ! 299: #define DATA_SECTION_ASM_OP "data" ! 300: ! 301: /* This says how to output an assembler line to define a global common symbol. ! 302: We use SIZE rather than ROUNDED, as this is what the native cc does. */ ! 303: ! 304: #undef ASM_OUTPUT_COMMON ! 305: #define ASM_OUTPUT_COMMON(FILE, NAME, SIZE, ROUNDED) \ ! 306: ( fputs ("\tcomm ", (FILE)), \ ! 307: assemble_name ((FILE), (NAME)), \ ! 308: fprintf ((FILE), ",%d\n", ((SIZE) == 0) ? (ROUNDED) : (SIZE))) ! 309: ! 310: /* This says how to output an assembler line to define a local common symbol. ! 311: We use SIZE rather than ROUNDED, as this is what the native cc does. */ ! 312: ! 313: #undef ASM_OUTPUT_LOCAL ! 314: #define ASM_OUTPUT_LOCAL(FILE, NAME, SIZE, ROUNDED) \ ! 315: ( fputs ("\tlcomm ", (FILE)), \ ! 316: assemble_name ((FILE), (NAME)), \ ! 317: fprintf ((FILE), ",%d\n", ((SIZE) == 0) ? (ROUNDED) : (SIZE))) ! 318: ! 319: /* Store in OUTPUT a string (made with alloca) containing ! 320: an assembler-name for a local static variable named NAME. ! 321: LABELNO is an integer which is different for each call. */ ! 322: ! 323: #undef ASM_FORMAT_PRIVATE_NAME ! 324: #define ASM_FORMAT_PRIVATE_NAME(OUTPUT, NAME, LABELNO) \ ! 325: ( (OUTPUT) = (char *) alloca (strlen ((NAME)) + 11), \ ! 326: sprintf ((OUTPUT), "%s%%%%%d", (NAME), (LABELNO))) ! 327: ! 328: /* This is the command to make the user-level label named NAME ! 329: defined for reference from other files. */ ! 330: ! 331: #undef GLOBAL_ASM_OP ! 332: #define GLOBAL_ASM_OP "global" ! 333: ! 334: #undef ASM_GENERATE_INTERNAL_LABEL ! 335: #define ASM_GENERATE_INTERNAL_LABEL(LABEL, PREFIX, NUM) \ ! 336: sprintf ((LABEL), "%s%%%d", (PREFIX), (NUM)) ! 337: ! 338: #undef ASM_OUTPUT_INTERNAL_LABEL ! 339: #define ASM_OUTPUT_INTERNAL_LABEL(FILE,PREFIX,NUM) \ ! 340: fprintf ((FILE), "%s%%%d:\n", (PREFIX), (NUM)) ! 341: ! 342: #undef ASM_OUTPUT_CASE_LABEL ! 343: #define ASM_OUTPUT_CASE_LABEL(FILE,PREFIX,NUM,TABLE) \ ! 344: fprintf (FILE, "\tswbeg &%d\n%s%%%d:\n", \ ! 345: XVECLEN (PATTERN (TABLE), 1), (PREFIX), (NUM)); \ ! 346: ! 347: #undef ASM_OUTPUT_DOUBLE ! 348: #define ASM_OUTPUT_DOUBLE(FILE,VALUE) \ ! 349: do { long l[2]; \ ! 350: REAL_VALUE_TO_TARGET_DOUBLE (VALUE, l); \ ! 351: fprintf (FILE, "\tlong 0x%x,0x%x\n", l[0], l[1]); \ ! 352: } while (0) ! 353: ! 354: #undef ASM_OUTPUT_LONG_DOUBLE ! 355: #define ASM_OUTPUT_LONG_DOUBLE(FILE,VALUE) \ ! 356: do { long l[3]; \ ! 357: REAL_VALUE_TO_TARGET_LONG_DOUBLE (VALUE, l); \ ! 358: fprintf (FILE, "\tlong 0x%x,0x%x,0x%x\n", l[0], l[1], l[2]); \ ! 359: } while (0) ! 360: ! 361: #undef ASM_OUTPUT_FLOAT ! 362: #define ASM_OUTPUT_FLOAT(FILE,VALUE) \ ! 363: do { long l; \ ! 364: REAL_VALUE_TO_TARGET_SINGLE (VALUE, l); \ ! 365: fprintf ((FILE), "\tlong 0x%x\n", l); \ ! 366: } while (0) ! 367: ! 368: /* This is how to output an assembler line defining an `int' constant. */ ! 369: ! 370: #undef ASM_OUTPUT_INT ! 371: #define ASM_OUTPUT_INT(FILE,VALUE) \ ! 372: ( fprintf (FILE, "\tlong "), \ ! 373: output_addr_const (FILE, (VALUE)), \ ! 374: fprintf (FILE, "\n")) ! 375: ! 376: /* Likewise for `char' and `short' constants. */ ! 377: ! 378: #undef ASM_OUTPUT_SHORT ! 379: #define ASM_OUTPUT_SHORT(FILE,VALUE) \ ! 380: ( fprintf (FILE, "\tshort "), \ ! 381: output_addr_const (FILE, (VALUE)), \ ! 382: fprintf (FILE, "\n")) ! 383: ! 384: #undef ASM_OUTPUT_CHAR ! 385: #define ASM_OUTPUT_CHAR(FILE,VALUE) \ ! 386: ( fprintf (FILE, "\tbyte "), \ ! 387: output_addr_const (FILE, (VALUE)), \ ! 388: fprintf (FILE, "\n")) ! 389: ! 390: /* This is how to output an assembler line for a numeric constant byte. */ ! 391: ! 392: #undef ASM_OUTPUT_BYTE ! 393: #define ASM_OUTPUT_BYTE(FILE,VALUE) \ ! 394: fprintf (FILE, "\tbyte 0x%x\n", (VALUE)) ! 395: ! 396: #undef ASM_OUTPUT_ADDR_VEC_ELT ! 397: #define ASM_OUTPUT_ADDR_VEC_ELT(FILE, VALUE) \ ! 398: fprintf (FILE, "\tlong L%%%d\n", (VALUE)) ! 399: ! 400: #undef ASM_OUTPUT_ADDR_DIFF_ELT ! 401: #define ASM_OUTPUT_ADDR_DIFF_ELT(FILE, VALUE, REL) \ ! 402: fprintf (FILE, "\tshort L%%%d-L%%%d\n", (VALUE), (REL)) ! 403: ! 404: #undef ASM_OUTPUT_ALIGN ! 405: #define ASM_OUTPUT_ALIGN(FILE,LOG) \ ! 406: if ((LOG) == 1) \ ! 407: fprintf (FILE, "\teven\n"); \ ! 408: else if ((LOG) != 0) \ ! 409: abort (); ! 410: ! 411: #undef ASM_OUTPUT_SKIP ! 412: #define ASM_OUTPUT_SKIP(FILE,SIZE) \ ! 413: fprintf (FILE, "\tspace %d\n", (SIZE)) ! 414: ! 415: #undef PRINT_OPERAND ! 416: #define PRINT_OPERAND(FILE, X, CODE) \ ! 417: { if (CODE == '.') fprintf (FILE, "."); \ ! 418: else if (CODE == '#') fprintf (FILE, "&"); \ ! 419: else if (CODE == '-') fprintf (FILE, "-(%%sp)"); \ ! 420: else if (CODE == '+') fprintf (FILE, "(%%sp)+"); \ ! 421: else if (CODE == '@') fprintf (FILE, "(%%sp)"); \ ! 422: else if (CODE == '!') fprintf (FILE, "%%fpcr"); \ ! 423: else if (CODE == '/') \ ! 424: fprintf (FILE, "%%"); \ ! 425: else if (CODE == '$') { if (TARGET_68040_ONLY) fprintf (FILE, "s"); } \ ! 426: else if (CODE == '&') { if (TARGET_68040_ONLY) fprintf (FILE, "d"); } \ ! 427: else if (GET_CODE (X) == REG) \ ! 428: fprintf (FILE, "%s", reg_names[REGNO (X)]); \ ! 429: else if (GET_CODE (X) == MEM) \ ! 430: output_address (XEXP (X, 0)); \ ! 431: else if (GET_CODE (X) == CONST_DOUBLE && GET_MODE (X) == SFmode) \ ! 432: { REAL_VALUE_TYPE r; long l; \ ! 433: REAL_VALUE_FROM_CONST_DOUBLE (r, X); \ ! 434: REAL_VALUE_TO_TARGET_SINGLE (r, l); \ ! 435: fprintf (FILE, "&0x%x", l); } \ ! 436: else if (GET_CODE (X) == CONST_DOUBLE && GET_MODE (X) == DFmode) \ ! 437: { REAL_VALUE_TYPE r; int i[2]; \ ! 438: REAL_VALUE_FROM_CONST_DOUBLE (r, X); \ ! 439: REAL_VALUE_TO_TARGET_DOUBLE (r, i); \ ! 440: fprintf (FILE, "&0x%x%08x", i[0], i[1]); } \ ! 441: else if (GET_CODE (X) == CONST_DOUBLE && GET_MODE (X) == XFmode) \ ! 442: { REAL_VALUE_TYPE r; \ ! 443: REAL_VALUE_FROM_CONST_DOUBLE (r, X); \ ! 444: ASM_OUTPUT_LONG_DOUBLE_OPERAND (FILE, r); } \ ! 445: else { putc ('&', FILE); output_addr_const (FILE, X); }} ! 446: ! 447: /* Note that this contains a kludge that knows that the only reason ! 448: we have an address (plus (label_ref...) (reg...)) ! 449: is in the insn before a tablejump, and we know that the table is ! 450: exactly 10 bytes away. */ ! 451: ! 452: #undef PRINT_OPERAND_ADDRESS ! 453: #define PRINT_OPERAND_ADDRESS(FILE, ADDR) \ ! 454: { register rtx reg1, reg2, breg, ireg; \ ! 455: register rtx addr = ADDR; \ ! 456: rtx offset; \ ! 457: switch (GET_CODE (addr)) \ ! 458: { \ ! 459: case REG: \ ! 460: fprintf (FILE, "(%s)", reg_names[REGNO (addr)]); \ ! 461: break; \ ! 462: case PRE_DEC: \ ! 463: fprintf (FILE, "-(%s)", reg_names[REGNO (XEXP (addr, 0))]); \ ! 464: break; \ ! 465: case POST_INC: \ ! 466: fprintf (FILE, "(%s)+", reg_names[REGNO (XEXP (addr, 0))]); \ ! 467: break; \ ! 468: case PLUS: \ ! 469: reg1 = 0; reg2 = 0; \ ! 470: ireg = 0; breg = 0; \ ! 471: offset = 0; \ ! 472: if (CONSTANT_ADDRESS_P (XEXP (addr, 0))) \ ! 473: { \ ! 474: offset = XEXP (addr, 0); \ ! 475: addr = XEXP (addr, 1); \ ! 476: } \ ! 477: else if (CONSTANT_ADDRESS_P (XEXP (addr, 1))) \ ! 478: { \ ! 479: offset = XEXP (addr, 1); \ ! 480: addr = XEXP (addr, 0); \ ! 481: } \ ! 482: if (GET_CODE (addr) != PLUS) ; \ ! 483: else if (GET_CODE (XEXP (addr, 0)) == SIGN_EXTEND) \ ! 484: { \ ! 485: reg1 = XEXP (addr, 0); \ ! 486: addr = XEXP (addr, 1); \ ! 487: } \ ! 488: else if (GET_CODE (XEXP (addr, 1)) == SIGN_EXTEND) \ ! 489: { \ ! 490: reg1 = XEXP (addr, 1); \ ! 491: addr = XEXP (addr, 0); \ ! 492: } \ ! 493: else if (GET_CODE (XEXP (addr, 0)) == MULT) \ ! 494: { \ ! 495: reg1 = XEXP (addr, 0); \ ! 496: addr = XEXP (addr, 1); \ ! 497: } \ ! 498: else if (GET_CODE (XEXP (addr, 1)) == MULT) \ ! 499: { \ ! 500: reg1 = XEXP (addr, 1); \ ! 501: addr = XEXP (addr, 0); \ ! 502: } \ ! 503: else if (GET_CODE (XEXP (addr, 0)) == REG) \ ! 504: { \ ! 505: reg1 = XEXP (addr, 0); \ ! 506: addr = XEXP (addr, 1); \ ! 507: } \ ! 508: else if (GET_CODE (XEXP (addr, 1)) == REG) \ ! 509: { \ ! 510: reg1 = XEXP (addr, 1); \ ! 511: addr = XEXP (addr, 0); \ ! 512: } \ ! 513: if (GET_CODE (addr) == REG || GET_CODE (addr) == MULT \ ! 514: || GET_CODE (addr) == SIGN_EXTEND) \ ! 515: { if (reg1 == 0) reg1 = addr; else reg2 = addr; addr = 0; } \ ! 516: /* for OLD_INDEXING \ ! 517: else if (GET_CODE (addr) == PLUS) \ ! 518: { \ ! 519: if (GET_CODE (XEXP (addr, 0)) == REG) \ ! 520: { \ ! 521: reg2 = XEXP (addr, 0); \ ! 522: addr = XEXP (addr, 1); \ ! 523: } \ ! 524: else if (GET_CODE (XEXP (addr, 1)) == REG) \ ! 525: { \ ! 526: reg2 = XEXP (addr, 1); \ ! 527: addr = XEXP (addr, 0); \ ! 528: } \ ! 529: } \ ! 530: */ \ ! 531: if (offset != 0) { if (addr != 0) abort (); addr = offset; } \ ! 532: if ((reg1 && (GET_CODE (reg1) == SIGN_EXTEND \ ! 533: || GET_CODE (reg1) == MULT)) \ ! 534: || (reg2 != 0 && REGNO_OK_FOR_BASE_P (REGNO (reg2)))) \ ! 535: { breg = reg2; ireg = reg1; } \ ! 536: else if (reg1 != 0 && REGNO_OK_FOR_BASE_P (REGNO (reg1))) \ ! 537: { breg = reg1; ireg = reg2; } \ ! 538: if (ireg != 0 && breg == 0 && GET_CODE (addr) == LABEL_REF) \ ! 539: { int scale = 1; \ ! 540: if (GET_CODE (ireg) == MULT) \ ! 541: { scale = INTVAL (XEXP (ireg, 1)); \ ! 542: ireg = XEXP (ireg, 0); } \ ! 543: if (GET_CODE (ireg) == SIGN_EXTEND) \ ! 544: fprintf (FILE, "10(%%pc,%s.w", \ ! 545: reg_names[REGNO (XEXP (ireg, 0))]); \ ! 546: else \ ! 547: fprintf (FILE, "10(%%pc,%s.l", \ ! 548: reg_names[REGNO (ireg)]); \ ! 549: if (scale != 1) fprintf (FILE, "*%d", scale); \ ! 550: putc (')', FILE); \ ! 551: break; } \ ! 552: if (ireg != 0 || breg != 0) \ ! 553: { int scale = 1; \ ! 554: if (breg == 0) \ ! 555: abort (); \ ! 556: if (addr != 0) \ ! 557: output_addr_const (FILE, addr); \ ! 558: fprintf (FILE, "(%s", reg_names[REGNO (breg)]); \ ! 559: if (ireg != 0) \ ! 560: putc (',', FILE); \ ! 561: if (ireg != 0 && GET_CODE (ireg) == MULT) \ ! 562: { scale = INTVAL (XEXP (ireg, 1)); \ ! 563: ireg = XEXP (ireg, 0); } \ ! 564: if (ireg != 0 && GET_CODE (ireg) == SIGN_EXTEND) \ ! 565: fprintf (FILE, "%s.w", reg_names[REGNO (XEXP (ireg, 0))]); \ ! 566: else if (ireg != 0) \ ! 567: fprintf (FILE, "%s.l", reg_names[REGNO (ireg)]); \ ! 568: if (scale != 1) fprintf (FILE, "*%d", scale); \ ! 569: putc (')', FILE); \ ! 570: break; \ ! 571: } \ ! 572: else if (reg1 != 0 && GET_CODE (addr) == LABEL_REF) \ ! 573: { fprintf (FILE, "10(%%pc,%s.w)", \ ! 574: reg_names[REGNO (reg1)]); \ ! 575: break; } \ ! 576: default: \ ! 577: output_addr_const (FILE, addr); \ ! 578: }} ! 579: ! 580: ! 581: ! 582: /* Override usual definitions of SDB output macros. ! 583: These definitions differ only in the absence of the period ! 584: at the beginning of the name of the directive ! 585: and in the use of `~' as the symbol for the current location. */ ! 586: ! 587: #define PUT_SDB_SCL(a) fprintf(asm_out_file, "\tscl\t%d;", (a)) ! 588: #define PUT_SDB_INT_VAL(a) fprintf (asm_out_file, "\tval\t%d;", (a)) ! 589: #define PUT_SDB_VAL(a) \ ! 590: ( fputs ("\tval\t", asm_out_file), \ ! 591: output_addr_const (asm_out_file, (a)), \ ! 592: fputc (';', asm_out_file)) ! 593: ! 594: #define PUT_SDB_DEF(a) \ ! 595: do { fprintf (asm_out_file, "\tdef\t"); \ ! 596: ASM_OUTPUT_LABELREF (asm_out_file, a); \ ! 597: fprintf (asm_out_file, ";"); } while (0) ! 598: ! 599: #define PUT_SDB_PLAIN_DEF(a) fprintf(asm_out_file,"\tdef\t~%s;",a) ! 600: #define PUT_SDB_ENDEF fputs("\tendef\n", asm_out_file) ! 601: #define PUT_SDB_TYPE(a) fprintf(asm_out_file, "\ttype\t0%o;", a) ! 602: #define PUT_SDB_SIZE(a) fprintf(asm_out_file, "\tsize\t%d;", a) ! 603: #define PUT_SDB_START_DIM fprintf(asm_out_file, "\tdim\t") ! 604: #define PUT_SDB_NEXT_DIM(a) fprintf(asm_out_file, "%d,", a) ! 605: #define PUT_SDB_LAST_DIM(a) fprintf(asm_out_file, "%d;", a) ! 606: ! 607: #define PUT_SDB_TAG(a) \ ! 608: do { fprintf (asm_out_file, "\ttag\t"); \ ! 609: ASM_OUTPUT_LABELREF (asm_out_file, a); \ ! 610: fprintf (asm_out_file, ";"); } while (0) ! 611: ! 612: #define PUT_SDB_BLOCK_START(LINE) \ ! 613: fprintf (asm_out_file, \ ! 614: "\tdef\t~bb;\tval\t~;\tscl\t100;\tline\t%d;\tendef\n", \ ! 615: (LINE)) ! 616: ! 617: #define PUT_SDB_BLOCK_END(LINE) \ ! 618: fprintf (asm_out_file, \ ! 619: "\tdef\t~eb;\tval\t~;\tscl\t100;\tline\t%d;\tendef\n", \ ! 620: (LINE)) ! 621: ! 622: #define PUT_SDB_FUNCTION_START(LINE) \ ! 623: fprintf (asm_out_file, \ ! 624: "\tdef\t~bf;\tval\t~;\tscl\t101;\tline\t%d;\tendef\n", \ ! 625: (LINE)) ! 626: ! 627: #define PUT_SDB_FUNCTION_END(LINE) \ ! 628: fprintf (asm_out_file, \ ! 629: "\tdef\t~ef;\tval\t~;\tscl\t101;\tline\t%d;\tendef\n", \ ! 630: (LINE)) ! 631: ! 632: #define PUT_SDB_EPILOGUE_END(NAME) \ ! 633: fprintf (asm_out_file, \ ! 634: "\tdef\t%s;\tval\t~;\tscl\t-1;\tendef\n", \ ! 635: (NAME)) ! 636: ! 637: #define SDB_GENERATE_FAKE(BUFFER, NUMBER) \ ! 638: sprintf ((BUFFER), "~%dfake", (NUMBER)); ! 639: ! 640: #define NO_DOLLAR_IN_LABEL ! 641: #define NO_DOT_IN_LABEL ! 642: ! 643: /* The usual definitions don't work because neither $ nor . is allowed. */ ! 644: #define CONSTRUCTOR_NAME_FORMAT "_GLOBAL_%%I\%%%s" ! 645: ! 646: /* Define a few machine-specific details ! 647: of the implementation of constructors. ! 648: ! 649: The __CTORS_LIST__ goes in the .init section. Define CTOR_LIST_BEGIN ! 650: and CTOR_LIST_END to contribute to the .init section an instruction to ! 651: push a word containing 0 (or some equivalent of that). ! 652: ! 653: ASM_OUTPUT_CONSTRUCTOR should be defined ! 654: to push the address of the constructor. */ ! 655: ! 656: #define ASM_LONG "\tlong" ! 657: #undef INIT_SECTION_ASM_OP ! 658: #define INIT_SECTION_ASM_OP "section\t~init" ! 659: #undef FINI_SECTION_ASM_OP ! 660: #define FINI_SECTION_ASM_OP "section\t~fini" ! 661: #undef CONST_SECTION_ASM_OP ! 662: #define CONST_SECTION_ASM_OP "section\t~rodata" ! 663: ! 664: #define CTOR_LIST_BEGIN \ ! 665: asm (INIT_SECTION_ASM_OP); \ ! 666: asm ("clr.l -(%sp)") ! 667: #define CTOR_LIST_END CTOR_LIST_BEGIN ! 668: ! 669: #define BSS_SECTION_ASM_OP "section\t~bss" ! 670: ! 671: #define ASM_OUTPUT_CONSTRUCTOR(FILE,NAME) \ ! 672: do { \ ! 673: init_section (); \ ! 674: fprintf (FILE, "\tmov.l &"); \ ! 675: assemble_name (FILE, NAME); \ ! 676: fprintf (FILE, ",-(%%sp)\n"); \ ! 677: } while (0) ! 678: ! 679: /* We do not want leading underscores. */ ! 680: ! 681: #undef ASM_OUTPUT_LABELREF ! 682: #define ASM_OUTPUT_LABELREF(FILE,NAME) \ ! 683: fprintf (FILE, "%s", NAME)
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.