MONO_ARCH_SAVE_REGS;
- addr = mono_compile_method (method);
+ addr = mono_create_jump_trampoline (mono_domain_get (), method, TRUE);
- return addr;
+ return mono_create_ftnptr (mono_domain_get (), addr);
}
+/*
+ * Same as mono_ldftn, but do not add a synchronized wrapper. Used in the
+ * synchronized wrappers to avoid infinite recursion.
+ */
static void*
-mono_ldvirtfn (MonoObject *obj, MonoMethod *method)
+mono_ldftn_nosync (MonoMethod *method)
{
+ gpointer addr;
+
MONO_ARCH_SAVE_REGS;
- method = mono_object_get_virtual_method (obj, method);
- if (method->iflags & METHOD_IMPL_ATTRIBUTE_SYNCHRONIZED)
- method = mono_marshal_get_synchronized_wrapper (method);
+ addr = mono_create_jump_trampoline (mono_domain_get (), method, FALSE);
- return mono_ldftn (method);
+ return mono_create_ftnptr (mono_domain_get (), addr);
}
-static void
-helper_initobj (void *addr, int size)
+static void*
+mono_ldvirtfn (MonoObject *obj, MonoMethod *method)
{
MONO_ARCH_SAVE_REGS;
- memset (addr, 0, size);
-}
-
-static void
-helper_memcpy (void *addr, void *src, int size)
-{
- MONO_ARCH_SAVE_REGS;
+ if (obj == NULL)
+ mono_raise_exception (mono_get_exception_null_reference ());
- memcpy (addr, src, size);
-}
-
-static void
-helper_memset (void *addr, int val, int size)
-{
- MONO_ARCH_SAVE_REGS;
+ method = mono_object_get_virtual_method (obj, method);
- memset (addr, val, size);
+ return mono_ldftn (method);
}
static void
mono_raise_exception (mono_get_exception_array_type_mismatch ());
}
+#ifndef MONO_ARCH_NO_EMULATE_LONG_MUL_OPTS
+
static gint64
mono_llmult (gint64 a, gint64 b)
{
return 0;
}
-
static guint64
mono_llmult_ovf (gint64 a, gint64 b)
{
*/
sign = ah ^ bh;
if (ah < 0) {
+ if (((guint32)ah == 0x80000000) && (al == 0)) {
+ /* This has no two's complement */
+ if (b == 0)
+ return 0;
+ else if (b == 1)
+ return a;
+ else
+ goto raise_exception;
+ }
+
/* flip the bits and add 1 */
ah ^= ~0;
if (al == 0)
}
if (bh < 0) {
+ if (((guint32)bh == 0x80000000) && (bl == 0)) {
+ /* This has no two's complement */
+ if (a == 0)
+ return 0;
+ else if (a == 1)
+ return b;
+ else
+ goto raise_exception;
+ }
+
/* flip the bits and add 1 */
bh ^= ~0;
if (bl == 0)
in a 64 bit result */
if (ah && bh)
goto raise_exception;
+ if ((gint64)((gint64)ah * (gint64)bl) > (gint64)0x80000000 || (gint64)((gint64)al * (gint64)bh) > (gint64)0x80000000)
+ goto raise_exception;
/* do the AlBl term first */
t1 = (gint64)al * (gint64)bl;
/* now do the [(Ah-Al)(Bl-Bh)+AlBl]R term */
t1 += (gint64)(ah - al) * (gint64)(bl - bh);
- t1 <<= 32;
/* check for overflow */
+ t1 <<= 32;
if (t1 > (0x7FFFFFFFFFFFFFFFLL - res))
goto raise_exception;
return 0;
}
+#if defined(MONO_ARCH_EMULATE_MUL_DIV) || defined(MONO_ARCH_EMULATE_DIV)
+
+static gint32
+mono_idiv (gint32 a, gint32 b)
+{
+ MONO_ARCH_SAVE_REGS;
+
+#ifdef MONO_ARCH_NEED_DIV_CHECK
+ if (!b)
+ mono_raise_exception (mono_get_exception_divide_by_zero ());
+ else if (b == -1 && a == (0x80000000))
+ mono_raise_exception (mono_get_exception_arithmetic ());
+#endif
+ return a / b;
+}
+
+static guint32
+mono_idiv_un (guint32 a, guint32 b)
+{
+ MONO_ARCH_SAVE_REGS;
+
+#ifdef MONO_ARCH_NEED_DIV_CHECK
+ if (!b)
+ mono_raise_exception (mono_get_exception_divide_by_zero ());
+#endif
+ return a / b;
+}
+
+static gint32
+mono_irem (gint32 a, gint32 b)
+{
+ MONO_ARCH_SAVE_REGS;
+
+#ifdef MONO_ARCH_NEED_DIV_CHECK
+ if (!b)
+ mono_raise_exception (mono_get_exception_divide_by_zero ());
+ else if (b == -1 && a == (0x80000000))
+ mono_raise_exception (mono_get_exception_arithmetic ());
+#endif
+
+ return a % b;
+}
+
+static guint32
+mono_irem_un (guint32 a, guint32 b)
+{
+ MONO_ARCH_SAVE_REGS;
+
+#ifdef MONO_ARCH_NEED_DIV_CHECK
+ if (!b)
+ mono_raise_exception (mono_get_exception_divide_by_zero ());
+#endif
+ return a % b;
+}
+
+#endif
+
+#ifdef MONO_ARCH_EMULATE_MUL_DIV
+
+static gint32
+mono_imul (gint32 a, gint32 b)
+{
+ MONO_ARCH_SAVE_REGS;
+
+ return a * b;
+}
+
+static gint32
+mono_imul_ovf (gint32 a, gint32 b)
+{
+ gint64 res;
+
+ MONO_ARCH_SAVE_REGS;
+
+ res = (gint64)a * (gint64)b;
+
+ if ((res > 0x7fffffffL) || (res < -2147483648))
+ mono_raise_exception (mono_get_exception_overflow ());
+
+ return res;
+}
+
+static gint32
+mono_imul_ovf_un (guint32 a, guint32 b)
+{
+ guint64 res;
+
+ MONO_ARCH_SAVE_REGS;
+
+ res = (guint64)a * (guint64)b;
+
+ if ((res >> 32))
+ mono_raise_exception (mono_get_exception_overflow ());
+
+ return res;
+}
+
+static double
+mono_fdiv (double a, double b)
+{
+ MONO_ARCH_SAVE_REGS;
+
+ return a / b;
+}
+#endif
+
static gint64
mono_lldiv (gint64 a, gint64 b)
{
MONO_ARCH_SAVE_REGS;
+#ifdef MONO_ARCH_NEED_DIV_CHECK
+ if (!b)
+ mono_raise_exception (mono_get_exception_divide_by_zero ());
+ else if (b == -1 && a == (-9223372036854775807LL - 1LL))
+ mono_raise_exception (mono_get_exception_arithmetic ());
+#endif
return a / b;
}
{
MONO_ARCH_SAVE_REGS;
+#ifdef MONO_ARCH_NEED_DIV_CHECK
+ if (!b)
+ mono_raise_exception (mono_get_exception_divide_by_zero ());
+ else if (b == -1 && a == (-9223372036854775807LL - 1LL))
+ mono_raise_exception (mono_get_exception_arithmetic ());
+#endif
return a % b;
}
{
MONO_ARCH_SAVE_REGS;
+#ifdef MONO_ARCH_NEED_DIV_CHECK
+ if (!b)
+ mono_raise_exception (mono_get_exception_divide_by_zero ());
+#endif
return a / b;
}
{
MONO_ARCH_SAVE_REGS;
+#ifdef MONO_ARCH_NEED_DIV_CHECK
+ if (!b)
+ mono_raise_exception (mono_get_exception_divide_by_zero ());
+#endif
return a % b;
}
+#endif
+
+#ifndef MONO_ARCH_NO_EMULATE_LONG_SHIFT_OPS
+
static guint64
mono_lshl (guint64 a, gint32 shamt)
{
return res;
}
+#endif
+
/**
* ves_array_element_address:
* @this: a pointer to the array object
}
esize *= ind;
- ea = (gpointer*)((char*)this->vector + esize);
+ ea = (gpointer*)(gpointer)((char*)this->vector + esize);
va_end(ap);
MONO_ARCH_SAVE_REGS;
- pcount = cm->signature->param_count;
+ pcount = mono_method_signature (cm)->param_count;
rank = cm->klass->rank;
va_start (ap, cm);
if (rank == pcount) {
/* Only lengths provided. */
- lower_bounds = NULL;
+ if (cm->klass->byval_arg.type == MONO_TYPE_ARRAY) {
+ lower_bounds = alloca (sizeof (guint32) * rank);
+ memset (lower_bounds, 0, sizeof (guint32) * rank);
+ } else {
+ lower_bounds = NULL;
+ }
} else {
g_assert (pcount == (rank * 2));
/* lower bounds are first. */
}
static gpointer
-mono_ldtoken_wrapper (MonoImage *image, int token)
+mono_ldtoken_wrapper (MonoImage *image, int token, MonoGenericContext *context)
{
MonoClass *handle_class;
gpointer res;
MONO_ARCH_SAVE_REGS;
- res = mono_ldtoken (image, token, &handle_class);
+ res = mono_ldtoken (image, token, &handle_class, context);
mono_class_init (handle_class);
return res;
static guint64
mono_fconv_u8 (double v)
{
- /* no need, no exceptions: MONO_ARCH_SAVE_REGS;*/
return (guint64)v;
}
return (guint32)v;
}
+#ifndef HAVE_TRUNC
+/* Solaris doesn't have trunc */
+#ifdef HAVE_AINTL
+extern long double aintl (long double);
+#define trunc aintl
+#else
+/* FIXME: This means we will never throw overflow exceptions */
+#define trunc(v) res
+#endif
+#endif /* HAVE_TRUNC */
+
static gint64
mono_fconv_ovf_i8 (double v)
{
res = (gint64)v;
- if (isnan(v) || v != res) {
+ if (isnan(v) || trunc (v) != res) {
mono_raise_exception (mono_get_exception_overflow ());
}
return res;
res = (guint64)v;
- if (isnan(v) || v != res) {
+ if (isnan(v) || trunc (v) != res) {
mono_raise_exception (mono_get_exception_overflow ());
}
return res;
}
#endif
+#ifdef MONO_ARCH_EMULATE_CONV_R8_UN
+static double
+mono_conv_to_r8_un (guint32 a)
+{
+ return (double)a;
+}
+#endif
+
#ifdef MONO_ARCH_EMULATE_LCONV_TO_R8_UN
static double
mono_lconv_to_r8_un (guint64 a)
}
#endif
+static gpointer
+helper_compile_generic_method (MonoObject *obj, MonoMethod *method, MonoGenericContext *context)
+{
+ MonoMethod *vmethod, *inflated;
+ gpointer addr;
+
+ vmethod = mono_object_get_virtual_method (obj, method);
+ inflated = mono_class_inflate_generic_method (vmethod, context);
+ inflated = mono_get_inflated_method (inflated);
+ addr = mono_compile_method (inflated);
+
+ return addr;
+}
+
+static MonoString*
+helper_ldstr (MonoImage *image, guint32 idx)
+{
+ return mono_ldstr (mono_domain_get (), image, idx);
+}