use the GetRequestStreamAsync method to avoid blocking on HttpClientHandler SendAsync
[mono.git] / mono / metadata / marshal.c
index 4c478c63cdb56d3a9ad911aaa15d810ec10705ca..3dd967b4534c7541de8bac8b0297ec68ca94f2a7 100644 (file)
@@ -124,8 +124,6 @@ mono_byvalarray_to_array (MonoArray *arr, gpointer native_arr, MonoClass *eltype
 static void
 mono_array_to_byvalarray (gpointer native_arr, MonoArray *arr, MonoClass *eltype, guint32 elnum);
 
-static MonoObject *
-mono_remoting_wrapper (MonoMethod *method, gpointer *params);
 
 static MonoAsyncResult *
 mono_delegate_begin_invoke (MonoDelegate *delegate, gpointer *params);
@@ -142,9 +140,15 @@ mono_marshal_set_domain_by_id (gint32 id, MonoBoolean push);
 static gboolean
 mono_marshal_check_domain_image (gint32 domain_id, MonoImage *image);
 
+#ifndef DISABLE_REMOTING
+static MonoObject *
+mono_remoting_wrapper (MonoMethod *method, gpointer *params);
+
 void
 mono_upgrade_remote_class_wrapper (MonoReflectionType *rtype, MonoTransparentProxy *tproxy);
 
+#endif
+
 static MonoReflectionType *
 type_from_handle (MonoType *handle);
 
@@ -248,7 +252,6 @@ mono_marshal_init (void)
                register_icall (g_free, "g_free", "void ptr", FALSE);
                register_icall (mono_object_isinst, "mono_object_isinst", "object object ptr", FALSE);
                register_icall (mono_struct_delete_old, "mono_struct_delete_old", "void ptr ptr", FALSE);
-               register_icall (mono_remoting_wrapper, "mono_remoting_wrapper", "object ptr ptr", FALSE);
                register_icall (mono_delegate_begin_invoke, "mono_delegate_begin_invoke", "object object ptr", FALSE);
                register_icall (mono_delegate_end_invoke, "mono_delegate_end_invoke", "object object ptr", FALSE);
                register_icall (mono_marshal_xdomain_copy_value, "mono_marshal_xdomain_copy_value", "object object", FALSE);
@@ -258,11 +261,14 @@ mono_marshal_init (void)
                register_icall (mono_compile_method, "mono_compile_method", "ptr ptr", FALSE);
                register_icall (mono_context_get, "mono_context_get", "object", FALSE);
                register_icall (mono_context_set, "mono_context_set", "void object", FALSE);
-               register_icall (mono_upgrade_remote_class_wrapper, "mono_upgrade_remote_class_wrapper", "void object object", FALSE);
                register_icall (type_from_handle, "type_from_handle", "object ptr", FALSE);
                register_icall (mono_gc_wbarrier_generic_nostore, "wb_generic", "void ptr", FALSE);
                register_icall (mono_gchandle_get_target, "mono_gchandle_get_target", "object int32", TRUE);
 
+#ifndef DISABLE_REMOTING
+               register_icall (mono_remoting_wrapper, "mono_remoting_wrapper", "object ptr ptr", FALSE);
+               register_icall (mono_upgrade_remote_class_wrapper, "mono_upgrade_remote_class_wrapper", "void object object", FALSE);
+#endif
                mono_cominterop_init ();
        }
 }
@@ -278,9 +284,9 @@ mono_marshal_cleanup (void)
        marshal_mutex_initialized = FALSE;
 }
 
-MonoClass *byte_array_class;
+#ifndef DISABLE_REMOTING
+static MonoClass *byte_array_class;
 static MonoMethod *method_rs_serialize, *method_rs_deserialize, *method_exc_fixexc, *method_rs_appdomain_target;
-static MonoMethod *method_set_context, *method_get_context;
 static MonoMethod *method_set_call_context, *method_needs_context_sink, *method_rs_serialize_exc;
 
 static void
@@ -302,11 +308,6 @@ mono_remoting_marshal_init (void)
                klass = mono_defaults.exception_class;
                method_exc_fixexc = mono_class_get_method_from_name (klass, "FixRemotingException", -1);
        
-               klass = mono_defaults.thread_class;
-               method_get_context = mono_class_get_method_from_name (klass, "get_CurrentContext", -1);
-       
-               klass = mono_defaults.appdomain_class;
-               method_set_context = mono_class_get_method_from_name (klass, "InternalSetContext", -1);
                byte_array_class = mono_array_class_get (mono_defaults.byte_class, 1);
        
                klass = mono_class_from_name (mono_defaults.corlib, "System.Runtime.Remoting.Messaging", "CallContext");
@@ -318,6 +319,7 @@ mono_remoting_marshal_init (void)
                module_initialized = TRUE;
        }
 }
+#endif
 
 gpointer
 mono_delegate_to_ftnptr (MonoDelegate *delegate)
@@ -1024,7 +1026,7 @@ mono_string_new_len_wrapper (const char *text, guint length)
 }
 
 #ifndef DISABLE_JIT
-
+#ifndef DISABLE_REMOTING
 static int
 mono_mb_emit_proxy_check (MonoMethodBuilder *mb, int branch_code)
 {
@@ -1074,6 +1076,7 @@ mono_mb_emit_contextbound_check (MonoMethodBuilder *mb, int branch_code)
        mono_mb_emit_icon (mb, 0);
        return mono_mb_emit_branch (mb, branch_code);
 }
+#endif /* DISABLE_REMOTING */
 
 /*
  * mono_mb_emit_exception_marshal_directive:
@@ -1094,6 +1097,8 @@ mono_mb_emit_exception_marshal_directive (MonoMethodBuilder *mb, char *msg)
        mono_mb_emit_exception_full (mb, "System.Runtime.InteropServices", "MarshalDirectiveException", s);
 }
 
+#endif /* !DISABLE_JIT */
+
 guint
 mono_type_to_ldind (MonoType *type)
 {
@@ -1205,6 +1210,8 @@ handle_enum:
        return -1;
 }
 
+#ifndef DISABLE_JIT
+
 static void
 emit_ptr_to_object_conv (MonoMethodBuilder *mb, MonoType *type, MonoMarshalConv conv, MonoMarshalSpec *mspec)
 {
@@ -2116,10 +2123,11 @@ mono_delegate_begin_invoke (MonoDelegate *delegate, gpointer *params)
        if (mcast_delegate->prev != NULL)
                mono_raise_exception (mono_get_exception_argument (NULL, "The delegate must have only one target"));
 
+#ifndef DISABLE_REMOTING
        if (delegate->target && mono_object_class (delegate->target) == mono_defaults.transparent_proxy_class) {
 
                MonoTransparentProxy* tp = (MonoTransparentProxy *)delegate->target;
-               if (!tp->remote_class->proxy_class->contextbound || tp->rp->context != (MonoObject *) mono_context_get ()) {
+               if (!mono_class_is_contextbound (tp->remote_class->proxy_class) || tp->rp->context != (MonoObject *) mono_context_get ()) {
 
                        /* If the target is a proxy, make a direct call. Is proxy's work
                        // to make the call asynchronous.
@@ -2143,6 +2151,7 @@ mono_delegate_begin_invoke (MonoDelegate *delegate, gpointer *params)
                        return ares;
                }
        }
+#endif
 
        klass = delegate->object.vtable->klass;
 
@@ -2655,6 +2664,59 @@ get_wrapper_target_class (MonoImage *image)
        return klass;
 }
 
+static MonoMethod*
+check_generic_delegate_wrapper_cache (GHashTable *cache, MonoMethod *orig_method, MonoMethod *def_method, MonoGenericContext *ctx)
+{
+       MonoMethod *res;
+       MonoMethod *inst, *def;
+
+       /*
+        * Look for the instance
+        */
+       res = mono_marshal_find_in_cache (cache, orig_method->klass);
+       if (res)
+               return res;
+
+       /*
+        * Look for the definition
+        */
+       def = mono_marshal_find_in_cache (cache, def_method->klass);
+       if (def) {
+               inst = mono_class_inflate_generic_method (def, ctx);
+               /* Cache it */
+               mono_memory_barrier ();
+               mono_marshal_lock ();
+               res = g_hash_table_lookup (cache, orig_method->klass);
+               if (!res) {
+                       g_hash_table_insert (cache, orig_method->klass, inst);
+                       res = inst;
+               }
+               mono_marshal_unlock ();
+               return res;
+       }
+       return NULL;
+}
+
+static MonoMethod*
+cache_generic_delegate_wrapper (GHashTable *cache, MonoMethod *orig_method, MonoMethod *def, MonoGenericContext *ctx)
+{
+       MonoMethod *inst, *res;
+
+       /*
+        * We use the same cache for the generic definition and the instances.
+        */
+       inst = mono_class_inflate_generic_method (def, ctx);
+       mono_memory_barrier ();
+       mono_marshal_lock ();
+       res = g_hash_table_lookup (cache, orig_method->klass);
+       if (!res) {
+               g_hash_table_insert (cache, orig_method->klass, inst);
+               res = inst;
+       }
+       mono_marshal_unlock ();
+       return res;
+}
+
 MonoMethod *
 mono_marshal_get_delegate_begin_invoke (MonoMethod *method)
 {
@@ -2664,22 +2726,46 @@ mono_marshal_get_delegate_begin_invoke (MonoMethod *method)
        GHashTable *cache;
        int params_var;
        char *name;
+       MonoGenericContext *ctx = NULL;
+       MonoMethod *orig_method = NULL;
 
        g_assert (method && method->klass->parent == mono_defaults.multicastdelegate_class &&
                  !strcmp (method->name, "BeginInvoke"));
 
+       /*
+        * For generic delegates, create a generic wrapper, and returns an instance to help AOT.
+        */
+       if (method->is_inflated) {
+               orig_method = method;
+               ctx = &((MonoMethodInflated*)method)->context;
+               method = ((MonoMethodInflated*)method)->declaring;
+       }
+
        sig = mono_signature_no_pinvoke (method);
 
-       cache = get_cache (&method->klass->image->delegate_begin_invoke_cache,
-                                          (GHashFunc)mono_signature_hash, 
-                                          (GCompareFunc)mono_metadata_signature_equal);
-       if ((res = mono_marshal_find_in_cache (cache, sig)))
-               return res;
+       /*
+        * Check cache
+        */
+       if (ctx) {
+               cache = get_cache (&method->klass->image->delegate_begin_invoke_generic_cache, mono_aligned_addr_hash, NULL);
+               res = check_generic_delegate_wrapper_cache (cache, orig_method, method, ctx);
+               if (res)
+                       return res;
+       } else {
+               cache = get_cache (&method->klass->image->delegate_begin_invoke_cache,
+                                                  (GHashFunc)mono_signature_hash, 
+                                                  (GCompareFunc)mono_metadata_signature_equal);
+               if ((res = mono_marshal_find_in_cache (cache, sig)))
+                       return res;
+       }
 
        g_assert (sig->hasthis);
 
        name = mono_signature_to_name (sig, "begin_invoke");
-       mb = mono_mb_new (get_wrapper_target_class (method->klass->image), name, MONO_WRAPPER_DELEGATE_BEGIN_INVOKE);
+       if (ctx)
+               mb = mono_mb_new (method->klass, name, MONO_WRAPPER_DELEGATE_BEGIN_INVOKE);
+       else
+               mb = mono_mb_new (get_wrapper_target_class (method->klass->image), name, MONO_WRAPPER_DELEGATE_BEGIN_INVOKE);
        g_free (name);
 
 #ifndef DISABLE_JIT
@@ -2691,7 +2777,14 @@ mono_marshal_get_delegate_begin_invoke (MonoMethod *method)
        mono_mb_emit_byte (mb, CEE_RET);
 #endif
 
-       res = mono_mb_create_and_cache (cache, sig, mb, sig, sig->param_count + 16);
+       if (ctx) {
+               MonoMethod *def;
+               def = mono_mb_create_and_cache (cache, method->klass, mb, sig, sig->param_count + 16);
+               res = cache_generic_delegate_wrapper (cache, orig_method, def, ctx);
+       } else {
+               res = mono_mb_create_and_cache (cache, sig, mb, sig, sig->param_count + 16);
+       }
+
        mono_mb_free (mb);
        return res;
 }
@@ -2739,14 +2832,17 @@ mono_delegate_end_invoke (MonoDelegate *delegate, gpointer *params)
                return NULL;
        }
 
-       if (delegate->target && mono_object_class (delegate->target) == mono_defaults.transparent_proxy_class) {
+#ifndef DISABLE_REMOTING
+       if (delegate->target && mono_object_is_transparent_proxy (delegate->target)) {
                MonoTransparentProxy* tp = (MonoTransparentProxy *)delegate->target;
                msg = (MonoMethodMessage *)mono_object_new (domain, mono_defaults.mono_method_message_class);
                mono_message_init (domain, msg, delegate->method_info, NULL);
                msg->call_type = CallType_EndInvoke;
                MONO_OBJECT_SETREF (msg, async_result, ares);
                res = mono_remoting_invoke ((MonoObject *)tp->rp, msg, &exc, &out_args);
-       } else {
+       } else
+#endif
+       {
                res = mono_thread_pool_finish (ares, &out_args, &exc);
        }
 
@@ -2815,6 +2911,12 @@ mono_mb_emit_restore_result (MonoMethodBuilder *mb, MonoType *return_type)
                mono_mb_emit_op (mb, CEE_LDOBJ, klass);
                break;
        }
+       case MONO_TYPE_VAR:
+       case MONO_TYPE_MVAR: {
+               MonoClass *klass = mono_class_from_mono_type (return_type);
+               mono_mb_emit_op (mb, CEE_UNBOX_ANY, klass);
+               break;
+       }
        default:
                g_warning ("type 0x%x not handled", return_type->type);
                g_assert_not_reached ();
@@ -2834,22 +2936,46 @@ mono_marshal_get_delegate_end_invoke (MonoMethod *method)
        GHashTable *cache;
        int params_var;
        char *name;
+       MonoGenericContext *ctx = NULL;
+       MonoMethod *orig_method = NULL;
 
        g_assert (method && method->klass->parent == mono_defaults.multicastdelegate_class &&
                  !strcmp (method->name, "EndInvoke"));
 
+       /*
+        * For generic delegates, create a generic wrapper, and returns an instance to help AOT.
+        */
+       if (method->is_inflated) {
+               orig_method = method;
+               ctx = &((MonoMethodInflated*)method)->context;
+               method = ((MonoMethodInflated*)method)->declaring;
+       }
+
        sig = mono_signature_no_pinvoke (method);
 
-       cache = get_cache (&method->klass->image->delegate_end_invoke_cache,
-                                          (GHashFunc)mono_signature_hash, 
-                                          (GCompareFunc)mono_metadata_signature_equal);
-       if ((res = mono_marshal_find_in_cache (cache, sig)))
-               return res;
+       /*
+        * Check cache
+        */
+       if (ctx) {
+               cache = get_cache (&method->klass->image->delegate_end_invoke_generic_cache, mono_aligned_addr_hash, NULL);
+               res = check_generic_delegate_wrapper_cache (cache, orig_method, method, ctx);
+               if (res)
+                       return res;
+       } else {
+               cache = get_cache (&method->klass->image->delegate_end_invoke_cache,
+                                                  (GHashFunc)mono_signature_hash, 
+                                                  (GCompareFunc)mono_metadata_signature_equal);
+               if ((res = mono_marshal_find_in_cache (cache, sig)))
+                       return res;
+       }
 
        g_assert (sig->hasthis);
 
        name = mono_signature_to_name (sig, "end_invoke");
-       mb = mono_mb_new (get_wrapper_target_class (method->klass->image), name, MONO_WRAPPER_DELEGATE_END_INVOKE);
+       if (ctx)
+               mb = mono_mb_new (method->klass, name, MONO_WRAPPER_DELEGATE_END_INVOKE);
+       else
+               mb = mono_mb_new (get_wrapper_target_class (method->klass->image), name, MONO_WRAPPER_DELEGATE_END_INVOKE);
        g_free (name);
 
 #ifndef DISABLE_JIT
@@ -2866,13 +2992,21 @@ mono_marshal_get_delegate_end_invoke (MonoMethod *method)
                mono_mb_emit_restore_result (mb, sig->ret);
 #endif
 
-       res = mono_mb_create_and_cache (cache, sig,
-                                                                                mb, sig, sig->param_count + 16);
+       if (ctx) {
+               MonoMethod *def;
+               def = mono_mb_create_and_cache (cache, method->klass, mb, sig, sig->param_count + 16);
+               res = cache_generic_delegate_wrapper (cache, orig_method, def, ctx);
+       } else {
+               res = mono_mb_create_and_cache (cache, sig,
+                                                                               mb, sig, sig->param_count + 16);
+       }
        mono_mb_free (mb);
 
        return res;
 }
 
+#ifndef DISABLE_REMOTING
+
 static MonoObject *
 mono_remoting_wrapper (MonoMethod *method, gpointer *params)
 {
@@ -2889,7 +3023,7 @@ mono_remoting_wrapper (MonoMethod *method, gpointer *params)
        /* skip the this pointer */
        params++;
 
-       if (this->remote_class->proxy_class->contextbound && this->rp->context == (MonoObject *) mono_context_get ())
+       if (mono_class_is_contextbound (this->remote_class->proxy_class) && this->rp->context == (MonoObject *) mono_context_get ())
        {
                int i;
                MonoMethodSignature *sig = mono_method_signature (method);
@@ -2943,11 +3077,11 @@ mono_marshal_get_remoting_invoke (MonoMethod *method)
 
        /* this seems to be the best plase to put this, as all remoting invokes seem to get filtered through here */
 #ifndef DISABLE_COM
-       if (method->klass->is_com_object || method->klass == mono_defaults.com_object_class) {
+       if (mono_class_is_com_object (method->klass) || method->klass == mono_defaults.com_object_class) {
                MonoVTable *vtable = mono_class_vtable (mono_domain_get (), method->klass);
                g_assert (vtable); /*FIXME do proper error handling*/
 
-               if (!vtable->remote) {
+               if (!mono_vtable_is_remote (vtable)) {
                        return mono_cominterop_get_invoke (method);
                }
        }
@@ -2988,6 +3122,9 @@ mono_marshal_get_remoting_invoke (MonoMethod *method)
        return res;
 }
 
+#endif /* DISABLE_REMOTING */
+
+
 /* mono_get_xdomain_marshal_type()
  * Returns the kind of marshalling that a type needs for cross domain calls.
  */
@@ -3126,7 +3263,9 @@ mono_marshal_xdomain_copy_out_value (MonoObject *src, MonoObject *dst)
        }
 }
 
-#ifndef DISABLE_JIT
+
+#if !(defined (DISABLE_JIT) || defined (DISABLE_REMOTING))
+
 static void
 mono_marshal_emit_xdomain_copy_value (MonoMethodBuilder *mb, MonoClass *pclass)
 {
@@ -3141,15 +3280,17 @@ mono_marshal_emit_xdomain_copy_out_value (MonoMethodBuilder *mb, MonoClass *pcla
 }
 #endif
 
+#ifndef DISABLE_REMOTING
 /* mono_marshal_supports_fast_xdomain()
  * Returns TRUE if the method can use the fast xdomain wrapper.
  */
 static gboolean
 mono_marshal_supports_fast_xdomain (MonoMethod *method)
 {
-       return !method->klass->contextbound &&
+       return !mono_class_is_contextbound (method->klass) &&
                   !((method->flags & METHOD_ATTRIBUTE_SPECIAL_NAME) && (strcmp (".ctor", method->name) == 0));
 }
+#endif
 
 static gint32
 mono_marshal_set_domain_by_id (gint32 id, MonoBoolean push)
@@ -3168,7 +3309,7 @@ mono_marshal_set_domain_by_id (gint32 id, MonoBoolean push)
        return current_domain->domain_id;
 }
 
-#ifndef DISABLE_JIT
+#if !(defined (DISABLE_JIT) || defined (DISABLE_REMOTING))
 static void
 mono_marshal_emit_switch_domain (MonoMethodBuilder *mb)
 {
@@ -3215,6 +3356,8 @@ mono_marshal_check_domain_image (gint32 domain_id, MonoImage *image)
        return tmp != NULL;
 }
 
+#ifndef DISABLE_REMOTING
+
 /* mono_marshal_get_xappdomain_dispatch ()
  * Generates a method that dispatches a method call from another domain into
  * the current domain.
@@ -3887,6 +4030,8 @@ mono_marshal_get_remoting_invoke_with_check (MonoMethod *method)
        return res;
 }
 
+#endif /* DISABLE_REMOTING */
+
 typedef struct
 {
        MonoMethodSignature *sig;
@@ -4001,33 +4146,10 @@ mono_marshal_get_delegate_invoke (MonoMethod *method, MonoDelegate *del)
         * Check cache
         */
        if (ctx) {
-               MonoMethod *def, *inst;
-
-               /*
-                * Look for the instance
-                */
                cache = get_cache (&method->klass->image->delegate_invoke_generic_cache, mono_aligned_addr_hash, NULL);
-               res = mono_marshal_find_in_cache (cache, orig_method->klass);
+               res = check_generic_delegate_wrapper_cache (cache, orig_method, method, ctx);
                if (res)
                        return res;
-
-               /*
-                * Look for the definition
-                */
-               def = mono_marshal_find_in_cache (cache, method->klass);
-               if (def) {
-                       inst = mono_class_inflate_generic_method (def, ctx);
-                       /* Cache it */
-                       mono_memory_barrier ();
-                       mono_marshal_lock ();
-                       res = g_hash_table_lookup (cache, orig_method->klass);
-                       if (!res) {
-                               g_hash_table_insert (cache, orig_method->klass, inst);
-                               res = inst;
-                       }
-                       mono_marshal_unlock ();
-                       return res;
-               }
        } else if (callvirt || static_method_with_first_arg_bound) {
                GHashTable **cache_ptr;
                if (static_method_with_first_arg_bound)
@@ -4176,22 +4298,10 @@ mono_marshal_get_delegate_invoke (MonoMethod *method, MonoDelegate *del)
 #endif /* DISABLE_JIT */
 
        if (ctx) {
-               MonoMethod *def, *inst;
+               MonoMethod *def;
 
-               /*
-                * We use the same cache for the generic definition and the instances.
-                */
                def = mono_mb_create_and_cache (cache, method->klass, mb, sig, sig->param_count + 16);
-
-               inst = mono_class_inflate_generic_method (def, ctx);
-               mono_memory_barrier ();
-               mono_marshal_lock ();
-               res = g_hash_table_lookup (cache, orig_method->klass);
-               if (!res) {
-                       g_hash_table_insert (cache, orig_method->klass, inst);
-                       res = inst;
-               }
-               mono_marshal_unlock ();
+               res = cache_generic_delegate_wrapper (cache, orig_method, def, ctx);
        } else if (static_method_with_first_arg_bound || callvirt) {
                // From mono_mb_create_and_cache
                newm = mono_mb_create_method (mb, sig, sig->param_count + 16);
@@ -4840,6 +4950,9 @@ mono_marshal_get_runtime_invoke (MonoMethod *method, gboolean virtual)
                                g_hash_table_insert (cache, callsig, res);
                                /* Can't insert it into wrapper_hash since the key is a signature */
                                g_hash_table_insert (method->klass->image->runtime_invoke_direct_cache, method, res);
+                               info = mono_wrapper_info_create (res, WRAPPER_SUBTYPE_RUNTIME_INVOKE_NORMAL);
+                               info->d.runtime_invoke.sig = callsig;
+                               mono_marshal_set_wrapper_info (res, info);
                        } else {
                                mono_free_method (newm);
                        }
@@ -4992,6 +5105,7 @@ mono_mb_emit_auto_layout_exception (MonoMethodBuilder *mb, MonoClass *klass)
 }
 #endif
 
+#ifndef DISABLE_REMOTING
 /*
  * mono_marshal_get_ldfld_remote_wrapper:
  * @klass: The return type
@@ -5546,6 +5660,7 @@ mono_marshal_get_stfld_wrapper (MonoType *type)
        
        return res;
 }
+#endif /* DISABLE_REMOTING */
 
 /*
  * generates IL code for the icall wrapper (the generated method
@@ -5580,7 +5695,9 @@ mono_marshal_get_icall_wrapper (MonoMethodSignature *sig, const char *name, gcon
        for (i = 0; i < sig->param_count; i++)
                mono_mb_emit_ldarg (mb, i + sig->hasthis);
 
-       mono_mb_emit_native_call (mb, csig2, (gpointer) func);
+       mono_mb_emit_byte (mb, MONO_CUSTOM_PREFIX);
+       mono_mb_emit_op (mb, CEE_MONO_JIT_ICALL_ADDR, (gpointer)func);
+       mono_mb_emit_calli (mb, csig2);
        if (check_exceptions)
                emit_thread_interrupt_checkpoint (mb);
        mono_mb_emit_byte (mb, CEE_RET);
@@ -5595,18 +5712,23 @@ mono_marshal_get_icall_wrapper (MonoMethodSignature *sig, const char *name, gcon
        mono_mb_free (mb);
 
        info = mono_wrapper_info_create (res, WRAPPER_SUBTYPE_ICALL_WRAPPER);
+       info->d.icall.func = (gpointer)func;
        mono_marshal_set_wrapper_info (res, info);
        
        return res;
 }
 
-#ifndef DISABLE_JIT
 static int
 emit_marshal_custom (EmitMarshalContext *m, int argnum, MonoType *t,
                                         MonoMarshalSpec *spec, 
                                         int conv_arg, MonoType **conv_arg_type, 
                                         MarshalAction action)
 {
+#ifdef DISABLE_JIT
+       if (action == MARSHAL_ACTION_CONV_IN && t->type == MONO_TYPE_VALUETYPE)
+               *conv_arg_type = &mono_defaults.int_class->byval_arg;
+       return conv_arg;
+#else
        MonoType *mtype;
        MonoClass *mklass;
        static MonoClass *ICustomMarshaler = NULL;
@@ -5900,8 +6022,9 @@ handle_exception:
        default:
                g_assert_not_reached ();
        }
-               
        return conv_arg;
+#endif
+
 }
 
 static int
@@ -5910,6 +6033,7 @@ emit_marshal_asany (EmitMarshalContext *m, int argnum, MonoType *t,
                                        int conv_arg, MonoType **conv_arg_type, 
                                        MarshalAction action)
 {
+#ifndef DISABLE_JIT
        MonoMethodBuilder *mb = m->mb;
 
        switch (action) {
@@ -5946,7 +6070,7 @@ emit_marshal_asany (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                g_assert_not_reached ();
        }
-
+#endif
        return conv_arg;
 }
 
@@ -5956,6 +6080,7 @@ emit_marshal_vtype (EmitMarshalContext *m, int argnum, MonoType *t,
                                        int conv_arg, MonoType **conv_arg_type, 
                                        MarshalAction action)
 {
+#ifndef DISABLE_JIT
        MonoMethodBuilder *mb = m->mb;
        MonoClass *klass, *date_time_class;
        int pos = 0, pos2;
@@ -6229,7 +6354,7 @@ emit_marshal_vtype (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                g_assert_not_reached ();
        }
-
+#endif
        return conv_arg;
 }
 
@@ -6239,6 +6364,16 @@ emit_marshal_string (EmitMarshalContext *m, int argnum, MonoType *t,
                                         int conv_arg, MonoType **conv_arg_type, 
                                         MarshalAction action)
 {
+#ifdef DISABLE_JIT
+       switch (action) {
+       case MARSHAL_ACTION_CONV_IN:
+               *conv_arg_type = &mono_defaults.int_class->byval_arg;
+               break;
+       case MARSHAL_ACTION_MANAGED_CONV_IN:
+               *conv_arg_type = &mono_defaults.int_class->byval_arg;
+               break;
+       }
+#else
        MonoMethodBuilder *mb = m->mb;
        MonoMarshalNative encoding = mono_marshal_get_string_encoding (m->piinfo, spec);
        MonoMarshalConv conv = mono_marshal_get_string_to_ptr_conv (m->piinfo, spec);
@@ -6396,15 +6531,20 @@ emit_marshal_string (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                g_assert_not_reached ();
        }
-
+#endif
        return conv_arg;
 }
 
+
 static int
 emit_marshal_safehandle (EmitMarshalContext *m, int argnum, MonoType *t, 
                         MonoMarshalSpec *spec, int conv_arg, 
                         MonoType **conv_arg_type, MarshalAction action)
 {
+#ifdef DISABLE_JIT
+       if (action == MARSHAL_ACTION_CONV_IN)
+               *conv_arg_type = &mono_defaults.int_class->byval_arg;
+#else
        MonoMethodBuilder *mb = m->mb;
 
        switch (action){
@@ -6555,15 +6695,20 @@ emit_marshal_safehandle (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                printf ("Unhandled case for MarshalAction: %d\n", action);
        }
-
+#endif
        return conv_arg;
 }
 
+
 static int
 emit_marshal_handleref (EmitMarshalContext *m, int argnum, MonoType *t, 
                        MonoMarshalSpec *spec, int conv_arg, 
                        MonoType **conv_arg_type, MarshalAction action)
 {
+#ifdef DISABLE_JIT
+       if (action == MARSHAL_ACTION_CONV_IN)
+               *conv_arg_type = &mono_defaults.int_class->byval_arg;
+#else
        MonoMethodBuilder *mb = m->mb;
 
        switch (action){
@@ -6616,16 +6761,21 @@ emit_marshal_handleref (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                fprintf (stderr, "Unhandled case for MarshalAction: %d\n", action);
        }
-
+#endif
        return conv_arg;
 }
 
+
 static int
 emit_marshal_object (EmitMarshalContext *m, int argnum, MonoType *t,
                     MonoMarshalSpec *spec, 
                     int conv_arg, MonoType **conv_arg_type, 
                     MarshalAction action)
 {
+#ifdef DISABLE_JIT
+       if (action == MARSHAL_ACTION_CONV_IN)
+               *conv_arg_type = &mono_defaults.int_class->byval_arg;
+#else
        MonoMethodBuilder *mb = m->mb;
        MonoClass *klass = mono_class_from_mono_type (t);
        int pos, pos2, loc;
@@ -7099,10 +7249,11 @@ emit_marshal_object (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                g_assert_not_reached ();
        }
-
+#endif
        return conv_arg;
 }
 
+#ifndef DISABLE_JIT
 
 #ifndef DISABLE_COM
 
@@ -7222,6 +7373,7 @@ emit_marshal_variant (EmitMarshalContext *m, int argnum, MonoType *t,
 }
 
 #endif /* DISABLE_COM */
+#endif /* DISABLE_JIT */
 
 static gboolean
 mono_pinvoke_is_unicode (MonoMethodPInvoke *piinfo)
@@ -7241,12 +7393,23 @@ mono_pinvoke_is_unicode (MonoMethodPInvoke *piinfo)
        }
 }
 
+
 static int
 emit_marshal_array (EmitMarshalContext *m, int argnum, MonoType *t,
                                        MonoMarshalSpec *spec, 
                                        int conv_arg, MonoType **conv_arg_type, 
                                        MarshalAction action)
 {
+#ifdef DISABLE_JIT
+       switch (action) {
+       case MARSHAL_ACTION_CONV_IN:
+               *conv_arg_type = &mono_defaults.object_class->byval_arg;
+               break;
+       case MARSHAL_ACTION_MANAGED_CONV_IN:
+               *conv_arg_type = &mono_defaults.int_class->byval_arg;
+               break;
+       }
+#else
        MonoMethodBuilder *mb = m->mb;
        MonoClass *klass = mono_class_from_mono_type (t);
        gboolean need_convert, need_free;
@@ -7919,16 +8082,82 @@ emit_marshal_array (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                g_assert_not_reached ();
        }
-
+#endif
        return conv_arg;
 }
 
+static MonoType*
+marshal_boolean_conv_in_get_local_type (MonoMarshalSpec *spec, guint8 *ldc_op /*out*/)
+{
+       if (spec == NULL) {
+               return &mono_defaults.int32_class->byval_arg;
+       } else {
+               switch (spec->native) {
+               case MONO_NATIVE_I1:
+               case MONO_NATIVE_U1:
+                       return &mono_defaults.byte_class->byval_arg;
+               case MONO_NATIVE_VARIANTBOOL:
+                       if (ldc_op) *ldc_op = CEE_LDC_I4_M1;
+                       return &mono_defaults.int16_class->byval_arg;
+               case MONO_NATIVE_BOOLEAN:
+                       return &mono_defaults.int32_class->byval_arg;
+               default:
+                       g_warning ("marshalling bool as native type %x is currently not supported", spec->native);
+                       return &mono_defaults.int32_class->byval_arg;
+               }
+       }
+}
+
+static MonoClass*
+marshal_boolean_managed_conv_in_get_conv_arg_class (MonoMarshalSpec *spec, guint8 *ldop/*out*/)
+{
+       MonoClass* conv_arg_class = mono_defaults.int32_class;
+       if (spec) {
+               switch (spec->native) {
+               case MONO_NATIVE_I1:
+               case MONO_NATIVE_U1:
+                       conv_arg_class = mono_defaults.byte_class;
+                       if (ldop) *ldop = CEE_LDIND_I1;
+                       break;
+               case MONO_NATIVE_VARIANTBOOL:
+                       conv_arg_class = mono_defaults.int16_class;
+                       if (ldop) *ldop = CEE_LDIND_I2;
+                       break;
+               case MONO_NATIVE_BOOLEAN:
+                       break;
+               default:
+                       g_warning ("marshalling bool as native type %x is currently not supported", spec->native);
+               }
+       }
+       return conv_arg_class;
+}
+
 static int
 emit_marshal_boolean (EmitMarshalContext *m, int argnum, MonoType *t,
                      MonoMarshalSpec *spec, 
                      int conv_arg, MonoType **conv_arg_type, 
                      MarshalAction action)
 {
+#ifdef DISABLE_JIT
+       switch (action) {
+       case MARSHAL_ACTION_CONV_IN:
+               if (t->byref)
+                       *conv_arg_type = &mono_defaults.int_class->byval_arg;
+               else
+                       *conv_arg_type = marshal_boolean_conv_in_get_local_type (spec, NULL);
+               break;
+
+       case MARSHAL_ACTION_MANAGED_CONV_IN: {
+               MonoClass* conv_arg_class = marshal_boolean_managed_conv_in_get_conv_arg_class (spec, NULL);
+               if (t->byref)
+                       *conv_arg_type = &conv_arg_class->this_arg;
+               else
+                       *conv_arg_type = &conv_arg_class->byval_arg;
+               break;
+       }
+
+       }
+#else
        MonoMethodBuilder *mb = m->mb;
 
        switch (action) {
@@ -7937,27 +8166,7 @@ emit_marshal_boolean (EmitMarshalContext *m, int argnum, MonoType *t,
                int label_false;
                guint8 ldc_op = CEE_LDC_I4_1;
 
-               if (spec == NULL) {
-                       local_type = &mono_defaults.int32_class->byval_arg;
-               } else {
-                       switch (spec->native) {
-                       case MONO_NATIVE_I1:
-                       case MONO_NATIVE_U1:
-                               local_type = &mono_defaults.byte_class->byval_arg;
-                               break;
-                       case MONO_NATIVE_VARIANTBOOL:
-                               local_type = &mono_defaults.int16_class->byval_arg;
-                               ldc_op = CEE_LDC_I4_M1;
-                               break;
-                       case MONO_NATIVE_BOOLEAN:
-                               local_type = &mono_defaults.int32_class->byval_arg;
-                               break;
-                       default:
-                               g_warning ("marshalling bool as native type %x is currently not supported", spec->native);
-                               local_type = &mono_defaults.int32_class->byval_arg;
-                               break;
-                       }
-               }
+               local_type = marshal_boolean_conv_in_get_local_type (spec, &ldc_op);
                if (t->byref)
                        *conv_arg_type = &mono_defaults.int_class->byval_arg;
                else
@@ -8015,26 +8224,9 @@ emit_marshal_boolean (EmitMarshalContext *m, int argnum, MonoType *t,
                guint8 ldop = CEE_LDIND_I4;
                int label_null, label_false;
 
+               conv_arg_class = marshal_boolean_managed_conv_in_get_conv_arg_class (spec, &ldop);
                conv_arg = mono_mb_add_local (mb, &mono_defaults.boolean_class->byval_arg);
 
-               if (spec) {
-                       switch (spec->native) {
-                       case MONO_NATIVE_I1:
-                       case MONO_NATIVE_U1:
-                               conv_arg_class = mono_defaults.byte_class;
-                               ldop = CEE_LDIND_I1;
-                               break;
-                       case MONO_NATIVE_VARIANTBOOL:
-                               conv_arg_class = mono_defaults.int16_class;
-                               ldop = CEE_LDIND_I2;
-                               break;
-                       case MONO_NATIVE_BOOLEAN:
-                               break;
-                       default:
-                               g_warning ("marshalling bool as native type %x is currently not supported", spec->native);
-                       }
-               }
-
                if (t->byref)
                        *conv_arg_type = &conv_arg_class->this_arg;
                else
@@ -8106,7 +8298,7 @@ emit_marshal_boolean (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                g_assert_not_reached ();
        }
-
+#endif
        return conv_arg;
 }
 
@@ -8115,6 +8307,7 @@ emit_marshal_ptr (EmitMarshalContext *m, int argnum, MonoType *t,
                  MonoMarshalSpec *spec, int conv_arg, 
                  MonoType **conv_arg_type, MarshalAction action)
 {
+#ifndef DISABLE_JIT
        MonoMethodBuilder *mb = m->mb;
 
        switch (action) {
@@ -8140,7 +8333,7 @@ emit_marshal_ptr (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                break;
        }
-
+#endif
        return conv_arg;
 }
 
@@ -8149,6 +8342,7 @@ emit_marshal_char (EmitMarshalContext *m, int argnum, MonoType *t,
                   MonoMarshalSpec *spec, int conv_arg, 
                   MonoType **conv_arg_type, MarshalAction action)
 {
+#ifndef DISABLE_JIT
        MonoMethodBuilder *mb = m->mb;
 
        switch (action) {
@@ -8167,7 +8361,7 @@ emit_marshal_char (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                break;
        }
-
+#endif
        return conv_arg;
 }
 
@@ -8176,6 +8370,7 @@ emit_marshal_scalar (EmitMarshalContext *m, int argnum, MonoType *t,
                     MonoMarshalSpec *spec, int conv_arg, 
                     MonoType **conv_arg_type, MarshalAction action)
 {
+#ifndef DISABLE_JIT
        MonoMethodBuilder *mb = m->mb;
 
        switch (action) {
@@ -8191,7 +8386,7 @@ emit_marshal_scalar (EmitMarshalContext *m, int argnum, MonoType *t,
        default:
                break;
        }
-
+#endif
        return conv_arg;
 }
 
@@ -8267,10 +8462,10 @@ emit_marshal (EmitMarshalContext *m, int argnum, MonoType *t,
                else
                        return emit_marshal_object (m, argnum, t, spec, conv_arg, conv_arg_type, action);
        }
-
        return conv_arg;
 }
 
+#ifndef DISABLE_JIT
 /**
  * mono_marshal_emit_native_wrapper:
  * @image: the image to use for looking up custom marshallers
@@ -8490,9 +8685,9 @@ mono_marshal_emit_native_wrapper (MonoImage *image, MonoMethodBuilder *mb, MonoM
 
        mono_mb_emit_byte (mb, CEE_RET);
 }
-
 #endif /* DISABLE_JIT */
 
+
 G_GNUC_UNUSED static void
 code_for (MonoMethod *method) {
        MonoMethodHeader *header = mono_method_get_header (method);
@@ -8613,21 +8808,26 @@ mono_marshal_get_native_wrapper (MonoMethod *method, gboolean check_exceptions,
 
        mb->method->save_lmf = 1;
 
-#ifndef DISABLE_JIT
        /*
         * In AOT mode and embedding scenarios, it is possible that the icall is not
         * registered in the runtime doing the AOT compilation.
         */
        if (!piinfo->addr && !aot) {
+#ifndef DISABLE_JIT
                mono_mb_emit_exception (mb, exc_class, exc_arg);
+#endif
                csig = signature_dup (method->klass->image, sig);
                csig->pinvoke = 0;
                res = mono_mb_create_and_cache (cache, method,
                                                                                mb, csig, csig->param_count + 16);
                mono_mb_free (mb);
+
+               info = mono_wrapper_info_create (res, WRAPPER_SUBTYPE_NONE);
+               info->d.managed_to_native.method = method;
+               mono_marshal_set_wrapper_info (res, info);
+
                return res;
        }
-#endif
 
        /* internal calls: we simply push all arguments and call the method (no conversions) */
        if (method->iflags & (METHOD_IMPL_ATTRIBUTE_INTERNAL_CALL | METHOD_IMPL_ATTRIBUTE_RUNTIME)) {
@@ -8828,7 +9028,6 @@ mono_marshal_get_native_func_wrapper_aot (MonoClass *klass)
        return res;
 }
 
-#ifndef DISABLE_JIT
 /*
  * mono_marshal_emit_managed_wrapper:
  *
@@ -8840,6 +9039,29 @@ mono_marshal_get_native_func_wrapper_aot (MonoClass *klass)
 void
 mono_marshal_emit_managed_wrapper (MonoMethodBuilder *mb, MonoMethodSignature *invoke_sig, MonoMarshalSpec **mspecs, EmitMarshalContext* m, MonoMethod *method, uint32_t target_handle)
 {
+#ifdef DISABLE_JIT
+       MonoMethodSignature *sig, *csig;
+       int i;
+
+       sig = m->sig;
+       csig = m->csig;
+
+       /* we first do all conversions */
+       for (i = 0; i < sig->param_count; i ++) {
+               MonoType *t = sig->params [i];
+
+               switch (t->type) {
+               case MONO_TYPE_OBJECT:
+               case MONO_TYPE_CLASS:
+               case MONO_TYPE_VALUETYPE:
+               case MONO_TYPE_ARRAY:
+               case MONO_TYPE_SZARRAY:
+               case MONO_TYPE_STRING:
+               case MONO_TYPE_BOOLEAN:
+                       emit_marshal (m, i, sig->params [i], mspecs [i + 1], 0, &csig->params [i], MARSHAL_ACTION_MANAGED_CONV_IN);
+               }
+       }
+#else
        MonoMethodSignature *sig, *csig;
        int i, *tmp_locals;
        gboolean closed = FALSE;
@@ -9021,8 +9243,8 @@ mono_marshal_emit_managed_wrapper (MonoMethodBuilder *mb, MonoMethodSignature *i
 
        if (closed)
                g_free (sig);
+#endif
 }
-#endif /* DISABLE_JIT */
 
 static void 
 mono_marshal_set_callconv_from_modopt (MonoMethod *method, MonoMethodSignature *csig)
@@ -9196,9 +9418,7 @@ mono_marshal_get_managed_wrapper (MonoMethod *method, MonoClass *delegate_klass,
                        mono_custom_attrs_free (cinfo);
        }
 
-#ifndef DISABLE_JIT
        mono_marshal_emit_managed_wrapper (mb, invoke_sig, mspecs, &m, method, target_handle);
-#endif
 
        if (!target_handle) {
                WrapperInfo *info;
@@ -9268,9 +9488,9 @@ mono_marshal_get_vtfixup_ftnptr (MonoImage *image, guint32 token, guint16 type)
 
                /* FIXME: Implement VTFIXUP_TYPE_FROM_UNMANAGED_RETAIN_APPDOMAIN. */
 
-#ifndef DISABLE_JIT
                mono_marshal_emit_managed_wrapper (mb, sig, mspecs, &m, method, 0);
 
+#ifndef DISABLE_JIT
                mb->dynamic = 1;
 #endif
                method = mono_mb_create_method (mb, csig, sig->param_count + 16);
@@ -9574,11 +9794,11 @@ mono_marshal_get_isinst (MonoClass *klass)
        mono_mb_emit_byte (mb, MONO_CUSTOM_PREFIX);
        mono_mb_emit_op (mb, CEE_MONO_CISINST, klass);
 
-       /* The result of MONO_ISINST can be:
+       /* The result of MONO_CISINST can be:
                0) the type check succeeded
                1) the type check did not succeed
                2) a CanCastTo call is needed */
-       
+#ifndef DISABLE_REMOTING
        mono_mb_emit_byte (mb, CEE_DUP);
        pos_was_ok = mono_mb_emit_branch (mb, CEE_BRFALSE);
 
@@ -9608,7 +9828,25 @@ mono_marshal_get_isinst (MonoClass *klass)
        mono_mb_patch_branch (mb, pos_end);
        mono_mb_patch_branch (mb, pos_end2);
        mono_mb_emit_byte (mb, CEE_RET);
-#endif
+#else
+       pos_was_ok = mono_mb_emit_branch (mb, CEE_BRFALSE);
+
+       /* fail */
+
+       mono_mb_emit_byte (mb, CEE_LDNULL);
+       pos_end = mono_mb_emit_branch (mb, CEE_BR);
+
+       /* success */
+
+       mono_mb_patch_branch (mb, pos_was_ok);
+       mono_mb_emit_ldarg (mb, 0);
+
+       /* the end */
+
+       mono_mb_patch_branch (mb, pos_end);
+       mono_mb_emit_byte (mb, CEE_RET);
+#endif /* DISABLE_REMOTING */
+#endif /* DISABLE_JIT */
 
        res = mono_mb_create_and_cache (cache, klass, mb, isint_sig, isint_sig->param_count + 16);
        mono_mb_free (mb);
@@ -9660,12 +9898,12 @@ mono_marshal_get_castclass (MonoClass *klass)
        mono_mb_emit_byte (mb, MONO_CUSTOM_PREFIX);
        mono_mb_emit_op (mb, CEE_MONO_CCASTCLASS, klass);
 
-       /* The result of MONO_ISINST can be:
+       /* The result of MONO_CCASTCLASS can be:
                0) the cast is valid
                1) cast of unknown proxy type
                or an exception if the cast is is invalid
        */
-       
+#ifndef DISABLE_REMOTING
        pos_was_ok = mono_mb_emit_branch (mb, CEE_BRFALSE);
 
        /* get the real proxy from the transparent proxy*/
@@ -9680,11 +9918,16 @@ mono_marshal_get_castclass (MonoClass *klass)
        /* success */
        mono_mb_patch_branch (mb, pos_was_ok);
        mono_mb_patch_branch (mb, pos_was_ok2);
+#else
+       /* MONO_CCASTCLASS leaves an int in the stack with the result, pop it. */
+       mono_mb_emit_byte (mb, CEE_POP);
+#endif /* DISABLE_REMOTING */
+
        mono_mb_emit_ldarg (mb, 0);
        
        /* the end */
        mono_mb_emit_byte (mb, CEE_RET);
-#endif
+#endif /* DISABLE_JIT */
 
        res = mono_mb_create_and_cache (cache, klass, mb, castclass_sig, castclass_sig->param_count + 16);
        mono_mb_free (mb);
@@ -9695,6 +9938,7 @@ mono_marshal_get_castclass (MonoClass *klass)
        return res;
 }
 
+#ifndef DISABLE_REMOTING
 MonoMethod *
 mono_marshal_get_proxy_cancast (MonoClass *klass)
 {
@@ -9783,6 +10027,7 @@ mono_upgrade_remote_class_wrapper (MonoReflectionType *rtype, MonoTransparentPro
        klass = mono_class_from_mono_type (rtype->type);
        mono_upgrade_remote_class (domain, (MonoObject*)tproxy, klass);
 }
+#endif /*DISABLE_REMOTING */
 
 /**
  * mono_marshal_get_struct_to_ptr:
@@ -10186,7 +10431,7 @@ get_virtual_stelemref_kind (MonoClass *element_class)
                return STELEMREF_INTERFACE;
 #endif
        /*Arrays are sealed but are covariant on their element type, We can't use any of the fast paths.*/
-       if (element_class->marshalbyref || element_class->rank || mono_class_has_variant_generic_params (element_class))
+       if (mono_class_is_marshalbyref (element_class) || element_class->rank || mono_class_has_variant_generic_params (element_class))
                return STELEMREF_COMPLEX;
        if (element_class->flags & TYPE_ATTRIBUTE_SEALED)
                return STELEMREF_SEALED_CLASS;
@@ -10247,14 +10492,13 @@ record_slot_vstore (MonoObject *array, size_t index, MonoObject *value)
  *     - Maybe mve some MonoClass field into the vtable to reduce the number of loads
  *     - Add a case for arrays of arrays.
  */
-MonoMethod*
-mono_marshal_get_virtual_stelemref (MonoClass *array_class)
+static MonoMethod*
+get_virtual_stelemref_wrapper (int kind)
 {
        static MonoMethod *cached_methods [STELEMREF_KIND_COUNT] = { NULL }; /*object iface sealed regular*/
        static MonoMethodSignature *signature;
        MonoMethodBuilder *mb;
        MonoMethod *res;
-       int kind;
        char *name;
        const char *param_names [16];
        guint32 b1, b2, b3;
@@ -10262,9 +10506,6 @@ mono_marshal_get_virtual_stelemref (MonoClass *array_class)
        int array_slot_addr;
        WrapperInfo *info;
 
-       g_assert (array_class->rank == 1);
-       kind = get_virtual_stelemref_kind (array_class->element_class);
-
        if (cached_methods [kind])
                return cached_methods [kind];
 
@@ -10632,6 +10873,30 @@ mono_marshal_get_virtual_stelemref (MonoClass *array_class)
        return cached_methods [kind];
 }
 
+MonoMethod*
+mono_marshal_get_virtual_stelemref (MonoClass *array_class)
+{
+       int kind;
+
+       g_assert (array_class->rank == 1);
+       kind = get_virtual_stelemref_kind (array_class->element_class);
+
+       return get_virtual_stelemref_wrapper (kind);
+}
+
+MonoMethod**
+mono_marshal_get_virtual_stelemref_wrappers (int *nwrappers)
+{
+       MonoMethod **res;
+       int i;
+
+       *nwrappers = STELEMREF_KIND_COUNT;
+       res = g_malloc0 (STELEMREF_KIND_COUNT * sizeof (MonoMethod*));
+       for (i = 0; i < STELEMREF_KIND_COUNT; ++i)
+               res [i] = get_virtual_stelemref_wrapper (i);
+       return res;
+}
+
 /*
  * The wrapper info for the wrapper is a WrapperInfo structure.
  */