* test-9.il: New test, test instaniating a class
[mono.git] / mono / dis / get.c
index 72b38040e28d589f626bf001abc62ee72f251fe9..4c36d14351386f457c2b7faa25fd83cc4cb0069f 100644 (file)
 #include "util.h"
 #include "get.h"
 #include <mono/metadata/class.h>
+#include <mono/metadata/marshal.h>
+
+extern gboolean substitute_with_mscorlib_p;
 
 char *
 get_typedef (MonoImage *m, int idx)
 {
        guint32 cols [MONO_TYPEDEF_SIZE];
+       const char *ns;
 
        mono_metadata_decode_row (&m->tables [MONO_TABLE_TYPEDEF], idx - 1, cols, MONO_TYPEDEF_SIZE);
 
+       ns = mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAMESPACE]);
        return g_strdup_printf (
-               "%s.%s",
-               mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAMESPACE]),
+               "%s%s%s", ns, *ns?".":"",
                mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAME]));
 }
 
@@ -40,7 +44,7 @@ get_module (MonoImage *m, int idx)
         */
        g_assert (idx == 1);
            
-       mono_metadata_decode_row (&m->tables [MONO_TABLE_MODULEREF], idx - 1, cols, MONO_MODULE_SIZE);
+       mono_metadata_decode_row (&m->tables [MONO_TABLE_MODULE], idx - 1, cols, MONO_MODULE_SIZE);
 
        return g_strdup (mono_metadata_string_heap (m, cols [MONO_MODULE_NAME]));
 }
@@ -203,35 +207,33 @@ get_typeref (MonoImage *m, int idx)
        t = mono_metadata_string_heap (m, cols [MONO_TYPEREF_NAME]);
        s = mono_metadata_string_heap (m, cols [MONO_TYPEREF_NAMESPACE]);
 
-       rs_idx = cols [MONO_TYPEREF_SCOPE] >> 2;
-       /*
-        * Two bits in Beta2.
-        * ECMA spec claims 3 bits
-        */
-       table = cols [MONO_TYPEREF_SCOPE] & 3;
+       rs_idx = cols [MONO_TYPEREF_SCOPE] >> RESOLTION_SCOPE_BITS;
+       table = cols [MONO_TYPEREF_SCOPE] & RESOLTION_SCOPE_MASK;
        
        switch (table){
-       case 0: /* Module */
+       case RESOLTION_SCOPE_MODULE: /* Module */
                x = get_module (m, rs_idx);
-               ret = g_strdup_printf ("TODO:TypeRef-Module [%s] %s.%s", x, s, t);
+               ret = g_strdup_printf ("[%s] %s%s%s", x, s, *s?".":"", t);
                g_free (x);
                break;
 
-       case 1: /* ModuleRef */
+       case RESOLTION_SCOPE_MODULEREF: /* ModuleRef */
                ret = g_strdup_printf ("TODO:TypeRef-ModuleRef (%s.%s)", s, t);
                break;
                              
-       case 2: /*
+       case RESOLTION_SCOPE_ASSEMBLYREF: /*
                 * AssemblyRef (ECMA docs claim it is 3, but it looks to
                 * me like it is 2 (tokens are prefixed with 0x23)
                 */
                x = get_assemblyref (m, rs_idx);
-               ret = g_strdup_printf ("[%s]%s.%s", x, s, t);
+               ret = g_strdup_printf ("[%s]%s%s%s", x, s, *s?".":"", t);
                g_free (x);
                break;
                
-       case 4: /* TypeRef */
-               ret =  g_strdup_printf ("TODO:TypeRef-TypeRef: TYPEREF! (%s.%s)", s, t);
+       case RESOLTION_SCOPE_TYPEREF: /* TypeRef */
+               x = get_typeref (m, rs_idx);
+               ret =  g_strdup_printf ("%s/%s", x, t);
+               g_free (x);
                break;
                
        default:
@@ -280,7 +282,7 @@ get_typedef_or_ref (MonoImage *m, guint32 dor_token)
                break;
 
        default:
-               g_error ("Unhandled encoding for typedef-or-ref coded index");
+               g_error ("Unhandled encoding for typedef-or-ref coded index 0x%08x", dor_token);
 
        }
        
@@ -433,7 +435,7 @@ dis_stringify_param (MonoImage *m, MonoType *param)
 {
        char *t;
        char *result;
-       char *out = param->attrs & 2 ? "[out] ": "";
+       const char *out = param->attrs & 2 ? "[out] ": "";
        t = dis_stringify_type (m, param);
        result = g_strconcat (out, t, NULL);
        g_free (t);
@@ -461,7 +463,7 @@ dis_stringify_method_signature (MonoImage *m, MonoMethodSignature *method, int m
                if (!method) {
                        const char *sig = mono_metadata_blob_heap (m, cols [MONO_METHOD_SIGNATURE]);
                        mono_metadata_decode_blob_size (sig, &sig);
-                       method = mono_metadata_parse_method_signature (m, 1, sig, &sig);
+                       method = mono_metadata_parse_method_signature (m, methoddef_row, sig, &sig);
                        free_method = 1;
                }
        }
@@ -498,31 +500,40 @@ dis_stringify_method_signature (MonoImage *m, MonoMethodSignature *method, int m
 }
 
 static char *
-dis_stringify_object (MonoImage *m, MonoType *type)
+dis_stringify_object_with_class (MonoImage *m, MonoClass *c)
 {
-       char *otype = type->type == MONO_TYPE_CLASS? "class" : "valuetype";
-       char *assemblyref = NULL;
-       MonoClass *c = type->data.klass;
-       if (!c)
-               return g_strdup ("Unknown");
+       /* FIXME: handle MONO_TYPE_OBJECT ... */
+       const char *otype = c->byval_arg.type == MONO_TYPE_CLASS? "class" : "valuetype";
+       char *assemblyref = NULL, *result;
        if (m != c->image) {
-               /* we cheat */
-               if (!strcmp ("corlib", c->image->assembly_name))
-                       assemblyref = g_strdup_printf ("[%s]", "mscorlib");
-               else
-                       assemblyref = g_strdup_printf ("[%s]", c->image->assembly_name);
+               if (c->image->assembly_name) {
+                       /* we cheat */
+                       if (substitute_with_mscorlib_p && !strcmp ("corlib", c->image->assembly_name))
+                               assemblyref = g_strdup_printf ("[%s]", "mscorlib");
+                       else
+                               assemblyref = g_strdup_printf ("[%s]", c->image->assembly->aname.name);
+               } else {
+                       assemblyref = g_strdup_printf ("[.module %s]", c->image->module_name);
+               }
        }
-       otype = g_strdup_printf ("%s %s%s%s%s", otype, assemblyref?assemblyref:"", c->name_space, 
+       result = g_strdup_printf ("%s %s%s%s%s", otype, assemblyref?assemblyref:"", c->name_space, 
                                *c->name_space?".":"", c->name);
        g_free (assemblyref);
-       return otype;
+       return result;
+}
+
+static char *
+dis_stringify_object (MonoImage *m, MonoType *type)
+{
+       MonoClass *c = mono_class_from_mono_type (type);
+       return dis_stringify_object_with_class (m, c);
 }
 
 char*
 dis_stringify_type (MonoImage *m, MonoType *type)
 {
-       char *bare = NULL, *pinned = "", *byref = "";
-       char *mods = NULL;
+       const char *pinned = "", *byref = "";
+       char *bare = NULL, *mods = NULL;
        char *result;
 
        if (type->num_mods)
@@ -640,7 +651,7 @@ get_field_signature (MonoImage *m, guint32 blob_signature)
        if (allocated_modifier_string)
                g_free (allocated_modifier_string);
        if (allocated_type_string)
-               g_free (allocated_modifier_string);
+               g_free (allocated_type_string);
        
        return res;
 }
@@ -990,10 +1001,11 @@ get_method (MonoImage *m, guint32 token)
        MonoMethod *mh;
 
        mh = mono_get_method (m, token, NULL);
-       if (mh)
-               name = g_strdup_printf ("%s.%s::%s", mh->klass->name_space, 
-                               mh->klass->name, mh->name); 
-       else
+       if (mh) {
+               sig = dis_stringify_object_with_class (m, mh->klass);
+               name = g_strdup_printf ("%s::%s", sig, mh->name);
+               g_free (sig);
+       } else
                name = NULL;
 
        switch (mono_metadata_token_code (token)){
@@ -1060,24 +1072,30 @@ get_constant (MonoImage *m, MonoTypeEnum t, guint32 blob_index)
        case MONO_TYPE_I4:
                return g_strdup_printf ("int32(%d)", read32 (ptr));
                
-       case MONO_TYPE_I8:
-               /*
-                * FIXME: This is not endian portable, does only 
-                * matter for debugging, but still.
-                * Need also to fix unaligned access.
-                */
-               return g_strdup_printf ("0x%08x%08x", *(guint32 *) ptr, *(guint32 *) (ptr + 4));
-               
-       case MONO_TYPE_U8:
-               return g_strdup_printf ("0x%08x%08x", *(guint32 *) ptr, *(guint32 *) (ptr + 4));                
-       case MONO_TYPE_R4:
-               return g_strdup_printf ("%g", (double) (* (float *) ptr));
-               
-       case MONO_TYPE_R8:
-               return g_strdup_printf ("%g", * (double *) ptr);
-               
+       case MONO_TYPE_I8: {
+               guint32 low, high;
+               low = read32 (ptr);
+               high = read32 (ptr + 4);
+               return g_strdup_printf ("0x%08x%08x", high, low);
+       }
+       case MONO_TYPE_U8: {
+               guint32 low, high;
+               low = read32 (ptr);
+               high = read32 (ptr + 4);
+               return g_strdup_printf ("0x%08x%08x", high, low);
+       }
+       case MONO_TYPE_R4: {
+               float r;
+               readr4 (ptr, &r);
+               return g_strdup_printf ("%g", (double) r);
+       }
+       case MONO_TYPE_R8: {
+               double r;
+               readr8 (ptr, &r);
+               return g_strdup_printf ("%g", r);
+       }
        case MONO_TYPE_STRING: {
-               int len, i, j, e;
+               int i, j, e;
                char *res;
                e = len = 0;
                for (i = 0; !ptr [i+1]; i += 2){
@@ -1189,7 +1207,7 @@ get_token_type (MonoImage *m, guint32 token)
                break;
 
        default:
-               g_error ("Unhandled encoding for typedef-or-ref coded index");
+               g_error ("Unhandled encoding for token 0x%08x", token);
 
        }
        
@@ -1198,3 +1216,176 @@ get_token_type (MonoImage *m, guint32 token)
 
        return s;
 }
+
+char *
+get_guid (MonoImage *m, guint32 guid_index)
+{
+       const unsigned char *guid;
+       char *result;
+
+       guid = mono_metadata_guid_heap (m, guid_index);
+
+       result = g_strdup_printf ("{%02X%02X%02X%02X-%02X%02X-%02X%02X-%02X%02X-%02X%02X%02X%02X%02X%02X}", 
+                       guid [3], guid [2], guid [1], guid [0], guid [5], guid [4], guid [7], guid [6],
+                       guid [8], guid [9], guid [10], guid [11], guid [12], guid [13], guid [14], guid [15]);
+       return result;
+}
+
+GList *
+dis_get_custom_attrs (MonoImage *m, guint32 token)
+{
+       GList *list = NULL;
+       guint32 idx, i, len, mtoken;
+       guint32 cols [MONO_CUSTOM_ATTR_SIZE];
+       MonoTableInfo *ca;
+       char *method;
+       GString *attr;
+       const char *val;
+
+       idx = mono_metadata_token_index (token);
+       idx <<= CUSTOM_ATTR_BITS;
+       
+       switch (mono_metadata_token_table (token)) {
+       case MONO_TABLE_TYPEDEF:
+               idx |= CUSTOM_ATTR_TYPEDEF;
+               break;
+       case MONO_TABLE_ASSEMBLY:
+               idx |= CUSTOM_ATTR_ASSEMBLY;
+               break;
+       case MONO_TABLE_PROPERTY:
+               idx |= CUSTOM_ATTR_PROPERTY;
+               break;
+       case MONO_TABLE_EVENT:
+               idx |= CUSTOM_ATTR_EVENT;
+               break;
+       case MONO_TABLE_FIELD:
+               idx |= CUSTOM_ATTR_FIELDDEF;
+               break;
+       case MONO_TABLE_METHOD:
+               idx |= CUSTOM_ATTR_METHODDEF;
+               break;
+       case MONO_TABLE_PARAM:
+               idx |= CUSTOM_ATTR_PARAMDEF;
+               break;
+       default:
+               g_print ("Missing custom attr get support for token 0x%08x\n", token);
+               return NULL;
+       }
+
+       ca = &m->tables [MONO_TABLE_CUSTOMATTRIBUTE];
+       /* the table is not sorted */
+       for (i = 0; i < ca->rows; ++i) {
+               char *dump;
+               mono_metadata_decode_row (ca, i, cols, MONO_CUSTOM_ATTR_SIZE);
+               if (cols [MONO_CUSTOM_ATTR_PARENT] != idx)
+                       continue;
+               mtoken = cols [MONO_CUSTOM_ATTR_TYPE] >> CUSTOM_ATTR_TYPE_BITS;
+               switch (cols [MONO_CUSTOM_ATTR_TYPE] & CUSTOM_ATTR_TYPE_MASK) {
+               case CUSTOM_ATTR_TYPE_METHODDEF:
+                       mtoken |= MONO_TOKEN_METHOD_DEF;
+                       break;
+               case CUSTOM_ATTR_TYPE_MEMBERREF:
+                       mtoken |= MONO_TOKEN_MEMBER_REF;
+                       break;
+               default:
+                       g_error ("Unknown table for custom attr type %08x", cols [MONO_CUSTOM_ATTR_TYPE]);
+                       break;
+               }
+               method = get_method (m, mtoken);
+               val = mono_metadata_blob_heap (m, cols [MONO_CUSTOM_ATTR_VALUE]);
+               len = mono_metadata_decode_value (val, &val);
+               attr = g_string_new (".custom ");
+               dump = data_dump (val, len, "\t\t");
+               g_string_sprintfa (attr, "%s = %s", method, dump);
+               g_free (dump);
+               list = g_list_append (list, attr->str);
+               g_string_free (attr, FALSE);
+               g_free (method);
+       }
+       return list;
+}
+
+char*
+get_marshal_info (MonoImage *m, const char *blob) {
+       int len, size = 0;
+
+       len = mono_metadata_decode_blob_size (blob, &blob);
+
+       switch (*blob) {
+       case MONO_NATIVE_BOOLEAN:
+               return g_strdup ("bool");
+       case MONO_NATIVE_I1:
+               return g_strdup ("int8");
+       case MONO_NATIVE_U1:
+               return g_strdup ("unsigned int8");
+       case MONO_NATIVE_I2:
+               return g_strdup ("int16");
+       case MONO_NATIVE_U2:
+               return g_strdup ("unsigned int16");
+       case MONO_NATIVE_I4:
+               return g_strdup ("int32");
+       case MONO_NATIVE_U4:
+               return g_strdup ("unsigned int32");
+       case MONO_NATIVE_I8:
+               return g_strdup ("int64");
+       case MONO_NATIVE_U8:
+               return g_strdup ("unsigned int64");
+       case MONO_NATIVE_R4:
+               return g_strdup ("float32");
+       case MONO_NATIVE_R8:
+               return g_strdup ("float64");
+       case MONO_NATIVE_CURRENCY:
+               return g_strdup ("currency");
+       case MONO_NATIVE_BSTR:
+               return g_strdup ("bstr");
+       case MONO_NATIVE_LPSTR:
+               return g_strdup ("lpstr");
+       case MONO_NATIVE_LPWSTR:
+               return g_strdup ("lpwstr");
+       case MONO_NATIVE_LPTSTR:
+               return g_strdup ("lptstr");
+       case MONO_NATIVE_BYVALTSTR:
+               size = mono_metadata_decode_value (blob + 1, &blob);
+               return g_strdup_printf ("fixed sysstring [%d]", size);
+       case MONO_NATIVE_IUNKNOWN:
+               return g_strdup ("iunknown");
+       case MONO_NATIVE_IDISPATCH:
+               return g_strdup ("idispatch");
+       case MONO_NATIVE_STRUCT:
+               return g_strdup ("struct");
+       case MONO_NATIVE_INTERFACE:
+               return g_strdup ("interface");
+       case MONO_NATIVE_SAFEARRAY:
+               return g_strdup ("safearray");
+       case MONO_NATIVE_BYVALARRAY:
+               size = mono_metadata_decode_value (blob + 1, &blob);
+               return g_strdup_printf ("fixed array [%d]", size);
+       case MONO_NATIVE_INT:
+               return g_strdup ("int");
+       case MONO_NATIVE_UINT:
+               return g_strdup ("unsigned int");
+       case MONO_NATIVE_VBBYREFSTR:
+               return g_strdup ("vbbyrefstr");
+       case MONO_NATIVE_ANSIBSTR:
+               return g_strdup ("ansi bstr");
+       case MONO_NATIVE_TBSTR:
+               return g_strdup ("tbstr");
+       case MONO_NATIVE_VARIANTBOOL:
+               return g_strdup ("variant bool");
+       case MONO_NATIVE_FUNC:
+               return g_strdup ("method");
+       case MONO_NATIVE_ASANY:
+               return g_strdup ("as any");
+       case MONO_NATIVE_LPARRAY:
+               return g_strdup ("[]");
+       case MONO_NATIVE_LPSTRUCT:
+               return g_strdup ("lpstruct");
+       case MONO_NATIVE_CUSTOM:
+               return g_strdup ("custom");
+       case MONO_NATIVE_ERROR:
+               return g_strdup ("error");
+       default:
+               return g_strdup ("unknown");
+       }
+}
+