2010-03-12 Jb Evain <jbevain@novell.com>
[mono.git] / mono / dis / get.c
1 /*
2  * get.c: Functions to get stringified values from the metadata tables.
3  *
4  * Author:
5  *   Miguel de Icaza (miguel@ximian.com)
6  *
7  * (C) 2001 Ximian, Inc.
8  */
9 #include <config.h>
10 #include <stdio.h>
11 #include <stdlib.h>
12 #include <string.h>
13 #include <ctype.h>
14 #include <glib.h>
15 #include <math.h>
16 #include "meta.h"
17 #include "util.h"
18 #include "get.h"
19 #include <mono/utils/mono-compiler.h>
20 #include <mono/metadata/class.h>
21 #include <mono/metadata/marshal.h>
22
23 extern gboolean substitute_with_mscorlib_p;
24
25 static MonoGenericContainer *
26 get_memberref_container (MonoImage *m, guint32 mrp_token, MonoGenericContainer *container);
27
28 static char *
29 get_memberref_parent (MonoImage *m, guint32 mrp_token, MonoGenericContainer *container);
30  
31 static gboolean
32 cant_print_generic_param_name (MonoGenericParam *gparam);
33
34 GHashTable *key_table = NULL;
35 GHashTable *mono_generic_params_with_ambiguous_names = NULL;
36 GHashTable *generic_containers = NULL;
37 gboolean show_method_tokens = FALSE;
38 gboolean show_tokens = FALSE;
39
40 char *
41 get_typedef (MonoImage *m, int idx)
42 {
43         guint32 cols [MONO_TYPEDEF_SIZE];
44         const char *ns;
45         char *tstring, *result;
46         guint32 token;
47         
48         if (idx == 1)
49                 /* <Module> */
50                 return NULL;
51
52         mono_metadata_decode_row (&m->tables [MONO_TABLE_TYPEDEF], idx - 1, cols, MONO_TYPEDEF_SIZE);
53
54         ns = mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAMESPACE]);
55
56         /* Check if this is a nested type */
57         token = MONO_TOKEN_TYPE_DEF | (idx);
58         token = mono_metadata_nested_in_typedef (m, token);
59         tstring = show_tokens ? g_strdup_printf ("/*%08x*/", token) : NULL;
60         if (token) {
61                 char *outer;
62                 
63                 outer = get_typedef (m, mono_metadata_token_index (token));
64                 result = g_strdup_printf (
65                         "%s/%s%s", outer,
66                         mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAME]),
67                         tstring ? tstring : "");
68                 g_free (outer);
69                 g_free (tstring);
70                 return result;
71         }
72         
73         
74         result = g_strdup_printf (
75                 "%s%s%s%s", ns, *ns?".":"",
76                 mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAME]),
77                 tstring ? tstring : "");
78         g_free (tstring);
79
80         return result;
81 }
82
83 char *
84 get_module (MonoImage *m, int idx)
85 {
86         guint32 cols [MONO_MODULE_SIZE];
87         
88         /*
89          * There MUST BE only one module in the Module table
90          */
91         g_assert (idx == 1);
92             
93         mono_metadata_decode_row (&m->tables [MONO_TABLE_MODULE], idx - 1, cols, MONO_MODULE_SIZE);
94
95         return get_escaped_name (mono_metadata_string_heap (m, cols [MONO_MODULE_NAME]));
96 }
97
98 char *
99 get_moduleref (MonoImage *m, int idx)
100 {
101         guint32 cols [MONO_MODULEREF_SIZE];
102         
103         mono_metadata_decode_row (&m->tables [MONO_TABLE_MODULEREF], idx - 1, cols, MONO_MODULEREF_SIZE);
104
105         return get_escaped_name (mono_metadata_string_heap (m, cols [MONO_MODULEREF_NAME]));
106 }
107
108 char *
109 get_assemblyref (MonoImage *m, int idx)
110 {
111         guint32 cols [MONO_ASSEMBLYREF_SIZE];
112         
113         mono_metadata_decode_row (&m->tables [MONO_TABLE_ASSEMBLYREF], idx - 1, cols, MONO_ASSEMBLYREF_SIZE);
114
115         return get_escaped_name (mono_metadata_string_heap (m, cols [MONO_ASSEMBLYREF_NAME]));
116 }
117
118 static char *
119 stringify_array (guint32 rank, guint32 num_sizes, guint32 num_lo_bounds, gint32 *sizes, gint32 *lo_bounds)
120 {
121         GString *res = g_string_new ("[");
122         int i;
123
124         for (i = 0; i < rank; i++) {
125                 if (i)
126                         g_string_append_c (res, ',');
127                 if (i < num_lo_bounds)
128                         g_string_append_printf (res, "%d...", lo_bounds [i]);
129                 if (i < num_sizes) {
130                         if (i < num_lo_bounds)
131                                 g_string_append_printf (res, "%d", lo_bounds [i] + sizes [i] - 1);
132                         else
133                                 g_string_append_printf (res, "%d", sizes [i]);
134                 }
135
136         }
137         g_string_append (res, "]");
138
139         return g_string_free (res, FALSE);
140 }
141
142 /*
143  *
144  * Returns a string representing the ArrayShape (22.2.16).
145  */
146 static const char *
147 get_array_shape (MonoImage *m, const char *ptr, char **result)
148 {
149         guint32 rank, num_sizes, num_lo_bounds;
150         gint32 *sizes = NULL, *lo_bounds = NULL;
151         int i;
152         
153         rank = mono_metadata_decode_value (ptr, &ptr);
154         num_sizes = mono_metadata_decode_value (ptr, &ptr);
155
156         if (num_sizes > 0)
157                 sizes = g_new (gint32, num_sizes);
158         
159         for (i = 0; i < num_sizes; i++)
160                 sizes [i] = mono_metadata_decode_value (ptr, &ptr);
161
162         num_lo_bounds = mono_metadata_decode_value (ptr, &ptr);
163         if (num_lo_bounds > 0)
164                 lo_bounds = g_new (gint32, num_lo_bounds);
165         
166         for (i = 0; i < num_lo_bounds; i++)
167                 lo_bounds [i] = mono_metadata_decode_signed_value (ptr, &ptr);
168
169         *result = stringify_array (rank, num_sizes, num_lo_bounds, sizes, lo_bounds);
170
171         g_free (sizes);
172         g_free (lo_bounds);
173
174         return ptr;
175 }
176
177 /**
178  * get_typespec:
179  * @m: metadata context
180  * @blob_idx: index into the blob heap
181  *
182  * Returns the stringified representation of a TypeSpec signature (22.2.17)
183  */
184 char *
185 get_typespec (MonoImage *m, guint32 idx, gboolean is_def, MonoGenericContainer *container)
186 {
187         guint32 cols [MONO_TYPESPEC_SIZE];
188         const char *ptr;
189         char *s, *result;
190         GString *res = g_string_new ("");
191         int len;
192         MonoMethodSignature *sig;
193
194         mono_metadata_decode_row (&m->tables [MONO_TABLE_TYPESPEC], idx-1, cols, MONO_TYPESPEC_SIZE);
195         ptr = mono_metadata_blob_heap (m, cols [MONO_TYPESPEC_SIGNATURE]);
196         len = mono_metadata_decode_value (ptr, &ptr);
197
198         switch (*ptr++){
199         case MONO_TYPE_PTR:
200                 ptr = get_custom_mod (m, ptr, &s);
201                 if (s){
202                         g_string_append (res, s);
203                         g_string_append_c (res, ' ');
204                         g_free (s);
205                 }
206
207                 if (*ptr == MONO_TYPE_VOID)
208                         g_string_append (res, "void");
209                 else {
210                         ptr = get_type (m, ptr, &s, is_def, container);
211                         if (s)
212                                 g_string_append (res, s);
213                 }
214                 g_string_append (res, "*");
215                 break;
216
217         case MONO_TYPE_FNPTR:
218                 sig = mono_metadata_parse_method_signature_full (m, container, 0, ptr, &ptr);
219                 s = dis_stringify_function_ptr (m, sig);
220                 g_string_append (res, "method ");
221                 g_string_append (res, s);
222                 g_free (s);
223                 break;
224
225         case MONO_TYPE_ARRAY:
226                 ptr = get_type (m, ptr, &s, is_def, container);
227                 g_string_append (res, s);
228                 g_free (s);
229                 g_string_append_c (res, ' ');
230                 ptr = get_array_shape (m, ptr, &s);
231                 g_string_append (res, s);
232                 g_free (s);
233                 break;
234
235         case MONO_TYPE_SZARRAY:
236                 ptr = get_custom_mod (m, ptr, &s);
237                 if (s){
238                         g_string_append (res, s);
239                         g_string_append_c (res, ' ');
240                         g_free (s);
241                 }
242                 ptr = get_type (m, ptr, &s, is_def, container);
243                 g_string_append (res, s);
244                 g_string_append (res, "[]");
245                 g_free (s);
246                 break;
247
248         default:
249                 ptr = get_type (m, ptr - 1, &s, is_def, container);
250                 g_string_append (res, s);
251                 g_free (s);
252                 break;
253         }
254
255         if (show_tokens) {
256                 int token = mono_metadata_make_token (MONO_TABLE_TYPESPEC, idx);
257                 result = g_strdup_printf ("%s/*%08x*/", res->str, token);
258         } else
259                 result = res->str;
260
261         g_string_free (res, FALSE);
262
263         return result;
264 }
265
266 char *
267 get_typeref (MonoImage *m, int idx)
268 {
269         guint32 cols [MONO_TYPEREF_SIZE];
270         const char *s, *t;
271         char *x, *ret;
272         guint32 rs_idx, table;
273         
274         mono_metadata_decode_row (&m->tables [MONO_TABLE_TYPEREF], idx - 1, cols, MONO_TYPEREF_SIZE);
275
276         t = mono_metadata_string_heap (m, cols [MONO_TYPEREF_NAME]);
277         s = mono_metadata_string_heap (m, cols [MONO_TYPEREF_NAMESPACE]);
278
279         rs_idx = cols [MONO_TYPEREF_SCOPE] >> MONO_RESOLTION_SCOPE_BITS;
280         table = cols [MONO_TYPEREF_SCOPE] & MONO_RESOLTION_SCOPE_MASK;
281         
282         switch (table){
283         case MONO_RESOLTION_SCOPE_MODULE: /* Module */
284                 x = get_module (m, rs_idx);
285                 ret = g_strdup_printf ("[%s] %s%s%s", x, s, *s?".":"", t);
286                 g_free (x);
287                 break;
288
289         case MONO_RESOLTION_SCOPE_MODULEREF: /* ModuleRef */
290                 x = get_moduleref (m, rs_idx);
291                 ret = g_strdup_printf ("[.module %s]%s%s%s", x, s, *s ? "." : "", t);
292                 break;
293                               
294         case MONO_RESOLTION_SCOPE_ASSEMBLYREF: /*
295                  * AssemblyRef (ECMA docs claim it is 3, but it looks to
296                  * me like it is 2 (tokens are prefixed with 0x23)
297                  */
298                 x = get_assemblyref (m, rs_idx);
299                 ret = g_strdup_printf ("[%s]%s%s%s", x, s, *s?".":"", t);
300                 g_free (x);
301                 break;
302                 
303         case MONO_RESOLTION_SCOPE_TYPEREF: /* TypeRef */
304                 x = get_typeref (m, rs_idx);
305                 ret =  g_strdup_printf ("%s/%s", x, t);
306                 g_free (x);
307                 break;
308                 
309         default:
310                 ret = g_strdup_printf ("Unknown table in TypeRef %d", table);
311         }
312
313         if (show_tokens) {
314                 int token = mono_metadata_make_token (MONO_TABLE_TYPEREF, idx);
315                 char *temp = g_strdup_printf ("%s/*%08x*/", ret, token);
316                 g_free (ret);
317                 ret = temp;
318         }
319
320         return ret;
321 }
322
323 /**
324  * get_typedef_or_ref:
325  * @m: metadata context
326  * @dor_token: def or ref encoded index
327  *
328  * Low two bits contain table to lookup from
329  * high bits contain the index into the def or ref table
330  *
331  * Returns: a stringified version of the MethodDef or MethodRef
332  * at (dor_token >> 2) 
333  */
334 char *
335 get_typedef_or_ref (MonoImage *m, guint32 dor_token, MonoGenericContainer *container)
336 {
337         char *temp = NULL, *s = NULL;
338         int table, idx;
339
340         /*
341          * low 2 bits contain encoding
342          */
343         table = dor_token & MONO_TYPEDEFORREF_MASK;
344         idx = dor_token >> MONO_TYPEDEFORREF_BITS;
345         
346         switch (table){
347         case 0: /* TypeDef */
348                 temp = get_typedef (m, idx);
349                 s = g_strdup_printf ("%s", temp);
350                 break;
351                 
352         case 1: /* TypeRef */
353                 temp = get_typeref (m, idx);
354                 s = g_strdup_printf ("%s", temp);
355                 break;
356                 
357         case 2: /* TypeSpec */
358                 s = get_typespec (m, idx, FALSE, container);
359                 break;
360
361         default:
362                 g_error ("Unhandled encoding for typedef-or-ref coded index 0x%08x", dor_token);
363
364         }
365         
366         if (temp)
367                 g_free (temp);
368
369         return s;
370 }
371
372 /**
373  * get_type_or_methdef
374  * @m: metadata context
375  * @dor_token: type or method def encoded index
376  *
377  * Low bit contains the table to lookup from
378  * high bits contain the index into the type def or method def table
379  *
380  * Returns: a stringified version of the TypeOrMethodDef token
381  */
382 char *
383 get_type_or_methdef (MonoImage *m, guint32 dor_token)
384 {
385         if (dor_token & MONO_TYPEORMETHOD_METHOD) /* MethodDef */
386                 return get_methoddef (m, dor_token >> MONO_TYPEORMETHOD_BITS);
387         else  /* TypeDef */
388                 return get_typedef (m, dor_token >> MONO_TYPEORMETHOD_BITS);
389 }
390
391 /** 
392  * get_encoded_typedef_or_ref:
393  * @m: metadata context 
394  * @ptr: location to decode from.
395  * @result: pointer to string where resulting decoded string is stored
396  *
397  * result will point to a g_malloc()ed string.
398  *
399  * Returns: the new ptr to continue decoding
400  */
401 const char *
402 get_encoded_typedef_or_ref (MonoImage *m, const char *ptr, char **result)
403 {
404         guint32 token;
405         
406         token = mono_metadata_decode_value (ptr, &ptr);
407
408         *result = get_typedef_or_ref (m, token, NULL);
409
410         return ptr;
411 }
412
413 /**
414  * get_custom_mod:
415  *
416  * Decodes a CustomMod (22.2.7)
417  *
418  * Returns: updated pointer location
419  */
420 const char *
421 get_custom_mod (MonoImage *m, const char *ptr, char **return_value)
422 {
423         char *s;
424         const char *mod;
425         
426         *return_value = NULL;
427         while ((*ptr == MONO_TYPE_CMOD_OPT) ||
428                    (*ptr == MONO_TYPE_CMOD_REQD)) {
429                 mod = (*ptr == MONO_TYPE_CMOD_REQD) ? "modreq" : "modopt";
430                 ptr++;
431                 ptr = get_encoded_typedef_or_ref (m, ptr, &s);
432
433                 if (*return_value == NULL)
434                         *return_value = g_strconcat (" ", mod, " (", s, ")", NULL);
435                 else
436                         *return_value = g_strconcat (*return_value, " ", mod, " (", s, ")", NULL);
437                 g_free (s);
438         }
439         return ptr;
440 }
441
442
443 static dis_map_t element_type_map [] = {
444         { MONO_TYPE_END        , "end" },
445         { MONO_TYPE_VOID       , "void" },
446         { MONO_TYPE_BOOLEAN    , "bool" },
447         { MONO_TYPE_CHAR       , "char" }, 
448         { MONO_TYPE_I1         , "int8" },
449         { MONO_TYPE_U1         , "unsigned int8" }, 
450         { MONO_TYPE_I2         , "int16" },
451         { MONO_TYPE_U2         , "unsigned int16" },
452         { MONO_TYPE_I4         , "int32" },
453         { MONO_TYPE_U4         , "unsigned int32" },
454         { MONO_TYPE_I8         , "int64" },
455         { MONO_TYPE_U8         , "unsigned int64" },
456         { MONO_TYPE_R4         , "float32" },
457         { MONO_TYPE_R8         , "float64" },
458         { MONO_TYPE_STRING     , "string" },
459         { MONO_TYPE_TYPEDBYREF , "typedref" },
460         { MONO_TYPE_I          , "native int" },
461         { MONO_TYPE_U          , "native unsigned int" },
462         { MONO_TYPE_OBJECT     , "object" },
463         { 0, NULL }
464 };
465
466 static dis_map_t call_conv_type_map [] = {
467         { MONO_CALL_DEFAULT     , "default" },
468         { MONO_CALL_C           , "unmanaged cdecl" },
469         { MONO_CALL_STDCALL     , "unmanaged stdcall" },
470         { MONO_CALL_THISCALL    , "unmanaged thiscall" },
471         { MONO_CALL_FASTCALL    , "unmanaged fastcall" },
472         { MONO_CALL_VARARG      , "vararg" },
473         { 0, NULL }
474 };
475
476 char*
477 dis_stringify_token (MonoImage *m, guint32 token)
478 {
479         guint idx = token & 0xffffff;
480         switch (token >> 24) {
481         case MONO_TABLE_TYPEDEF: return get_typedef (m, idx);
482         case MONO_TABLE_TYPEREF: return get_typeref (m, idx);
483         case MONO_TABLE_TYPESPEC: return get_typespec (m, idx, FALSE, NULL);
484         default:
485                  break;
486         }
487         return g_strdup_printf("0x%08x", token);
488 }
489
490 char*
491 dis_stringify_array (MonoImage *m, MonoArrayType *array, gboolean is_def)
492 {
493         char *type, *arr_str, *ret;
494         
495         type = dis_stringify_type (m, &array->eklass->byval_arg, is_def);
496         arr_str = stringify_array (array->rank, array->numsizes, array->numlobounds, array->sizes, array->lobounds);
497
498         ret = g_strconcat (type, arr_str, NULL);
499         
500         g_free (arr_str);
501         g_free (type);
502         return ret;
503 }
504
505 char*
506 dis_stringify_modifiers (MonoImage *m, int n, MonoCustomMod *mod)
507 {
508         GString *s = g_string_new("");
509         char *result;
510         int i;
511         for (i = 0; i < n; ++i) {
512                 char *tok = dis_stringify_token (m, mod[i].token);
513                 if (i > 0)
514                         g_string_append_printf (s, " ");
515                 g_string_append_printf (s, " %s (%s)", mod[i].required ? "modreq": "modopt", tok);
516                 g_free (tok);
517         }
518         g_string_append_c (s, ' ');
519         result = s->str;
520         g_string_free (s, FALSE);
521         return result;
522 }
523
524 char*
525 dis_stringify_param (MonoImage *m, MonoType *param) 
526 {
527         char *t;
528         char *result;
529         char *attribs;
530         const char *in = param->attrs & PARAM_ATTRIBUTE_IN ? "[in]" : "";
531         const char *out = param->attrs & PARAM_ATTRIBUTE_OUT ? "[out]": "";
532         const char *opt = param->attrs & PARAM_ATTRIBUTE_OPTIONAL ? "[opt]": "";
533         attribs = g_strconcat(in, out, opt, NULL);
534         t = dis_stringify_type (m, param, TRUE);
535         result = g_strjoin(attribs[0] ? " ":"", attribs, t, NULL);
536         g_free (t);
537         g_free (attribs);
538         return result;
539 }
540
541 static char*
542 dis_stringify_variant_type (MonoMarshalVariant variant)
543 {
544         switch (variant) {
545         case MONO_VARIANT_EMPTY:
546                 return g_strdup ("");
547         case MONO_VARIANT_NULL:
548                 return g_strdup ("null");
549         case MONO_VARIANT_I2:
550                 return g_strdup ("int16");
551         case MONO_VARIANT_I4:
552                 return g_strdup ("int32");
553         case MONO_VARIANT_R4:
554                 return g_strdup ("float32");
555         case MONO_VARIANT_R8:
556                 return g_strdup ("float64");
557         case MONO_VARIANT_CY:
558                 return g_strdup ("currency");
559         case MONO_VARIANT_DATE:
560                 return g_strdup ("date");
561         case MONO_VARIANT_BSTR:
562                 return g_strdup ("bstr");
563         case MONO_VARIANT_DISPATCH:
564                 return g_strdup ("idispatch");
565         case MONO_VARIANT_ERROR:
566                 return g_strdup ("error");
567         case MONO_VARIANT_BOOL:
568                 return g_strdup ("bool");
569         case MONO_VARIANT_VARIANT:
570                 return g_strdup ("variant");
571         case MONO_VARIANT_UNKNOWN:
572                 return g_strdup ("iunknown");
573         case MONO_VARIANT_DECIMAL:
574                 return g_strdup ("decimal");
575         case MONO_VARIANT_I1:
576                 return g_strdup ("int8");
577         case MONO_VARIANT_UI1:
578                 return g_strdup ("unsigned int8");
579         case MONO_VARIANT_UI2:
580                 return g_strdup ("unsigned int16");
581         case MONO_VARIANT_UI4:
582                 return g_strdup ("unsigned int32");
583         case MONO_VARIANT_I8:
584                 return g_strdup ("int64");
585         case MONO_VARIANT_UI8:
586                 return g_strdup ("unsigned int64");
587         case MONO_VARIANT_INT:
588                 return g_strdup ("int");
589         case MONO_VARIANT_UINT:
590                 return g_strdup ("unsigned int");
591         case MONO_VARIANT_VOID:
592                 return g_strdup ("void");
593         case MONO_VARIANT_HRESULT:
594                 return g_strdup ("hresult");
595         case MONO_VARIANT_PTR:
596                 return g_strdup ("*");
597         case MONO_VARIANT_SAFEARRAY:
598                 return g_strdup ("safearray");
599         case MONO_VARIANT_CARRAY:
600                 return g_strdup ("carray");
601         case MONO_VARIANT_USERDEFINED:
602                 return g_strdup ("userdefined");
603         case MONO_VARIANT_LPSTR:
604                 return g_strdup ("lpstr");
605         case MONO_VARIANT_LPWSTR:
606                 return g_strdup ("lpwstr");
607         case MONO_VARIANT_RECORD:
608                 return g_strdup ("record");
609         case MONO_VARIANT_FILETIME:
610                 return g_strdup ("filetime");
611         case MONO_VARIANT_BLOB:
612                 return g_strdup ("blob");
613         case MONO_VARIANT_STREAM:
614                 return g_strdup ("stream");
615         case MONO_VARIANT_STORAGE:
616                 return g_strdup ("storage");
617         case MONO_VARIANT_STREAMED_OBJECT:
618                 return g_strdup ("streamed_object");
619         case MONO_VARIANT_STORED_OBJECT:
620                 return g_strdup ("stored_object");
621         case MONO_VARIANT_BLOB_OBJECT:
622                 return g_strdup ("blob_object");
623         case MONO_VARIANT_CF:
624                 return g_strdup ("cf");
625         case MONO_VARIANT_CLSID:
626                 return g_strdup ("clsid");
627         case MONO_VARIANT_VECTOR:
628                 /* FIXME: output: <v_type> vector */
629                 return g_strdup ("vector");
630         case MONO_VARIANT_ARRAY:
631                 /* FIXME: output: <v_type> [ ] */
632                 return g_strdup ("[]");
633         case MONO_VARIANT_BYREF:
634                 /* FIXME: output: <v_type> & */
635                 return g_strdup ("&");
636         default:
637                 return g_strdup ("unknown");
638         }
639 }
640
641 static char*
642 dis_stringify_native_type (MonoMarshalNative native)
643 {
644         switch (native) {
645         case MONO_NATIVE_BOOLEAN:
646                 return g_strdup ("bool");
647         case MONO_NATIVE_I1:
648                 return g_strdup ("int8");
649         case MONO_NATIVE_U1:
650                 return g_strdup ("unsigned int8");
651         case MONO_NATIVE_I2:
652                 return g_strdup ("int16");
653         case MONO_NATIVE_U2:
654                 return g_strdup ("unsigned int16");
655         case MONO_NATIVE_I4:
656                 return g_strdup ("int32");
657         case MONO_NATIVE_U4:
658                 return g_strdup ("unsigned int32");
659         case MONO_NATIVE_I8:
660                 return g_strdup ("int64");
661         case MONO_NATIVE_U8:
662                 return g_strdup ("unsigned int64");
663         case MONO_NATIVE_R4:
664                 return g_strdup ("float32");
665         case MONO_NATIVE_R8:
666                 return g_strdup ("float64");
667         case MONO_NATIVE_CURRENCY:
668                 return g_strdup ("currency");
669         case MONO_NATIVE_BSTR:
670                 return g_strdup ("bstr");
671         case MONO_NATIVE_LPSTR:
672                 return g_strdup ("lpstr");
673         case MONO_NATIVE_LPWSTR:
674                 return g_strdup ("lpwstr");
675         case MONO_NATIVE_LPTSTR:
676                 return g_strdup ("lptstr");
677         case MONO_NATIVE_IUNKNOWN:
678                 return g_strdup ("iunknown");
679         case MONO_NATIVE_IDISPATCH:
680                 return g_strdup ("idispatch");
681         case MONO_NATIVE_STRUCT:
682                 return g_strdup ("struct");
683         case MONO_NATIVE_INTERFACE:
684                 return g_strdup ("interface");
685         case MONO_NATIVE_SAFEARRAY:
686                 return g_strdup ("safearray");
687         case MONO_NATIVE_INT:
688                 return g_strdup ("int");
689         case MONO_NATIVE_UINT:
690                 return g_strdup ("unsigned int");
691         case MONO_NATIVE_VBBYREFSTR:
692                 return g_strdup ("vbbyrefstr");
693         case MONO_NATIVE_ANSIBSTR:
694                 return g_strdup ("ansi bstr");
695         case MONO_NATIVE_TBSTR:
696                 return g_strdup ("tbstr");
697         case MONO_NATIVE_VARIANTBOOL:
698                 return g_strdup ("variant bool");
699         case MONO_NATIVE_FUNC:
700                 return g_strdup ("method");
701         case MONO_NATIVE_ASANY:
702                 return g_strdup ("as any");
703         case MONO_NATIVE_LPSTRUCT:
704                 return g_strdup ("lpstruct");
705         case MONO_NATIVE_CUSTOM:
706                 return g_strdup ("custom");
707         case MONO_NATIVE_ERROR:
708                 return g_strdup ("error");
709         case MONO_NATIVE_MAX:
710                 return g_strdup ("");
711         default:
712                 return g_strdup ("unknown");
713         }
714 }
715
716 char*
717 dis_stringify_marshal_spec (MonoMarshalSpec *spec)
718 {
719         switch (spec->native) {
720         case MONO_NATIVE_BYVALTSTR:
721                 return g_strdup_printf (" marshal (fixed sysstring [%d])", spec->data.array_data.num_elem);
722         case MONO_NATIVE_BYVALARRAY:
723                 return g_strdup_printf (" marshal (fixed array [%d])", spec->data.array_data.num_elem);
724         case MONO_NATIVE_LPARRAY: {
725                 char *elem_type, *elems, *ret;
726                 guint32 num_elem = spec->data.array_data.num_elem;
727                 guint32 param_num = spec->data.array_data.param_num;
728
729                 elem_type = dis_stringify_native_type (spec->data.array_data.elem_type);
730                 if (num_elem == -1 && param_num == -1)
731                         elems = g_strdup ("");
732                 else if ((param_num == -1) || (spec->data.array_data.elem_mult == 0))
733                         elems = g_strdup_printf ("%d", num_elem);
734                 else if ((num_elem == -1) || (num_elem == 0))
735                         elems = g_strdup_printf ("+ %d", param_num);
736                 else
737                         elems = g_strdup_printf ("%d + %d", num_elem, param_num);
738                         
739                 ret = g_strdup_printf (" marshal (%s[%s])", elem_type, elems);
740                 g_free (elem_type);
741                 g_free (elems);
742                 return ret;
743         }
744         case MONO_NATIVE_SAFEARRAY: {
745                 char *elem_type = NULL, *ret;
746                 
747                 if (spec->data.safearray_data.elem_type != 0)
748                         elem_type = dis_stringify_variant_type (spec->data.safearray_data.elem_type);
749                 ret = g_strdup_printf (" marshal (safearray %s)", elem_type ? elem_type : "");
750                 
751                 g_free (elem_type);
752                 return ret;
753         }
754         case MONO_NATIVE_CUSTOM:
755                 return g_strdup_printf (" marshal (custom (\"%s\", \"%s\"))", 
756                         spec->data.custom_data.custom_name ? spec->data.custom_data.custom_name : "", 
757                         spec->data.custom_data.cookie ? spec->data.custom_data.cookie : "");
758         default: {
759                 char *native_type, *ret;
760                 native_type = dis_stringify_native_type (spec->native);
761                 ret = g_strdup_printf (" marshal (%s)", native_type);
762                 g_free (native_type);
763                 return ret;
764         }       
765         }
766 }
767
768 /**
769  * get_generic_param
770  * @m: metadata context 
771  * @table_type: The type of table we are getting generics for (0 for typedef, 1 for method)
772  * @row: The row in the table
773  *
774  * Returns: Allocated stringified generic parameters
775  */
776 char*
777 get_generic_param (MonoImage *m, MonoGenericContainer *container)
778 {
779         GString *result;
780         char *retval;
781         int i;
782
783         if (!container)
784                 return NULL;
785
786         result = g_string_new ("");
787
788         g_string_append_c (result, '<');
789         for (i = 0; i < container->type_argc; i++) {
790                 MonoGenericParam *param = mono_generic_container_get_param (container, i);
791                 MonoGenericParamInfo *param_info = mono_generic_param_info (param);
792                 MonoClass **constr;
793                 int first = 1;
794                 guint16 flags;
795                 char *esname;
796
797                 if (i > 0)
798                         g_string_append (result, ",");
799                 
800                 flags = param_info->flags & GENERIC_PARAMETER_ATTRIBUTE_VARIANCE_MASK;
801                 if ((flags & GENERIC_PARAMETER_ATTRIBUTE_COVARIANT) == GENERIC_PARAMETER_ATTRIBUTE_COVARIANT)
802                         g_string_append (result, "+ ");
803                 if ((flags & GENERIC_PARAMETER_ATTRIBUTE_CONTRAVARIANT) == GENERIC_PARAMETER_ATTRIBUTE_CONTRAVARIANT)
804                         g_string_append (result, "- ");
805
806                 flags = param_info->flags & GENERIC_PARAMETER_ATTRIBUTE_SPECIAL_CONSTRAINTS_MASK;
807                 if ((flags & GENERIC_PARAMETER_ATTRIBUTE_REFERENCE_TYPE_CONSTRAINT) == GENERIC_PARAMETER_ATTRIBUTE_REFERENCE_TYPE_CONSTRAINT)
808                         g_string_append (result, "class ");
809                 if ((flags & GENERIC_PARAMETER_ATTRIBUTE_VALUE_TYPE_CONSTRAINT) == GENERIC_PARAMETER_ATTRIBUTE_VALUE_TYPE_CONSTRAINT)
810                         g_string_append (result, "valuetype ");
811                 if ((flags & GENERIC_PARAMETER_ATTRIBUTE_CONSTRUCTOR_CONSTRAINT) == GENERIC_PARAMETER_ATTRIBUTE_CONSTRUCTOR_CONSTRAINT)
812                         g_string_append (result, ".ctor ");
813
814                 for (constr = param_info->constraints; constr && *constr; constr++) {
815                         char *sig;
816
817                         if (first) {
818                                 g_string_append_c (result, '(');
819                                 first = 0;
820                         } else
821                                 g_string_append (result, ", ");
822                         sig = dis_stringify_type (m, &((*constr)->byval_arg), TRUE);
823                         g_string_append (result, sig);
824                         g_free (sig);
825                 }
826
827                 if (!first)
828                         g_string_append (result, ") ");
829
830                 esname = get_escaped_name (mono_generic_param_info (param)->name);
831                 g_string_append (result, esname);
832                 g_free (esname);
833         }
834
835         g_string_append_c (result, '>');
836
837         retval = result->str;
838         g_string_free (result, FALSE);
839         return retval;
840 }
841
842 char*
843 dis_stringify_method_signature (MonoImage *m, MonoMethodSignature *method, int methoddef_row,
844                                 MonoGenericContainer *container, gboolean fully_qualified)
845 {
846         return dis_stringify_method_signature_full (m, method, methoddef_row, container, fully_qualified, TRUE);
847 }
848
849 /* 
850  * @m: metadata context
851  * @method: MonoMethodSignature to dis-stringify
852  * @methoddef_row: row index in the Method table
853  * @context: generic context, generic method's context in case of a Generic method
854  *           or a generic type's context. if !@context, treats it as a non-generic method
855  * @fully_qualified: TRUE to print type name also.
856  *
857  * Returns: Allocated stringified method signature
858  */
859 char*
860 dis_stringify_method_signature_full (MonoImage *m, MonoMethodSignature *method, int methoddef_row,
861                                 MonoGenericContainer *container, gboolean fully_qualified, gboolean with_marshal_info)
862 {
863         guint32 cols [MONO_METHOD_SIZE];
864         guint32 pcols [MONO_PARAM_SIZE];
865         guint32 param_index = 0, next_param_index = 0;
866         gboolean has_param_row;
867         const char *method_name = "";
868         int free_method = 0;
869         char *retval, *esname;
870         char *type = NULL;
871         char *marshal_info = NULL, *ret_marshal_info = NULL;
872         char *gen_param = NULL;
873         GString *result = g_string_new ("");
874         GString *result_ret = g_string_new ("");
875         int i, start;
876         
877         g_assert (method || methoddef_row);
878
879         if (methoddef_row) {
880                 mono_metadata_decode_row (&m->tables [MONO_TABLE_METHOD], methoddef_row -1, cols, MONO_METHOD_SIZE);
881                 if (fully_qualified) {
882                         guint32 type_idx = mono_metadata_typedef_from_method (m, methoddef_row);
883                         if (type_idx)
884                                 type = get_typedef (m, type_idx);
885                         else
886                                 type = g_strdup ("<invalid>");
887                 }
888                 method_name = mono_metadata_string_heap (m, cols [MONO_METHOD_NAME]);
889                 param_index = cols [MONO_METHOD_PARAMLIST];
890                 if (!method) {
891                         const char *sig = mono_metadata_blob_heap (m, cols [MONO_METHOD_SIGNATURE]);
892
893                         container = mono_metadata_load_generic_params (m, MONO_TOKEN_METHOD_DEF | methoddef_row, container);
894                         if (container)
895                                 mono_metadata_load_generic_param_constraints (m, MONO_TOKEN_METHOD_DEF | methoddef_row, container);
896
897                         mono_metadata_decode_blob_size (sig, &sig);
898                         method = mono_metadata_parse_method_signature_full (m, container, methoddef_row, sig, &sig);
899                         free_method = 1;
900                 }
901
902                 if (container && container->is_method)
903                         gen_param = get_generic_param (m, container);
904                 
905                 if (methoddef_row < m->tables [MONO_TABLE_METHOD].rows) {
906                         mono_metadata_decode_row (&m->tables [MONO_TABLE_METHOD], methoddef_row, cols, MONO_METHOD_SIZE);
907                         next_param_index = cols [MONO_METHOD_PARAMLIST];
908                 } else {
909                         next_param_index = m->tables [MONO_TABLE_PARAM].rows + 1;
910                 }
911         } 
912
913         start = method->hasthis ? 0 : 1;
914         for (i = 0; i < method->param_count + 1; ++i) {
915                 marshal_info = NULL;
916                 has_param_row = param_index && param_index < next_param_index;
917                 esname = NULL;
918
919                 if (method->param_count == 0 && !has_param_row)
920                         /* method has zero parameters, and no row for return val in the PARAM table */
921                         continue;
922                 
923                 if (has_param_row)
924                         mono_metadata_decode_row (&m->tables [MONO_TABLE_PARAM], param_index - 1, pcols, MONO_PARAM_SIZE);
925                 
926                 if (has_param_row && i == pcols [MONO_PARAM_SEQUENCE]) {
927                         if (i)
928                                 esname = get_escaped_name (
929                                                 mono_metadata_string_heap (m, pcols [MONO_PARAM_NAME]));
930
931                         if (with_marshal_info && (pcols [MONO_PARAM_FLAGS] & PARAM_ATTRIBUTE_HAS_FIELD_MARSHAL)) {
932                                 const char *tp;
933                                 MonoMarshalSpec *spec;
934                                 tp = mono_metadata_get_marshal_info (m, param_index - 1, FALSE);
935                                 if (tp) {
936                                         spec = mono_metadata_parse_marshal_spec (m, tp);
937         
938                                         if (i)
939                                                 marshal_info = dis_stringify_marshal_spec (spec);
940                                         else
941                                                 ret_marshal_info = dis_stringify_marshal_spec (spec);
942                                 } else {
943                                         if (i)
944                                                 marshal_info = g_strdup ("(missing)");
945                                         else
946                                                 ret_marshal_info = g_strdup ("(missing)");
947                                 }
948                         }
949                         param_index ++;
950                 } else {
951                         if (i)
952                                 /* A_[0-9]* does not require escaping */
953                                 esname = g_strdup_printf ("A_%i", i - start);
954                 }
955
956                 if (!i)
957                         continue;
958
959                 if (i > 1)
960                         g_string_append (result, ", ");
961
962                 retval = dis_stringify_param (m, method->params [i - 1]);
963
964                 g_string_append_printf (result, "%s%s %s", retval, marshal_info ? marshal_info : "", esname);
965                 g_free (retval);
966                 g_free (esname);
967                 g_free (marshal_info);
968         }
969         g_string_append (result, ") ");
970
971         retval = dis_stringify_param (m, method->ret);
972
973         if (method->hasthis)
974                 g_string_append (result_ret, "instance ");
975         g_string_append (result_ret, map (method->call_convention, call_conv_type_map));
976         g_string_append_printf (result_ret, " %s%s ", retval, ret_marshal_info ? ret_marshal_info :"");
977         g_free (ret_marshal_info);
978         if (type) {
979                 char *estype = get_escaped_name (type);
980                 g_string_append_printf (result_ret, "%s::", estype);
981                 g_free (estype);
982                 g_free (type);
983         }
984         esname = get_escaped_name (method_name);
985         g_string_append (result_ret, esname);
986         g_free (esname);
987         if (gen_param) {
988                 g_string_append (result_ret, gen_param);
989                 g_free (gen_param);
990         }
991         g_string_append (result_ret, " (");
992         g_free (retval);
993
994         g_string_prepend (result, result_ret->str);
995         g_string_free (result_ret, FALSE);
996
997         if (show_method_tokens && methoddef_row)
998                 g_string_append_printf (result, " /* 0x%X */ ",
999                                 (methoddef_row >> MONO_TYPEORMETHOD_BITS) | MONO_TOKEN_METHOD_DEF);
1000
1001         if (free_method)
1002                 mono_metadata_free_method_signature (method);
1003         retval = result->str;
1004         g_string_free (result, FALSE);
1005
1006         return retval;
1007 }
1008
1009 char*
1010 dis_stringify_function_ptr (MonoImage *m, MonoMethodSignature *method)
1011 {
1012         char *retval;
1013         GString *result = g_string_new ("");
1014         int i;
1015         
1016         g_assert (method);
1017
1018         g_string_append (result, map (method->call_convention, call_conv_type_map));
1019
1020         retval = dis_stringify_param (m, method->ret);
1021         g_string_append_printf (result, " %s ", retval);
1022         g_free (retval);
1023
1024         g_string_append (result, " *(");
1025         for (i = 0; i < method->param_count; ++i) {
1026                 if (i)
1027                         g_string_append (result, ", ");
1028                 retval = dis_stringify_param (m, method->params [i]);
1029                 g_string_append (result, retval);
1030                 g_free (retval);
1031         }
1032         g_string_append (result, ") ");
1033
1034         retval = result->str;
1035         g_string_free (result, FALSE);
1036
1037         return retval;
1038 }
1039
1040 static char *
1041 get_escaped_class_name (MonoClass *c)
1042 {
1043         char *result, *esname;
1044
1045         if (c->type_token == mono_metadata_make_token (MONO_TABLE_TYPEDEF, 1))
1046                 /* <Module> */
1047                 return NULL;
1048
1049         if (c->rank || c->byval_arg.type == MONO_TYPE_PTR) 
1050                 g_assert (0);
1051
1052         esname = get_escaped_name (c->name);
1053
1054         if (c->nested_in){
1055                 char *part_a = get_escaped_class_name (c->nested_in);
1056
1057                 result = g_strdup_printf ("%s/%s", part_a, esname);
1058                 g_free (part_a);
1059         } else if (*c->name_space)
1060                 result = g_strdup_printf ("%s.%s", c->name_space, esname);
1061         else
1062                 result = g_strdup (esname);
1063
1064         g_free (esname);
1065         return result;
1066 }
1067
1068 char *
1069 dis_stringify_object_with_class (MonoImage *m, MonoClass *c, gboolean prefix, gboolean is_def)
1070 {
1071         /* FIXME: handle MONO_TYPE_OBJECT ... */
1072         MonoType *type = &c->byval_arg;
1073         const char *otype = type->type == MONO_TYPE_VALUETYPE ? "valuetype " : "class " ;
1074         char *assemblyref = NULL, *result, *esname, *generic = NULL;
1075         
1076         if (c->type_token == mono_metadata_make_token (MONO_TABLE_TYPEDEF, 1))
1077                 /* <Module> */
1078                 return NULL;
1079
1080         if (m != c->image) {
1081                 if (c->image->assembly_name) {
1082                         /* we cheat */
1083                         if (substitute_with_mscorlib_p && !strcmp ("corlib", c->image->assembly_name))
1084                                 assemblyref = g_strdup_printf ("[%s]", "mscorlib");
1085                         else {
1086                                 char *esc = get_escaped_name (c->image->assembly->aname.name);
1087                                 
1088                                 assemblyref = g_strdup_printf ("[%s]", esc);
1089                                 g_free (esc);
1090                         }
1091                 } else {
1092                         assemblyref = g_strdup_printf ("[.module %s]", c->image->module_name);
1093                 }
1094         }
1095
1096         esname = get_escaped_class_name (c);
1097
1098         if (c->generic_class) {
1099                 MonoGenericClass *gclass = c->generic_class;
1100                 MonoGenericInst *inst = gclass->context.class_inst;
1101                 GString *str = g_string_new ("");
1102                 int i;
1103
1104                 for (i = 0; i < inst->type_argc; i++){
1105                         char *t = dis_stringify_type (m, inst->type_argv [i], is_def);
1106
1107                         g_string_append (str, t);
1108                         if (i+1 != inst->type_argc)
1109                                 g_string_append (str, ", ");
1110                         g_free (t);
1111                 }
1112                 generic = g_strdup_printf ("<%s>", str->str);
1113                 g_string_free (str, TRUE);
1114         }
1115
1116
1117         result = g_strdup_printf ("%s%s%s%s", prefix ? otype : "", assemblyref?assemblyref:"",
1118                                   esname, generic?generic:"");
1119         
1120         g_free (generic);
1121         g_free (assemblyref);
1122         g_free (esname);
1123         
1124         return result;
1125 }
1126
1127 static char *
1128 dis_stringify_object (MonoImage *m, MonoType *type, gboolean is_def)
1129 {
1130         MonoClass *c = mono_class_from_mono_type (type);
1131         return dis_stringify_object_with_class (m, c, TRUE, is_def);
1132 }
1133
1134 char*
1135 dis_stringify_type (MonoImage *m, MonoType *type, gboolean is_def)
1136 {
1137         const char *pinned = "", *byref = "";
1138         char *bare = NULL, *mods = NULL;
1139         char *result;
1140
1141         if (type->num_mods)
1142                 mods = dis_stringify_modifiers (m, type->num_mods, type->modifiers);
1143
1144         switch (type->type){
1145         case MONO_TYPE_BOOLEAN:
1146         case MONO_TYPE_CHAR:
1147         case MONO_TYPE_I1:
1148         case MONO_TYPE_U1:
1149         case MONO_TYPE_I2:
1150         case MONO_TYPE_U2:
1151         case MONO_TYPE_I4:
1152         case MONO_TYPE_U4:
1153         case MONO_TYPE_I8:
1154         case MONO_TYPE_U8:
1155         case MONO_TYPE_R4:
1156         case MONO_TYPE_R8:
1157         case MONO_TYPE_I:
1158         case MONO_TYPE_U:
1159         case MONO_TYPE_STRING:
1160         case MONO_TYPE_OBJECT:
1161         case MONO_TYPE_TYPEDBYREF:
1162                 bare = g_strdup (map (type->type, element_type_map));
1163                 break;
1164                 
1165         case MONO_TYPE_VALUETYPE:
1166         case MONO_TYPE_CLASS:
1167                 bare = dis_stringify_object (m, type, is_def);
1168                 break;
1169         case MONO_TYPE_FNPTR: {
1170                 char *child_type;
1171                 child_type = dis_stringify_function_ptr (m, type->data.method);
1172                 bare = g_strdup_printf ("method %s", child_type);
1173                 g_free (child_type);
1174                 break;
1175         }
1176         case MONO_TYPE_PTR: {
1177                 char *child_type;
1178                 child_type = dis_stringify_type (m, type->data.type, is_def);
1179                 
1180                 bare = g_strdup_printf ("%s*", child_type);
1181                 g_free (child_type);
1182                 break;
1183         }
1184         case MONO_TYPE_SZARRAY: {
1185                 char *child_type;
1186                 child_type = dis_stringify_type (m, &type->data.klass->byval_arg, is_def);
1187                 
1188                 bare = g_strdup_printf ("%s[]", child_type);
1189                 g_free (child_type);
1190                 break;
1191         }
1192         case MONO_TYPE_ARRAY:
1193                 bare = dis_stringify_array (m, type->data.array, is_def);
1194                 break;
1195         case MONO_TYPE_VOID:
1196                 bare = g_strdup ("void");
1197                 break;
1198         case MONO_TYPE_MVAR:
1199                 if (is_def && !cant_print_generic_param_name (type->data.generic_param))
1200                         bare = g_strdup_printf ("!!%s", get_escaped_name (mono_generic_param_info (type->data.generic_param)->name));
1201                 else
1202                         bare = g_strdup_printf ("!!%d", mono_type_get_generic_param_num (type));
1203                 break;
1204         case MONO_TYPE_VAR:
1205                 if (is_def && !cant_print_generic_param_name (type->data.generic_param))
1206                         bare = g_strdup_printf ("!%s", get_escaped_name (mono_generic_param_info (type->data.generic_param)->name));
1207                 else
1208                         bare = g_strdup_printf ("!%d", mono_type_get_generic_param_num (type));
1209                 break;
1210         case MONO_TYPE_GENERICINST: {
1211                 GString *str = g_string_new ("");
1212                 MonoGenericInst *inst;
1213                 int i;
1214                 char *generic_type = dis_stringify_type (
1215                         m, &type->data.generic_class->container_class->byval_arg, is_def);
1216                 inst = type->data.generic_class->context.class_inst;
1217                 for (i = 0; i < inst->type_argc; i++){
1218                         char *t = dis_stringify_type (m, inst->type_argv [i], is_def);
1219
1220                         g_string_append (str, t);
1221                         if (i+1 != inst->type_argc)
1222                                 g_string_append (str, ", ");
1223                         g_free (t);
1224                 }
1225                 bare = g_strdup_printf ("%s<%s>", generic_type, str->str);
1226                 g_string_free (str, TRUE);
1227                 break;
1228         }
1229                 
1230         default:
1231                 g_error ("Do not know how to stringify type 0x%x", type->type);
1232         }
1233         
1234         if (type->pinned)
1235                 pinned = " pinned";
1236
1237         if (type->byref)
1238                 byref = "&";
1239         
1240         if (!bare)
1241                 /* bare is NULL, for <Module> */
1242                 return bare;
1243                 
1244         result = g_strconcat (bare, byref, pinned, mods ? mods : "", NULL);
1245
1246         g_free (bare);
1247
1248         return result;
1249 }
1250
1251 /**
1252  * get_type:
1253  * @m: metadata context 
1254  * @ptr: location to decode from.
1255  * @result: pointer to string where resulting decoded string is stored
1256  *
1257  * This routine returs in @result the stringified type pointed by @ptr.
1258  * (22.2.12)
1259  *
1260  * Returns: the new ptr to continue decoding
1261  */
1262 const char *
1263 get_type (MonoImage *m, const char *ptr, char **result, gboolean is_def, MonoGenericContainer *container)
1264 {
1265         const char *start = ptr;
1266         guint32 type;
1267         MonoType *t;
1268
1269         if (*ptr == MONO_TYPE_BYREF)
1270                 ++ptr;
1271
1272         type = mono_metadata_decode_value (ptr, &ptr);
1273
1274         switch (type){
1275         case MONO_TYPE_VALUETYPE:
1276         case MONO_TYPE_CLASS: {
1277                 guint32 token = mono_metadata_parse_typedef_or_ref (m, ptr, &ptr);
1278                 MonoClass *klass = mono_class_get (m, token);
1279                 char *temp;
1280                 if (klass)
1281                         temp = dis_stringify_object_with_class (m, klass, TRUE, FALSE);
1282                 else
1283                         temp = g_strdup_printf ("<BROKEN CLASS token_%8x>", token);
1284
1285                 if (show_tokens) {
1286                         *result = g_strdup_printf ("%s/*%08x*/", temp, token);
1287                         g_free (temp);
1288                 } else
1289                         *result = temp;
1290                 break;
1291         }
1292
1293         case MONO_TYPE_GENERICINST: {
1294                 GString *str = g_string_new ("");
1295                 int count, i;
1296                 char *temp;
1297
1298                 ptr = get_type (m, ptr, &temp, is_def, container);
1299                 g_string_append (str, temp);
1300                 g_free (temp);
1301
1302                 count = mono_metadata_decode_value (ptr, &ptr);
1303                 g_string_append (str, "<");
1304
1305                 for (i = 0; i < count; i++) {
1306                         if (i)
1307                                 g_string_append (str, ",");
1308                         ptr = get_type (m, ptr, &temp, is_def, container);
1309                         g_string_append (str, temp);
1310                 }
1311
1312                 g_string_append (str, ">");
1313                 *result = str->str;
1314                 g_string_free (str, FALSE);
1315                 break;
1316         }
1317
1318         default:
1319                 t = mono_metadata_parse_type_full (m, container, MONO_PARSE_TYPE, 0, start, &ptr);
1320                 if (t) {
1321                         *result = dis_stringify_type (m, t, is_def);
1322                 } else {
1323                         GString *err = g_string_new ("@!#$<InvalidType>$#!@");
1324                         if (container)
1325                                 t = mono_metadata_parse_type_full (m, NULL, MONO_PARSE_TYPE, 0, start, &ptr);
1326                         if (t) {
1327                                 char *name = dis_stringify_type (m, t, is_def);
1328                                 g_warning ("Encountered a generic type inappropriate for its context");
1329                                 g_string_append (err, " // ");
1330                                 g_string_append (err, name);
1331                                 g_free (name);
1332                         } else {
1333                                 g_warning ("Encountered an invalid type");
1334                         }
1335                         *result = g_string_free (err, FALSE);
1336                 }
1337
1338                 break;
1339         }
1340
1341         return ptr;
1342 }
1343
1344 /**
1345  * 
1346  * Returns a stringified representation of a FieldSig (22.2.4)
1347  */
1348 char *
1349 get_field_signature (MonoImage *m, guint32 blob_signature, MonoGenericContainer *container)
1350 {
1351         char *allocated_modifier_string, *allocated_type_string;
1352         const char *ptr = mono_metadata_blob_heap (m, blob_signature);
1353         const char *base;
1354         char *res;
1355         int len;
1356         
1357         len = mono_metadata_decode_value (ptr, &ptr);
1358         base = ptr;
1359         /* FIELD is 0x06 */
1360         g_assert (*ptr == 0x06);
1361 /*      hex_dump (ptr, 0, len); */
1362         ptr++; len--;
1363         
1364         ptr = get_custom_mod (m, ptr, &allocated_modifier_string);
1365         ptr = get_type (m, ptr, &allocated_type_string, FALSE, container);
1366
1367         res = g_strdup_printf (
1368                 "%s%s",
1369                 allocated_type_string,
1370                 allocated_modifier_string ? allocated_modifier_string : "");
1371         
1372         if (allocated_modifier_string)
1373                 g_free (allocated_modifier_string);
1374         if (allocated_type_string)
1375                 g_free (allocated_type_string);
1376         
1377         return res;
1378 }
1379
1380 MonoTypeEnum
1381 get_field_literal_type (MonoImage *m, guint32 blob_signature)
1382 {
1383         const char *ptr = mono_metadata_blob_heap (m, blob_signature);
1384         int len;
1385         char *allocated_modifier_string;
1386         
1387         len = mono_metadata_decode_value (ptr, &ptr);
1388
1389         /* FIELD is 0x06 */
1390         g_assert (*ptr == 0x06);
1391         ptr++; len--;
1392         
1393         ptr = get_custom_mod (m, ptr, &allocated_modifier_string);
1394         if (allocated_modifier_string)
1395                 g_free (allocated_modifier_string);
1396
1397         return (MonoTypeEnum) *ptr;
1398         
1399 }
1400
1401 /**
1402  * decode_literal:
1403  * @m: metadata context
1404  * @token: token to decode
1405  *
1406  * decodes the literal indexed by @token.
1407  */
1408 char *
1409 decode_literal (MonoImage *m, guint32 token)
1410 {
1411         return g_strdup ("LITERAL_VALUE");
1412 }
1413
1414 /**
1415  * get_ret_type:
1416  * @m: metadata context 
1417  * @ptr: location to decode from.
1418  * @result: pointer to string where resulting decoded string is stored
1419  *
1420  * This routine returns in @result the stringified RetType (22.2.11)
1421  *
1422  * Returns: the new ptr to continue decoding.
1423  */
1424 const char *
1425 get_ret_type (MonoImage *m, const char *ptr, char **ret_type, MonoGenericContainer *container)
1426 {
1427         GString *str = g_string_new ("");
1428         char *mod = NULL;
1429         char *allocated_type_string;
1430         int has_byref = 0;
1431         
1432         ptr = get_custom_mod (m, ptr, &mod);
1433
1434         if (*ptr == MONO_TYPE_TYPEDBYREF){
1435                 g_string_append (str, "typedref");
1436                 ptr++;
1437         } else if (*ptr == MONO_TYPE_VOID){
1438                  g_string_append (str, "void");
1439                  ptr++;
1440         } else {
1441                 if (*ptr == MONO_TYPE_BYREF){
1442                         has_byref = 1;
1443                         ptr++;
1444                 }
1445
1446                 ptr = get_type (m, ptr, &allocated_type_string, FALSE, container);
1447                 g_string_append (str, allocated_type_string);
1448                 if (has_byref)
1449                         g_string_append (str, "& ");
1450                 g_free (allocated_type_string);
1451         }
1452
1453         if (mod){
1454                 g_string_append (str, mod);
1455                 g_string_append_c (str, ' ');
1456                 g_free (mod);
1457         }
1458
1459         *ret_type = str->str;
1460         g_string_free (str, FALSE);
1461
1462         return ptr;
1463 }
1464
1465 /**
1466  * get_param:
1467  * @m: metadata context 
1468  * @ptr: location to decode from.
1469  * @result: pointer to string where resulting decoded string is stored
1470  *
1471  * This routine returns in @result the stringified Param (22.2.10)
1472  *
1473  * Returns: the new ptr to continue decoding.
1474  */
1475 const char *
1476 get_param (MonoImage *m, const char *ptr, char **retval, MonoGenericContainer *container)
1477 {
1478         GString *str = g_string_new ("");
1479         char *allocated_mod_string, *allocated_type_string;
1480         
1481         ptr = get_custom_mod (m, ptr, &allocated_mod_string);
1482         
1483         if (*ptr == MONO_TYPE_TYPEDBYREF){
1484                 g_string_append (str, " typedref ");
1485                 ptr++;
1486         } else {
1487                 gboolean by_ref = 0;
1488                 if (*ptr == MONO_TYPE_BYREF){
1489                         g_string_append (str, "[out] ");
1490                         ptr++;
1491                         by_ref = 1;
1492                 }
1493                 ptr = get_type (m, ptr, &allocated_type_string, FALSE, container);
1494                 g_string_append (str, allocated_type_string);
1495                 if (by_ref)
1496                         g_string_append_c (str, '&');
1497                 g_free (allocated_type_string);
1498         }
1499
1500         if (allocated_mod_string){
1501                 g_string_append (str, allocated_mod_string);
1502                 g_string_append_c (str, ' ');
1503                 g_free (allocated_mod_string);
1504         }
1505
1506         *retval = str->str;
1507         g_string_free (str, FALSE);
1508         return ptr;
1509 }
1510
1511 /**
1512  * str_escape 
1513  * 
1514  * @str: string to process
1515  * @list: list of chars to escape
1516  * 
1517  * Returns: an allocated escaped string.
1518  */
1519 static char*
1520 str_escape (const char *str, const char *list)
1521 {
1522         const char *p = str;
1523         GString *res;
1524
1525         res = g_string_sized_new (strlen (str));
1526
1527         for (;;) {
1528                 while (*p && !strchr (list, *p))
1529                         ++p;
1530                 g_string_append_len (res, str, p - str);
1531                 if (!*p)
1532                         break;
1533                 g_string_append_c (res, '\\');
1534                 str = p;
1535                 ++p;
1536         }
1537                         
1538         return g_string_free (res, FALSE);
1539 }
1540
1541 /**
1542  * get_escaped_name
1543  *
1544  * Returns: An allocated escaped name. A name needs to be escaped
1545  * because it might be an ilasm keyword.
1546  */
1547 char*
1548 get_escaped_name (const char *name)
1549 {
1550         const char *s;
1551         char *ret, *esc;
1552
1553         if (!name)
1554                 return NULL;
1555
1556         g_assert (key_table);
1557
1558         if (strlen (name) == 0)
1559                 return g_strdup (name);
1560
1561         for (s = name; *s; s++) {
1562                 char *first, *result;
1563
1564                 if (*s != '/')
1565                         continue;
1566
1567                 first = g_strndup (name, s-name);
1568                 result = g_strdup_printf ("%s/%s", get_escaped_name (first), get_escaped_name (s+1));
1569                 g_free (first);
1570
1571                 return result;
1572         }
1573
1574         for (s = name; *s; s++) {
1575                 if (isalnum (*s) || *s == '_' || *s == '$' || *s == '@' ||
1576                     *s == '?' || (*s == '.' && s != name) || *s == 0 || *s == '!' || *s == '`')
1577                         continue;
1578
1579                 esc = str_escape (name, "'\\");
1580                 ret = g_strdup_printf ("'%s'", esc);
1581                 g_free (esc);
1582                 return ret;
1583         }
1584         
1585         if (g_hash_table_lookup (key_table, name))
1586                 return g_strdup_printf ("'%s'", name);
1587                         
1588         return str_escape (name, "'\\");
1589 }
1590
1591 static dis_map_t param_map [] = {
1592         { PARAM_ATTRIBUTE_IN,                "[in] " },
1593         { PARAM_ATTRIBUTE_OUT,               "[out] " },
1594         { PARAM_ATTRIBUTE_OPTIONAL,          "optional " },
1595         { PARAM_ATTRIBUTE_HAS_DEFAULT,       "hasdefault " },
1596         { PARAM_ATTRIBUTE_HAS_FIELD_MARSHAL, "fieldmarshal " },
1597         { 0, NULL }
1598 };
1599
1600 char *
1601 param_flags (guint32 f)
1602 {
1603         return g_strdup (flags (f, param_map));
1604 }
1605
1606 static dis_map_t field_access_map [] = {
1607         { FIELD_ATTRIBUTE_COMPILER_CONTROLLED, "privatescope " },
1608         { FIELD_ATTRIBUTE_PRIVATE,             "private " },
1609         { FIELD_ATTRIBUTE_FAM_AND_ASSEM,       "famandassem " },
1610         { FIELD_ATTRIBUTE_ASSEMBLY,            "assembly " },
1611         { FIELD_ATTRIBUTE_FAMILY,              "family " },
1612         { FIELD_ATTRIBUTE_FAM_OR_ASSEM,        "famorassem " },
1613         { FIELD_ATTRIBUTE_PUBLIC,              "public " },
1614         { 0, NULL }
1615 };
1616
1617 static dis_map_t field_flags_map [] = {
1618         { FIELD_ATTRIBUTE_STATIC,              "static " },
1619         { FIELD_ATTRIBUTE_INIT_ONLY,           "initonly " },
1620         { FIELD_ATTRIBUTE_LITERAL,             "literal " },
1621         { FIELD_ATTRIBUTE_NOT_SERIALIZED,      "notserialized " },
1622         { FIELD_ATTRIBUTE_SPECIAL_NAME,        "specialname " },
1623         { FIELD_ATTRIBUTE_PINVOKE_IMPL,        "FIXME:pinvokeimpl " },
1624         { FIELD_ATTRIBUTE_RT_SPECIAL_NAME,        "rtspecialname " },
1625         /*{ FIELD_ATTRIBUTE_HAS_FIELD_MARSHAL,        "hasfieldmarshal " },*/
1626         { 0, NULL }
1627 };
1628
1629 /**
1630  * field_flags:
1631  *
1632  * Returns a stringified version of a Field's flags
1633  */
1634 char *
1635 field_flags (guint32 f)
1636 {
1637         char buffer [1024];
1638         int access = f & FIELD_ATTRIBUTE_FIELD_ACCESS_MASK;
1639         
1640         buffer [0] = 0;
1641
1642         strcat (buffer, map (access, field_access_map));
1643         strcat (buffer, flags (f, field_flags_map));
1644         return g_strdup (buffer);
1645 }
1646
1647 /**
1648  * Returns a stringifed representation of a MethodRefSig (22.2.2)
1649  */
1650 char *
1651 get_methodref_signature (MonoImage *m, guint32 blob_signature, const char *fancy_name)
1652 {
1653         GString *res = g_string_new ("");
1654         const char *ptr = mono_metadata_blob_heap (m, blob_signature);
1655         char *allocated_ret_type, *s;
1656         const char *cconv_str;
1657         gboolean seen_vararg = 0;
1658         int param_count, signature_len;
1659         int i, gen_count = 0;
1660         int cconv;
1661
1662         signature_len = mono_metadata_decode_value (ptr, &ptr);
1663
1664         if (*ptr & 0x20){
1665                 if (*ptr & 0x40)
1666                         g_string_append (res, "explicit-this ");
1667                 else
1668                         g_string_append (res, "instance "); /* has-this */
1669         }
1670
1671         if (*ptr & 0x10)
1672                 gen_count = 1;
1673         cconv = *ptr & 0x0f;
1674         cconv_str = map (cconv, call_conv_type_map);
1675         if (strcmp (cconv_str, "default") != 0) {
1676                 g_string_append (res, cconv_str);
1677                 g_string_append (res, " ");
1678         }
1679
1680         ptr++;
1681         if (gen_count)
1682                 gen_count = mono_metadata_decode_value (ptr, &ptr);
1683         param_count = mono_metadata_decode_value (ptr, &ptr);
1684         if (cconv != 0xa) {
1685                 ptr = get_ret_type (m, ptr, &allocated_ret_type, NULL);
1686                 g_string_append (res, allocated_ret_type);
1687                 g_free (allocated_ret_type);
1688         }
1689
1690         if (fancy_name){
1691                 g_string_append_c (res, ' ');
1692                 g_string_append (res, fancy_name);
1693         }
1694         
1695         g_string_append (res, "(");
1696
1697         /*
1698          * param_count describes parameters *before* and *after*
1699          * the vararg sentinel
1700          */
1701         for (i = 0; i < param_count; i++){
1702                 char *param = NULL;
1703                 
1704                 /*
1705                  * If ptr is a SENTINEL
1706                  */
1707                 if (*ptr == 0x41){
1708                         if (!seen_vararg){
1709                                 g_string_append (res, "..., ");
1710                                 seen_vararg = 1;
1711                         }
1712
1713                         ptr++;
1714                 }
1715
1716                 ptr = get_param (m, ptr, &param, NULL);
1717                 g_string_append (res, param);
1718                 if (i+1 != param_count)
1719                         g_string_append (res, ", ");
1720                 g_free (param);
1721         }
1722         g_string_append (res, ")");
1723         
1724         /*
1725          * cleanup and return
1726          */
1727         s = res->str;
1728         g_string_free (res, FALSE);
1729         return s;
1730 }
1731
1732 /**
1733  * Returns a stringifed representation of a field ref
1734  */
1735 char *
1736 get_fieldref_signature (MonoImage *m, int idx, MonoGenericContainer *container)
1737 {
1738         guint32 cols [MONO_MEMBERREF_SIZE];
1739         MonoGenericContainer *new_container;
1740         char *type, *esname;
1741         char *sig;
1742         char *full_sig;
1743
1744         mono_metadata_decode_row (&m->tables [MONO_TABLE_MEMBERREF],
1745                                   idx - 1, cols, MONO_MEMBERREF_SIZE);
1746
1747         new_container = get_memberref_container (m, cols [MONO_MEMBERREF_CLASS], container);
1748         sig = get_field_signature (m, cols [MONO_MEMBERREF_SIGNATURE], new_container);
1749
1750         type = get_memberref_parent (m, cols [MONO_MEMBERREF_CLASS], container);
1751         esname = get_escaped_name (mono_metadata_string_heap (m, cols [MONO_MEMBERREF_NAME]));
1752
1753         full_sig = g_strdup_printf ("%s %s%s%s", 
1754                         sig, 
1755                         type ? type : "", 
1756                         type ? "::" : "",
1757                         esname);
1758         g_free (sig);
1759         g_free (type);
1760         g_free (esname);
1761         
1762         return full_sig;
1763 }
1764
1765 /**
1766  * get_field:
1767  * @m: metadata context
1768  * @token: a FIELD_DEF token
1769  *
1770  * This routine has to locate the TypeDef that "owns" this Field.
1771  * Since there is no backpointer in the Field table, we have to scan
1772  * the TypeDef table and locate the actual "owner" of the field
1773  */
1774 char *
1775 get_field (MonoImage *m, guint32 token, MonoGenericContainer *container)
1776 {
1777         int idx = mono_metadata_token_index (token);
1778         guint32 cols [MONO_FIELD_SIZE];
1779         char *sig, *res, *type, *estype, *esname;
1780         guint32 type_idx;
1781
1782         /*
1783          * We can get here also with a MenberRef token (for a field
1784          * defined in another module/assembly, just like in get_method ()
1785          */
1786         if (mono_metadata_token_code (token) == MONO_TOKEN_MEMBER_REF) {
1787                 return get_fieldref_signature (m, idx, container);
1788         }
1789         g_assert (mono_metadata_token_code (token) == MONO_TOKEN_FIELD_DEF);
1790
1791         mono_metadata_decode_row (&m->tables [MONO_TABLE_FIELD], idx - 1, cols, MONO_FIELD_SIZE);
1792         sig = get_field_signature (m, cols [MONO_FIELD_SIGNATURE], container);
1793
1794         /*
1795          * To locate the actual "container" for this field, we have to scan
1796          * the TypeDef table.  LAME!
1797          */
1798         type_idx = mono_metadata_typedef_from_field (m, idx);
1799         if (!type_idx) {
1800                 res = g_strdup_printf ("<invalid> %s", sig);
1801                 g_free (sig);
1802                 return res;
1803         }
1804
1805         type = get_typedef (m, type_idx);
1806         estype = get_escaped_name (type);
1807         esname = get_escaped_name (mono_metadata_string_heap (m, cols [MONO_FIELD_NAME]));
1808         res = g_strdup_printf ("%s %s%s%s",
1809                         sig, 
1810                         estype ? estype : "",
1811                         estype ? "::" : "",
1812                         esname);
1813
1814         g_free (type);
1815         g_free (sig);
1816         g_free (estype);
1817         g_free (esname);
1818
1819         return res;
1820 }
1821
1822 static MonoGenericContainer *
1823 get_memberref_container (MonoImage *m, guint32 mrp_token, MonoGenericContainer *container)
1824 {
1825         MonoClass *klass;
1826
1827         /*
1828          * mrp_index is a MemberRefParent coded index
1829          */
1830         guint32 table = mrp_token & 7;
1831         guint32 idx = mrp_token >> 3;
1832
1833         switch (table){
1834         case 0: /* TypeDef */
1835                 return mono_metadata_load_generic_params (m, MONO_TOKEN_TYPE_DEF | idx, NULL);
1836                 
1837         case 1: /* TypeRef */
1838                 return NULL;
1839                 
1840         case 4: /* TypeSpec */
1841                 klass = mono_class_get_full (m, MONO_TOKEN_TYPE_SPEC | idx, (MonoGenericContext *) container);
1842                 g_assert (klass);
1843                 return klass->generic_class ? klass->generic_class->container_class->generic_container : NULL;
1844         }
1845         g_assert_not_reached ();
1846         return NULL;
1847 }
1848
1849 static char *
1850 get_memberref_parent (MonoImage *m, guint32 mrp_token, MonoGenericContainer *container)
1851 {
1852         /*
1853          * mrp_index is a MemberRefParent coded index
1854          */
1855         guint32 table = mrp_token & 7;
1856         guint32 idx = mrp_token >> 3;
1857
1858         switch (table){
1859         case 0: /* TypeDef */
1860                 return get_typedef (m, idx);
1861                 
1862         case 1: /* TypeRef */
1863                 return get_typeref (m, idx);
1864                 
1865         case 2: /* ModuleRef */
1866                 return g_strdup_printf ("TODO:MemberRefParent-ModuleRef");
1867                 
1868         case 3: /* MethodDef */
1869                 return g_strdup ("TODO:MethodDef");
1870                 
1871         case 4: /* TypeSpec */
1872                 return get_typespec (m, idx, FALSE, container);
1873         }
1874         g_assert_not_reached ();
1875         return NULL;
1876 }
1877
1878 /**
1879  * get_method:
1880  * @m: metadata context
1881  * @token: a METHOD_DEF or MEMBER_REF token
1882  *
1883  * This routine has to locate the TypeDef that "owns" this Field.
1884  * Since there is no backpointer in the Field table, we have to scan
1885  * the TypeDef table and locate the actual "owner" of the field
1886  */
1887 static char *
1888 get_method_core (MonoImage *m, guint32 token, gboolean fullsig, MonoGenericContainer *container)
1889 {
1890         int idx = mono_metadata_token_index (token);
1891         guint32 member_cols [MONO_MEMBERREF_SIZE], method_cols [MONO_METHOD_SIZE];
1892         char *sig = NULL, *esname;
1893         char *name;
1894
1895         MonoMethod *mh;
1896         MonoGenericContainer *type_container = container;
1897
1898         mh = mono_get_method_full (m, token, NULL, (MonoGenericContext *) container);
1899         if (mh) {
1900                 if (mono_method_signature (mh)->is_inflated)
1901                         container = mono_method_get_generic_container (((MonoMethodInflated *) mh)->declaring);
1902                 esname = get_escaped_name (mh->name);
1903                 sig = dis_stringify_type (m, &mh->klass->byval_arg, TRUE);
1904                 if (show_tokens)
1905                         name = g_strdup_printf ("%s/*%08x*/%s%s", sig ? sig : "", token, sig ? "::" : "", esname);
1906                 else
1907                         name = g_strdup_printf ("%s%s%s", sig ? sig : "", sig ? "::" : "", esname);
1908                 g_free (sig);
1909                 g_free (esname);
1910         } else
1911                 name = NULL;
1912
1913         switch (mono_metadata_token_code (token)){
1914         case MONO_TOKEN_METHOD_DEF:
1915                 mono_metadata_decode_row (&m->tables [MONO_TABLE_METHOD], 
1916                                           idx - 1, method_cols, MONO_METHOD_SIZE);
1917
1918                 sig = get_methodref_signature (m, method_cols [MONO_METHOD_SIGNATURE], name);
1919                 break;
1920                 
1921         case MONO_TOKEN_MEMBER_REF: {
1922                 mono_metadata_decode_row (&m->tables [MONO_TABLE_MEMBERREF],
1923                                           idx - 1, member_cols, MONO_MEMBERREF_SIZE);
1924                 if (!name) {
1925                         char *parent = get_memberref_parent (m, member_cols [MONO_MEMBERREF_CLASS], container);
1926                         name = g_strdup_printf ("%s%s%s",
1927                                                 parent ? parent : "", 
1928                                                 parent ? "::" : "",
1929                                                 mono_metadata_string_heap (m, member_cols [MONO_MEMBERREF_NAME]));
1930                         g_free (parent);
1931                 }
1932                 if (mh) {
1933                         int arity = 0;
1934
1935                         if (mh->is_generic)
1936                                 arity = mono_method_get_generic_container (mh)->type_argc;
1937                         else
1938                         if (mh->is_inflated && ((MonoMethodInflated *)mh)->declaring->is_generic)
1939                                 arity = mono_method_get_generic_container (((MonoMethodInflated*) mh)->declaring)->type_argc;
1940
1941                         if (arity > 0) {
1942                                 char *str = g_strdup_printf ("%s <[%d]>", name, arity);
1943                                 g_free (name);
1944                                 name = str;
1945                         }
1946                 }
1947                                 
1948                 sig = get_methodref_signature (
1949                         m, member_cols [MONO_MEMBERREF_SIGNATURE], name);
1950                 break;
1951         }
1952         case MONO_TOKEN_METHOD_SPEC: {
1953                 mono_metadata_decode_row (&m->tables [MONO_TABLE_METHODSPEC],
1954                                 idx - 1, member_cols, MONO_METHODSPEC_SIZE);
1955                 token = member_cols [MONO_METHODSPEC_METHOD];
1956                 sig = get_methodspec (m, idx, token, name, type_container);
1957                 break;
1958         }
1959
1960         default:
1961                 g_assert_not_reached ();
1962         }
1963
1964         if (fullsig)
1965                 g_free (name);
1966         else {
1967                 g_free (sig);
1968                 return name;
1969         }
1970         
1971         if (show_tokens) {
1972                 char *retval = g_strdup_printf ("%s /* %08x */", sig, token);
1973                 g_free (sig);
1974                 return retval;
1975         } else
1976                 return sig;
1977 }
1978
1979 char *
1980 get_method (MonoImage *m, guint32 token, MonoGenericContainer *container)
1981 {
1982         return get_method_core (m, token, TRUE, container);
1983 }
1984
1985 /**
1986  * get_methoddef
1987  * @m: metadata context
1988  * @idx: index into the method table
1989  *
1990  * Returns: A stringified version of the method signature.
1991  */
1992 char *
1993 get_methoddef (MonoImage *m, guint32 idx)
1994 {
1995         guint32 cols [MONO_METHOD_SIZE];
1996         char *sig;
1997         const char *name;
1998
1999         MonoMethod *mh;
2000
2001         mh = mono_get_method (m, MONO_TOKEN_METHOD_DEF | idx, NULL);
2002         if (mh) {
2003                 sig = dis_stringify_type (m, &mh->klass->byval_arg, FALSE);
2004                 name = g_strdup_printf ("%s%s%s", 
2005                                 sig ? sig : "",
2006                                 sig ? "::" : "",
2007                                 mh->name);
2008                 g_free (sig);
2009         } else
2010                 name = NULL;
2011         mono_metadata_decode_row (&m->tables [MONO_TABLE_METHOD], 
2012                         idx - 1, cols, MONO_METHOD_SIZE);
2013         sig = get_methodref_signature (m, cols [MONO_METHOD_SIGNATURE], name);
2014         
2015         return sig;
2016 }
2017
2018 char *
2019 get_method_type_param (MonoImage *m, guint32 blob_signature, MonoGenericContainer *container)
2020 {
2021         GString *res = g_string_new ("");
2022         const char *ptr = mono_metadata_blob_heap (m, blob_signature);
2023         int param_count;
2024         int i = 0;
2025         char *s;
2026         
2027         mono_metadata_decode_value (ptr, &ptr);
2028         ptr++;
2029         param_count = mono_metadata_decode_value (ptr, &ptr);
2030         
2031         g_string_append_c (res, '<');
2032         
2033         for (i = 0; i < param_count; i++){
2034                 char *param = NULL;
2035
2036                 ptr = get_param (m, ptr, &param, container);
2037                 g_string_append (res, param);
2038                 if (i+1 != param_count)
2039                         g_string_append (res, ", ");
2040                 g_free (param);
2041         }
2042         g_string_append_c (res, '>');
2043         
2044         s = res->str;
2045         g_string_free (res, FALSE);
2046         return s;
2047 }
2048
2049 /**
2050  * get_methodspec
2051  *
2052  * Returns: An allocated stringified version of the methodspec signature.
2053  */
2054
2055 char *
2056 get_methodspec (MonoImage *m, int idx, guint32 token, const char *fancy_name, MonoGenericContainer *type_container)
2057 {
2058         GString *res = g_string_new ("");
2059         guint32 member_cols [MONO_MEMBERREF_SIZE], method_cols [MONO_METHOD_SIZE];
2060         char *s, *type_param;
2061         const char *ptr;
2062         guint32 sig = 0;
2063         int param_count, cconv, i, gen_count = 0;
2064         MonoGenericContainer *container;
2065         MonoMethod *mh = NULL;
2066
2067         switch (token & MONO_METHODDEFORREF_MASK) {
2068         case MONO_METHODDEFORREF_METHODDEF:
2069                 mono_metadata_decode_row (&m->tables [MONO_TABLE_METHOD], 
2070                                           (token >> MONO_METHODDEFORREF_BITS) - 1,
2071                                           method_cols, MONO_METHOD_SIZE);
2072                 sig = method_cols [MONO_METHOD_SIGNATURE];
2073                 break;
2074         case MONO_METHODDEFORREF_METHODREF:
2075                 mono_metadata_decode_row (&m->tables [MONO_TABLE_MEMBERREF], 
2076                                           (token >> MONO_METHODDEFORREF_BITS) - 1,
2077                                           member_cols, MONO_MEMBERREF_SIZE);
2078                 sig = member_cols [MONO_MEMBERREF_SIGNATURE];
2079                 break;
2080         default:
2081                 g_assert_not_reached ();
2082         }
2083
2084         ptr = mono_metadata_blob_heap (m, sig);
2085         mono_metadata_decode_value (ptr, &ptr);
2086         
2087         mh = mono_get_method_full (m, method_dor_to_token (token), NULL, (MonoGenericContext *) type_container);
2088         g_assert (mh);
2089         container = mono_method_get_generic_container (mh);
2090         if (!container)
2091                 container = type_container;
2092
2093         if (*ptr & 0x20){
2094                 if (*ptr & 0x40)
2095                         g_string_append (res, "explicit-this ");
2096                 else
2097                         g_string_append (res, "instance "); /* has-this */
2098         }
2099
2100         if (*ptr & 0x10)
2101                 gen_count = 1;
2102         cconv = *ptr & 0x0f;
2103         ptr++;
2104         if (gen_count)
2105                 gen_count = mono_metadata_decode_value (ptr, &ptr);
2106         param_count = mono_metadata_decode_value (ptr, &ptr);
2107         if (cconv != 0xa) {
2108                 char *allocated_ret_type;
2109                 ptr = get_ret_type (m, ptr, &allocated_ret_type, container);
2110                 g_string_append (res, allocated_ret_type);
2111                 g_free (allocated_ret_type);
2112         }
2113
2114         if (fancy_name){
2115                 g_string_append_c (res, ' ');
2116                 g_string_append (res, fancy_name);
2117         }
2118
2119         mono_metadata_decode_row (&m->tables [MONO_TABLE_METHODSPEC],
2120                         idx - 1, member_cols, MONO_METHODSPEC_SIZE);
2121         token = member_cols [MONO_METHODSPEC_SIGNATURE];
2122         type_param = get_method_type_param (m, token, type_container);
2123         g_string_append (res, type_param);
2124         g_string_append (res, " (");
2125
2126         /*
2127          * methodspecs can not be varargs so we don't need to worry about that here
2128          */
2129          
2130         for (i = 0; i < param_count; i++){
2131                 char *param = NULL;
2132                 
2133                 ptr = get_param (m, ptr, &param, container);
2134                 g_string_append (res, param);
2135                 if (i+1 != param_count)
2136                         g_string_append (res, ", ");
2137                 g_free (param);
2138         }
2139         g_string_append (res, ")");
2140         
2141         /*
2142          * cleanup and return
2143          */
2144         s = res->str;
2145         g_string_free (res, FALSE);
2146         return s;
2147 }
2148
2149 /*
2150  * get_encoded_user_string_bytearray:
2151  * @ptr: pointer into the US heap
2152  * @len: length of string in the heap.
2153  *
2154  * Strings on the US heap are encoded using UTF-16. Print a bytearray.
2155  */
2156 static char*
2157 get_encoded_user_string_bytearray (const unsigned char* ptr, int len)
2158 {
2159         gchar *str;
2160         int i, j, tspaces = (len % 16);
2161         GString *res;
2162
2163         if (len == 0)
2164                 return g_strdup_printf ("\"\"");
2165
2166         res = g_string_new ("bytearray (\n\t");
2167
2168         for (i = 1; i <= len; ++i) {
2169                 g_string_append_printf (res, "%02x ", ptr [i-1]);
2170
2171                 if (i % 16 == 0) {
2172                         if (i == len)
2173                                 g_string_append (res, ")// ");
2174                         else
2175                                 g_string_append (res, " // ");
2176
2177                         for(j = i - 16; j < i; ++j) 
2178                                 g_string_append_printf (res, "%c", isprint (ptr [j]) ? ptr [j] : '.');
2179                         g_string_append (res, "\n\t");
2180                 }
2181         }
2182
2183         if (tspaces) {
2184                 g_string_append (res, ")  ");
2185                 for (i = tspaces + 1; i < 16; ++i)
2186                         g_string_append_printf (res, "   ");
2187
2188                 g_string_append (res, " // ");
2189                 for(i = len - tspaces; i < len; ++i)
2190                         g_string_append_printf (res, "%c", isprint (ptr [i]) ? ptr [i] : '.');
2191                 g_string_append (res, "\n\t");
2192         } 
2193
2194         str = res->str;
2195         g_string_free (res, FALSE);
2196         return str;
2197 }
2198
2199 /*
2200  * get_encoded_user_string_or_bytearray:
2201  * @ptr: pointer into the US heap
2202  *
2203  * Strings on the US heap are encoded using UTF-16. Print as string
2204  * if possible, else emit a bytearray.
2205  */
2206 char*
2207 get_encoded_user_string_or_bytearray (const unsigned char *ptr, int len)
2208 {
2209         char *res, *eres, *result;
2210         int i;
2211
2212         res = g_malloc ((len >> 1) + 1);
2213
2214         /*
2215          * I should really use some kind of libunicode here
2216          */
2217         for (i = 0; i + 1 < len; i += 2) {
2218                 if (ptr [i + 1] || 
2219                     (!isprint (ptr [i]) && ptr [i] != '\\' && ptr [i] != '"' && 
2220                      ptr [i] != '\r' && ptr [i] != '\n' && ptr [i] != '\t')) {
2221                         g_free (res);
2222                         return get_encoded_user_string_bytearray (ptr, len);
2223                 }
2224
2225                 res [i >> 1] = ptr [i];
2226         }       
2227
2228         res [len >> 1] = 0;
2229
2230         eres = g_strescape (res, NULL);
2231         result = g_strdup_printf ("\"%s\"", eres);
2232         g_free (res);
2233         g_free (eres);
2234         
2235         return result;
2236 }
2237
2238 char *
2239 stringify_double (double r)
2240 {
2241         char *ret, *ptr;
2242
2243         ret = g_strdup_printf ("%.17g.", r);
2244         ptr = ret + strlen (ret) - 1;
2245         if (strpbrk (ret, ".eE") != ptr)
2246                 *ptr = '\0';
2247
2248         return ret;
2249 }
2250
2251 /**
2252  * get_constant:
2253  * @m: metadata context
2254  * @blob_index: index into the blob where the constant is stored
2255  *
2256  * Returns: An allocated value representing a stringified version of the
2257  * constant.
2258  */
2259 char *
2260 get_constant (MonoImage *m, MonoTypeEnum t, guint32 blob_index)
2261 {
2262         const char *ptr = mono_metadata_blob_heap (m, blob_index);
2263         int len;
2264         
2265         len = mono_metadata_decode_value (ptr, &ptr);
2266         
2267         switch (t){
2268         case MONO_TYPE_BOOLEAN:
2269                 return g_strdup_printf ("%s", *ptr ? "bool(true)" : "bool(false)");
2270                 
2271         case MONO_TYPE_CHAR:
2272                 return g_strdup_printf ("char(0x%04x)", read16(ptr)); 
2273                 
2274         case MONO_TYPE_U1:
2275         case MONO_TYPE_I1:
2276                 return g_strdup_printf ("int8(0x%02x)", (int) ((*ptr) & 0xFF));
2277                 break;
2278                 
2279         case MONO_TYPE_U2:
2280         case MONO_TYPE_I2:
2281                 return g_strdup_printf ("int16(0x%08x)", (int) read16 (ptr));
2282                 
2283         case MONO_TYPE_U4:
2284         case MONO_TYPE_I4:
2285                 return g_strdup_printf ("int32(0x%08x)", read32 (ptr));
2286                 
2287         case MONO_TYPE_U8:
2288         case MONO_TYPE_I8: {
2289                 guint32 low, high;
2290                 low = read32 (ptr);
2291                 high = read32 (ptr + 4);
2292                 return g_strdup_printf ("int64(0x%08x%08x)", high, low);
2293         }
2294         case MONO_TYPE_R4: {
2295                 gboolean normal;
2296                 float r;
2297                 readr4 (ptr, &r);
2298
2299                 /* Crazy solaris systems doesn't have isnormal */
2300 #ifdef HAVE_FINITE
2301                 normal = finite (r);
2302 #else
2303                 normal = isnormal (r);
2304 #endif
2305                 if (!normal) {
2306                         return g_strdup_printf ("float32(0x%08x)", read32 (ptr));
2307                 } else {
2308                         char *str = stringify_double ((double) r);
2309                         char *ret = g_strdup_printf ("float32(%s)", str);
2310                         g_free (str);
2311                         return ret;
2312                 }
2313         }       
2314         case MONO_TYPE_R8: {
2315                 gboolean normal;
2316                 double r;
2317                 readr8 (ptr, &r);
2318
2319                 /* Crazy solaris systems doesn't have isnormal */
2320 #ifdef HAVE_FINITE
2321                 normal = finite (r);
2322 #else
2323                 normal = isnormal (r);
2324 #endif
2325                 if (!normal) {
2326                         guint32 low, high;
2327                         low = read32 (ptr);
2328                         high = read32 (ptr + 4);
2329                         return g_strdup_printf ("float64(0x%08x%08x)", high, low);
2330                 } else {
2331                         char *str = stringify_double (r);
2332                         char *ret = g_strdup_printf ("float64(%s)", str);
2333                         g_free (str);
2334                         return ret;
2335                 }
2336         }
2337         case MONO_TYPE_STRING:
2338                 return get_encoded_user_string_or_bytearray ((const guchar*)ptr, len);
2339                 
2340         case MONO_TYPE_CLASS:
2341                 return g_strdup ("nullref");
2342                 
2343         default:
2344                 g_error ("Unknown MONO_TYPE (%d) on constant at Blob index (0x%08x)\n",
2345                          (int) *ptr, blob_index);
2346                 return g_strdup_printf ("Unknown");
2347         }
2348
2349 }
2350
2351 /**
2352  * get_token:
2353  * @m: metadata context
2354  * @token: token that we want to decode.
2355  *
2356  * Returns: An allocated value representing a stringified version of the
2357  * constant.
2358  */
2359 char *
2360 get_token (MonoImage *m, guint32 token, MonoGenericContainer *container)
2361 {
2362         char *temp, *result;
2363         guint32 idx = mono_metadata_token_index (token);
2364
2365         switch (mono_metadata_token_code (token)){
2366         case MONO_TOKEN_FIELD_DEF:
2367                 temp = get_field (m, token, container);
2368                 result = g_strdup_printf ("field %s", temp);
2369                 g_free (temp);
2370                 return result;
2371         case MONO_TOKEN_METHOD_DEF:
2372         case MONO_TOKEN_METHOD_SPEC:
2373                 temp = get_method (m, token, container);
2374                 result = g_strdup_printf ("method %s", temp);
2375                 g_free (temp);
2376                 return result;
2377         case MONO_TOKEN_TYPE_DEF:
2378                 temp = get_typedef (m, idx);
2379                 result = get_escaped_name (temp);
2380                 g_free (temp);
2381                 return result;
2382         case MONO_TOKEN_TYPE_REF:
2383                 return get_typeref (m, idx);
2384         case MONO_TOKEN_TYPE_SPEC:
2385                 return get_typespec (m, idx, TRUE, container);
2386         case MONO_TOKEN_MEMBER_REF: {
2387                 guint32 cols [MONO_MEMBERREF_SIZE];
2388                 const char *sig;
2389                 mono_metadata_decode_row (&m->tables [MONO_TABLE_MEMBERREF], mono_metadata_token_index (token) - 1, cols, MONO_MEMBERREF_SIZE);
2390                 sig = mono_metadata_blob_heap (m, cols [MONO_MEMBERREF_SIGNATURE]);
2391                 mono_metadata_decode_blob_size (sig, &sig);
2392                 if (*sig == 0x6) { /* it's a field */
2393                         temp = get_field (m, token, container);
2394                         result = g_strdup_printf ("field %s", temp);
2395                         g_free (temp);
2396                         return result;
2397                 } else {
2398                         temp = get_method (m, token, container);
2399                         result = g_strdup_printf ("method %s", temp);
2400                         g_free (temp);
2401                         return result;
2402                 }
2403                 break;
2404         }
2405         default:                
2406                 g_error ("Do not know how to decode tokens of type 0x%08x", token);
2407         }
2408
2409         g_assert_not_reached ();
2410         return g_strdup ("ERROR");
2411 }
2412
2413 /**
2414  * get_token_type:
2415  * @m: metadata context
2416  * @token: the token can belong to any of the following tables:
2417  * MONO_TOKEN_TYPE_REF, MONO_TOKEN_TYPE_DEF, MONO_TOKEN_TYPE_SPEC
2418  *
2419  * Returns: a stringified version of the MethodDef or MethodRef or TypeSpecn
2420  * at (token & 0xffffff) 
2421  */
2422 char *
2423 get_token_type (MonoImage *m, guint32 token, MonoGenericContainer *container)
2424 {
2425         char *temp = NULL, *s = NULL;
2426         int idx;
2427
2428         idx = mono_metadata_token_index (token);
2429         
2430         switch (mono_metadata_token_code (token)){
2431         case MONO_TOKEN_TYPE_DEF:
2432                 temp = get_typedef (m, idx);
2433                 s = g_strdup_printf ("%s", temp);
2434                 break;
2435                 
2436         case MONO_TOKEN_TYPE_REF: 
2437                 temp = get_typeref (m, idx);
2438                 s = g_strdup_printf ("%s", temp);
2439                 break;
2440                 
2441         case MONO_TOKEN_TYPE_SPEC:
2442                 s = get_typespec (m, idx, FALSE, container);
2443                 break;
2444
2445         default:
2446                 g_error ("Unhandled encoding for token 0x%08x", token);
2447
2448         }
2449         
2450         if (temp)
2451                 g_free (temp);
2452
2453         return s;
2454 }
2455
2456 char *
2457 get_guid (MonoImage *m, guint32 guid_index)
2458 {
2459         const unsigned char *guid;
2460         char *result;
2461
2462         guid = (const guchar*)mono_metadata_guid_heap (m, guid_index);
2463
2464         result = g_strdup_printf ("{%02X%02X%02X%02X-%02X%02X-%02X%02X-%02X%02X-%02X%02X%02X%02X%02X%02X}", 
2465                         guid [3], guid [2], guid [1], guid [0], guid [5], guid [4], guid [7], guid [6],
2466                         guid [8], guid [9], guid [10], guid [11], guid [12], guid [13], guid [14], guid [15]);
2467         return result;
2468 }
2469
2470 GList *
2471 dis_get_custom_attrs (MonoImage *m, guint32 token)
2472 {
2473         GList *list = NULL;
2474         guint32 idx, i, len, mtoken;
2475         guint32 cols [MONO_CUSTOM_ATTR_SIZE];
2476         MonoTableInfo *ca;
2477         char *method;
2478         GString *attr;
2479         const char *val;
2480
2481         idx = mono_metadata_token_index (token);
2482         idx <<= MONO_CUSTOM_ATTR_BITS;
2483         
2484         switch (mono_metadata_token_table (token)) {
2485         case MONO_TABLE_TYPEDEF:
2486                 idx |= MONO_CUSTOM_ATTR_TYPEDEF;
2487                 break;
2488         case MONO_TABLE_ASSEMBLY:
2489                 idx |= MONO_CUSTOM_ATTR_ASSEMBLY;
2490                 break;
2491         case MONO_TABLE_ASSEMBLYREF:
2492                 idx |= MONO_CUSTOM_ATTR_ASSEMBLYREF;
2493                 break;
2494         case MONO_TABLE_MODULE:
2495                 idx |= MONO_CUSTOM_ATTR_MODULE;
2496                 break;
2497         case MONO_TABLE_PROPERTY:
2498                 idx |= MONO_CUSTOM_ATTR_PROPERTY;
2499                 break;
2500         case MONO_TABLE_EVENT:
2501                 idx |= MONO_CUSTOM_ATTR_EVENT;
2502                 break;
2503         case MONO_TABLE_FIELD:
2504                 idx |= MONO_CUSTOM_ATTR_FIELDDEF;
2505                 break;
2506         case MONO_TABLE_METHOD:
2507                 idx |= MONO_CUSTOM_ATTR_METHODDEF;
2508                 break;
2509         case MONO_TABLE_PARAM:
2510                 idx |= MONO_CUSTOM_ATTR_PARAMDEF;
2511                 break;
2512         case MONO_TABLE_GENERICPARAM:
2513                 idx |= MONO_CUSTOM_ATTR_GENERICPAR;
2514                 break;
2515         default:
2516                 g_print ("Missing custom attr get support for token 0x%08x\n", token);
2517                 return NULL;
2518         }
2519
2520         ca = &m->tables [MONO_TABLE_CUSTOMATTRIBUTE];
2521         /* the table is not sorted */
2522         for (i = 0; i < ca->rows; ++i) {
2523                 char *dump;
2524                 mono_metadata_decode_row (ca, i, cols, MONO_CUSTOM_ATTR_SIZE);
2525                 if (cols [MONO_CUSTOM_ATTR_PARENT] != idx)
2526                         continue;
2527                 mtoken = cols [MONO_CUSTOM_ATTR_TYPE] >> MONO_CUSTOM_ATTR_TYPE_BITS;
2528                 switch (cols [MONO_CUSTOM_ATTR_TYPE] & MONO_CUSTOM_ATTR_TYPE_MASK) {
2529                 case MONO_CUSTOM_ATTR_TYPE_METHODDEF:
2530                         mtoken |= MONO_TOKEN_METHOD_DEF;
2531                         break;
2532                 case MONO_CUSTOM_ATTR_TYPE_MEMBERREF:
2533                         mtoken |= MONO_TOKEN_MEMBER_REF;
2534                         break;
2535                 default:
2536                         g_error ("Unknown table for custom attr type %08x", cols [MONO_CUSTOM_ATTR_TYPE]);
2537                         break;
2538                 }
2539                 method = get_method (m, mtoken, NULL);
2540                 val = mono_metadata_blob_heap (m, cols [MONO_CUSTOM_ATTR_VALUE]);
2541                 len = mono_metadata_decode_value (val, &val);
2542                 attr = g_string_new (".custom ");
2543                 dump = data_dump (val, len, "\t\t");
2544                 g_string_append_printf (attr, "%s = %s", method, dump);
2545                 g_free (dump);
2546                 list = g_list_append (list, attr->str);
2547                 g_string_free (attr, FALSE);
2548                 g_free (method);
2549         }
2550         return list;
2551 }
2552
2553 char*
2554 get_marshal_info (MonoImage *m, const char *blob) {
2555         int len, size = 0;
2556
2557         len = mono_metadata_decode_blob_size (blob, &blob);
2558
2559         switch (*blob) {
2560         case MONO_NATIVE_BOOLEAN:
2561                 return g_strdup ("bool");
2562         case MONO_NATIVE_I1:
2563                 return g_strdup ("int8");
2564         case MONO_NATIVE_U1:
2565                 return g_strdup ("unsigned int8");
2566         case MONO_NATIVE_I2:
2567                 return g_strdup ("int16");
2568         case MONO_NATIVE_U2:
2569                 return g_strdup ("unsigned int16");
2570         case MONO_NATIVE_I4:
2571                 return g_strdup ("int32");
2572         case MONO_NATIVE_U4:
2573                 return g_strdup ("unsigned int32");
2574         case MONO_NATIVE_I8:
2575                 return g_strdup ("int64");
2576         case MONO_NATIVE_U8:
2577                 return g_strdup ("unsigned int64");
2578         case MONO_NATIVE_R4:
2579                 return g_strdup ("float32");
2580         case MONO_NATIVE_R8:
2581                 return g_strdup ("float64");
2582         case MONO_NATIVE_CURRENCY:
2583                 return g_strdup ("currency");
2584         case MONO_NATIVE_BSTR:
2585                 return g_strdup ("bstr");
2586         case MONO_NATIVE_LPSTR:
2587                 return g_strdup ("lpstr");
2588         case MONO_NATIVE_LPWSTR:
2589                 return g_strdup ("lpwstr");
2590         case MONO_NATIVE_LPTSTR:
2591                 return g_strdup ("lptstr");
2592         case MONO_NATIVE_BYVALTSTR:
2593                 size = mono_metadata_decode_value (blob + 1, &blob);
2594                 return g_strdup_printf ("fixed sysstring [%d]", size);
2595         case MONO_NATIVE_IUNKNOWN:
2596                 return g_strdup ("iunknown");
2597         case MONO_NATIVE_IDISPATCH:
2598                 return g_strdup ("idispatch");
2599         case MONO_NATIVE_STRUCT:
2600                 return g_strdup ("struct");
2601         case MONO_NATIVE_INTERFACE:
2602                 return g_strdup ("interface");
2603         case MONO_NATIVE_SAFEARRAY:
2604                 return g_strdup ("safearray");
2605         case MONO_NATIVE_BYVALARRAY:
2606                 size = mono_metadata_decode_value (blob + 1, &blob);
2607                 return g_strdup_printf ("fixed array [%d]", size);
2608         case MONO_NATIVE_INT:
2609                 return g_strdup ("int");
2610         case MONO_NATIVE_UINT:
2611                 return g_strdup ("unsigned int");
2612         case MONO_NATIVE_VBBYREFSTR:
2613                 return g_strdup ("vbbyrefstr");
2614         case MONO_NATIVE_ANSIBSTR:
2615                 return g_strdup ("ansi bstr");
2616         case MONO_NATIVE_TBSTR:
2617                 return g_strdup ("tbstr");
2618         case MONO_NATIVE_VARIANTBOOL:
2619                 return g_strdup ("variant bool");
2620         case MONO_NATIVE_FUNC:
2621                 return g_strdup ("method");
2622         case MONO_NATIVE_ASANY:
2623                 return g_strdup ("as any");
2624         case MONO_NATIVE_LPARRAY:
2625                 return g_strdup ("[]");
2626         case MONO_NATIVE_LPSTRUCT:
2627                 return g_strdup ("lpstruct");
2628         case MONO_NATIVE_CUSTOM:
2629                 return g_strdup ("custom");
2630         case MONO_NATIVE_ERROR:
2631                 return g_strdup ("error");
2632         default:
2633                 return g_strdup ("unknown");
2634         }
2635 }
2636
2637 void
2638 init_key_table (void)
2639 {
2640         key_table = g_hash_table_new (g_str_hash, g_str_equal);
2641
2642         g_hash_table_insert (key_table, (char *) "9", GINT_TO_POINTER (TRUE));
2643         g_hash_table_insert (key_table, (char *) "abstract", GINT_TO_POINTER (TRUE));
2644         g_hash_table_insert (key_table, (char *) "add", GINT_TO_POINTER (TRUE));
2645         g_hash_table_insert (key_table, (char *) "add.ovf", GINT_TO_POINTER (TRUE));
2646         g_hash_table_insert (key_table, (char *) "add.ovf.un", GINT_TO_POINTER (TRUE));
2647         g_hash_table_insert (key_table, (char *) "algorithm", GINT_TO_POINTER (TRUE));
2648         g_hash_table_insert (key_table, (char *) "alignment", GINT_TO_POINTER (TRUE));
2649         g_hash_table_insert (key_table, (char *) "and", GINT_TO_POINTER (TRUE));
2650         g_hash_table_insert (key_table, (char *) "ansi", GINT_TO_POINTER (TRUE));
2651         g_hash_table_insert (key_table, (char *) "any", GINT_TO_POINTER (TRUE));
2652         g_hash_table_insert (key_table, (char *) "arglist", GINT_TO_POINTER (TRUE));
2653         g_hash_table_insert (key_table, (char *) "array", GINT_TO_POINTER (TRUE));
2654         g_hash_table_insert (key_table, (char *) "as", GINT_TO_POINTER (TRUE));
2655         g_hash_table_insert (key_table, (char *) "assembly", GINT_TO_POINTER (TRUE));
2656         g_hash_table_insert (key_table, (char *) "assert", GINT_TO_POINTER (TRUE));
2657         g_hash_table_insert (key_table, (char *) "at", GINT_TO_POINTER (TRUE));
2658         g_hash_table_insert (key_table, (char *) "autochar", GINT_TO_POINTER (TRUE));
2659         g_hash_table_insert (key_table, (char *) "auto", GINT_TO_POINTER (TRUE));
2660         g_hash_table_insert (key_table, (char *) "beforefieldinit", GINT_TO_POINTER (TRUE));
2661         g_hash_table_insert (key_table, (char *) "bestfit", GINT_TO_POINTER (TRUE));
2662         g_hash_table_insert (key_table, (char *) "beq", GINT_TO_POINTER (TRUE));
2663         g_hash_table_insert (key_table, (char *) "beq.s", GINT_TO_POINTER (TRUE));
2664         g_hash_table_insert (key_table, (char *) "bge", GINT_TO_POINTER (TRUE));
2665         g_hash_table_insert (key_table, (char *) "bge.s", GINT_TO_POINTER (TRUE));
2666         g_hash_table_insert (key_table, (char *) "bge.un", GINT_TO_POINTER (TRUE));
2667         g_hash_table_insert (key_table, (char *) "bge.un.s", GINT_TO_POINTER (TRUE));
2668         g_hash_table_insert (key_table, (char *) "bgt", GINT_TO_POINTER (TRUE));
2669         g_hash_table_insert (key_table, (char *) "bgt.s", GINT_TO_POINTER (TRUE));
2670         g_hash_table_insert (key_table, (char *) "bgt.un", GINT_TO_POINTER (TRUE));
2671         g_hash_table_insert (key_table, (char *) "bgt.un.s", GINT_TO_POINTER (TRUE));
2672         g_hash_table_insert (key_table, (char *) "ble", GINT_TO_POINTER (TRUE));
2673         g_hash_table_insert (key_table, (char *) "ble.s", GINT_TO_POINTER (TRUE));
2674         g_hash_table_insert (key_table, (char *) "ble.un", GINT_TO_POINTER (TRUE));
2675         g_hash_table_insert (key_table, (char *) "ble.un.s", GINT_TO_POINTER (TRUE));
2676         g_hash_table_insert (key_table, (char *) "blob", GINT_TO_POINTER (TRUE));
2677         g_hash_table_insert (key_table, (char *) "blob_object", GINT_TO_POINTER (TRUE));
2678         g_hash_table_insert (key_table, (char *) "blt", GINT_TO_POINTER (TRUE));
2679         g_hash_table_insert (key_table, (char *) "blt.s", GINT_TO_POINTER (TRUE));
2680         g_hash_table_insert (key_table, (char *) "blt.un", GINT_TO_POINTER (TRUE));
2681         g_hash_table_insert (key_table, (char *) "blt.un.s", GINT_TO_POINTER (TRUE));
2682         g_hash_table_insert (key_table, (char *) "bne.un", GINT_TO_POINTER (TRUE));
2683         g_hash_table_insert (key_table, (char *) "bne.un.s", GINT_TO_POINTER (TRUE));
2684         g_hash_table_insert (key_table, (char *) "bool", GINT_TO_POINTER (TRUE));
2685         g_hash_table_insert (key_table, (char *) "box", GINT_TO_POINTER (TRUE));
2686         g_hash_table_insert (key_table, (char *) "break", GINT_TO_POINTER (TRUE));
2687         g_hash_table_insert (key_table, (char *) "brfalse", GINT_TO_POINTER (TRUE));
2688         g_hash_table_insert (key_table, (char *) "brfalse.s", GINT_TO_POINTER (TRUE));
2689         g_hash_table_insert (key_table, (char *) "br", GINT_TO_POINTER (TRUE));
2690         g_hash_table_insert (key_table, (char *) "brinst", GINT_TO_POINTER (TRUE));
2691         g_hash_table_insert (key_table, (char *) "brinst.s", GINT_TO_POINTER (TRUE));
2692         g_hash_table_insert (key_table, (char *) "brnull", GINT_TO_POINTER (TRUE));
2693         g_hash_table_insert (key_table, (char *) "brnull.s", GINT_TO_POINTER (TRUE));
2694         g_hash_table_insert (key_table, (char *) "br.s", GINT_TO_POINTER (TRUE));
2695         g_hash_table_insert (key_table, (char *) "brtrue", GINT_TO_POINTER (TRUE));
2696         g_hash_table_insert (key_table, (char *) "brtrue.s", GINT_TO_POINTER (TRUE));
2697         g_hash_table_insert (key_table, (char *) "brzero", GINT_TO_POINTER (TRUE));
2698         g_hash_table_insert (key_table, (char *) "brzero.s", GINT_TO_POINTER (TRUE));
2699         g_hash_table_insert (key_table, (char *) "bstr", GINT_TO_POINTER (TRUE));
2700         g_hash_table_insert (key_table, (char *) "bytearray", GINT_TO_POINTER (TRUE));
2701         g_hash_table_insert (key_table, (char *) "byvalstr", GINT_TO_POINTER (TRUE));
2702         g_hash_table_insert (key_table, (char *) "call", GINT_TO_POINTER (TRUE));
2703         g_hash_table_insert (key_table, (char *) "callconv", GINT_TO_POINTER (TRUE));
2704         g_hash_table_insert (key_table, (char *) "calli", GINT_TO_POINTER (TRUE));
2705         g_hash_table_insert (key_table, (char *) "callmostderived", GINT_TO_POINTER (TRUE));
2706         g_hash_table_insert (key_table, (char *) "callvirt", GINT_TO_POINTER (TRUE));
2707         g_hash_table_insert (key_table, (char *) "carray", GINT_TO_POINTER (TRUE));
2708         g_hash_table_insert (key_table, (char *) "castclass", GINT_TO_POINTER (TRUE));
2709         g_hash_table_insert (key_table, (char *) "catch", GINT_TO_POINTER (TRUE));
2710         g_hash_table_insert (key_table, (char *) "cdecl", GINT_TO_POINTER (TRUE));
2711         g_hash_table_insert (key_table, (char *) "ceq", GINT_TO_POINTER (TRUE));
2712         g_hash_table_insert (key_table, (char *) "cf", GINT_TO_POINTER (TRUE));
2713         g_hash_table_insert (key_table, (char *) "cgt", GINT_TO_POINTER (TRUE));
2714         g_hash_table_insert (key_table, (char *) "cgt.un", GINT_TO_POINTER (TRUE));
2715         g_hash_table_insert (key_table, (char *) "char", GINT_TO_POINTER (TRUE));
2716         g_hash_table_insert (key_table, (char *) "charmaperror", GINT_TO_POINTER (TRUE));
2717         g_hash_table_insert (key_table, (char *) "cil", GINT_TO_POINTER (TRUE));
2718         g_hash_table_insert (key_table, (char *) "ckfinite", GINT_TO_POINTER (TRUE));
2719         g_hash_table_insert (key_table, (char *) "class", GINT_TO_POINTER (TRUE));
2720         g_hash_table_insert (key_table, (char *) "clsid", GINT_TO_POINTER (TRUE));
2721         g_hash_table_insert (key_table, (char *) "clt", GINT_TO_POINTER (TRUE));
2722         g_hash_table_insert (key_table, (char *) "clt.un", GINT_TO_POINTER (TRUE));
2723         g_hash_table_insert (key_table, (char *) "Compilercontrolled", GINT_TO_POINTER (TRUE));
2724         g_hash_table_insert (key_table, (char *) "const", GINT_TO_POINTER (TRUE));
2725         g_hash_table_insert (key_table, (char *) "conv.i1", GINT_TO_POINTER (TRUE));
2726         g_hash_table_insert (key_table, (char *) "conv.i2", GINT_TO_POINTER (TRUE));
2727         g_hash_table_insert (key_table, (char *) "conv.i4", GINT_TO_POINTER (TRUE));
2728         g_hash_table_insert (key_table, (char *) "conv.i8", GINT_TO_POINTER (TRUE));
2729         g_hash_table_insert (key_table, (char *) "conv.i", GINT_TO_POINTER (TRUE));
2730         g_hash_table_insert (key_table, (char *) "conv.ovf.i1", GINT_TO_POINTER (TRUE));
2731         g_hash_table_insert (key_table, (char *) "conv.ovf.i1.un", GINT_TO_POINTER (TRUE));
2732         g_hash_table_insert (key_table, (char *) "conv.ovf.i2", GINT_TO_POINTER (TRUE));
2733         g_hash_table_insert (key_table, (char *) "conv.ovf.i2.un", GINT_TO_POINTER (TRUE));
2734         g_hash_table_insert (key_table, (char *) "conv.ovf.i4", GINT_TO_POINTER (TRUE));
2735         g_hash_table_insert (key_table, (char *) "conv.ovf.i4.un", GINT_TO_POINTER (TRUE));
2736         g_hash_table_insert (key_table, (char *) "conv.ovf.i8", GINT_TO_POINTER (TRUE));
2737         g_hash_table_insert (key_table, (char *) "conv.ovf.i8.un", GINT_TO_POINTER (TRUE));
2738         g_hash_table_insert (key_table, (char *) "conv.ovf.i", GINT_TO_POINTER (TRUE));
2739         g_hash_table_insert (key_table, (char *) "conv.ovf.i.un", GINT_TO_POINTER (TRUE));
2740         g_hash_table_insert (key_table, (char *) "conv.ovf.u1", GINT_TO_POINTER (TRUE));
2741         g_hash_table_insert (key_table, (char *) "conv.ovf.u1.un", GINT_TO_POINTER (TRUE));
2742         g_hash_table_insert (key_table, (char *) "conv.ovf.u2", GINT_TO_POINTER (TRUE));
2743         g_hash_table_insert (key_table, (char *) "conv.ovf.u2.un", GINT_TO_POINTER (TRUE));
2744         g_hash_table_insert (key_table, (char *) "conv.ovf.u4", GINT_TO_POINTER (TRUE));
2745         g_hash_table_insert (key_table, (char *) "conv.ovf.u4.un", GINT_TO_POINTER (TRUE));
2746         g_hash_table_insert (key_table, (char *) "conv.ovf.u8", GINT_TO_POINTER (TRUE));
2747         g_hash_table_insert (key_table, (char *) "conv.ovf.u8.un", GINT_TO_POINTER (TRUE));
2748         g_hash_table_insert (key_table, (char *) "conv.ovf.u", GINT_TO_POINTER (TRUE));
2749         g_hash_table_insert (key_table, (char *) "conv.ovf.u.un", GINT_TO_POINTER (TRUE));
2750         g_hash_table_insert (key_table, (char *) "conv.r4", GINT_TO_POINTER (TRUE));
2751         g_hash_table_insert (key_table, (char *) "conv.r8", GINT_TO_POINTER (TRUE));
2752         g_hash_table_insert (key_table, (char *) "conv.r.un", GINT_TO_POINTER (TRUE));
2753         g_hash_table_insert (key_table, (char *) "conv.u1", GINT_TO_POINTER (TRUE));
2754         g_hash_table_insert (key_table, (char *) "conv.u2", GINT_TO_POINTER (TRUE));
2755         g_hash_table_insert (key_table, (char *) "conv.u4", GINT_TO_POINTER (TRUE));
2756         g_hash_table_insert (key_table, (char *) "conv.u8", GINT_TO_POINTER (TRUE));
2757         g_hash_table_insert (key_table, (char *) "conv.u", GINT_TO_POINTER (TRUE));
2758         g_hash_table_insert (key_table, (char *) "cpblk", GINT_TO_POINTER (TRUE));
2759         g_hash_table_insert (key_table, (char *) "cpobj", GINT_TO_POINTER (TRUE));
2760         g_hash_table_insert (key_table, (char *) "currency", GINT_TO_POINTER (TRUE));
2761         g_hash_table_insert (key_table, (char *) "custom", GINT_TO_POINTER (TRUE));
2762         g_hash_table_insert (key_table, (char *) "date", GINT_TO_POINTER (TRUE));
2763         g_hash_table_insert (key_table, (char *) "decimal", GINT_TO_POINTER (TRUE));
2764         g_hash_table_insert (key_table, (char *) "default", GINT_TO_POINTER (TRUE));
2765         g_hash_table_insert (key_table, (char *) "demand", GINT_TO_POINTER (TRUE));
2766         g_hash_table_insert (key_table, (char *) "deny", GINT_TO_POINTER (TRUE));
2767         g_hash_table_insert (key_table, (char *) "div", GINT_TO_POINTER (TRUE));
2768         g_hash_table_insert (key_table, (char *) "div.un", GINT_TO_POINTER (TRUE));
2769         g_hash_table_insert (key_table, (char *) "dup", GINT_TO_POINTER (TRUE));
2770         g_hash_table_insert (key_table, (char *) "endfault", GINT_TO_POINTER (TRUE));
2771         g_hash_table_insert (key_table, (char *) "endfilter", GINT_TO_POINTER (TRUE));
2772         g_hash_table_insert (key_table, (char *) "endfinally", GINT_TO_POINTER (TRUE));
2773         g_hash_table_insert (key_table, (char *) "endmac", GINT_TO_POINTER (TRUE));
2774         g_hash_table_insert (key_table, (char *) "enum", GINT_TO_POINTER (TRUE));
2775         g_hash_table_insert (key_table, (char *) "error", GINT_TO_POINTER (TRUE));
2776         g_hash_table_insert (key_table, (char *) "explicit", GINT_TO_POINTER (TRUE));
2777         g_hash_table_insert (key_table, (char *) "extends", GINT_TO_POINTER (TRUE));
2778         g_hash_table_insert (key_table, (char *) "extern", GINT_TO_POINTER (TRUE));
2779         g_hash_table_insert (key_table, (char *) "false", GINT_TO_POINTER (TRUE));
2780         g_hash_table_insert (key_table, (char *) "famandassem", GINT_TO_POINTER (TRUE));
2781         g_hash_table_insert (key_table, (char *) "family", GINT_TO_POINTER (TRUE));
2782         g_hash_table_insert (key_table, (char *) "famorassem", GINT_TO_POINTER (TRUE));
2783         g_hash_table_insert (key_table, (char *) "fastcall", GINT_TO_POINTER (TRUE));
2784         g_hash_table_insert (key_table, (char *) "fault", GINT_TO_POINTER (TRUE));
2785         g_hash_table_insert (key_table, (char *) "field", GINT_TO_POINTER (TRUE));
2786         g_hash_table_insert (key_table, (char *) "filetime", GINT_TO_POINTER (TRUE));
2787         g_hash_table_insert (key_table, (char *) "filter", GINT_TO_POINTER (TRUE));
2788         g_hash_table_insert (key_table, (char *) "final", GINT_TO_POINTER (TRUE));
2789         g_hash_table_insert (key_table, (char *) "finally", GINT_TO_POINTER (TRUE));
2790         g_hash_table_insert (key_table, (char *) "fixed", GINT_TO_POINTER (TRUE));
2791         g_hash_table_insert (key_table, (char *) "flags", GINT_TO_POINTER (TRUE));
2792         g_hash_table_insert (key_table, (char *) "float32", GINT_TO_POINTER (TRUE));
2793         g_hash_table_insert (key_table, (char *) "float64", GINT_TO_POINTER (TRUE));
2794         g_hash_table_insert (key_table, (char *) "float", GINT_TO_POINTER (TRUE));
2795         g_hash_table_insert (key_table, (char *) "forwardref", GINT_TO_POINTER (TRUE));
2796         g_hash_table_insert (key_table, (char *) "fromunmanaged", GINT_TO_POINTER (TRUE));
2797         g_hash_table_insert (key_table, (char *) "handler", GINT_TO_POINTER (TRUE));
2798         g_hash_table_insert (key_table, (char *) "hidebysig", GINT_TO_POINTER (TRUE));
2799         g_hash_table_insert (key_table, (char *) "hresult", GINT_TO_POINTER (TRUE));
2800         g_hash_table_insert (key_table, (char *) "idispatch", GINT_TO_POINTER (TRUE));
2801         g_hash_table_insert (key_table, (char *) "il", GINT_TO_POINTER (TRUE));
2802         g_hash_table_insert (key_table, (char *) "illegal", GINT_TO_POINTER (TRUE));
2803         g_hash_table_insert (key_table, (char *) "implements", GINT_TO_POINTER (TRUE));
2804         g_hash_table_insert (key_table, (char *) "implicitcom", GINT_TO_POINTER (TRUE));
2805         g_hash_table_insert (key_table, (char *) "implicitres", GINT_TO_POINTER (TRUE));
2806         g_hash_table_insert (key_table, (char *) "import", GINT_TO_POINTER (TRUE));
2807         g_hash_table_insert (key_table, (char *) "in", GINT_TO_POINTER (TRUE));
2808         g_hash_table_insert (key_table, (char *) "inheritcheck", GINT_TO_POINTER (TRUE));
2809         g_hash_table_insert (key_table, (char *) "initblk", GINT_TO_POINTER (TRUE));
2810         g_hash_table_insert (key_table, (char *) "init", GINT_TO_POINTER (TRUE));
2811         g_hash_table_insert (key_table, (char *) "initobj", GINT_TO_POINTER (TRUE));
2812         g_hash_table_insert (key_table, (char *) "initonly", GINT_TO_POINTER (TRUE));
2813         g_hash_table_insert (key_table, (char *) "instance", GINT_TO_POINTER (TRUE));
2814         g_hash_table_insert (key_table, (char *) "int16", GINT_TO_POINTER (TRUE));
2815         g_hash_table_insert (key_table, (char *) "int32", GINT_TO_POINTER (TRUE));
2816         g_hash_table_insert (key_table, (char *) "int64", GINT_TO_POINTER (TRUE));
2817         g_hash_table_insert (key_table, (char *) "int8", GINT_TO_POINTER (TRUE));
2818         g_hash_table_insert (key_table, (char *) "interface", GINT_TO_POINTER (TRUE));
2819         g_hash_table_insert (key_table, (char *) "internalcall", GINT_TO_POINTER (TRUE));
2820         g_hash_table_insert (key_table, (char *) "int", GINT_TO_POINTER (TRUE));
2821         g_hash_table_insert (key_table, (char *) "isinst", GINT_TO_POINTER (TRUE));
2822         g_hash_table_insert (key_table, (char *) "iunknown", GINT_TO_POINTER (TRUE));
2823         g_hash_table_insert (key_table, (char *) "jmp", GINT_TO_POINTER (TRUE));
2824         g_hash_table_insert (key_table, (char *) "lasterr", GINT_TO_POINTER (TRUE));
2825         g_hash_table_insert (key_table, (char *) "lcid", GINT_TO_POINTER (TRUE));
2826         g_hash_table_insert (key_table, (char *) "ldarg.0", GINT_TO_POINTER (TRUE));
2827         g_hash_table_insert (key_table, (char *) "ldarg.1", GINT_TO_POINTER (TRUE));
2828         g_hash_table_insert (key_table, (char *) "ldarg.2", GINT_TO_POINTER (TRUE));
2829         g_hash_table_insert (key_table, (char *) "ldarg.3", GINT_TO_POINTER (TRUE));
2830         g_hash_table_insert (key_table, (char *) "ldarga", GINT_TO_POINTER (TRUE));
2831         g_hash_table_insert (key_table, (char *) "ldarga.s", GINT_TO_POINTER (TRUE));
2832         g_hash_table_insert (key_table, (char *) "ldarg", GINT_TO_POINTER (TRUE));
2833         g_hash_table_insert (key_table, (char *) "ldarg.s", GINT_TO_POINTER (TRUE));
2834         g_hash_table_insert (key_table, (char *) "ldc.i4.0", GINT_TO_POINTER (TRUE));
2835         g_hash_table_insert (key_table, (char *) "ldc.i4.1", GINT_TO_POINTER (TRUE));
2836         g_hash_table_insert (key_table, (char *) "ldc.i4.2", GINT_TO_POINTER (TRUE));
2837         g_hash_table_insert (key_table, (char *) "ldc.i4.3", GINT_TO_POINTER (TRUE));
2838         g_hash_table_insert (key_table, (char *) "ldc.i4.4", GINT_TO_POINTER (TRUE));
2839         g_hash_table_insert (key_table, (char *) "ldc.i4.5", GINT_TO_POINTER (TRUE));
2840         g_hash_table_insert (key_table, (char *) "ldc.i4.6", GINT_TO_POINTER (TRUE));
2841         g_hash_table_insert (key_table, (char *) "ldc.i4.7", GINT_TO_POINTER (TRUE));
2842         g_hash_table_insert (key_table, (char *) "ldc.i4.8", GINT_TO_POINTER (TRUE));
2843         g_hash_table_insert (key_table, (char *) "ldc.i4", GINT_TO_POINTER (TRUE));
2844         g_hash_table_insert (key_table, (char *) "ldc.i4.m1", GINT_TO_POINTER (TRUE));
2845         g_hash_table_insert (key_table, (char *) "ldc.i4.M1", GINT_TO_POINTER (TRUE));
2846         g_hash_table_insert (key_table, (char *) "ldc.i4.s", GINT_TO_POINTER (TRUE));
2847         g_hash_table_insert (key_table, (char *) "ldc.i8", GINT_TO_POINTER (TRUE));
2848         g_hash_table_insert (key_table, (char *) "ldc.r4", GINT_TO_POINTER (TRUE));
2849         g_hash_table_insert (key_table, (char *) "ldc.r8", GINT_TO_POINTER (TRUE));
2850         g_hash_table_insert (key_table, (char *) "ldelem", GINT_TO_POINTER (TRUE));
2851         g_hash_table_insert (key_table, (char *) "ldelema", GINT_TO_POINTER (TRUE));
2852         g_hash_table_insert (key_table, (char *) "ldelem.i1", GINT_TO_POINTER (TRUE));
2853         g_hash_table_insert (key_table, (char *) "ldelem.i2", GINT_TO_POINTER (TRUE));
2854         g_hash_table_insert (key_table, (char *) "ldelem.i4", GINT_TO_POINTER (TRUE));
2855         g_hash_table_insert (key_table, (char *) "ldelem.i8", GINT_TO_POINTER (TRUE));
2856         g_hash_table_insert (key_table, (char *) "ldelem.i", GINT_TO_POINTER (TRUE));
2857         g_hash_table_insert (key_table, (char *) "ldelem.r4", GINT_TO_POINTER (TRUE));
2858         g_hash_table_insert (key_table, (char *) "ldelem.r8", GINT_TO_POINTER (TRUE));
2859         g_hash_table_insert (key_table, (char *) "ldelem.ref", GINT_TO_POINTER (TRUE));
2860         g_hash_table_insert (key_table, (char *) "ldelem.u1", GINT_TO_POINTER (TRUE));
2861         g_hash_table_insert (key_table, (char *) "ldelem.u2", GINT_TO_POINTER (TRUE));
2862         g_hash_table_insert (key_table, (char *) "ldelem.u4", GINT_TO_POINTER (TRUE));
2863         g_hash_table_insert (key_table, (char *) "ldelem.u8", GINT_TO_POINTER (TRUE));
2864         g_hash_table_insert (key_table, (char *) "ldflda", GINT_TO_POINTER (TRUE));
2865         g_hash_table_insert (key_table, (char *) "ldfld", GINT_TO_POINTER (TRUE));
2866         g_hash_table_insert (key_table, (char *) "ldftn", GINT_TO_POINTER (TRUE));
2867         g_hash_table_insert (key_table, (char *) "ldind.i1", GINT_TO_POINTER (TRUE));
2868         g_hash_table_insert (key_table, (char *) "ldind.i2", GINT_TO_POINTER (TRUE));
2869         g_hash_table_insert (key_table, (char *) "ldind.i4", GINT_TO_POINTER (TRUE));
2870         g_hash_table_insert (key_table, (char *) "ldind.i8", GINT_TO_POINTER (TRUE));
2871         g_hash_table_insert (key_table, (char *) "ldind.i", GINT_TO_POINTER (TRUE));
2872         g_hash_table_insert (key_table, (char *) "ldind.r4", GINT_TO_POINTER (TRUE));
2873         g_hash_table_insert (key_table, (char *) "ldind.r8", GINT_TO_POINTER (TRUE));
2874         g_hash_table_insert (key_table, (char *) "ldind.ref", GINT_TO_POINTER (TRUE));
2875         g_hash_table_insert (key_table, (char *) "ldind.u1", GINT_TO_POINTER (TRUE));
2876         g_hash_table_insert (key_table, (char *) "ldind.u2", GINT_TO_POINTER (TRUE));
2877         g_hash_table_insert (key_table, (char *) "ldind.u4", GINT_TO_POINTER (TRUE));
2878         g_hash_table_insert (key_table, (char *) "ldind.u8", GINT_TO_POINTER (TRUE));
2879         g_hash_table_insert (key_table, (char *) "ldlen", GINT_TO_POINTER (TRUE));
2880         g_hash_table_insert (key_table, (char *) "ldloc.0", GINT_TO_POINTER (TRUE));
2881         g_hash_table_insert (key_table, (char *) "ldloc.1", GINT_TO_POINTER (TRUE));
2882         g_hash_table_insert (key_table, (char *) "ldloc.2", GINT_TO_POINTER (TRUE));
2883         g_hash_table_insert (key_table, (char *) "ldloc.3", GINT_TO_POINTER (TRUE));
2884         g_hash_table_insert (key_table, (char *) "ldloca", GINT_TO_POINTER (TRUE));
2885         g_hash_table_insert (key_table, (char *) "ldloca.s", GINT_TO_POINTER (TRUE));
2886         g_hash_table_insert (key_table, (char *) "ldloc", GINT_TO_POINTER (TRUE));
2887         g_hash_table_insert (key_table, (char *) "ldloc.s", GINT_TO_POINTER (TRUE));
2888         g_hash_table_insert (key_table, (char *) "ldnull", GINT_TO_POINTER (TRUE));
2889         g_hash_table_insert (key_table, (char *) "ldobj", GINT_TO_POINTER (TRUE));
2890         g_hash_table_insert (key_table, (char *) "ldsflda", GINT_TO_POINTER (TRUE));
2891         g_hash_table_insert (key_table, (char *) "ldsfld", GINT_TO_POINTER (TRUE));
2892         g_hash_table_insert (key_table, (char *) "ldstr", GINT_TO_POINTER (TRUE));
2893         g_hash_table_insert (key_table, (char *) "ldtoken", GINT_TO_POINTER (TRUE));
2894         g_hash_table_insert (key_table, (char *) "ldvirtftn", GINT_TO_POINTER (TRUE));
2895         g_hash_table_insert (key_table, (char *) "leave", GINT_TO_POINTER (TRUE));
2896         g_hash_table_insert (key_table, (char *) "leave.s", GINT_TO_POINTER (TRUE));
2897         g_hash_table_insert (key_table, (char *) "linkcheck", GINT_TO_POINTER (TRUE));
2898         g_hash_table_insert (key_table, (char *) "literal", GINT_TO_POINTER (TRUE));
2899         g_hash_table_insert (key_table, (char *) "localloc", GINT_TO_POINTER (TRUE));
2900         g_hash_table_insert (key_table, (char *) "lpstr", GINT_TO_POINTER (TRUE));
2901         g_hash_table_insert (key_table, (char *) "lpstruct", GINT_TO_POINTER (TRUE));
2902         g_hash_table_insert (key_table, (char *) "lptstr", GINT_TO_POINTER (TRUE));
2903         g_hash_table_insert (key_table, (char *) "lpvoid", GINT_TO_POINTER (TRUE));
2904         g_hash_table_insert (key_table, (char *) "lpwstr", GINT_TO_POINTER (TRUE));
2905         g_hash_table_insert (key_table, (char *) "managed", GINT_TO_POINTER (TRUE));
2906         g_hash_table_insert (key_table, (char *) "marshal", GINT_TO_POINTER (TRUE));
2907         g_hash_table_insert (key_table, (char *) "method", GINT_TO_POINTER (TRUE));
2908         g_hash_table_insert (key_table, (char *) "mkrefany", GINT_TO_POINTER (TRUE));
2909         g_hash_table_insert (key_table, (char *) "modopt", GINT_TO_POINTER (TRUE));
2910         g_hash_table_insert (key_table, (char *) "modreq", GINT_TO_POINTER (TRUE));
2911         g_hash_table_insert (key_table, (char *) "mul", GINT_TO_POINTER (TRUE));
2912         g_hash_table_insert (key_table, (char *) "mul.ovf", GINT_TO_POINTER (TRUE));
2913         g_hash_table_insert (key_table, (char *) "mul.ovf.un", GINT_TO_POINTER (TRUE));
2914         g_hash_table_insert (key_table, (char *) "native", GINT_TO_POINTER (TRUE));
2915         g_hash_table_insert (key_table, (char *) "neg", GINT_TO_POINTER (TRUE));
2916         g_hash_table_insert (key_table, (char *) "nested", GINT_TO_POINTER (TRUE));
2917         g_hash_table_insert (key_table, (char *) "newarr", GINT_TO_POINTER (TRUE));
2918         g_hash_table_insert (key_table, (char *) "newobj", GINT_TO_POINTER (TRUE));
2919         g_hash_table_insert (key_table, (char *) "newslot", GINT_TO_POINTER (TRUE));
2920         g_hash_table_insert (key_table, (char *) "noappdomain", GINT_TO_POINTER (TRUE));
2921         g_hash_table_insert (key_table, (char *) "noinlining", GINT_TO_POINTER (TRUE));
2922         g_hash_table_insert (key_table, (char *) "nomachine", GINT_TO_POINTER (TRUE));
2923         g_hash_table_insert (key_table, (char *) "nomangle", GINT_TO_POINTER (TRUE));
2924         g_hash_table_insert (key_table, (char *) "nometadata", GINT_TO_POINTER (TRUE));
2925         g_hash_table_insert (key_table, (char *) "noncasdemand", GINT_TO_POINTER (TRUE));
2926         g_hash_table_insert (key_table, (char *) "noncasinheritance", GINT_TO_POINTER (TRUE));
2927         g_hash_table_insert (key_table, (char *) "noncaslinkdemand", GINT_TO_POINTER (TRUE));
2928         g_hash_table_insert (key_table, (char *) "nop", GINT_TO_POINTER (TRUE));
2929         g_hash_table_insert (key_table, (char *) "noprocess", GINT_TO_POINTER (TRUE));
2930         g_hash_table_insert (key_table, (char *) "not", GINT_TO_POINTER (TRUE));
2931         g_hash_table_insert (key_table, (char *) "not_in_gc_heap", GINT_TO_POINTER (TRUE));
2932         g_hash_table_insert (key_table, (char *) "notremotable", GINT_TO_POINTER (TRUE));
2933         g_hash_table_insert (key_table, (char *) "notserialized", GINT_TO_POINTER (TRUE));
2934         g_hash_table_insert (key_table, (char *) "null", GINT_TO_POINTER (TRUE));
2935         g_hash_table_insert (key_table, (char *) "nullref", GINT_TO_POINTER (TRUE));
2936         g_hash_table_insert (key_table, (char *) "object", GINT_TO_POINTER (TRUE));
2937         g_hash_table_insert (key_table, (char *) "objectref", GINT_TO_POINTER (TRUE));
2938         g_hash_table_insert (key_table, (char *) "off", GINT_TO_POINTER (TRUE));
2939         g_hash_table_insert (key_table, (char *) "on", GINT_TO_POINTER (TRUE));
2940         g_hash_table_insert (key_table, (char *) "opt", GINT_TO_POINTER (TRUE));
2941         g_hash_table_insert (key_table, (char *) "optil", GINT_TO_POINTER (TRUE));
2942         g_hash_table_insert (key_table, (char *) "or", GINT_TO_POINTER (TRUE));
2943         g_hash_table_insert (key_table, (char *) "out", GINT_TO_POINTER (TRUE));
2944         g_hash_table_insert (key_table, (char *) "permitonly", GINT_TO_POINTER (TRUE));
2945         g_hash_table_insert (key_table, (char *) "pinned", GINT_TO_POINTER (TRUE));
2946         g_hash_table_insert (key_table, (char *) "pinvokeimpl", GINT_TO_POINTER (TRUE));
2947         g_hash_table_insert (key_table, (char *) "pop", GINT_TO_POINTER (TRUE));
2948         g_hash_table_insert (key_table, (char *) "prefix1", GINT_TO_POINTER (TRUE));
2949         g_hash_table_insert (key_table, (char *) "prefix2", GINT_TO_POINTER (TRUE));
2950         g_hash_table_insert (key_table, (char *) "prefix3", GINT_TO_POINTER (TRUE));
2951         g_hash_table_insert (key_table, (char *) "prefix4", GINT_TO_POINTER (TRUE));
2952         g_hash_table_insert (key_table, (char *) "prefix5", GINT_TO_POINTER (TRUE));
2953         g_hash_table_insert (key_table, (char *) "prefix6", GINT_TO_POINTER (TRUE));
2954         g_hash_table_insert (key_table, (char *) "prefix7", GINT_TO_POINTER (TRUE));
2955         g_hash_table_insert (key_table, (char *) "prefixref", GINT_TO_POINTER (TRUE));
2956         g_hash_table_insert (key_table, (char *) "prejitdeny", GINT_TO_POINTER (TRUE));
2957         g_hash_table_insert (key_table, (char *) "prejitgrant", GINT_TO_POINTER (TRUE));
2958         g_hash_table_insert (key_table, (char *) "preservesig", GINT_TO_POINTER (TRUE));
2959         g_hash_table_insert (key_table, (char *) "private", GINT_TO_POINTER (TRUE));
2960         g_hash_table_insert (key_table, (char *) "privatescope", GINT_TO_POINTER (TRUE));
2961         g_hash_table_insert (key_table, (char *) "property", GINT_TO_POINTER (TRUE));
2962         g_hash_table_insert (key_table, (char *) "protected", GINT_TO_POINTER (TRUE));
2963         g_hash_table_insert (key_table, (char *) "public", GINT_TO_POINTER (TRUE));
2964         g_hash_table_insert (key_table, (char *) "readonly", GINT_TO_POINTER (TRUE));
2965         g_hash_table_insert (key_table, (char *) "record", GINT_TO_POINTER (TRUE));
2966         g_hash_table_insert (key_table, (char *) "refany", GINT_TO_POINTER (TRUE));
2967         g_hash_table_insert (key_table, (char *) "refanytype", GINT_TO_POINTER (TRUE));
2968         g_hash_table_insert (key_table, (char *) "refanyval", GINT_TO_POINTER (TRUE));
2969         g_hash_table_insert (key_table, (char *) "rem", GINT_TO_POINTER (TRUE));
2970         g_hash_table_insert (key_table, (char *) "rem.un", GINT_TO_POINTER (TRUE));
2971         g_hash_table_insert (key_table, (char *) "reqmin", GINT_TO_POINTER (TRUE));
2972         g_hash_table_insert (key_table, (char *) "reqopt", GINT_TO_POINTER (TRUE));
2973         g_hash_table_insert (key_table, (char *) "reqrefuse", GINT_TO_POINTER (TRUE));
2974         g_hash_table_insert (key_table, (char *) "reqsecobj", GINT_TO_POINTER (TRUE));
2975         g_hash_table_insert (key_table, (char *) "request", GINT_TO_POINTER (TRUE));
2976         g_hash_table_insert (key_table, (char *) "ret", GINT_TO_POINTER (TRUE));
2977         g_hash_table_insert (key_table, (char *) "rethrow", GINT_TO_POINTER (TRUE));
2978         g_hash_table_insert (key_table, (char *) "retval", GINT_TO_POINTER (TRUE));
2979         g_hash_table_insert (key_table, (char *) "rtspecialname", GINT_TO_POINTER (TRUE));
2980         g_hash_table_insert (key_table, (char *) "runtime", GINT_TO_POINTER (TRUE));
2981         g_hash_table_insert (key_table, (char *) "safearray", GINT_TO_POINTER (TRUE));
2982         g_hash_table_insert (key_table, (char *) "sealed", GINT_TO_POINTER (TRUE));
2983         g_hash_table_insert (key_table, (char *) "sequential", GINT_TO_POINTER (TRUE));
2984         g_hash_table_insert (key_table, (char *) "serializable", GINT_TO_POINTER (TRUE));
2985         g_hash_table_insert (key_table, (char *) "shl", GINT_TO_POINTER (TRUE));
2986         g_hash_table_insert (key_table, (char *) "shr", GINT_TO_POINTER (TRUE));
2987         g_hash_table_insert (key_table, (char *) "shr.un", GINT_TO_POINTER (TRUE));
2988         g_hash_table_insert (key_table, (char *) "sizeof", GINT_TO_POINTER (TRUE));
2989         g_hash_table_insert (key_table, (char *) "special", GINT_TO_POINTER (TRUE));
2990         g_hash_table_insert (key_table, (char *) "specialname", GINT_TO_POINTER (TRUE));
2991         g_hash_table_insert (key_table, (char *) "starg", GINT_TO_POINTER (TRUE));
2992         g_hash_table_insert (key_table, (char *) "starg.s", GINT_TO_POINTER (TRUE));
2993         g_hash_table_insert (key_table, (char *) "static", GINT_TO_POINTER (TRUE));
2994         g_hash_table_insert (key_table, (char *) "stdcall", GINT_TO_POINTER (TRUE));
2995         g_hash_table_insert (key_table, (char *) "stelem", GINT_TO_POINTER (TRUE));
2996         g_hash_table_insert (key_table, (char *) "stelem.i1", GINT_TO_POINTER (TRUE));
2997         g_hash_table_insert (key_table, (char *) "stelem.i2", GINT_TO_POINTER (TRUE));
2998         g_hash_table_insert (key_table, (char *) "stelem.i4", GINT_TO_POINTER (TRUE));
2999         g_hash_table_insert (key_table, (char *) "stelem.i8", GINT_TO_POINTER (TRUE));
3000         g_hash_table_insert (key_table, (char *) "stelem.i", GINT_TO_POINTER (TRUE));
3001         g_hash_table_insert (key_table, (char *) "stelem.r4", GINT_TO_POINTER (TRUE));
3002         g_hash_table_insert (key_table, (char *) "stelem.r8", GINT_TO_POINTER (TRUE));
3003         g_hash_table_insert (key_table, (char *) "stelem.ref", GINT_TO_POINTER (TRUE));
3004         g_hash_table_insert (key_table, (char *) "stfld", GINT_TO_POINTER (TRUE));
3005         g_hash_table_insert (key_table, (char *) "stind.i1", GINT_TO_POINTER (TRUE));
3006         g_hash_table_insert (key_table, (char *) "stind.i2", GINT_TO_POINTER (TRUE));
3007         g_hash_table_insert (key_table, (char *) "stind.i4", GINT_TO_POINTER (TRUE));
3008         g_hash_table_insert (key_table, (char *) "stind.i8", GINT_TO_POINTER (TRUE));
3009         g_hash_table_insert (key_table, (char *) "stind.i", GINT_TO_POINTER (TRUE));
3010         g_hash_table_insert (key_table, (char *) "stind.r4", GINT_TO_POINTER (TRUE));
3011         g_hash_table_insert (key_table, (char *) "stind.r8", GINT_TO_POINTER (TRUE));
3012         g_hash_table_insert (key_table, (char *) "stloc", GINT_TO_POINTER (TRUE));
3013         g_hash_table_insert (key_table, (char *) "stobj", GINT_TO_POINTER (TRUE));
3014         g_hash_table_insert (key_table, (char *) "storage", GINT_TO_POINTER (TRUE));
3015         g_hash_table_insert (key_table, (char *) "stored_object", GINT_TO_POINTER (TRUE));
3016         g_hash_table_insert (key_table, (char *) "streamed_object", GINT_TO_POINTER (TRUE));
3017         g_hash_table_insert (key_table, (char *) "stream", GINT_TO_POINTER (TRUE));
3018         g_hash_table_insert (key_table, (char *) "strict", GINT_TO_POINTER (TRUE));
3019         g_hash_table_insert (key_table, (char *) "string", GINT_TO_POINTER (TRUE));
3020         g_hash_table_insert (key_table, (char *) "struct", GINT_TO_POINTER (TRUE));
3021         g_hash_table_insert (key_table, (char *) "stsfld", GINT_TO_POINTER (TRUE));
3022         g_hash_table_insert (key_table, (char *) "sub", GINT_TO_POINTER (TRUE));
3023         g_hash_table_insert (key_table, (char *) "sub.ovf", GINT_TO_POINTER (TRUE));
3024         g_hash_table_insert (key_table, (char *) "sub.ovf.un", GINT_TO_POINTER (TRUE));
3025         g_hash_table_insert (key_table, (char *) "switch", GINT_TO_POINTER (TRUE));
3026         g_hash_table_insert (key_table, (char *) "synchronized", GINT_TO_POINTER (TRUE));
3027         g_hash_table_insert (key_table, (char *) "syschar", GINT_TO_POINTER (TRUE));
3028         g_hash_table_insert (key_table, (char *) "sysstring", GINT_TO_POINTER (TRUE));
3029         g_hash_table_insert (key_table, (char *) "tbstr", GINT_TO_POINTER (TRUE));
3030         g_hash_table_insert (key_table, (char *) "thiscall", GINT_TO_POINTER (TRUE));
3031         g_hash_table_insert (key_table, (char *) "tls", GINT_TO_POINTER (TRUE));
3032         g_hash_table_insert (key_table, (char *) "to", GINT_TO_POINTER (TRUE));
3033         g_hash_table_insert (key_table, (char *) "true", GINT_TO_POINTER (TRUE));
3034         g_hash_table_insert (key_table, (char *) "type", GINT_TO_POINTER (TRUE));
3035         g_hash_table_insert (key_table, (char *) "typedref", GINT_TO_POINTER (TRUE));
3036         g_hash_table_insert (key_table, (char *) "uint", GINT_TO_POINTER (TRUE));
3037         g_hash_table_insert (key_table, (char *) "uint8", GINT_TO_POINTER (TRUE));
3038         g_hash_table_insert (key_table, (char *) "uint16", GINT_TO_POINTER (TRUE));
3039         g_hash_table_insert (key_table, (char *) "uint32", GINT_TO_POINTER (TRUE));
3040         g_hash_table_insert (key_table, (char *) "uint64", GINT_TO_POINTER (TRUE));
3041         g_hash_table_insert (key_table, (char *) "unbox", GINT_TO_POINTER (TRUE));
3042         g_hash_table_insert (key_table, (char *) "unicode", GINT_TO_POINTER (TRUE));
3043         g_hash_table_insert (key_table, (char *) "unmanagedexp", GINT_TO_POINTER (TRUE));
3044         g_hash_table_insert (key_table, (char *) "unmanaged", GINT_TO_POINTER (TRUE));
3045         g_hash_table_insert (key_table, (char *) "unsigned", GINT_TO_POINTER (TRUE));
3046         g_hash_table_insert (key_table, (char *) "userdefined", GINT_TO_POINTER (TRUE));
3047         g_hash_table_insert (key_table, (char *) "value", GINT_TO_POINTER (TRUE));
3048         g_hash_table_insert (key_table, (char *) "valuetype", GINT_TO_POINTER (TRUE));
3049         g_hash_table_insert (key_table, (char *) "vararg", GINT_TO_POINTER (TRUE));
3050         g_hash_table_insert (key_table, (char *) "variant", GINT_TO_POINTER (TRUE));
3051         g_hash_table_insert (key_table, (char *) "vector", GINT_TO_POINTER (TRUE));
3052         g_hash_table_insert (key_table, (char *) "virtual", GINT_TO_POINTER (TRUE));
3053         g_hash_table_insert (key_table, (char *) "void", GINT_TO_POINTER (TRUE));
3054         g_hash_table_insert (key_table, (char *) "wchar", GINT_TO_POINTER (TRUE));
3055         g_hash_table_insert (key_table, (char *) "winapi", GINT_TO_POINTER (TRUE));
3056         g_hash_table_insert (key_table, (char *) "with", GINT_TO_POINTER (TRUE));
3057         g_hash_table_insert (key_table, (char *) "xor", GINT_TO_POINTER (TRUE));
3058 }
3059
3060 guint32
3061 method_dor_to_token (guint32 idx) {
3062         switch (idx & MONO_METHODDEFORREF_MASK) {
3063         case MONO_METHODDEFORREF_METHODDEF:
3064                 return MONO_TOKEN_METHOD_DEF | (idx >> MONO_METHODDEFORREF_BITS);
3065         case MONO_METHODDEFORREF_METHODREF:
3066                 return MONO_TOKEN_MEMBER_REF | (idx >> MONO_METHODDEFORREF_BITS);
3067         }
3068         return -1;
3069 }
3070
3071 char *
3072 get_method_override (MonoImage *m, guint32 token, MonoGenericContainer *container)
3073 {
3074         MonoTableInfo *t = &m->tables [MONO_TABLE_METHODIMPL];
3075         int i;
3076
3077         for (i = 1; i <= t->rows; i++){
3078                 guint32 cols [MONO_METHODIMPL_SIZE];
3079                 guint32 decl, impl;
3080
3081                 mono_metadata_decode_row (t, i - 1, cols, MONO_METHODIMPL_SIZE);
3082
3083                 impl = method_dor_to_token (cols [MONO_METHODIMPL_BODY]);
3084                 decl = method_dor_to_token (cols [MONO_METHODIMPL_DECLARATION]);
3085
3086                 if (token == impl) {
3087                         MonoMethod *mh = NULL;
3088                         mh = mono_get_method_full (m, decl, NULL, (MonoGenericContext *) container);
3089
3090                         if (mh && (mh->klass && (mh->klass->generic_class || mh->klass->generic_container))) {
3091                                 char *meth_str;
3092                                 char *ret;
3093                                 
3094                                 meth_str = get_method_core (m, decl, TRUE, container);
3095                                 ret = g_strdup_printf ("method %s", meth_str);
3096                                 g_free (meth_str);
3097                                 return ret;
3098                         } else {
3099                                 return get_method_core (m, decl, FALSE, container);
3100                         }
3101                 }
3102         }
3103
3104         return NULL;
3105 }
3106
3107 static void
3108 check_ambiguous_genparams (MonoGenericContainer *container)
3109 {
3110         GSList *dup_list = NULL, *l;
3111         GHashTable *table = NULL;
3112         gpointer *p;
3113         int i;
3114
3115         if (!container)
3116                 return;
3117         
3118         if (generic_containers && g_hash_table_lookup (generic_containers, container))
3119                 /* Already been checked for ambiguous gen params */
3120                 return;
3121
3122         table = g_hash_table_new (g_str_hash, g_str_equal);
3123         for (i = 0; i < container->type_argc; i++) {
3124                 MonoGenericParam *param = mono_generic_container_get_param (container, i);
3125
3126                 if ((p = g_hash_table_lookup (table, mono_generic_param_info (param)->name)))
3127                         dup_list = g_slist_prepend (g_slist_prepend (dup_list, GUINT_TO_POINTER (i + 1)), p);
3128                 else
3129                         g_hash_table_insert (table, (char*)mono_generic_param_info (param)->name, GUINT_TO_POINTER (i + 1));
3130         }
3131
3132         if (dup_list) {
3133                 if (!mono_generic_params_with_ambiguous_names)
3134                         mono_generic_params_with_ambiguous_names = g_hash_table_new (NULL, NULL);
3135                 for (l = dup_list; l; l = l->next) {
3136                         int param = GPOINTER_TO_UINT (l->data);
3137                         g_hash_table_insert (mono_generic_params_with_ambiguous_names,
3138                                 mono_generic_container_get_param (container, param-1),
3139                                 mono_generic_container_get_param (container, param-1));
3140                 }
3141                 g_slist_free (dup_list);
3142         }
3143
3144         if (!generic_containers)
3145                 generic_containers = g_hash_table_new (NULL, NULL);
3146
3147         g_hash_table_insert (generic_containers, container, container);
3148         g_hash_table_destroy (table);
3149 }
3150         
3151 static gboolean
3152 cant_print_generic_param_name (MonoGenericParam *gparam)
3153 {
3154         MonoGenericContainer *container;
3155         g_assert (gparam);
3156
3157         container = mono_generic_param_owner (gparam);
3158         check_ambiguous_genparams (container);
3159         return (!container || (mono_generic_params_with_ambiguous_names &&
3160                         g_hash_table_lookup (mono_generic_params_with_ambiguous_names, gparam)));
3161 }
3162
3163
3164 static dis_map_t method_impl_map [] = {
3165         { METHOD_IMPL_ATTRIBUTE_IL,              "cil " },
3166         { METHOD_IMPL_ATTRIBUTE_NATIVE,          "native " },
3167         { METHOD_IMPL_ATTRIBUTE_OPTIL,           "optil " },
3168         { METHOD_IMPL_ATTRIBUTE_RUNTIME,         "runtime " },
3169         { 0, NULL }
3170 };
3171
3172 static dis_map_t managed_type_map [] = {
3173         { METHOD_IMPL_ATTRIBUTE_UNMANAGED,       "unmanaged " },
3174         { METHOD_IMPL_ATTRIBUTE_MANAGED,         "managed " },
3175         { 0, NULL }
3176 };
3177
3178 static dis_map_t managed_impl_flags [] = {
3179         { METHOD_IMPL_ATTRIBUTE_FORWARD_REF,     "fwdref " },
3180         { METHOD_IMPL_ATTRIBUTE_PRESERVE_SIG,    "preservesig " },
3181         { METHOD_IMPL_ATTRIBUTE_INTERNAL_CALL,   "internalcall " },
3182         { METHOD_IMPL_ATTRIBUTE_SYNCHRONIZED,    "synchronized " },
3183         { METHOD_IMPL_ATTRIBUTE_NOINLINING,      "noinlining " },
3184         { 0, NULL }
3185 };
3186
3187 char *
3188 get_method_impl_flags (guint32 f)
3189 {
3190         GString *str = g_string_new ("");
3191         char *s;
3192         int code_type = f & METHOD_IMPL_ATTRIBUTE_CODE_TYPE_MASK;
3193         int managed_type = f & METHOD_IMPL_ATTRIBUTE_MANAGED_MASK;
3194
3195         g_string_append (str, map (code_type, method_impl_map));
3196         g_string_append (str, map (managed_type, managed_type_map));
3197         g_string_append (str, flags (f, managed_impl_flags));
3198         
3199         s = str->str;
3200         g_string_free (str, FALSE);
3201         return s;
3202 }
3203