[utils] Another try to fix the windows build.
[mono.git] / mono / metadata / class.c
index c9cdaf1df37814ec68f3c5a133cace22929c1f39..041b9e3dcd61c72b0fd2bd820763b8f53c62e59c 100644 (file)
@@ -72,10 +72,6 @@ static guint32 mono_field_resolve_flags (MonoClassField *field);
 static void mono_class_setup_vtable_full (MonoClass *class, GList *in_setup);
 static void mono_generic_class_setup_parent (MonoClass *klass, MonoClass *gklass);
 
-
-void (*mono_debugger_class_init_func) (MonoClass *klass) = NULL;
-
-
 /*
 We use gclass recording to allow recursive system f types to be referenced by a parent.
 
@@ -191,9 +187,9 @@ mono_class_from_typeref_checked (MonoImage *image, guint32 type_token, MonoError
        name = mono_metadata_string_heap (image, cols [MONO_TYPEREF_NAME]);
        nspace = mono_metadata_string_heap (image, cols [MONO_TYPEREF_NAMESPACE]);
 
-       idx = cols [MONO_TYPEREF_SCOPE] >> MONO_RESOLTION_SCOPE_BITS;
-       switch (cols [MONO_TYPEREF_SCOPE] & MONO_RESOLTION_SCOPE_MASK) {
-       case MONO_RESOLTION_SCOPE_MODULE:
+       idx = cols [MONO_TYPEREF_SCOPE] >> MONO_RESOLUTION_SCOPE_BITS;
+       switch (cols [MONO_TYPEREF_SCOPE] & MONO_RESOLUTION_SCOPE_MASK) {
+       case MONO_RESOLUTION_SCOPE_MODULE:
                /*
                LAMESPEC The spec says that a null module resolution scope should go through the exported type table.
                This is not the observed behavior of existing implementations.
@@ -203,13 +199,13 @@ mono_class_from_typeref_checked (MonoImage *image, guint32 type_token, MonoError
                res = mono_class_from_name (image, nspace, name); /*FIXME proper error handling*/
                goto done;
 
-       case MONO_RESOLTION_SCOPE_MODULEREF:
+       case MONO_RESOLUTION_SCOPE_MODULEREF:
                module = mono_image_load_module (image, idx);
                if (module)
                        res = mono_class_from_name (module, nspace, name); /*FIXME proper error handling*/
                goto done;
 
-       case MONO_RESOLTION_SCOPE_TYPEREF: {
+       case MONO_RESOLUTION_SCOPE_TYPEREF: {
                MonoClass *enclosing;
                GList *tmp;
 
@@ -246,7 +242,7 @@ mono_class_from_typeref_checked (MonoImage *image, guint32 type_token, MonoError
                g_warning ("TypeRef ResolutionScope not yet handled (%d) for %s.%s in image %s", idx, nspace, name, image->name);
                goto done;
        }
-       case MONO_RESOLTION_SCOPE_ASSEMBLYREF:
+       case MONO_RESOLUTION_SCOPE_ASSEMBLYREF:
                break;
        }
 
@@ -1082,10 +1078,8 @@ mono_class_inflate_generic_method_full_checked (MonoMethod *method, MonoClass *k
                        iresult->context.class_inst = iresult->declaring->klass->generic_class->context.class_inst;
        }
 
-       mono_loader_lock ();
        cached = mono_method_inflated_lookup (iresult, FALSE);
        if (cached) {
-               mono_loader_unlock ();
                g_free (iresult);
                return (MonoMethod*)cached;
        }
@@ -1156,12 +1150,9 @@ mono_class_inflate_generic_method_full_checked (MonoMethod *method, MonoClass *k
         * is_generic_method_definition().
         */
 
-       mono_method_inflated_lookup (iresult, TRUE);
-       mono_loader_unlock ();
-       return result;
+       return (MonoMethod*)mono_method_inflated_lookup (iresult, TRUE);
 
 fail:
-       mono_loader_unlock ();
        g_free (iresult);
        return NULL;
 }
@@ -1241,7 +1232,7 @@ mono_method_get_generic_container (MonoMethod *method)
  * mono_method_set_generic_container:
  *
  *   Sets the generic container of METHOD to CONTAINER.
- * LOCKING: Acquires the loader lock.
+ * LOCKING: Acquires the image lock.
  */
 void
 mono_method_set_generic_container (MonoMethod *method, MonoGenericContainer* container)
@@ -1259,7 +1250,7 @@ mono_method_set_generic_container (MonoMethod *method, MonoGenericContainer* con
  * in a separate function since it is cheaper than calling mono_class_setup_fields.
  */
 static MonoType*
-mono_class_find_enum_basetype (MonoClass *class)
+mono_class_find_enum_basetype (MonoClass *class, MonoError *error)
 {
        MonoGenericContainer *container = NULL;
        MonoImage *m = class->image; 
@@ -1268,6 +1259,8 @@ mono_class_find_enum_basetype (MonoClass *class)
 
        g_assert (class->enumtype);
 
+       mono_error_init (error);
+
        if (class->generic_container)
                container = class->generic_container;
        else if (class->generic_class) {
@@ -1292,27 +1285,41 @@ mono_class_find_enum_basetype (MonoClass *class)
                if (cols [MONO_FIELD_FLAGS] & FIELD_ATTRIBUTE_STATIC) //no need to decode static fields
                        continue;
 
-               if (!mono_verifier_verify_field_signature (class->image, cols [MONO_FIELD_SIGNATURE], NULL))
-                       return NULL;
+               if (!mono_verifier_verify_field_signature (class->image, cols [MONO_FIELD_SIGNATURE], NULL)) {
+                       mono_error_set_bad_image (error, class->image, "Invalid field signature %x", cols [MONO_FIELD_SIGNATURE]);
+                       goto fail;
+               }
 
                sig = mono_metadata_blob_heap (m, cols [MONO_FIELD_SIGNATURE]);
                mono_metadata_decode_value (sig, &sig);
                /* FIELD signature == 0x06 */
-               if (*sig != 0x06)
-                       return NULL;
+               if (*sig != 0x06) {
+                       mono_error_set_bad_image (error, class->image, "Invalid field signature %x, expected 0x6 but got %x", cols [MONO_FIELD_SIGNATURE], *sig);
+                       goto fail;
+               }
 
                ftype = mono_metadata_parse_type_full (m, container, MONO_PARSE_FIELD, cols [MONO_FIELD_FLAGS], sig + 1, &sig);
-               if (!ftype)
-                       return NULL;
+               if (!ftype) {
+                       if (mono_loader_get_last_error ()) /*FIXME plug the above to not leak errors*/
+                               mono_error_set_from_loader_error (error);
+                       else
+                               mono_error_set_bad_image (error, class->image, "Could not parse type for field signature %x", cols [MONO_FIELD_SIGNATURE]);
+                       goto fail;
+               }
                if (class->generic_class) {
                        //FIXME do we leak here?
-                       ftype = mono_class_inflate_generic_type (ftype, mono_class_get_context (class));
+                       ftype = mono_class_inflate_generic_type_checked (ftype, mono_class_get_context (class), error);
+                       if (!mono_error_ok (error))
+                               goto fail;
                        ftype->attrs = cols [MONO_FIELD_FLAGS];
                }
 
                return ftype;
        }
+       mono_error_set_type_load_class (error, class, "Could not find base type");
 
+fail:
+       g_assert (!mono_loader_get_last_error ());
        return NULL;
 }
 
@@ -1552,8 +1559,8 @@ mono_class_setup_fields (MonoClass *class)
        explicit_size = mono_metadata_packing_from_typedef (class->image, class->type_token, &packing_size, &real_size);
 
        if (explicit_size) {
-               if ((packing_size & 0xfffffff0) != 0) {
-                       char *err_msg = g_strdup_printf ("Could not load struct '%s' with packing size %d >= 16", class->name, packing_size);
+               if ((packing_size & 0xffffff00) != 0) {
+                       char *err_msg = g_strdup_printf ("Could not load struct '%s' with packing size %d >= 256", class->name, packing_size);
                        mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, err_msg);
                        return;
                }
@@ -1626,6 +1633,10 @@ mono_class_setup_fields (MonoClass *class)
                                        mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, g_strdup_printf ("Invalid negative field offset %d for %s", field->offset, field->name));
                                        break;
                                }
+                               if (class->generic_container) {
+                                       mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, g_strdup_printf ("Generic class cannot have explicit layout."));
+                                       break;
+                               }
                        }
                }
 
@@ -1923,7 +1934,9 @@ mono_class_layout_fields (MonoClass *class)
                        }
                }
                break;
-       case TYPE_ATTRIBUTE_EXPLICIT_LAYOUT:
+       case TYPE_ATTRIBUTE_EXPLICIT_LAYOUT: {
+               guint8 *ref_bitmap;
+
                real_size = 0;
                for (i = 0; i < top; i++) {
                        gint32 align;
@@ -1936,7 +1949,6 @@ mono_class_layout_fields (MonoClass *class)
                         * There must be info about all the fields in a type if it
                         * uses explicit layout.
                         */
-
                        if (mono_field_is_deleted (field))
                                continue;
                        if (field->type->attrs & FIELD_ATTRIBUTE_STATIC)
@@ -1966,6 +1978,43 @@ mono_class_layout_fields (MonoClass *class)
                         */
                        real_size = MAX (real_size, size + field->offset);
                }
+
+               if (class->has_references) {
+                       ref_bitmap = g_new0 (guint8, real_size / sizeof (gpointer));
+
+                       /* Check for overlapping reference and non-reference fields */
+                       for (i = 0; i < top; i++) {
+                               MonoType *ftype;
+
+                               field = &class->fields [i];
+
+                               if (mono_field_is_deleted (field))
+                                       continue;
+                               if (field->type->attrs & FIELD_ATTRIBUTE_STATIC)
+                                       continue;
+                               ftype = mono_type_get_underlying_type (field->type);
+                               if (MONO_TYPE_IS_REFERENCE (ftype))
+                                       ref_bitmap [field->offset / sizeof (gpointer)] = 1;
+                       }
+                       for (i = 0; i < top; i++) {
+                               field = &class->fields [i];
+
+                               if (mono_field_is_deleted (field))
+                                       continue;
+                               if (field->type->attrs & FIELD_ATTRIBUTE_STATIC)
+                                       continue;
+
+                               // FIXME: Too much code does this
+#if 0
+                               if (!MONO_TYPE_IS_REFERENCE (field->type) && ref_bitmap [field->offset / sizeof (gpointer)]) {
+                                       char *err_msg = g_strdup_printf ("Could not load type '%s' because it contains an object field at offset %d that is incorrectly aligned or overlapped by a non-object field.", class->name, field->offset);
+                                       mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, err_msg);
+                               }
+#endif
+                       }
+                       g_free (ref_bitmap);
+               }
+
                class->instance_size = MAX (real_size, class->instance_size);
                if (class->instance_size & (class->min_align - 1)) {
                        class->instance_size += class->min_align - 1;
@@ -1973,6 +2022,7 @@ mono_class_layout_fields (MonoClass *class)
                }
                break;
        }
+       }
 
        if (layout != TYPE_ATTRIBUTE_EXPLICIT_LAYOUT) {
                /*
@@ -2098,12 +2148,18 @@ mono_class_setup_methods (MonoClass *class)
                MonoMethodSignature *sig;
                int count_generic = 0, first_generic = 0;
                int method_num = 0;
+               gboolean jagged_ctor = FALSE;
 
                count = 3 + (class->rank > 1? 2: 1);
 
                mono_class_setup_interfaces (class, &error);
                g_assert (mono_error_ok (&error)); /*FIXME can this fail for array types?*/
 
+               if (class->rank == 1 && class->element_class->rank) {
+                       jagged_ctor = TRUE;
+                       class->method.count ++;
+               }
+
                if (class->interface_count) {
                        count_generic = generic_array_methods (class);
                        first_generic = count;
@@ -2132,6 +2188,19 @@ mono_class_setup_methods (MonoClass *class)
                        amethod = create_array_method (class, ".ctor", sig);
                        methods [method_num++] = amethod;
                }
+
+               if (jagged_ctor) {
+                       /* Jagged arrays have an extra ctor in .net which creates an array of arrays */
+                       sig = mono_metadata_signature_alloc (class->image, class->rank + 1);
+                       sig->ret = &mono_defaults.void_class->byval_arg;
+                       sig->pinvoke = TRUE;
+                       sig->hasthis = TRUE;
+                       for (i = 0; i < class->rank + 1; ++i)
+                               sig->params [i] = &mono_defaults.int32_class->byval_arg;
+                       amethod = create_array_method (class, ".ctor", sig);
+                       methods [method_num++] = amethod;
+               }
+
                /* element Get (idx11, [idx2, ...]) */
                sig = mono_metadata_signature_alloc (class->image, class->rank);
                sig->ret = &class->element_class->byval_arg;
@@ -2457,7 +2526,7 @@ mono_class_setup_events (MonoClass *class)
 
        if (class->generic_class) {
                MonoClass *gklass = class->generic_class->container_class;
-               MonoGenericContext *context;
+               MonoGenericContext *context = NULL;
 
                mono_class_setup_events (gklass);
                if (gklass->exception_type) {
@@ -5192,9 +5261,6 @@ mono_class_init (MonoClass *class)
 
        mono_loader_unlock ();
 
-       if (mono_debugger_class_init_func)
-               mono_debugger_class_init_func (class);
-
        return class->exception_type == MONO_EXCEPTION_NONE;
 }
 
@@ -5682,10 +5748,12 @@ mono_class_create_from_typedef (MonoImage *image, guint32 type_token, MonoError
                        class->byval_arg.data.klass = class;
                        class->byval_arg.type = MONO_TYPE_CLASS;
                }
-               parent = mono_class_get_full (image, parent_token, context);
+               parent = mono_class_get_checked (image, parent_token, error);
+               if (parent && context) /* Always inflate */
+                       parent = mono_class_inflate_generic_class_checked (parent, context, error);
 
                if (parent == NULL) {
-                       mono_class_set_failure_from_loader_error (class, error, g_strdup_printf ("Could not load parent, token is %x", parent_token));
+                       mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, g_strdup (mono_error_get_message (error)));
                        goto parent_failure;
                }
 
@@ -5784,11 +5852,11 @@ mono_class_create_from_typedef (MonoImage *image, guint32 type_token, MonoError
        }
 
        if (class->enumtype) {
-               MonoType *enum_basetype = mono_class_find_enum_basetype (class);
+               MonoType *enum_basetype = mono_class_find_enum_basetype (class, error);
                if (!enum_basetype) {
                        /*set it to a default value as the whole runtime can't handle this to be null*/
                        class->cast_class = class->element_class = mono_defaults.int32_class;
-                       mono_class_set_failure_and_error (class, error, "Could not enum basetype");
+                       mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, g_strdup (mono_error_get_message (error)));
                        mono_loader_unlock ();
                        mono_profiler_class_loaded (class, MONO_PROFILE_FAILED);
                        g_assert (!mono_loader_get_last_error ());
@@ -5802,8 +5870,8 @@ mono_class_create_from_typedef (MonoImage *image, guint32 type_token, MonoError
         * We must do this after the class has been constructed to make certain recursive scenarios
         * work.
         */
-       if (class->generic_container && !mono_metadata_load_generic_param_constraints_full (image, type_token, class->generic_container)){
-               mono_class_set_failure_from_loader_error (class, error, g_strdup ("Could not load generic parameter constraints"));
+       if (class->generic_container && !mono_metadata_load_generic_param_constraints_checked (image, type_token, class->generic_container, error)) {
+               mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, g_strdup_printf ("Could not load generic parameter constrains due to %s", mono_error_get_message (error)));
                mono_loader_unlock ();
                mono_profiler_class_loaded (class, MONO_PROFILE_FAILED);
                g_assert (!mono_loader_get_last_error ());
@@ -6057,8 +6125,11 @@ make_generic_param_class (MonoGenericParam *param, MonoImage *image, gboolean is
 
 #define FAST_CACHE_SIZE 16
 
+/*
+ * LOCKING: Takes the image lock depending on @take_lock.
+ */
 static MonoClass *
-get_anon_gparam_class (MonoGenericParam *param, gboolean is_mvar)
+get_anon_gparam_class (MonoGenericParam *param, gboolean is_mvar, gboolean take_lock)
 {
        int n = mono_generic_param_num (param) | ((guint32)param->serial << 16);
        MonoImage *image = param->image;
@@ -6072,26 +6143,33 @@ get_anon_gparam_class (MonoGenericParam *param, gboolean is_mvar)
                else
                        return image->var_cache_fast ? image->var_cache_fast [n] : NULL;
        } else {
+               MonoClass *klass = NULL;
                ht = is_mvar ? image->mvar_cache_slow : image->var_cache_slow;
-               return ht ? g_hash_table_lookup (ht, GINT_TO_POINTER (n)) : NULL;
+               if (ht) {
+                       if (take_lock)
+                               mono_image_lock (image);
+                       klass = g_hash_table_lookup (ht, GINT_TO_POINTER (n));
+                       if (take_lock)
+                               mono_image_unlock (image);
+               }
+               return klass;
        }
 }
 
 /*
- * LOCKING: Acquires the loader lock.
+ * LOCKING: Image lock (param->image) must be held
  */
 static void
 set_anon_gparam_class (MonoGenericParam *param, gboolean is_mvar, MonoClass *klass)
 {
        int n = mono_generic_param_num (param) | ((guint32)param->serial << 16);
        MonoImage *image = param->image;
-       GHashTable *ht;
 
        g_assert (image);
 
        if (n < FAST_CACHE_SIZE) {
                if (is_mvar) {
-                       /* No locking needed */
+                       /* Requires locking to avoid droping an already published class */
                        if (!image->mvar_cache_fast)
                                image->mvar_cache_fast = mono_image_alloc0 (image, sizeof (MonoClass*) * FAST_CACHE_SIZE);
                        image->mvar_cache_fast [n] = klass;
@@ -6100,54 +6178,42 @@ set_anon_gparam_class (MonoGenericParam *param, gboolean is_mvar, MonoClass *kla
                                image->var_cache_fast = mono_image_alloc0 (image, sizeof (MonoClass*) * FAST_CACHE_SIZE);
                        image->var_cache_fast [n] = klass;
                }
-               return;
-       }
-       ht = is_mvar ? image->mvar_cache_slow : image->var_cache_slow;
-       if (!ht) {
-               mono_image_lock (image);
-               ht = is_mvar ? image->mvar_cache_slow : image->var_cache_slow;
+       } else {
+               GHashTable *ht = is_mvar ? image->mvar_cache_slow : image->var_cache_slow;
                if (!ht) {
-                       ht = g_hash_table_new (NULL, NULL);
-                       mono_memory_barrier ();
-                       if (is_mvar)
-                               image->mvar_cache_slow = ht;
-                       else
-                               image->var_cache_slow = ht;
+                       ht = is_mvar ? image->mvar_cache_slow : image->var_cache_slow;
+                       if (!ht) {
+                               ht = g_hash_table_new (NULL, NULL);
+                               mono_memory_barrier ();
+                               if (is_mvar)
+                                       image->mvar_cache_slow = ht;
+                               else
+                                       image->var_cache_slow = ht;
+                       }
                }
-               mono_image_unlock (image);
+               g_hash_table_insert (ht, GINT_TO_POINTER (n), klass);
        }
-
-       g_hash_table_insert (ht, GINT_TO_POINTER (n), klass);
 }
 
 /*
- * LOCKING: Acquires the loader lock.
+ * LOCKING: Acquires the image lock (@image).
  */
 MonoClass *
 mono_class_from_generic_parameter (MonoGenericParam *param, MonoImage *image, gboolean is_mvar)
 {
        MonoGenericContainer *container = mono_generic_param_owner (param);
-       MonoGenericParamInfo *pinfo;
-       MonoClass *klass;
-
-       mono_loader_lock ();
+       MonoGenericParamInfo *pinfo = NULL;
+       MonoClass *klass, *klass2;
 
        if (container) {
                pinfo = mono_generic_param_info (param);
-               if (pinfo->pklass) {
-                       mono_loader_unlock ();
-                       return pinfo->pklass;
-               }
+               klass = pinfo->pklass;
        } else {
-               pinfo = NULL;
                image = NULL;
-
-               klass = get_anon_gparam_class (param, is_mvar);
-               if (klass) {
-                       mono_loader_unlock ();
-                       return klass;
-               }
+               klass = get_anon_gparam_class (param, is_mvar, TRUE);
        }
+       if (klass)
+               return klass;
 
        if (!image && container) {
                if (is_mvar) {
@@ -6165,15 +6231,30 @@ mono_class_from_generic_parameter (MonoGenericParam *param, MonoImage *image, gb
 
        mono_memory_barrier ();
 
+       if (!image) //FIXME is this only needed by monodis? Can't we fix monodis instead of having this hack?
+               image = mono_defaults.corlib;
+
+       mono_image_lock (image);
        if (container)
-               pinfo->pklass = klass;
+               klass2 = pinfo->pklass;
        else
-               set_anon_gparam_class (param, is_mvar, klass);
+               klass2 = get_anon_gparam_class (param, is_mvar, FALSE);
 
-       mono_loader_unlock ();
+       if (klass2) {
+               klass = klass2;
+       } else {
+               if (container)
+                       pinfo->pklass = klass;
+               else
+                       set_anon_gparam_class (param, is_mvar, klass);
+       }
+       mono_image_unlock (image);
 
        /* FIXME: Should this go inside 'make_generic_param_klass'? */
-       mono_profiler_class_loaded (klass, MONO_PROFILE_OK);
+       if (klass2)
+               mono_profiler_class_loaded (klass2, MONO_PROFILE_FAILED);
+       else
+               mono_profiler_class_loaded (klass, MONO_PROFILE_OK);
 
        return klass;
 }
@@ -6189,15 +6270,15 @@ mono_ptr_class_get (MonoType *type)
        el_class = mono_class_from_mono_type (type);
        image = el_class->image;
 
-       mono_loader_lock ();
-
-       if (!image->ptr_cache)
-               image->ptr_cache = g_hash_table_new (mono_aligned_addr_hash, NULL);
-
-       if ((result = g_hash_table_lookup (image->ptr_cache, el_class))) {
-               mono_loader_unlock ();
-               return result;
+       mono_image_lock (image);
+       if (image->ptr_cache) {
+               if ((result = g_hash_table_lookup (image->ptr_cache, el_class))) {
+                       mono_image_unlock (image);
+                       return result;
+               }
        }
+       mono_image_unlock (image);
+       
        result = mono_image_alloc0 (image, sizeof (MonoClass));
 
        classes_size += sizeof (MonoClass);
@@ -6224,9 +6305,19 @@ mono_ptr_class_get (MonoType *type)
 
        mono_class_setup_supertypes (result);
 
+       mono_image_lock (image);
+       if (image->ptr_cache) {
+               MonoClass *result2;
+               if ((result2 = g_hash_table_lookup (image->ptr_cache, el_class))) {
+                       mono_image_unlock (image);
+                       mono_profiler_class_loaded (result, MONO_PROFILE_FAILED);
+                       return result2;
+               }
+       } else {
+               image->ptr_cache = g_hash_table_new (mono_aligned_addr_hash, NULL);
+       }
        g_hash_table_insert (image->ptr_cache, el_class, result);
-
-       mono_loader_unlock ();
+       mono_image_unlock (image);
 
        mono_profiler_class_loaded (result, MONO_PROFILE_OK);
 
@@ -6356,23 +6447,20 @@ mono_class_from_mono_type (MonoType *type)
 static MonoType *
 mono_type_retrieve_from_typespec (MonoImage *image, guint32 type_spec, MonoGenericContext *context, gboolean *did_inflate, MonoError *error)
 {
-       MonoType *t = mono_type_create_from_typespec (image, type_spec);
+       MonoType *t = mono_type_create_from_typespec_checked (image, type_spec, error);
 
-       mono_error_init (error);
        *did_inflate = FALSE;
 
-       if (!t) {
-               char *name = mono_class_name_from_token (image, type_spec);
-               char *assembly = mono_assembly_name_from_token (image, type_spec);
-               mono_error_set_type_load_name (error, name, assembly, "Could not resolve typespec token %08x", type_spec);
+       if (!t)
                return NULL;
-       }
 
        if (context && (context->class_inst || context->method_inst)) {
                MonoType *inflated = inflate_generic_type (NULL, t, context, error);
 
-               if (!mono_error_ok (error))
+               if (!mono_error_ok (error)) {
+                       g_assert (!mono_loader_get_last_error ());
                        return NULL;
+               }
 
                if (inflated) {
                        t = inflated;
@@ -7085,18 +7173,18 @@ mono_assembly_name_from_token (MonoImage *image, guint32 type_token)
                }
                mono_metadata_decode_row (t, idx-1, cols, MONO_TYPEREF_SIZE);
 
-               idx = cols [MONO_TYPEREF_SCOPE] >> MONO_RESOLTION_SCOPE_BITS;
-               switch (cols [MONO_TYPEREF_SCOPE] & MONO_RESOLTION_SCOPE_MASK) {
-               case MONO_RESOLTION_SCOPE_MODULE:
+               idx = cols [MONO_TYPEREF_SCOPE] >> MONO_RESOLUTION_SCOPE_BITS;
+               switch (cols [MONO_TYPEREF_SCOPE] & MONO_RESOLUTION_SCOPE_MASK) {
+               case MONO_RESOLUTION_SCOPE_MODULE:
                        /* FIXME: */
                        return g_strdup ("");
-               case MONO_RESOLTION_SCOPE_MODULEREF:
+               case MONO_RESOLUTION_SCOPE_MODULEREF:
                        /* FIXME: */
                        return g_strdup ("");
-               case MONO_RESOLTION_SCOPE_TYPEREF:
+               case MONO_RESOLUTION_SCOPE_TYPEREF:
                        /* FIXME: */
                        return g_strdup ("");
-               case MONO_RESOLTION_SCOPE_ASSEMBLYREF:
+               case MONO_RESOLUTION_SCOPE_ASSEMBLYREF:
                        mono_assembly_get_assemblyref (image, idx - 1, &aname);
                        return mono_stringify_assembly_name (&aname);
                default:
@@ -7129,15 +7217,31 @@ mono_class_get_full (MonoImage *image, guint32 type_token, MonoGenericContext *c
        MonoError error;
        MonoClass *class;
        class = mono_class_get_checked (image, type_token, &error);
-       g_assert (mono_error_ok (&error)); /*FIXME remove this function */
 
-       if (context && mono_metadata_token_table (type_token) == MONO_TABLE_TYPESPEC) {
+       if (class && context && mono_metadata_token_table (type_token) == MONO_TABLE_TYPESPEC)
                class = mono_class_inflate_generic_class_checked (class, context, &error);
-               g_assert (mono_error_ok (&error)); /*FIXME remove this function */
+
+       if (!class) {
+               mono_loader_set_error_from_mono_error (&error);
+               mono_error_cleanup (&error); /*FIXME don't swallow this error */
        }
        return class;
 }
 
+
+MonoClass *
+mono_class_get_and_inflate_typespec_checked (MonoImage *image, guint32 type_token, MonoGenericContext *context, MonoError *error)
+{
+       MonoClass *class;
+
+       mono_error_init (error);
+       class = mono_class_get_checked (image, type_token, error);
+
+       if (class && context && mono_metadata_token_table (type_token) == MONO_TABLE_TYPESPEC)
+               class = mono_class_inflate_generic_class_checked (class, context, error);
+
+       return class;
+}
 /**
  * mono_class_get_checked:
  * @image: the image where the class resides
@@ -7191,33 +7295,33 @@ done:
 
 
 /**
- * mono_type_get_full:
+ * mono_type_get_checked:
  * @image: the image where the type resides
  * @type_token: the token for the type
  * @context: the generic context used to evaluate generic instantiations in
+ * @error: Error handling context
  *
  * This functions exists to fullfill the fact that sometimes it's desirable to have access to the 
  * 
  * Returns: the MonoType that represents @type_token in @image
  */
 MonoType *
-mono_type_get_full (MonoImage *image, guint32 type_token, MonoGenericContext *context)
+mono_type_get_checked (MonoImage *image, guint32 type_token, MonoGenericContext *context, MonoError *error)
 {
-       MonoError error;
        MonoType *type = NULL;
        gboolean inflated = FALSE;
 
+       mono_error_init (error);
+
        //FIXME: this will not fix the very issue for which mono_type_get_full exists -but how to do it then?
        if (image_is_dynamic (image))
                return mono_class_get_type (mono_lookup_dynamic_token (image, type_token, context));
 
        if ((type_token & 0xff000000) != MONO_TOKEN_TYPE_SPEC) {
-               MonoClass *class = mono_class_get_checked (image, type_token, &error);
+               MonoClass *class = mono_class_get_checked (image, type_token, error);
 
-               if (!mono_error_ok (&error)) {
-                       mono_loader_set_error_from_mono_error (&error);
-                       /*FIXME don't swallow the error message*/
-                       mono_error_cleanup (&error);
+               if (!class) {
+                       g_assert (!mono_loader_get_last_error ());
                        return NULL;
                }
 
@@ -7225,17 +7329,10 @@ mono_type_get_full (MonoImage *image, guint32 type_token, MonoGenericContext *co
                return mono_class_get_type (class);
        }
 
-       type = mono_type_retrieve_from_typespec (image, type_token, context, &inflated, &error);
-
-       if (!mono_error_ok (&error)) {
-               char *name = mono_class_name_from_token (image, type_token);
-               char *assembly = mono_assembly_name_from_token (image, type_token);
-
-               g_warning ("Error loading type %s from %s due to %s", name, assembly, mono_error_get_message (&error));
+       type = mono_type_retrieve_from_typespec (image, type_token, context, &inflated, error);
 
-               mono_loader_set_error_type_load (name, assembly);
-               /*FIXME don't swallow the error message.*/
-               mono_error_cleanup (&error);
+       if (!type) {
+               g_assert (!mono_loader_get_last_error ());
                return NULL;
        }
 
@@ -7395,6 +7492,7 @@ find_nocase (gpointer key, gpointer value, gpointer user_data)
  * @image: The MonoImage where the type is looked up in
  * @name_space: the type namespace
  * @name: the type short name.
+ * @deprecated: use the _checked variant
  *
  * Obtains a MonoClass with a given namespace and a given name which
  * is located in the given MonoImage.   The namespace and name
@@ -7402,6 +7500,15 @@ find_nocase (gpointer key, gpointer value, gpointer user_data)
  */
 MonoClass *
 mono_class_from_name_case (MonoImage *image, const char* name_space, const char *name)
+{
+       MonoError error;
+       MonoClass *res = mono_class_from_name_case_checked (image, name_space, name, &error);
+       g_assert (!mono_error_ok (&error));
+       return res;
+}
+
+MonoClass *
+mono_class_from_name_case_checked (MonoImage *image, const char* name_space, const char *name, MonoError *error)
 {
        MonoTableInfo  *t = &image->tables [MONO_TABLE_TYPEDEF];
        guint32 cols [MONO_TYPEDEF_SIZE];
@@ -7409,6 +7516,8 @@ mono_class_from_name_case (MonoImage *image, const char* name_space, const char
        const char *nspace;
        guint32 i, visib;
 
+       mono_error_init (error);
+
        if (image_is_dynamic (image)) {
                guint32 token = 0;
                FindUserData user_data;
@@ -7437,7 +7546,7 @@ mono_class_from_name_case (MonoImage *image, const char* name_space, const char
                mono_image_unlock (image);
                
                if (token)
-                       return mono_class_get (image, MONO_TOKEN_TYPE_DEF | token);
+                       return mono_class_get_checked (image, MONO_TOKEN_TYPE_DEF | token, error);
                else
                        return NULL;
 
@@ -7456,7 +7565,7 @@ mono_class_from_name_case (MonoImage *image, const char* name_space, const char
                n = mono_metadata_string_heap (image, cols [MONO_TYPEDEF_NAME]);
                nspace = mono_metadata_string_heap (image, cols [MONO_TYPEDEF_NAMESPACE]);
                if (mono_utf8_strcasecmp (n, name) == 0 && mono_utf8_strcasecmp (nspace, name_space) == 0)
-                       return mono_class_get (image, MONO_TOKEN_TYPE_DEF | i);
+                       return mono_class_get_checked (image, MONO_TOKEN_TYPE_DEF | i, error);
        }
        return NULL;
 }
@@ -8330,26 +8439,34 @@ mono_ldtoken (MonoImage *image, guint32 token, MonoClass **handle_class,
        case MONO_TOKEN_TYPE_DEF:
        case MONO_TOKEN_TYPE_REF:
        case MONO_TOKEN_TYPE_SPEC: {
+               MonoError error;
                MonoType *type;
                if (handle_class)
                        *handle_class = mono_defaults.typehandle_class;
-               type = mono_type_get_full (image, token, context);
-               if (!type)
+               type = mono_type_get_checked (image, token, context, &error);
+               if (!type) {
+                       mono_loader_set_error_from_mono_error (&error);
+                       mono_error_cleanup (&error); /* FIXME Don't swallow the error */
                        return NULL;
+               }
                mono_class_init (mono_class_from_mono_type (type));
                /* We return a MonoType* as handle */
                return type;
        }
        case MONO_TOKEN_FIELD_DEF: {
                MonoClass *class;
+               MonoError error;
                guint32 type = mono_metadata_typedef_from_field (image, mono_metadata_token_index (token));
                if (!type)
                        return NULL;
                if (handle_class)
                        *handle_class = mono_defaults.fieldhandle_class;
-               class = mono_class_get_full (image, MONO_TOKEN_TYPE_DEF | type, context);
-               if (!class)
+               class = mono_class_get_and_inflate_typespec_checked (image, MONO_TOKEN_TYPE_DEF | type, context, &error);
+               if (!class) {
+                       mono_loader_set_error_from_mono_error (&error);
+                       mono_error_cleanup (&error); /* FIXME Don't swallow the error */
                        return NULL;
+               }
                mono_class_init (class);
                return mono_class_get_field (class, token);
        }
@@ -8368,11 +8485,16 @@ mono_ldtoken (MonoImage *image, guint32 token, MonoClass **handle_class,
                sig = mono_metadata_blob_heap (image, cols [MONO_MEMBERREF_SIGNATURE]);
                mono_metadata_decode_blob_size (sig, &sig);
                if (*sig == 0x6) { /* it's a field */
+                       MonoError error;
                        MonoClass *klass;
                        MonoClassField *field;
-                       field = mono_field_from_token (image, token, &klass, context);
+                       field = mono_field_from_token_checked (image, token, &klass, context, &error);
                        if (handle_class)
                                *handle_class = mono_defaults.fieldhandle_class;
+                       if (!field) {
+                               mono_loader_set_error_from_mono_error (&error);
+                               mono_error_cleanup (&error); /* FIXME Don't swallow the error */
+                       }
                        return field;
                } else {
                        MonoMethod *meth;