[forwarded] RS/6000 patch adding some features for PPC compiler

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 




Here is the message I've posted few months ago on gcc-patches. Unfortunatelly there was no answer. Perhaps someone could help me here.


with best regards,



Subject: RS/6000 patch adding some features for PPC compiler
Date: Fri, 30 May 2003 12:02:37 +0200
From: Michal Schulz <michal.schulz@xxxxxxxxxxxxxxx>
To: gcc-patches@xxxxxxxxxxx


Hello.


First of all I would like to introduce myself. I am one of AROS
developers, working on AmigaOS compatible operating system. Currently my
interests are to port our OS to PPC based platform (PReP and CHRP machines).

In order to do that (and to simplify porting issues), together with
Fabio Alemagna (one of AROS developers) I have patched gcc so that it
understands "stackparm" attribute in functions. Function defined with
stackparm doesn not create the parameters frame used by __builtin_va*
functions but instead pushes all the parameters on the stack
(function_arg() returns always NULL_RTX);

The second part of the patch is related with compatibility issues. Since
on original AmigaOS parameters to system functions were passed in
registers, I wanted to follow this tradition on PPC architectures too.
First of all standard ABI allows only up to 8 parameters to be passed in
registers (some library functions take even 12!), secondly, this way it
would be easier to implement M68K emulation into the system (there would
be no need to trace how and where parameters should be passed, just
simple mapping, eg. D0-D7 => R16-R23, A0-A7 => R24-R31).

In order to do that, I have added "regarg" attribute. It requires one
parameter which is the register name where argument is supposed to be.
the function may look like this:

int foo(int arg __attribute__((regarg("r16"))))
{
	[...]
}

The attached patch is for gcc-3.2.3.

I have two problems with that:

1) If the vararg function (without stackparm) is called from function
with registerized parameter, it does not work. I have no clue why
especially that I haven't change anything within vararg handling part of
rs6000.c file

2) By strong optimisations (-O2, -O3) gcc "forgets" somehow that it
should initialize all registers with given parameters. Looking at asm
output I see the initialization (some of them even 100 instructions
before the call), but then the content of given registers is
overwritten. Is it my fault or gcc's?

I hope there is someone who could help me.

PS. Yes, I have read the gccint docs, but they couldn't help me solving
my problems.

with best regards,

--
Michal Schulz
Institut für Metallurgie
TU Clausthal
Robert-Koch-Straße 42
38678 Clausthal-Zellerfeld

diff -Naur gcc-3.2.3-org/gcc/config/rs6000/rs6000.c gcc-3.2.3/gcc/config/rs6000/rs6000.c
--- gcc-3.2.3-org/gcc/config/rs6000/rs6000.c	2003-04-08 22:03:23.000000000 +0000
+++ gcc-3.2.3/gcc/config/rs6000/rs6000.c	2003-05-29 23:07:00.000000000 +0000
@@ -141,6 +141,7 @@
 static bool rs6000_assemble_integer PARAMS ((rtx, unsigned int, int));
 static int rs6000_ra_ever_killed PARAMS ((void));
 static tree rs6000_handle_longcall_attribute PARAMS ((tree *, tree, tree, int, bool *));
+static tree rs6000_handle_register_attribute PARAMS ((tree *, tree, tree, int, bool *));
 const struct attribute_spec rs6000_attribute_table[];
 static void rs6000_output_function_prologue PARAMS ((FILE *, HOST_WIDE_INT));
 static void rs6000_output_function_epilogue PARAMS ((FILE *, HOST_WIDE_INT));
@@ -2510,6 +2511,13 @@
   if (fntype && lookup_attribute ("longcall", TYPE_ATTRIBUTES (fntype)))
     cum->call_cookie = CALL_LONG;
 
+  cum->int_regmask = 0x00003fff;	/* r14 - r31 are free for use */
+  cum->fp_regmask =  0x00003fff;	/* f14 - f31 are free for use */
+  cum->altivec_regmask = 0;		/* all altivec regs free */
+
+  cum->stackparm = fntype && lookup_attribute("stackparm",
+					      TYPE_ATTRIBUTES(fntype));
+
   if (TARGET_DEBUG_ARG)
     {
       fprintf (stderr, "\ninit_cumulative_args:");
@@ -2585,8 +2593,49 @@
      tree type;
      int named;
 {
+  tree regarg = lookup_attribute("regarg", TYPE_ATTRIBUTES(type));
+
   cum->nargs_prototype--;
 
+  /*
+    If regarg attribute is defined and we are within function prototype,
+    allocate space for given reg.
+  */
+  if (regarg && cum->prototype)
+  {
+    int reg = decode_reg_name(TREE_STRING_POINTER(TREE_VALUE(TREE_VALUE(regarg))));
+    int *mask = NULL;
+    int size = 0;
+    
+    if (INT_REGNO_P(reg))
+    {
+	mask = &cum->int_regmask;
+	size = RS6000_ARG_SIZE(mode, type);
+    }
+    else if (FP_REGNO_P(reg))
+    {
+	mask = &cum->fp_regmask;
+	reg -= 32;
+    }
+    else if (ALTIVEC_REGNO_P(reg))
+    {
+	mask = &cum->altivec_regmask;
+	reg -= FIRST_ALTIVEC_REGNO;
+    }
+    
+    if ((*mask & (1 << reg)) || ((size == 2) && (*mask & (3 << (reg)))))
+    {
+      error("Register %s used twice in function call",
+        TREE_STRING_POINTER(TREE_VALUE(TREE_VALUE(regarg))));
+    }
+    
+    if (size == 2)
+	*mask |= (3 << (reg));
+    else
+	*mask |= (1 << reg);
+  } 
+  else
+
   if (TARGET_ALTIVEC_ABI && ALTIVEC_VECTOR_MODE (mode))
     {
       if (cum->vregno <= ALTIVEC_ARG_MAX_REG && cum->nargs_prototype >= 0)
@@ -2702,6 +2751,9 @@
 {
   enum rs6000_abi abi = DEFAULT_ABI;
 
+  tree regarg;
+  
+
   /* Return a marker to indicate whether CR1 needs to set or clear the
      bit that V.4 uses to say fp args were passed in registers.
      Assume that we don't need the marker for software floating point,
@@ -2722,6 +2774,35 @@
       return GEN_INT (cum->call_cookie);
     }
 
+  /*
+    Test whether there is "regarg" attribute defined for this parameter.
+    if we are called as prototype, rework it to register format, ignore it
+    otherwise.
+  */
+    regarg = lookup_attribute("regarg", TYPE_ATTRIBUTES(type));
+  if (regarg) // && cum->prototype)
+  {
+    int reg = decode_reg_name(TREE_STRING_POINTER(TREE_VALUE(TREE_VALUE(regarg))));
+
+    if (reg < 0)
+    {
+      error("Invalid register name '%s'", 
+    	    TREE_STRING_POINTER(TREE_VALUE(TREE_VALUE(regarg))));
+      return NULL_RTX;
+    }
+    else if (!HARD_REGNO_MODE_OK(reg, mode))
+    {
+      error("Invalid register '%s' for argument", 
+    	    TREE_STRING_POINTER(TREE_VALUE(TREE_VALUE(regarg))));
+      return NULL_RTX;
+    }
+
+    return gen_rtx_REG(mode, reg);
+  }
+
+    if (cum->stackparm)
+	return NULL_RTX;
+
   if (TARGET_ALTIVEC_ABI && ALTIVEC_VECTOR_MODE (mode))
     {
       if (named && cum->vregno <= ALTIVEC_ARG_MAX_REG)
@@ -2902,6 +2983,9 @@
   tree fntype;
   int stdarg_p;
 
+  if (cum->stackparm)
+    return;
+
   fntype = TREE_TYPE (current_function_decl);
   stdarg_p = (TYPE_ARG_TYPES (fntype) != 0
 	      && (TREE_VALUE (tree_last (TYPE_ARG_TYPES (fntype)))
@@ -10740,6 +10824,8 @@
 {
   /* { name, min_len, max_len, decl_req, type_req, fn_type_req, handler } */
   { "longcall", 0, 0, false, true,  true,  rs6000_handle_longcall_attribute },
+  { "stackparm",0, 0, false, true,  true,  rs6000_handle_longcall_attribute },
+  { "regarg",   1, 1, false, false, false, rs6000_handle_register_attribute },
   { NULL,       0, 0, false, false, false, NULL }
 };
 
@@ -10766,6 +10852,78 @@
   return NULL_TREE;
 }
 
+static tree
+rs6000_handle_register_attribute (node, name, args, flags, no_add_attrs)
+     tree *node;
+     tree name;
+     tree args ATTRIBUTE_UNUSED;
+     int flags ATTRIBUTE_UNUSED;
+     bool *no_add_attrs;
+{
+    if (TREE_CODE (*node) != PARM_DECL)
+    {
+       warning ("`%s' attribute only applies to function parameters",
+                IDENTIFIER_POINTER (name));
+       *no_add_attrs = true;
+     }
+   else
+     {
+       tree cst;
+ 
+       cst = TREE_VALUE (args);
+       if (TREE_CODE (cst) != STRING_CST)
+         {
+           error ("`%s' attribute requires a string constant argument specifying a register name",
+                    IDENTIFIER_POINTER (name));
+ 
+           *no_add_attrs = true;
+         }
+       else
+         {
+           /* Ok, this is a PARM_DECL which has a register attribute
+              with the proper argument. However, we need to set the attribute
+              to the DECL's TYPE, rather than to the DECL itself, because
+              the TYPE is what we deal with in function_arg(). The framework
+              doesn't seem to allow for this or, rather, it allows to transfer
+              a DECL's attribute to its TYPE, but the way it does it doesn't let
+              this function know to which DECL that TYPE belongs, and since we
+              only allow "regarg" to be specified for PARM_DECL's, that's not
+              good.
+ 
+              So here we do the same work that decl_attributes() in attribs.c
+              does, only we do it the right way.  */
+ 
+           tree old_attrs;
+           tree a;
+ 
+           old_attrs = TYPE_ATTRIBUTES (TREE_TYPE(*node));
+ 
+           for (a = lookup_attribute ("regarg", old_attrs);
+                a != NULL_TREE;
+                a = lookup_attribute ("regarg", TREE_CHAIN (a)))
+             {
+               if (simple_cst_equal (TREE_VALUE (a), args) == 1)
+                 break;
+             }
+ 
+           if (a == NULL_TREE)
+             {
+               /* This attribute isn't already in the list.  */
+               TREE_TYPE(*node) = build_type_attribute_variant (TREE_TYPE(*node),
+                                                           tree_cons (name, args,
+                                                                     old_attrs));
+             }
+ 
+             /* Now that the attribute has been added to the DECL's TYPE, don't
+                add it to the DECL itself.  */
+ 
+             *no_add_attrs = true;
+         }
+      }
+  
+    return NULL_TREE;
+}
+
 /* Return a reference suitable for calling a function with the
    longcall attribute.  */
 
diff -Naur gcc-3.2.3-org/gcc/config/rs6000/rs6000.h gcc-3.2.3/gcc/config/rs6000/rs6000.h
--- gcc-3.2.3-org/gcc/config/rs6000/rs6000.h	2003-03-29 12:39:20.000000000 +0000
+++ gcc-3.2.3/gcc/config/rs6000/rs6000.h	2003-05-29 21:15:07.000000000 +0000
@@ -1594,6 +1594,10 @@
   int nargs_prototype;		/* # args left in the current prototype */
   int orig_nargs;		/* Original value of nargs_prototype */
   int prototype;		/* Whether a prototype was defined */
+  int stackparm;
+  int int_regmask;
+  int fp_regmask;
+  int altivec_regmask;
   int call_cookie;		/* Do special things for this call */
   int sysv_gregno;		/* next available GP register */
 } CUMULATIVE_ARGS;


[Index of Archives]     [Linux C Programming]     [Linux Kernel]     [eCos]     [Fedora Development]     [Fedora Announce]     [Autoconf]     [The DWARVES Debugging Tools]     [Yosemite Campsites]     [Yosemite News]     [Linux GCC]

  Powered by Linux