2009-11-18 Sebastien Pouliot <sebastien@ximian.com>
[mono.git] / mono / metadata / class.c
index 91bae5b79c6bf1faa1fd08307c2306beced7ef2a..717f93c7b8ce1558c4944335b1099534e26f6e34 100644 (file)
@@ -40,6 +40,7 @@
 #include <mono/metadata/mono-debug.h>
 #include <mono/utils/mono-counters.h>
 #include <mono/utils/mono-string.h>
+#include <mono/utils/mono-error-internals.h>
 
 MonoStats mono_stats;
 
@@ -492,8 +493,10 @@ mono_class_is_open_constructed_type (MonoType *t)
 }
 
 static MonoType*
-inflate_generic_type (MonoImage *image, MonoType *type, MonoGenericContext *context)
+inflate_generic_type (MonoImage *image, MonoType *type, MonoGenericContext *context, MonoError *error)
 {
+       mono_error_init (error);
+
        switch (type->type) {
        case MONO_TYPE_MVAR: {
                MonoType *nt;
@@ -503,8 +506,9 @@ inflate_generic_type (MonoImage *image, MonoType *type, MonoGenericContext *cont
                        return NULL;
                if (num >= inst->type_argc) {
                        MonoGenericParamInfo *info = mono_generic_param_info (type->data.generic_param);
-                       g_error ("MVAR %d (%s) cannot be expanded in this context with %d instantiations",
+                       mono_error_set_bad_image (error, image->module_name, "MVAR %d (%s) cannot be expanded in this context with %d instantiations",
                                num, info ? info->name : "", inst->type_argc);
+                       return NULL;
                }
 
                /*
@@ -525,8 +529,9 @@ inflate_generic_type (MonoImage *image, MonoType *type, MonoGenericContext *cont
                        return NULL;
                if (num >= inst->type_argc) {
                        MonoGenericParamInfo *info = mono_generic_param_info (type->data.generic_param);
-                       g_error ("VAR %d (%s) cannot be expanded in this context with %d instantiations",
+                       mono_error_set_bad_image (error, image->module_name, "VAR %d (%s) cannot be expanded in this context with %d instantiations",
                                num, info ? info->name : "", inst->type_argc);
+                       return NULL;
                }
                nt = mono_metadata_type_dup (image, inst->type_argv [num]);
                nt->byref = type->byref;
@@ -535,8 +540,8 @@ inflate_generic_type (MonoImage *image, MonoType *type, MonoGenericContext *cont
        }
        case MONO_TYPE_SZARRAY: {
                MonoClass *eclass = type->data.klass;
-               MonoType *nt, *inflated = inflate_generic_type (NULL, &eclass->byval_arg, context);
-               if (!inflated)
+               MonoType *nt, *inflated = inflate_generic_type (NULL, &eclass->byval_arg, context, error);
+               if (!inflated || !mono_error_ok (error))
                        return NULL;
                nt = mono_metadata_type_dup (image, type);
                nt->data.klass = mono_class_from_mono_type (inflated);
@@ -545,8 +550,8 @@ inflate_generic_type (MonoImage *image, MonoType *type, MonoGenericContext *cont
        }
        case MONO_TYPE_ARRAY: {
                MonoClass *eclass = type->data.array->eklass;
-               MonoType *nt, *inflated = inflate_generic_type (NULL, &eclass->byval_arg, context);
-               if (!inflated)
+               MonoType *nt, *inflated = inflate_generic_type (NULL, &eclass->byval_arg, context, error);
+               if (!inflated || !mono_error_ok (error))
                        return NULL;
                nt = mono_metadata_type_dup (image, type);
                nt->data.array = g_memdup (nt->data.array, sizeof (MonoArrayType));
@@ -644,6 +649,7 @@ mono_class_get_generic_class (MonoClass *klass)
  * @mempool: a mempool
  * @type: a type
  * @context: a generics context
+ * @error: error context
  *
  * The same as mono_class_inflate_generic_type, but allocates the MonoType
  * from mempool if it is non-NULL.  If it is NULL, the MonoType is
@@ -652,12 +658,15 @@ mono_class_get_generic_class (MonoClass *klass)
  * modified by the caller, and it should be freed using mono_metadata_free_type ().
  */
 MonoType*
-mono_class_inflate_generic_type_with_mempool (MonoImage *image, MonoType *type, MonoGenericContext *context)
+mono_class_inflate_generic_type_with_mempool (MonoImage *image, MonoType *type, MonoGenericContext *context, MonoError *error)
 {
-       MonoType *inflated = NULL; 
+       MonoType *inflated = NULL;
+       mono_error_init (error);
 
        if (context)
-               inflated = inflate_generic_type (image, type, context);
+               inflated = inflate_generic_type (image, type, context, error);
+       if (!mono_error_ok (error))
+               return NULL;
 
        if (!inflated) {
                MonoType *shared = mono_metadata_get_shared_type (type);
@@ -682,12 +691,40 @@ mono_class_inflate_generic_type_with_mempool (MonoImage *image, MonoType *type,
  * generics context @context.
  *
  * Returns: the instantiated type or a copy of @type. The returned MonoType is allocated
- * on the heap and is owned by the caller.
+ * on the heap and is owned by the caller. Returns NULL on error.
+ *
+ * @deprecated Please use mono_class_inflate_generic_type_checked instead
  */
 MonoType*
 mono_class_inflate_generic_type (MonoType *type, MonoGenericContext *context)
 {
-       return mono_class_inflate_generic_type_with_mempool (NULL, type, context);
+       MonoError error;
+       MonoType *result;
+       result = mono_class_inflate_generic_type_checked (type, context, &error);
+
+       if (!mono_error_ok (&error)) {
+               mono_error_cleanup (&error);
+               return NULL;
+       }
+       return result;
+}
+
+/*
+ * mono_class_inflate_generic_type:
+ * @type: a type
+ * @context: a generics context
+ * @error: error context to use
+ *
+ * If @type is a generic type and @context is not NULL, instantiate it using the 
+ * generics context @context.
+ *
+ * Returns: the instantiated type or a copy of @type. The returned MonoType is allocated
+ * on the heap and is owned by the caller.
+ */
+MonoType*
+mono_class_inflate_generic_type_checked (MonoType *type, MonoGenericContext *context, MonoError *error)
+{
+       return mono_class_inflate_generic_type_with_mempool (NULL, type, context, error);
 }
 
 /*
@@ -699,10 +736,13 @@ mono_class_inflate_generic_type (MonoType *type, MonoGenericContext *context)
 static MonoType*
 mono_class_inflate_generic_type_no_copy (MonoImage *image, MonoType *type, MonoGenericContext *context)
 {
+       MonoError error;
        MonoType *inflated = NULL; 
 
-       if (context)
-               inflated = inflate_generic_type (image, type, context);
+       if (context) {
+               inflated = inflate_generic_type (image, type, context, &error);
+               g_assert (mono_error_ok (&error)); /*FIXME proper error handling*/
+       }
 
        if (!inflated)
                return type;
@@ -719,10 +759,12 @@ mono_class_inflate_generic_type_no_copy (MonoImage *image, MonoType *type, MonoG
 MonoClass*
 mono_class_inflate_generic_class (MonoClass *gklass, MonoGenericContext *context)
 {
+       MonoError error;
        MonoClass *res;
        MonoType *inflated;
 
-       inflated = mono_class_inflate_generic_type (&gklass->byval_arg, context);
+       inflated = mono_class_inflate_generic_type_checked (&gklass->byval_arg, context, &error);
+       g_assert (mono_error_ok (&error)); /*FIXME proper error handling*/
 
        res = mono_class_from_mono_type (inflated);
        mono_metadata_free_type (inflated);
@@ -892,7 +934,11 @@ mono_class_inflate_generic_method_full (MonoMethod *method, MonoClass *klass_hin
                result->klass = klass_hint;
 
        if (!result->klass) {
-               MonoType *inflated = inflate_generic_type (NULL, &method->klass->byval_arg, context);
+               MonoError error;
+               MonoType *inflated = inflate_generic_type (NULL, &method->klass->byval_arg, context, &error);
+
+               g_assert (mono_error_ok (&error)); /*FIXME proper error handling*/
+
                result->klass = inflated ? mono_class_from_mono_type (inflated) : method->klass;
                if (inflated)
                        mono_metadata_free_type (inflated);
@@ -1106,8 +1152,13 @@ mono_class_setup_fields (MonoClass *class)
        if (class->parent) {
                /* For generic instances, class->parent might not have been initialized */
                mono_class_init (class->parent);
-               if (!class->parent->size_inited)
+               if (!class->parent->size_inited) {
                        mono_class_setup_fields (class->parent);
+                       if (class->parent->exception_type) {
+                               mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, NULL);
+                               return;
+                       }
+               }
                class->instance_size += class->parent->instance_size;
                class->min_align = class->parent->min_align;
                /* we use |= since it may have been set already */
@@ -1217,8 +1268,13 @@ mono_class_setup_fields (MonoClass *class)
                                blittable = FALSE;
                        } else {
                                MonoClass *field_class = mono_class_from_mono_type (field->type);
-                               if (field_class)
+                               if (field_class) {
                                        mono_class_setup_fields (field_class);
+                                       if (field_class->exception_type) {
+                                               mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, NULL);
+                                               break;
+                                       }
+                               }
                                if (!field_class || !field_class->blittable)
                                        blittable = FALSE;
                        }
@@ -1698,9 +1754,14 @@ mono_class_setup_methods (MonoClass *class)
                }
        }
 
-       if (MONO_CLASS_IS_INTERFACE (class))
-               for (i = 0; i < class->method.count; ++i)
-                       methods [i]->slot = i;
+       if (MONO_CLASS_IS_INTERFACE (class)) {
+               int slot = 0;
+               /*Only assign slots to virtual methods as interfaces are allowed to have static methods.*/
+               for (i = 0; i < class->method.count; ++i) {
+                       if (methods [i]->flags & METHOD_ATTRIBUTE_VIRTUAL)
+                               methods [i]->slot = slot++;
+               }
+       }
 
        /* Needed because of the double-checking locking pattern */
        mono_memory_barrier ();
@@ -3238,7 +3299,10 @@ mono_class_setup_vtable_general (MonoClass *class, MonoMethod **overrides, int o
                for (i = 0; i < parent->interface_offsets_count; i++) {
                        MonoClass *parent_interface = parent->interfaces_packed [i];
                        int interface_offset = mono_class_interface_offset (class, parent_interface);
-                       
+                       /*FIXME this is now dead code as this condition will never hold true.
+                       Since interface offsets are inherited then the offset of an interface implemented
+                       by a parent will never be the out of it's vtable boundary.
+                       */
                        if (interface_offset >= parent->vtable_size) {
                                int parent_interface_offset = mono_class_interface_offset (parent, parent_interface);
                                int j;
@@ -3537,7 +3601,7 @@ mono_class_setup_vtable_general (MonoClass *class, MonoMethod **overrides, int o
                printf ("VTable %s (vtable entries = %d, interfaces = %d)\n", mono_type_full_name (&class->byval_arg), 
                        class->vtable_size, icount); 
 
-               for (i = 0; i < class->vtable_size; ++i) {
+               for (i = 0; i < cur_slot; ++i) {
                        MonoMethod *cm;
               
                        cm = vtable [i];
@@ -3556,15 +3620,15 @@ mono_class_setup_vtable_general (MonoClass *class, MonoMethod **overrides, int o
                                ic = class->interfaces [i];
                                printf ("  slot offset: %03d, method count: %03d, iid: %03d %s\n",  
                                        mono_class_interface_offset (class, ic),
-                                       ic->method.count, ic->interface_id, mono_type_full_name (&ic->byval_arg));
+                                       count_virtual_methods (ic), ic->interface_id, mono_type_full_name (&ic->byval_arg));
                        }
 
                        for (k = class->parent; k ; k = k->parent) {
                                for (i = 0; i < k->interface_count; i++) {
                                        ic = k->interfaces [i]; 
-                                       printf ("  slot offset: %03d, method count: %03d, iid: %03d %s\n",  
+                                       printf ("  parent slot offset: %03d, method count: %03d, iid: %03d %s\n",  
                                                mono_class_interface_offset (class, ic),
-                                               ic->method.count, ic->interface_id, mono_type_full_name (&ic->byval_arg));
+                                               count_virtual_methods (ic), ic->interface_id, mono_type_full_name (&ic->byval_arg));
                                }
                        }
                }
@@ -4219,6 +4283,30 @@ mono_class_setup_mono_type (MonoClass *class)
 
 }
 
+/*
+ * COM initialization (using mono_init_com_types) is delayed until needed. 
+ * However when a [ComImport] attribute is present on a type it will trigger
+ * the initialization. This is not a problem unless the BCL being executed 
+ * lacks the types that COM depends on (e.g. Variant on Silverlight).
+ */
+static void
+init_com_from_comimport (MonoClass *class)
+{
+       /* we don't always allow COM initialization under the CoreCLR (e.g. Moonlight does not require it) */
+       if ((mono_security_get_mode () == MONO_SECURITY_MODE_CORE_CLR)) {
+               /* but some other CoreCLR user could requires it for their platform (i.e. trusted) code */
+               if (!mono_security_core_clr_determine_platform_image (class->image)) {
+                       /* but it can not be made available for application (i.e. user code) since all COM calls
+                        * are considered native calls. In this case we fail with a TypeLoadException (just like
+                        * Silverlight 2 does */
+                       mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, NULL);
+                       return;
+               }
+       }
+       /* FIXME : we should add an extra checks to ensure COM can be initialized properly before continuing */
+       mono_init_com_types ();
+}
+
 /*
  * LOCKING: this assumes the loader lock is held
  */
@@ -4245,7 +4333,7 @@ mono_class_setup_parent (MonoClass *class, MonoClass *parent)
        if (!MONO_CLASS_IS_INTERFACE (class)) {
                /* Imported COM Objects always derive from __ComObject. */
                if (MONO_CLASS_IS_IMPORT (class)) {
-                       mono_init_com_types ();
+                       init_com_from_comimport (class);
                        if (parent == mono_defaults.object_class)
                                parent = mono_defaults.com_object_class;
                }
@@ -4296,7 +4384,7 @@ mono_class_setup_parent (MonoClass *class, MonoClass *parent)
        } else {
                /* initialize com types if COM interfaces are present */
                if (MONO_CLASS_IS_IMPORT (class))
-                       mono_init_com_types ();
+                       init_com_from_comimport (class);
                class->parent = NULL;
        }
 
@@ -4436,7 +4524,7 @@ mono_class_create_from_typedef (MonoImage *image, guint32 type_token)
        if ((class->flags & TYPE_ATTRIBUTE_STRING_FORMAT_MASK) == TYPE_ATTRIBUTE_UNICODE_CLASS)
                class->unicode = 1;
 
-#if PLATFORM_WIN32
+#ifdef PLATFORM_WIN32
        if ((class->flags & TYPE_ATTRIBUTE_STRING_FORMAT_MASK) == TYPE_ATTRIBUTE_AUTO_CLASS)
                class->unicode = 1;
 #endif
@@ -4983,11 +5071,14 @@ 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);
        if (!t)
                return NULL;
        if (context && (context->class_inst || context->method_inst)) {
-               MonoType *inflated = inflate_generic_type (NULL, t, context);
+               MonoType *inflated = inflate_generic_type (NULL, t, context, &error);
+               g_assert (mono_error_ok (&error)); /*FIXME proper error handling*/
+
                if (inflated) {
                        t = inflated;
                        *did_inflate = TRUE;
@@ -5130,6 +5221,9 @@ mono_bounded_array_class_get (MonoClass *eclass, guint32 rank, gboolean bounded)
                mono_class_init (eclass);
        if (!eclass->size_inited)
                mono_class_setup_fields (eclass);
+       if (eclass->exception_type) /*FIXME we fail the array type, but we have to let other fields be set.*/
+               mono_class_set_failure (class, MONO_EXCEPTION_TYPE_LOAD, NULL);
+
        class->has_references = MONO_TYPE_IS_REFERENCE (&eclass->byval_arg) || eclass->has_references? TRUE: FALSE;
 
        class->rank = rank;
@@ -5283,6 +5377,8 @@ static MonoClassField *
 mono_class_get_field_idx (MonoClass *class, int idx)
 {
        mono_class_setup_fields_locking (class);
+       if (class->exception_type)
+               return NULL;
 
        while (class) {
                if (class->image->uncompressed_metadata) {
@@ -5363,6 +5459,9 @@ mono_class_get_field_from_name_full (MonoClass *klass, const char *name, MonoTyp
        int i;
 
        mono_class_setup_fields_locking (klass);
+       if (klass->exception_type)
+               return NULL;
+
        while (klass) {
                for (i = 0; i < klass->field.count; ++i) {
                        MonoClassField *field = &klass->fields [i];
@@ -5398,6 +5497,9 @@ mono_class_get_field_token (MonoClassField *field)
        int i;
 
        mono_class_setup_fields_locking (klass);
+       if (klass->exception_type)
+               return 0;
+
        while (klass) {
                for (i = 0; i < klass->field.count; ++i) {
                        if (&klass->fields [i] == field) {
@@ -5808,6 +5910,7 @@ mono_image_add_to_name_cache (MonoImage *image, const char *nspace,
 {
        GHashTable *nspace_table;
        GHashTable *name_cache;
+       guint32 old_index;
 
        mono_image_lock (image);
 
@@ -5819,6 +5922,10 @@ mono_image_add_to_name_cache (MonoImage *image, const char *nspace,
                nspace_table = g_hash_table_new (g_str_hash, g_str_equal);
                g_hash_table_insert (name_cache, (char *)nspace, (char *)nspace_table);
        }
+
+       if ((old_index = GPOINTER_TO_UINT (g_hash_table_lookup (nspace_table, (char*) name))))
+               g_error ("overrwritting old token %x on image %s for type %s::%s", old_index, image->name, nspace, name);
+
        g_hash_table_insert (nspace_table, (char *) name, GUINT_TO_POINTER (index));
 
        mono_image_unlock (image);
@@ -7704,6 +7811,14 @@ can_access_internals (MonoAssembly *accessing, MonoAssembly* accessed)
                return TRUE;
        if (!accessed || !accessing)
                return FALSE;
+
+       /* extra safety under CoreCLR - the runtime does not verify the strongname signatures
+        * anywhere so untrusted friends are not safe to access platform's code internals */
+       if (mono_security_get_mode () == MONO_SECURITY_MODE_CORE_CLR) {
+               if (!mono_security_core_clr_can_access_internals (accessing->image, accessed->image))
+                       return FALSE;
+       }
+
        mono_assembly_load_friends (accessed);
        for (tmp = accessed->friend_assembly_names; tmp; tmp = tmp->next) {
                MonoAssemblyName *friend = tmp->data;