⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 out-spur.c

📁 这是完整的gcc源代码
💻 C
字号:
/* Subroutines for insn-output.c for SPUR.  Adapted from routines for   the Motorola 68000 family.   Copyright (C) 1988 Free Software Foundation, Inc.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 1, 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, 675 Mass Ave, Cambridge, MA 02139, USA.  */static rtx find_addr_reg ();char *output_compare (operands, opcode, exchange_opcode, 		neg_opcode, neg_exchange_opcode)     rtx *operands;     char *opcode;     char *exchange_opcode;     char *neg_opcode;     char *neg_exchange_opcode;{  static char buf[100];  operands[2] = operands[0];  if (GET_CODE (cc_prev_status.value1) == CONST_INT)    {      operands[1] = cc_prev_status.value1;      operands[0] = cc_prev_status.value2;      opcode = exchange_opcode, neg_opcode = neg_exchange_opcode;    }  else    {      operands[0] = cc_prev_status.value1;      operands[1] = cc_prev_status.value2;    }  if (TARGET_LONG_JUMPS)    sprintf (buf,	     "cmp_br_delayed %s,%%0,%%1,1f\n\tnop\n\tjump %%l2\n\tnop\n1:",	     neg_opcode);  else     sprintf (buf, "cmp_br_delayed %s,%%0,%%1,%%l2\n\tnop", opcode);  return buf;}/* Return the best assembler insn template   for moving operands[1] into operands[0] as a fullword.  */static char *singlemove_string (operands)     rtx *operands;{  if (GET_CODE (operands[0]) == MEM)    return "st_32 %r1,%0";  if (GET_CODE (operands[1]) == MEM)    return "ld_32 %0,%1\n\tnop";  if (GET_CODE (operands[1]) == REG)    return "add_nt %0,%1,$0";  return "add_nt %0,r0,%1";}/* Output assembler code to perform a doubleword move insn   with operands OPERANDS.  */char *output_move_double (operands)     rtx *operands;{  enum { REGOP, OFFSOP, MEMOP, PUSHOP, POPOP, CNSTOP, RNDOP } optype0, optype1;  rtx latehalf[2];  rtx addreg0 = 0, addreg1 = 0;  /* First classify both operands.  */  if (REG_P (operands[0]))    optype0 = REGOP;  else if (offsettable_memref_p (operands[0]))    optype0 = OFFSOP;  else if (GET_CODE (operands[0]) == MEM)    optype0 = MEMOP;  else    optype0 = RNDOP;  if (REG_P (operands[1]))    optype1 = REGOP;  else if (CONSTANT_P (operands[1])	   || GET_CODE (operands[1]) == CONST_DOUBLE)    optype1 = CNSTOP;  else if (offsettable_memref_p (operands[1]))    optype1 = OFFSOP;  else if (GET_CODE (operands[1]) == MEM)    optype1 = MEMOP;  else    optype1 = RNDOP;  /* Check for the cases that the operand constraints are not     supposed to allow to happen.  Abort if we get one,     because generating code for these cases is painful.  */  if (optype0 == RNDOP || optype1 == RNDOP)    abort ();  /* If an operand is an unoffsettable memory ref, find a register     we can increment temporarily to make it refer to the second word.  */  if (optype0 == MEMOP)    addreg0 = find_addr_reg (XEXP (operands[0], 0));  if (optype1 == MEMOP)    addreg1 = find_addr_reg (XEXP (operands[1], 0));  /* Ok, we can do one word at a time.     Normally we do the low-numbered word first,     but if either operand is autodecrementing then we     do the high-numbered word first.     In either case, set up in LATEHALF the operands to use     for the high-numbered word and in some cases alter the     operands in OPERANDS to be suitable for the low-numbered word.  */  if (optype0 == REGOP)    latehalf[0] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1);  else if (optype0 == OFFSOP)    latehalf[0] = adj_offsettable_operand (operands[0], 4);  else    latehalf[0] = operands[0];  if (optype1 == REGOP)    latehalf[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1);  else if (optype1 == OFFSOP)    latehalf[1] = adj_offsettable_operand (operands[1], 4);  else if (optype1 == CNSTOP)    {      if (CONSTANT_P (operands[1]))	latehalf[1] = const0_rtx;      else if (GET_CODE (operands[1]) == CONST_DOUBLE)	{	  latehalf[1] = gen_rtx (CONST_INT, VOIDmode,				 CONST_DOUBLE_HIGH (operands[1]));	  operands[1] = gen_rtx (CONST_INT, VOIDmode,				 CONST_DOUBLE_LOW (operands[1]));	}    }  else    latehalf[1] = operands[1];  /* If the first move would clobber the source of the second one,     do them in the other order.  This happens only for registers;     such overlap can't happen in memory unless the user explicitly     sets it up, and that is an undefined circumstance.  */  if (optype0 == REGOP && optype1 == REGOP      && REGNO (operands[0]) == REGNO (latehalf[1]))    {      /* Make any unoffsettable addresses point at high-numbered word.  */      if (addreg0)	output_asm_insn ("add_nt %0,%0,$4", &addreg0);      if (addreg1)	output_asm_insn ("add_nt %0,%0,$4", &addreg1);      /* Do that word.  */      output_asm_insn (singlemove_string (latehalf), latehalf);      /* Undo the adds we just did.  */      if (addreg0)	output_asm_insn ("add_nt %0,%0,$-4", &addreg0);      if (addreg1)	output_asm_insn ("add_nt %0,%0,$-4", &addreg0);      /* Do low-numbered word.  */      return singlemove_string (operands);    }  /* Normal case: do the two words, low-numbered first.  */  output_asm_insn (singlemove_string (operands), operands);  /* Make any unoffsettable addresses point at high-numbered word.  */  if (addreg0)    output_asm_insn ("add_nt %0,%0,$4", &addreg0);  if (addreg1)    output_asm_insn ("add_nt %0,%0,$4", &addreg1);  /* Do that word.  */  output_asm_insn (singlemove_string (latehalf), latehalf);  /* Undo the adds we just did.  */  if (addreg0)    output_asm_insn ("add_nt %0,%0,$-4", &addreg0);  if (addreg1)    output_asm_insn ("add_nt %0,%0,$-4", &addreg1);  return "";}static char *output_fp_move_double (operands)     rtx *operands;{  if (FP_REG_P (operands[0]))    {      if (FP_REG_P (operands[1]))	return "fmov %0,%1";      if (GET_CODE (operands[1]) == REG)	{	  rtx xoperands[2];	  int offset = - get_frame_size () - 8;	  xoperands[1] = gen_rtx (REG, SImode, REGNO (operands[1]) + 1);	  xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset + 4);	  output_asm_insn ("st_32 %1,r25,%0", xoperands);	  xoperands[1] = operands[1];	  xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset);	  output_asm_insn ("st_32 %1,r25,%0", xoperands);	  xoperands[1] = operands[0];	  output_asm_insn ("ld_dbl %1,r25,%0\n\tnop", xoperands);	  return "";	}      return "ld_dbl %0,%1\n\tnop";    }  else if (FP_REG_P (operands[1]))    {      if (GET_CODE (operands[0]) == REG)	{	  rtx xoperands[2];	  int offset = - get_frame_size () - 8;	  xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset);	  xoperands[1] = operands[1];	  output_asm_insn ("st_dbl %1,r25,%0", xoperands);	  xoperands[1] = operands[0];	  output_asm_insn ("ld_32 %1,r25,%0\n\tnop", xoperands);	  xoperands[1] = gen_rtx (REG, SImode, REGNO (operands[0]) + 1);	  xoperands[0] = gen_rtx (CONST_INT, VOIDmode, offset + 4);	  output_asm_insn ("ld_32 %1,r25,%0\n\tnop", xoperands);	  return "";	}      return "st_dbl %1,%0";    }}/* Return a REG that occurs in ADDR with coefficient 1.   ADDR can be effectively incremented by incrementing REG.  */static rtxfind_addr_reg (addr)     rtx addr;{  while (GET_CODE (addr) == PLUS)    {      if (GET_CODE (XEXP (addr, 0)) == REG)	addr = XEXP (addr, 0);      else if (GET_CODE (XEXP (addr, 1)) == REG)	addr = XEXP (addr, 1);      else if (CONSTANT_P (XEXP (addr, 0)))	addr = XEXP (addr, 1);      else if (CONSTANT_P (XEXP (addr, 1)))	addr = XEXP (addr, 0);      else	abort ();    }  if (GET_CODE (addr) == REG)    return addr;  abort ();}/* Generate code to add a large integer constant to register, reg, storing * the result in a register, target.  Offset must be 27-bit signed quantity */static char *output_add_large_offset (target, reg, offset)     rtx target, reg;     int offset;{  rtx operands[3];  int high, n, i;  operands[0] = target, operands[1] = reg;      for (high = offset, n = 0;        (unsigned) (high + 0x2000) >= 0x4000;        high >>= 1, n += 1)    ;  operands[2] = gen_rtx (CONST_INT, VOIDmode, high);  output_asm_insn ("add_nt r2,r0,%2", operands);  i = n;  while (i >= 3)    output_asm_insn ("sll r2,r2,$3", operands), i -= 3;  if (i == 2)     output_asm_insn ("sll r2,r2,$2", operands);  else if (i == 1)    output_asm_insn ("sll r2,r2,$1", operands);  output_asm_insn ("add_nt %0,r2,%1", operands);  if (offset - (high << n) != 0)    {      operands[2] = gen_rtx (CONST_INT, VOIDmode, offset - (high << n));      output_asm_insn ("add_nt %0,%0,%2", operands);    }  return "";}/* Additional TESTFN for matching. Like immediate_operand, but matches big * constants */intbig_immediate_operand (op, mode)     rtx op;     enum machine_mode mode;{  return (GET_CODE (op) == CONST_INT);}

⌨️ 快捷键说明

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