(v9 sparc_builtin_saveregs): Save quadword fp regs on 16 byte boundaries.

(v9 sparc_builtin_saveregs): Save quadword fp regs on
16 byte boundaries.  Handle non-zero number of named fp args.

From-SVN: r7855
This commit is contained in:
Doug Evans 1994-08-04 01:34:34 +00:00
parent 29421402b7
commit a7acd9111d

View File

@ -3086,27 +3086,36 @@ sparc_builtin_saveregs (arglist)
{ {
tree fntype = TREE_TYPE (current_function_decl); tree fntype = TREE_TYPE (current_function_decl);
/* First unnamed integer register. */ /* First unnamed integer register. */
int first_intreg = current_function_args_info.arg_count[0]; int first_intreg = current_function_args_info.arg_count[(int) SPARC_ARG_INT];
/* Number of integer registers we need to save. */ /* Number of integer registers we need to save. */
int n_intregs = MAX (0, NPARM_REGS (SImode) - first_intreg); int n_intregs = MAX (0, NPARM_REGS (SImode) - first_intreg);
/* First unnamed SFmode float reg (no, you can't pass SFmode floats as /* First unnamed SFmode float reg (no, you can't pass SFmode floats as
unnamed arguments, we just number them that way). */ unnamed arguments, we just number them that way). We must round up to
int first_floatreg = current_function_args_info.arg_count[1] + 1 & ~1; the next double word float reg - that is the first one to save. */
int first_floatreg = current_function_args_info.arg_count[(int) SPARC_ARG_FLOAT] + 1 & ~1;
/* Number of SFmode float regs to save. */ /* Number of SFmode float regs to save. */
int n_floatregs = MAX (0, NPARM_REGS (SFmode) - first_floatreg); int n_floatregs = MAX (0, NPARM_REGS (SFmode) - first_floatreg);
int ptrsize = GET_MODE_SIZE (Pmode); int ptrsize = GET_MODE_SIZE (Pmode);
rtx valist, regbuf, fpregs; rtx valist, regbuf, fpregs;
int regno; int bufsize, adjust, regno;
/* Allocate block of memory for the regs. /* Allocate block of memory for the regs.
We only allocate as much as we need. */ We only allocate as much as we need, but we must ensure quadword float
regs are stored with the appropriate alignment. */
/* ??? If n_intregs + n_floatregs == 0, should we allocate at least 1 byte? /* ??? If n_intregs + n_floatregs == 0, should we allocate at least 1 byte?
Or can assign_stack_local accept a 0 SIZE argument? */ Or can assign_stack_local accept a 0 SIZE argument? */
regbuf = assign_stack_local (BLKmode, bufsize = (n_intregs * UNITS_PER_WORD) + (n_floatregs * (UNITS_PER_WORD / 2));
(n_intregs * UNITS_PER_WORD /* Add space in front of the int regs to ensure proper alignment of quadword
+ n_floatregs * FLOAT_TYPE_SIZE/BITS_PER_UNIT), fp regs. We must add the space in front because va_start assumes this. */
BITS_PER_WORD); if (n_floatregs >= 4)
adjust = ((n_intregs + first_floatreg / 2) % 2) * UNITS_PER_WORD;
else
adjust = 0;
regbuf = assign_stack_local (BLKmode, bufsize + adjust,
GET_MODE_BITSIZE (TFmode));
regbuf = gen_rtx (MEM, BLKmode, plus_constant (XEXP (regbuf, 0), adjust));
MEM_IN_STRUCT_P (regbuf) = 1; MEM_IN_STRUCT_P (regbuf) = 1;
/* Save int args. /* Save int args.
@ -3130,9 +3139,10 @@ sparc_builtin_saveregs (arglist)
for (regno = first_floatreg; regno < NPARM_REGS (SFmode); regno += 2) for (regno = first_floatreg; regno < NPARM_REGS (SFmode); regno += 2)
emit_move_insn (gen_rtx (MEM, DFmode, emit_move_insn (gen_rtx (MEM, DFmode,
plus_constant (fpregs, plus_constant (fpregs,
GET_MODE_SIZE (SFmode) * regno)), GET_MODE_SIZE (SFmode)
gen_rtx (REG, DFmode, BASE_INCOMING_ARG_REG (DFmode) * (regno - first_floatreg))),
+ regno)); gen_rtx (REG, DFmode,
BASE_INCOMING_ARG_REG (DFmode) + regno));
/* Return the address of the regbuf. */ /* Return the address of the regbuf. */