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