* test-9.il: New test, test instaniating a class
[mono.git] / mono / dis / get.c
index 033e64a5dfc8575ff82912489a6305b5fa0830a5..4c36d14351386f457c2b7faa25fd83cc4cb0069f 100644 (file)
 #include "meta.h"
 #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 (metadata_t *m, int idx)
+get_typedef (MonoImage *m, int idx)
 {
-       guint32 cols [6];
+       guint32 cols [MONO_TYPEDEF_SIZE];
+       const char *ns;
 
-       mono_metadata_decode_row (&m->tables [META_TABLE_TYPEDEF], idx - 1, cols, CSIZE (cols));
+       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 [2]),
-               mono_metadata_string_heap (m, cols [1]));
+               "%s%s%s", ns, *ns?".":"",
+               mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAME]));
 }
 
 char *
-get_module (metadata_t *m, int idx)
+get_module (MonoImage *m, int idx)
 {
-       guint32 cols [5];
+       guint32 cols [MONO_MODULE_SIZE];
        
        /*
         * There MUST BE only one module in the Module table
         */
        g_assert (idx == 1);
            
-       mono_metadata_decode_row (&m->tables [META_TABLE_MODULEREF], idx - 1, cols, CSIZE (cols));
+       mono_metadata_decode_row (&m->tables [MONO_TABLE_MODULE], idx - 1, cols, MONO_MODULE_SIZE);
 
-       return g_strdup (mono_metadata_string_heap (m, cols [6]));
+       return g_strdup (mono_metadata_string_heap (m, cols [MONO_MODULE_NAME]));
 }
 
 char *
-get_assemblyref (metadata_t *m, int idx)
+get_assemblyref (MonoImage *m, int idx)
 {
-       guint32 cols [9];
+       guint32 cols [MONO_ASSEMBLYREF_SIZE];
        
-       mono_metadata_decode_row (&m->tables [META_TABLE_ASSEMBLYREF], idx - 1, cols, CSIZE (cols));
+       mono_metadata_decode_row (&m->tables [MONO_TABLE_ASSEMBLYREF], idx - 1, cols, MONO_ASSEMBLYREF_SIZE);
 
-       return g_strdup (mono_metadata_string_heap (m, cols [6]));
+       return g_strdup (mono_metadata_string_heap (m, cols [MONO_ASSEMBLYREF_NAME]));
 }
 
 /*
@@ -59,7 +64,7 @@ get_assemblyref (metadata_t *m, int idx)
  * Returns a string representing the ArrayShape (22.2.16).
  */
 static const char *
-get_array_shape (metadata_t *m, const char *ptr, char **result)
+get_array_shape (MonoImage *m, const char *ptr, char **result)
 {
        GString *res = g_string_new ("[");
        guint32 rank, num_sizes, num_lo_bounds;
@@ -119,20 +124,20 @@ get_array_shape (metadata_t *m, const char *ptr, char **result)
  * Returns the stringified representation of a TypeSpec signature (22.2.17)
  */
 char *
-get_typespec (metadata_t *m, guint32 idx)
+get_typespec (MonoImage *m, guint32 idx)
 {
-       guint32 cols [1];
+       guint32 cols [MONO_TYPESPEC_SIZE];
        const char *ptr;
        char *s, *result;
        GString *res = g_string_new ("");
        int len;
 
-       mono_metadata_decode_row (&m->tables [META_TABLE_TYPESPEC], idx-1, cols, CSIZE (cols));
-       ptr = mono_metadata_blob_heap (m, cols [0]);
+       mono_metadata_decode_row (&m->tables [MONO_TABLE_TYPESPEC], idx-1, cols, MONO_TYPESPEC_SIZE);
+       ptr = mono_metadata_blob_heap (m, cols [MONO_TYPESPEC_SIGNATURE]);
        len = mono_metadata_decode_value (ptr, &ptr);
        
        switch (*ptr++){
-       case ELEMENT_TYPE_PTR:
+       case MONO_TYPE_PTR:
                ptr = get_custom_mod (m, ptr, &s);
                if (s){
                        g_string_append (res, s);
@@ -140,7 +145,7 @@ get_typespec (metadata_t *m, guint32 idx)
                        g_free (s);
                }
                
-               if (*ptr == ELEMENT_TYPE_VOID)
+               if (*ptr == MONO_TYPE_VOID)
                        g_string_append (res, "void");
                else {
                        ptr = get_type (m, ptr, &s);
@@ -149,7 +154,7 @@ get_typespec (metadata_t *m, guint32 idx)
                }
                break;
                
-       case ELEMENT_TYPE_FNPTR:
+       case MONO_TYPE_FNPTR:
                g_string_append (res, "FNPTR ");
                /*
                 * we assume MethodRefSig, as we do not know
@@ -160,7 +165,7 @@ get_typespec (metadata_t *m, guint32 idx)
                hex_dump (ptr, 0, 40);
                break;
                        
-       case ELEMENT_TYPE_ARRAY:
+       case MONO_TYPE_ARRAY:
                ptr = get_type (m, ptr, &s);
                g_string_append (res, s);
                g_free (s);
@@ -170,7 +175,7 @@ get_typespec (metadata_t *m, guint32 idx)
                g_free (s);
                break;
                
-       case ELEMENT_TYPE_SZARRAY:
+       case MONO_TYPE_SZARRAY:
                ptr = get_custom_mod (m, ptr, &s);
                if (s){
                        g_string_append (res, s);
@@ -190,47 +195,45 @@ get_typespec (metadata_t *m, guint32 idx)
 }
 
 char *
-get_typeref (metadata_t *m, int idx)
+get_typeref (MonoImage *m, int idx)
 {
-       guint32 cols [3];
+       guint32 cols [MONO_TYPEREF_SIZE];
        const char *s, *t;
        char *x, *ret;
        guint32 rs_idx, table;
        
-       mono_metadata_decode_row (&m->tables [META_TABLE_TYPEREF], idx - 1, cols, CSIZE (cols));
+       mono_metadata_decode_row (&m->tables [MONO_TABLE_TYPEREF], idx - 1, cols, MONO_TYPEREF_SIZE);
 
-       t = mono_metadata_string_heap (m, cols [1]);
-       s = mono_metadata_string_heap (m, cols [2]);
+       t = mono_metadata_string_heap (m, cols [MONO_TYPEREF_NAME]);
+       s = mono_metadata_string_heap (m, cols [MONO_TYPEREF_NAMESPACE]);
 
-       rs_idx = cols [0] >> 2;
-       /*
-        * Two bits in Beta2.
-        * ECMA spec claims 3 bits
-        */
-       table = cols [0] & 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:
@@ -252,7 +255,7 @@ get_typeref (metadata_t *m, int idx)
  * at (dor_token >> 2) 
  */
 char *
-get_typedef_or_ref (metadata_t *m, guint32 dor_token)
+get_typedef_or_ref (MonoImage *m, guint32 dor_token)
 {
        char *temp = NULL, *s;
        int table, idx;
@@ -279,7 +282,7 @@ get_typedef_or_ref (metadata_t *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);
 
        }
        
@@ -300,7 +303,7 @@ get_typedef_or_ref (metadata_t *m, guint32 dor_token)
  * Returns: the new ptr to continue decoding
  */
 const char *
-get_encoded_typedef_or_ref (metadata_t *m, const char *ptr, char **result)
+get_encoded_typedef_or_ref (MonoImage *m, const char *ptr, char **result)
 {
        guint32 token;
        
@@ -319,12 +322,12 @@ get_encoded_typedef_or_ref (metadata_t *m, const char *ptr, char **result)
  * Returns: updated pointer location
  */
 const char *
-get_custom_mod (metadata_t *m, const char *ptr, char **return_value)
+get_custom_mod (MonoImage *m, const char *ptr, char **return_value)
 {
        char *s;
        
-       if ((*ptr == ELEMENT_TYPE_CMOD_OPT) ||
-           (*ptr == ELEMENT_TYPE_CMOD_REQD)){
+       if ((*ptr == MONO_TYPE_CMOD_OPT) ||
+           (*ptr == MONO_TYPE_CMOD_REQD)){
                ptr++;
                ptr = get_encoded_typedef_or_ref (m, ptr, &s);
 
@@ -337,25 +340,25 @@ get_custom_mod (metadata_t *m, const char *ptr, char **return_value)
 
 
 static map_t element_type_map [] = {
-       { ELEMENT_TYPE_END        , "end" },
-       { ELEMENT_TYPE_VOID       , "void" },
-       { ELEMENT_TYPE_BOOLEAN    , "bool" },
-       { ELEMENT_TYPE_CHAR       , "char" }, 
-       { ELEMENT_TYPE_I1         , "sbyte" },
-       { ELEMENT_TYPE_U1         , "byte" }, 
-       { ELEMENT_TYPE_I2         , "int16" },
-       { ELEMENT_TYPE_U2         , "uint16" },
-       { ELEMENT_TYPE_I4         , "int32" },
-       { ELEMENT_TYPE_U4         , "uint32" },
-       { ELEMENT_TYPE_I8         , "int64" },
-       { ELEMENT_TYPE_U8         , "uint64" },
-       { ELEMENT_TYPE_R4         , "float32" },
-       { ELEMENT_TYPE_R8         , "float64" },
-       { ELEMENT_TYPE_STRING     , "string" },
-       { ELEMENT_TYPE_TYPEDBYREF , "TypedByRef" },
-       { ELEMENT_TYPE_I          , "native int" },
-       { ELEMENT_TYPE_U          , "native unsigned int" },
-       { ELEMENT_TYPE_OBJECT     , "object" },
+       { MONO_TYPE_END        , "end" },
+       { MONO_TYPE_VOID       , "void" },
+       { MONO_TYPE_BOOLEAN    , "bool" },
+       { MONO_TYPE_CHAR       , "char" }, 
+       { MONO_TYPE_I1         , "int8" },
+       { MONO_TYPE_U1         , "unsigned int8" }, 
+       { MONO_TYPE_I2         , "int16" },
+       { MONO_TYPE_U2         , "uint16" },
+       { MONO_TYPE_I4         , "int32" },
+       { MONO_TYPE_U4         , "uint32" },
+       { MONO_TYPE_I8         , "int64" },
+       { MONO_TYPE_U8         , "uint64" },
+       { MONO_TYPE_R4         , "float32" },
+       { MONO_TYPE_R8         , "float64" },
+       { MONO_TYPE_STRING     , "string" },
+       { MONO_TYPE_TYPEDBYREF , "TypedByRef" },
+       { MONO_TYPE_I          , "native int" },
+       { MONO_TYPE_U          , "native unsigned int" },
+       { MONO_TYPE_OBJECT     , "object" },
        { 0, NULL }
 };
 
@@ -370,13 +373,13 @@ static map_t call_conv_type_map [] = {
 };
 
 char*
-dis_stringify_token (metadata_t *m, guint32 token)
+dis_stringify_token (MonoImage *m, guint32 token)
 {
        guint idx = token & 0xffffff;
        switch (token >> 24) {
-       case META_TABLE_TYPEDEF: return get_typedef (m, idx);
-       case META_TABLE_TYPEREF: return get_typeref (m, idx);
-       case META_TABLE_TYPESPEC: return get_typespec (m, idx);
+       case MONO_TABLE_TYPEDEF: return get_typedef (m, idx);
+       case MONO_TABLE_TYPEREF: return get_typeref (m, idx);
+       case MONO_TABLE_TYPESPEC: return get_typespec (m, idx);
        default:
                 break;
        }
@@ -384,7 +387,7 @@ dis_stringify_token (metadata_t *m, guint32 token)
 }
 
 char*
-dis_stringify_array (metadata_t *m, MonoArray *array) 
+dis_stringify_array (MonoImage *m, MonoArrayType *array) 
 {
        char *type;
        GString *s = g_string_new("");
@@ -411,14 +414,14 @@ dis_stringify_array (metadata_t *m, MonoArray *array)
 }
 
 char*
-dis_stringify_modifiers (metadata_t *m, int n, MonoCustomMod *mod)
+dis_stringify_modifiers (MonoImage *m, int n, MonoCustomMod *mod)
 {
        GString *s = g_string_new("");
        char *result;
        int i;
        for (i = 0; i < n; ++i) {
                char *tok = dis_stringify_token (m, mod[i].token);
-               g_string_sprintfa (s, "%s %s", mod[i].mod == ELEMENT_TYPE_CMOD_OPT ? "opt": "reqd", tok);
+               g_string_sprintfa (s, "%s %s", mod[i].required ? "opt": "reqd", tok);
                g_free (tok);
        }
        g_string_append_c (s, ' ');
@@ -428,98 +431,172 @@ dis_stringify_modifiers (metadata_t *m, int n, MonoCustomMod *mod)
 }
 
 char*
-dis_stringify_param (metadata_t *m, MonoParam *param) 
+dis_stringify_param (MonoImage *m, MonoType *param) 
 {
-       char *mods = NULL;
        char *t;
        char *result;
-       if (param->num_modifiers)
-               mods = dis_stringify_modifiers (m, param->num_modifiers, param->modifiers);
-       if (param->typedbyref)
-               t = g_strdup ("TypedByRef");
-       else if (!param->type)
-               t = g_strdup ("void");
-       else
-               t = dis_stringify_type (m, param->type);
-       result = g_strjoin (mods ? mods : "", t, NULL);
+       const char *out = param->attrs & 2 ? "[out] ": "";
+       t = dis_stringify_type (m, param);
+       result = g_strconcat (out, t, NULL);
        g_free (t);
-       g_free (mods);
        return result;
 }
 
 char*
-dis_stringify_method_signature (metadata_t *m, MonoMethodSignature *method)
+dis_stringify_method_signature (MonoImage *m, MonoMethodSignature *method, int methoddef_row)
+{
+       guint32 cols [MONO_METHOD_SIZE];
+       guint32 pcols [MONO_PARAM_SIZE];
+       guint32 param_index = 0;
+       const char *name = "";
+       int free_method = 0;
+       char *retval;
+       GString *result = g_string_new ("");
+       int i;
+
+       g_assert (method || methoddef_row);
+
+       if (methoddef_row) {
+               mono_metadata_decode_row (&m->tables [MONO_TABLE_METHOD], methoddef_row -1, cols, MONO_METHOD_SIZE);
+               name = mono_metadata_string_heap (m, cols [MONO_METHOD_NAME]);
+               param_index = cols [MONO_METHOD_PARAMLIST];
+               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, methoddef_row, sig, &sig);
+                       free_method = 1;
+               }
+       }
+       
+       retval = dis_stringify_param (m, method->ret);
+       if (method->hasthis)
+               g_string_append (result, "instance ");
+       g_string_append (result, map (method->call_convention, call_conv_type_map));
+       g_string_sprintfa (result, " %s %s(", retval, name);
+       g_free (retval);
+       for (i = 0; i < method->param_count; ++i) {
+               if (param_index && param_index <= m->tables [MONO_TABLE_PARAM].rows) {
+                       mono_metadata_decode_row (&m->tables [MONO_TABLE_PARAM], param_index - 1, pcols, MONO_PARAM_SIZE);
+                       name = mono_metadata_string_heap (m, pcols [MONO_PARAM_NAME]);
+                       method->params [i]->attrs = pcols [MONO_PARAM_FLAGS];
+                       param_index++;
+               } else {
+                       name = "";
+               }
+               if (i)
+                       g_string_append (result, ", ");
+               retval = dis_stringify_param (m, method->params [i]);
+               g_string_sprintfa (result, "%s %s", retval, name);
+               g_free (retval);
+       }
+       g_string_append (result, ") ");
+
+       if (free_method)
+               mono_metadata_free_method_signature (method);
+       retval = result->str;
+       g_string_free (result, FALSE);
+
+       return retval;
+}
+
+static char *
+dis_stringify_object_with_class (MonoImage *m, MonoClass *c)
 {
-       return g_strdup ("method-signature");
+       /* 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) {
+               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);
+               }
+       }
+       result = g_strdup_printf ("%s %s%s%s%s", otype, assemblyref?assemblyref:"", c->name_space, 
+                               *c->name_space?".":"", c->name);
+       g_free (assemblyref);
+       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 (metadata_t *m, MonoType *type)
+dis_stringify_type (MonoImage *m, MonoType *type)
 {
-       char *bare = NULL;
-       char *byref;
+       const char *pinned = "", *byref = "";
+       char *bare = NULL, *mods = NULL;
        char *result;
 
-       if (!type)
-               return g_strdup ("void");
-
-       byref = type->byref ? "ref " : "";
+       if (type->num_mods)
+               mods = dis_stringify_modifiers (m, type->num_mods, type->modifiers);
 
        switch (type->type){
-       case ELEMENT_TYPE_BOOLEAN:
-       case ELEMENT_TYPE_CHAR:
-       case ELEMENT_TYPE_I1:
-       case ELEMENT_TYPE_U1:
-       case ELEMENT_TYPE_I2:
-       case ELEMENT_TYPE_U2:
-       case ELEMENT_TYPE_I4:
-       case ELEMENT_TYPE_U4:
-       case ELEMENT_TYPE_I8:
-       case ELEMENT_TYPE_U8:
-       case ELEMENT_TYPE_R4:
-       case ELEMENT_TYPE_R8:
-       case ELEMENT_TYPE_I:
-       case ELEMENT_TYPE_U:
-       case ELEMENT_TYPE_STRING:
-       case ELEMENT_TYPE_OBJECT:
-       case ELEMENT_TYPE_TYPEDBYREF:
+       case MONO_TYPE_BOOLEAN:
+       case MONO_TYPE_CHAR:
+       case MONO_TYPE_I1:
+       case MONO_TYPE_U1:
+       case MONO_TYPE_I2:
+       case MONO_TYPE_U2:
+       case MONO_TYPE_I4:
+       case MONO_TYPE_U4:
+       case MONO_TYPE_I8:
+       case MONO_TYPE_U8:
+       case MONO_TYPE_R4:
+       case MONO_TYPE_R8:
+       case MONO_TYPE_I:
+       case MONO_TYPE_U:
+       case MONO_TYPE_STRING:
+       case MONO_TYPE_OBJECT:
+       case MONO_TYPE_TYPEDBYREF:
                bare = g_strdup (map (type->type, element_type_map));
                break;
                
-       case ELEMENT_TYPE_VALUETYPE:
-       case ELEMENT_TYPE_CLASS:
-               bare = dis_stringify_token (m, type->data.token);
+       case MONO_TYPE_VALUETYPE:
+       case MONO_TYPE_CLASS:
+               bare = dis_stringify_object (m, type);
                break;
                
-       case ELEMENT_TYPE_FNPTR:
-               bare = dis_stringify_method_signature (m, type->data.method);
+       case MONO_TYPE_FNPTR:
+               bare = dis_stringify_method_signature (m, type->data.method, 0);
                break;
-       case ELEMENT_TYPE_PTR:
-       case ELEMENT_TYPE_SZARRAY: {
+       case MONO_TYPE_PTR:
+       case MONO_TYPE_SZARRAY: {
                char *child_type;
-               char *mods;
-               if (type->custom_mod) {
-                       mods = dis_stringify_modifiers (m, type->data.mtype->num_modifiers, type->data.mtype->modifiers);       
-                       child_type = dis_stringify_type (m, type->data.mtype->type);
-               } else {
-                       mods = g_strdup("");
-                       child_type = dis_stringify_type (m, type->data.type);
-               }
+               child_type = dis_stringify_type (m, type->data.type);
                
-               bare = g_strdup_printf (type->type == ELEMENT_TYPE_PTR ? "%s%s*" : "%s%s[]", mods, child_type);
+               bare = g_strdup_printf (type->type == MONO_TYPE_PTR ? "%s*" : "%s[]", child_type);
                g_free (child_type);
-               g_free (mods);
                break;
        }
-       case ELEMENT_TYPE_ARRAY:
+       case MONO_TYPE_ARRAY:
                bare = dis_stringify_array (m, type->data.array);
                break;
+       case MONO_TYPE_VOID:
+               bare = g_strdup ("void");
+               break;
        default:
                g_error ("Do not know how to stringify type 0x%x", type->type);
        }
+       
+       if (type->pinned)
+               pinned = " pinned";
+
+       if (type->byref)
+               byref = "&";
+               
+       result = g_strconcat (mods ? mods : "", bare, byref, pinned, NULL);
 
-       result = g_strjoin (byref, bare, NULL);
        g_free (bare);
+
        return result;
 }
 
@@ -535,9 +612,9 @@ dis_stringify_type (metadata_t *m, MonoType *type)
  * Returns: the new ptr to continue decoding
  */
 const char *
-get_type (metadata_t *m, const char *ptr, char **result)
+get_type (MonoImage *m, const char *ptr, char **result)
 {
-       MonoType *type = mono_metadata_parse_type (m, ptr, &ptr);
+       MonoType *type = mono_metadata_parse_type (m, MONO_PARSE_TYPE, 0, ptr, &ptr);
        *result = dis_stringify_type (m, type);
        mono_metadata_free_type (type);
        return ptr;
@@ -548,7 +625,7 @@ get_type (metadata_t *m, const char *ptr, char **result)
  * Returns a stringified representation of a FieldSig (22.2.4)
  */
 char *
-get_field_signature (metadata_t *m, guint32 blob_signature)
+get_field_signature (MonoImage *m, guint32 blob_signature)
 {
        char *allocated_modifier_string, *allocated_type_string;
        const char *ptr = mono_metadata_blob_heap (m, blob_signature);
@@ -574,13 +651,13 @@ get_field_signature (metadata_t *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;
 }
 
-ElementTypeEnum
-get_field_literal_type (metadata_t *m, guint32 blob_signature)
+MonoTypeEnum
+get_field_literal_type (MonoImage *m, guint32 blob_signature)
 {
        const char *ptr = mono_metadata_blob_heap (m, blob_signature);
        int len;
@@ -596,7 +673,7 @@ get_field_literal_type (metadata_t *m, guint32 blob_signature)
        if (allocated_modifier_string)
                g_free (allocated_modifier_string);
 
-       return (ElementTypeEnum) *ptr;
+       return (MonoTypeEnum) *ptr;
        
 }
 
@@ -608,7 +685,7 @@ get_field_literal_type (metadata_t *m, guint32 blob_signature)
  * decodes the literal indexed by @token.
  */
 char *
-decode_literal (metadata_t *m, guint32 token)
+decode_literal (MonoImage *m, guint32 token)
 {
        return g_strdup ("LITERAL_VALUE");
 }
@@ -624,7 +701,7 @@ decode_literal (metadata_t *m, guint32 token)
  * Returns: the new ptr to continue decoding.
  */
 const char *
-get_ret_type (metadata_t *m, const char *ptr, char **ret_type)
+get_ret_type (MonoImage *m, const char *ptr, char **ret_type)
 {
        GString *str = g_string_new ("");
        char *mod = NULL;
@@ -637,15 +714,15 @@ get_ret_type (metadata_t *m, const char *ptr, char **ret_type)
                g_free (mod);
        }
 
-       if (*ptr == ELEMENT_TYPE_TYPEDBYREF){
+       if (*ptr == MONO_TYPE_TYPEDBYREF){
                /* TODO: what does `typedbyref' mean? */
                g_string_append (str, "/* FIXME: What does this mean? */ typedbyref ");
                ptr++;
-       } else if (*ptr == ELEMENT_TYPE_VOID){
+       } else if (*ptr == MONO_TYPE_VOID){
                 g_string_append (str, "void");
                 ptr++;
        } else {
-               if (*ptr == ELEMENT_TYPE_BYREF){
+               if (*ptr == MONO_TYPE_BYREF){
                        g_string_append (str, "[out] ");
                        ptr++;
                }
@@ -672,7 +749,7 @@ get_ret_type (metadata_t *m, const char *ptr, char **ret_type)
  * Returns: the new ptr to continue decoding.
  */
 const char *
-get_param (metadata_t *m, const char *ptr, char **retval)
+get_param (MonoImage *m, const char *ptr, char **retval)
 {
        GString *str = g_string_new ("");
        char *allocated_mod_string, *allocated_type_string;
@@ -684,11 +761,11 @@ get_param (metadata_t *m, const char *ptr, char **retval)
                g_free (allocated_mod_string);
        }
        
-       if (*ptr == ELEMENT_TYPE_TYPEDBYREF){
+       if (*ptr == MONO_TYPE_TYPEDBYREF){
                g_string_append (str, "/*FIXME: what does typedbyref mean? */ typedbyref ");
                ptr++;
        } else {
-               if (*ptr == ELEMENT_TYPE_BYREF){
+               if (*ptr == MONO_TYPE_BYREF){
                        g_string_append (str, "[out] ");
                        ptr++;
                }
@@ -735,6 +812,10 @@ static map_t field_flags_map [] = {
        { FIELD_ATTRIBUTE_NOT_SERIALIZED,      "notserialized " },
        { FIELD_ATTRIBUTE_SPECIAL_NAME,        "specialname " },
        { FIELD_ATTRIBUTE_PINVOKE_IMPL,        "FIXME:pinvokeimpl " },
+       { FIELD_ATTRIBUTE_RT_SPECIAL_NAME,        "rtspecialname " },
+       { FIELD_ATTRIBUTE_HAS_FIELD_MARSHAL,        "hasfieldmarshal " },
+       { FIELD_ATTRIBUTE_HAS_DEFAULT,        "hasdefault " },
+       { FIELD_ATTRIBUTE_HAS_FIELD_RVA,        "hasfieldrva " },
        { 0, NULL }
 };
 
@@ -746,7 +827,7 @@ static map_t field_flags_map [] = {
 char *
 field_flags (guint32 f)
 {
-       static char buffer [1024];
+       char buffer [1024];
        int access = f & FIELD_ATTRIBUTE_FIELD_ACCESS_MASK;
        
        buffer [0] = 0;
@@ -760,7 +841,7 @@ field_flags (guint32 f)
  * Returns a stringifed representation of a MethodRefSig (22.2.2)
  */
 char *
-get_methodref_signature (metadata_t *m, guint32 blob_signature, const char *fancy_name)
+get_methodref_signature (MonoImage *m, guint32 blob_signature, const char *fancy_name)
 {
        GString *res = g_string_new ("");
        const char *ptr = mono_metadata_blob_heap (m, blob_signature);
@@ -792,7 +873,7 @@ get_methodref_signature (metadata_t *m, guint32 blob_signature, const char *fanc
                g_string_append (res, fancy_name);
        }
        
-       g_string_append (res, " (");
+       g_string_append (res, "(");
        
        /*
         * param_count describes parameters *before* and *after*
@@ -836,11 +917,10 @@ get_methodref_signature (metadata_t *m, guint32 blob_signature, const char *fanc
  * the TypeDef table and locate the actual "owner" of the field
  */
 char *
-get_field (metadata_t *m, guint32 token)
+get_field (MonoImage *m, guint32 token)
 {
        int idx = mono_metadata_token_index (token);
-       metadata_tableinfo_t *tdef = &m->tables [META_TABLE_TYPEDEF];
-       guint32 cols [3];
+       guint32 cols [MONO_FIELD_SIZE];
        char *sig, *res, *type;
        guint32 type_idx;
 
@@ -848,13 +928,13 @@ get_field (metadata_t *m, guint32 token)
         * We can get here also with a MenberRef token (for a field
         * defined in another module/assembly, just like in get_method ()
         */
-       if (mono_metadata_token_code (token) == TOKEN_TYPE_MEMBER_REF) {
+       if (mono_metadata_token_code (token) == MONO_TOKEN_MEMBER_REF) {
                return g_strdup_printf ("fieldref-0x%08x", token);
        }
-       g_assert (mono_metadata_token_code (token) == TOKEN_TYPE_FIELD_DEF);
+       g_assert (mono_metadata_token_code (token) == MONO_TOKEN_FIELD_DEF);
 
-       mono_metadata_decode_row (&m->tables [META_TABLE_FIELD], idx - 1, cols, CSIZE (cols));
-       sig = get_field_signature (m, cols [2]);
+       mono_metadata_decode_row (&m->tables [MONO_TABLE_FIELD], idx - 1, cols, MONO_FIELD_SIZE);
+       sig = get_field_signature (m, cols [MONO_FIELD_SIGNATURE]);
 
        /*
         * To locate the actual "container" for this field, we have to scan
@@ -863,9 +943,9 @@ get_field (metadata_t *m, guint32 token)
        type_idx = mono_metadata_typedef_from_field (m, idx);
 
        type = get_typedef (m, type_idx);
-       res = g_strdup_printf ("%s %s.%s",
+       res = g_strdup_printf ("%s %s::%s",
                               sig, type,
-                              mono_metadata_string_heap (m, cols [1]));
+                              mono_metadata_string_heap (m, cols [MONO_FIELD_NAME]));
        g_free (type);
        g_free (sig);
 
@@ -873,7 +953,7 @@ get_field (metadata_t *m, guint32 token)
 }
 
 static char *
-get_memberref_parent (metadata_t *m, guint32 mrp_token)
+get_memberref_parent (MonoImage *m, guint32 mrp_token)
 {
        /*
         * mrp_index is a MemberRefParent coded index
@@ -911,44 +991,44 @@ get_memberref_parent (metadata_t *m, guint32 mrp_token)
  * the TypeDef table and locate the actual "owner" of the field
  */
 char *
-get_method (metadata_t *m, guint32 token)
+get_method (MonoImage *m, guint32 token)
 {
        int idx = mono_metadata_token_index (token);
-       guint32 member_cols [3], method_cols [6];
-       char *res, *class, *fancy_name, *sig;
-       
-       switch (mono_metadata_token_code (token)){
-       case TOKEN_TYPE_METHOD_DEF:
+       guint32 member_cols [MONO_MEMBERREF_SIZE], method_cols [MONO_METHOD_SIZE];
+       char *sig;
+       const char *name;
 
-               mono_metadata_decode_row (&m->tables [META_TABLE_METHOD], 
-                                         idx - 1, method_cols, 6);
+       MonoMethod *mh;
 
-               fancy_name = mono_metadata_string_heap (m, method_cols [3]);
+       mh = mono_get_method (m, token, NULL);
+       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;
 
-               sig = get_methodref_signature (m, method_cols [4], fancy_name);
+       switch (mono_metadata_token_code (token)){
+       case MONO_TOKEN_METHOD_DEF:
+               mono_metadata_decode_row (&m->tables [MONO_TABLE_METHOD], 
+                                         idx - 1, method_cols, MONO_METHOD_SIZE);
 
+               sig = get_methodref_signature (m, method_cols [MONO_METHOD_SIGNATURE], name);
                return sig;
                
-       case TOKEN_TYPE_MEMBER_REF: {
-               
-               mono_metadata_decode_row (&m->tables [META_TABLE_MEMBERREF],
-                                         idx - 1, member_cols,
-                                         CSIZE (member_cols));
-               class = get_memberref_parent (m, member_cols [0]);
-               fancy_name = g_strconcat (
-                       class, "::",
-                       mono_metadata_string_heap (m, member_cols [1]),
-                       NULL);
+       case MONO_TOKEN_MEMBER_REF: {
                
+               mono_metadata_decode_row (&m->tables [MONO_TABLE_MEMBERREF],
+                                         idx - 1, member_cols, MONO_MEMBERREF_SIZE);
+               if (!name)
+                       name = g_strdup_printf ("%s::%s",
+                                       get_memberref_parent (m, member_cols [MONO_MEMBERREF_CLASS]),
+                                       mono_metadata_string_heap (m, member_cols [MONO_MEMBERREF_NAME]));
                sig = get_methodref_signature (
-                       m, member_cols [2], fancy_name);
-               g_free (fancy_name);
-
-               res = g_strdup_printf ("%s", sig);
-               g_free (sig);
-               return res;
+                       m, member_cols [MONO_MEMBERREF_SIGNATURE], name);
+               return sig;
        }
-               
+              
        default:
                g_assert_not_reached ();
        }
@@ -965,7 +1045,7 @@ get_method (metadata_t *m, guint32 token)
  * constant.
  */
 char *
-get_constant (metadata_t *m, ElementTypeEnum t, guint32 blob_index)
+get_constant (MonoImage *m, MonoTypeEnum t, guint32 blob_index)
 {
        const char *ptr = mono_metadata_blob_heap (m, blob_index);
        int len;
@@ -973,39 +1053,49 @@ get_constant (metadata_t *m, ElementTypeEnum t, guint32 blob_index)
        len = mono_metadata_decode_value (ptr, &ptr);
        
        switch (t){
-       case ELEMENT_TYPE_BOOLEAN:
+       case MONO_TYPE_BOOLEAN:
                return g_strdup_printf ("%s", *ptr ? "true" : "false");
                
-       case ELEMENT_TYPE_CHAR:
-               return g_strdup_printf ("%c", *ptr);
+       case MONO_TYPE_CHAR:
+               return g_strdup_printf ("%c", *ptr); /* FIXME: unicode char */
                
-       case ELEMENT_TYPE_U1:
-               return g_strdup_printf ("0x%02x", (int) (*ptr));
+       case MONO_TYPE_U1:
+       case MONO_TYPE_I1:
+               return g_strdup_printf ("int8(0x%02x)", (int) (*ptr));
                break;
                
-       case ELEMENT_TYPE_I2:
-               return g_strdup_printf ("%d", (int) (*(gint16 *) ptr));
-               
-       case ELEMENT_TYPE_I4:
-               return g_strdup_printf ("%d", *(gint32 *) ptr);
-               
-       case ELEMENT_TYPE_I8:
-               /*
-                * FIXME: This is not endian portable, does only 
-                * matter for debugging, but still.
-                */
-               return g_strdup_printf ("0x%08x%08x", *(guint32 *) ptr, *(guint32 *) (ptr + 4));
+       case MONO_TYPE_U2:
+       case MONO_TYPE_I2:
+               return g_strdup_printf ("int16(0x%08x)", (int) read16 (ptr));
                
-       case ELEMENT_TYPE_U8:
-               return g_strdup_printf ("0x%08x%08x", *(guint32 *) ptr, *(guint32 *) (ptr + 4));                
-       case ELEMENT_TYPE_R4:
-               return g_strdup_printf ("%g", (double) (* (float *) ptr));
+       case MONO_TYPE_U4:
+       case MONO_TYPE_I4:
+               return g_strdup_printf ("int32(%d)", read32 (ptr));
                
-       case ELEMENT_TYPE_R8:
-               return g_strdup_printf ("%g", * (double *) ptr);
-               
-       case ELEMENT_TYPE_STRING: {
-               int len, i, j, e;
+       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 i, j, e;
                char *res;
                e = len = 0;
                for (i = 0; !ptr [i+1]; i += 2){
@@ -1043,23 +1133,11 @@ get_constant (metadata_t *m, ElementTypeEnum t, guint32 blob_index)
                return res;
        }
                
-       case ELEMENT_TYPE_CLASS:
+       case MONO_TYPE_CLASS:
                return g_strdup ("CLASS CONSTANT.  MUST BE ZERO");
                
-               /*
-                * These are non CLS compliant:
-                */
-       case ELEMENT_TYPE_I1:
-               return g_strdup_printf ("%d", (int) *ptr);
-
-       case ELEMENT_TYPE_U2:
-               return g_strdup_printf ("0x%04x", (unsigned int) (*(guint16 *) ptr));
-               
-       case ELEMENT_TYPE_U4:
-               return g_strdup_printf ("0x%04x", (unsigned int) (*(guint32 *) ptr));
-               
        default:
-               g_error ("Unknown ELEMENT_TYPE (%d) on constant at Blob index (0x%08x)\n",
+               g_error ("Unknown MONO_TYPE (%d) on constant at Blob index (0x%08x)\n",
                         (int) *ptr, blob_index);
                return g_strdup_printf ("Unknown");
        }
@@ -1075,13 +1153,20 @@ get_constant (metadata_t *m, ElementTypeEnum t, guint32 blob_index)
  * constant.
  */
 char *
-get_token (metadata_t *m, guint32 token)
+get_token (MonoImage *m, guint32 token)
 {
+       guint32 idx = mono_metadata_token_index (token);
+
        switch (mono_metadata_token_code (token)){
-       case TOKEN_TYPE_FIELD_DEF:
+       case MONO_TOKEN_FIELD_DEF:
                return (get_field (m, token));
-               
-       default:
+       case MONO_TOKEN_TYPE_DEF:
+               return get_typedef (m, idx);
+       case MONO_TOKEN_TYPE_REF:
+               return get_typeref (m, idx);
+       case MONO_TOKEN_TYPE_SPEC:
+               return get_typespec (m, idx);
+       default:                
                g_error ("Do not know how to decode tokens of type 0x%08x", token);
        }
 
@@ -1093,13 +1178,13 @@ get_token (metadata_t *m, guint32 token)
  * get_token_type:
  * @m: metadata context
  * @token: the token can belong to any of the following tables:
- * TOKEN_TYPE_TYPE_REF, TOKEN_TYPE_TYPE_DEF, TOKEN_TYPE_TYPE_SPEC
+ * MONO_TOKEN_TYPE_REF, MONO_TOKEN_TYPE_DEF, MONO_TOKEN_TYPE_SPEC
  *
  * Returns: a stringified version of the MethodDef or MethodRef or TypeSpecn
  * at (token & 0xffffff) 
  */
 char *
-get_token_type (metadata_t *m, guint32 token)
+get_token_type (MonoImage *m, guint32 token)
 {
        char *temp = NULL, *s;
        int idx;
@@ -1107,22 +1192,22 @@ get_token_type (metadata_t *m, guint32 token)
        idx = mono_metadata_token_index (token);
        
        switch (mono_metadata_token_code (token)){
-       case TOKEN_TYPE_TYPE_DEF:
+       case MONO_TOKEN_TYPE_DEF:
                temp = get_typedef (m, idx);
                s = g_strdup_printf ("%s", temp);
                break;
                
-       case TOKEN_TYPE_TYPE_REF: 
+       case MONO_TOKEN_TYPE_REF: 
                temp = get_typeref (m, idx);
                s = g_strdup_printf ("%s", temp);
                break;
                
-       case TOKEN_TYPE_TYPE_SPEC:
+       case MONO_TOKEN_TYPE_SPEC:
                s = get_typespec (m, idx);
                break;
 
        default:
-               g_error ("Unhandled encoding for typedef-or-ref coded index");
+               g_error ("Unhandled encoding for token 0x%08x", token);
 
        }
        
@@ -1131,3 +1216,176 @@ get_token_type (metadata_t *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");
+       }
+}
+