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