2009-05-12 Zoltan Varga <vargaz@gmail.com>
[mono.git] / mono / mini / exceptions-sparc.c
index 0c9a21e276c49f737b714a5990b2f81108b712d6..e721463495cf603d1ebbf0027ab7bcd9f30ac9d5 100644 (file)
@@ -1,5 +1,5 @@
 /*
- * exceptions-sparc.c: exception support for 64 bit sparc
+ * exceptions-sparc.c: exception support for sparc
  *
  * Authors:
  *   Mark Crichton (crichton@gimp.org)
@@ -12,6 +12,7 @@
 #include <glib.h>
 #include <signal.h>
 #include <string.h>
+#include <sys/ucontext.h>
 
 #include <mono/arch/sparc/sparc-codegen.h>
 #include <mono/metadata/appdomain.h>
 #include <mono/metadata/debug-helpers.h>
 #include <mono/metadata/exception.h>
 #include <mono/metadata/mono-debug.h>
+#include <mono/metadata/gc-internal.h>
+#include <mono/metadata/tokentype.h>
 
 #include "mini.h"
 #include "mini-sparc.h"
 
-#warning NotReady
+#ifndef REG_SP
+#define REG_SP REG_O6
+#endif
+
+#define MONO_SPARC_WINDOW_ADDR(sp) ((gpointer*)(((guint8*)(sp)) + MONO_SPARC_STACK_BIAS))
+
+/*
+ * mono_arch_get_restore_context:
+ *
+ * Returns a pointer to a method which restores a previously saved sigcontext.
+ */
+gpointer
+mono_arch_get_restore_context (void)
+{
+       static guint32 *start;
+       static int inited = 0;
+       guint32 *code;
 
-gboolean  mono_arch_handle_exception (struct sigcontext *ctx, gpointer obj, gboolean test_only);
+       if (inited)
+               return start;
 
-typedef struct sigcontext MonoContext;
+       code = start = mono_global_codeman_reserve (32 * sizeof (guint32));
 
-#define MONO_CONTEXT_SET_IP(ctx,ip) do { (ctx)->SC_EIP = (long)ip; } while (0); 
-#define MONO_CONTEXT_SET_BP(ctx,bp) do { (ctx)->SC_EBP = (long)bp; } while (0); 
+       sparc_ldi_imm (code, sparc_o0, G_STRUCT_OFFSET (MonoContext, ip), sparc_i7);
+       sparc_ldi_imm (code, sparc_o0, G_STRUCT_OFFSET (MonoContext, sp), sparc_i6);
 
-#define MONO_CONTEXT_GET_IP(ctx) ((gpointer)((ctx)->SC_EIP))
-#define MONO_CONTEXT_GET_BP(ctx) ((gpointer)((ctx)->SC_EBP))
+       sparc_jmpl_imm (code, sparc_i7, 0, sparc_g0);
+       /* FIXME: This does not return to the correct window */
+       sparc_restore_imm (code, sparc_g0, 0, sparc_g0);
 
-#ifdef MONO_USE_EXC_TABLES
+       g_assert ((code - start) < 32);
 
-/*************************************/
-/*    STACK UNWINDING STUFF          */
-/*************************************/
+       mono_arch_flush_icache ((guint8*)start, (guint8*)code - (guint8*)start);
 
-/* These definitions are from unwind-dw2.c in glibc 2.2.5 */
+       inited = 1;
 
-/* For x86 */
-#define DWARF_FRAME_REGISTERS 17
+       return start;
+}
 
-typedef struct frame_state
+/*
+ * mono_arch_get_call_filter:
+ *
+ * Returns a pointer to a method which calls an exception filter. We
+ * also use this function to call finally handlers (we pass NULL as 
+ * @exc object in this case).
+ *
+ * call_filter (MonoContext *ctx, gpointer ip)
+ */
+gpointer
+mono_arch_get_call_filter (void)
 {
-  void *cfa;
-  void *eh_ptr;
-  long cfa_offset;
-  long args_size;
-  long reg_or_offset[DWARF_FRAME_REGISTERS+1];
-  unsigned short cfa_reg;
-  unsigned short retaddr_column;
-  char saved[DWARF_FRAME_REGISTERS+1];
-} frame_state;
+       static guint32 *start;
+       static int inited = 0;
+       guint32 *code;
+       int i;
 
+       if (inited)
+               return start;
 
-typedef struct frame_state * (*framesf) (void *, struct frame_state *);
+       code = start = mono_global_codeman_reserve (64 * sizeof (guint32));
 
-static framesf frame_state_for = NULL;
+       /*
+        * There are two frames here:
+        * - the first frame is used by call_filter
+        * - the second frame is used to run the filter code
+        */
 
-static gboolean inited = FALSE;
+       /* Create first frame */
+       sparc_save_imm (code, sparc_sp, -256, sparc_sp);
 
-typedef char ** (*get_backtrace_symbols_type) (void *__const *__array, int __size);
+       sparc_mov_reg_reg (code, sparc_i1, sparc_o0);
+       sparc_ldi_imm (code, sparc_i0, G_STRUCT_OFFSET (MonoContext, sp), sparc_o1);
 
-static get_backtrace_symbols_type get_backtrace_symbols = NULL;
+       /* Create second frame */
+       sparc_save_imm (code, sparc_sp, -256, sparc_sp);
 
-static void
-init_frame_state_for (void)
-{
-       GModule *module;
+       sparc_mov_reg_reg (code, sparc_i0, sparc_o0);
+       sparc_mov_reg_reg (code, sparc_i1, sparc_o1);
 
        /*
-        * There are two versions of __frame_state_for: one in libgcc.a and the
-        * other in glibc.so. We need the version from glibc.
-        * For more info, see this:
-        * http://gcc.gnu.org/ml/gcc/2002-08/msg00192.html
+        * We need to change %fp to point to the stack frame of the method
+        * containing the filter. But changing %fp also changes the %sp of
+        * the parent frame (the first frame), so if the OS saves the first frame,
+        * it saves it to the stack frame of the method, which is not good.
+        * So flush all register windows to memory before changing %fp.
         */
-       if ((module = g_module_open ("libc.so.6", G_MODULE_BIND_LAZY))) {
-       
-               if (!g_module_symbol (module, "__frame_state_for", (gpointer*)&frame_state_for))
-                       frame_state_for = NULL;
+       sparc_flushw (code);
 
-               if (!g_module_symbol (module, "backtrace_symbols", (gpointer*)&get_backtrace_symbols)) {
-                       get_backtrace_symbols = NULL;
-                       frame_state_for = NULL;
-               }
+       sparc_mov_reg_reg (code, sparc_fp, sparc_o7);
 
-               g_module_close (module);
+       /* 
+        * Modify the second frame so it is identical to the one used in the
+        * method containing the filter.
+        */
+       for (i = 0; i < 16; ++i)
+               sparc_ldi_imm (code, sparc_o1, MONO_SPARC_STACK_BIAS + i * sizeof (gpointer), sparc_l0 + i);
+
+       /* Save %fp to a location reserved in mono_arch_allocate_vars */
+       sparc_sti_imm (code, sparc_o7, sparc_fp, MONO_SPARC_STACK_BIAS - sizeof (gpointer));
+
+       /* Call the filter code, after this returns, %o0 will hold the result */
+       sparc_call_imm (code, sparc_o0, 0);
+       sparc_nop (code);
+
+       /* Restore original %fp */
+       sparc_ldi_imm (code, sparc_fp, MONO_SPARC_STACK_BIAS - sizeof (gpointer), sparc_fp);
+
+       sparc_mov_reg_reg (code, sparc_o0, sparc_i0);
+
+       /* Return to first frame */
+       sparc_restore (code, sparc_g0, sparc_g0, sparc_g0);
+
+       /* FIXME: Save locals to the stack */
+
+       /* Return to caller */
+       sparc_ret (code);
+       /* Return result in delay slot */
+       sparc_restore (code, sparc_o0, sparc_g0, sparc_o0);
+
+       g_assert ((code - start) < 64);
+
+       mono_arch_flush_icache ((guint8*)start, (guint8*)code - (guint8*)start);
+
+       inited = 1;
+
+       return start;
+}
+
+static void
+throw_exception (MonoObject *exc, gpointer sp, gpointer ip, gboolean rethrow)
+{
+       MonoContext ctx;
+       static void (*restore_context) (MonoContext *);
+       gpointer *window;
+       
+       if (!restore_context)
+               restore_context = mono_arch_get_restore_context ();
+
+       window = MONO_SPARC_WINDOW_ADDR (sp);
+       ctx.sp = (gpointer*)sp;
+       ctx.ip = ip;
+       ctx.fp = (gpointer*)(MONO_SPARC_WINDOW_ADDR (sp) [sparc_i6 - 16]);
+
+       if (mono_object_isinst (exc, mono_defaults.exception_class)) {
+               MonoException *mono_ex = (MonoException*)exc;
+               if (!rethrow)
+                       mono_ex->stack_trace = NULL;
        }
+       mono_handle_exception (&ctx, exc, ip, FALSE);
+       restore_context (&ctx);
 
-       inited = TRUE;
+       g_assert_not_reached ();
 }
 
-/* mono_arch_has_unwind_info:
+static gpointer 
+get_throw_exception (gboolean rethrow)
+{
+       guint32 *start, *code;
+
+       code = start = mono_global_codeman_reserve (16 * sizeof (guint32));
+
+       sparc_save_imm (code, sparc_sp, -512, sparc_sp);
+
+       sparc_flushw (code);
+       sparc_mov_reg_reg (code, sparc_i0, sparc_o0);
+       sparc_mov_reg_reg (code, sparc_fp, sparc_o1);
+       sparc_mov_reg_reg (code, sparc_i7, sparc_o2);
+       sparc_set (code, rethrow, sparc_o3);
+       sparc_set (code, throw_exception, sparc_o7);
+       sparc_jmpl (code, sparc_o7, sparc_g0, sparc_callsite);
+       sparc_nop (code);
+
+       g_assert ((code - start) <= 16);
+
+       mono_arch_flush_icache ((guint8*)start, (guint8*)code - (guint8*)start);
+
+       return start;
+}
+
+/**
+ * mono_arch_get_throw_exception:
  *
- * Tests if a function has an DWARF exception table able to restore
- * all caller saved registers. 
+ * Returns a function pointer which can be used to raise exceptions.
+ * The returned function has the following 
+ * signature: void (*func) (MonoException *exc); 
  */
-gboolean
-mono_arch_has_unwind_info (gconstpointer addr)
+gpointer 
+mono_arch_get_throw_exception (void)
 {
-       return FALSE;
+       static guint32* start;
+       static int inited = 0;
+
+       if (inited)
+               return start;
+
+       inited = 1;
+
+       start = get_throw_exception (FALSE);
+
+       return start;
 }
 
-struct stack_frame
+gpointer 
+mono_arch_get_rethrow_exception (void)
 {
-  void *next;
-  void *return_address;
-};
-#endif
+       static guint32* start;
+       static int inited = 0;
+
+       if (inited)
+               return start;
+
+       inited = 1;
+
+       start = get_throw_exception (TRUE);
 
+       return start;
+}
+
+/**
+ * mono_arch_get_throw_exception_by_name:
+ *
+ * Returns a function pointer which can be used to raise 
+ * corlib exceptions. The returned function has the following 
+ * signature: void (*func) (char *exc_name, gpointer ip); 
+ */
 gpointer 
-mono_arch_get_throw_exception (void)
+mono_arch_get_throw_exception_by_name (void)
 {
-       return 0xdeadbeef;
+       static guint32 *start;
+       static int inited = 0;
+       guint32 *code;
+       int reg;
+
+       if (inited)
+               return start;
+
+       inited = 1;
+       code = start = mono_global_codeman_reserve (64 * sizeof (guint32));
+
+#ifdef SPARCV9
+       reg = sparc_g4;
+#else
+       reg = sparc_g1;
+#endif
+
+       sparc_save_imm (code, sparc_sp, -160, sparc_sp);
+
+       sparc_mov_reg_reg (code, sparc_i0, sparc_o2);
+       sparc_set (code, mono_defaults.corlib, sparc_o0);
+       sparc_set (code, "System", sparc_o1);
+       sparc_set (code, mono_exception_from_name, sparc_o7);
+       sparc_jmpl (code, sparc_o7, sparc_g0, sparc_callsite);
+       sparc_nop (code);
+
+       /* Return to the caller, so exception handling does not see this frame */
+       sparc_restore (code, sparc_o0, sparc_g0, sparc_o0);
+
+       /* Put original return address into %o7 */
+       sparc_mov_reg_reg (code, sparc_o1, sparc_o7);
+       sparc_set (code, mono_arch_get_throw_exception (), reg);
+       /* Use a jmp instead of a call so o7 is preserved */
+       sparc_jmpl_imm (code, reg, 0, sparc_g0);
+       sparc_nop (code);
+
+       g_assert ((code - start) < 32);
+
+       mono_arch_flush_icache ((guint8*)start, (guint8*)code - (guint8*)start);
+
+       return start;
 }
 
 /**
- * arch_get_throw_exception_by_name:
+ * mono_arch_get_throw_corlib_exception:
  *
  * Returns a function pointer which can be used to raise 
  * corlib exceptions. The returned function has the following 
- * signature: void (*func) (char *exc_name); 
- * For example to raise an arithmetic exception you can use:
- *
- * x86_push_imm (code, "ArithmeticException"); 
- * x86_call_code (code, arch_get_throw_exception_by_name ()); 
- *
+ * signature: void (*func) (guint32 ex_token, guint32 offset); 
+ * Here, offset is the offset which needs to be substracted from the caller IP 
+ * to get the IP of the throw. Passing the offset has the advantage that it 
+ * needs no relocations in the caller.
  */
 gpointer 
-mono_arch_get_throw_exception_by_name (void)
+mono_arch_get_throw_corlib_exception (void)
 {
-       return 0xdecafbad;
-}      
+       static guint32 *start;
+       static int inited = 0;
+       guint32 *code;
+       int reg;
+
+       if (inited)
+               return start;
+
+       inited = 1;
+       code = start = mono_global_codeman_reserve (64 * sizeof (guint32));
+
+#ifdef SPARCV9
+       reg = sparc_g4;
+#else
+       reg = sparc_g1;
+#endif
+
+       sparc_mov_reg_reg (code, sparc_o7, sparc_o2);
+       sparc_save_imm (code, sparc_sp, -160, sparc_sp);
+
+       sparc_set (code, MONO_TOKEN_TYPE_DEF, sparc_o7);
+       sparc_add (code, FALSE, sparc_i0, sparc_o7, sparc_o1);
+       sparc_set (code, mono_defaults.exception_class->image, sparc_o0);
+       sparc_set (code, mono_exception_from_token, sparc_o7);
+       sparc_jmpl (code, sparc_o7, sparc_g0, sparc_callsite);
+       sparc_nop (code);
+
+       /* Return to the caller, so exception handling does not see this frame */
+       sparc_restore (code, sparc_o0, sparc_g0, sparc_o0);
+
+       /* Compute throw ip */
+       sparc_sll_imm (code, sparc_o1, 2, sparc_o1);
+       sparc_sub (code, 0, sparc_o2, sparc_o1, sparc_o7);
 
-static MonoArray *
-glist_to_array (GList *list) 
+       sparc_set (code, mono_arch_get_throw_exception (), reg);
+       /* Use a jmp instead of a call so o7 is preserved */
+       sparc_jmpl_imm (code, reg, 0, sparc_g0);
+       sparc_nop (code);
+
+       g_assert ((code - start) < 32);
+
+       mono_arch_flush_icache ((guint8*)start, (guint8*)code - (guint8*)start);
+
+       return start;
+}
+
+/* mono_arch_find_jit_info:
+ *
+ * This function is used to gather information from @ctx. It return the 
+ * MonoJitInfo of the corresponding function, unwinds one stack frame and
+ * stores the resulting context into @new_ctx. It also stores a string 
+ * describing the stack location into @trace (if not NULL), and modifies
+ * the @lmf if necessary. @native_offset return the IP offset from the 
+ * start of the function or -1 if that info is not available.
+ */
+MonoJitInfo *
+mono_arch_find_jit_info (MonoDomain *domain, MonoJitTlsData *jit_tls, MonoJitInfo *res, MonoJitInfo *prev_ji, MonoContext *ctx, 
+                        MonoContext *new_ctx, MonoLMF **lmf, gboolean *managed)
 {
-       MonoDomain *domain = mono_domain_get ();
-       MonoArray *res;
-       int len, i;
+       MonoJitInfo *ji;
+       gpointer ip = MONO_CONTEXT_GET_IP (ctx);
+       gpointer *window;
+
+       /* Avoid costly table lookup during stack overflow */
+       if (prev_ji && (ip > prev_ji->code_start && ((guint8*)ip < ((guint8*)prev_ji->code_start) + prev_ji->code_size)))
+               ji = prev_ji;
+       else
+               ji = mono_jit_info_table_find (domain, ip);
+
+       if (managed)
+               *managed = FALSE;
+
+       if (ji != NULL) {
+               *new_ctx = *ctx;
+
+               if (managed)
+                       if (!ji->method->wrapper_type)
+                               *managed = TRUE;
+
+               if (*lmf && (MONO_CONTEXT_GET_BP (ctx) >= (gpointer)(*lmf)->ebp)) {
+                       /* remove any unused lmf */
+                       *lmf = (*lmf)->previous_lmf;
+               }
+
+               /* Restore ip and sp from the saved register window */
+               window = MONO_SPARC_WINDOW_ADDR (ctx->sp);
+               new_ctx->ip = window [sparc_i7 - 16];
+               new_ctx->sp = (gpointer*)(window [sparc_i6 - 16]);
+               new_ctx->fp = (gpointer*)(MONO_SPARC_WINDOW_ADDR (new_ctx->sp) [sparc_i6 - 16]);
+
+               return ji;
+       }
+       else {
+               if (!(*lmf))
+                       return NULL;
+
+               *new_ctx = *ctx;
+
+               if (!(*lmf)->method)
+                       return (gpointer)-1;
 
-       if (!list)
-               return NULL;
+               if ((ji = mono_jit_info_table_find (domain, (gpointer)(*lmf)->ip))) {
+               } else {
+                       memset (res, 0, sizeof (MonoJitInfo));
+                       res->method = (*lmf)->method;
+               }
 
-       len = g_list_length (list);
-       res = mono_array_new (domain, mono_defaults.int_class, len);
+               new_ctx->ip = (*lmf)->ip;
+               new_ctx->sp = (*lmf)->sp;
+               new_ctx->fp = (*lmf)->ebp;
 
-       for (i = 0; list; list = list->next, i++)
-               mono_array_set (res, gpointer, i, list->data);
+               *lmf = (*lmf)->previous_lmf;
 
-       return res;
+               return ji ? ji : res;
+       }
 }
 
-MonoArray *
-ves_icall_get_trace (MonoException *exc, gint32 skip, MonoBoolean need_file_info)
+gboolean
+mono_arch_has_unwind_info (gconstpointer addr)
 {
-       return NULL;
+       return FALSE;
 }
 
-void
-mono_jit_walk_stack (MonoStackWalk func, gpointer user_data) {
+#ifdef __linux__
+
+gboolean
+mono_arch_handle_exception (void *sigctx, gpointer obj, gboolean test_only)
+{
+       MonoContext mctx;
+       struct sigcontext *sc = sigctx;
+       gpointer *window;
+
+#ifdef SPARCV9
+       mctx.ip = (gpointer) sc->sigc_regs.tpc;
+       mctx.sp = (gpointer) sc->sigc_regs.u_regs[14];
+#else
+       mctx.ip = (gpointer) sc->si_regs.pc;
+       mctx.sp = (gpointer) sc->si_regs.u_regs[14];
+#endif
+
+       window = (gpointer*)(((guint8*)mctx.sp) + MONO_SPARC_STACK_BIAS);
+       mctx.fp = window [sparc_fp - 16];
+
+       mono_handle_exception (&mctx, obj, mctx.ip, test_only);
+
+#ifdef SPARCV9
+       sc->sigc_regs.tpc = (unsigned long) mctx.ip;
+       sc->sigc_regs.tnpc = (unsigned long) (mctx.ip + 4);
+       sc->sigc_regs.u_regs[14] = (unsigned long) mctx.sp;
+#else
+       sc->si_regs.pc = (unsigned long) mctx.ip;
+       sc->si_regs.npc = (unsigned long) (mctx.ip + 4);
+       sc->si_regs.u_regs[14] = (unsigned long) mctx.sp;
+#endif
+
+       window = (gpointer*)(((guint8*)mctx.sp) + MONO_SPARC_STACK_BIAS);
+       window [sparc_fp - 16] = mctx.fp;
+
+       return TRUE;
 }
 
-MonoBoolean
-ves_icall_get_frame_info (gint32 skip, MonoBoolean need_file_info, 
-                         MonoReflectionMethod **method, 
-                         gint32 *iloffset, gint32 *native_offset,
-                         MonoString **file, gint32 *line, gint32 *column)
+gpointer
+mono_arch_ip_from_context (void *sigctx)
 {
-       return FALSE;
+       struct sigcontext *sc = sigctx;
+       gpointer *ret;
+
+#ifdef SPARCV9
+       ret = (gpointer) sc->sigc_regs.tpc;
+#else
+       ret = (gpointer) sc->si_regs.pc;
+#endif
+
+       return ret;
 }
 
-/**
- * arch_handle_exception:
- * @ctx: saved processor state
- * @obj: the exception object
- * @test_only: only test if the exception is caught, but dont call handlers
- *
- *
- */
+#else /* !__linux__ */
+
 gboolean
-mono_arch_handle_exception (MonoContext *ctx, gpointer obj, gboolean test_only)
+mono_arch_handle_exception (void *sigctx, gpointer obj, gboolean test_only)
 {
-       g_assert_not_reached ();
+       MonoContext mctx;
+       ucontext_t *ctx = (ucontext_t*)sigctx;
+       gpointer *window;
+
+       /*
+        * Access to the machine state using the ucontext_t parameter is somewhat
+        * under documented under solaris. The code below seems to work under
+        * Solaris 9.
+        */
+       g_assert (!ctx->uc_mcontext.gwins);
+
+       mctx.ip = ctx->uc_mcontext.gregs [REG_PC];
+       mctx.sp = ctx->uc_mcontext.gregs [REG_SP];
+       window = (gpointer*)(((guint8*)mctx.sp) + MONO_SPARC_STACK_BIAS);
+       mctx.fp = window [sparc_fp - 16];
+
+       mono_handle_exception (&mctx, obj, mctx.ip, test_only);
+       
+       /* We can't use restore_context to return from a signal handler */
+       ctx->uc_mcontext.gregs [REG_PC] = mctx.ip;
+       ctx->uc_mcontext.gregs [REG_nPC] = mctx.ip + 4;
+       ctx->uc_mcontext.gregs [REG_SP] = mctx.sp;
+       window = (gpointer*)(((guint8*)mctx.sp) + MONO_SPARC_STACK_BIAS);
+       window [sparc_fp - 16] = mctx.fp;
+
+       return TRUE;
 }
+
+gpointer
+mono_arch_ip_from_context (void *sigctx)
+{
+       ucontext_t *ctx = (ucontext_t*)sigctx;
+       return (gpointer)ctx->uc_mcontext.gregs [REG_PC];
+}
+
+#endif