#include "util.h"
#include "get.h"
#include <mono/metadata/class.h>
+#include <mono/metadata/marshal.h>
+
+extern gboolean substitute_with_mscorlib_p;
char *
get_typedef (MonoImage *m, int idx)
{
guint32 cols [MONO_TYPEDEF_SIZE];
+ const char *ns;
mono_metadata_decode_row (&m->tables [MONO_TABLE_TYPEDEF], idx - 1, cols, MONO_TYPEDEF_SIZE);
+ ns = mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAMESPACE]);
return g_strdup_printf (
- "%s.%s",
- mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAMESPACE]),
+ "%s%s%s", ns, *ns?".":"",
mono_metadata_string_heap (m, cols [MONO_TYPEDEF_NAME]));
}
*/
g_assert (idx == 1);
- mono_metadata_decode_row (&m->tables [MONO_TABLE_MODULEREF], idx - 1, cols, MONO_MODULE_SIZE);
+ mono_metadata_decode_row (&m->tables [MONO_TABLE_MODULE], idx - 1, cols, MONO_MODULE_SIZE);
return g_strdup (mono_metadata_string_heap (m, cols [MONO_MODULE_NAME]));
}
t = mono_metadata_string_heap (m, cols [MONO_TYPEREF_NAME]);
s = mono_metadata_string_heap (m, cols [MONO_TYPEREF_NAMESPACE]);
- rs_idx = cols [MONO_TYPEREF_SCOPE] >> 2;
- /*
- * Two bits in Beta2.
- * ECMA spec claims 3 bits
- */
- table = cols [MONO_TYPEREF_SCOPE] & 3;
+ rs_idx = cols [MONO_TYPEREF_SCOPE] >> RESOLTION_SCOPE_BITS;
+ table = cols [MONO_TYPEREF_SCOPE] & RESOLTION_SCOPE_MASK;
switch (table){
- case 0: /* Module */
+ case RESOLTION_SCOPE_MODULE: /* Module */
x = get_module (m, rs_idx);
- ret = g_strdup_printf ("TODO:TypeRef-Module [%s] %s.%s", x, s, t);
+ ret = g_strdup_printf ("[%s] %s%s%s", x, s, *s?".":"", t);
g_free (x);
break;
- case 1: /* ModuleRef */
+ case RESOLTION_SCOPE_MODULEREF: /* ModuleRef */
ret = g_strdup_printf ("TODO:TypeRef-ModuleRef (%s.%s)", s, t);
break;
- case 2: /*
+ case RESOLTION_SCOPE_ASSEMBLYREF: /*
* AssemblyRef (ECMA docs claim it is 3, but it looks to
* me like it is 2 (tokens are prefixed with 0x23)
*/
x = get_assemblyref (m, rs_idx);
- ret = g_strdup_printf ("[%s]%s.%s", x, s, t);
+ ret = g_strdup_printf ("[%s]%s%s%s", x, s, *s?".":"", t);
g_free (x);
break;
- case 4: /* TypeRef */
- ret = g_strdup_printf ("TODO:TypeRef-TypeRef: TYPEREF! (%s.%s)", s, t);
+ case RESOLTION_SCOPE_TYPEREF: /* TypeRef */
+ x = get_typeref (m, rs_idx);
+ ret = g_strdup_printf ("%s/%s", x, t);
+ g_free (x);
break;
default:
break;
default:
- g_error ("Unhandled encoding for typedef-or-ref coded index");
+ g_error ("Unhandled encoding for typedef-or-ref coded index 0x%08x", dor_token);
}
{
char *t;
char *result;
- char *out = param->attrs & 2 ? "[out] ": "";
+ const char *out = param->attrs & 2 ? "[out] ": "";
t = dis_stringify_type (m, param);
result = g_strconcat (out, t, NULL);
g_free (t);
if (!method) {
const char *sig = mono_metadata_blob_heap (m, cols [MONO_METHOD_SIGNATURE]);
mono_metadata_decode_blob_size (sig, &sig);
- method = mono_metadata_parse_method_signature (m, 1, sig, &sig);
+ method = mono_metadata_parse_method_signature (m, methoddef_row, sig, &sig);
free_method = 1;
}
}
}
static char *
-dis_stringify_object (MonoImage *m, MonoType *type)
+dis_stringify_object_with_class (MonoImage *m, MonoClass *c)
{
- char *otype = type->type == MONO_TYPE_CLASS? "class" : "valuetype";
- char *assemblyref = NULL;
- MonoClass *c = type->data.klass;
- if (!c)
- return g_strdup ("Unknown");
+ /* FIXME: handle MONO_TYPE_OBJECT ... */
+ const char *otype = c->byval_arg.type == MONO_TYPE_CLASS? "class" : "valuetype";
+ char *assemblyref = NULL, *result;
if (m != c->image) {
- /* we cheat */
- if (!strcmp ("corlib", c->image->assembly_name))
- assemblyref = g_strdup_printf ("[%s]", "mscorlib");
- else
- assemblyref = g_strdup_printf ("[%s]", c->image->assembly_name);
+ if (c->image->assembly_name) {
+ /* we cheat */
+ if (substitute_with_mscorlib_p && !strcmp ("corlib", c->image->assembly_name))
+ assemblyref = g_strdup_printf ("[%s]", "mscorlib");
+ else
+ assemblyref = g_strdup_printf ("[%s]", c->image->assembly->aname.name);
+ } else {
+ assemblyref = g_strdup_printf ("[.module %s]", c->image->module_name);
+ }
}
- otype = g_strdup_printf ("%s %s%s%s%s", otype, assemblyref?assemblyref:"", c->name_space,
+ result = g_strdup_printf ("%s %s%s%s%s", otype, assemblyref?assemblyref:"", c->name_space,
*c->name_space?".":"", c->name);
g_free (assemblyref);
- return otype;
+ return result;
+}
+
+static char *
+dis_stringify_object (MonoImage *m, MonoType *type)
+{
+ MonoClass *c = mono_class_from_mono_type (type);
+ return dis_stringify_object_with_class (m, c);
}
char*
dis_stringify_type (MonoImage *m, MonoType *type)
{
- char *bare = NULL, *pinned = "", *byref = "";
- char *mods = NULL;
+ const char *pinned = "", *byref = "";
+ char *bare = NULL, *mods = NULL;
char *result;
if (type->num_mods)
if (allocated_modifier_string)
g_free (allocated_modifier_string);
if (allocated_type_string)
- g_free (allocated_modifier_string);
+ g_free (allocated_type_string);
return res;
}
MonoMethod *mh;
mh = mono_get_method (m, token, NULL);
- if (mh)
- name = g_strdup_printf ("%s.%s::%s", mh->klass->name_space,
- mh->klass->name, mh->name);
- else
+ if (mh) {
+ sig = dis_stringify_object_with_class (m, mh->klass);
+ name = g_strdup_printf ("%s::%s", sig, mh->name);
+ g_free (sig);
+ } else
name = NULL;
switch (mono_metadata_token_code (token)){
case MONO_TYPE_I4:
return g_strdup_printf ("int32(%d)", read32 (ptr));
- case MONO_TYPE_I8:
- /*
- * FIXME: This is not endian portable, does only
- * matter for debugging, but still.
- * Need also to fix unaligned access.
- */
- return g_strdup_printf ("0x%08x%08x", *(guint32 *) ptr, *(guint32 *) (ptr + 4));
-
- case MONO_TYPE_U8:
- return g_strdup_printf ("0x%08x%08x", *(guint32 *) ptr, *(guint32 *) (ptr + 4));
- case MONO_TYPE_R4:
- return g_strdup_printf ("%g", (double) (* (float *) ptr));
-
- case MONO_TYPE_R8:
- return g_strdup_printf ("%g", * (double *) ptr);
-
+ case MONO_TYPE_I8: {
+ guint32 low, high;
+ low = read32 (ptr);
+ high = read32 (ptr + 4);
+ return g_strdup_printf ("0x%08x%08x", high, low);
+ }
+ case MONO_TYPE_U8: {
+ guint32 low, high;
+ low = read32 (ptr);
+ high = read32 (ptr + 4);
+ return g_strdup_printf ("0x%08x%08x", high, low);
+ }
+ case MONO_TYPE_R4: {
+ float r;
+ readr4 (ptr, &r);
+ return g_strdup_printf ("%g", (double) r);
+ }
+ case MONO_TYPE_R8: {
+ double r;
+ readr8 (ptr, &r);
+ return g_strdup_printf ("%g", r);
+ }
case MONO_TYPE_STRING: {
- int len, i, j, e;
+ int i, j, e;
char *res;
e = len = 0;
for (i = 0; !ptr [i+1]; i += 2){
break;
default:
- g_error ("Unhandled encoding for typedef-or-ref coded index");
+ g_error ("Unhandled encoding for token 0x%08x", token);
}
return s;
}
+
+char *
+get_guid (MonoImage *m, guint32 guid_index)
+{
+ const unsigned char *guid;
+ char *result;
+
+ guid = mono_metadata_guid_heap (m, guid_index);
+
+ result = g_strdup_printf ("{%02X%02X%02X%02X-%02X%02X-%02X%02X-%02X%02X-%02X%02X%02X%02X%02X%02X}",
+ guid [3], guid [2], guid [1], guid [0], guid [5], guid [4], guid [7], guid [6],
+ guid [8], guid [9], guid [10], guid [11], guid [12], guid [13], guid [14], guid [15]);
+ return result;
+}
+
+GList *
+dis_get_custom_attrs (MonoImage *m, guint32 token)
+{
+ GList *list = NULL;
+ guint32 idx, i, len, mtoken;
+ guint32 cols [MONO_CUSTOM_ATTR_SIZE];
+ MonoTableInfo *ca;
+ char *method;
+ GString *attr;
+ const char *val;
+
+ idx = mono_metadata_token_index (token);
+ idx <<= CUSTOM_ATTR_BITS;
+
+ switch (mono_metadata_token_table (token)) {
+ case MONO_TABLE_TYPEDEF:
+ idx |= CUSTOM_ATTR_TYPEDEF;
+ break;
+ case MONO_TABLE_ASSEMBLY:
+ idx |= CUSTOM_ATTR_ASSEMBLY;
+ break;
+ case MONO_TABLE_PROPERTY:
+ idx |= CUSTOM_ATTR_PROPERTY;
+ break;
+ case MONO_TABLE_EVENT:
+ idx |= CUSTOM_ATTR_EVENT;
+ break;
+ case MONO_TABLE_FIELD:
+ idx |= CUSTOM_ATTR_FIELDDEF;
+ break;
+ case MONO_TABLE_METHOD:
+ idx |= CUSTOM_ATTR_METHODDEF;
+ break;
+ case MONO_TABLE_PARAM:
+ idx |= CUSTOM_ATTR_PARAMDEF;
+ break;
+ default:
+ g_print ("Missing custom attr get support for token 0x%08x\n", token);
+ return NULL;
+ }
+
+ ca = &m->tables [MONO_TABLE_CUSTOMATTRIBUTE];
+ /* the table is not sorted */
+ for (i = 0; i < ca->rows; ++i) {
+ char *dump;
+ mono_metadata_decode_row (ca, i, cols, MONO_CUSTOM_ATTR_SIZE);
+ if (cols [MONO_CUSTOM_ATTR_PARENT] != idx)
+ continue;
+ mtoken = cols [MONO_CUSTOM_ATTR_TYPE] >> CUSTOM_ATTR_TYPE_BITS;
+ switch (cols [MONO_CUSTOM_ATTR_TYPE] & CUSTOM_ATTR_TYPE_MASK) {
+ case CUSTOM_ATTR_TYPE_METHODDEF:
+ mtoken |= MONO_TOKEN_METHOD_DEF;
+ break;
+ case CUSTOM_ATTR_TYPE_MEMBERREF:
+ mtoken |= MONO_TOKEN_MEMBER_REF;
+ break;
+ default:
+ g_error ("Unknown table for custom attr type %08x", cols [MONO_CUSTOM_ATTR_TYPE]);
+ break;
+ }
+ method = get_method (m, mtoken);
+ val = mono_metadata_blob_heap (m, cols [MONO_CUSTOM_ATTR_VALUE]);
+ len = mono_metadata_decode_value (val, &val);
+ attr = g_string_new (".custom ");
+ dump = data_dump (val, len, "\t\t");
+ g_string_sprintfa (attr, "%s = %s", method, dump);
+ g_free (dump);
+ list = g_list_append (list, attr->str);
+ g_string_free (attr, FALSE);
+ g_free (method);
+ }
+ return list;
+}
+
+char*
+get_marshal_info (MonoImage *m, const char *blob) {
+ int len, size = 0;
+
+ len = mono_metadata_decode_blob_size (blob, &blob);
+
+ switch (*blob) {
+ case MONO_NATIVE_BOOLEAN:
+ return g_strdup ("bool");
+ case MONO_NATIVE_I1:
+ return g_strdup ("int8");
+ case MONO_NATIVE_U1:
+ return g_strdup ("unsigned int8");
+ case MONO_NATIVE_I2:
+ return g_strdup ("int16");
+ case MONO_NATIVE_U2:
+ return g_strdup ("unsigned int16");
+ case MONO_NATIVE_I4:
+ return g_strdup ("int32");
+ case MONO_NATIVE_U4:
+ return g_strdup ("unsigned int32");
+ case MONO_NATIVE_I8:
+ return g_strdup ("int64");
+ case MONO_NATIVE_U8:
+ return g_strdup ("unsigned int64");
+ case MONO_NATIVE_R4:
+ return g_strdup ("float32");
+ case MONO_NATIVE_R8:
+ return g_strdup ("float64");
+ case MONO_NATIVE_CURRENCY:
+ return g_strdup ("currency");
+ case MONO_NATIVE_BSTR:
+ return g_strdup ("bstr");
+ case MONO_NATIVE_LPSTR:
+ return g_strdup ("lpstr");
+ case MONO_NATIVE_LPWSTR:
+ return g_strdup ("lpwstr");
+ case MONO_NATIVE_LPTSTR:
+ return g_strdup ("lptstr");
+ case MONO_NATIVE_BYVALTSTR:
+ size = mono_metadata_decode_value (blob + 1, &blob);
+ return g_strdup_printf ("fixed sysstring [%d]", size);
+ case MONO_NATIVE_IUNKNOWN:
+ return g_strdup ("iunknown");
+ case MONO_NATIVE_IDISPATCH:
+ return g_strdup ("idispatch");
+ case MONO_NATIVE_STRUCT:
+ return g_strdup ("struct");
+ case MONO_NATIVE_INTERFACE:
+ return g_strdup ("interface");
+ case MONO_NATIVE_SAFEARRAY:
+ return g_strdup ("safearray");
+ case MONO_NATIVE_BYVALARRAY:
+ size = mono_metadata_decode_value (blob + 1, &blob);
+ return g_strdup_printf ("fixed array [%d]", size);
+ case MONO_NATIVE_INT:
+ return g_strdup ("int");
+ case MONO_NATIVE_UINT:
+ return g_strdup ("unsigned int");
+ case MONO_NATIVE_VBBYREFSTR:
+ return g_strdup ("vbbyrefstr");
+ case MONO_NATIVE_ANSIBSTR:
+ return g_strdup ("ansi bstr");
+ case MONO_NATIVE_TBSTR:
+ return g_strdup ("tbstr");
+ case MONO_NATIVE_VARIANTBOOL:
+ return g_strdup ("variant bool");
+ case MONO_NATIVE_FUNC:
+ return g_strdup ("method");
+ case MONO_NATIVE_ASANY:
+ return g_strdup ("as any");
+ case MONO_NATIVE_LPARRAY:
+ return g_strdup ("[]");
+ case MONO_NATIVE_LPSTRUCT:
+ return g_strdup ("lpstruct");
+ case MONO_NATIVE_CUSTOM:
+ return g_strdup ("custom");
+ case MONO_NATIVE_ERROR:
+ return g_strdup ("error");
+ default:
+ return g_strdup ("unknown");
+ }
+}
+