elxsi.c

来自「gcc3.2.1源代码」· C语言 代码 · 共 276 行

C
276
字号
/* Subroutines for insn-output.c for GNU compiler.  Elxsi version.   Copyright (C) 1987, 1992, 1998, 1999, 2000 Free Software Foundation, Inc   Contributrd by Mike Stump <mrs@cygnus.com> in 1988 and is the first   64 bit port of GNU CC.   Based upon the VAX port.This file is part of GNU CC.GNU CC is free software; you can redistribute it and/or modifyit under the terms of the GNU General Public License as published bythe Free Software Foundation; either version 2, or (at your option)any later version.GNU CC is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY; without even the implied warranty ofMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See theGNU General Public License for more details.You should have received a copy of the GNU General Public Licensealong with GNU CC; see the file COPYING.  If not, write tothe Free Software Foundation, 59 Temple Place - Suite 330,Boston, MA 02111-1307, USA.  */#include "config.h"#include "system.h"#include "rtl.h"#include "function.h"#include "output.h"#include "tree.h"#include "expr.h"#include "regs.h"#include "flags.h"#include "hard-reg-set.h"#include "tm_p.h"#include "target.h"#include "target-def.h"extern const char *reg_names[];rtx cmp_op0=0, cmp_op1=0;/* table of relations for compares and branches */static const char *const cmp_tab[] = {    "gt", "gt", "eq", "eq", "ge", "ge", "lt", "lt", "ne", "ne",    "le", "le" };static bool elxsi_assemble_integer PARAMS ((rtx, unsigned int, int));static void elxsi_output_function_prologue PARAMS ((FILE *, HOST_WIDE_INT));static void elxsi_output_function_epilogue PARAMS ((FILE *, HOST_WIDE_INT));/* Initialize the GCC target structure.  */#undef TARGET_ASM_BYTE_OP#define TARGET_ASM_BYTE_OP NULL#undef TARGET_ASM_ALIGNED_HI_OP#define TARGET_ASM_ALIGNED_HI_OP NULL#undef TARGET_ASM_ALIGNED_SI_OP#define TARGET_ASM_ALIGNED_SI_OP NULL#undef TARGET_ASM_INTEGER#define TARGET_ASM_INTEGER elxsi_assemble_integer#undef TARGET_ASM_FUNCTION_PROLOGUE#define TARGET_ASM_FUNCTION_PROLOGUE elxsi_output_function_prologue#undef TARGET_ASM_FUNCTION_EPILOGUE#define TARGET_ASM_FUNCTION_EPILOGUE elxsi_output_function_epiloguestruct gcc_target targetm = TARGET_INITIALIZER;/* Target hook for assembling integer objects.  The ELXSI assembler   syntax uses a suffix to indicate the size of data, so we can't use   the usual string hooks.  */static boolelxsi_assemble_integer (x, size, aligned_p)     rtx x;     unsigned int size;     int aligned_p;{  if (aligned_p)    switch (size)      {      case 1:      case 2:      case 4:	fputs ("\t.data\t", asm_out_file);	output_addr_const (asm_out_file, x);	fprintf (asm_out_file, "{%d}\n", size * BITS_PER_UNIT);	return true;      }  return default_assemble_integer (x, size, aligned_p);}/* Generate the assembly code for function entry.  FILE is a stdio   stream to output the code to.  SIZE is an int: how many units of   temporary storage to allocate.   Refer to the array `regs_ever_live' to determine which registers to   save; `regs_ever_live[I]' is nonzero if register number I is ever   used in the function.  This function is responsible for knowing   which registers should not be saved even if used.  */static voidelxsi_output_function_prologue (file, size)     FILE *file;     HOST_WIDE_INT size;{  register int regno;  register int cnt = 0;  /* the below two lines are a HACK, and should be deleted, but     for now are very much needed (1.35) */  if (frame_pointer_needed)    regs_ever_live[14] = 1, call_used_regs[14] = 0;  for (regno = 0; regno < FIRST_PSEUDO_REGISTER; regno++)    if (regs_ever_live[regno] && !call_used_regs[regno])      cnt += 8;  if (size + cnt)    fprintf (file, "\tadd.64\t.sp,=%d\n", -size - cnt);  cnt = 0;  for (regno = 0; regno < FIRST_PSEUDO_REGISTER; regno++)    if (regs_ever_live[regno] && !call_used_regs[regno])      fprintf (file, "\tst.64\t.r%d,[.sp]%d\n", regno, (cnt += 8) - 12);  if (frame_pointer_needed)    fprintf (file, "\tadd.64\t.r14,.sp,=%d\n", size + cnt);}/* This function generates the assembly code for function exit.   Args are as for output_function_prologue ().   The function epilogue should not depend on the current stack   pointer!  It should use the frame pointer only.  This is mandatory   because of alloca; we also take advantage of it to omit stack   adjustments before returning. */static voidelxsi_output_function_epilogue (file, size)     FILE *file;     HOST_WIDE_INT size;{  register int regno;  register int cnt = 0;  /* this conditional is ONLY here because there is a BUG;     EXIT_IGNORE_STACK is ignored itself when the first part of     the condition is true! (at least in version 1.35) */  /* the 8*10 is for 64 bits of .r5 - .r14 */  if (current_function_calls_alloca || size >= (256 - 8 * 10))    {      /* use .r4 as a temporary! Ok for now.... */      fprintf (file, "\tld.64\t.r4,.r14\n");      for (regno = FIRST_PSEUDO_REGISTER-1; regno >= 0; --regno)	if (regs_ever_live[regno] && !call_used_regs[regno])	  cnt += 8;      for (regno = 0; regno < FIRST_PSEUDO_REGISTER; ++regno)	if (regs_ever_live[regno] && !call_used_regs[regno])	  fprintf (file, "\tld.64\t.r%d,[.r14]%d\n", regno,		   -((cnt -= 8) + 8) - 4 - size);      fprintf (file, "\tld.64\t.sp,.r4\n\texit\t0\n");    }  else    {      for (regno = 0; regno < FIRST_PSEUDO_REGISTER; ++regno)	if (regs_ever_live[regno] && !call_used_regs[regno])	  fprintf (file, "\tld.64\t.r%d,[.sp]%d\n", regno, (cnt += 8) - 12);      fprintf (file, "\texit\t%d\n", size + cnt);    }}/* type is the index into the above table *//* s is "" for signed, or "u" for unsigned */const char *cmp_jmp (s, type, where)     const char *s;     int type;     rtx where;{    rtx br_ops[3];    char template[50];    const char *f = "";    const char *bits = "64";    if (GET_MODE (cmp_op0) == SFmode) f = "f", bits = "32";    if (GET_MODE (cmp_op0) == DFmode) f = "f";    br_ops[0] = where;    br_ops[1] = cmp_op0;    br_ops[2] = cmp_op1;    if (cmp_op1)	sprintf(template, "%scmp%s.br.%s\t%%1,%%2:j%s\t%%l0",		f, s, bits, cmp_tab[type]);    else if (*f)	sprintf(template, "fcmp.br.%s\t%%1,=0:j%s\t%%l0",		bits, cmp_tab[type]);    else if (*s) /* can turn the below in to a jmp ... */	sprintf(template, "cmpu.br.64\t%%1,=0:j%s\t%%l0", s);    else	sprintf(template, "jmp.%s\t%%1,%%l0", cmp_tab[type+1]);    output_asm_insn(template, br_ops);    return "";}const char *cmp_set (s, type, reg)     const char *s, *type;     rtx reg;{    rtx br_ops[3];    char template[50];    const char *f = "";    const char *bits = "64";    if (GET_MODE (cmp_op0) == SFmode) f = "f", bits = "32";    else if (GET_MODE (cmp_op0) == DFmode) f = "f";    else if (GET_MODE (cmp_op0) == SImode) bits = "32";    else if (GET_MODE (cmp_op0) == HImode) bits = "16";    else if (GET_MODE (cmp_op0) == QImode) bits = "8";    br_ops[0] = reg;    br_ops[1] = cmp_op0;    br_ops[2] = cmp_op1;    if (cmp_op1)	sprintf(template, "%scmp%s.%s\t%%0,%%1,%%2:%s",		f, s, bits, type);    else	sprintf(template, "%scmp%s.%s\t%%0,%%1,=0:%s",		f, s, bits, type);    output_asm_insn(template, br_ops);    return "";}voidprint_operand_address (file, addr)     FILE *file;     register rtx addr;{  register rtx reg1, reg2, breg, ireg;  rtx offset;  switch (GET_CODE (addr))    {    case MEM:      if (GET_CODE (XEXP (addr, 0)) == REG)        fprintf (file, "%s", reg_names[REGNO (addr)]);      else abort();      break;    case REG:      fprintf (file, "[%s]", reg_names[REGNO (addr)]);      break;    case PLUS:      reg1 = 0;	reg2 = 0;      ireg = 0;	breg = 0;      offset = 0;      if (GET_CODE (XEXP (addr, 0)) == REG)	{	  offset = XEXP (addr, 1);	  addr = XEXP (addr, 0);	}      else if (GET_CODE (XEXP (addr, 1)) == REG)	{	  offset = XEXP (addr, 0);	  addr = XEXP (addr, 1);	}      fprintf (file, "[%s]", reg_names[REGNO (addr)]);      output_address (offset);      break;    default:      output_addr_const (file, addr);    }}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?