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