entered into RCS
From-SVN: r188
This commit is contained in:
parent
2042e9d2d1
commit
e1c8bbf44b
220
gcc/config/convex/convex.c
Normal file
220
gcc/config/convex/convex.c
Normal file
@ -0,0 +1,220 @@
|
|||||||
|
/* Subroutines for insn-output.c for Convex.
|
||||||
|
Copyright (C) 1989,1991 Free Software Foundation, Inc.
|
||||||
|
|
||||||
|
This file is part of GNU CC.
|
||||||
|
|
||||||
|
GNU CC is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the 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 of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with GNU CC; see the file COPYING. If not, write to
|
||||||
|
the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
#include "rtl.h"
|
||||||
|
#include "regs.h"
|
||||||
|
#include "hard-reg-set.h"
|
||||||
|
#include "real.h"
|
||||||
|
#include "insn-config.h"
|
||||||
|
#include "conditions.h"
|
||||||
|
#include "insn-flags.h"
|
||||||
|
#include "output.h"
|
||||||
|
#include "insn-attr.h"
|
||||||
|
|
||||||
|
/* Boolean to keep track of whether the current section is .text or not.
|
||||||
|
Used by .align handler in convex.h. */
|
||||||
|
|
||||||
|
int current_section_is_text;
|
||||||
|
|
||||||
|
/* set_cmp saves the operands of a "cmp" insn, along with the type character
|
||||||
|
* to be used in the compare instruction.
|
||||||
|
*
|
||||||
|
* gen_cmp finds out what comparison is to be performed and outputs the
|
||||||
|
* necessary instructions, e.g.
|
||||||
|
* "eq.w a1,a2\;jbra.t L5"
|
||||||
|
* for (cmpsi a1 a2) (beq L5) */
|
||||||
|
|
||||||
|
static rtx xop0, xop1;
|
||||||
|
static char typech, regch;
|
||||||
|
|
||||||
|
char *
|
||||||
|
set_cmp (op0, op1, typechr)
|
||||||
|
rtx op0, op1;
|
||||||
|
char typechr;
|
||||||
|
{
|
||||||
|
xop0 = op0;
|
||||||
|
xop1 = op1;
|
||||||
|
typech = typechr;
|
||||||
|
if (GET_CODE (op0) == REG)
|
||||||
|
regch = A_REGNO_P (REGNO (op0)) ? 'a' : 's';
|
||||||
|
else if (GET_CODE (op1) == REG)
|
||||||
|
regch = A_REGNO_P (REGNO (op1)) ? 'a' : 's';
|
||||||
|
else abort ();
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
char *
|
||||||
|
gen_cmp (label, cmpop, tf)
|
||||||
|
rtx label;
|
||||||
|
char *cmpop;
|
||||||
|
char tf;
|
||||||
|
{
|
||||||
|
char buf[80];
|
||||||
|
char revop[4];
|
||||||
|
rtx ops[3];
|
||||||
|
|
||||||
|
ops[2] = label;
|
||||||
|
|
||||||
|
/* Constant must be first; swap operands if necessary.
|
||||||
|
If lt, le, ltu, leu are swapped, change to le, lt, leu, ltu
|
||||||
|
and reverse the sense of the jump. */
|
||||||
|
|
||||||
|
if (CONSTANT_P (xop1))
|
||||||
|
{
|
||||||
|
ops[0] = xop1;
|
||||||
|
ops[1] = xop0;
|
||||||
|
if (cmpop[0] == 'l')
|
||||||
|
{
|
||||||
|
bcopy (cmpop, revop, sizeof revop);
|
||||||
|
revop[1] ^= 'e' ^ 't';
|
||||||
|
tf ^= 't' ^ 'f';
|
||||||
|
cmpop = revop;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ops[0] = xop0;
|
||||||
|
ops[1] = xop1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sprintf (buf, "%s.%c %%0,%%1\n\tjbr%c.%c %%l2", cmpop, typech, regch, tf);
|
||||||
|
output_asm_insn (buf, ops);
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Routines to separate CONST_DOUBLEs into component parts. */
|
||||||
|
|
||||||
|
int
|
||||||
|
const_double_high_int (x)
|
||||||
|
rtx x;
|
||||||
|
{
|
||||||
|
if (GET_MODE_CLASS (GET_MODE (x)) == MODE_FLOAT)
|
||||||
|
return CONST_DOUBLE_LOW (x);
|
||||||
|
else
|
||||||
|
return CONST_DOUBLE_HIGH (x);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
const_double_low_int (x)
|
||||||
|
rtx x;
|
||||||
|
{
|
||||||
|
if (GET_MODE_CLASS (GET_MODE (x)) == MODE_FLOAT)
|
||||||
|
return CONST_DOUBLE_HIGH (x);
|
||||||
|
else
|
||||||
|
return CONST_DOUBLE_LOW (x);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Return the number of args in the call insn X. */
|
||||||
|
|
||||||
|
static int
|
||||||
|
call_num_args (x)
|
||||||
|
rtx x;
|
||||||
|
{
|
||||||
|
if (GET_CODE (x) == CALL)
|
||||||
|
return INTVAL (x->fld[1].rtx);
|
||||||
|
if (GET_CODE (x) == SET)
|
||||||
|
return call_num_args (SET_SRC (x));
|
||||||
|
abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scan forward from a call to decide whether we need to reload AP
|
||||||
|
from 12(FP) after it. We need to if there can be a reference to
|
||||||
|
arg_pointer_rtx before the next call, which will clobber AP.
|
||||||
|
Look forward in the instruction list until encountering a call
|
||||||
|
(don't need the load), or a reference to AP (do need it), or
|
||||||
|
a jump (don't know, do the load). */
|
||||||
|
|
||||||
|
static int
|
||||||
|
ap_reload_needed (insn)
|
||||||
|
rtx insn;
|
||||||
|
{
|
||||||
|
for (;;)
|
||||||
|
{
|
||||||
|
insn = NEXT_INSN (insn);
|
||||||
|
switch (GET_CODE (insn))
|
||||||
|
{
|
||||||
|
case JUMP_INSN:
|
||||||
|
/* Basic block ends. If return, no AP needed, else assume it is. */
|
||||||
|
return GET_CODE (PATTERN (insn)) != RETURN;
|
||||||
|
case CALL_INSN:
|
||||||
|
/* A subsequent call. AP isn't needed unless the call itself
|
||||||
|
requires it. But zero-arg calls don't clobber AP, so
|
||||||
|
don't terminate the search in that case. */
|
||||||
|
if (reg_mentioned_p (arg_pointer_rtx, PATTERN (insn)))
|
||||||
|
return 1;
|
||||||
|
if (! TARGET_ARGCOUNT && call_num_args (PATTERN (insn)) == 0)
|
||||||
|
break;
|
||||||
|
return 0;
|
||||||
|
case BARRIER:
|
||||||
|
/* Barrier, don't need AP. */
|
||||||
|
return 0;
|
||||||
|
case INSN:
|
||||||
|
/* Other insn may need AP; if not, keep looking. */
|
||||||
|
if (reg_mentioned_p (arg_pointer_rtx, PATTERN (insn)))
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Output the insns needed to do a call. */
|
||||||
|
|
||||||
|
char *
|
||||||
|
output_call (insn, address, argcount)
|
||||||
|
rtx insn, address, argcount;
|
||||||
|
{
|
||||||
|
int set_ap = TARGET_ARGCOUNT || argcount != const0_rtx;
|
||||||
|
|
||||||
|
/* If AP is used by the call address, evaluate the address into a temp. */
|
||||||
|
if (reg_mentioned_p (arg_pointer_rtx, address))
|
||||||
|
if (set_ap)
|
||||||
|
{
|
||||||
|
address = XEXP (address, 0);
|
||||||
|
output_asm_insn ("ld.w %0,a1", &address);
|
||||||
|
address = gen_rtx (MEM, QImode, gen_rtx (REG, Pmode, 9));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If there are args, point AP to them. */
|
||||||
|
if (set_ap)
|
||||||
|
output_asm_insn ("mov sp,ap");
|
||||||
|
|
||||||
|
/* If we are passing an arg count, convert it to words and push it. */
|
||||||
|
if (TARGET_ARGCOUNT)
|
||||||
|
{
|
||||||
|
argcount = gen_rtx (CONST_INT, VOIDmode, (INTVAL (argcount) + 3) / 4);
|
||||||
|
output_asm_insn ("pshea %a0", &argcount);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The call. */
|
||||||
|
output_asm_insn ("calls %0", &address);
|
||||||
|
|
||||||
|
/* If we clobbered AP, reload it if it is live. */
|
||||||
|
if (set_ap)
|
||||||
|
if (ap_reload_needed (insn))
|
||||||
|
output_asm_insn ("ld.w 12(fp),ap");
|
||||||
|
|
||||||
|
/* If we pushed an arg count, pop it and the args. */
|
||||||
|
if (TARGET_ARGCOUNT)
|
||||||
|
{
|
||||||
|
argcount = gen_rtx (CONST_INT, VOIDmode, INTVAL (argcount) * 4 + 4);
|
||||||
|
output_asm_insn ("add.w %0,sp", &argcount);
|
||||||
|
}
|
||||||
|
|
||||||
|
return "";
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user