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