* Removed all Id tags.
[cacao.git] / src / vm / jit / mips / md-abi.c
index d539eef67d2d48ac43689e3a0a547f446da2cf15..1b3065edf3b80ce477aa745496fc87d644b46d82 100644 (file)
    Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
    02110-1301, USA.
 
-   $Id: md-abi.c 7713 2007-04-15 21:49:48Z twisti $
-
 */
 
 
 #include "config.h"
+
+#include <stdarg.h>
+
 #include "vm/types.h"
 
 #include "vm/jit/mips/md-abi.h"
 
+#include "mm/memory.h"
+
 #include "vm/global.h"
 
 #include "vm/jit/abi.h"
 
 #include "vmcore/descriptor.h"
+#include "vmcore/method.h"
 
 
 /* register descripton array **************************************************/
@@ -264,31 +268,24 @@ void md_param_alloc(methoddesc *md)
        s4         i;
        s4         reguse;
        s4         stacksize;
-#if SIZEOF_VOID_P == 4 && !defined(ENABLE_SOFT_FLOAT)
-       s4         t;
-       bool       a0_is_float;
-#endif
 
        /* set default values */
 
        reguse      = 0;
        stacksize   = 0;
-#if SIZEOF_VOID_P == 4 && !defined(ENABLE_SOFT_FLOAT)
-       a0_is_float = false;
-#endif
 
        /* get params field of methoddesc */
 
        pd = md->params;
 
-       for (i = 0; i < md->paramcount; i++, pd++) {
 #if SIZEOF_VOID_P == 8
 
+       for (i = 0; i < md->paramcount; i++, pd++) {
                switch (md->paramtypes[i].type) {
                case TYPE_INT:
                case TYPE_ADR:
                case TYPE_LNG:
-                       if (i < INT_ARG_CNT) {
+                       if (reguse < INT_ARG_CNT) {
                                pd->inmemory = false;
                                pd->regoff   = abi_registers_integer_argument[reguse];
                                reguse++;
@@ -296,13 +293,14 @@ void md_param_alloc(methoddesc *md)
                        }
                        else {
                                pd->inmemory = true;
-                               pd->regoff   = stacksize;
+                               pd->regoff   = stacksize * 8;
                                stacksize++;
                        }
                        break;
+
                case TYPE_FLT:
                case TYPE_DBL:
-                       if (i < FLT_ARG_CNT) {
+                       if (reguse < FLT_ARG_CNT) {
                                pd->inmemory = false;
                                pd->regoff   = abi_registers_float_argument[reguse];
                                reguse++;
@@ -310,7 +308,7 @@ void md_param_alloc(methoddesc *md)
                        }
                        else {
                                pd->inmemory = true;
-                               pd->regoff   = stacksize;
+                               pd->regoff   = stacksize * 8;
                                stacksize++;
                        }
                        break;
@@ -319,13 +317,187 @@ void md_param_alloc(methoddesc *md)
                /* register type is the same as java type */
 
                pd->type = md->paramtypes[i].type;
+       }
 
 #else /* SIZEOF_VOID_P == 8 */
 
-#if !defined(ENABLE_SOFT_FLOAT)
+# if !defined(ENABLE_SOFT_FLOAT)
+
+       /* Set stacksize to 2, as 4 32-bit argument registers can be
+          stored. */
+       /* XXX maybe this should be done in stack.c? */
+
+       stacksize = 2;
+
+       for (i = 0; i < md->paramcount; i++, pd++) {
+               switch (md->paramtypes[i].type) {
+               case TYPE_INT:
+               case TYPE_ADR:
+                       if (reguse < INT_ARG_CNT) {
+                               pd->inmemory = false;
+                               pd->index    = reguse;
+                               pd->regoff   = abi_registers_integer_argument[reguse];
+                               reguse++;
+                               md->argintreguse = reguse;
+                       }
+                       else {
+                               pd->inmemory = true;
+                               pd->index    = stacksize;
+                               pd->regoff   = stacksize * 8;
+                               stacksize++;
+                       }
+                       break;
+
+               case TYPE_LNG:
+                       ALIGN_2(reguse);
+
+                       if (reguse < INT_ARG_CNT) {
+                               pd->inmemory = false;
+#  if WORDS_BIGENDIAN == 1
+                               pd->index    = PACK_REGS(reguse + 1, reguse);
+                               pd->regoff   =
+                                       PACK_REGS(abi_registers_integer_argument[reguse + 1],
+                                                         abi_registers_integer_argument[reguse]);
+#  else
+                               pd->index    = PACK_REGS(reguse, reguse + 1);
+                               pd->regoff   =
+                                       PACK_REGS(abi_registers_integer_argument[reguse],
+                                                         abi_registers_integer_argument[reguse + 1]);
+#  endif
+                               reguse += 2;
+                               md->argintreguse = reguse;
+                       }
+                       else {
+                               pd->inmemory = true;
+                               pd->index    = stacksize;
+                               pd->regoff   = stacksize * 8;
+                               stacksize++;
+                       }
+                       break;
+
+               case TYPE_FLT:
+               case TYPE_DBL:
+                       if (reguse < FLT_ARG_CNT) {
+                               pd->inmemory = false;
+                               pd->index    = reguse;
+                               pd->regoff   = abi_registers_float_argument[reguse];
+                               reguse++;
+                               md->argfltreguse = reguse;
+                       }
+                       else {
+                               pd->inmemory = true;
+                               pd->index    = stacksize;
+                               pd->regoff   = stacksize * 8;
+                               stacksize++;
+                       }
+                       break;
+               }
+
+               /* register type is the same as java type */
+
+               pd->type = md->paramtypes[i].type;
+       }
+
+# else /* !defined(ENABLE_SOFT_FLOAT) */
+#  error never actually tested!
+
+       for (i = 0; i < md->paramcount; i++, pd++) {
+               switch (md->paramtypes[i].type) {
+               case TYPE_INT:
+               case TYPE_ADR:
+               case TYPE_FLT:
+                       pd->type = TYPE_INT;
+
+                       if (reguse < INT_ARG_CNT) {
+                               pd->inmemory = false;
+                               pd->regoff   = abi_registers_integer_argument[reguse];
+                               reguse++;
+                               md->argintreguse = reguse;
+                       }
+                       else {
+                               pd->inmemory = true;
+                               pd->regoff   = stacksize * 8;
+                       }
+                       stacksize++;
+                       break;
+
+               case TYPE_LNG:
+               case TYPE_DBL:
+                       pd->type = TYPE_LNG;
 
-#define ALIGN_2_WORD(s)    ((s) & 1) ? ++(s) : (s)
+                       if (reguse < INT_ARG_CNT) {
+                               pd->inmemory = false;
+#  if WORDS_BIGENDIAN == 1
+                               pd->regoff   =
+                                       PACK_REGS(abi_registers_integer_argument[reguse + 1],
+                                                         abi_registers_integer_argument[reguse]);
+#  else
+                               pd->regoff   =
+                                       PACK_REGS(abi_registers_integer_argument[reguse],
+                                                         abi_registers_integer_argument[reguse + 1]);
+#  endif
+                               reguse += 2;
+                               md->argintreguse = reguse;
+                       }
+                       else {
+                               pd->inmemory = true;
+                               pd->regoff   = stacksize * 8;
+                       }
+                       stacksize += 2;
+                       break;
+               }
+       }
 
+# endif /* !defined(ENABLE_SOFT_FLOAT) */
+#endif /* SIZEOF_VOID_P == 8 */
+
+       /* fill register and stack usage */
+
+       md->memuse = stacksize;
+}
+
+
+/* md_param_alloc_native *******************************************************
+
+   Pre-allocate arguments according the native ABI.
+
+*******************************************************************************/
+
+void md_param_alloc_native(methoddesc *md)
+{
+#if SIZEOF_VOID_P == 8
+
+       /* On MIPS n64 we use the same ABI for JIT method calls as for
+          native method calls. */
+
+       md_param_alloc(md);
+
+#else /* SIZEOF_VOID_P == 8 */
+
+       paramdesc *pd;
+       s4         i;
+       s4         reguse;
+       s4         stacksize;
+# if !defined(ENABLE_SOFT_FLOAT)
+       s4         t;
+       bool       a0_is_float;
+# endif
+
+       /* set default values */
+
+       reguse      = 0;
+       stacksize   = 0;
+# if !defined(ENABLE_SOFT_FLOAT)
+       a0_is_float = false;
+# endif
+
+       /* get params field of methoddesc */
+
+       pd = md->params;
+
+# if !defined(ENABLE_SOFT_FLOAT)
+
+       for (i = 0; i < md->paramcount; i++, pd++) {
                t = md->paramtypes[i].type;
 
                if (IS_FLT_DBL_TYPE(t) &&
@@ -348,26 +520,28 @@ void md_param_alloc(methoddesc *md)
                }
                else {
                        if (IS_2_WORD_TYPE(t)) {
-                               ALIGN_2_WORD(reguse);
+                               ALIGN_2(reguse);
                                pd->type = TYPE_LNG;
 
                                if (reguse < INT_ARG_CNT) {
                                        pd->inmemory = false;
-# if WORDS_BIGENDIAN == 1
+#  if WORDS_BIGENDIAN == 1
                                        pd->regoff   =
                                                PACK_REGS(abi_registers_integer_argument[reguse + 1],
                                                                  abi_registers_integer_argument[reguse]);
-# else
+#  else
                                        pd->regoff   =
                                                PACK_REGS(abi_registers_integer_argument[reguse],
                                                                  abi_registers_integer_argument[reguse + 1]);
-# endif
+#  endif
                                        reguse += 2;
                                        md->argintreguse = reguse;
                                }
                                else {
+                                       ALIGN_2(stacksize);
+
                                        pd->inmemory = true;
-                                       pd->regoff   = ALIGN_2_WORD(stacksize);
+                                       pd->regoff   = stacksize * 4;
                                }
                                stacksize += 2;
                        }
@@ -382,15 +556,17 @@ void md_param_alloc(methoddesc *md)
                                }
                                else {
                                        pd->inmemory = true;
-                                       pd->regoff   = stacksize;
+                                       pd->regoff   = stacksize * 4;
                                }
                                stacksize++;
                        }
                }
+       }
 
-#else /* !defined(ENABLE_SOFT_FLOAT) */
-#error never actually tested!
+# else /* !defined(ENABLE_SOFT_FLOAT) */
+#  error never actually tested!
 
+       for (i = 0; i < md->paramcount; i++, pd++) {
                switch (md->paramtypes[i].type) {
                case TYPE_INT:
                case TYPE_ADR:
@@ -405,7 +581,7 @@ void md_param_alloc(methoddesc *md)
                        }
                        else {
                                pd->inmemory = true;
-                               pd->regoff   = stacksize;
+                               pd->regoff   = stacksize * 4;
                        }
                        stacksize++;
                        break;
@@ -415,50 +591,34 @@ void md_param_alloc(methoddesc *md)
 
                        if (i < INT_ARG_CNT) {
                                pd->inmemory = false;
-#if WORDS_BIGENDIAN == 1
+#  if WORDS_BIGENDIAN == 1
                                pd->regoff   =
                                        PACK_REGS(abi_registers_integer_argument[reguse + 1],
                                                          abi_registers_integer_argument[reguse]);
-#else
+#  else
                                pd->regoff   =
                                        PACK_REGS(abi_registers_integer_argument[reguse],
                                                          abi_registers_integer_argument[reguse + 1]);
-#endif
+#  endif
                                reguse += 2;
                                md->argintreguse = reguse;
                        }
                        else {
                                pd->inmemory = true;
-                               pd->regoff   = stacksize;
+                               pd->regoff   = stacksize * 4;
                        }
                        stacksize += 2;
                        break;
                }
-
-
-#endif /* !defined(ENABLE_SOFT_FLOAT) */
-
-#endif /* SIZEOF_VOID_P == 8 */
        }
 
+# endif /* !defined(ENABLE_SOFT_FLOAT) */
+
        /* fill register and stack usage */
 
        md->memuse = stacksize;
-}
-
 
-/* md_param_alloc_native *******************************************************
-
-   Pre-allocate arguments according the native ABI.
-
-*******************************************************************************/
-
-void md_param_alloc_native(methoddesc *md)
-{
-       /* On MIPS we use the same ABI for JIT method calls as for native
-          method calls. */
-
-       md_param_alloc(md);
+#endif /* SIZEOF_VOID_P == 8 */
 }