/* * mini-mips.c: MIPS backend for the Mono code generator * * Authors: * Mark Mason (mason@broadcom.com) * * Based on mini-ppc.c by * Paolo Molaro (lupus@ximian.com) * Dietmar Maurer (dietmar@ximian.com) * * (C) 2006 Broadcom * (C) 2003 Ximian, Inc. */ #include "mini.h" #include #include #include #include #include "mini-mips.h" #include "inssel.h" #include "cpu-mips.h" #include "trace.h" #undef SAVE_FP_REGS #define SAVE_LMF #define ALWAYS_USE_FP 1 #define ALWAYS_SAVE_RA 1 /* call-handler & switch currently clobber ra */ enum { TLS_MODE_DETECT, TLS_MODE_FAILED, TLS_MODE_LTHREADS, TLS_MODE_NPTL }; int mono_exc_esp_offset = 0; static int tls_mode = TLS_MODE_DETECT; static int lmf_pthread_key = -1; static int monothread_key = -1; static int monodomain_key = -1; #undef DEBUG #define DEBUG(a) if (cfg->verbose_level > 1) a #undef DEBUG #define DEBUG(a) a #undef DEBUG #define DEBUG(a) #define NOT_IMPLEMENTED(x) \ g_error ("FIXME: %s is not yet implemented. (trampoline)", x); /* emit an exception if condition is failed * * We assign the extra code used to throw the implicit exceptions * to cfg->bb_exit as far as the big branch handling is concerned */ #define EMIT_SYSTEM_EXCEPTION_NAME_LEN 5 #define EMIT_SYSTEM_EXCEPTION_NAME(exc_name) \ do { \ mono_add_patch_info (cfg, code - cfg->native_code, \ MONO_PATCH_INFO_EXC, exc_name); \ mips_lui (code, mips_t9, mips_zero, 0); \ mips_addiu (code, mips_t9, mips_t9, 0); \ mips_jalr (code, mips_t9, mips_ra); \ mips_nop (code); \ cfg->bb_exit->max_offset += 16; \ } while (0) /* XXX - need to actually test the condition - not just bne */ #define EMIT_COND_SYSTEM_EXCEPTION_FLAGS(b0,b1,exc_name) \ do { \ mips_bne (code, mips_at, mips_zero, 5); \ mips_nop (code); \ EMIT_SYSTEM_EXCEPTION_NAME(exc_name); \ cfg->bb_exit->max_offset += 8; \ } while (0) #define EMIT_COND_SYSTEM_EXCEPTION(cond,exc_name) \ do { \ switch (cond) { \ case CEE_BEQ: \ mips_bne (code, mips_at, mips_zero, 5); \ break; \ default: \ g_assert_not_reached(); \ } \ mips_nop (code); \ EMIT_SYSTEM_EXCEPTION_NAME(exc_name); \ cfg->bb_exit->max_offset += 8; \ } while (0) #define emit_linuxthreads_tls(code,dreg,key) do {\ int off1, off2; \ off1 = offsets_from_pthread_key ((key), &off2); \ ppc_lwz ((code), (dreg), off1, ppc_r2); \ ppc_lwz ((code), (dreg), off2, (dreg)); \ } while (0); #define emit_tls_access(code,dreg,key) do { \ switch (tls_mode) { \ case TLS_MODE_LTHREADS: emit_linuxthreads_tls(code,dreg,key); break; \ default: g_assert_not_reached (); \ } \ } while (0) typedef struct InstList InstList; struct InstList { InstList *prev; InstList *next; MonoInst *data; }; #define ALWAYS_ON_STACK(s) s #define FP_ALSO_IN_REG(s) s #define ALIGN_DOUBLES enum { RegTypeGeneral, RegTypeBase, RegTypeFP, RegTypeStructByVal, RegTypeStructByAddr }; typedef struct { gint32 offset; guint16 vtsize; /* in param area */ guint8 reg; guint8 regtype : 4; /* 0 general, 1 basereg, 2 floating point register, see RegType* */ guint8 size : 4; /* 1, 2, 4, 8, or regs used by RegTypeStructByVal */ } ArgInfo; typedef struct { int nargs; int gr; int fr; int gr_passed; int fr_passed; gboolean on_stack; int stack_size; guint32 stack_usage; guint32 struct_ret; ArgInfo ret; ArgInfo sig_cookie; ArgInfo args [1]; } CallInfo; static const char*const * ins_spec = mips_desc; void patch_lui_addiu(guint32 *ip, guint32 val); void mono_arch_flush_icache (guint8 *code, gint size) { /* Linux/MIPS specific */ cacheflush (code, size, BCACHE); } #ifndef CUSTOM_STACK_WALK void mono_arch_flush_register_windows (void) { } #endif /* XXX - big-endian dependent? */ void patch_lui_addiu(guint32 *ip, guint32 val) { guint16 *__lui_addiu = (guint16*)(void *)(ip); #if 0 printf ("patch_lui_addiu ip=0x%08x (0x%08x, 0x%08x) to point to 0x%08x\n", ip, ((guint32 *)ip)[0], ((guint32 *)ip)[1], val); fflush (stdout); #endif if (((guint32)(val)) & (1 << 15)) __lui_addiu [1] = ((((guint32)(val)) >> 16) & 0xffff) + 1; else __lui_addiu [1] = (((guint32)(val)) >> 16) & 0xffff; __lui_addiu [3] = ((guint32)(val)) & 0xffff; mono_arch_flush_icache ((guint8 *)ip, 8); } void mips_patch (guint32 *code, guint32 target) { guint32 ins = *code; guint32 op = ins >> 26; guint32 diff, offset; //printf ("patching 0x%08x (0x%08x) to point to 0x%08x\n", code, ins, target); switch (op) { case 0x00: /* jr ra */ if (ins == 0x3e00008) break; g_assert_not_reached (); break; case 0x02: /* j */ case 0x03: /* jal */ ins = (ins & 0xfc000000) | (((target) >> 2) & 0x03ffffff); *code = ins; mono_arch_flush_icache ((guint8 *)code, 4); break; case 0x01: /* BLTZ */ case 0x04: /* BEQ */ case 0x05: /* BNE */ case 0x06: /* BLEZ */ case 0x07: /* BGTZ */ case 0x11: /* bc1t */ diff = target - (guint32)(code + 1); offset = diff >> 2; ins = (ins & 0xffff0000) | (offset & 0x0000ffff); *code = ins; mono_arch_flush_icache ((guint8 *)code, 4); break; case 0x0f: /* LUI / ADDIU pair */ patch_lui_addiu (code, target); mono_arch_flush_icache ((guint8 *)code, 8); break; default: printf ("unknown op 0x%02x (0x%08x) @ %p\n", op, ins, code); g_assert_not_reached (); } } static int offsets_from_pthread_key (guint32 key, int *offset2) { int idx1 = key / 32; int idx2 = key % 32; *offset2 = idx2 * sizeof (gpointer); return 284 + idx1 * sizeof (gpointer); } const char* mono_arch_regname (int reg) { static const char * rnames[] = { "zero", "at", "v0", "v1", "a0", "a1", "a2", "a3", "t0", "t1", "t2", "t3", "t4", "t5", "t6", "t7", "s0", "s1", "s2", "s3", "s4", "s5", "s6", "s7", "t8", "t9", "k0", "k1", "gp", "sp", "fp", "ra" }; if (reg >= 0 && reg < 32) return rnames [reg]; return "unknown"; } const char* mono_arch_fregname (int reg) { static const char * rnames[] = { "f0", "f1", "f2", "f3", "f4", "f5", "f6", "f7", "f8", "f9", "f10", "f11", "f12", "f13", "f14", "f15", "f16", "f17", "f18", "f19", "f20", "f21", "f22", "f23", "f24", "f25", "f26", "f27", "f28", "f29", "f30", "f31" }; if (reg >= 0 && reg < 32) return rnames [reg]; return "unknown"; } /* this function overwrites at */ static guint8* emit_memcpy (guint8 *code, int size, int dreg, int doffset, int sreg, int soffset) { /* XXX write a loop, not an unrolled loop */ while (size > 0) { mips_lw (code, mips_at, sreg, soffset); mips_sw (code, mips_at, dreg, doffset); size -= 4; soffset += 4; doffset += 4; } return code; } /* * mono_arch_get_argument_info: * @csig: a method signature * @param_count: the number of parameters to consider * @arg_info: an array to store the result infos * * Gathers information on parameters such as size, alignment and * padding. arg_info should be large enought to hold param_count + 1 entries. * * Returns the size of the activation frame. */ int mono_arch_get_argument_info (MonoMethodSignature *csig, int param_count, MonoJitArgumentInfo *arg_info) { int k, frame_size = 0; guint32 size, align, pad; int offset = 0; if (MONO_TYPE_ISSTRUCT (csig->ret)) { frame_size += sizeof (gpointer); offset += 4; } arg_info [0].offset = offset; if (csig->hasthis) { frame_size += sizeof (gpointer); offset += 4; } arg_info [0].size = frame_size; for (k = 0; k < param_count; k++) { if (csig->pinvoke) size = mono_type_native_stack_size (csig->params [k], &align); else size = mono_type_stack_size (csig->params [k], &align); /* ignore alignment for now */ align = 1; frame_size += pad = (align - (frame_size & (align - 1))) & (align - 1); arg_info [k].pad = pad; frame_size += size; arg_info [k + 1].pad = 0; arg_info [k + 1].size = size; offset += pad; arg_info [k + 1].offset = offset; offset += size; } align = MONO_ARCH_FRAME_ALIGNMENT; frame_size += pad = (align - (frame_size & (align - 1))) & (align - 1); arg_info [k].pad = pad; return frame_size; } /* * Initialize the cpu to execute managed code. */ void mono_arch_cpu_init (void) { } /* * This function returns the optimizations supported on this cpu. */ guint32 mono_arch_cpu_optimizazions (guint32 *exclude_mask) { guint32 opts = 0; /* no mips-specific optimizations yet */ *exclude_mask = 0; return opts; } static gboolean is_regsize_var (MonoType *t) { if (t->byref) return TRUE; t = mono_type_get_underlying_type (t); switch (t->type) { case MONO_TYPE_I4: case MONO_TYPE_U4: case MONO_TYPE_I: case MONO_TYPE_U: case MONO_TYPE_PTR: case MONO_TYPE_FNPTR: return TRUE; case MONO_TYPE_OBJECT: case MONO_TYPE_STRING: case MONO_TYPE_CLASS: case MONO_TYPE_SZARRAY: case MONO_TYPE_ARRAY: return TRUE; case MONO_TYPE_GENERICINST: if (!mono_type_generic_inst_is_valuetype (t)) return TRUE; return FALSE; case MONO_TYPE_VALUETYPE: return FALSE; } return FALSE; } GList * mono_arch_get_allocatable_int_vars (MonoCompile *cfg) { GList *vars = NULL; int i; for (i = 0; i < cfg->num_varinfo; i++) { MonoInst *ins = cfg->varinfo [i]; MonoMethodVar *vmv = MONO_VARINFO (cfg, i); /* unused vars */ if (vmv->range.first_use.abs_pos >= vmv->range.last_use.abs_pos) continue; if (ins->flags & (MONO_INST_VOLATILE|MONO_INST_INDIRECT) || (ins->opcode != OP_LOCAL && ins->opcode != OP_ARG)) continue; /* we can only allocate 32 bit values */ if (is_regsize_var (ins->inst_vtype)) { g_assert (MONO_VARINFO (cfg, i)->reg == -1); g_assert (i == vmv->idx); vars = mono_varlist_insert_sorted (cfg, vars, vmv, FALSE); } } return vars; } GList * mono_arch_get_global_int_regs (MonoCompile *cfg) { GList *regs = NULL; regs = g_list_prepend (regs, (gpointer)mips_s0); regs = g_list_prepend (regs, (gpointer)mips_s1); regs = g_list_prepend (regs, (gpointer)mips_s2); regs = g_list_prepend (regs, (gpointer)mips_s3); regs = g_list_prepend (regs, (gpointer)mips_s4); regs = g_list_prepend (regs, (gpointer)mips_s5); regs = g_list_prepend (regs, (gpointer)mips_s6); regs = g_list_prepend (regs, (gpointer)mips_s7); return regs; } /* * mono_arch_regalloc_cost: * * Return the cost, in number of memory references, of the action of * allocating the variable VMV into a register during global register * allocation. */ guint32 mono_arch_regalloc_cost (MonoCompile *cfg, MonoMethodVar *vmv) { /* FIXME: */ return 2; } static void args_onto_stack (CallInfo *info, gboolean force) { if (!info->on_stack) { if ((info->gr > MIPS_LAST_ARG_REG) || (info->fr > (MIPS_LAST_FPARG_REG+1))) { force = TRUE; } if (force) { info->on_stack = TRUE; info->stack_size = MIPS_STACK_PARAM_OFFSET; } } } /* * O32 calling convention version */ static void add_int32_arg (CallInfo *info, ArgInfo *ainfo) { /* First, see if we need to drop onto the stack */ args_onto_stack (info, FALSE); /* Now, place the argument */ ainfo->offset = info->stack_size; info->stack_size += 4; if (info->on_stack) { ainfo->regtype = RegTypeBase; ainfo->reg = mips_sp; /* in the caller */ } else { ainfo->reg = info->gr; info->gr += 1; info->gr_passed += 1; /* FP and GP slots do not overlap */ info->fr += 1; } } static void add_int64_arg (CallInfo *info, ArgInfo *ainfo) { /* First, see if we need to drop onto the stack */ args_onto_stack (info, FALSE); /* Now, place the argument */ if (info->stack_size & 0x7) { ArgInfo dummy; /* foo (int, long) -- need to align 2nd arg */ add_int32_arg (info, &dummy); args_onto_stack (info, FALSE); } ainfo->offset = info->stack_size; info->stack_size += 8; if (info->on_stack) { ainfo->regtype = RegTypeBase; ainfo->reg = mips_sp; /* in the caller */ } else { ainfo->reg = info->gr; info->gr += 2; info->gr_passed += 2; /* FP and GP slots do not overlap */ info->fr += 2; } } static void add_float32_arg (CallInfo *info, ArgInfo *ainfo) { /* First, see if we need to drop onto the stack */ args_onto_stack (info, FALSE); /* Now, place the argument */ ainfo->offset = info->stack_size; if (info->on_stack) { ainfo->regtype = RegTypeBase; ainfo->reg = mips_sp; /* in the caller */ info->stack_size += 8; } else { /* Only use FP regs for args if no int args passed yet */ if (!info->gr_passed) { ainfo->regtype = RegTypeFP; ainfo->reg = info->fr; info->stack_size += 8; /* Even though it's a single-precision float, it takes up two FP regs */ info->fr += 2; info->fr_passed += 1; /* FP and GP slots do not overlap */ info->gr += 2; } else { /* Passing single-precision float arg in a GP register * such as: func (0, 1.0, 2, 3); * In this case, only one 'gr' register is consumed. */ ainfo->regtype = RegTypeGeneral; ainfo->reg = info->gr; info->stack_size += 4; /* Even though it's a single-precision float, it takes up two FP regs */ info->fr += 1; info->fr_passed += 1; /* FP and GP slots do not overlap */ info->gr += 1; } } } static void add_float64_arg (CallInfo *info, ArgInfo *ainfo) { /* First, see if we need to drop onto the stack */ args_onto_stack (info, FALSE); /* Now, place the argument */ #ifdef ALIGN_DOUBLES // info->stack_size += (info->stack_size % 8); #endif ainfo->offset = info->stack_size; info->stack_size += 8; if (info->on_stack) { ainfo->regtype = RegTypeBase; ainfo->reg = mips_sp; /* in the caller */ } else { /* Only use FP regs for args if no int args passed yet */ if (!info->gr_passed) { ainfo->regtype = RegTypeFP; ainfo->reg = info->fr; } else { ainfo->regtype = RegTypeGeneral; ainfo->reg = info->gr; } info->fr += 2; info->fr_passed += 2; /* FP and GP slots do not overlap */ info->gr += 2; } } static CallInfo* calculate_sizes (MonoMethodSignature *sig, gboolean is_pinvoke) { guint i; int n = sig->hasthis + sig->param_count; guint32 simpletype; CallInfo *cinfo = g_malloc0 (sizeof (CallInfo) + sizeof (ArgInfo) * n); cinfo->fr = MIPS_FIRST_FPARG_REG; cinfo->gr = MIPS_FIRST_ARG_REG; cinfo->stack_size = 0; DEBUG(printf("calculate_sizes\n")); /* handle returning a struct */ if (MONO_TYPE_ISSTRUCT (sig->ret)) { cinfo->struct_ret = cinfo->gr; add_int32_arg (cinfo, &cinfo->ret); } n = 0; if (sig->hasthis) { add_int32_arg (cinfo, cinfo->args + n); n++; } DEBUG(printf("params: %d\n", sig->param_count)); for (i = 0; i < sig->param_count; ++i) { if ((sig->call_convention == MONO_CALL_VARARG) && (i == sig->sentinelpos)) { /* Prevent implicit arguments and sig_cookie from being passed in registers */ args_onto_stack (cinfo, TRUE); /* Emit the signature cookie just before the implicit arguments */ add_int32_arg (cinfo, &cinfo->sig_cookie); } DEBUG(printf("param %d: ", i)); if (sig->params [i]->byref) { DEBUG(printf("byref\n")); add_int32_arg (cinfo, &cinfo->args[n]); n++; continue; } simpletype = mono_type_get_underlying_type (sig->params [i])->type; switch (simpletype) { case MONO_TYPE_BOOLEAN: case MONO_TYPE_I1: case MONO_TYPE_U1: DEBUG(printf("1 byte\n")); cinfo->args [n].size = 1; add_int32_arg (cinfo, &cinfo->args[n]); n++; break; case MONO_TYPE_CHAR: case MONO_TYPE_I2: case MONO_TYPE_U2: DEBUG(printf("2 bytes\n")); cinfo->args [n].size = 2; add_int32_arg (cinfo, &cinfo->args[n]); n++; break; case MONO_TYPE_I4: case MONO_TYPE_U4: DEBUG(printf("4 bytes\n")); cinfo->args [n].size = 4; add_int32_arg (cinfo, &cinfo->args[n]); n++; break; case MONO_TYPE_I: case MONO_TYPE_U: case MONO_TYPE_PTR: case MONO_TYPE_FNPTR: case MONO_TYPE_CLASS: case MONO_TYPE_OBJECT: case MONO_TYPE_STRING: case MONO_TYPE_SZARRAY: case MONO_TYPE_ARRAY: cinfo->args [n].size = sizeof (gpointer); add_int32_arg (cinfo, &cinfo->args[n]); n++; break; case MONO_TYPE_GENERICINST: if (!mono_type_generic_inst_is_valuetype (sig->params [i])) { cinfo->args [n].size = sizeof (gpointer); add_int32_arg (cinfo, &cinfo->args[n]); n++; break; } /* Fall through */ case MONO_TYPE_VALUETYPE: { int j; int align_size; int nwords = 0; ArgInfo dummy_arg; gint size, alignment; MonoClass *klass; klass = mono_class_from_mono_type (sig->params [i]); if (is_pinvoke) size = mono_class_native_size (klass, NULL); else size = mono_class_value_size (klass, NULL); alignment = mono_class_min_align (klass); DEBUG(printf ("load %d bytes struct\n", mono_class_native_size (sig->params [i]->data.klass, NULL))); #if MIPS_PASS_STRUCTS_BY_VALUE cinfo->args [n].regtype = RegTypeStructByVal; #if 1 /* Need to do alignment if struct contains long or double */ if (cinfo->stack_size & (alignment - 1)) { add_int32_arg (cinfo, &dummy_arg); } g_assert (!(cinfo->stack_size & (alignment - 1))); #endif align_size = size; align_size += (sizeof (gpointer) - 1); align_size &= ~(sizeof (gpointer) - 1); nwords = (align_size + sizeof (gpointer) -1 ) / sizeof (gpointer); for (j = 0; j < nwords; ++j) { if (j == 0) add_int32_arg (cinfo, &cinfo->args [n]); else add_int32_arg (cinfo, &dummy_arg); if (cinfo->on_stack) cinfo->args [n].vtsize += 1; else cinfo->args [n].size += 1; } #else add_int32_arg (cinfo, &cinfo->args[n]); cinfo->args [n].regtype = RegTypeStructByAddr; #endif n++; break; } case MONO_TYPE_TYPEDBYREF: { /* keep in sync or merge with the valuetype case */ #if MIPS_PASS_STRUCTS_BY_VALUE { int size = sizeof (MonoTypedRef); int nwords = (size + sizeof (gpointer) -1 ) / sizeof (gpointer); cinfo->args [n].regtype = RegTypeStructByVal; if (cinfo->gr <= MIPS_LAST_ARG_REG) { int rest = MIPS_LAST_ARG_REG - cinfo->gr + 1; int n_in_regs = rest >= nwords? nwords: rest; cinfo->args [n].size = n_in_regs; cinfo->args [n].vtsize = nwords - n_in_regs; cinfo->args [n].reg = cinfo->gr; cinfo->gr += n_in_regs; } else { cinfo->args [n].size = 0; cinfo->args [n].vtsize = nwords; } cinfo->args [n].offset = MIPS_STACK_PARAM_OFFSET + cinfo->stack_size; g_print ("offset for arg %d at %d\n", n, MIPS_STACK_PARAM_OFFSET + cinfo->stack_size); cinfo->stack_size += nwords * sizeof (gpointer); } #else add_int32_arg (cinfo, &cinfo->args[n]); cinfo->args [n].regtype = RegTypeStructByAddr; #endif n++; break; } case MONO_TYPE_U8: case MONO_TYPE_I8: cinfo->args [n].size = 8; add_int64_arg (cinfo, &cinfo->args[n]); n++; break; case MONO_TYPE_R4: cinfo->args [n].size = 4; add_float32_arg (cinfo, &cinfo->args[n]); n++; break; case MONO_TYPE_R8: cinfo->args [n].size = 8; add_float64_arg (cinfo, &cinfo->args[n]); n++; break; default: g_error ("Can't trampoline 0x%x", sig->params [i]->type); } } { simpletype = mono_type_get_underlying_type (sig->ret)->type; switch (simpletype) { case MONO_TYPE_BOOLEAN: case MONO_TYPE_I1: case MONO_TYPE_U1: case MONO_TYPE_I2: case MONO_TYPE_U2: case MONO_TYPE_CHAR: case MONO_TYPE_I4: case MONO_TYPE_U4: case MONO_TYPE_I: case MONO_TYPE_U: case MONO_TYPE_PTR: case MONO_TYPE_FNPTR: case MONO_TYPE_CLASS: case MONO_TYPE_OBJECT: case MONO_TYPE_SZARRAY: case MONO_TYPE_ARRAY: case MONO_TYPE_STRING: cinfo->ret.reg = mips_v0; break; case MONO_TYPE_U8: case MONO_TYPE_I8: cinfo->ret.reg = mips_v0; break; case MONO_TYPE_R4: case MONO_TYPE_R8: cinfo->ret.reg = mips_f0; cinfo->ret.regtype = RegTypeFP; break; case MONO_TYPE_GENERICINST: if (!mono_type_generic_inst_is_valuetype (sig->ret)) { cinfo->ret.reg = mips_v0; break; } break; case MONO_TYPE_VALUETYPE: break; case MONO_TYPE_TYPEDBYREF: case MONO_TYPE_VOID: break; default: g_error ("Can't handle as return value 0x%x", sig->ret->type); } } /* align stack size to 16 */ cinfo->stack_size = (cinfo->stack_size + 15) & ~15; cinfo->stack_usage = cinfo->stack_size; return cinfo; } /* * Set var information according to the calling convention. mips version. * The locals var stuff should most likely be split in another method. */ void mono_arch_allocate_vars (MonoCompile *cfg) { MonoMethodSignature *sig; MonoMethodHeader *header; MonoInst *inst; int i, offset, size, align, curinst; int frame_reg = mips_sp; guint32 iregs_to_save = 0; guint32 fregs_to_restore; cfg->flags |= MONO_CFG_HAS_SPILLUP; /* allow room for the vararg method args: void* and long/double */ if (mono_jit_trace_calls != NULL && mono_trace_eval (cfg->method)) cfg->param_area = MAX (cfg->param_area, sizeof (gpointer)*8); /* this is bug #60332: remove when #59509 is fixed, so no weird vararg * call convs needs to be handled this way. */ if (cfg->flags & MONO_CFG_HAS_VARARGS) cfg->param_area = MAX (cfg->param_area, sizeof (gpointer)*8); /* gtk-sharp and other broken code will dllimport vararg functions even with * non-varargs signatures. Since there is little hope people will get this right * we assume they won't. */ if (cfg->method->wrapper_type == MONO_WRAPPER_MANAGED_TO_NATIVE) cfg->param_area = MAX (cfg->param_area, sizeof (gpointer)*8); cfg->param_area = MAX (cfg->param_area, 16); header = mono_method_get_header (cfg->method); sig = mono_method_signature (cfg->method); /* * We use the frame register also for any method that has * exception clauses. This way, when the handlers are called, * the code will reference local variables using the frame reg instead of * the stack pointer: if we had to restore the stack pointer, we'd * corrupt the method frames that are already on the stack (since * filters get called before stack unwinding happens) when the filter * code would call any method (this also applies to finally etc.). */ if ((cfg->flags & MONO_CFG_HAS_ALLOCA) || header->num_clauses || ALWAYS_USE_FP) frame_reg = mips_fp; cfg->frame_reg = frame_reg; if (frame_reg != mips_sp) { cfg->used_int_regs |= 1 << frame_reg; } offset = 0; curinst = 0; if (!MONO_TYPE_ISSTRUCT (sig->ret)) { /* FIXME: handle long and FP values */ switch (mono_type_get_underlying_type (sig->ret)->type) { case MONO_TYPE_VOID: break; default: cfg->ret->opcode = OP_REGVAR; cfg->ret->inst_c0 = mips_v0; break; } } /* Space for outgoing parameters, including a0-a3 */ offset += cfg->param_area; /* allow room to save the return value (if it's a struct) */ if (mono_jit_trace_calls != NULL && mono_trace_eval (cfg->method)) offset += 8; if (sig->call_convention == MONO_CALL_VARARG) { cfg->sig_cookie = MIPS_STACK_PARAM_OFFSET; } /* Now handle the local variables */ curinst = cfg->locals_start; for (i = curinst; i < cfg->num_varinfo; ++i) { inst = cfg->varinfo [i]; if ((inst->flags & MONO_INST_IS_DEAD) || inst->opcode == OP_REGVAR) continue; /* inst->backend.is_pinvoke indicates native sized value types, this is used by the * pinvoke wrappers when they call functions returning structure */ if (inst->backend.is_pinvoke && MONO_TYPE_ISSTRUCT (inst->inst_vtype) && inst->inst_vtype->type != MONO_TYPE_TYPEDBYREF) size = mono_class_native_size (mono_class_from_mono_type (inst->inst_vtype), &align); else size = mono_type_size (inst->inst_vtype, &align); offset += align - 1; offset &= ~(align - 1); inst->inst_offset = offset; inst->opcode = OP_REGOFFSET; inst->inst_basereg = frame_reg; offset += size; // g_print ("allocating local %d to %d\n", i, inst->inst_offset); } /* Space for LMF (if needed) */ #ifdef SAVE_LMF if (cfg->method->save_lmf) { /* align the offset to 16 bytes */ offset = (offset + MIPS_STACK_ALIGNMENT - 1) & ~(MIPS_STACK_ALIGNMENT - 1); cfg->arch.lmf_offset = offset; offset += sizeof (MonoLMF); } #endif #if 1 /* XXX - Saved S-regs seem to be getting clobbered by some calls with struct * args or return vals. Extra stack space avoids this in a lot of cases. */ offset += 64; #endif /* Space for saved registers */ cfg->arch.iregs_offset = offset; iregs_to_save = (cfg->used_int_regs & MONO_ARCH_CALLEE_SAVED_REGS); if (iregs_to_save) { for (i = MONO_MAX_IREGS-1; i >= 0; --i) { if (iregs_to_save & (1 << i)) { offset += sizeof (gulong); } } } #if 1 /* XXX - Saved S-regs seem to be getting clobbered by some calls with struct * args or return vals. Extra stack space avoids this in a lot of cases. */ offset += 64; #endif /* saved float registers */ #ifdef SAVE_FP_REGS fregs_to_restore = (cfg->used_float_regs & MONO_ARCH_CALLEE_SAVED_FREGS); if (fregs_to_restore) { for (i = MONO_MAX_FREGS-1; i >= 0; --i) { if (fregs_to_restore & (1 << i)) { offset += sizeof (double); } } } #endif #if 0 /* XXX - pre-allocate spill locations */ offset = (offset + MIPS_STACK_ALIGNMENT - 1) & ~(MIPS_STACK_ALIGNMENT - 1); cfg->arch.spillvar_offset = offset; offset += 64; cfg->arch.spillvar_offset_float = offset; offset += 64; #endif /* Now add space for saving the ra */ offset += 4; /* change sign? */ offset = (offset + MIPS_STACK_ALIGNMENT - 1) & ~(MIPS_STACK_ALIGNMENT - 1); cfg->stack_offset = offset; /* * Now allocate stack slots for the int arg regs (a0 - a3) * On MIPS o32, these are just above the incoming stack pointer * Even if the arg has been assigned to a regvar, it gets a stack slot */ /* Return struct-by-value results in a hidden first argument */ if (MONO_TYPE_ISSTRUCT (sig->ret)) { cfg->ret->opcode = OP_REGOFFSET; cfg->ret->inst_c0 = mips_a0; cfg->ret->inst_offset = offset; cfg->ret->inst_basereg = frame_reg; offset += 4; } for (i = 0; i < sig->param_count + sig->hasthis; ++i) { inst = cfg->varinfo [i]; if (inst->opcode != OP_REGVAR) { MonoType *arg_type; if (sig->hasthis && (i == 0)) arg_type = &mono_defaults.object_class->byval_arg; else arg_type = sig->params [i - sig->hasthis]; inst->opcode = OP_REGOFFSET; size = mono_type_size (arg_type, &align); /* Need to take references to R4 into account */ /* If it's a single-precision float, allocate 8 bytes of stack for it */ if ((arg_type->type == MONO_TYPE_R4) && !arg_type->byref) { align = 8; size = 8; } if (size < 4) { size = 4; align = 4; } inst->inst_basereg = frame_reg; offset = (offset + align - 1) & ~(align - 1); inst->inst_offset = offset; offset += size; if ((sig->call_convention == MONO_CALL_VARARG) && (i < sig->sentinelpos)) cfg->sig_cookie += size; // g_print ("allocating param %d to %d\n", i, inst->inst_offset); } else { /* Even a0-a3 get stack slots */ size = sizeof (gpointer); align = sizeof (gpointer); inst->inst_basereg = frame_reg; offset = (offset + align - 1) & ~(align - 1); inst->inst_offset = offset; offset += size; if ((sig->call_convention == MONO_CALL_VARARG) && (i < sig->sentinelpos)) cfg->sig_cookie += size; // g_print ("allocating param %d to %d\n", i, inst->inst_offset); } } #if 0 if (sig->hasthis) { inst = cfg->varinfo [curinst]; if (inst->opcode != OP_REGVAR) { inst->opcode = OP_REGOFFSET; inst->inst_basereg = frame_reg; offset += sizeof (gpointer) - 1; offset &= ~(sizeof (gpointer) - 1); inst->inst_offset = offset; offset += sizeof (gpointer); if (sig->call_convention == MONO_CALL_VARARG) cfg->sig_cookie += sizeof (gpointer); // g_print ("allocating this to %d\n", inst->inst_offset); } curinst++; } #endif } /* Fixme: we need an alignment solution for enter_method and mono_arch_call_opcode, * currently alignment in mono_arch_call_opcode is computed without arch_get_argument_info */ /* * take the arguments and generate the arch-specific * instructions to properly call the function in call. * This includes pushing, moving arguments to the right register * etc. * Issue: who does the spilling if needed, and when? */ MonoCallInst* mono_arch_call_opcode (MonoCompile *cfg, MonoBasicBlock* bb, MonoCallInst *call, int is_virtual) { MonoInst *arg, *in; MonoMethodSignature *sig; int i, n; CallInfo *cinfo; ArgInfo *ainfo; sig = call->signature; n = sig->param_count + sig->hasthis; cinfo = calculate_sizes (sig, sig->pinvoke); if (cinfo->struct_ret) call->used_iregs |= 1 << cinfo->struct_ret; for (i = 0; i < n; ++i) { ainfo = cinfo->args + i; if ((sig->call_convention == MONO_CALL_VARARG) && (i == sig->sentinelpos)) { MonoInst *sig_arg; cfg->disable_aot = TRUE; MONO_INST_NEW (cfg, sig_arg, OP_ICONST); sig_arg->inst_p0 = call->signature; MONO_INST_NEW (cfg, arg, OP_OUTARG); arg->inst_imm = cinfo->sig_cookie.offset; arg->inst_left = sig_arg; /* prepend, so they get reversed */ arg->next = call->out_args; call->out_args = arg; } if (is_virtual && i == 0) { /* the argument will be attached to the call instrucion */ in = call->args [i]; call->used_iregs |= 1 << ainfo->reg; } else { MONO_INST_NEW (cfg, arg, OP_OUTARG); in = call->args [i]; arg->cil_code = in->cil_code; arg->inst_left = in; arg->inst_call = call; arg->type = in->type; /* prepend, we'll need to reverse them later */ arg->next = call->out_args; call->out_args = arg; if (ainfo->regtype == RegTypeGeneral) { arg->backend.reg3 = ainfo->reg; call->used_iregs |= 1 << ainfo->reg; if (arg->type == STACK_I8) call->used_iregs |= 1 << (ainfo->reg + 1); } else if (ainfo->regtype == RegTypeStructByAddr) { /* FIXME: where is the data allocated? */ arg->backend.reg3 = ainfo->reg; call->used_iregs |= 1 << ainfo->reg; } else if (ainfo->regtype == RegTypeStructByVal) { int cur_reg; MonoMIPSArgInfo *ai = mono_mempool_alloc0 (cfg->mempool, sizeof (MonoMIPSArgInfo)); /* mark the used regs */ for (cur_reg = 0; cur_reg < ainfo->size; ++cur_reg) { call->used_iregs |= 1 << (ainfo->reg + cur_reg); } arg->opcode = OP_OUTARG_VT; ai->reg = ainfo->reg; ai->size = ainfo->size; ai->vtsize = ainfo->vtsize; ai->offset = ainfo->offset; arg->backend.data = ai; } else if (ainfo->regtype == RegTypeBase) { MonoMIPSArgInfo *ai = mono_mempool_alloc0 (cfg->mempool, sizeof (MonoMIPSArgInfo)); arg->opcode = OP_OUTARG_MEMBASE; ai->reg = ainfo->reg; ai->size = ainfo->size; ai->offset = ainfo->offset; arg->backend.data = ai; } else if (ainfo->regtype == RegTypeFP) { arg->opcode = OP_OUTARG_R8; arg->backend.reg3 = ainfo->reg; call->used_fregs |= 1 << ainfo->reg; if (ainfo->size == 4) { arg->opcode = OP_OUTARG_R4; /* we reduce the precision */ /*MonoInst *conv; MONO_INST_NEW (cfg, conv, OP_FCONV_TO_R4); conv->inst_left = arg->inst_left; arg->inst_left = conv;*/ } } else { g_assert_not_reached (); } } } /* * Reverse the call->out_args list. */ { MonoInst *prev = NULL, *list = call->out_args, *next; while (list) { next = list->next; list->next = prev; prev = list; list = next; } call->out_args = prev; } call->stack_usage = cinfo->stack_usage; cfg->param_area = MAX (cfg->param_area, cinfo->stack_usage); cfg->param_area = MAX (cfg->param_area, 16); /* a0-a3 always present */ cfg->param_area = (cfg->param_area + MIPS_STACK_ALIGNMENT - 1) & ~(MIPS_STACK_ALIGNMENT - 1); cfg->flags |= MONO_CFG_HAS_CALLS; /* * should set more info in call, such as the stack space * used by the args that needs to be added back to esp */ g_free (cinfo); return call; } static void peephole_pass (MonoCompile *cfg, MonoBasicBlock *bb) { MonoInst *ins, *last_ins = NULL; ins = bb->code; while (ins) { switch (ins->opcode) { case OP_MUL_IMM: /* remove unnecessary multiplication with 1 */ if (ins->inst_imm == 1) { if (ins->dreg != ins->sreg1) { ins->opcode = OP_MOVE; } else { last_ins->next = ins->next; ins = ins->next; continue; } } else { int power2 = mono_is_power_of_two (ins->inst_imm); if (power2 > 0) { ins->opcode = OP_SHL_IMM; ins->inst_imm = power2; } } break; case OP_LOAD_MEMBASE: case OP_LOADI4_MEMBASE: /* * OP_STORE_MEMBASE_REG reg, offset(basereg) * OP_LOAD_MEMBASE offset(basereg), reg */ if (last_ins && (last_ins->opcode == OP_STOREI4_MEMBASE_REG || last_ins->opcode == OP_STORE_MEMBASE_REG) && ins->inst_basereg == last_ins->inst_destbasereg && ins->inst_offset == last_ins->inst_offset) { if (ins->dreg == last_ins->sreg1) { last_ins->next = ins->next; ins = ins->next; continue; } else { //static int c = 0; printf ("MATCHX %s %d\n", cfg->method->name,c++); ins->opcode = OP_MOVE; ins->sreg1 = last_ins->sreg1; } /* * Note: reg1 must be different from the basereg in the second load * OP_LOAD_MEMBASE offset(basereg), reg1 * OP_LOAD_MEMBASE offset(basereg), reg2 * --> * OP_LOAD_MEMBASE offset(basereg), reg1 * OP_MOVE reg1, reg2 */ } if (last_ins && (last_ins->opcode == OP_LOADI4_MEMBASE || last_ins->opcode == OP_LOAD_MEMBASE) && ins->inst_basereg != last_ins->dreg && ins->inst_basereg == last_ins->inst_basereg && ins->inst_offset == last_ins->inst_offset) { if (ins->dreg == last_ins->dreg) { last_ins->next = ins->next; ins = ins->next; continue; } else { ins->opcode = OP_MOVE; ins->sreg1 = last_ins->dreg; } //g_assert_not_reached (); #if 0 /* * OP_STORE_MEMBASE_IMM imm, offset(basereg) * OP_LOAD_MEMBASE offset(basereg), reg * --> * OP_STORE_MEMBASE_IMM imm, offset(basereg) * OP_ICONST reg, imm */ } else if (last_ins && (last_ins->opcode == OP_STOREI4_MEMBASE_IMM || last_ins->opcode == OP_STORE_MEMBASE_IMM) && ins->inst_basereg == last_ins->inst_destbasereg && ins->inst_offset == last_ins->inst_offset) { //static int c = 0; printf ("MATCHX %s %d\n", cfg->method->name,c++); ins->opcode = OP_ICONST; ins->inst_c0 = last_ins->inst_imm; g_assert_not_reached (); // check this rule #endif } break; case OP_LOADU1_MEMBASE: case OP_LOADI1_MEMBASE: if (last_ins && (last_ins->opcode == OP_STOREI1_MEMBASE_REG) && ins->inst_basereg == last_ins->inst_destbasereg && ins->inst_offset == last_ins->inst_offset) { if (ins->dreg == last_ins->sreg1) { last_ins->next = ins->next; ins = ins->next; continue; } else { //static int c = 0; printf ("MATCHX %s %d\n", cfg->method->name,c++); ins->opcode = OP_MOVE; ins->sreg1 = last_ins->sreg1; } } break; case OP_LOADU2_MEMBASE: case OP_LOADI2_MEMBASE: if (last_ins && (last_ins->opcode == OP_STOREI2_MEMBASE_REG) && ins->inst_basereg == last_ins->inst_destbasereg && ins->inst_offset == last_ins->inst_offset) { if (ins->dreg == last_ins->sreg1) { last_ins->next = ins->next; ins = ins->next; continue; } else { //static int c = 0; printf ("MATCHX %s %d\n", cfg->method->name,c++); ins->opcode = OP_MOVE; ins->sreg1 = last_ins->sreg1; } } break; case CEE_CONV_I4: case CEE_CONV_U4: case OP_MOVE: case OP_SETREG: ins->opcode = OP_MOVE; /* * OP_MOVE reg, reg */ if (ins->dreg == ins->sreg1) { if (last_ins) last_ins->next = ins->next; ins = ins->next; continue; } /* * OP_MOVE sreg, dreg * OP_MOVE dreg, sreg */ if (last_ins && last_ins->opcode == OP_MOVE && ins->sreg1 == last_ins->dreg && ins->dreg == last_ins->sreg1) { last_ins->next = ins->next; ins = ins->next; continue; } break; } last_ins = ins; ins = ins->next; } bb->last_ins = last_ins; } static inline InstList* inst_list_prepend (MonoMemPool *pool, InstList *list, MonoInst *data) { InstList *item = mono_mempool_alloc (pool, sizeof (InstList)); item->data = data; item->prev = NULL; item->next = list; if (list) list->prev = item; return item; } static void insert_after_ins (MonoBasicBlock *bb, MonoInst *ins, MonoInst *to_insert) { if (ins == NULL) { ins = bb->code; bb->code = to_insert; to_insert->next = ins; } else { to_insert->next = ins->next; ins->next = to_insert; } } #define NEW_INS(cfg,dest,op) do { \ (dest) = mono_mempool_alloc0 ((cfg)->mempool, sizeof (MonoInst)); \ (dest)->opcode = (op); \ insert_after_ins (bb, last_ins, (dest)); \ } while (0) static int map_to_reg_reg_op (int op) { switch (op) { case OP_ADD_IMM: return CEE_ADD; case OP_SUB_IMM: return CEE_SUB; case OP_AND_IMM: return CEE_AND; case OP_COMPARE_IMM: return OP_COMPARE; case OP_ADDCC_IMM: return OP_ADDCC; case OP_ADC_IMM: return OP_ADC; case OP_SUBCC_IMM: return OP_SUBCC; case OP_SBB_IMM: return OP_SBB; case OP_OR_IMM: return CEE_OR; case OP_XOR_IMM: return CEE_XOR; case OP_MUL_IMM: return CEE_MUL; case OP_LOAD_MEMBASE: return OP_LOAD_MEMINDEX; case OP_LOADI4_MEMBASE: return OP_LOADI4_MEMINDEX; case OP_LOADU4_MEMBASE: return OP_LOADU4_MEMINDEX; case OP_LOADU1_MEMBASE: return OP_LOADU1_MEMINDEX; case OP_LOADI2_MEMBASE: return OP_LOADI2_MEMINDEX; case OP_LOADU2_MEMBASE: return OP_LOADU2_MEMINDEX; case OP_LOADI1_MEMBASE: return OP_LOADI1_MEMINDEX; case OP_LOADR4_MEMBASE: return OP_LOADR4_MEMINDEX; case OP_LOADR8_MEMBASE: return OP_LOADR8_MEMINDEX; case OP_STOREI1_MEMBASE_REG: return OP_STOREI1_MEMINDEX; case OP_STOREI2_MEMBASE_REG: return OP_STOREI2_MEMINDEX; case OP_STOREI4_MEMBASE_REG: return OP_STOREI4_MEMINDEX; case OP_STORE_MEMBASE_REG: return OP_STORE_MEMINDEX; case OP_STORER4_MEMBASE_REG: return OP_STORER4_MEMINDEX; case OP_STORER8_MEMBASE_REG: return OP_STORER8_MEMINDEX; case OP_STORE_MEMBASE_IMM: return OP_STORE_MEMBASE_REG; case OP_STOREI1_MEMBASE_IMM: return OP_STOREI1_MEMBASE_REG; case OP_STOREI2_MEMBASE_IMM: return OP_STOREI2_MEMBASE_REG; case OP_STOREI4_MEMBASE_IMM: return OP_STOREI4_MEMBASE_REG; } g_assert_not_reached (); } /* * Remove from the instruction list the instructions that can't be * represented with very simple instructions with no register * requirements. */ static void mono_arch_lowering_pass (MonoCompile *cfg, MonoBasicBlock *bb) { MonoInst *ins, *next, *temp, *last_ins = NULL; int imm; /* setup the virtual reg allocator */ if (bb->max_ireg > cfg->rs->next_vireg) cfg->rs->next_vireg = bb->max_ireg; ins = bb->code; while (ins) { loop_start: switch (ins->opcode) { case OP_ADD_IMM: case OP_ADDCC_IMM: if (!mips_is_imm16 (ins->inst_imm)) { NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_imm; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg2 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); } break; case OP_SUB_IMM: #if 0 if (!mips_is_imm16 (-ins->inst_imm)) { NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_imm; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg2 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); } #endif break; case OP_AND_IMM: case OP_OR_IMM: case OP_XOR_IMM: #if 0 if ((ins->inst_imm & 0xffff0000) && (ins->inst_imm & 0xffff)) { NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_imm; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg2 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); } #endif break; case OP_SBB_IMM: case OP_SUBCC_IMM: case OP_ADC_IMM: #if 0 NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_imm; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg2 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); break; #endif case OP_COMPARE_IMM: #if 0 if (compare_opcode_is_unsigned (ins->next->opcode)) { if (!ppc_is_uimm16 (ins->inst_imm)) { NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_imm; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg2 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); } } else { if (!ppc_is_imm16 (ins->inst_imm)) { NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_imm; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg2 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); } } #endif break; case OP_MUL_IMM: #if 0 if (ins->inst_imm == 1) { ins->opcode = OP_MOVE; break; } if (ins->inst_imm == 0) { ins->opcode = OP_ICONST; ins->inst_c0 = 0; break; } imm = mono_is_power_of_two (ins->inst_imm); if (imm > 0) { ins->opcode = OP_SHL_IMM; ins->inst_imm = imm; break; } if (!ppc_is_imm16 (ins->inst_imm)) { NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_imm; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg2 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); } #endif break; case OP_LOAD_MEMBASE: case OP_LOADI4_MEMBASE: case OP_LOADU4_MEMBASE: case OP_LOADI2_MEMBASE: case OP_LOADU2_MEMBASE: case OP_LOADI1_MEMBASE: case OP_LOADU1_MEMBASE: case OP_LOADR4_MEMBASE: case OP_LOADR8_MEMBASE: case OP_STORE_MEMBASE_REG: case OP_STOREI4_MEMBASE_REG: case OP_STOREI2_MEMBASE_REG: case OP_STOREI1_MEMBASE_REG: case OP_STORER4_MEMBASE_REG: case OP_STORER8_MEMBASE_REG: #if 0 /* we can do two things: load the immed in a register * and use an indexed load, or see if the immed can be * represented as an ad_imm + a load with a smaller offset * that fits. We just do the first for now, optimize later. */ if (ppc_is_imm16 (ins->inst_offset)) break; NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_offset; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg2 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); #endif break; case OP_STORE_MEMBASE_IMM: case OP_STOREI1_MEMBASE_IMM: case OP_STOREI2_MEMBASE_IMM: case OP_STOREI4_MEMBASE_IMM: #if 0 NEW_INS (cfg, temp, OP_ICONST); temp->inst_c0 = ins->inst_imm; temp->dreg = mono_regstate_next_int (cfg->rs); ins->sreg1 = temp->dreg; ins->opcode = map_to_reg_reg_op (ins->opcode); last_ins = temp; goto loop_start; /* make it handle the possibly big ins->inst_offset */ #endif #if 1 ; #endif } last_ins = ins; ins = ins->next; } bb->last_ins = last_ins; bb->max_ireg = cfg->rs->next_vireg; } void mono_arch_local_regalloc (MonoCompile *cfg, MonoBasicBlock *bb) { if (!bb->code) return; mono_arch_lowering_pass (cfg, bb); mono_local_regalloc (cfg, bb); #if 0 mono_arch_allocate_vars (cfg); #endif } static guchar* emit_float_to_int (MonoCompile *cfg, guchar *code, int dreg, int sreg, int size, gboolean is_signed) { /* sreg is a float, dreg is an integer reg. mips_at is used a scratch */ #if 1 mips_truncwd (code, mips_ftemp, sreg); #else mips_cvtwd (code, mips_ftemp, sreg); #endif mips_mfc1 (code, dreg, mips_ftemp); if (!is_signed) { if (size == 1) mips_andi (code, dreg, dreg, 0xff); else if (size == 2) { mips_sll (code, dreg, dreg, 16); mips_srl (code, dreg, dreg, 16); } } else { if (size == 1) { mips_sll (code, dreg, dreg, 24); mips_sra (code, dreg, dreg, 24); } else if (size == 2) { mips_sll (code, dreg, dreg, 16); mips_sra (code, dreg, dreg, 16); } } return code; } void mono_arch_output_basic_block (MonoCompile *cfg, MonoBasicBlock *bb) { MonoInst *ins; MonoCallInst *call; guint offset; guint8 *code = cfg->native_code + cfg->code_len; MonoInst *last_ins = NULL; guint last_offset = 0; int max_len, cpos; int ins_cnt = 0; if (cfg->opt & MONO_OPT_PEEPHOLE) peephole_pass (cfg, bb); /* we don't align basic blocks of loops on mips */ if (cfg->verbose_level > 2) g_print ("Basic block %d starting at offset 0x%x\n", bb->block_num, bb->native_offset); cpos = bb->max_offset; #if 0 if (cfg->prof_options & MONO_PROFILE_COVERAGE) { MonoCoverageInfo *cov = mono_get_coverage_info (cfg->method); g_assert (!mono_compile_aot); cpos += 6; if (bb->cil_code) cov->data [bb->dfn].iloffset = bb->cil_code - cfg->cil_code; /* this is not thread save, but good enough */ /* fixme: howto handle overflows? */ x86_inc_mem (code, &cov->data [bb->dfn].count); } #endif ins = bb->code; while (ins) { offset = code - cfg->native_code; max_len = ((guint8 *)ins_spec [ins->opcode])[MONO_INST_LEN]; if (offset > (cfg->code_size - max_len - 16)) { cfg->code_size *= 2; cfg->native_code = g_realloc (cfg->native_code, cfg->code_size); code = cfg->native_code + offset; } mono_debug_record_line_number (cfg, ins, offset); if (cfg->verbose_level > 2) { g_print (" @ 0x%x\t", offset); mono_print_ins (ins_cnt++, ins); } switch (ins->opcode) { case OP_TLS_GET: g_assert_not_reached(); #if 0 emit_tls_access (code, ins->dreg, ins->inst_offset); #endif break; case OP_BIGMUL: mips_mult (code, ins->sreg1, ins->sreg2); mips_mflo (code, ins->dreg); mips_mfhi (code, ins->dreg+1); break; case OP_BIGMUL_UN: mips_multu (code, ins->sreg1, ins->sreg2); mips_mflo (code, ins->dreg); mips_mfhi (code, ins->dreg+1); break; case OP_MEMORY_BARRIER: #if 0 ppc_sync (code); #endif break; case OP_STOREI1_MEMBASE_IMM: mips_load_const (code, mips_temp, ins->inst_imm); if (mips_is_imm16 (ins->inst_offset)) { mips_sb (code, mips_temp, ins->inst_destbasereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_sb (code, mips_temp, mips_at, ins->inst_destbasereg); } break; case OP_STOREI2_MEMBASE_IMM: mips_load_const (code, mips_temp, ins->inst_imm); if (mips_is_imm16 (ins->inst_offset)) { mips_sh (code, mips_temp, ins->inst_destbasereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_sh (code, mips_temp, mips_at, ins->inst_destbasereg); } break; case OP_STORE_MEMBASE_IMM: case OP_STOREI4_MEMBASE_IMM: mips_load_const (code, mips_temp, ins->inst_imm); if (mips_is_imm16 (ins->inst_offset)) { mips_sw (code, mips_temp, ins->inst_destbasereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_sw (code, mips_temp, mips_at, ins->inst_destbasereg); } break; case OP_STOREI1_MEMBASE_REG: if (mips_is_imm16 (ins->inst_offset)) { mips_sb (code, ins->sreg1, ins->inst_destbasereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_destbasereg); mips_sb (code, ins->sreg1, mips_at, 0); } break; case OP_STOREI2_MEMBASE_REG: if (mips_is_imm16 (ins->inst_offset)) { mips_sh (code, ins->sreg1, ins->inst_destbasereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_destbasereg); mips_sh (code, ins->sreg1, mips_at, 0); } break; case OP_STORE_MEMBASE_REG: case OP_STOREI4_MEMBASE_REG: if (mips_is_imm16 (ins->inst_offset)) { mips_sw (code, ins->sreg1, ins->inst_destbasereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_destbasereg); mips_sw (code, ins->sreg1, mips_at, 0); } break; case CEE_LDIND_I: case CEE_LDIND_I4: case CEE_LDIND_U4: g_assert_not_reached (); //x86_mov_reg_mem (code, ins->dreg, ins->inst_p0, 4); break; case OP_LOADU4_MEM: g_assert_not_reached (); //x86_mov_reg_imm (code, ins->dreg, ins->inst_p0); //x86_mov_reg_membase (code, ins->dreg, ins->dreg, 0, 4); break; case OP_LOAD_MEMBASE: case OP_LOADI4_MEMBASE: case OP_LOADU4_MEMBASE: if (mips_is_imm16 (ins->inst_offset)) { mips_lw (code, ins->dreg, ins->inst_basereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_basereg); mips_lw (code, ins->dreg, mips_at, 0); } break; case OP_LOADI1_MEMBASE: if (mips_is_imm16 (ins->inst_offset)) { mips_lb (code, ins->dreg, ins->inst_basereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_basereg); mips_lb (code, ins->dreg, mips_at, 0); } break; case OP_LOADU1_MEMBASE: if (mips_is_imm16 (ins->inst_offset)) { mips_lbu (code, ins->dreg, ins->inst_basereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_basereg); mips_lbu (code, ins->dreg, mips_at, 0); } break; case OP_LOADI2_MEMBASE: if (mips_is_imm16 (ins->inst_offset)) { mips_lh (code, ins->dreg, ins->inst_basereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_basereg); mips_lh (code, ins->dreg, mips_at, 0); } break; case OP_LOADU2_MEMBASE: if (mips_is_imm16 (ins->inst_offset)) { mips_lhu (code, ins->dreg, ins->inst_basereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_basereg); mips_lhu (code, ins->dreg, mips_at, 0); } break; case CEE_CONV_I1: mips_sll (code, mips_at, ins->sreg1, 24); mips_sra (code, ins->dreg, mips_at, 24); break; case CEE_CONV_I2: mips_sll (code, mips_at, ins->sreg1, 16); mips_sra (code, ins->dreg, mips_at, 16); break; case CEE_CONV_U1: mips_andi (code, ins->dreg, ins->sreg1, 0xff); break; case CEE_CONV_U2: mips_sll (code, mips_at, ins->sreg1, 16); mips_srl (code, ins->dreg, mips_at, 16); break; case OP_MIPS_SLT: mips_slt (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_MIPS_SLTI: g_assert (mips_is_imm16 (ins->inst_imm)); mips_slti (code, ins->dreg, ins->sreg1, ins->inst_imm); break; case OP_MIPS_SLTU: mips_sltu (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_MIPS_SLTIU: g_assert (mips_is_imm16 (ins->inst_imm)); mips_sltiu (code, ins->dreg, ins->sreg1, ins->inst_imm); break; #if 1 case OP_COMPARE_IMM: if (mips_is_imm16 (ins->inst_imm)) { mips_addiu (code, mips_at, ins->sreg1, ((-ins->inst_imm) & 0xffff)); } else { mips_load_const (code, mips_at, ins->inst_imm); mips_subu (code, mips_at, ins->sreg1, mips_at); } break; case OP_COMPARE: mips_subu (code, mips_at, ins->sreg1, ins->sreg2); break; #else case OP_COMPARE_IMM: case OP_COMPARE: { gboolean emitted = FALSE; guint cmp_reg = ins->sreg2; gboolean direct_imm = 0; guint32 imm = 0; if (ins->opcode == OP_COMPARE_IMM) { if (!mips_is_imm16 (ins->inst_imm)) { mips_load_const (code, mips_at, ins->inst_imm); cmp_reg = mips_at; } else { direct_imm = TRUE; imm = ins->inst_imm & 0xffff; } } #if 0 if (ins->next) { switch (ins->next->opcode) { case CEE_BGE_UN: if (direct_imm) { mips_sltiu (code, mips_at, ins->sreg1, imm); } else { mips_sltu (code, mips_at, ins->sreg1, cmp_reg); } mips_xori (code, mips_at, mips_at, 1); emitted = TRUE; break; case CEE_BLT_UN: if (direct_imm) { mips_sltiu (code, mips_at, ins->sreg1, imm); } else { mips_sltu (code, mips_at, ins->sreg1, cmp_reg); } emitted = TRUE; break; } mips_sll (code, mips_at, mips_at, 31); } #endif if (!emitted) { if (ins->inst_imm == 0) { mips_move (code, mips_at, ins->sreg1); } else if (mips_is_imm16 (-ins->inst_imm)) { mips_addiu (code, mips_at, ins->sreg1, ((-ins->inst_imm) & 0xffff)); } else if (ins->opcode == OP_COMPARE_IMM) { mips_load_const (code, mips_at, ins->inst_imm); mips_subu (code, mips_at, ins->sreg1, mips_at); } else { mips_subu (code, mips_at, ins->sreg1, ins->sreg2); } emitted = TRUE; } break; } #endif case CEE_BREAK: mips_break (code, 0xfd); break; case OP_ADDCC: /* XXX ppc_addc */ mips_addu (code, ins->dreg, ins->sreg1, ins->sreg2); break; case CEE_ADD: mips_addu (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_ADC: /* XXX ppc_adde */ mips_addu (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_ADDCC_IMM: /* XXX ppc_addc/ppc_addic */ if (mips_is_imm16 (ins->inst_imm)) { mips_addiu (code, ins->dreg, ins->sreg1, ins->inst_imm); } else { mips_load_const (code, mips_at, ins->inst_imm); mips_addu (code, ins->dreg, ins->sreg1, mips_at); } break; case OP_ADD_IMM: if (mips_is_imm16 (ins->inst_imm)) { mips_addiu (code, ins->dreg, ins->sreg1, ins->inst_imm); } else { mips_load_const (code, mips_at, ins->inst_imm); mips_add (code, ins->dreg, ins->sreg1, mips_at); } break; case OP_ADC_IMM: /* XXX ppc_adde */ mips_load_const (code, mips_at, ins->inst_imm); mips_addu (code, ins->dreg, ins->sreg1, mips_at); break; case CEE_ADD_OVF: mips_addu (code, ins->dreg, ins->sreg1, ins->sreg2); /* XXX - Throw exception if overflowed */ break; case CEE_ADD_OVF_UN: mips_addu (code, ins->dreg, ins->sreg1, ins->sreg2); /* XXX - Throw exception if overflowed */ break; case CEE_SUB_OVF: mips_subu (code, ins->dreg, ins->sreg1, ins->sreg2); /* XXX - Throw exception if overflowed */ break; case CEE_SUB_OVF_UN: mips_subu (code, ins->dreg, ins->sreg1, ins->sreg2); /* XXX - Throw exception if overflowed */ break; case OP_ADD_OVF_CARRY: mips_addu (code, ins->dreg, ins->sreg1, ins->sreg2); /* XXX - Throw exception if overflowed */ break; case OP_ADD_OVF_UN_CARRY: mips_addu (code, ins->dreg, ins->sreg1, ins->sreg2); /* XXX - Throw exception if overflowed */ break; case OP_SUB_OVF_CARRY: mips_subu (code, ins->dreg, ins->sreg1, ins->sreg2); /* XXX - Throw exception if overflowed */ break; case OP_SUB_OVF_UN_CARRY: mips_subu (code, ins->dreg, ins->sreg1, ins->sreg2); /* XXX - Throw exception if overflowed */ break; case OP_SUBCC: mips_subu (code, ins->dreg, ins->sreg2, ins->sreg1); break; case OP_SUBCC_IMM: /* XXX */ mips_load_const (code, mips_at, ins->inst_imm); mips_subu (code, ins->dreg, mips_at, ins->sreg1); break; case CEE_SUB: mips_subu (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_SBB: /* XXX */ mips_subu (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_SUB_IMM: // we add the negated value if (mips_is_imm16 (-ins->inst_imm)) mips_addi (code, ins->dreg, ins->sreg1, -ins->inst_imm); else { mips_load_const (code, mips_at, ins->inst_imm); mips_subu (code, ins->dreg, ins->sreg1, mips_at); } break; case OP_SBB_IMM: mips_load_const (code, mips_at, ins->inst_imm); /* XXX */ mips_subu (code, ins->dreg, mips_at, ins->sreg1); break; case CEE_AND: mips_and (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_AND_IMM: if (mips_is_imm16 (ins->inst_imm)) { mips_andi (code, ins->dreg, ins->sreg1, ins->inst_imm); } else { mips_load_const (code, mips_at, ins->inst_imm); mips_and (code, ins->dreg, ins->sreg1, mips_at); } break; case CEE_DIV: { guint32 *divisor_is_zero = (guint32 *)(void *)code; /* Put divide in branch delay slot */ mips_bne (code, ins->sreg2, mips_zero, 0); mips_div (code, ins->sreg1, ins->sreg2); /* Divide by zero -- throw exception */ EMIT_SYSTEM_EXCEPTION_NAME("DivideByZeroException"); mips_patch (divisor_is_zero, (guint32)code); mips_mflo (code, ins->dreg); break; } case CEE_DIV_UN: { guint32 *divisor_is_zero = (guint32 *)(void *)code; /* Put divide in branch delay slot */ mips_bne (code, ins->sreg2, mips_zero, 0); mips_divu (code, ins->sreg1, ins->sreg2); /* Divide by zero -- throw exception */ EMIT_SYSTEM_EXCEPTION_NAME("DivideByZeroException"); mips_patch (divisor_is_zero, (guint32)code); mips_mflo (code, ins->dreg); break; } case OP_DIV_IMM: g_assert_not_reached (); #if 0 ppc_load (code, ppc_r11, ins->inst_imm); ppc_divwod (code, ins->dreg, ins->sreg1, ppc_r11); ppc_mfspr (code, ppc_r0, ppc_xer); ppc_andisd (code, ppc_r0, ppc_r0, (1<<14)); /* FIXME: use OverflowException for 0x80000000/-1 */ EMIT_COND_SYSTEM_EXCEPTION_FLAGS (PPC_BR_FALSE, PPC_BR_EQ, "DivideByZeroException"); #endif g_assert_not_reached(); break; case CEE_REM: { /* XXX */ #if 1 guint32 *divisor_is_zero = (guint32 *)(void *)code; /* Put divide in branch delay slot */ mips_bne (code, ins->sreg2, mips_zero, 0); mips_div (code, ins->sreg1, ins->sreg2); /* Divide by zero -- throw exception */ EMIT_SYSTEM_EXCEPTION_NAME("DivideByZeroException"); mips_patch (divisor_is_zero, (guint32)code); mips_mfhi (code, ins->dreg); break; #else guint32 *divisor_is_m1; ppc_cmpi (code, 0, 0, ins->sreg2, -1); divisor_is_m1 = (guint32 *)(void *)code; ppc_bc (code, PPC_BR_FALSE | PPC_BR_LIKELY, PPC_BR_EQ, 0); ppc_lis (code, ppc_r11, 0x8000); ppc_cmp (code, 0, 0, ins->sreg1, ppc_r11); EMIT_COND_SYSTEM_EXCEPTION_FLAGS (PPC_BR_TRUE, PPC_BR_EQ, "ArithmeticException"); mips_patch (divisor_is_m1, (guint32)code); ppc_divwod (code, ppc_r11, ins->sreg1, ins->sreg2); ppc_mfspr (code, ppc_r0, ppc_xer); ppc_andisd (code, ppc_r0, ppc_r0, (1<<14)); /* FIXME: use OverflowException for 0x80000000/-1 */ EMIT_COND_SYSTEM_EXCEPTION_FLAGS (PPC_BR_FALSE, PPC_BR_EQ, "DivideByZeroException"); ppc_mullw (code, ppc_r11, ppc_r11, ins->sreg2); ppc_subf (code, ins->dreg, ppc_r11, ins->sreg1); #endif break; } case CEE_REM_UN: { guint32 *divisor_is_zero = (guint32 *)(void *)code; /* Put divide in branch delay slot */ mips_bne (code, ins->sreg2, mips_zero, 0); mips_divu (code, ins->sreg1, ins->sreg2); /* Divide by zero -- throw exception */ EMIT_SYSTEM_EXCEPTION_NAME("DivideByZeroException"); mips_patch (divisor_is_zero, (guint32)code); mips_mfhi (code, ins->dreg); break; } case OP_REM_IMM: g_assert_not_reached (); case CEE_OR: mips_or (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_OR_IMM: if (mips_is_imm16 (ins->inst_imm)) { mips_ori (code, ins->sreg1, ins->dreg, ins->inst_imm); } else { mips_load_const (code, mips_at, ins->inst_imm); mips_or (code, ins->dreg, ins->sreg1, mips_at); } break; case CEE_XOR: mips_xor (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_XOR_IMM: /* unsigned 16-bit immediate */ if ((ins->inst_imm & 0xffff) == ins->inst_imm) { mips_xori (code, ins->dreg, ins->sreg1, ins->inst_imm); } else { mips_load_const (code, mips_at, ins->inst_imm); mips_xor (code, ins->dreg, ins->sreg1, mips_at); } break; case OP_MIPS_XORI: g_assert (mips_is_imm16 (ins->inst_imm)); mips_xori (code, ins->dreg, ins->sreg1, ins->inst_imm); break; case CEE_SHL: mips_sllv (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_SHL_IMM: mips_sll (code, ins->dreg, ins->sreg1, ins->inst_imm & 0x1f); break; case CEE_SHR: mips_srav (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_SHR_IMM: mips_sra (code, ins->dreg, ins->sreg1, ins->inst_imm & 0x1f); break; case OP_SHR_UN_IMM: mips_srl (code, ins->dreg, ins->sreg1, ins->inst_imm & 0x1f); break; case CEE_SHR_UN: mips_srlv (code, ins->dreg, ins->sreg1, ins->sreg2); break; case CEE_NOT: mips_nor (code, ins->dreg, mips_zero, ins->sreg1); break; case CEE_NEG: mips_subu (code, ins->dreg, mips_zero, ins->sreg1); break; case CEE_MUL: #if 1 mips_mul (code, ins->dreg, ins->sreg1, ins->sreg2); #else mips_mult (code, ins->sreg1, ins->sreg2); mips_mflo (code, ins->dreg); mips_nop (code); mips_nop (code); #endif break; case OP_MUL_IMM: mips_load_const (code, mips_at, ins->inst_imm); #if 1 mips_mul (code, ins->dreg, ins->sreg1, mips_at); #else mips_mult (code, ins->sreg1, mips_at); mips_mflo (code, ins->dreg); mips_nop (code); mips_nop (code); #endif break; case CEE_MUL_OVF: #if 1 mips_mul (code, ins->dreg, ins->sreg1, ins->sreg2); #else mips_mult (code, ins->sreg1, ins->sreg2); mips_mflo (code, ins->dreg); mips_mfhi (code, mips_at); mips_nop (code); mips_nop (code); #endif /* XXX - Throw exception if we overflowed */ break; case CEE_MUL_OVF_UN: #if 1 mips_mul (code, ins->dreg, ins->sreg1, ins->sreg2); #else mips_mult (code, ins->sreg1, ins->sreg2); mips_mflo (code, ins->dreg); mips_mfhi (code, mips_at); mips_nop (code); mips_nop (code); #endif /* XXX - Throw exception if we overflowed */ break; case OP_ICONST: case OP_SETREGIMM: mips_load_const (code, ins->dreg, ins->inst_c0); break; case OP_AOTCONST: mono_add_patch_info (cfg, offset, (MonoJumpInfoType)ins->inst_i1, ins->inst_p0); mips_load (code, ins->dreg, 0); break; case OP_MIPS_MTC1S: mips_mtc1 (code, ins->dreg, ins->sreg1); break; case OP_MIPS_MFC1S: mips_mfc1 (code, ins->dreg, ins->sreg1); break; case OP_MIPS_MTC1D: mips_dmtc1 (code, ins->dreg, ins->sreg1); break; case OP_MIPS_MFC1D: mips_dmfc1 (code, ins->dreg, ins->sreg1); break; case CEE_CONV_I4: case CEE_CONV_U4: case OP_MOVE: case OP_SETREG: if (ins->dreg != ins->sreg1) mips_move (code, ins->dreg, ins->sreg1); break; case OP_SETLRET: /* Get sreg1 into v1, sreg2 into v0 */ if (ins->sreg1 == mips_v0) { if (ins->sreg1 != mips_at) mips_move (code, mips_at, ins->sreg1); if (ins->sreg2 != mips_v0) mips_move (code, mips_v0, ins->sreg2); mips_move (code, mips_v1, mips_at); } else { if (ins->sreg2 != mips_v0) mips_move (code, mips_v0, ins->sreg2); if (ins->sreg1 != mips_v1) mips_move (code, mips_v1, ins->sreg1); } break; case OP_SETFREG: if (ins->dreg != ins->sreg1) { mips_fmovd (code, ins->dreg, ins->sreg1); } break; case OP_FMOVE: if (ins->dreg != ins->sreg1) { mips_fmovd (code, ins->dreg, ins->sreg1); } break; case OP_MIPS_CVTSD: /* Convert from double to float and leave it there */ mips_cvtsd (code, ins->dreg, ins->sreg1); break; case OP_FCONV_TO_R4: /* Convert from double to float and back again */ mips_cvtsd (code, ins->dreg, ins->sreg1); mips_cvtds (code, ins->dreg, ins->dreg); break; case CEE_JMP: { int i, pos = 0; /* * Keep in sync with mono_arch_emit_epilog */ g_assert (!cfg->method->save_lmf); g_assert_not_reached(); #if 0 if (1 || cfg->flags & MONO_CFG_HAS_CALLS) { if (ppc_is_imm16 (cfg->stack_usage + MIPS_RET_ADDR_OFFSET)) { ppc_lwz (code, ppc_r0, cfg->stack_usage + MIPS_RET_ADDR_OFFSET, cfg->frame_reg); } else { ppc_load (code, ppc_r11, cfg->stack_usage + MIPS_RET_ADDR_OFFSET); ppc_lwzx (code, ppc_r0, cfg->frame_reg, ppc_r11); } ppc_mtlr (code, ppc_r0); } if (ppc_is_imm16 (cfg->stack_usage)) { ppc_addic (code, ppc_sp, cfg->frame_reg, cfg->stack_usage); } else { ppc_load (code, ppc_r11, cfg->stack_usage); ppc_add (code, ppc_sp, cfg->frame_reg, ppc_r11); } if (!cfg->method->save_lmf) { /*for (i = 31; i >= 14; --i) { if (cfg->used_float_regs & (1 << i)) { pos += sizeof (double); ppc_lfd (code, i, -pos, cfg->frame_reg); } }*/ for (i = 31; i >= 13; --i) { if (cfg->used_int_regs & (1 << i)) { pos += sizeof (gulong); ppc_lwz (code, i, -pos, cfg->frame_reg); } } } else { /* FIXME restore from MonoLMF: though this can't happen yet */ } #endif mono_add_patch_info (cfg, (guint8*) code - cfg->native_code, MONO_PATCH_INFO_METHOD_JUMP, ins->inst_p0); mips_beq (code, mips_zero, mips_zero, 0); mips_nop (code); break; } case OP_CHECK_THIS: /* ensure ins->sreg1 is not NULL */ mips_lw (code, mips_zero, ins->sreg1, 0); break; case OP_ARGLIST: { if (mips_is_imm16 (cfg->sig_cookie + cfg->stack_usage)) { mips_addiu (code, mips_at, cfg->frame_reg, cfg->sig_cookie + cfg->stack_usage); } else { mips_load_const (code, mips_at, cfg->sig_cookie + cfg->stack_usage); mips_add (code, mips_at, cfg->frame_reg, mips_at); } mips_sw (code, mips_at, ins->sreg1, 0); break; } case OP_FCALL: case OP_LCALL: case OP_VCALL: case OP_VOIDCALL: case CEE_CALL: case OP_FCALL_REG: case OP_LCALL_REG: case OP_VCALL_REG: case OP_VOIDCALL_REG: case OP_CALL_REG: case OP_FCALL_MEMBASE: case OP_LCALL_MEMBASE: case OP_VCALL_MEMBASE: case OP_VOIDCALL_MEMBASE: case OP_CALL_MEMBASE: switch (ins->opcode) { case OP_FCALL: case OP_LCALL: case OP_VCALL: case OP_VOIDCALL: case CEE_CALL: call = (MonoCallInst*)ins; if (ins->flags & MONO_INST_HAS_METHOD) mono_add_patch_info (cfg, offset, MONO_PATCH_INFO_METHOD, call->method); else mono_add_patch_info (cfg, offset, MONO_PATCH_INFO_ABS, call->fptr); mips_lui (code, mips_t9, mips_zero, 0); mips_addiu (code, mips_t9, mips_t9, 0); break; case OP_FCALL_REG: case OP_LCALL_REG: case OP_VCALL_REG: case OP_VOIDCALL_REG: case OP_CALL_REG: mips_move (code, mips_t9, ins->sreg1); break; case OP_FCALL_MEMBASE: case OP_LCALL_MEMBASE: case OP_VCALL_MEMBASE: case OP_VOIDCALL_MEMBASE: case OP_CALL_MEMBASE: mips_lw (code, mips_t9, ins->sreg1, ins->inst_offset); break; } mips_jalr (code, mips_t9, mips_ra); mips_nop (code); break; case OP_OUTARG: g_assert_not_reached (); break; case OP_LOCALLOC: { /* keep alignment */ int alloca_waste = MIPS_STACK_PARAM_OFFSET + cfg->param_area + 31; int area_offset = alloca_waste; area_offset &= ~31; mips_addiu (code, mips_at, ins->sreg1, -alloca_waste); #if 0 ppc_rlwinm (code, ppc_r11, ppc_r11, 0, 0, 27); /* use ctr to store the number of words to 0 if needed */ if (ins->flags & MONO_INST_INIT) { /* we zero 4 bytes at a time */ ppc_addi (code, ppc_r0, ins->sreg1, 3); ppc_srawi (code, ppc_r0, ppc_r0, 2); ppc_mtctr (code, ppc_r0); } ppc_lwz (code, ppc_r0, 0, ppc_sp); ppc_neg (code, ppc_r11, ppc_r11); ppc_stwux (code, ppc_r0, ppc_sp, ppc_r11); #endif if (ins->flags & MONO_INST_INIT) { mips_sw (code, mips_zero, ins->dreg, 0); mips_addiu (code, mips_at, mips_at, -4); mips_bne (code, mips_at, mips_zero, -3); mips_nop (code); } mips_addiu (code, ins->dreg, mips_sp, area_offset); break; } case CEE_RET: mips_jr (code, mips_ra); mips_nop (code); break; case CEE_THROW: { gpointer addr = mono_arch_get_throw_exception(); mips_move (code, mips_a0, ins->sreg1); mips_load_const (code, mips_t9, addr); mips_jr (code, mips_t9); mips_nop (code); mips_break (code, 0xfc); break; } case OP_RETHROW: { gpointer addr = mono_arch_get_rethrow_exception(); mips_move (code, mips_a0, ins->sreg1); mips_load_const (code, mips_t9, addr); mips_jr (code, mips_t9); mips_nop (code); mips_break (code, 0xfb); break; } case OP_START_HANDLER: /* * The START_HANDLER instruction marks the beginning of a handler * block. It is called using a call instruction, so mips_ra contains * the return address. Since the handler executes in the same stack * frame as the method itself, we can't use save/restore to save * the return address. Instead, we save it into a dedicated * variable. */ if (mips_is_imm16 (ins->inst_left->inst_offset)) { mips_sw (code, mips_ra, ins->inst_left->inst_basereg, ins->inst_left->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_left->inst_offset); mips_add (code, mips_at, mips_at, ins->inst_left->inst_basereg); mips_sw (code, mips_ra, mips_at, 0); } break; case OP_ENDFILTER: if (ins->sreg1 != mips_v0) mips_move (code, mips_v0, ins->sreg1); if (mips_is_imm16 (ins->inst_left->inst_offset)) { mips_lw (code, mips_ra, ins->inst_left->inst_basereg, ins->inst_left->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_left->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_left->inst_basereg); mips_lw (code, mips_ra, mips_at, 0); } mips_jr (code, mips_ra); mips_nop (code); break; case CEE_ENDFINALLY: mips_lw (code, mips_t9, ins->inst_left->inst_basereg, ins->inst_left->inst_offset); mips_jalr (code, mips_t9, mips_ra); mips_nop (code); break; case OP_CALL_HANDLER: mono_add_patch_info (cfg, offset, MONO_PATCH_INFO_BB, ins->inst_target_bb); mips_lui (code, mips_t9, mips_zero, 0); mips_addiu (code, mips_t9, mips_t9, 0); mips_jalr (code, mips_t9, mips_ra); mips_nop (code); break; case OP_LABEL: ins->inst_c0 = code - cfg->native_code; break; case CEE_BR: if (ins->flags & MONO_INST_BRLABEL) { mono_add_patch_info (cfg, offset, MONO_PATCH_INFO_LABEL, ins->inst_i0); mips_beq (code, mips_zero, mips_zero, 0); mips_nop (code); } else { mono_add_patch_info (cfg, offset, MONO_PATCH_INFO_BB, ins->inst_target_bb); mips_beq (code, mips_zero, mips_zero, 0); mips_nop (code); } break; case OP_BR_REG: mips_jr (code, ins->sreg1); mips_nop (code); break; case CEE_SWITCH: { int i; max_len += 4 * GPOINTER_TO_INT (ins->klass); if (offset > (cfg->code_size - max_len - 16)) { cfg->code_size += max_len; cfg->code_size *= 2; cfg->native_code = g_realloc (cfg->native_code, cfg->code_size); code = cfg->native_code + offset; } g_assert (ins->sreg1 != -1); mips_sll (code, mips_at, ins->sreg1, 2); if (1 || !(cfg->flags & MONO_CFG_HAS_CALLS)) mips_move (code, mips_t8, mips_ra); mips_bgezal (code, mips_zero, 1); /* bal */ mips_nop (code); mips_addu (code, mips_t9, mips_ra, mips_at); /* Table is 16 or 20 bytes from target of bal above */ if (1 || !(cfg->flags & MONO_CFG_HAS_CALLS)) { mips_move (code, mips_ra, mips_t8); mips_lw (code, mips_t9, mips_t9, 20); } else mips_lw (code, mips_t9, mips_t9, 16); mips_jalr (code, mips_t9, mips_t8); mips_nop (code); for (i = 0; i < GPOINTER_TO_INT (ins->klass); ++i) mips_emit32 (code, 0xfefefefe); break; } case OP_CEQ: /* XXX */ mips_addiu (code, ins->dreg, mips_zero, 1); mips_beq (code, mips_at, mips_zero, 2); mips_nop (code); mips_move (code, ins->dreg, mips_zero); break; case OP_CLT: case OP_CLT_UN: /* XXX */ mips_addiu (code, ins->dreg, mips_zero, 1); mips_bltz (code, mips_at, 2); mips_nop (code); mips_move (code, ins->dreg, mips_zero); break; case OP_CGT: case OP_CGT_UN: /* XXX */ mips_addiu (code, ins->dreg, mips_zero, 1); mips_bgtz (code, mips_at, 2); mips_nop (code); mips_move (code, ins->dreg, mips_zero); break; case OP_COND_EXC_EQ: case OP_COND_EXC_NE_UN: case OP_COND_EXC_LT: case OP_COND_EXC_LT_UN: case OP_COND_EXC_GT: case OP_COND_EXC_GT_UN: case OP_COND_EXC_GE: case OP_COND_EXC_GE_UN: case OP_COND_EXC_LE: case OP_COND_EXC_LE_UN: #if 0 /* Don't raise conditional exceptions at the moment */ switch (ins->opcode) { case OP_COND_EXC_EQ: mips_bne (code, mips_at, mips_zero, 5); break; case OP_COND_EXC_NE_UN: mips_beq (code, mips_at, mips_zero, 5); break; case OP_COND_EXC_LT: case OP_COND_EXC_LT_UN: mips_bgez (code, mips_at, 5); break; case OP_COND_EXC_GT: case OP_COND_EXC_GT_UN: mips_blez (code, mips_at, 5); break; case OP_COND_EXC_GE: case OP_COND_EXC_GE_UN: mips_bltz (code, mips_at, 5); break; case OP_COND_EXC_LE: case OP_COND_EXC_LE_UN: mips_bgtz (code, mips_at, 5); break; } mips_nop (code); mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_EXC, ins->inst_p1); mips_lui (code, mips_t9, mips_zero, 0); mips_addiu (code, mips_t9, mips_t9, 0); mips_jalr (code, mips_t9, mips_ra); mips_nop (code); cfg->bb_exit->max_offset += 24; #endif break; case OP_COND_EXC_C: g_assert_not_reached(); #if 0 /* check XER [0-3] (SO, OV, CA): we can't use mcrxr */ ppc_mfspr (code, ppc_r0, ppc_xer); ppc_andisd (code, ppc_r0, ppc_r0, (1<<14)); EMIT_COND_SYSTEM_EXCEPTION_FLAGS (PPC_BR_FALSE, PPC_BR_EQ, "OverflowException"); #endif break; case OP_COND_EXC_OV: g_assert_not_reached(); #if 0 ppc_mcrxr (code, 0); EMIT_COND_SYSTEM_EXCEPTION (CEE_BGT - CEE_BEQ, ins->inst_p1); #endif break; case OP_COND_EXC_NC: case OP_COND_EXC_NO: g_assert_not_reached (); break; case CEE_BEQ: case CEE_BNE_UN: case CEE_BLT: case CEE_BLT_UN: case CEE_BGT: case CEE_BGT_UN: case CEE_BGE: case CEE_BGE_UN: case CEE_BLE: case CEE_BLE_UN: /* Should be re-mapped to OP_MIPS_B* by *.inssel-mips.brg */ g_assert_not_reached (); break; case OP_MIPS_BEQ: if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_beq (code, ins->sreg1, ins->sreg2, 0); mips_nop (code); break; case OP_MIPS_BNE: if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_bne (code, ins->sreg1, ins->sreg2, 0); mips_nop (code); break; case OP_MIPS_BGEZ: if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_bgez (code, ins->sreg1, 0); mips_nop (code); break; case OP_MIPS_BGTZ: if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_bgtz (code, ins->sreg1, 0); mips_nop (code); break; case OP_MIPS_BLEZ: if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_blez (code, ins->sreg1, 0); mips_nop (code); break; case OP_MIPS_BLTZ: if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_bltz (code, ins->sreg1, 0); mips_nop (code); break; /* floating point opcodes */ case OP_R8CONST: if (((guint32)ins->inst_p0) & (1 << 15)) mips_lui (code, mips_at, mips_zero, (((guint32)ins->inst_p0)>>16)+1); else mips_lui (code, mips_at, mips_zero, (((guint32)ins->inst_p0)>>16)); mips_ldc1 (code, ins->dreg, mips_at, ((guint32)ins->inst_p0) & 0xffff); break; case OP_R4CONST: if (((guint32)ins->inst_p0) & (1 << 15)) mips_lui (code, mips_at, mips_zero, (((guint32)ins->inst_p0)>>16)+1); else mips_lui (code, mips_at, mips_zero, (((guint32)ins->inst_p0)>>16)); mips_lwc1 (code, ins->dreg, mips_at, ((guint32)ins->inst_p0) & 0xffff); mips_cvtds (code, ins->dreg, ins->dreg); break; case OP_STORER8_MEMBASE_REG: if (mips_is_imm16 (ins->inst_offset)) { #if 1 mips_sdc1 (code, ins->sreg1, ins->inst_destbasereg, ins->inst_offset); #else mips_swc1 (code, ins->sreg1, ins->inst_destbasereg, ins->inst_offset+4); mips_swc1 (code, ins->sreg1+1, ins->inst_destbasereg, ins->inst_offset); #endif } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_destbasereg); mips_swc1 (code, ins->sreg1, mips_at, 4); mips_swc1 (code, ins->sreg1+1, mips_at, 0); } break; case OP_LOADR8_MEMBASE: if (mips_is_imm16 (ins->inst_offset)) { #if 1 mips_ldc1 (code, ins->dreg, ins->inst_basereg, ins->inst_offset); #else mips_lwc1 (code, ins->dreg, ins->inst_basereg, ins->inst_offset+4); mips_lwc1 (code, ins->dreg+1, ins->inst_basereg, ins->inst_offset); #endif } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_basereg); mips_lwc1 (code, ins->dreg, mips_at, 4); mips_lwc1 (code, ins->dreg+1, mips_at, 0); } break; case OP_STORER4_MEMBASE_REG: /* XXX Need to convert ins->sreg1 to single-precision first */ mips_cvtsd (code, mips_ftemp, ins->sreg1); if (mips_is_imm16 (ins->inst_offset)) { mips_swc1 (code, mips_ftemp, ins->inst_destbasereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_destbasereg); mips_swc1 (code, mips_ftemp, mips_at, 0); } break; case OP_MIPS_LWC1: if (mips_is_imm16 (ins->inst_offset)) { mips_lwc1 (code, ins->dreg, ins->inst_basereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_basereg); mips_lwc1 (code, ins->dreg, mips_at, 0); } break; case OP_LOADR4_MEMBASE: if (mips_is_imm16 (ins->inst_offset)) { mips_lwc1 (code, ins->dreg, ins->inst_basereg, ins->inst_offset); } else { mips_load_const (code, mips_at, ins->inst_offset); mips_addu (code, mips_at, mips_at, ins->inst_basereg); mips_lwc1 (code, ins->dreg, mips_at, 0); } /* Convert to double precision in place */ mips_cvtds (code, ins->dreg, ins->dreg); break; case CEE_CONV_R_UN: { g_assert_not_reached(); #if 0 static const guint64 adjust_val = 0x4330000000000000ULL; ppc_addis (code, ppc_r0, ppc_r0, 0x4330); ppc_stw (code, ppc_r0, -8, ppc_sp); ppc_stw (code, ins->sreg1, -4, ppc_sp); ppc_load (code, ppc_r11, &adjust_val); ppc_lfd (code, ins->dreg, -8, ppc_sp); ppc_lfd (code, ppc_f0, 0, ppc_r11); ppc_fsub (code, ins->dreg, ins->dreg, ppc_f0); #endif break; } case CEE_CONV_R4: mips_mtc1 (code, mips_ftemp, ins->sreg1); mips_cvtsw (code, ins->dreg, mips_ftemp); mips_cvtds (code, ins->dreg, ins->dreg); break; case CEE_CONV_R8: mips_mtc1 (code, mips_ftemp, ins->sreg1); mips_cvtdw (code, ins->dreg, mips_ftemp); break; case OP_FCONV_TO_I1: code = emit_float_to_int (cfg, code, ins->dreg, ins->sreg1, 1, TRUE); break; case OP_FCONV_TO_U1: code = emit_float_to_int (cfg, code, ins->dreg, ins->sreg1, 1, FALSE); break; case OP_FCONV_TO_I2: code = emit_float_to_int (cfg, code, ins->dreg, ins->sreg1, 2, TRUE); break; case OP_FCONV_TO_U2: code = emit_float_to_int (cfg, code, ins->dreg, ins->sreg1, 2, FALSE); break; case OP_FCONV_TO_I4: case OP_FCONV_TO_I: code = emit_float_to_int (cfg, code, ins->dreg, ins->sreg1, 4, TRUE); break; case OP_FCONV_TO_U4: case OP_FCONV_TO_U: code = emit_float_to_int (cfg, code, ins->dreg, ins->sreg1, 4, FALSE); break; case OP_FCONV_TO_I8: case OP_FCONV_TO_U8: g_assert_not_reached (); /* Implemented as helper calls */ break; case OP_LCONV_TO_R_UN: g_assert_not_reached (); /* Implemented as helper calls */ break; case OP_LCONV_TO_OVF_I: { #if 1 if (ins->dreg != ins->sreg1) mips_move (code, ins->dreg, ins->sreg1); #else guint32 *negative_branch, *msword_positive_branch, *msword_negative_branch, *ovf_ex_target; g_assert_not_reached(); // Check if its negative ppc_cmpi (code, 0, 0, ins->sreg1, 0); negative_branch = (guint32 *)(void *)code; ppc_bc (code, PPC_BR_TRUE, PPC_BR_LT, 0); // Its positive msword == 0 ppc_cmpi (code, 0, 0, ins->sreg2, 0); msword_positive_branch = (guint32 *)(void *)code; ppc_bc (code, PPC_BR_TRUE, PPC_BR_EQ, 0); ovf_ex_target = (guint32 *)(void *)code; EMIT_COND_SYSTEM_EXCEPTION_FLAGS (PPC_BR_ALWAYS, 0, "OverflowException"); // Negative mips_patch (negative_branch, (guint32)code); ppc_cmpi (code, 0, 0, ins->sreg2, -1); msword_negative_branch = (guint32 *)(void *)code; ppc_bc (code, PPC_BR_FALSE, PPC_BR_EQ, 0); mips_patch (msword_negative_branch, (guint32)ovf_ex_target); mips_patch (msword_positive_branch, (guint32)code); if (ins->dreg != ins->sreg1) ppc_mr (code, ins->dreg, ins->sreg1); #endif break; } case OP_SQRT: mips_fsqrtd (code, ins->dreg, ins->sreg1); break; case OP_FADD: mips_faddd (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_FSUB: mips_fsubd (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_FMUL: mips_fmuld (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_FDIV: mips_fdivd (code, ins->dreg, ins->sreg1, ins->sreg2); break; case OP_FNEG: mips_fnegd (code, ins->dreg, ins->sreg1); break; case OP_FREM: /* emulated */ g_assert_not_reached (); break; case OP_FCOMPARE: g_assert_not_reached(); break; case OP_FCEQ: mips_fcmpd (code, MIPS_FPU_EQ, ins->sreg1, ins->sreg2); mips_addiu (code, ins->dreg, mips_zero, 1); mips_fbtrue (code, 2); mips_nop (code); mips_move (code, ins->dreg, mips_zero); break; case OP_FCLT: mips_fcmpd (code, MIPS_FPU_LT, ins->sreg1, ins->sreg2); mips_addiu (code, ins->dreg, mips_zero, 1); mips_fbtrue (code, 2); mips_nop (code); mips_move (code, ins->dreg, mips_zero); break; case OP_FCLT_UN: /* Less than, or Unordered */ mips_fcmpd (code, MIPS_FPU_ULT, ins->sreg1, ins->sreg2); mips_addiu (code, ins->dreg, mips_zero, 1); mips_fbtrue (code, 2); mips_nop (code); mips_move (code, ins->dreg, mips_zero); break; case OP_FCGT: mips_fcmpd (code, MIPS_FPU_ULE, ins->sreg1, ins->sreg2); mips_move (code, ins->dreg, mips_zero); mips_fbtrue (code, 2); mips_nop (code); mips_addiu (code, ins->dreg, mips_zero, 1); break; case OP_FCGT_UN: /* Greater than, or Unordered */ mips_fcmpd (code, MIPS_FPU_OLE, ins->sreg1, ins->sreg2); mips_move (code, ins->dreg, mips_zero); mips_fbtrue (code, 2); mips_nop (code); mips_addiu (code, ins->dreg, mips_zero, 1); break; case OP_FBEQ: mips_fcmpd (code, MIPS_FPU_EQ, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbtrue (code, 0); mips_nop (code); break; case OP_FBNE_UN: mips_fcmpd (code, MIPS_FPU_EQ, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbfalse (code, 0); mips_nop (code); break; case OP_FBLT: mips_fcmpd (code, MIPS_FPU_LT, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbtrue (code, 0); mips_nop (code); break; case OP_FBLT_UN: mips_fcmpd (code, MIPS_FPU_ULT, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbtrue (code, 0); mips_nop (code); break; case OP_FBGT: mips_fcmpd (code, MIPS_FPU_LE, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbfalse (code, 0); mips_nop (code); break; case OP_FBGT_UN: mips_fcmpd (code, MIPS_FPU_OLE, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbfalse (code, 0); mips_nop (code); break; case OP_FBGE: mips_fcmpd (code, MIPS_FPU_LT, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbfalse (code, 0); mips_nop (code); break; case OP_FBGE_UN: mips_fcmpd (code, MIPS_FPU_OLT, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbfalse (code, 0); mips_nop (code); break; case OP_FBLE: mips_fcmpd (code, MIPS_FPU_OLE, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbtrue (code, 0); mips_nop (code); break; case OP_FBLE_UN: mips_fcmpd (code, MIPS_FPU_ULE, ins->sreg1, ins->sreg2); mips_nop (code); if (ins->flags & MONO_INST_BRLABEL) mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_LABEL, ins->inst_i0); else mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_BB, ins->inst_true_bb); mips_fbtrue (code, 0); mips_nop (code); break; case CEE_CKFINITE: { g_assert_not_reached(); #if 0 ppc_stfd (code, ins->sreg1, -8, ppc_sp); ppc_lwz (code, ppc_r11, -8, ppc_sp); ppc_rlwinm (code, ppc_r11, ppc_r11, 0, 1, 31); ppc_addis (code, ppc_r11, ppc_r11, -32752); ppc_rlwinmd (code, ppc_r11, ppc_r11, 1, 31, 31); EMIT_COND_SYSTEM_EXCEPTION (CEE_BEQ - CEE_BEQ, "ArithmeticException"); #endif break; } default: g_warning ("unknown opcode %s in %s()\n", mono_inst_name (ins->opcode), __FUNCTION__); g_assert_not_reached (); } if ((cfg->opt & MONO_OPT_BRANCH) && ((code - cfg->native_code - offset) > max_len)) { g_warning ("wrong maximal instruction length of instruction %s (expected %d, got %d)", mono_inst_name (ins->opcode), max_len, code - cfg->native_code - offset); g_assert_not_reached (); } cpos += max_len; last_ins = ins; last_offset = offset; ins = ins->next; } cfg->code_len = code - cfg->native_code; } void mono_arch_register_lowlevel_calls (void) { } void mono_arch_patch_code (MonoMethod *method, MonoDomain *domain, guint8 *code, MonoJumpInfo *ji, gboolean run_cctors) { MonoJumpInfo *patch_info; for (patch_info = ji; patch_info; patch_info = patch_info->next) { unsigned char *ip = patch_info->ip.i + code; const unsigned char *target; target = mono_resolve_patch_target (method, domain, code, patch_info, run_cctors); switch (patch_info->type) { case MONO_PATCH_INFO_IP: patch_lui_addiu ((guint32 *)(void *)ip, (guint32)ip); continue; case MONO_PATCH_INFO_SWITCH: { /* jt is the inlined jump table, 7 or 9 instructions after ip * In the normal case we store the absolute addresses. * otherwise the displacements. */ int i; gpointer *table = (gpointer *)patch_info->data.table->table; gpointer *jt = ((gpointer*)(void *)ip) + 7; if (1 /* || !(cfg->->flags & MONO_CFG_HAS_CALLS) */) jt += 2; for (i = 0; i < patch_info->data.table->table_size; i++) { jt [i] = code + (int)table [i]; } continue; } case MONO_PATCH_INFO_METHODCONST: case MONO_PATCH_INFO_CLASS: case MONO_PATCH_INFO_IMAGE: case MONO_PATCH_INFO_FIELD: case MONO_PATCH_INFO_VTABLE: case MONO_PATCH_INFO_IID: case MONO_PATCH_INFO_SFLDA: case MONO_PATCH_INFO_LDSTR: case MONO_PATCH_INFO_TYPE_FROM_HANDLE: case MONO_PATCH_INFO_LDTOKEN: case MONO_PATCH_INFO_R4: case MONO_PATCH_INFO_R8: /* from OP_AOTCONST : lui + addiu */ patch_lui_addiu ((guint32 *)(void *)ip, (guint32)target); continue; case MONO_PATCH_INFO_EXC_NAME: g_assert_not_reached (); *((gconstpointer *)(void *)(ip + 1)) = patch_info->data.name; continue; case MONO_PATCH_INFO_NONE: case MONO_PATCH_INFO_BB_OVF: case MONO_PATCH_INFO_EXC_OVF: /* everything is dealt with at epilog output time */ continue; default: break; } mips_patch ((guint32 *)(void *)ip, (guint32)target); } } /* * Allow tracing to work with this interface (with an optional argument) * * This code is expected to be inserted just after the 'real' prolog code, * and before the first basic block. We need to allocate a 2nd, temporary * stack frame so that we can preserve f12-f15 as well as a0-a3. */ void* mono_arch_instrument_prolog (MonoCompile *cfg, void *func, void *p, gboolean enable_arguments) { guchar *code = p; int fp_stack_offset = 0; mips_nop (code); mips_nop (code); mips_nop (code); mips_sw (code, mips_a0, mips_sp, cfg->stack_offset + 0); mips_sw (code, mips_a1, mips_sp, cfg->stack_offset + 4); mips_sw (code, mips_a2, mips_sp, cfg->stack_offset + 8); mips_sw (code, mips_a3, mips_sp, cfg->stack_offset + 12); #if 0 #if 0 fp_stack_offset = MIPS_STACK_PARAM_OFFSET; mips_addiu (code, mips_sp, mips_sp, -64); mips_swc1 (code, mips_f12, mips_sp, fp_stack_offset + 16); mips_swc1 (code, mips_f13, mips_sp, fp_stack_offset + 20); mips_swc1 (code, mips_f14, mips_sp, fp_stack_offset + 24); mips_swc1 (code, mips_f15, mips_sp, fp_stack_offset + 28); #else mips_fmovs (code, mips_f22, mips_f12); mips_fmovs (code, mips_f23, mips_f13); mips_fmovs (code, mips_f24, mips_f14); mips_fmovs (code, mips_f25, mips_f15); #endif #endif mips_load_const (code, mips_a0, cfg->method); mips_addiu (code, mips_a1, mips_sp, cfg->stack_offset + fp_stack_offset); mips_load_const (code, mips_t9, func); mips_jalr (code, mips_t9, mips_ra); mips_nop (code); mips_lw (code, mips_a0, mips_sp, cfg->stack_offset + 0); mips_lw (code, mips_a1, mips_sp, cfg->stack_offset + 4); mips_lw (code, mips_a2, mips_sp, cfg->stack_offset + 8); mips_lw (code, mips_a3, mips_sp, cfg->stack_offset + 12); #if 0 #if 0 mips_lwc1 (code, mips_f12, mips_sp, fp_stack_offset + 16); mips_lwc1 (code, mips_f13, mips_sp, fp_stack_offset + 20); mips_lwc1 (code, mips_f14, mips_sp, fp_stack_offset + 24); mips_lwc1 (code, mips_f15, mips_sp, fp_stack_offset + 28); mips_addiu (code, mips_sp, mips_sp, 64); #else mips_fmovs (code, mips_f12, mips_f22); mips_fmovs (code, mips_f13, mips_f23); mips_fmovs (code, mips_f14, mips_f24); mips_fmovs (code, mips_f15, mips_f25); #endif #endif mips_nop (code); mips_nop (code); mips_nop (code); return code; } /* * Stack frame layout: * * ------------------- sp + cfg->stack_usage + MIPS_STACK_PARAM_OFFSET + cfg->param_area * param area incoming * ------------------- sp + cfg->stack_usage + MIPS_STACK_PARAM_OFFSET * a0-a3 incoming * ------------------- sp + cfg->stack_usage * ra * ------------------- sp + cfg->stack_usage-4 * spilled regs * ------------------- sp + * MonoLMF structure optional * ------------------- sp + cfg->arch.lmf_offset * saved registers s0-s8 * ------------------- sp + cfg->arch.iregs_offset * locals * ------------------- sp + cfg->param_area * param area outgoing * ------------------- sp + 16 * a0-a3 outgoing * ------------------- sp * red zone */ guint8 * mono_arch_emit_prolog (MonoCompile *cfg) { MonoMethod *method = cfg->method; MonoMethodSignature *sig; MonoInst *inst; int alloc_size, pos, i; guint8 *code; CallInfo *cinfo; int tracing = 0; guint32 iregs_to_save = 0; guint32 fregs_to_save = 0; if (mono_jit_trace_calls != NULL && mono_trace_eval (method)) tracing = 1; if (tracing) cfg->flags |= MONO_CFG_HAS_CALLS; sig = mono_method_signature (method); cfg->code_size = 256 + sig->param_count * 20; code = cfg->native_code = g_malloc (cfg->code_size); alloc_size = cfg->stack_offset; g_assert ((alloc_size & (MIPS_STACK_ALIGNMENT-1)) == 0); /* re-align cfg->stack_offset if needed (due to var spilling in mini-codegen.c) */ cfg->stack_offset = (cfg->stack_offset + MIPS_STACK_ALIGNMENT - 1) & ~(MIPS_STACK_ALIGNMENT - 1); /* stack_offset should not be changed here. */ alloc_size = cfg->stack_offset; cfg->stack_usage = alloc_size; iregs_to_save = (cfg->used_int_regs & MONO_ARCH_CALLEE_SAVED_REGS); #ifdef SAVE_FP_REGS #if 0 fregs_to_save = (cfg->used_float_regs & MONO_ARCH_CALLEE_SAVED_FREGS); #else fregs_to_save = MONO_ARCH_CALLEE_SAVED_FREGS; fregs_to_save |= (fregs_to_save << 1); #endif #endif if (alloc_size) { if (mips_is_imm16 (-alloc_size)) { mips_addiu (code, mips_sp, mips_sp, -alloc_size); } else { mips_load_const (code, mips_at, -alloc_size); mips_addu (code, mips_sp, mips_sp, mips_at); } } if ((cfg->flags & MONO_CFG_HAS_CALLS) || ALWAYS_SAVE_RA) mips_sw (code, mips_ra, mips_sp, alloc_size + MIPS_RET_ADDR_OFFSET); /* Do instrumentation before assigning regvars to registers. Because they may be assigned * to the t* registers, which would be clobbered by the instrumentation calls. */ if (tracing) code = mono_arch_instrument_prolog (cfg, mono_trace_enter_method, code, TRUE); /* XXX - optimize this later to not save all regs if LMF constructed */ if (iregs_to_save) { /* save used registers in own stack frame (at pos) */ pos = cfg->arch.iregs_offset; for (i = MONO_MAX_IREGS-1; i >= 0; --i) { if (iregs_to_save & (1 << i)) { g_assert (pos < cfg->stack_usage - 4); mips_sw (code, i, mips_sp, pos); pos += sizeof (gulong); } } } #ifdef SAVE_FP_REGS /* Save float registers */ if (fregs_to_save) { for (i = MONO_MAX_FREGS-1; i >= 0; --i) { if (fregs_to_save & (1 << i)) { g_assert (pos < cfg->stack_usage - MIPS_STACK_ALIGNMENT); mips_swc1 (code, i, mips_sp, pos); pos += sizeof (gulong); } } } #endif #ifdef SAVE_LMF if (method->save_lmf) { #if 0 ppc_stmw (code, ppc_r13, ppc_r1, ofs); for (i = 14; i < 32; i++) { ppc_stfd (code, i, (-pos + G_STRUCT_OFFSET(MonoLMF, fregs) + ((i-14) * sizeof (gdouble))), ppc_r1); } #endif } #endif if (cfg->frame_reg != mips_sp) mips_move (code, cfg->frame_reg, mips_sp); /* load arguments allocated to register from the stack */ pos = 0; cinfo = calculate_sizes (sig, sig->pinvoke); if (MONO_TYPE_ISSTRUCT (sig->ret)) { ArgInfo *ainfo = &cinfo->ret; inst = cfg->ret; if (inst->opcode == OP_REGVAR) mips_move (code, inst->dreg, ainfo->reg); else if (mips_is_imm16 (inst->inst_offset)) { mips_sw (code, ainfo->reg, inst->inst_basereg, inst->inst_offset); } else { mips_load_const (code, mips_at, inst->inst_offset); mips_addu (code, mips_at, mips_at, inst->inst_basereg); mips_sw (code, ainfo->reg, mips_at, 0); } } for (i = 0; i < sig->param_count + sig->hasthis; ++i) { ArgInfo *ainfo = cinfo->args + i; inst = cfg->varinfo [pos]; if (cfg->verbose_level > 2) g_print ("Saving argument %d (type: %d)\n", i, ainfo->regtype); if (inst->opcode == OP_REGVAR) { /* Argument ends up in a register */ if (ainfo->regtype == RegTypeGeneral) mips_move (code, inst->dreg, ainfo->reg); else if (ainfo->regtype == RegTypeFP) { g_assert_not_reached(); #if 0 ppc_fmr (code, inst->dreg, ainfo->reg); #endif } else if (ainfo->regtype == RegTypeBase) { mips_lw (code, inst->dreg, mips_sp, cfg->stack_usage + ainfo->offset); } else g_assert_not_reached (); if (cfg->verbose_level > 2) g_print ("Argument %d assigned to register %s\n", pos, mono_arch_regname (inst->dreg)); } else { /* Argument ends up on the stack */ if (ainfo->regtype == RegTypeGeneral) { /* Incoming parameters should be above this frame */ g_assert (inst->inst_offset >= alloc_size); g_assert (mips_is_imm16 (inst->inst_offset)); switch (ainfo->size) { case 1: mips_sb (code, ainfo->reg, inst->inst_basereg, inst->inst_offset); break; case 2: mips_sh (code, ainfo->reg, inst->inst_basereg, inst->inst_offset); break; case 0: /* XXX */ case 4: mips_sw (code, ainfo->reg, inst->inst_basereg, inst->inst_offset); break; case 8: mips_sw (code, ainfo->reg, inst->inst_basereg, inst->inst_offset); mips_sw (code, ainfo->reg + 1, inst->inst_basereg, inst->inst_offset + 4); break; default: g_assert_not_reached (); break; } } else if (ainfo->regtype == RegTypeBase) { /* * Argument comes in on the stack, and ends up on the stack * 1 and 2 byte args are passed as 32-bit quantities, but used as * 8 and 16 bit quantities. Shorten them in place. */ g_assert (mips_is_imm16 (inst->inst_offset)); switch (ainfo->size) { case 1: mips_lw (code, mips_at, inst->inst_basereg, inst->inst_offset); mips_sb (code, mips_at, inst->inst_basereg, inst->inst_offset); break; case 2: mips_lw (code, mips_at, inst->inst_basereg, inst->inst_offset); mips_sh (code, mips_at, inst->inst_basereg, inst->inst_offset); break; case 0: /* XXX */ case 4: case 8: break; default: g_assert_not_reached (); } } else if (ainfo->regtype == RegTypeFP) { g_assert (mips_is_imm16 (inst->inst_offset)); if (ainfo->size == 8) { #if 1 mips_sdc1 (code, ainfo->reg, inst->inst_basereg, inst->inst_offset); #else mips_swc1 (code, ainfo->reg, inst->inst_basereg, inst->inst_offset+4); mips_swc1 (code, ainfo->reg+1, inst->inst_basereg, inst->inst_offset); #endif } else if (ainfo->size == 4) mips_swc1 (code, ainfo->reg, inst->inst_basereg, inst->inst_offset); else g_assert_not_reached (); } else if (ainfo->regtype == RegTypeStructByVal) { int i; int doffset = inst->inst_offset; g_assert (mips_is_imm16 (inst->inst_offset)); g_assert (mips_is_imm16 (inst->inst_offset + ainfo->size * sizeof (gpointer))); /* Push the argument registers into their stack slots */ for (i = 0; i < ainfo->size; ++i) { mips_sw (code, ainfo->reg + i, inst->inst_basereg, doffset); doffset += sizeof (gpointer); } } else if (ainfo->regtype == RegTypeStructByAddr) { g_assert (mips_is_imm16 (inst->inst_offset)); /* FIXME: handle overrun! with struct sizes not multiple of 4 */ code = emit_memcpy (code, ainfo->vtsize * sizeof (gpointer), inst->inst_basereg, inst->inst_offset, ainfo->reg, 0); } else g_assert_not_reached (); } pos++; } if (method->wrapper_type == MONO_WRAPPER_NATIVE_TO_MANAGED) { g_assert_not_reached(); #if 0 ppc_load (code, ppc_r3, cfg->domain); mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_INTERNAL_METHOD, (gpointer)"mono_jit_thread_attach"); ppc_bl (code, 0); #endif } #ifdef SAVE_LMF if (method->save_lmf) { if (lmf_pthread_key != -1) { g_assert_not_reached(); #if 0 emit_tls_access (code, mips_temp, lmf_pthread_key); #endif if (G_STRUCT_OFFSET (MonoJitTlsData, lmf)) mips_addiu (code, mips_a0, mips_temp, G_STRUCT_OFFSET (MonoJitTlsData, lmf)); } else { mips_load_const (code, mips_t9, (gpointer)mono_get_lmf_addr); mips_jalr (code, mips_t9, mips_ra); mips_nop (code); } /* we build the MonoLMF structure on the stack - see mini-mips.h * The pointer to the struct is put in mips_t2 (new_lmf). * (t2 = r11, v0 = r3 in PPC) */ /* lmf_offset is the offset of the LMF from our stack pointer. * The callee-saved registers are already in the MonoLMF structure */ mips_addiu (code, mips_t2, mips_sp, cfg->arch.lmf_offset); mips_sw (code, mips_a0, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_a0])); mips_sw (code, mips_a1, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_a1])); mips_sw (code, mips_a2, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_a2])); mips_sw (code, mips_a3, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_a3])); mips_sw (code, mips_s0, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_s0])); mips_sw (code, mips_s1, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_s1])); mips_sw (code, mips_s2, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_s2])); mips_sw (code, mips_s3, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_s3])); mips_sw (code, mips_s4, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_s4])); mips_sw (code, mips_s5, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_s5])); mips_sw (code, mips_s6, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_s6])); mips_sw (code, mips_s7, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_s7])); mips_sw (code, mips_fp, mips_t2, G_STRUCT_OFFSET(MonoLMF, iregs[mips_fp])); /* mips_v0 is the result from mono_get_lmf_addr () (MonoLMF **) */ mips_sw (code, mips_v0, mips_t2, G_STRUCT_OFFSET(MonoLMF, lmf_addr)); /* new_lmf->previous_lmf = *lmf_addr */ mips_lw (code, mips_at, mips_v0, 0); mips_sw (code, mips_at, mips_t2, G_STRUCT_OFFSET(MonoLMF, previous_lmf)); /* *(lmf_addr) = t2 */ mips_sw (code, mips_t2, mips_v0, 0); /* save method info */ mips_load_const (code, mips_at, method); mips_sw (code, mips_at, mips_t2, G_STRUCT_OFFSET(MonoLMF, method)); mips_sw (code, mips_sp, mips_t2, G_STRUCT_OFFSET(MonoLMF, ebp)); /* save the current IP */ mono_add_patch_info (cfg, code - cfg->native_code, MONO_PATCH_INFO_IP, NULL); mips_load_const (code, mips_at, 0x01010101); mips_sw (code, mips_at, mips_t2, G_STRUCT_OFFSET(MonoLMF, eip)); } #endif cfg->code_len = code - cfg->native_code; g_assert (cfg->code_len < cfg->code_size); g_free (cinfo); return code; } enum { SAVE_NONE, SAVE_STRUCT, SAVE_ONE, SAVE_TWO, SAVE_FP }; void* mono_arch_instrument_epilog (MonoCompile *cfg, void *func, void *p, gboolean enable_arguments) { guchar *code = p; int save_mode = SAVE_NONE; int offset; MonoMethod *method = cfg->method; int rtype = mono_type_get_underlying_type (mono_method_signature (method)->ret)->type; int save_offset = MIPS_STACK_PARAM_OFFSET + cfg->param_area; save_offset += 15; save_offset &= ~15; offset = code - cfg->native_code; /* we need about 16 instructions */ if (offset > (cfg->code_size - 16 * 4)) { cfg->code_size *= 2; cfg->native_code = g_realloc (cfg->native_code, cfg->code_size); code = cfg->native_code + offset; } mips_nop (code); mips_nop (code); mips_nop (code); switch (rtype) { case MONO_TYPE_VOID: /* special case string .ctor icall */ if (strcmp (".ctor", method->name) && method->klass == mono_defaults.string_class) save_mode = SAVE_ONE; else save_mode = SAVE_NONE; break; case MONO_TYPE_I8: case MONO_TYPE_U8: save_mode = SAVE_TWO; break; case MONO_TYPE_R4: case MONO_TYPE_R8: save_mode = SAVE_FP; break; case MONO_TYPE_VALUETYPE: save_mode = SAVE_STRUCT; break; default: save_mode = SAVE_ONE; break; } switch (save_mode) { case SAVE_TWO: mips_sw (code, mips_v0, cfg->frame_reg, save_offset); mips_sw (code, mips_v1, cfg->frame_reg, save_offset + 4); if (enable_arguments) { mips_move (code, mips_a1, mips_v0); mips_move (code, mips_a2, mips_v1); } break; case SAVE_ONE: mips_sw (code, mips_v0, cfg->frame_reg, save_offset); if (enable_arguments) { mips_move (code, mips_a1, mips_v0); } break; case SAVE_FP: g_assert_not_reached(); #if 0 ppc_stfd (code, ppc_f1, save_offset, cfg->frame_reg); if (enable_arguments) { /* FIXME: what reg? */ ppc_fmr (code, ppc_f3, ppc_f1); ppc_lwz (code, ppc_r4, save_offset, cfg->frame_reg); ppc_lwz (code, ppc_r5, save_offset + 4, cfg->frame_reg); } #endif break; case SAVE_STRUCT: case SAVE_NONE: default: break; } mips_load_const (code, mips_a0, cfg->method); mips_load_const (code, mips_t9, func); mips_jalr (code, mips_t9, mips_ra); mips_nop (code); switch (save_mode) { case SAVE_TWO: mips_lw (code, mips_v0, cfg->frame_reg, save_offset); mips_lw (code, mips_v1, cfg->frame_reg, save_offset + 4); break; case SAVE_ONE: mips_lw (code, mips_v0, cfg->frame_reg, save_offset); break; case SAVE_FP: g_assert_not_reached(); #if 0 ppc_lfd (code, ppc_f1, save_offset, cfg->frame_reg); #endif break; case SAVE_STRUCT: case SAVE_NONE: default: break; } mips_nop (code); mips_nop (code); mips_nop (code); return code; } void mono_arch_emit_epilog (MonoCompile *cfg) { MonoMethod *method = cfg->method; int pos, i; int max_epilog_size = 16 + 20*4; guint8 *code; guint32 iregs_to_restore; guint32 fregs_to_restore; #ifdef SAVE_LMF if (cfg->method->save_lmf) max_epilog_size += 128; #endif if (mono_jit_trace_calls != NULL) max_epilog_size += 50; if (cfg->prof_options & MONO_PROFILE_ENTER_LEAVE) max_epilog_size += 50; while (cfg->code_len + max_epilog_size > (cfg->code_size - 16)) { cfg->code_size *= 2; cfg->native_code = g_realloc (cfg->native_code, cfg->code_size); mono_jit_stats.code_reallocs++; } /* * Keep in sync with CEE_JMP */ code = cfg->native_code + cfg->code_len; if (mono_jit_trace_calls != NULL && mono_trace_eval (method)) { code = mono_arch_instrument_epilog (cfg, mono_trace_leave_method, code, TRUE); } pos = cfg->arch.iregs_offset; iregs_to_restore = (cfg->used_int_regs & MONO_ARCH_CALLEE_SAVED_REGS); if (iregs_to_restore) { for (i = MONO_MAX_IREGS-1; i >= 0; --i) { if (iregs_to_restore & (1 << i)) { mips_lw (code, i, mips_sp, pos); pos += sizeof (gulong); } } } #ifdef SAVE_FP_REGS #if 0 fregs_to_restore = (cfg->used_float_regs & MONO_ARCH_CALLEE_SAVED_FREGS); #else fregs_to_restore = MONO_ARCH_CALLEE_SAVED_FREGS; fregs_to_restore |= (fregs_to_restore << 1); #endif if (fregs_to_restore) { for (i = MONO_MAX_FREGS-1; i >= 0; --i) { if (fregs_to_restore & (1 << i)) { g_assert (pos < cfg->stack_usage - MIPS_STACK_ALIGNMENT); mips_lwc1 (code, i, mips_sp, pos); pos += sizeof (gulong); } } } #endif #ifdef SAVE_LMF /* Unlink the LMF if necessary */ if (method->save_lmf) { int lmf_offset = cfg->arch.lmf_offset; /* t0 = current_lmf->previous_lmf */ mips_lw (code, mips_temp, mips_sp, lmf_offset + G_STRUCT_OFFSET(MonoLMF, previous_lmf)); /* t1 = lmf_addr */ mips_lw (code, mips_t1, mips_sp, lmf_offset + G_STRUCT_OFFSET(MonoLMF, lmf_addr)); /* (*lmf_addr) = previous_lmf */ mips_sw (code, mips_temp, mips_t1, 0); } #endif #if 0 /* Restore the fp */ mips_lw (code, mips_fp, mips_sp, cfg->stack_usage + MIPS_FP_ADDR_OFFSET); #endif /* Correct the stack pointer, and return */ if ((cfg->flags & MONO_CFG_HAS_CALLS) || ALWAYS_SAVE_RA) mips_lw (code, mips_ra, mips_sp, cfg->stack_usage + MIPS_RET_ADDR_OFFSET); mips_addiu (code, mips_sp, mips_sp, cfg->stack_usage); mips_jr (code, mips_ra); mips_nop (code); cfg->code_len = code - cfg->native_code; g_assert (cfg->code_len < cfg->code_size); } /* remove once throw_exception_by_name is eliminated */ static int exception_id_by_name (const char *name) { if (strcmp (name, "IndexOutOfRangeException") == 0) return MONO_EXC_INDEX_OUT_OF_RANGE; if (strcmp (name, "OverflowException") == 0) return MONO_EXC_OVERFLOW; if (strcmp (name, "ArithmeticException") == 0) return MONO_EXC_ARITHMETIC; if (strcmp (name, "DivideByZeroException") == 0) return MONO_EXC_DIVIDE_BY_ZERO; if (strcmp (name, "InvalidCastException") == 0) return MONO_EXC_INVALID_CAST; if (strcmp (name, "NullReferenceException") == 0) return MONO_EXC_NULL_REF; if (strcmp (name, "ArrayTypeMismatchException") == 0) return MONO_EXC_ARRAY_TYPE_MISMATCH; g_error ("Unknown intrinsic exception %s\n", name); return 0; } void mono_arch_emit_exceptions (MonoCompile *cfg) { MonoJumpInfo *patch_info; int i; guint8 *code; const guint8* exc_throw_pos [MONO_EXC_INTRINS_NUM] = {NULL}; guint8 exc_throw_found [MONO_EXC_INTRINS_NUM] = {0}; int max_epilog_size = 50; /* count the number of exception infos */ /* * make sure we have enough space for exceptions * 24 is the simulated call to throw_exception_by_name */ for (patch_info = cfg->patch_info; patch_info; patch_info = patch_info->next) { if (patch_info->type == MONO_PATCH_INFO_EXC) { i = exception_id_by_name (patch_info->data.target); g_assert (i < MONO_EXC_INTRINS_NUM); if (!exc_throw_found [i]) { max_epilog_size += 12; exc_throw_found [i] = TRUE; } } else if (patch_info->type == MONO_PATCH_INFO_BB_OVF) max_epilog_size += 12; #if 0 else if (patch_info->type == MONO_PATCH_INFO_EXC_OVF) { MonoOvfJump *ovfj = (MonoOvfJump *)patch_info->data.target; i = exception_id_by_name (ovfj->data.exception); g_assert (i < MONO_EXC_INTRINS_NUM); if (!exc_throw_found [i]) { max_epilog_size += 12; exc_throw_found [i] = TRUE; } max_epilog_size += 8; } #endif } while (cfg->code_len + max_epilog_size > (cfg->code_size - 16)) { cfg->code_size *= 2; cfg->native_code = g_realloc (cfg->native_code, cfg->code_size); mono_jit_stats.code_reallocs++; } code = cfg->native_code + cfg->code_len; /* add code to raise exceptions */ for (patch_info = cfg->patch_info; patch_info; patch_info = patch_info->next) { switch (patch_info->type) { #if 0 case MONO_PATCH_INFO_BB_OVF: { MonoOvfJump *ovfj = (MonoOvfJump *)patch_info->data.target; guint32 *ip = (guint32 *)(void *)(patch_info->ip.i + cfg->native_code); /* patch the initial jump */ mips_patch (ip, (guint32)code); mips_bne (code, ovfj->b0_cond, ovfj->b1_cond, 2); mips_nop (code); mips_lui (code, mips_at, mips_zero, 0); mips_addiu (code, mips_at, mips_at, 0); mips_jr (code, mips_at); mips_patch ((guint32 *)(void *)(code - 4), (guint32)(ip + 1)); /* jump back after the initial branch */ mips_nop (code); /* jump back to the true target */ mips_jump (code, 0); ip = (guint32 *)(void *)(ovfj->data.bb->native_offset + cfg->native_code); mips_patch ((guint32 *)(void *)(code - 4), (guint32)ip); mips_nop (code); break; } case MONO_PATCH_INFO_EXC_OVF: { MonoOvfJump *ovfj = (MonoOvfJump *)patch_info->data.target; MonoJumpInfo *newji; guint32 *ip = (guint32 *)(void *)(patch_info->ip.i + cfg->native_code); unsigned char *bcl = code; /* patch the initial jump: we arrived here with a call */ mips_patch (ip, (guint32)code); mips_bne (code, ovfj->b0_cond, ovfj->b1_cond, 0); mips_nop (code); mips_jump (code, 0); /* jump back after the initial branch */ mips_patch ((guint32 *)(void *)(code - 4), (guint32)(ip + 1)); mips_nop (code); /* patch the conditional jump to the right handler */ /* make it processed next */ newji = mono_mempool_alloc (cfg->mempool, sizeof (MonoJumpInfo)); newji->type = MONO_PATCH_INFO_EXC; newji->ip.i = bcl - cfg->native_code; newji->data.target = ovfj->data.exception; newji->next = patch_info->next; patch_info->next = newji; break; } #endif case MONO_PATCH_INFO_EXC: { //unsigned char *ip = patch_info->ip.i + cfg->native_code; i = exception_id_by_name (patch_info->data.target); g_assert (i >= 0 && i < MONO_EXC_INTRINS_NUM); if (!exc_throw_pos [i]) { guint32 addr; exc_throw_pos [i] = code; //g_print ("exc: writing stub at %p\n", code); mips_load_const (code, mips_a0, patch_info->data.target); addr = (guint32) mono_arch_get_throw_exception_by_name (); mips_load_const (code, mips_t9, addr); mips_jr (code, mips_t9); mips_nop (code); } //g_print ("exc: patch %p to %p\n", ip, exc_throw_pos[i]); /* Turn into a Relative patch, pointing at code stub */ patch_info->type = MONO_PATCH_INFO_METHOD_REL; patch_info->data.offset = exc_throw_pos[i] - cfg->native_code; break; } default: /* do nothing */ break; } } cfg->code_len = code - cfg->native_code; g_assert (cfg->code_len < cfg->code_size); } /* * Thread local storage support */ static void setup_tls_access (void) { guint32 ptk; //guint32 *ins, *code; if (tls_mode == TLS_MODE_FAILED) return; if (g_getenv ("MONO_NO_TLS")) { tls_mode = TLS_MODE_FAILED; return; } if (tls_mode == TLS_MODE_DETECT) { /* XXX */ tls_mode = TLS_MODE_FAILED; return; #if 0 ins = (guint32*)pthread_getspecific; /* uncond branch to the real method */ if ((*ins >> 26) == 18) { gint32 val; val = (*ins & ~3) << 6; val >>= 6; if (*ins & 2) { /* absolute */ ins = (guint32*)val; } else { ins = (guint32*) ((char*)ins + val); } } code = &cmplwi_1023; ppc_cmpli (code, 0, 0, ppc_r3, 1023); code = &li_0x48; ppc_li (code, ppc_r4, 0x48); code = &blr_ins; ppc_blr (code); if (*ins == cmplwi_1023) { int found_lwz_284 = 0; for (ptk = 0; ptk < 20; ++ptk) { ++ins; if (!*ins || *ins == blr_ins) break; if ((guint16)*ins == 284 && (*ins >> 26) == 32) { found_lwz_284 = 1; break; } } if (!found_lwz_284) { tls_mode = TLS_MODE_FAILED; return; } tls_mode = TLS_MODE_LTHREADS; } else if (*ins == li_0x48) { ++ins; /* uncond branch to the real method */ if ((*ins >> 26) == 18) { gint32 val; val = (*ins & ~3) << 6; val >>= 6; if (*ins & 2) { /* absolute */ ins = (guint32*)val; } else { ins = (guint32*) ((char*)ins + val); } code = &val; ppc_li (code, ppc_r0, 0x7FF2); if (ins [1] == val) { /* Darwin on G4, implement */ tls_mode = TLS_MODE_FAILED; return; } else { code = &val; ppc_mfspr (code, ppc_r3, 104); if (ins [1] != val) { tls_mode = TLS_MODE_FAILED; return; } tls_mode = TLS_MODE_DARWIN_G5; } } else { tls_mode = TLS_MODE_FAILED; return; } } else { tls_mode = TLS_MODE_FAILED; return; } #endif } if (monodomain_key == -1) { ptk = mono_domain_get_tls_key (); if (ptk < 1024) { ptk = mono_pthread_key_for_tls (ptk); if (ptk < 1024) { monodomain_key = ptk; } } } if (lmf_pthread_key == -1) { ptk = mono_pthread_key_for_tls (mono_jit_tls_id); if (ptk < 1024) { /*g_print ("MonoLMF at: %d\n", ptk);*/ /*if (!try_offset_access (mono_get_lmf_addr (), ptk)) { init_tls_failed = 1; return; }*/ lmf_pthread_key = ptk; } } if (monothread_key == -1) { ptk = mono_thread_get_tls_key (); if (ptk < 1024) { ptk = mono_pthread_key_for_tls (ptk); if (ptk < 1024) { monothread_key = ptk; /*g_print ("thread inited: %d\n", ptk);*/ } } else { /*g_print ("thread not inited yet %d\n", ptk);*/ } } } void mono_arch_setup_jit_tls_data (MonoJitTlsData *tls) { setup_tls_access (); } void mono_arch_free_jit_tls_data (MonoJitTlsData *tls) { } void mono_arch_emit_this_vret_args (MonoCompile *cfg, MonoCallInst *inst, int this_reg, int this_type, int vt_reg) { int this_dreg = mips_a0; if (vt_reg != -1) this_dreg = mips_a1; /* add the this argument */ if (this_reg != -1) { MonoInst *this; MONO_INST_NEW (cfg, this, OP_SETREG); this->type = this_type; this->sreg1 = this_reg; this->dreg = mono_regstate_next_int (cfg->rs); mono_bblock_add_inst (cfg->cbb, this); mono_call_inst_add_outarg_reg (cfg, inst, this->dreg, this_dreg, FALSE); } if (vt_reg != -1) { MonoInst *vtarg; MONO_INST_NEW (cfg, vtarg, OP_SETREG); vtarg->type = STACK_MP; vtarg->sreg1 = vt_reg; vtarg->dreg = mono_regstate_next_int (cfg->rs); mono_bblock_add_inst (cfg->cbb, vtarg); mono_call_inst_add_outarg_reg (cfg, inst, vtarg->dreg, mips_a0, FALSE); } } MonoInst* mono_arch_get_inst_for_method (MonoCompile *cfg, MonoMethod *cmethod, MonoMethodSignature *fsig, MonoInst **args) { MonoInst *ins = NULL; if (cmethod->klass == mono_defaults.thread_class && strcmp (cmethod->name, "MemoryBarrier") == 0) { MONO_INST_NEW (cfg, ins, OP_MEMORY_BARRIER); } /*if (cmethod->klass == mono_defaults.math_class) { if (strcmp (cmethod->name, "Sqrt") == 0) { MONO_INST_NEW (cfg, ins, OP_SQRT); ins->inst_i0 = args [0]; } }*/ return ins; } gboolean mono_arch_print_tree (MonoInst *tree, int arity) { return 0; } MonoInst* mono_arch_get_domain_intrinsic (MonoCompile* cfg) { MonoInst* ins; setup_tls_access (); if (monodomain_key == -1) return NULL; MONO_INST_NEW (cfg, ins, OP_TLS_GET); ins->inst_offset = monodomain_key; return ins; } MonoInst* mono_arch_get_thread_intrinsic (MonoCompile* cfg) { MonoInst* ins; setup_tls_access (); if (monothread_key == -1) return NULL; MONO_INST_NEW (cfg, ins, OP_TLS_GET); ins->inst_offset = monothread_key; return ins; }