| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352 | ## Author: Zoltan Varga (vargaz@gmail.com)# License: MIT/X11### This is a mono support mode for gdb 7.0 and later# Usage:# - copy/symlink this file to the directory where the mono executable lives.# - run mono under gdb, or attach to a mono process started with --debug=gdb using gdb.#from __future__ import print_functionimport osclass StringPrinter:    "Print a C# string"    def __init__(self, val):        self.val = val    def to_string(self):        if int(self.val.cast (gdb.lookup_type ("guint64"))) == 0:            return "null"        obj = self.val.cast (gdb.lookup_type ("MonoString").pointer ()).dereference ()        len = obj ['length']        chars = obj ['chars']        i = 0        res = ['"']        while i < len:            val = (chars.cast(gdb.lookup_type ("gint64")) + (i * 2)).cast(gdb.lookup_type ("gunichar2").pointer ()).dereference ()            if val >= 256:                c = unichr (val)            else:                c = chr (val)            res.append (c)            i = i + 1        res.append ('"')        return ''.join (res)def stringify_class_name(ns, name):    if ns == "System":        if name == "Byte":            return "byte"        if name == "String":            return "string"    if ns == "":        return name    else:        return "{0}.{1}".format (ns, name)class ArrayPrinter:    "Print a C# array"    def __init__(self, val, class_ns, class_name):        self.val = val        self.class_ns = class_ns        self.class_name = class_name    def to_string(self):        obj = self.val.cast (gdb.lookup_type ("MonoArray").pointer ()).dereference ()        length = obj ['max_length']        return "{0} [{1}]".format (stringify_class_name (self.class_ns, self.class_name [0:len(self.class_name) - 2]), int(length))        class ObjectPrinter:    "Print a C# object"    def __init__(self, val):        if str(val.type)[-1] == "&":            self.val = val.address.cast (gdb.lookup_type ("MonoObject").pointer ())        else:            self.val = val.cast (gdb.lookup_type ("MonoObject").pointer ())    class _iterator:        def __init__(self,obj):            self.obj = obj            self.iter = self.obj.type.fields ().__iter__ ()            pass        def __iter__(self):            return self        def next(self):            field = self.iter.next ()            try:                if str(self.obj [field.name].type) == "object":                    # Avoid recursion                    return (field.name, self.obj [field.name].cast (gdb.lookup_type ("void").pointer ()))                else:                    return (field.name, self.obj [field.name])            except:                # Superclass                return (field.name, self.obj.cast (gdb.lookup_type ("{0}".format (field.name))))    def children(self):        # FIXME: It would be easier if gdb.Value would support iteration itself        # It would also be better if we could return None        if int(self.val.cast (gdb.lookup_type ("guint64"))) == 0:            return {}.__iter__ ()        try:            obj = self.val.dereference ()            class_ns = obj ['vtable'].dereference ()['klass'].dereference ()['name_space'].string ()            class_name = obj ['vtable'].dereference ()['klass'].dereference ()['name'].string ()            if class_name [-2:len(class_name)] == "[]":                return {}.__iter__ ()            try:                gdb_type = gdb.lookup_type ("struct {0}_{1}".format (class_ns.replace (".", "_"), class_name))                return self._iterator(obj.cast (gdb_type))            except:                return {}.__iter__ ()        except:            print (sys.exc_info ()[0])            print (sys.exc_info ()[1])            return {}.__iter__ ()    def to_string(self):        if int(self.val.cast (gdb.lookup_type ("guint64"))) == 0:            return "null"        try:            obj = self.val.dereference ()            class_ns = obj ['vtable'].dereference ()['klass'].dereference ()['name_space'].string ()            class_name = obj ['vtable'].dereference ()['klass'].dereference ()['name'].string ()            if class_ns == "System" and class_name == "String":                return StringPrinter (self.val).to_string ()            if class_name [-2:len(class_name)] == "[]":                return ArrayPrinter (self.val,class_ns,class_name).to_string ()            if class_ns != "":                try:                    gdb_type = gdb.lookup_type ("struct {0}.{1}".format (class_ns, class_name))                except:                    # Maybe there is no debug info for that type                    return "{0}.{1}".format (class_ns, class_name)                #return obj.cast (gdb_type)                return "{0}.{1}".format (class_ns, class_name)            return class_name        except:            print (sys.exc_info ()[0])            print (sys.exc_info ()[1])            # FIXME: This can happen because we don't have liveness information            return self.val.cast (gdb.lookup_type ("guint64"))        class MonoMethodPrinter:    "Print a MonoMethod structure"    def __init__(self, val):        self.val = val    def to_string(self):        if int(self.val.cast (gdb.lookup_type ("guint64"))) == 0:            return "0x0"        val = self.val.dereference ()        klass = val ["klass"].dereference ()        class_name = stringify_class_name (klass ["name_space"].string (), klass ["name"].string ())        return "\"{0}:{1} ()\"".format (class_name, val ["name"].string ())        # This returns more info but requires calling into the inferior        #return "\"{0}\"".format (gdb.parse_and_eval ("mono_method_full_name ({0}, 1)".format (str (int (self.val.cast (gdb.lookup_type ("guint64")))))).string ())class MonoClassPrinter:    "Print a MonoClass structure"    def __init__(self, val):        self.val = val    def to_string_inner(self, add_quotes):        if int(self.val.cast (gdb.lookup_type ("guint64"))) == 0:            return "0x0"        klass = self.val.dereference ()        class_name = stringify_class_name (klass ["name_space"].string (), klass ["name"].string ())        if add_quotes:            return "\"{0}\"".format (class_name)        else:            return class_name        # This returns more info but requires calling into the inferior        #return "\"{0}\"".format (gdb.parse_and_eval ("mono_type_full_name (&((MonoClass*){0})->byval_arg)".format (str (int ((self.val).cast (gdb.lookup_type ("guint64")))))))    def to_string(self):        try:            return self.to_string_inner (True)        except:            #print (sys.exc_info ()[0])            #print (sys.exc_info ()[1])            return str(self.val.cast (gdb.lookup_type ("gpointer")))class MonoGenericInstPrinter:    "Print a MonoGenericInst structure"    def __init__(self, val):        self.val = val    def to_string(self):        if int(self.val.cast (gdb.lookup_type ("guint64"))) == 0:            return "0x0"        inst = self.val.dereference ()        inst_len = inst ["type_argc"]        inst_args = inst ["type_argv"]        inst_str = ""        for i in range(0, inst_len):            # print (inst_args)            type_printer = MonoTypePrinter (inst_args [i])            if i > 0:                inst_str = inst_str + ", "            inst_str = inst_str + type_printer.to_string ()        return inst_strclass MonoGenericClassPrinter:    "Print a MonoGenericClass structure"    def __init__(self, val):        self.val = val    def to_string_inner(self):        gclass = self.val.dereference ()        container_str = str(gclass ["container_class"])        class_inst = gclass ["context"]["class_inst"]        class_inst_str = ""        if int(class_inst.cast (gdb.lookup_type ("guint64"))) != 0:            class_inst_str  = str(class_inst)        method_inst = gclass ["context"]["method_inst"]        method_inst_str = ""        if int(method_inst.cast (gdb.lookup_type ("guint64"))) != 0:            method_inst_str  = str(method_inst)        return "{0}, [{1}], [{2}]>".format (container_str, class_inst_str, method_inst_str)    def to_string(self):        try:            return self.to_string_inner ()        except:            #print (sys.exc_info ()[0])            #print (sys.exc_info ()[1])            return str(self.val.cast (gdb.lookup_type ("gpointer")))class MonoTypePrinter:    "Print a MonoType structure"    def __init__(self, val):        self.val = val    def to_string_inner(self, csharp):        try:            t = self.val.referenced_value ()            kind = str (t ["type"]).replace ("MONO_TYPE_", "").lower ()            info = ""            if kind == "class":                p = MonoClassPrinter(t ["data"]["klass"])                info = p.to_string ()            elif kind == "genericinst":                info = str(t ["data"]["generic_class"])            if info != "":                return "{{{0}, {1}}}".format (kind, info)            else:                return "{{{0}}}".format (kind)        except:            #print (sys.exc_info ()[0])            #print (sys.exc_info ()[1])            return str(self.val.cast (gdb.lookup_type ("gpointer")))    def to_string(self):        return self.to_string_inner (False)class MonoMethodRgctxPrinter:    "Print a MonoMethodRgctx structure"    def __init__(self, val):        self.val = val    def to_string(self):        rgctx = self.val.dereference ()        klass = rgctx ["class_vtable"].dereference () ["klass"]        klass_printer = MonoClassPrinter (klass)        inst = rgctx ["method_inst"].dereference ()        inst_len = inst ["type_argc"]        inst_args = inst ["type_argv"]        inst_str = ""        for i in range(0, inst_len):            # print (inst_args)            type_printer = MonoTypePrinter (inst_args [i])            if i > 0:                inst_str = inst_str + ", "            inst_str = inst_str + type_printer.to_string ()        return "MRGCTX[{0}, [{1}]]".format (klass_printer.to_string(), inst_str)class MonoVTablePrinter:    "Print a MonoVTable structure"    def __init__(self, val):        self.val = val    def to_string(self):        if int(self.val.cast (gdb.lookup_type ("guint64"))) == 0:            return "0x0"        vtable = self.val.dereference ()        klass = vtable ["klass"]        klass_printer = MonoClassPrinter (klass)        return "vtable({0})".format (klass_printer.to_string ())def lookup_pretty_printer(val):    t = str (val.type)    if t == "object":        return ObjectPrinter (val)    if t[0:5] == "class" and t[-1] == "&":        return ObjectPrinter (val)        if t == "string":        return StringPrinter (val)    if t == "MonoString *":        return StringPrinter (val)    if t == "MonoMethod *":        return MonoMethodPrinter (val)    if t == "MonoClass *":        return MonoClassPrinter (val)    if t == "MonoType *":        return MonoTypePrinter (val)    if t == "MonoGenericInst *":        return MonoGenericInstPrinter (val)    if t == "MonoGenericClass *":        return MonoGenericClassPrinter (val)    if t == "MonoMethodRuntimeGenericContext *":        return MonoMethodRgctxPrinter (val)    if t == "MonoVTable *":        return MonoVTablePrinter (val)    return Nonedef register_csharp_printers(obj):    "Register C# pretty-printers with objfile Obj."    if obj == None:        obj = gdb    obj.pretty_printers.append (lookup_pretty_printer)# This command will flush the debugging info collected by the runtimeclass XdbCommand (gdb.Command):    def __init__ (self):        super (XdbCommand, self).__init__ ("xdb", gdb.COMMAND_NONE,                                           gdb.COMPLETE_COMMAND)    def invoke(self, arg, from_tty):        gdb.execute ("call mono_xdebug_flush ()")register_csharp_printers (gdb.current_objfile())XdbCommand ()gdb.execute ("set environment MONO_XDEBUG gdb")print ("Mono support loaded.")
 |