Commit d60a05a1 by Richard Kenner

(alpha_sa_size): Round size to 16-byte boundary.

(add_long_const): Add new arg, TEMP_REG.
(output_{pro,epi}log): Rework to put save area between outgoing args and
local variables.

From-SVN: r7606
parent 9ff3516a
/* Subroutines used for code generation on the DEC Alpha. /* Subroutines used for code generation on the DEC Alpha.
Copyright (C) 1992, 1993, 1994 Free Software Foundation, Inc. Copyright (C) 1992, 1993, 1994 Free Software Foundation, Inc.
Contributed by Richard Kenner (kenner@nyu.edu) Contributed by Richard Kenner (kenner@vlsi1.ultra.nyu.edu)
This file is part of GNU CC. This file is part of GNU CC.
...@@ -58,6 +58,10 @@ static int inside_function = FALSE; ...@@ -58,6 +58,10 @@ static int inside_function = FALSE;
int alpha_function_needs_gp; int alpha_function_needs_gp;
extern char *version_string; extern char *version_string;
/* Declarations of static functions. */
static void alpha_set_memflags_1 PROTO((rtx, int, int, int));
static void add_long_const PROTO((FILE *, HOST_WIDE_INT, int, int, int));
/* Returns 1 if VALUE is a mask that contains full bytes of zero or ones. */ /* Returns 1 if VALUE is a mask that contains full bytes of zero or ones. */
...@@ -1127,6 +1131,10 @@ alpha_sa_size () ...@@ -1127,6 +1131,10 @@ alpha_sa_size ()
if (size != 0 && ! regs_ever_live[26]) if (size != 0 && ! regs_ever_live[26])
size++; size++;
/* Our size must be even (multiple of 16 bytes). */
if (size & 1)
size ++;
return size * 8; return size * 8;
} }
...@@ -1162,13 +1170,16 @@ alpha_write_verstamp (file) ...@@ -1162,13 +1170,16 @@ alpha_write_verstamp (file)
} }
/* Write code to add constant C to register number IN_REG (possibly 31) /* Write code to add constant C to register number IN_REG (possibly 31)
and put the result into OUT_REG. Write the code to FILE. */ and put the result into OUT_REG. Use TEMP_REG as a scratch register;
usually this will be OUT_REG, but should not be if OUT_REG is
STACK_POINTER_REGNUM, since it must be updated in a single instruction.
Write the code to FILE. */
static void static void
add_long_const (file, c, in_reg, out_reg) add_long_const (file, c, in_reg, out_reg, temp_reg)
HOST_WIDE_INT c;
int in_reg, out_reg;
FILE *file; FILE *file;
HOST_WIDE_INT c;
int in_reg, out_reg, temp_reg;
{ {
HOST_WIDE_INT low = (c & 0xffff) - 2 * (c & 0x8000); HOST_WIDE_INT low = (c & 0xffff) - 2 * (c & 0x8000);
HOST_WIDE_INT tmp1 = c - low; HOST_WIDE_INT tmp1 = c - low;
...@@ -1194,17 +1205,22 @@ add_long_const (file, c, in_reg, out_reg) ...@@ -1194,17 +1205,22 @@ add_long_const (file, c, in_reg, out_reg)
if (low != 0) if (low != 0)
{ {
int result_reg = (extra == 0 && high == 0) ? out_reg : temp_reg;
if (low >= 0 && low < 255) if (low >= 0 && low < 255)
fprintf (file, "\taddq $%d,%d,$%d\n", in_reg, low, out_reg); fprintf (file, "\taddq $%d,%d,$%d\n", in_reg, low, result_reg);
else else
fprintf (file, "\tlda $%d,%d($%d)\n", out_reg, low, in_reg); fprintf (file, "\tlda $%d,%d($%d)\n", result_reg, low, in_reg);
in_reg = out_reg;
in_reg = result_reg;
} }
if (extra) if (extra)
{ {
fprintf (file, "\tldah $%d,%d($%d)\n", out_reg, extra, in_reg); int result_reg = (high == 0) ? out_reg : temp_reg;
in_reg = out_reg;
fprintf (file, "\tldah $%d,%d($%d)\n", result_reg, extra, in_reg);
in_reg = result_reg;
} }
if (high) if (high)
...@@ -1218,20 +1234,21 @@ output_prolog (file, size) ...@@ -1218,20 +1234,21 @@ output_prolog (file, size)
FILE *file; FILE *file;
int size; int size;
{ {
HOST_WIDE_INT vars_size = (size + 7) & ~7; HOST_WIDE_INT out_args_size
HOST_WIDE_INT frame_size = ((vars_size + current_function_outgoing_args_size = ALPHA_ROUND (current_function_outgoing_args_size);
+ current_function_pretend_args_size HOST_WIDE_INT sa_size = alpha_sa_size ();
+ alpha_sa_size () + 15) & ~15); HOST_WIDE_INT frame_size
HOST_WIDE_INT reg_offset = vars_size + current_function_outgoing_args_size; = (out_args_size + sa_size
+ ALPHA_ROUND (size + current_function_pretend_args_size));
HOST_WIDE_INT reg_offset = out_args_size;
HOST_WIDE_INT start_reg_offset = reg_offset; HOST_WIDE_INT start_reg_offset = reg_offset;
HOST_WIDE_INT actual_start_reg_offset = start_reg_offset; HOST_WIDE_INT actual_start_reg_offset = start_reg_offset;
int int_reg_save_area_size = 0; int int_reg_save_area_size = 0;
rtx insn; rtx insn;
int reg_offset_base_reg = 30;
unsigned reg_mask = 0; unsigned reg_mask = 0;
int i; int i;
/* Ecoff can handle multiple .file directives, put out file and lineno. /* Ecoff can handle multiple .file directives, so put out file and lineno.
We have to do that before the .ent directive as we cannot switch We have to do that before the .ent directive as we cannot switch
files within procedures with native ecoff because line numbers are files within procedures with native ecoff because line numbers are
linked to procedure descriptors. linked to procedure descriptors.
...@@ -1244,7 +1261,8 @@ output_prolog (file, size) ...@@ -1244,7 +1261,8 @@ output_prolog (file, size)
ASM_OUTPUT_SOURCE_FILENAME (file, ASM_OUTPUT_SOURCE_FILENAME (file,
DECL_SOURCE_FILE (current_function_decl)); DECL_SOURCE_FILE (current_function_decl));
if (debug_info_level != DINFO_LEVEL_TERSE) if (debug_info_level != DINFO_LEVEL_TERSE)
ASM_OUTPUT_SOURCE_LINE (file, DECL_SOURCE_LINE (current_function_decl)); ASM_OUTPUT_SOURCE_LINE (file,
DECL_SOURCE_LINE (current_function_decl));
} }
/* The assembly language programmer's guide states that the second argument /* The assembly language programmer's guide states that the second argument
...@@ -1300,14 +1318,15 @@ output_prolog (file, size) ...@@ -1300,14 +1318,15 @@ output_prolog (file, size)
if (frame_size > 4096) if (frame_size > 4096)
{ {
int probed = 4096; int probed = 4096;
int regnum = 2; int regnum = 1;
fprintf (file, "\tldq $%d,-%d($30)\n", regnum++, probed); fprintf (file, "\tldq $%d,-%d($30)\n", regnum++, probed);
while (probed + 8192 < frame_size) while (probed + 8192 < frame_size)
fprintf (file, "\tldq $%d,-%d($30)\n", regnum++, probed += 8192); fprintf (file, "\tldq $%d,-%d($30)\n", regnum++, probed += 8192);
if (probed + 4096 < frame_size) /* We only have to do this probe if we aren't saving registers. */
if (sa_size == 0 && probed + 4096 < frame_size)
fprintf (file, "\tldq $%d,-%d($30)\n", regnum++, probed += 4096); fprintf (file, "\tldq $%d,-%d($30)\n", regnum++, probed += 4096);
if (regnum > 9) if (regnum > 9)
...@@ -1322,12 +1341,13 @@ output_prolog (file, size) ...@@ -1322,12 +1341,13 @@ output_prolog (file, size)
/* Here we generate code to set R4 to SP + 4096 and set R5 to the /* Here we generate code to set R4 to SP + 4096 and set R5 to the
number of 8192 byte blocks to probe. We then probe each block number of 8192 byte blocks to probe. We then probe each block
in the loop and then set SP to the proper location. If the in the loop and then set SP to the proper location. If the
amount remaining is > 4096, we have to do one more probe. */ amount remaining is > 4096, we have to do one more probe if we
are not saving any registers. */
HOST_WIDE_INT blocks = (frame_size + 4096) / 8192; HOST_WIDE_INT blocks = (frame_size + 4096) / 8192;
HOST_WIDE_INT leftover = frame_size + 4096 - blocks * 8192; HOST_WIDE_INT leftover = frame_size + 4096 - blocks * 8192;
add_long_const (file, blocks, 31, 5); add_long_const (file, blocks, 31, 5, 5);
fprintf (file, "\tlda $4,4096($30)\n"); fprintf (file, "\tlda $4,4096($30)\n");
...@@ -1344,31 +1364,21 @@ output_prolog (file, size) ...@@ -1344,31 +1364,21 @@ output_prolog (file, size)
fprintf (file, "\tlda $30,-%d($4)\n", leftover); fprintf (file, "\tlda $30,-%d($4)\n", leftover);
if (leftover > 4096) if (leftover > 4096 && sa_size == 0)
fprintf (file, "\tldq $2,%d($30)\n", leftover - 4096); fprintf (file, "\tldq $2,%d($30)\n", leftover - 4096);
} }
/* Describe our frame. */ /* Describe our frame. */
fprintf (file, "\t.frame $%d,%d,$26,%d\n", fprintf (file, "\t.frame $%d,%d,$26,%d\n",
frame_pointer_needed ? FRAME_POINTER_REGNUM : STACK_POINTER_REGNUM, (frame_pointer_needed
? HARD_FRAME_POINTER_REGNUM : STACK_POINTER_REGNUM),
frame_size, current_function_pretend_args_size); frame_size, current_function_pretend_args_size);
/* If reg_offset is "close enough" to 2**15 that one of the offsets would /* Save register 26 if any other register needs to be saved. */
overflow a store instruction, compute the base of the register save if (sa_size != 0)
area into $28. */
if (reg_offset >= 32768 - alpha_sa_size () && alpha_sa_size () != 0)
{
add_long_const (file, reg_offset, 30, 28);
reg_offset_base_reg = 28;
reg_offset = start_reg_offset = 0;
}
/* Save register 26 if it is used or if any other register needs to
be saved. */
if (regs_ever_live[26] || alpha_sa_size () != 0)
{ {
reg_mask |= 1 << 26; reg_mask |= 1 << 26;
fprintf (file, "\tstq $26,%d($%d)\n", reg_offset, reg_offset_base_reg); fprintf (file, "\tstq $26,%d($30)\n", reg_offset);
reg_offset += 8; reg_offset += 8;
int_reg_save_area_size += 8; int_reg_save_area_size += 8;
} }
...@@ -1378,8 +1388,7 @@ output_prolog (file, size) ...@@ -1378,8 +1388,7 @@ output_prolog (file, size)
if (! fixed_regs[i] && ! call_used_regs[i] && regs_ever_live[i] && i != 26) if (! fixed_regs[i] && ! call_used_regs[i] && regs_ever_live[i] && i != 26)
{ {
reg_mask |= 1 << i; reg_mask |= 1 << i;
fprintf (file, "\tstq $%d,%d($%d)\n", fprintf (file, "\tstq $%d,%d($30)\n", i, reg_offset);
i, reg_offset, reg_offset_base_reg);
reg_offset += 8; reg_offset += 8;
int_reg_save_area_size += 8; int_reg_save_area_size += 8;
} }
...@@ -1397,8 +1406,7 @@ output_prolog (file, size) ...@@ -1397,8 +1406,7 @@ output_prolog (file, size)
&& regs_ever_live[i + 32]) && regs_ever_live[i + 32])
{ {
reg_mask |= 1 << i; reg_mask |= 1 << i;
fprintf (file, "\tstt $f%d,%d($%d)\n", fprintf (file, "\tstt $f%d,%d($30)\n", i, reg_offset);
i, reg_offset, reg_offset_base_reg);
reg_offset += 8; reg_offset += 8;
} }
...@@ -1424,13 +1432,16 @@ output_epilog (file, size) ...@@ -1424,13 +1432,16 @@ output_epilog (file, size)
int size; int size;
{ {
rtx insn = get_last_insn (); rtx insn = get_last_insn ();
HOST_WIDE_INT vars_size = (size + 7) & ~7; HOST_WIDE_INT out_args_size
HOST_WIDE_INT frame_size = ((vars_size + current_function_outgoing_args_size = ALPHA_ROUND (current_function_outgoing_args_size);
+ current_function_pretend_args_size HOST_WIDE_INT sa_size = alpha_sa_size ();
+ alpha_sa_size () + 15) & ~15); HOST_WIDE_INT frame_size
HOST_WIDE_INT reg_offset = vars_size + current_function_outgoing_args_size; = (out_args_size + sa_size
+ ALPHA_ROUND (size + current_function_pretend_args_size));
HOST_WIDE_INT reg_offset = out_args_size;
HOST_WIDE_INT frame_size_from_reg_save = frame_size - reg_offset; HOST_WIDE_INT frame_size_from_reg_save = frame_size - reg_offset;
int reg_offset_base_reg = 30; int restore_fp
= frame_pointer_needed && regs_ever_live[HARD_FRAME_POINTER_REGNUM];
int i; int i;
/* If the last insn was a BARRIER, we don't have to write anything except /* If the last insn was a BARRIER, we don't have to write anything except
...@@ -1445,21 +1456,11 @@ output_epilog (file, size) ...@@ -1445,21 +1456,11 @@ output_epilog (file, size)
if (frame_pointer_needed) if (frame_pointer_needed)
fprintf (file, "\tbis $15,$15,$30\n"); fprintf (file, "\tbis $15,$15,$30\n");
/* If the register save area is out of range, put its address into
$28. */
if (reg_offset >= 32768 - alpha_sa_size () && alpha_sa_size () != 0)
{
add_long_const (file, reg_offset, 30, 28);
reg_offset_base_reg = 28;
reg_offset = 0;
}
/* Restore all the registers, starting with the return address /* Restore all the registers, starting with the return address
register. */ register. */
if (regs_ever_live[26] || alpha_sa_size () != 0) if (sa_size != 0)
{ {
fprintf (file, "\tldq $26,%d($%d)\n", fprintf (file, "\tldq $26,%d($30)\n", reg_offset);
reg_offset, reg_offset_base_reg);
reg_offset += 8; reg_offset += 8;
} }
...@@ -1471,11 +1472,10 @@ output_epilog (file, size) ...@@ -1471,11 +1472,10 @@ output_epilog (file, size)
if (! fixed_regs[i] && ! call_used_regs[i] && regs_ever_live[i] if (! fixed_regs[i] && ! call_used_regs[i] && regs_ever_live[i]
&& i != 26) && i != 26)
{ {
if (i == FRAME_POINTER_REGNUM && frame_pointer_needed) if (i == HARD_FRAME_POINTER_REGNUM && frame_pointer_needed)
fp_offset = reg_offset; fp_offset = reg_offset;
else else
fprintf (file, "\tldq $%d,%d($%d)\n", fprintf (file, "\tldq $%d,%d($30)\n", i, reg_offset);
i, reg_offset, reg_offset_base_reg);
reg_offset += 8; reg_offset += 8;
} }
...@@ -1483,47 +1483,32 @@ output_epilog (file, size) ...@@ -1483,47 +1483,32 @@ output_epilog (file, size)
if (! fixed_regs[i + 32] && ! call_used_regs[i + 32] if (! fixed_regs[i + 32] && ! call_used_regs[i + 32]
&& regs_ever_live[i + 32]) && regs_ever_live[i + 32])
{ {
fprintf (file, "\tldt $f%d,%d($%d)\n", fprintf (file, "\tldt $f%d,%d($30)\n", i, reg_offset);
i, reg_offset, reg_offset_base_reg);
reg_offset += 8; reg_offset += 8;
} }
/* If the stack size is large, compute the size of the stack into /* If the stack size is large and we have a frame pointer, compute the
a register because the old FP restore, stack pointer adjust, size of the stack into a register because the old FP restore, stack
and return are required to be consecutive instructions. pointer adjust, and return are required to be consecutive
However, if the new stack pointer can be computed by adding the instructions. */
a constant to the start of the register save area, we can do if (frame_size > 32767 && restore_fp)
it that way. */ add_long_const (file, frame_size, 31, 1, 1);
if (frame_size > 32767
&& ! (reg_offset_base_reg != 30
&& frame_size_from_reg_save < 32768))
add_long_const (file, frame_size, 31, 1);
/* If we needed a frame pointer and we have to restore it, do it /* If we needed a frame pointer and we have to restore it, do it
now. This must be done in one instruction immediately now. This must be done in one instruction immediately
before the SP update. */ before the SP update. */
if (frame_pointer_needed && regs_ever_live[FRAME_POINTER_REGNUM]) if (restore_fp)
fprintf (file, "\tldq $15,%d($%d)\n", fp_offset, reg_offset_base_reg); fprintf (file, "\tldq $15,%d($30)\n", fp_offset);
/* Now update the stack pointer, if needed. This must be done in /* Now update the stack pointer, if needed. Only one instruction must
one, stylized, instruction. */ modify the stack pointer. It must be the last instruction in the
if (frame_size > 32768) sequence and must be an ADDQ or LDA instruction. If the frame
{ pointer was loaded above, we may only put one instruction here. */
if (reg_offset_base_reg != 30
&& frame_size_from_reg_save < 32768) if (frame_size > 32768 && restore_fp)
{ fprintf (file, "\taddq $1,$30,$30\n");
if (frame_size_from_reg_save < 255) else
fprintf (file, "\taddq $%d,%d,$30\n", add_long_const (file, frame_size, 30, 30, 1);
reg_offset_base_reg, frame_size_from_reg_save);
else
fprintf (file, "\tlda %30,%d($%d)\n",
frame_size_from_reg_save, reg_offset_base_reg);
}
else
fprintf (file, "\taddq $1,$30,$30\n");
}
else if (frame_size != 0)
fprintf (file, "\tlda $30,%d($30)\n", frame_size);
/* Finally return to the caller. */ /* Finally return to the caller. */
fprintf (file, "\tret $31,($26),1\n"); fprintf (file, "\tret $31,($26),1\n");
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment