f5a2889838
From-SVN: r5535
1958 lines
53 KiB
C
1958 lines
53 KiB
C
/* Subroutines used for code generation on IBM RS/6000.
|
||
Copyright (C) 1991, 1993 Free Software Foundation, Inc.
|
||
Contributed by Richard Kenner (kenner@nyu.edu)
|
||
|
||
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 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 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 <stdio.h>
|
||
#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"
|
||
#include "flags.h"
|
||
#include "recog.h"
|
||
#include "expr.h"
|
||
#include "obstack.h"
|
||
#include "tree.h"
|
||
|
||
extern char *language_string;
|
||
extern int profile_block_flag;
|
||
|
||
#define min(A,B) ((A) < (B) ? (A) : (B))
|
||
#define max(A,B) ((A) > (B) ? (A) : (B))
|
||
|
||
/* Target cpu type */
|
||
|
||
enum processor_type rs6000_cpu;
|
||
char *rs6000_cpu_string;
|
||
|
||
/* Set to non-zero by "fix" operation to indicate that itrunc and
|
||
uitrunc must be defined. */
|
||
|
||
int rs6000_trunc_used;
|
||
|
||
/* Set to non-zero once they have been defined. */
|
||
|
||
static int trunc_defined;
|
||
|
||
/* Save information from a "cmpxx" operation until the branch or scc is
|
||
emitted. */
|
||
|
||
rtx rs6000_compare_op0, rs6000_compare_op1;
|
||
int rs6000_compare_fp_p;
|
||
|
||
/* Override command line options. Mostly we process the processor
|
||
type and sometimes adjust other TARGET_ options. */
|
||
|
||
void
|
||
rs6000_override_options ()
|
||
{
|
||
int i;
|
||
|
||
static struct ptt
|
||
{
|
||
char *name; /* Canonical processor name. */
|
||
enum processor_type processor; /* Processor type enum value. */
|
||
int target_enable; /* Target flags to enable. */
|
||
int target_disable; /* Target flags to disable. */
|
||
} processor_target_table[]
|
||
= {{"all", PROCESSOR_DEFAULT,
|
||
0, MASK_POWER | MASK_POWERPC | MASK_POWERPC64},
|
||
{"rios", PROCESSOR_RIOS1,
|
||
MASK_POWER, MASK_POWERPC | MASK_POWERPC64},
|
||
{"rios1", PROCESSOR_RIOS1,
|
||
MASK_POWER, MASK_POWERPC | MASK_POWERPC64},
|
||
{"rios2", PROCESSOR_RIOS2,
|
||
MASK_POWER, MASK_POWERPC | MASK_POWERPC64},
|
||
{"601", PROCESSOR_PPC601,
|
||
MASK_POWER | MASK_POWERPC | MASK_NEW_MNEMONICS, MASK_POWERPC64},
|
||
{"mpc601", PROCESSOR_PPC601,
|
||
MASK_POWER | MASK_POWERPC | MASK_NEW_MNEMONICS, MASK_POWERPC64},
|
||
{"603", PROCESSOR_PPC603,
|
||
MASK_POWERPC | MASK_NEW_MNEMONICS, MASK_POWER | MASK_POWERPC64},
|
||
{"mpc603", PROCESSOR_PPC603,
|
||
MASK_POWERPC | MASK_NEW_MNEMONICS, MASK_POWER | MASK_POWERPC64},
|
||
{"604", PROCESSOR_PPC604,
|
||
MASK_POWERPC | MASK_NEW_MNEMONICS, MASK_POWER | MASK_POWERPC64},
|
||
{"mpc604", PROCESSOR_PPC604,
|
||
MASK_POWERPC | MASK_NEW_MNEMONICS, MASK_POWER | MASK_POWERPC64},
|
||
{"620", PROCESSOR_PPC620,
|
||
MASK_POWERPC | MASK_POWERPC64 | MASK_NEW_MNEMONICS, MASK_POWER},
|
||
{"mpc620", PROCESSOR_PPC620,
|
||
MASK_POWERPC | MASK_POWERPC64 | MASK_NEW_MNEMONICS, MASK_POWER}};
|
||
|
||
int ptt_size = sizeof (processor_target_table) / sizeof (struct ptt);
|
||
|
||
profile_block_flag = 0;
|
||
|
||
/* Identify the processor type */
|
||
if (rs6000_cpu_string == 0)
|
||
rs6000_cpu = PROCESSOR_DEFAULT;
|
||
else
|
||
{
|
||
for (i = 0; i < ptt_size; i++)
|
||
if (! strcmp (rs6000_cpu_string, processor_target_table[i].name))
|
||
{
|
||
rs6000_cpu = processor_target_table[i].processor;
|
||
target_flags |= processor_target_table[i].target_enable;
|
||
target_flags &= ~processor_target_table[i].target_disable;
|
||
break;
|
||
}
|
||
|
||
if (i == ptt_size)
|
||
{
|
||
error ("bad value (%s) for -mcpu= switch", rs6000_cpu_string);
|
||
rs6000_cpu_string = "default";
|
||
rs6000_cpu = PROCESSOR_DEFAULT;
|
||
}
|
||
}
|
||
}
|
||
|
||
/* Return non-zero if this function is known to have a null epilogue. */
|
||
|
||
int
|
||
direct_return ()
|
||
{
|
||
return (reload_completed
|
||
&& first_reg_to_save () == 32
|
||
&& first_fp_reg_to_save () == 64
|
||
&& ! regs_ever_live[65]
|
||
&& ! rs6000_pushes_stack ());
|
||
}
|
||
|
||
/* Returns 1 always. */
|
||
|
||
int
|
||
any_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return 1;
|
||
}
|
||
|
||
/* Return 1 if OP is a constant that can fit in a D field. */
|
||
|
||
int
|
||
short_cint_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (GET_CODE (op) == CONST_INT
|
||
&& (unsigned) (INTVAL (op) + 0x8000) < 0x10000);
|
||
}
|
||
|
||
/* Similar for a unsigned D field. */
|
||
|
||
int
|
||
u_short_cint_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (GET_CODE (op) == CONST_INT && (INTVAL (op) & 0xffff0000) == 0);
|
||
}
|
||
|
||
/* Return 1 if OP is a CONST_INT that cannot fit in a signed D field. */
|
||
|
||
int
|
||
non_short_cint_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (GET_CODE (op) == CONST_INT
|
||
&& (unsigned) (INTVAL (op) + 0x8000) >= 0x10000);
|
||
}
|
||
|
||
/* Returns 1 if OP is a register that is not special (i.e., not MQ,
|
||
ctr, or lr). */
|
||
|
||
int
|
||
gpc_reg_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (register_operand (op, mode)
|
||
&& (GET_CODE (op) != REG || REGNO (op) >= 67 || REGNO (op) < 64));
|
||
}
|
||
|
||
/* Returns 1 if OP is either a pseudo-register or a register denoting a
|
||
CR field. */
|
||
|
||
int
|
||
cc_reg_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (register_operand (op, mode)
|
||
&& (GET_CODE (op) != REG
|
||
|| REGNO (op) >= FIRST_PSEUDO_REGISTER
|
||
|| CR_REGNO_P (REGNO (op))));
|
||
}
|
||
|
||
/* Returns 1 if OP is either a constant integer valid for a D-field or a
|
||
non-special register. If a register, it must be in the proper mode unless
|
||
MODE is VOIDmode. */
|
||
|
||
int
|
||
reg_or_short_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return short_cint_operand (op, mode) || gpc_reg_operand (op, mode);
|
||
}
|
||
|
||
/* Similar, except check if the negation of the constant would be valid for
|
||
a D-field. */
|
||
|
||
int
|
||
reg_or_neg_short_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
if (GET_CODE (op) == CONST_INT)
|
||
return CONST_OK_FOR_LETTER_P (INTVAL (op), 'P');
|
||
|
||
return gpc_reg_operand (op, mode);
|
||
}
|
||
|
||
/* Return 1 if the operand is either a register or an integer whose high-order
|
||
16 bits are zero. */
|
||
|
||
int
|
||
reg_or_u_short_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
if (GET_CODE (op) == CONST_INT
|
||
&& (INTVAL (op) & 0xffff0000) == 0)
|
||
return 1;
|
||
|
||
return gpc_reg_operand (op, mode);
|
||
}
|
||
|
||
/* Return 1 is the operand is either a non-special register or ANY
|
||
constant integer. */
|
||
|
||
int
|
||
reg_or_cint_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return GET_CODE (op) == CONST_INT || gpc_reg_operand (op, mode);
|
||
}
|
||
|
||
/* Return 1 if the operand is a CONST_DOUBLE and it can be put into a
|
||
register with one instruction per word. For SFmode, this means that
|
||
the low 16-bits are zero. For DFmode, it means the low 16-bits of
|
||
the first word are zero and the high 16 bits of the second word
|
||
are zero (usually all bits in the low-order word will be zero).
|
||
|
||
We only do this if we can safely read CONST_DOUBLE_{LOW,HIGH}. */
|
||
|
||
int
|
||
easy_fp_constant (op, mode)
|
||
register rtx op;
|
||
register enum machine_mode mode;
|
||
{
|
||
rtx low, high;
|
||
|
||
if (GET_CODE (op) != CONST_DOUBLE
|
||
|| GET_MODE (op) != mode
|
||
|| GET_MODE_CLASS (mode) != MODE_FLOAT)
|
||
return 0;
|
||
|
||
high = operand_subword (op, 0, 0, mode);
|
||
low = operand_subword (op, 1, 0, mode);
|
||
|
||
if (high == 0 || GET_CODE (high) != CONST_INT || (INTVAL (high) & 0xffff))
|
||
return 0;
|
||
|
||
return (mode == SFmode
|
||
|| (low != 0 && GET_CODE (low) == CONST_INT
|
||
&& (INTVAL (low) & 0xffff0000) == 0));
|
||
}
|
||
|
||
/* Return 1 if the operand is either a floating-point register, a pseudo
|
||
register, or memory. */
|
||
|
||
int
|
||
fp_reg_or_mem_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (memory_operand (op, mode)
|
||
|| (register_operand (op, mode)
|
||
&& (GET_CODE (op) != REG
|
||
|| REGNO (op) >= FIRST_PSEUDO_REGISTER
|
||
|| FP_REGNO_P (REGNO (op)))));
|
||
}
|
||
|
||
/* Return 1 if the operand is either an easy FP constant (see above) or
|
||
memory. */
|
||
|
||
int
|
||
mem_or_easy_const_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return memory_operand (op, mode) || easy_fp_constant (op, mode);
|
||
}
|
||
|
||
/* Return 1 if the operand is either a non-special register or an item
|
||
that can be used as the operand of an SI add insn. */
|
||
|
||
int
|
||
add_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (reg_or_short_operand (op, mode)
|
||
|| (GET_CODE (op) == CONST_INT && (INTVAL (op) & 0xffff) == 0));
|
||
}
|
||
|
||
/* Return 1 if OP is a constant but not a valid add_operand. */
|
||
|
||
int
|
||
non_add_cint_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (GET_CODE (op) == CONST_INT
|
||
&& (unsigned) (INTVAL (op) + 0x8000) >= 0x10000
|
||
&& (INTVAL (op) & 0xffff) != 0);
|
||
}
|
||
|
||
/* Return 1 if the operand is a non-special register or a constant that
|
||
can be used as the operand of an OR or XOR insn on the RS/6000. */
|
||
|
||
int
|
||
logical_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (gpc_reg_operand (op, mode)
|
||
|| (GET_CODE (op) == CONST_INT
|
||
&& ((INTVAL (op) & 0xffff0000) == 0
|
||
|| (INTVAL (op) & 0xffff) == 0)));
|
||
}
|
||
|
||
/* Return 1 if C is a constant that is not a logical operand (as
|
||
above). */
|
||
|
||
int
|
||
non_logical_cint_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (GET_CODE (op) == CONST_INT
|
||
&& (INTVAL (op) & 0xffff0000) != 0
|
||
&& (INTVAL (op) & 0xffff) != 0);
|
||
}
|
||
|
||
/* Return 1 if C is a constant that can be encoded in a mask on the
|
||
RS/6000. It is if there are no more than two 1->0 or 0->1 transitions.
|
||
Reject all ones and all zeros, since these should have been optimized
|
||
away and confuse the making of MB and ME. */
|
||
|
||
int
|
||
mask_constant (c)
|
||
register int c;
|
||
{
|
||
int i;
|
||
int last_bit_value;
|
||
int transitions = 0;
|
||
|
||
if (c == 0 || c == ~0)
|
||
return 0;
|
||
|
||
last_bit_value = c & 1;
|
||
|
||
for (i = 1; i < 32; i++)
|
||
if (((c >>= 1) & 1) != last_bit_value)
|
||
last_bit_value ^= 1, transitions++;
|
||
|
||
return transitions <= 2;
|
||
}
|
||
|
||
/* Return 1 if the operand is a constant that is a mask on the RS/6000. */
|
||
|
||
int
|
||
mask_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return GET_CODE (op) == CONST_INT && mask_constant (INTVAL (op));
|
||
}
|
||
|
||
/* Return 1 if the operand is either a non-special register or a
|
||
constant that can be used as the operand of an RS/6000 logical AND insn. */
|
||
|
||
int
|
||
and_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (reg_or_short_operand (op, mode)
|
||
|| logical_operand (op, mode)
|
||
|| mask_operand (op, mode));
|
||
}
|
||
|
||
/* Return 1 if the operand is a constant but not a valid operand for an AND
|
||
insn. */
|
||
|
||
int
|
||
non_and_cint_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return GET_CODE (op) == CONST_INT && ! and_operand (op, mode);
|
||
}
|
||
|
||
/* Return 1 if the operand is a general register or memory operand. */
|
||
|
||
int
|
||
reg_or_mem_operand (op, mode)
|
||
register rtx op;
|
||
register enum machine_mode mode;
|
||
{
|
||
return gpc_reg_operand (op, mode) || memory_operand (op, mode);
|
||
}
|
||
|
||
/* Return 1 if the operand, used inside a MEM, is a valid first argument
|
||
to CALL. This is a SYMBOL_REF or a pseudo-register, which will be
|
||
forced to lr. */
|
||
|
||
int
|
||
call_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
if (mode != VOIDmode && GET_MODE (op) != mode)
|
||
return 0;
|
||
|
||
return (GET_CODE (op) == SYMBOL_REF
|
||
|| (GET_CODE (op) == REG && REGNO (op) >= FIRST_PSEUDO_REGISTER));
|
||
}
|
||
|
||
|
||
/* Return 1 if the operand is a SYMBOL_REF for a function known to be in
|
||
this file. */
|
||
|
||
int
|
||
current_file_function_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
return (GET_CODE (op) == SYMBOL_REF
|
||
&& (SYMBOL_REF_FLAG (op)
|
||
|| op == XEXP (DECL_RTL (current_function_decl), 0)));
|
||
}
|
||
|
||
|
||
/* Return 1 if this operand is a valid input for a move insn. */
|
||
|
||
int
|
||
input_operand (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
/* Memory is always valid. */
|
||
if (memory_operand (op, mode))
|
||
return 1;
|
||
|
||
/* For floating-point, easy constants are valid. */
|
||
if (GET_MODE_CLASS (mode) == MODE_FLOAT
|
||
&& CONSTANT_P (op)
|
||
&& easy_fp_constant (op, mode))
|
||
return 1;
|
||
|
||
/* For floating-point or multi-word mode, the only remaining valid type
|
||
is a register. */
|
||
if (GET_MODE_CLASS (mode) == MODE_FLOAT
|
||
|| GET_MODE_SIZE (mode) > UNITS_PER_WORD)
|
||
return register_operand (op, mode);
|
||
|
||
/* The only cases left are integral modes one word or smaller (we
|
||
do not get called for MODE_CC values). These can be in any
|
||
register. */
|
||
if (register_operand (op, mode))
|
||
return 1;
|
||
|
||
/* For HImode and QImode, any constant is valid. */
|
||
if ((mode == HImode || mode == QImode)
|
||
&& GET_CODE (op) == CONST_INT)
|
||
return 1;
|
||
|
||
/* Otherwise, we will be doing this SET with an add, so anything valid
|
||
for an add will be valid. */
|
||
return add_operand (op, mode);
|
||
}
|
||
|
||
/* Return 1 if OP is a load multiple operation. It is known to be a
|
||
PARALLEL and the first section will be tested. */
|
||
|
||
int
|
||
load_multiple_operation (op, mode)
|
||
rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
int count = XVECLEN (op, 0);
|
||
int dest_regno;
|
||
rtx src_addr;
|
||
int i;
|
||
|
||
/* Perform a quick check so we don't blow up below. */
|
||
if (count <= 1
|
||
|| GET_CODE (XVECEXP (op, 0, 0)) != SET
|
||
|| GET_CODE (SET_DEST (XVECEXP (op, 0, 0))) != REG
|
||
|| GET_CODE (SET_SRC (XVECEXP (op, 0, 0))) != MEM)
|
||
return 0;
|
||
|
||
dest_regno = REGNO (SET_DEST (XVECEXP (op, 0, 0)));
|
||
src_addr = XEXP (SET_SRC (XVECEXP (op, 0, 0)), 0);
|
||
|
||
for (i = 1; i < count; i++)
|
||
{
|
||
rtx elt = XVECEXP (op, 0, i);
|
||
|
||
if (GET_CODE (elt) != SET
|
||
|| GET_CODE (SET_DEST (elt)) != REG
|
||
|| GET_MODE (SET_DEST (elt)) != SImode
|
||
|| REGNO (SET_DEST (elt)) != dest_regno + i
|
||
|| GET_CODE (SET_SRC (elt)) != MEM
|
||
|| GET_MODE (SET_SRC (elt)) != SImode
|
||
|| GET_CODE (XEXP (SET_SRC (elt), 0)) != PLUS
|
||
|| ! rtx_equal_p (XEXP (XEXP (SET_SRC (elt), 0), 0), src_addr)
|
||
|| GET_CODE (XEXP (XEXP (SET_SRC (elt), 0), 1)) != CONST_INT
|
||
|| INTVAL (XEXP (XEXP (SET_SRC (elt), 0), 1)) != i * 4)
|
||
return 0;
|
||
}
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Similar, but tests for store multiple. Here, the second vector element
|
||
is a CLOBBER. It will be tested later. */
|
||
|
||
int
|
||
store_multiple_operation (op, mode)
|
||
rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
int count = XVECLEN (op, 0) - 1;
|
||
int src_regno;
|
||
rtx dest_addr;
|
||
int i;
|
||
|
||
/* Perform a quick check so we don't blow up below. */
|
||
if (count <= 1
|
||
|| GET_CODE (XVECEXP (op, 0, 0)) != SET
|
||
|| GET_CODE (SET_DEST (XVECEXP (op, 0, 0))) != MEM
|
||
|| GET_CODE (SET_SRC (XVECEXP (op, 0, 0))) != REG)
|
||
return 0;
|
||
|
||
src_regno = REGNO (SET_SRC (XVECEXP (op, 0, 0)));
|
||
dest_addr = XEXP (SET_DEST (XVECEXP (op, 0, 0)), 0);
|
||
|
||
for (i = 1; i < count; i++)
|
||
{
|
||
rtx elt = XVECEXP (op, 0, i + 1);
|
||
|
||
if (GET_CODE (elt) != SET
|
||
|| GET_CODE (SET_SRC (elt)) != REG
|
||
|| GET_MODE (SET_SRC (elt)) != SImode
|
||
|| REGNO (SET_SRC (elt)) != src_regno + i
|
||
|| GET_CODE (SET_DEST (elt)) != MEM
|
||
|| GET_MODE (SET_DEST (elt)) != SImode
|
||
|| GET_CODE (XEXP (SET_DEST (elt), 0)) != PLUS
|
||
|| ! rtx_equal_p (XEXP (XEXP (SET_DEST (elt), 0), 0), dest_addr)
|
||
|| GET_CODE (XEXP (XEXP (SET_DEST (elt), 0), 1)) != CONST_INT
|
||
|| INTVAL (XEXP (XEXP (SET_DEST (elt), 0), 1)) != i * 4)
|
||
return 0;
|
||
}
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Return 1 if OP is a comparison operation that is valid for a branch insn.
|
||
We only check the opcode against the mode of the CC value here. */
|
||
|
||
int
|
||
branch_comparison_operator (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
enum rtx_code code = GET_CODE (op);
|
||
enum machine_mode cc_mode;
|
||
|
||
if (GET_RTX_CLASS (code) != '<')
|
||
return 0;
|
||
|
||
cc_mode = GET_MODE (XEXP (op, 0));
|
||
if (GET_MODE_CLASS (cc_mode) != MODE_CC)
|
||
return 0;
|
||
|
||
if ((code == GT || code == LT || code == GE || code == LE)
|
||
&& cc_mode == CCUNSmode)
|
||
return 0;
|
||
|
||
if ((code == GTU || code == LTU || code == GEU || code == LEU)
|
||
&& (cc_mode != CCUNSmode))
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Return 1 if OP is a comparison operation that is valid for an scc insn.
|
||
We check the opcode against the mode of the CC value and disallow EQ or
|
||
NE comparisons for integers. */
|
||
|
||
int
|
||
scc_comparison_operator (op, mode)
|
||
register rtx op;
|
||
enum machine_mode mode;
|
||
{
|
||
enum rtx_code code = GET_CODE (op);
|
||
enum machine_mode cc_mode;
|
||
|
||
if (GET_MODE (op) != mode && mode != VOIDmode)
|
||
return 0;
|
||
|
||
if (GET_RTX_CLASS (code) != '<')
|
||
return 0;
|
||
|
||
cc_mode = GET_MODE (XEXP (op, 0));
|
||
if (GET_MODE_CLASS (cc_mode) != MODE_CC)
|
||
return 0;
|
||
|
||
if (code == NE && cc_mode != CCFPmode)
|
||
return 0;
|
||
|
||
if ((code == GT || code == LT || code == GE || code == LE)
|
||
&& cc_mode == CCUNSmode)
|
||
return 0;
|
||
|
||
if ((code == GTU || code == LTU || code == GEU || code == LEU)
|
||
&& (cc_mode != CCUNSmode))
|
||
return 0;
|
||
|
||
if (cc_mode == CCEQmode && code != EQ && code != NE)
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Return 1 if ANDOP is a mask that has no bits on that are not in the
|
||
mask required to convert the result of a rotate insn into a shift
|
||
left insn of SHIFTOP bits. Both are known to be CONST_INT. */
|
||
|
||
int
|
||
includes_lshift_p (shiftop, andop)
|
||
register rtx shiftop;
|
||
register rtx andop;
|
||
{
|
||
int shift_mask = (~0 << INTVAL (shiftop));
|
||
|
||
return (INTVAL (andop) & ~shift_mask) == 0;
|
||
}
|
||
|
||
/* Similar, but for right shift. */
|
||
|
||
int
|
||
includes_rshift_p (shiftop, andop)
|
||
register rtx shiftop;
|
||
register rtx andop;
|
||
{
|
||
unsigned shift_mask = ~0;
|
||
|
||
shift_mask >>= INTVAL (shiftop);
|
||
|
||
return (INTVAL (andop) & ~ shift_mask) == 0;
|
||
}
|
||
|
||
/* Return the register class of a scratch register needed to copy IN into
|
||
or out of a register in CLASS in MODE. If it can be done directly,
|
||
NO_REGS is returned. */
|
||
|
||
enum reg_class
|
||
secondary_reload_class (class, mode, in)
|
||
enum reg_class class;
|
||
enum machine_mode mode;
|
||
rtx in;
|
||
{
|
||
int regno = true_regnum (in);
|
||
|
||
if (regno >= FIRST_PSEUDO_REGISTER)
|
||
regno = -1;
|
||
|
||
/* We can place anything into GENERAL_REGS and can put GENERAL_REGS
|
||
into anything. */
|
||
if (class == GENERAL_REGS || class == BASE_REGS
|
||
|| (regno >= 0 && INT_REGNO_P (regno)))
|
||
return NO_REGS;
|
||
|
||
/* Constants, memory, and FP registers can go into FP registers. */
|
||
if ((regno == -1 || FP_REGNO_P (regno))
|
||
&& (class == FLOAT_REGS || class == NON_SPECIAL_REGS))
|
||
return NO_REGS;
|
||
|
||
/* We can copy among the CR registers. */
|
||
if ((class == CR_REGS || class == CR0_REGS)
|
||
&& regno >= 0 && CR_REGNO_P (regno))
|
||
return NO_REGS;
|
||
|
||
/* Otherwise, we need GENERAL_REGS. */
|
||
return GENERAL_REGS;
|
||
}
|
||
|
||
/* Given a comparison operation, return the bit number in CCR to test. We
|
||
know this is a valid comparison.
|
||
|
||
SCC_P is 1 if this is for an scc. That means that %D will have been
|
||
used instead of %C, so the bits will be in different places.
|
||
|
||
Return -1 if OP isn't a valid comparison for some reason. */
|
||
|
||
int
|
||
ccr_bit (op, scc_p)
|
||
register rtx op;
|
||
int scc_p;
|
||
{
|
||
enum rtx_code code = GET_CODE (op);
|
||
enum machine_mode cc_mode;
|
||
int cc_regnum;
|
||
int base_bit;
|
||
|
||
if (GET_RTX_CLASS (code) != '<')
|
||
return -1;
|
||
|
||
cc_mode = GET_MODE (XEXP (op, 0));
|
||
cc_regnum = REGNO (XEXP (op, 0));
|
||
base_bit = 4 * (cc_regnum - 68);
|
||
|
||
/* In CCEQmode cases we have made sure that the result is always in the
|
||
third bit of the CR field. */
|
||
|
||
if (cc_mode == CCEQmode)
|
||
return base_bit + 3;
|
||
|
||
switch (code)
|
||
{
|
||
case NE:
|
||
return scc_p ? base_bit + 3 : base_bit + 2;
|
||
case EQ:
|
||
return base_bit + 2;
|
||
case GT: case GTU:
|
||
return base_bit + 1;
|
||
case LT: case LTU:
|
||
return base_bit;
|
||
|
||
case GE: case GEU:
|
||
/* If floating-point, we will have done a cror to put the bit in the
|
||
unordered position. So test that bit. For integer, this is ! LT
|
||
unless this is an scc insn. */
|
||
return cc_mode == CCFPmode || scc_p ? base_bit + 3 : base_bit;
|
||
|
||
case LE: case LEU:
|
||
return cc_mode == CCFPmode || scc_p ? base_bit + 3 : base_bit + 1;
|
||
|
||
default:
|
||
abort ();
|
||
}
|
||
}
|
||
|
||
/* Print an operand. Recognize special options, documented below. */
|
||
|
||
void
|
||
print_operand (file, x, code)
|
||
FILE *file;
|
||
rtx x;
|
||
char code;
|
||
{
|
||
int i;
|
||
int val;
|
||
|
||
/* These macros test for integers and extract the low-order bits. */
|
||
#define INT_P(X) \
|
||
((GET_CODE (X) == CONST_INT || GET_CODE (X) == CONST_DOUBLE) \
|
||
&& GET_MODE (X) == VOIDmode)
|
||
|
||
#define INT_LOWPART(X) \
|
||
(GET_CODE (X) == CONST_INT ? INTVAL (X) : CONST_DOUBLE_LOW (X))
|
||
|
||
switch (code)
|
||
{
|
||
case '.':
|
||
/* Write out the bit number for "cror" after a call. This differs
|
||
between AIX 3.2 and earlier versions. */
|
||
fprintf (file, "%d", RS6000_CROR_BIT_NUMBER);
|
||
return;
|
||
|
||
case 'A':
|
||
/* If X is a constant integer whose low-order 5 bits are zero,
|
||
write 'l'. Otherwise, write 'r'. This is a kludge to fix a bug
|
||
in the AIX assembler where "sri" with a zero shift count
|
||
write a trash instruction. */
|
||
if (GET_CODE (x) == CONST_INT && (INTVAL (x) & 31) == 0)
|
||
putc ('l', file);
|
||
else
|
||
putc ('r', file);
|
||
return;
|
||
|
||
case 'b':
|
||
/* Low-order 16 bits of constant, unsigned. */
|
||
if (! INT_P (x))
|
||
output_operand_lossage ("invalid %%b value");
|
||
|
||
fprintf (file, "%d", INT_LOWPART (x) & 0xffff);
|
||
return;
|
||
|
||
case 'C':
|
||
/* This is an optional cror needed for LE or GE floating-point
|
||
comparisons. Otherwise write nothing. */
|
||
if ((GET_CODE (x) == LE || GET_CODE (x) == GE)
|
||
&& GET_MODE (XEXP (x, 0)) == CCFPmode)
|
||
{
|
||
int base_bit = 4 * (REGNO (XEXP (x, 0)) - 68);
|
||
|
||
fprintf (file, "cror %d,%d,%d\n\t", base_bit + 3,
|
||
base_bit + 2, base_bit + (GET_CODE (x) == GE));
|
||
}
|
||
return;
|
||
|
||
case 'D':
|
||
/* Similar, except that this is for an scc, so we must be able to
|
||
encode the test in a single bit that is one. We do the above
|
||
for any LE, GE, GEU, or LEU and invert the bit for NE. */
|
||
if (GET_CODE (x) == LE || GET_CODE (x) == GE
|
||
|| GET_CODE (x) == LEU || GET_CODE (x) == GEU)
|
||
{
|
||
int base_bit = 4 * (REGNO (XEXP (x, 0)) - 68);
|
||
|
||
fprintf (file, "cror %d,%d,%d\n\t", base_bit + 3,
|
||
base_bit + 2,
|
||
base_bit + (GET_CODE (x) == GE || GET_CODE (x) == GEU));
|
||
}
|
||
|
||
else if (GET_CODE (x) == NE)
|
||
{
|
||
int base_bit = 4 * (REGNO (XEXP (x, 0)) - 68);
|
||
|
||
fprintf (file, "crnor %d,%d,%d\n\t", base_bit + 3,
|
||
base_bit + 2, base_bit + 2);
|
||
}
|
||
return;
|
||
|
||
case 'E':
|
||
/* X is a CR register. Print the number of the third bit of the CR */
|
||
if (GET_CODE (x) != REG || ! CR_REGNO_P (REGNO (x)))
|
||
output_operand_lossage ("invalid %%E value");
|
||
|
||
fprintf(file, "%d", 4 * (REGNO (x) - 68) + 3);
|
||
break;
|
||
|
||
case 'f':
|
||
/* X is a CR register. Print the shift count needed to move it
|
||
to the high-order four bits. */
|
||
if (GET_CODE (x) != REG || ! CR_REGNO_P (REGNO (x)))
|
||
output_operand_lossage ("invalid %%f value");
|
||
else
|
||
fprintf (file, "%d", 4 * (REGNO (x) - 68));
|
||
return;
|
||
|
||
case 'F':
|
||
/* Similar, but print the count for the rotate in the opposite
|
||
direction. */
|
||
if (GET_CODE (x) != REG || ! CR_REGNO_P (REGNO (x)))
|
||
output_operand_lossage ("invalid %%F value");
|
||
else
|
||
fprintf (file, "%d", 32 - 4 * (REGNO (x) - 68));
|
||
return;
|
||
|
||
case 'G':
|
||
/* X is a constant integer. If it is negative, print "m",
|
||
otherwise print "z". This is to make a aze or ame insn. */
|
||
if (GET_CODE (x) != CONST_INT)
|
||
output_operand_lossage ("invalid %%G value");
|
||
else if (INTVAL (x) >= 0)
|
||
putc ('z', file);
|
||
else
|
||
putc ('m', file);
|
||
return;
|
||
|
||
case 'h':
|
||
/* If constant, output low-order five bits. Otherwise,
|
||
write normally. */
|
||
if (INT_P (x))
|
||
fprintf (file, "%d", INT_LOWPART (x) & 31);
|
||
else
|
||
print_operand (file, x, 0);
|
||
return;
|
||
|
||
case 'H':
|
||
/* X must be a constant. Output the low order 5 bits plus 24. */
|
||
if (! INT_P (x))
|
||
output_operand_lossage ("invalid %%H value");
|
||
|
||
fprintf (file, "%d", (INT_LOWPART (x) + 24) & 31);
|
||
return;
|
||
|
||
case 'I':
|
||
/* Print `i' if this is a constant, else nothing. */
|
||
if (INT_P (x))
|
||
putc ('i', file);
|
||
return;
|
||
|
||
case 'j':
|
||
/* Write the bit number in CCR for jump. */
|
||
i = ccr_bit (x, 0);
|
||
if (i == -1)
|
||
output_operand_lossage ("invalid %%j code");
|
||
else
|
||
fprintf (file, "%d", i);
|
||
return;
|
||
|
||
case 'J':
|
||
/* Similar, but add one for shift count in rlinm for scc and pass
|
||
scc flag to `ccr_bit'. */
|
||
i = ccr_bit (x, 1);
|
||
if (i == -1)
|
||
output_operand_lossage ("invalid %%J code");
|
||
else
|
||
/* If we want bit 31, write a shift count of zero, not 32. */
|
||
fprintf (file, "%d", i == 31 ? 0 : i + 1);
|
||
return;
|
||
|
||
case 'k':
|
||
/* X must be a constant. Write the 1's complement of the
|
||
constant. */
|
||
if (! INT_P (x))
|
||
output_operand_lossage ("invalid %%k value");
|
||
|
||
fprintf (file, "%d", ~ INT_LOWPART (x));
|
||
return;
|
||
|
||
case 'L':
|
||
/* Write second word of DImode or DFmode reference. Works on register
|
||
or non-indexed memory only. */
|
||
if (GET_CODE (x) == REG)
|
||
fprintf (file, "%d", REGNO (x) + 1);
|
||
else if (GET_CODE (x) == MEM)
|
||
{
|
||
/* Handle possible auto-increment. Since it is pre-increment and
|
||
we have already done it, we can just use an offset of four. */
|
||
if (GET_CODE (XEXP (x, 0)) == PRE_INC
|
||
|| GET_CODE (XEXP (x, 0)) == PRE_DEC)
|
||
output_address (plus_constant (XEXP (XEXP (x, 0), 0), 4));
|
||
else
|
||
output_address (plus_constant (XEXP (x, 0), 4));
|
||
}
|
||
return;
|
||
|
||
case 'm':
|
||
/* MB value for a mask operand. */
|
||
if (! mask_operand (x, VOIDmode))
|
||
output_operand_lossage ("invalid %%m value");
|
||
|
||
val = INT_LOWPART (x);
|
||
|
||
/* If the high bit is set and the low bit is not, the value is zero.
|
||
If the high bit is zero, the value is the first 1 bit we find from
|
||
the left. */
|
||
if (val < 0 && (val & 1) == 0)
|
||
{
|
||
fprintf (file, "0");
|
||
return;
|
||
}
|
||
else if (val >= 0)
|
||
{
|
||
for (i = 1; i < 32; i++)
|
||
if ((val <<= 1) < 0)
|
||
break;
|
||
fprintf (file, "%d", i);
|
||
return;
|
||
}
|
||
|
||
/* Otherwise, look for the first 0 bit from the right. The result is its
|
||
number plus 1. We know the low-order bit is one. */
|
||
for (i = 0; i < 32; i++)
|
||
if (((val >>= 1) & 1) == 0)
|
||
break;
|
||
|
||
/* If we ended in ...01, I would be 0. The correct value is 31, so
|
||
we want 31 - i. */
|
||
fprintf (file, "%d", 31 - i);
|
||
return;
|
||
|
||
case 'M':
|
||
/* ME value for a mask operand. */
|
||
if (! mask_operand (x, VOIDmode))
|
||
output_operand_lossage ("invalid %%m value");
|
||
|
||
val = INT_LOWPART (x);
|
||
|
||
/* If the low bit is set and the high bit is not, the value is 31.
|
||
If the low bit is zero, the value is the first 1 bit we find from
|
||
the right. */
|
||
if ((val & 1) && val >= 0)
|
||
{
|
||
fputs ("31", file);
|
||
return;
|
||
}
|
||
else if ((val & 1) == 0)
|
||
{
|
||
for (i = 0; i < 32; i++)
|
||
if ((val >>= 1) & 1)
|
||
break;
|
||
|
||
/* If we had ....10, I would be 0. The result should be
|
||
30, so we need 30 - i. */
|
||
fprintf (file, "%d", 30 - i);
|
||
return;
|
||
}
|
||
|
||
/* Otherwise, look for the first 0 bit from the left. The result is its
|
||
number minus 1. We know the high-order bit is one. */
|
||
for (i = 0; i < 32; i++)
|
||
if ((val <<= 1) >= 0)
|
||
break;
|
||
|
||
fprintf (file, "%d", i);
|
||
return;
|
||
|
||
case 'N':
|
||
/* Write the number of elements in the vector times 4. */
|
||
if (GET_CODE (x) != PARALLEL)
|
||
output_operand_lossage ("invalid %%N value");
|
||
|
||
fprintf (file, "%d", XVECLEN (x, 0) * 4);
|
||
return;
|
||
|
||
case 'O':
|
||
/* Similar, but subtract 1 first. */
|
||
if (GET_CODE (x) != PARALLEL)
|
||
output_operand_lossage ("invalid %%N value");
|
||
|
||
fprintf (file, "%d", (XVECLEN (x, 0) - 1) * 4);
|
||
return;
|
||
|
||
case 'p':
|
||
/* X is a CONST_INT that is a power of two. Output the logarithm. */
|
||
if (! INT_P (x)
|
||
|| (i = exact_log2 (INT_LOWPART (x))) < 0)
|
||
output_operand_lossage ("invalid %%p value");
|
||
|
||
fprintf (file, "%d", i);
|
||
return;
|
||
|
||
case 'P':
|
||
/* The operand must be an indirect memory reference. The result
|
||
is the register number. */
|
||
if (GET_CODE (x) != MEM || GET_CODE (XEXP (x, 0)) != REG
|
||
|| REGNO (XEXP (x, 0)) >= 32)
|
||
output_operand_lossage ("invalid %%P value");
|
||
|
||
fprintf (file, "%d", REGNO (XEXP (x, 0)));
|
||
return;
|
||
|
||
case 'R':
|
||
/* X is a CR register. Print the mask for `mtcrf'. */
|
||
if (GET_CODE (x) != REG || ! CR_REGNO_P (REGNO (x)))
|
||
output_operand_lossage ("invalid %%R value");
|
||
else
|
||
fprintf (file, "%d", 128 >> (REGNO (x) - 68));
|
||
return;
|
||
|
||
case 's':
|
||
/* Low 5 bits of 32 - value */
|
||
if (! INT_P (x))
|
||
output_operand_lossage ("invalid %%s value");
|
||
|
||
fprintf (file, "%d", (32 - INT_LOWPART (x)) & 31);
|
||
return;
|
||
|
||
case 'S':
|
||
/* Low 5 bits of 31 - value */
|
||
if (! INT_P (x))
|
||
output_operand_lossage ("invalid %%S value");
|
||
|
||
fprintf (file, "%d", (31 - INT_LOWPART (x)) & 31);
|
||
return;
|
||
|
||
case 't':
|
||
/* Write 12 if this jump operation will branch if true, 4 otherwise.
|
||
All floating-point operations except NE branch true and integer
|
||
EQ, LT, GT, LTU and GTU also branch true. */
|
||
if (GET_RTX_CLASS (GET_CODE (x)) != '<')
|
||
output_operand_lossage ("invalid %%t value");
|
||
|
||
else if ((GET_MODE (XEXP (x, 0)) == CCFPmode
|
||
&& GET_CODE (x) != NE)
|
||
|| GET_CODE (x) == EQ
|
||
|| GET_CODE (x) == LT || GET_CODE (x) == GT
|
||
|| GET_CODE (x) == LTU || GET_CODE (x) == GTU)
|
||
fputs ("12", file);
|
||
else
|
||
putc ('4', file);
|
||
return;
|
||
|
||
case 'T':
|
||
/* Opposite of 't': write 4 if this jump operation will branch if true,
|
||
12 otherwise. */
|
||
if (GET_RTX_CLASS (GET_CODE (x)) != '<')
|
||
output_operand_lossage ("invalid %%t value");
|
||
|
||
else if ((GET_MODE (XEXP (x, 0)) == CCFPmode
|
||
&& GET_CODE (x) != NE)
|
||
|| GET_CODE (x) == EQ
|
||
|| GET_CODE (x) == LT || GET_CODE (x) == GT
|
||
|| GET_CODE (x) == LTU || GET_CODE (x) == GTU)
|
||
putc ('4', file);
|
||
else
|
||
fputs ("12", file);
|
||
return;
|
||
|
||
case 'u':
|
||
/* High-order 16 bits of constant. */
|
||
if (! INT_P (x))
|
||
output_operand_lossage ("invalid %%u value");
|
||
|
||
fprintf (file, "0x%x", (INT_LOWPART (x) >> 16) & 0xffff);
|
||
return;
|
||
|
||
case 'U':
|
||
/* Print `u' if this has an auto-increment or auto-decrement. */
|
||
if (GET_CODE (x) == MEM
|
||
&& (GET_CODE (XEXP (x, 0)) == PRE_INC
|
||
|| GET_CODE (XEXP (x, 0)) == PRE_DEC))
|
||
putc ('u', file);
|
||
return;
|
||
|
||
case 'w':
|
||
/* If constant, low-order 16 bits of constant, signed. Otherwise, write
|
||
normally. */
|
||
if (INT_P (x))
|
||
fprintf (file, "%d",
|
||
(INT_LOWPART (x) & 0xffff) - 2 * (INT_LOWPART (x) & 0x8000));
|
||
else
|
||
print_operand (file, x, 0);
|
||
return;
|
||
|
||
case 'W':
|
||
/* If constant, low-order 16 bits of constant, unsigned.
|
||
Otherwise, write normally. */
|
||
if (INT_P (x))
|
||
fprintf (file, "%d", INT_LOWPART (x) & 0xffff);
|
||
else
|
||
print_operand (file, x, 0);
|
||
return;
|
||
|
||
case 'X':
|
||
if (GET_CODE (x) == MEM
|
||
&& LEGITIMATE_INDEXED_ADDRESS_P (XEXP (x, 0)))
|
||
putc ('x', file);
|
||
return;
|
||
|
||
case 'Y':
|
||
/* Like 'L', for third word of TImode */
|
||
if (GET_CODE (x) == REG)
|
||
fprintf (file, "%d", REGNO (x) + 2);
|
||
else if (GET_CODE (x) == MEM)
|
||
{
|
||
if (GET_CODE (XEXP (x, 0)) == PRE_INC
|
||
|| GET_CODE (XEXP (x, 0)) == PRE_DEC)
|
||
output_address (plus_constant (XEXP (XEXP (x, 0), 0), 8));
|
||
else
|
||
output_address (plus_constant (XEXP (x, 0), 8));
|
||
}
|
||
return;
|
||
|
||
case 'z':
|
||
/* X is a SYMBOL_REF. Write out the name preceded by a
|
||
period and without any trailing data in brackets. Used for function
|
||
names. */
|
||
if (GET_CODE (x) != SYMBOL_REF)
|
||
abort ();
|
||
|
||
putc ('.', file);
|
||
RS6000_OUTPUT_BASENAME (file, XSTR (x, 0));
|
||
return;
|
||
|
||
case 'Z':
|
||
/* Like 'L', for last word of TImode. */
|
||
if (GET_CODE (x) == REG)
|
||
fprintf (file, "%d", REGNO (x) + 3);
|
||
else if (GET_CODE (x) == MEM)
|
||
{
|
||
if (GET_CODE (XEXP (x, 0)) == PRE_INC
|
||
|| GET_CODE (XEXP (x, 0)) == PRE_DEC)
|
||
output_address (plus_constant (XEXP (XEXP (x, 0), 0), 12));
|
||
else
|
||
output_address (plus_constant (XEXP (x, 0), 12));
|
||
}
|
||
return;
|
||
|
||
case 0:
|
||
if (GET_CODE (x) == REG)
|
||
fprintf (file, "%s", reg_names[REGNO (x)]);
|
||
else if (GET_CODE (x) == MEM)
|
||
{
|
||
/* We need to handle PRE_INC and PRE_DEC here, since we need to
|
||
know the width from the mode. */
|
||
if (GET_CODE (XEXP (x, 0)) == PRE_INC)
|
||
fprintf (file, "%d(%d)", GET_MODE_SIZE (GET_MODE (x)),
|
||
REGNO (XEXP (XEXP (x, 0), 0)));
|
||
else if (GET_CODE (XEXP (x, 0)) == PRE_DEC)
|
||
fprintf (file, "%d(%d)", - GET_MODE_SIZE (GET_MODE (x)),
|
||
REGNO (XEXP (XEXP (x, 0), 0)));
|
||
else
|
||
output_address (XEXP (x, 0));
|
||
}
|
||
else
|
||
output_addr_const (file, x);
|
||
break;
|
||
|
||
default:
|
||
output_operand_lossage ("invalid %%xn code");
|
||
}
|
||
}
|
||
|
||
/* Print the address of an operand. */
|
||
|
||
void
|
||
print_operand_address (file, x)
|
||
FILE *file;
|
||
register rtx x;
|
||
{
|
||
if (GET_CODE (x) == REG)
|
||
fprintf (file, "0(%d)", REGNO (x));
|
||
else if (GET_CODE (x) == SYMBOL_REF || GET_CODE (x) == CONST)
|
||
{
|
||
output_addr_const (file, x);
|
||
/* When TARGET_MINIMAL_TOC, use the indirected toc table pointer instead
|
||
of the toc pointer. */
|
||
if (TARGET_MINIMAL_TOC)
|
||
fprintf (file, "(30)");
|
||
else
|
||
fprintf (file, "(2)");
|
||
}
|
||
else if (GET_CODE (x) == PLUS && GET_CODE (XEXP (x, 1)) == REG)
|
||
{
|
||
if (REGNO (XEXP (x, 0)) == 0)
|
||
fprintf (file, "%d,%d", REGNO (XEXP (x, 1)), REGNO (XEXP (x, 0)));
|
||
else
|
||
fprintf (file, "%d,%d", REGNO (XEXP (x, 0)), REGNO (XEXP (x, 1)));
|
||
}
|
||
else if (GET_CODE (x) == PLUS && GET_CODE (XEXP (x, 1)) == CONST_INT)
|
||
fprintf (file, "%d(%d)", INTVAL (XEXP (x, 1)), REGNO (XEXP (x, 0)));
|
||
else
|
||
abort ();
|
||
}
|
||
|
||
/* This page contains routines that are used to determine what the function
|
||
prologue and epilogue code will do and write them out. */
|
||
|
||
/* Return the first fixed-point register that is required to be saved. 32 if
|
||
none. */
|
||
|
||
int
|
||
first_reg_to_save ()
|
||
{
|
||
int first_reg;
|
||
|
||
/* Find lowest numbered live register. */
|
||
for (first_reg = 13; first_reg <= 31; first_reg++)
|
||
if (regs_ever_live[first_reg])
|
||
break;
|
||
|
||
/* If profiling, then we must save/restore every register that contains
|
||
a parameter before/after the .mcount call. Use registers from 30 down
|
||
to 23 to do this. Don't use the frame pointer in reg 31.
|
||
|
||
For now, save enough room for all of the parameter registers. */
|
||
if (profile_flag)
|
||
if (first_reg > 23)
|
||
first_reg = 23;
|
||
|
||
return first_reg;
|
||
}
|
||
|
||
/* Similar, for FP regs. */
|
||
|
||
int
|
||
first_fp_reg_to_save ()
|
||
{
|
||
int first_reg;
|
||
|
||
/* Find lowest numbered live register. */
|
||
for (first_reg = 14 + 32; first_reg <= 63; first_reg++)
|
||
if (regs_ever_live[first_reg])
|
||
break;
|
||
|
||
return first_reg;
|
||
}
|
||
|
||
/* Return 1 if we need to save CR. */
|
||
|
||
int
|
||
must_save_cr ()
|
||
{
|
||
return regs_ever_live[70] || regs_ever_live[71] || regs_ever_live[72];
|
||
}
|
||
|
||
/* Compute the size of the save area in the stack, including the space for
|
||
the fixed area. */
|
||
|
||
int
|
||
rs6000_sa_size ()
|
||
{
|
||
int size;
|
||
|
||
/* We have the six fixed words, plus the size of the register save
|
||
areas, rounded to a double-word. */
|
||
size = 6 + (32 - first_reg_to_save ()) + (64 - first_fp_reg_to_save ()) * 2;
|
||
if (size & 1)
|
||
size++;
|
||
|
||
return size * 4;
|
||
}
|
||
|
||
/* Return non-zero if this function makes calls. */
|
||
|
||
int
|
||
rs6000_makes_calls ()
|
||
{
|
||
rtx insn;
|
||
|
||
/* If we are profiling, we will be making a call to mcount. */
|
||
if (profile_flag)
|
||
return 1;
|
||
|
||
for (insn = get_insns (); insn; insn = next_insn (insn))
|
||
if (GET_CODE (insn) == CALL_INSN)
|
||
return 1;
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Return non-zero if this function needs to push space on the stack. */
|
||
|
||
int
|
||
rs6000_pushes_stack ()
|
||
{
|
||
int total_size = (rs6000_sa_size () + get_frame_size ()
|
||
+ current_function_outgoing_args_size);
|
||
|
||
/* We need to push the stack if a frame pointer is needed (because the
|
||
stack might be dynamically adjusted), if we are debugging, if the
|
||
total stack size is more than 220 bytes, or if we make calls. */
|
||
|
||
return (frame_pointer_needed || write_symbols != NO_DEBUG
|
||
|| total_size > 220
|
||
|| rs6000_makes_calls ());
|
||
}
|
||
|
||
/* Write function prologue. */
|
||
|
||
void
|
||
output_prolog (file, size)
|
||
FILE *file;
|
||
int size;
|
||
{
|
||
int first_reg = first_reg_to_save ();
|
||
int must_push = rs6000_pushes_stack ();
|
||
int first_fp_reg = first_fp_reg_to_save ();
|
||
int basic_size = rs6000_sa_size ();
|
||
int total_size = (basic_size + size + current_function_outgoing_args_size);
|
||
|
||
/* Round size to multiple of 8 bytes. */
|
||
total_size = (total_size + 7) & ~7;
|
||
|
||
/* Write .extern for any function we will call to save and restore fp
|
||
values. */
|
||
if (first_fp_reg < 62)
|
||
fprintf (file, "\t.extern ._savef%d\n\t.extern ._restf%d\n",
|
||
first_fp_reg - 32, first_fp_reg - 32);
|
||
|
||
/* Write .extern for truncation routines, if needed. */
|
||
if (rs6000_trunc_used && ! trunc_defined)
|
||
{
|
||
fprintf (file, "\t.extern .itrunc\n\t.extern .uitrunc\n");
|
||
trunc_defined = 1;
|
||
}
|
||
|
||
/* If we have to call a function to save fpr's, or if we are doing profiling,
|
||
then we will be using LR. */
|
||
if (first_fp_reg < 62 || profile_flag)
|
||
regs_ever_live[65] = 1;
|
||
|
||
/* If we use the link register, get it into r0. */
|
||
if (regs_ever_live[65])
|
||
fprintf (file, "\tmflr 0\n");
|
||
|
||
/* If we need to save CR, put it into r12. */
|
||
if (must_save_cr ())
|
||
fprintf (file, "\tmfcr 12\n");
|
||
|
||
/* Do any required saving of fpr's. If only one or two to save, do it
|
||
ourself. Otherwise, call function. Note that since they are statically
|
||
linked, we do not need a nop following them. */
|
||
if (first_fp_reg == 62)
|
||
fprintf (file, "\tstfd 30,-16(1)\n\tstfd 31,-8(1)\n");
|
||
else if (first_fp_reg == 63)
|
||
fprintf (file, "\tstfd 31,-8(1)\n");
|
||
else if (first_fp_reg != 64)
|
||
fprintf (file, "\tbl ._savef%d\n", first_fp_reg - 32);
|
||
|
||
/* Now save gpr's. */
|
||
if (first_reg == 31)
|
||
fprintf (file, "\tst 31,%d(1)\n", -4 - (64 - first_fp_reg) * 8);
|
||
else if (first_reg != 32)
|
||
fprintf (file, "\tstm %d,%d(1)\n", first_reg,
|
||
- (32 - first_reg) * 4 - (64 - first_fp_reg) * 8);
|
||
|
||
/* Save lr if we used it. */
|
||
if (regs_ever_live[65])
|
||
fprintf (file, "\tst 0,8(1)\n");
|
||
|
||
/* Save CR if we use any that must be preserved. */
|
||
if (must_save_cr ())
|
||
fprintf (file, "\tst 12,4(1)\n");
|
||
|
||
/* Update stack and set back pointer. */
|
||
if (must_push)
|
||
{
|
||
if (total_size < 32767)
|
||
fprintf (file, "\tstu 1,%d(1)\n", - total_size);
|
||
else
|
||
{
|
||
fprintf (file, "\tcau 0,0,%d\n\toril 0,0,%d\n",
|
||
(total_size >> 16) & 0xffff, total_size & 0xffff);
|
||
fprintf (file, "\tsf 12,0,1\n\tst 1,0(12)\n\toril 1,12,0\n");
|
||
}
|
||
}
|
||
|
||
/* Set frame pointer, if needed. */
|
||
if (frame_pointer_needed)
|
||
fprintf (file, "\toril 31,1,0\n");
|
||
|
||
/* If TARGET_MINIMAL_TOC, and the constant pool is needed, then load the
|
||
TOC_TABLE address into register 30. */
|
||
if (TARGET_MINIMAL_TOC && get_pool_size () != 0)
|
||
fprintf (file, "\tl 30,LCTOC..0(2)\n");
|
||
}
|
||
|
||
/* Write function epilogue. */
|
||
|
||
void
|
||
output_epilog (file, size)
|
||
FILE *file;
|
||
int size;
|
||
{
|
||
int first_reg = first_reg_to_save ();
|
||
int must_push = rs6000_pushes_stack ();
|
||
int first_fp_reg = first_fp_reg_to_save ();
|
||
int basic_size = rs6000_sa_size ();
|
||
int total_size = (basic_size + size + current_function_outgoing_args_size);
|
||
rtx insn = get_last_insn ();
|
||
|
||
/* Round size to multiple of 8 bytes. */
|
||
total_size = (total_size + 7) & ~7;
|
||
|
||
/* If the last insn was a BARRIER, we don't have to write anything except
|
||
the trace table. */
|
||
if (GET_CODE (insn) == NOTE)
|
||
insn = prev_nonnote_insn (insn);
|
||
if (insn == 0 || GET_CODE (insn) != BARRIER)
|
||
{
|
||
/* If we have a frame pointer, a call to alloca, or a large stack
|
||
frame, restore the old stack pointer using the backchain. Otherwise,
|
||
we know what size to update it with. */
|
||
if (frame_pointer_needed || current_function_calls_alloca
|
||
|| total_size > 32767)
|
||
fprintf (file, "\tl 1,0(1)\n");
|
||
else if (must_push)
|
||
fprintf (file, "\tai 1,1,%d\n", total_size);
|
||
|
||
/* Get the old lr if we saved it. */
|
||
if (regs_ever_live[65])
|
||
fprintf (file, "\tl 0,8(1)\n");
|
||
|
||
/* Get the old cr if we saved it. */
|
||
if (must_save_cr ())
|
||
fprintf (file, "\tl 12,4(1)\n");
|
||
|
||
/* Set LR here to try to overlap restores below. */
|
||
if (regs_ever_live[65])
|
||
fprintf (file, "\tmtlr 0\n");
|
||
|
||
/* Restore gpr's. */
|
||
if (first_reg == 31)
|
||
fprintf (file, "\tl 31,%d(1)\n", -4 - (64 - first_fp_reg) * 8);
|
||
else if (first_reg != 32)
|
||
fprintf (file, "\tlm %d,%d(1)\n", first_reg,
|
||
- (32 - first_reg) * 4 - (64 - first_fp_reg) * 8);
|
||
|
||
/* Restore fpr's if we can do it without calling a function. */
|
||
if (first_fp_reg == 62)
|
||
fprintf (file, "\tlfd 30,-16(1)\n\tlfd 31,-8(1)\n");
|
||
else if (first_fp_reg == 63)
|
||
fprintf (file, "\tlfd 31,-8(1)\n");
|
||
|
||
/* If we saved cr, restore it here. Just those of cr2, cr3, and cr4
|
||
that were used. */
|
||
if (must_save_cr ())
|
||
fprintf (file, "\tmtcrf %d,12\n",
|
||
(regs_ever_live[70] != 0) * 0x20
|
||
+ (regs_ever_live[71] != 0) * 0x10
|
||
+ (regs_ever_live[72] != 0) * 0x8);
|
||
|
||
/* If we have to restore more than two FP registers, branch to the
|
||
restore function. It will return to our caller. */
|
||
if (first_fp_reg < 62)
|
||
fprintf (file, "\tb ._restf%d\n", first_fp_reg - 32);
|
||
else
|
||
fprintf (file, "\tbr\n");
|
||
}
|
||
|
||
/* Output a traceback table here. See /usr/include/sys/debug.h for info
|
||
on its format. */
|
||
{
|
||
char *fname = XSTR (XEXP (DECL_RTL (current_function_decl), 0), 0);
|
||
int fixed_parms, float_parms, parm_info;
|
||
int i;
|
||
|
||
/* Need label immediately before tbtab, so we can compute its offset
|
||
from the function start. */
|
||
if (*fname == '*')
|
||
++fname;
|
||
fprintf (file, "LT..");
|
||
ASM_OUTPUT_LABEL (file, fname);
|
||
|
||
/* The .tbtab pseudo-op can only be used for the first eight
|
||
expressions, since it can't handle the possibly variable length
|
||
fields that follow. However, if you omit the optional fields,
|
||
the assembler outputs zeros for all optional fields anyways, giving each
|
||
variable length field is minimum length (as defined in sys/debug.h).
|
||
Thus we can not use the .tbtab pseudo-op at all. */
|
||
|
||
/* An all-zero word flags the start of the tbtab, for debuggers that have
|
||
to find it by searching forward from the entry point or from the
|
||
current pc. */
|
||
fprintf (file, "\t.long 0\n");
|
||
|
||
/* Tbtab format type. Use format type 0. */
|
||
fprintf (file, "\t.byte 0,");
|
||
|
||
/* Language type. Unfortunately, there doesn't seem to be any official way
|
||
to get this info, so we use language_string. C is 0. C++ is 9.
|
||
No number defined for Obj-C, so use the value for C for now. */
|
||
if (! strcmp (language_string, "GNU C")
|
||
|| ! strcmp (language_string, "GNU Obj-C"))
|
||
i = 0;
|
||
else if (! strcmp (language_string, "GNU F77"))
|
||
i = 1;
|
||
else if (! strcmp (language_string, "GNU Ada"))
|
||
i = 3;
|
||
else if (! strcmp (language_string, "GNU PASCAL"))
|
||
i = 2;
|
||
else if (! strcmp (language_string, "GNU C++"))
|
||
i = 9;
|
||
else
|
||
abort ();
|
||
fprintf (file, "%d,", i);
|
||
|
||
/* 8 single bit fields: global linkage (not set for C extern linkage,
|
||
apparently a PL/I convention?), out-of-line epilogue/prologue, offset
|
||
from start of procedure stored in tbtab, internal function, function
|
||
has controlled storage, function has no toc, function uses fp,
|
||
function logs/aborts fp operations. */
|
||
/* Assume that fp operations are used if any fp reg must be saved. */
|
||
fprintf (file, "%d,", (1 << 5) | ((first_fp_reg != 64) << 1));
|
||
|
||
/* 6 bitfields: function is interrupt handler, name present in proc table,
|
||
function calls alloca, on condition directives (controls stack walks,
|
||
3 bits), saves condition reg, saves link reg. */
|
||
/* The `function calls alloca' bit seems to be set whenever reg 31 is
|
||
set up as a frame pointer, even when there is no alloca call. */
|
||
fprintf (file, "%d,",
|
||
((1 << 6) | (frame_pointer_needed << 5)
|
||
| (must_save_cr () << 1) | (regs_ever_live[65])));
|
||
|
||
/* 3 bitfields: saves backchain, spare bit, number of fpr saved
|
||
(6 bits). */
|
||
fprintf (file, "%d,",
|
||
(must_push << 7) | (64 - first_fp_reg_to_save ()));
|
||
|
||
/* 2 bitfields: spare bits (2 bits), number of gpr saved (6 bits). */
|
||
fprintf (file, "%d,", (32 - first_reg_to_save ()));
|
||
|
||
{
|
||
/* Compute the parameter info from the function decl argument list. */
|
||
tree decl;
|
||
int next_parm_info_bit;
|
||
|
||
next_parm_info_bit = 31;
|
||
parm_info = 0;
|
||
fixed_parms = 0;
|
||
float_parms = 0;
|
||
|
||
for (decl = DECL_ARGUMENTS (current_function_decl);
|
||
decl; decl = TREE_CHAIN (decl))
|
||
{
|
||
rtx parameter = DECL_INCOMING_RTL (decl);
|
||
enum machine_mode mode = GET_MODE (parameter);
|
||
|
||
if (GET_CODE (parameter) == REG)
|
||
{
|
||
if (GET_MODE_CLASS (mode) == MODE_FLOAT)
|
||
{
|
||
int bits;
|
||
|
||
float_parms++;
|
||
|
||
if (mode == SFmode)
|
||
bits = 0x2;
|
||
else if (mode == DFmode)
|
||
bits = 0x3;
|
||
else
|
||
abort ();
|
||
|
||
/* If only one bit will fit, don't or in this entry. */
|
||
if (next_parm_info_bit > 0)
|
||
parm_info |= (bits << (next_parm_info_bit - 1));
|
||
next_parm_info_bit -= 2;
|
||
}
|
||
else
|
||
{
|
||
fixed_parms += ((GET_MODE_SIZE (mode) + (UNITS_PER_WORD - 1))
|
||
/ UNITS_PER_WORD);
|
||
next_parm_info_bit -= 1;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/* Number of fixed point parameters. */
|
||
/* This is actually the number of words of fixed point parameters; thus
|
||
an 8 byte struct counts as 2; and thus the maximum value is 8. */
|
||
fprintf (file, "%d,", fixed_parms);
|
||
|
||
/* 2 bitfields: number of floating point parameters (7 bits), parameters
|
||
all on stack. */
|
||
/* This is actually the number of fp registers that hold parameters;
|
||
and thus the maximum value is 13. */
|
||
/* Set parameters on stack bit if parameters are not in their original
|
||
registers, regardless of whether they are on the stack? Xlc
|
||
seems to set the bit when not optimizing. */
|
||
fprintf (file, "%d\n", ((float_parms << 1) | (! optimize)));
|
||
|
||
/* Optional fields follow. Some are variable length. */
|
||
|
||
/* Parameter types, left adjusted bit fields: 0 fixed, 10 single float,
|
||
11 double float. */
|
||
/* There is an entry for each parameter in a register, in the order that
|
||
they occur in the parameter list. Any intervening arguments on the
|
||
stack are ignored. If the list overflows a long (max possible length
|
||
34 bits) then completely leave off all elements that don't fit. */
|
||
/* Only emit this long if there was at least one parameter. */
|
||
if (fixed_parms || float_parms)
|
||
fprintf (file, "\t.long %d\n", parm_info);
|
||
|
||
/* Offset from start of code to tb table. */
|
||
fprintf (file, "\t.long LT..");
|
||
RS6000_OUTPUT_BASENAME (file, fname);
|
||
fprintf (file, "-.");
|
||
RS6000_OUTPUT_BASENAME (file, fname);
|
||
fprintf (file, "\n");
|
||
|
||
/* Interrupt handler mask. */
|
||
/* Omit this long, since we never set the interrupt handler bit above. */
|
||
|
||
/* Number of CTL (controlled storage) anchors. */
|
||
/* Omit this long, since the has_ctl bit is never set above. */
|
||
|
||
/* Displacement into stack of each CTL anchor. */
|
||
/* Omit this list of longs, because there are no CTL anchors. */
|
||
|
||
/* Length of function name. */
|
||
fprintf (file, "\t.short %d\n", strlen (fname));
|
||
|
||
/* Function name. */
|
||
assemble_string (fname, strlen (fname));
|
||
|
||
/* Register for alloca automatic storage; this is always reg 31.
|
||
Only emit this if the alloca bit was set above. */
|
||
if (frame_pointer_needed)
|
||
fprintf (file, "\t.byte 31\n");
|
||
}
|
||
}
|
||
|
||
/* Output a TOC entry. We derive the entry name from what is
|
||
being written. */
|
||
|
||
void
|
||
output_toc (file, x, labelno)
|
||
FILE *file;
|
||
rtx x;
|
||
int labelno;
|
||
{
|
||
char buf[256];
|
||
char *name = buf;
|
||
rtx base = x;
|
||
int offset = 0;
|
||
|
||
ASM_OUTPUT_INTERNAL_LABEL (file, "LC", labelno);
|
||
|
||
/* Handle FP constants specially. Note that if we have a minimal
|
||
TOC, things we put here aren't actually in the TOC, so we can allow
|
||
FP constants. */
|
||
if (GET_CODE (x) == CONST_DOUBLE
|
||
&& GET_MODE (x) == DFmode
|
||
&& TARGET_FLOAT_FORMAT == HOST_FLOAT_FORMAT
|
||
&& BITS_PER_WORD == HOST_BITS_PER_INT
|
||
&& ! (TARGET_NO_FP_IN_TOC && ! TARGET_MINIMAL_TOC))
|
||
{
|
||
if (TARGET_MINIMAL_TOC)
|
||
fprintf (file, "\t.long %d\n\t.long %d\n",
|
||
CONST_DOUBLE_LOW (x), CONST_DOUBLE_HIGH (x));
|
||
else
|
||
fprintf (file, "\t.tc FD_%x_%x[TC],%d,%d\n",
|
||
CONST_DOUBLE_LOW (x), CONST_DOUBLE_HIGH (x),
|
||
CONST_DOUBLE_LOW (x), CONST_DOUBLE_HIGH (x));
|
||
return;
|
||
}
|
||
else if (GET_CODE (x) == CONST_DOUBLE && GET_MODE (x) == SFmode
|
||
&& ! (TARGET_NO_FP_IN_TOC && ! TARGET_MINIMAL_TOC))
|
||
{
|
||
rtx val = operand_subword (x, 0, 0, SFmode);
|
||
|
||
if (val == 0 || GET_CODE (val) != CONST_INT)
|
||
abort ();
|
||
|
||
if (TARGET_MINIMAL_TOC)
|
||
fprintf (file, "\t.long %d\n", INTVAL (val));
|
||
else
|
||
fprintf (file, "\t.tc FS_%x[TC],%d\n", INTVAL (val), INTVAL (val));
|
||
return;
|
||
}
|
||
|
||
if (GET_CODE (x) == CONST)
|
||
{
|
||
base = XEXP (XEXP (x, 0), 0);
|
||
offset = INTVAL (XEXP (XEXP (x, 0), 1));
|
||
}
|
||
|
||
if (GET_CODE (base) == SYMBOL_REF)
|
||
name = XSTR (base, 0);
|
||
else if (GET_CODE (base) == LABEL_REF)
|
||
ASM_GENERATE_INTERNAL_LABEL (buf, "L", CODE_LABEL_NUMBER (XEXP (base, 0)));
|
||
else if (GET_CODE (base) == CODE_LABEL)
|
||
ASM_GENERATE_INTERNAL_LABEL (buf, "L", CODE_LABEL_NUMBER (base));
|
||
else
|
||
abort ();
|
||
|
||
if (TARGET_MINIMAL_TOC)
|
||
fprintf (file, "\t.long ");
|
||
else
|
||
{
|
||
fprintf (file, "\t.tc ");
|
||
RS6000_OUTPUT_BASENAME (file, name);
|
||
|
||
if (offset < 0)
|
||
fprintf (file, ".N%d", - offset);
|
||
else if (offset)
|
||
fprintf (file, ".P%d", offset);
|
||
|
||
fprintf (file, "[TC],");
|
||
}
|
||
output_addr_const (file, x);
|
||
fprintf (file, "\n");
|
||
}
|
||
|
||
/* Output an assembler pseudo-op to write an ASCII string of N characters
|
||
starting at P to FILE.
|
||
|
||
On the RS/6000, we have to do this using the .byte operation and
|
||
write out special characters outside the quoted string.
|
||
Also, the assembler is broken; very long strings are truncated,
|
||
so we must artificially break them up early. */
|
||
|
||
void
|
||
output_ascii (file, p, n)
|
||
FILE *file;
|
||
char *p;
|
||
int n;
|
||
{
|
||
char c;
|
||
int i, count_string;
|
||
char *for_string = "\t.byte \"";
|
||
char *for_decimal = "\t.byte ";
|
||
char *to_close = NULL;
|
||
|
||
count_string = 0;
|
||
for (i = 0; i < n; i++)
|
||
{
|
||
c = *p++;
|
||
if (c >= ' ' && c < 0177)
|
||
{
|
||
if (for_string)
|
||
fputs (for_string, file);
|
||
putc (c, file);
|
||
|
||
/* Write two quotes to get one. */
|
||
if (c == '"')
|
||
{
|
||
putc (c, file);
|
||
++count_string;
|
||
}
|
||
|
||
for_string = NULL;
|
||
for_decimal = "\"\n\t.byte ";
|
||
to_close = "\"\n";
|
||
++count_string;
|
||
|
||
if (count_string >= 512)
|
||
{
|
||
fputs (to_close, file);
|
||
|
||
for_string = "\t.byte \"";
|
||
for_decimal = "\t.byte ";
|
||
to_close = NULL;
|
||
count_string = 0;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
if (for_decimal)
|
||
fputs (for_decimal, file);
|
||
fprintf (file, "%d", c);
|
||
|
||
for_string = "\n\t.byte \"";
|
||
for_decimal = ", ";
|
||
to_close = "\n";
|
||
count_string = 0;
|
||
}
|
||
}
|
||
|
||
/* Now close the string if we have written one. Then end the line. */
|
||
if (to_close)
|
||
fprintf (file, to_close);
|
||
}
|
||
|
||
/* Generate a unique section name for FILENAME for a section type
|
||
represented by SECTION_DESC. Output goes into BUF.
|
||
|
||
SECTION_DESC can be any string, as long as it is different for each
|
||
possible section type.
|
||
|
||
We name the section in the same manner as xlc. The name begins with an
|
||
underscore followed by the filename (after stripping any leading directory
|
||
names) with the last period replaced by the string SECTION_DESC. If
|
||
FILENAME does not contain a period, SECTION_DESC is appended to the end of
|
||
the name. */
|
||
|
||
void
|
||
rs6000_gen_section_name (buf, filename, section_desc)
|
||
char **buf;
|
||
char *filename;
|
||
char *section_desc;
|
||
{
|
||
char *q, *after_last_slash, *last_period;
|
||
char *p;
|
||
int len;
|
||
|
||
after_last_slash = filename;
|
||
for (q = filename; *q; q++)
|
||
{
|
||
if (*q == '/')
|
||
after_last_slash = q + 1;
|
||
else if (*q == '.')
|
||
last_period = q;
|
||
}
|
||
|
||
len = strlen (after_last_slash) + strlen (section_desc) + 2;
|
||
*buf = (char *) permalloc (len);
|
||
|
||
p = *buf;
|
||
*p++ = '_';
|
||
|
||
for (q = after_last_slash; *q; q++)
|
||
{
|
||
if (q == last_period)
|
||
{
|
||
strcpy (p, section_desc);
|
||
p += strlen (section_desc);
|
||
}
|
||
|
||
else if (isalnum (*q))
|
||
*p++ = *q;
|
||
}
|
||
|
||
if (last_period == 0)
|
||
strcpy (p, section_desc);
|
||
else
|
||
*p = '\0';
|
||
}
|
||
|
||
/* Write function profiler code. */
|
||
|
||
void
|
||
output_function_profiler (file, labelno)
|
||
FILE *file;
|
||
int labelno;
|
||
{
|
||
/* The last used parameter register. */
|
||
int last_parm_reg;
|
||
int i, j;
|
||
|
||
/* Set up a TOC entry for the profiler label. */
|
||
toc_section ();
|
||
fprintf (file, "LPC..%d:\n\t.tc\tLP..%d[TC],LP..%d\n",
|
||
labelno, labelno, labelno);
|
||
text_section ();
|
||
|
||
/* Figure out last used parameter register. The proper thing to do is
|
||
to walk incoming args of the function. A function might have live
|
||
parameter registers even if it has no incoming args. */
|
||
|
||
for (last_parm_reg = 10;
|
||
last_parm_reg > 2 && ! regs_ever_live [last_parm_reg];
|
||
last_parm_reg--)
|
||
;
|
||
|
||
/* Save parameter registers in regs 23-30. Don't overwrite reg 31, since
|
||
it might be set up as the frame pointer. */
|
||
|
||
for (i = 3, j = 30; i <= last_parm_reg; i++, j--)
|
||
fprintf (file, "\tai %d,%d,0\n", j, i);
|
||
|
||
/* Load location address into r3, and call mcount. */
|
||
|
||
fprintf (file, "\tl 3,LPC..%d(2)\n\tbl .mcount\n", labelno);
|
||
|
||
/* Restore parameter registers. */
|
||
|
||
for (i = 3, j = 30; i <= last_parm_reg; i++, j--)
|
||
fprintf (file, "\tai %d,%d,0\n", i, j);
|
||
}
|