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