forked from qt-creator/qt-creator
		
	
		
			
				
	
	
		
			950 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			950 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| 
 | |
| #Note: Keep name-type-value-numchild-extra order
 | |
| 
 | |
| #return
 | |
| 
 | |
| import sys
 | |
| import traceback
 | |
| import gdb
 | |
| import base64
 | |
| import curses.ascii
 | |
| 
 | |
| # only needed for gdb 7.0/7.0.1 that do not implement parse_and_eval
 | |
| import os
 | |
| import tempfile
 | |
| 
 | |
| verbosity = 0
 | |
| verbosity = 1
 | |
| 
 | |
| def select(condition, if_expr, else_expr):
 | |
|     if condition:
 | |
|         return if_expr
 | |
|     return else_expr
 | |
| 
 | |
| def qmin(n, m):
 | |
|     if n < m:
 | |
|         return n
 | |
|     return m
 | |
| 
 | |
| def isGoodGdb():
 | |
|     #return gdb.VERSION.startswith("6.8.50.2009") \
 | |
|     #   and gdb.VERSION != "6.8.50.20090630-cvs"
 | |
|     return 'parse_and_eval' in dir(gdb)
 | |
| 
 | |
| def cleanAddress(addr):
 | |
|     # we cannot use str(addr) as it yields rubbish for char pointers
 | |
|     # that might trigger Unicode encoding errors
 | |
|     return addr.cast(gdb.lookup_type("void").pointer())
 | |
| 
 | |
| def parseAndEvaluate(exp):
 | |
|     if isGoodGdb():
 | |
|         return gdb.parse_and_eval(exp)
 | |
|     # Work around non-existing gdb.parse_and_eval as in released 7.0
 | |
|     gdb.execute("set logging redirect on")
 | |
|     gdb.execute("set logging on")
 | |
|     try:
 | |
|         gdb.execute("print %s" % exp)
 | |
|     except:
 | |
|         gdb.execute("set logging off")
 | |
|         return None
 | |
|     gdb.execute("set logging off")
 | |
|     return gdb.history(0)
 | |
| 
 | |
| def listOfLocals():
 | |
|     try:
 | |
|         frame = gdb.selected_frame()
 | |
|         #warn("FRAME %s: " % frame)
 | |
|     except RuntimeError:
 | |
|         return []
 | |
| 
 | |
|     items = []
 | |
|     if isGoodGdb():
 | |
|         # archer-tromey-python
 | |
|         block = frame.block()
 | |
|         while True:
 | |
|             if block is None:
 | |
|                 warn("UNEXPECTED 'None' BLOCK")
 | |
|                 break
 | |
|             for symbol in block:
 | |
|                 name = symbol.print_name
 | |
| 
 | |
|                 if name == "__in_chrg":
 | |
|                     continue
 | |
| 
 | |
|                 # "NotImplementedError: Symbol type not yet supported in
 | |
|                 # Python scripts."
 | |
|                 #warn("SYMBOL %s: " % symbol.value)
 | |
|                 #warn("SYMBOL %s  (%s): " % (symbol, name))
 | |
|                 item = Item(0, "local", name, name)
 | |
|                 try:
 | |
|                     item.value = frame.read_var(name)  # this is a gdb value
 | |
|                 except RuntimeError:
 | |
|                     # happens for  void foo() { std::string s; std::wstring w; }
 | |
|                     #warn("  FRAME READ VAR ERROR: %s (%s): " % (symbol, name))
 | |
|                     continue
 | |
|                 #warn("ITEM %s: " % item.value)
 | |
|                 items.append(item)
 | |
|             # The outermost block in a function has the function member
 | |
|             # FIXME: check whether this is guaranteed.
 | |
|             if not block.function is None:
 | |
|                 break
 | |
| 
 | |
|             block = block.superblock
 | |
|     else:
 | |
|         # Assuming gdb 7.0 release or 6.8-symbianelf.
 | |
|         file = tempfile.mkstemp(prefix="gdbpy_")
 | |
|         filename = file[1]
 | |
|         gdb.execute("set logging off")
 | |
|         gdb.execute("set logging redirect off")
 | |
|         gdb.execute("set logging file %s" % filename)
 | |
|         gdb.execute("set logging redirect on")
 | |
|         gdb.execute("set logging on")
 | |
|         gdb.execute("info locals")
 | |
|         gdb.execute("info args")
 | |
|         gdb.execute("set logging off")
 | |
|         gdb.execute("set logging redirect off")
 | |
|         file = open(filename, "r")
 | |
|         for line in file:
 | |
|             if len(line) == 0 or line.startswith(" "):
 | |
|                 continue
 | |
|             pos = line.find(" = ")
 | |
|             if pos < 0:
 | |
|                 continue
 | |
|             name = line[0:pos]
 | |
|             item = Item(0, "local", name, name)
 | |
|             try:
 | |
|                 item.value = frame.read_var(name)  # this is a gdb value
 | |
|             except RuntimeError:
 | |
|                 continue
 | |
|             items.append(item)
 | |
|         file.close()
 | |
|         os.remove(filename)
 | |
| 
 | |
|     return items
 | |
| 
 | |
| 
 | |
| def value(expr):
 | |
|     value = parseAndEvaluate(expr)
 | |
|     try:
 | |
|         return int(value)
 | |
|     except:
 | |
|         return str(value)
 | |
| 
 | |
| 
 | |
| def isSimpleType(typeobj):
 | |
|     if typeobj.code == gdb.TYPE_CODE_PTR:
 | |
|         return False
 | |
|     type = str(typeobj)
 | |
|     return type == "bool" \
 | |
|         or type == "char" \
 | |
|         or type == "double" \
 | |
|         or type == "float" \
 | |
|         or type == "int" \
 | |
|         or type == "long" or type.startswith("long ") \
 | |
|         or type == "short" or type.startswith("short ") \
 | |
|         or type == "signed" or type.startswith("signed ") \
 | |
|         or type == "unsigned" or type.startswith("unsigned ")
 | |
| 
 | |
| 
 | |
| def isStringType(d, typeobj):
 | |
|     type = str(typeobj)
 | |
|     return type == d.ns + "QString" \
 | |
|         or type == d.ns + "QByteArray" \
 | |
|         or type == "std::string" \
 | |
|         or type == "std::wstring" \
 | |
|         or type == "wstring"
 | |
| 
 | |
| def warn(message):
 | |
|     if verbosity > 0:
 | |
|         print "XXX: %s " % message.encode("latin1")
 | |
|     pass
 | |
| 
 | |
| def check(exp):
 | |
|     if not exp:
 | |
|         raise RuntimeError("Check failed")
 | |
| 
 | |
| def checkRef(ref):
 | |
|     count = ref["_q_value"]
 | |
|     check(count > 0)
 | |
|     check(count < 1000000) # assume there aren't a million references to any object
 | |
| 
 | |
| #def couldBePointer(p, align):
 | |
| #    type = gdb.lookup_type("unsigned int")
 | |
| #    ptr = gdb.Value(p).cast(type)
 | |
| #    d = int(str(ptr))
 | |
| #    warn("CHECKING : %s %d " % (p, ((d & 3) == 0 and (d > 1000 or d == 0))))
 | |
| #    return (d & (align - 1)) and (d > 1000 or d == 0)
 | |
| 
 | |
| 
 | |
| def checkAccess(p, align = 1):
 | |
|     return p.dereference()
 | |
| 
 | |
| def checkContents(p, expected, align = 1):
 | |
|     if int(p.dereference()) != expected:
 | |
|         raise RuntimeError("Contents check failed")
 | |
| 
 | |
| def checkPointer(p, align = 1):
 | |
|     if not isNull(p):
 | |
|         p.dereference()
 | |
| 
 | |
| 
 | |
| def isNull(p):
 | |
|     # The following can cause evaluation to abort with "UnicodeEncodeError"
 | |
|     # for invalid char *, as their "contents" is being examined
 | |
|     #s = str(p)
 | |
|     #return s == "0x0" or s.startswith("0x0 ")
 | |
|     return p.cast(gdb.lookup_type("void").pointer()) == 0
 | |
| 
 | |
| movableTypes = set([
 | |
|     "QBrush", "QBitArray", "QByteArray",
 | |
|     "QCustomTypeInfo", "QChar",
 | |
|     "QDate", "QDateTime",
 | |
|     "QFileInfo", "QFixed", "QFixedPoint", "QFixedSize",
 | |
|     "QHashDummyValue",
 | |
|     "QIcon", "QImage",
 | |
|     "QLine", "QLineF", "QLatin1Char", "QLocal",
 | |
|     "QMatrix", "QModelIndex",
 | |
|     "QPoint", "QPointF", "QPen", "QPersistentModelIndex",
 | |
|     "QResourceRoot", "QRect", "QRectF", "QRegExp",
 | |
|     "QSize", "QSizeF", "QString",
 | |
|     "QTime", "QTextBlock",
 | |
|     "QUrl",
 | |
|     "QVariant",
 | |
|     "QXmlStreamAttribute", "QXmlStreamNamespaceDeclaration",
 | |
|     "QXmlStreamNotationDeclaration", "QXmlStreamEntityDeclaration"])
 | |
| 
 | |
| 
 | |
| def stripClassTag(type):
 | |
|     if type.startswith("class "):
 | |
|         return type[6:]
 | |
|     return type
 | |
| 
 | |
| def checkPointerRange(p, n):
 | |
|     for i in xrange(n):
 | |
|         checkPointer(p)
 | |
|         ++p
 | |
| 
 | |
| def call(value, func):
 | |
|     #warn("CALL: %s -> %s" % (value, func))
 | |
|     type = stripClassTag(str(value.type))
 | |
|     if type.find(":") >= 0:
 | |
|         type = "'" + type + "'"
 | |
|     exp = "((%s*)%s)->%s" % (type, value.address, func)
 | |
|     #warn("CALL: %s" % exp)
 | |
|     result = parseAndEvaluate(exp)
 | |
|     #warn("  -> %s" % result)
 | |
|     return result
 | |
| 
 | |
| def qtNamespace():
 | |
|     try:
 | |
|         type = str(parseAndEvaluate("&QString::null").type.target().unqualified())
 | |
|         return type[0:len(type) - len("QString::null")]
 | |
|     except RuntimeError:
 | |
|         return ""
 | |
| 
 | |
| def encodeCharArray(p, size):
 | |
|     s = ""
 | |
|     p = p.cast(gdb.lookup_type("unsigned char").pointer())
 | |
|     for i in xrange(size):
 | |
|         s += "%02x" % int(p.dereference())
 | |
|         p += 1
 | |
|     return s
 | |
| 
 | |
| def encodeByteArray(value):
 | |
|     d_ptr = value['d'].dereference()
 | |
|     data = d_ptr['data']
 | |
|     size = d_ptr['size']
 | |
|     alloc = d_ptr['alloc']
 | |
|     check(0 <= size and size <= alloc and alloc <= 100*1000*1000)
 | |
|     checkRef(d_ptr["ref"])
 | |
|     if size > 0:
 | |
|         checkAccess(data, 4)
 | |
|         checkAccess(data + size) == 0
 | |
| 
 | |
|     innerType = gdb.lookup_type("char")
 | |
|     p = gdb.Value(data.cast(innerType.pointer()))
 | |
|     return encodeCharArray(p, size)
 | |
| 
 | |
| def encodeString(value):
 | |
|     d_ptr = value['d'].dereference()
 | |
|     data = d_ptr['data']
 | |
|     size = d_ptr['size']
 | |
|     alloc = d_ptr['alloc']
 | |
|     check(0 <= size and size <= alloc and alloc <= 100*1000*1000)
 | |
|     if size > 0:
 | |
|         checkAccess(data, 4)
 | |
|         checkAccess(data + size * 2) == 0
 | |
|     checkRef(d_ptr["ref"])
 | |
|     p = gdb.Value(d_ptr["data"])
 | |
|     s = ""
 | |
|     for i in xrange(size):
 | |
|         val = int(p.dereference())
 | |
|         s += "%02x" % (val % 256)
 | |
|         s += "%02x" % (val / 256)
 | |
|         p += 1
 | |
|     return s
 | |
| 
 | |
| #######################################################################
 | |
| #
 | |
| # Item
 | |
| #
 | |
| #######################################################################
 | |
| 
 | |
| class Item:
 | |
|     def __init__(self, value, parentiname, iname, name = None):
 | |
|         self.value = value
 | |
|         if iname is None:
 | |
|             self.iname = parentiname
 | |
|         else:
 | |
|             self.iname = "%s.%s" % (parentiname, iname)
 | |
|         self.name = name
 | |
| 
 | |
| 
 | |
| #######################################################################
 | |
| #
 | |
| # FrameCommand
 | |
| #
 | |
| #######################################################################
 | |
| 
 | |
| class FrameCommand(gdb.Command):
 | |
|     """Do fancy stuff. Usage bb --verbose expandedINames"""
 | |
| 
 | |
|     def __init__(self):
 | |
|         super(FrameCommand, self).__init__("bb", gdb.COMMAND_OBSCURE)
 | |
| 
 | |
|     def invoke(self, arg, from_tty):
 | |
|         args = arg.split(' ')
 | |
|         #warn("ARG: %s" % arg)
 | |
|         #warn("ARGS: %s" % args)
 | |
|         options = args[0].split(",")
 | |
|         useFancy = "fancy" in options
 | |
|         expandedINames = set()
 | |
|         if len(args) > 1:
 | |
|             expandedINames = set(args[1].split(","))
 | |
|         #warn("EXPANDED INAMES: %s" % expandedINames)
 | |
|         module = sys.modules[__name__]
 | |
|         self.dumpers = {}
 | |
| 
 | |
|         if useFancy == -1:
 | |
|             output = "dumpers=["
 | |
|             for key, value in module.__dict__.items():
 | |
|                 if key.startswith("qdump__"):
 | |
|                     if output != "dumpers=[":
 | |
|                         output += ","
 | |
|                     output += '"' + key[7:] + '"'
 | |
|             output += "],"
 | |
|             #output += "qtversion=[%d,%d,%d]"
 | |
|             output += "qtversion=[4,6,0],"
 | |
|             output += "namespace=\"%s\"," % qtNamespace()
 | |
|             output += "dumperversion=\"2.0\","
 | |
|             output += "sizes=[],"
 | |
|             output += "expressions=[]"
 | |
|             output += "]"
 | |
|             print output
 | |
|             return
 | |
| 
 | |
| 
 | |
|         if useFancy:
 | |
|             for key, value in module.__dict__.items():
 | |
|                 #if callable(value):
 | |
|                 if key.startswith("qdump__"):
 | |
|                     self.dumpers[key[7:]] = value
 | |
| 
 | |
|         d = Dumper()
 | |
|         d.dumpers = self.dumpers
 | |
|         d.useFancy = useFancy
 | |
|         d.passExceptions = "passexceptions" in options
 | |
|         d.autoDerefPointers = "autoderef" in options
 | |
|         d.ns = qtNamespace()
 | |
|         d.expandedINames = expandedINames
 | |
|         #warn(" NAMESPACE IS: '%s'" % d.ns)
 | |
| 
 | |
|         #
 | |
|         # Locals
 | |
|         #
 | |
|         for item in listOfLocals():
 | |
|             #warn("ITEM NAME %s: " % item.name)
 | |
|             #warn("ITEM VALUE %s: " % item.value)
 | |
| 
 | |
|             type = item.value.type
 | |
|             if type.code == gdb.TYPE_CODE_PTR \
 | |
|                     and item.name == "argv" and str(type) == "char **":
 | |
|                 # Special handling for char** argv:
 | |
|                 n = 0
 | |
|                 p = item.value
 | |
|                 while not isNull(p.dereference()) and n <= 100:
 | |
|                     p += 1
 | |
|                     n += 1
 | |
| 
 | |
|                 d.beginHash()
 | |
|                 d.put('iname="%s",' % item.iname)
 | |
|                 d.putName(item.name)
 | |
|                 d.putItemCount(select(n <= 100, n, "> 100"))
 | |
|                 d.putType(type)
 | |
|                 d.putNumChild(n)
 | |
|                 if d.isExpanded(item):
 | |
|                     p = item.value
 | |
|                     d.beginChildren(n)
 | |
|                     for i in xrange(n):
 | |
|                         value = p.dereference()
 | |
|                         d.putItem(Item(value, item.iname, i, None))
 | |
|                         p += 1
 | |
|                     if n > 100:
 | |
|                         d.putEllipsis()
 | |
|                     d.endChildren()
 | |
|                 d.endHash()
 | |
| 
 | |
|             else:
 | |
|                 # A "normal" local variable or parameter
 | |
|                 try:
 | |
|                     addr = cleanAddress(item.value.address)
 | |
|                     d.beginHash()
 | |
|                     d.put('iname="%s",' % item.iname)
 | |
|                     d.put('addr="%s",' % addr)
 | |
|                     d.safePutItemHelper(item)
 | |
|                     d.endHash()
 | |
|                 except AttributeError:
 | |
|                     # thrown by cleanAddreas with message
 | |
|                     # "'NoneType' object has no attribute 'cast'"
 | |
|                     # for optimized-out values
 | |
|                     d.beginHash()
 | |
|                     d.put('iname="%s",' % item.iname)
 | |
|                     d.put('name="%s",' % item.name)
 | |
|                     d.put('addr="<optimized out>",')
 | |
|                     d.put('value="<optimized out>",')
 | |
|                     d.put('type="%s"' % item.value.type)
 | |
|                     d.endHash()
 | |
| 
 | |
|         d.pushOutput()
 | |
|         locals = d.safeoutput
 | |
| 
 | |
| 
 | |
|         #
 | |
|         # Watchers
 | |
|         #
 | |
|         d.safeoutput = ""
 | |
|         watchers = ""
 | |
|         if len(args) > 2:
 | |
|             watchers = base64.b16decode(args[2], True)
 | |
|         if len(watchers) > 0:
 | |
|             for watcher in watchers.split("##"):
 | |
|                 (exp, iname) = watcher.split("#")
 | |
|                 self.handleWatch(d, exp, iname)
 | |
|         d.pushOutput()
 | |
|         watchers = d.safeoutput
 | |
| 
 | |
|         print('data=[' + locals + ',' + watchers + ']\n')
 | |
| 
 | |
| 
 | |
|     def handleWatch(self, d, exp, iname):
 | |
|         #warn("HANDLING WATCH %s, INAME: '%s'" % (exp, iname))
 | |
|         if exp.startswith("["):
 | |
|             warn("EVAL: EXP: %s" % exp)
 | |
|             d.beginHash()
 | |
|             d.put('iname="%s",' % iname)
 | |
|             d.put('name="%s",' % exp)
 | |
|             d.put('exp="%s",' % exp)
 | |
|             try:
 | |
|                 list = eval(exp)
 | |
|                 #warn("EVAL: LIST: %s" % list)
 | |
|                 d.put('value=" "')
 | |
|                 d.put('type=" "')
 | |
|                 d.put('numchild="%d"' % len(list))
 | |
|                 # This is a list of expressions to evaluate
 | |
|                 d.beginChildren(len(list))
 | |
|                 itemNumber = 0
 | |
|                 for item in list:
 | |
|                     self.handleWatch(d, item, "%s.%d" % (iname, itemNumber))
 | |
|                     itemNumber += 1
 | |
|                 d.endChildren()
 | |
|             except:
 | |
|                 warn("EVAL: ERROR CAUGHT")
 | |
|                 d.put('value="<syntax error>"')
 | |
|                 d.put('type=" "')
 | |
|                 d.put('numchild="0"')
 | |
|                 d.beginChildren(0)
 | |
|                 d.endChildren()
 | |
|             d.endHash()
 | |
|             return
 | |
| 
 | |
|         d.beginHash()
 | |
|         d.put('iname="%s",' % iname)
 | |
|         d.put('name="%s",' % exp)
 | |
|         d.put('exp="%s",' % exp)
 | |
|         handled = False
 | |
|         if exp == "<Edit>":
 | |
|             d.put(',value=" ",')
 | |
|             d.put('type=" ",numchild="0"')
 | |
|         else:
 | |
|             try:
 | |
|                 value = parseAndEvaluate(exp)
 | |
|                 item = Item(value, iname, None, None)
 | |
|                 d.putItemHelper(item)
 | |
|             except RuntimeError:
 | |
|                 d.put(',value="<invalid>",')
 | |
|                 d.put('type="<unknown>",numchild="0"')
 | |
|         d.endHash()
 | |
| 
 | |
| 
 | |
| FrameCommand()
 | |
| 
 | |
| 
 | |
| 
 | |
| #######################################################################
 | |
| #
 | |
| # The Dumper Class
 | |
| #
 | |
| #######################################################################
 | |
| 
 | |
| class Dumper:
 | |
|     def __init__(self):
 | |
|         self.output = ""
 | |
|         self.safeoutput = ""
 | |
|         self.childTypes = [""]
 | |
|         self.childNumChilds = [-1]
 | |
|         self.maxNumChilds = [-1]
 | |
|         self.numChilds = [-1]
 | |
| 
 | |
|     def put(self, value):
 | |
|         self.output += value
 | |
| 
 | |
|     def putCommaIfNeeded(self):
 | |
|         c = self.output[-1:]
 | |
|         if c == '}' or c == '"' or c == ']' or c == '\n':
 | |
|             self.put(',')
 | |
|         #warn("C:'%s' COND:'%d' OUT:'%s'" %
 | |
|         #    (c, c == '}' or c == '"' or c == ']' or c == '\n', self.output))
 | |
| 
 | |
|     def putField(self, name, value):
 | |
|         self.putCommaIfNeeded()
 | |
|         self.put('%s="%s"' % (name, value))
 | |
| 
 | |
|     def beginHash(self):
 | |
|         self.putCommaIfNeeded()
 | |
|         self.put('{')
 | |
| 
 | |
|     def endHash(self):
 | |
|         self.put('}')
 | |
| 
 | |
|     def beginItem(self, name):
 | |
|         self.putCommaIfNeeded()
 | |
|         self.put(name)
 | |
|         self.put('="')
 | |
| 
 | |
|     def endItem(self):
 | |
|         self.put('"')
 | |
| 
 | |
|     def beginChildren(self, numChild_ = 1, childType_ = None, childNumChild_ = None):
 | |
|         childType = ""
 | |
|         childNumChild = -1
 | |
|         if type(numChild_) is list:
 | |
|             numChild = numChild_[0]
 | |
|             maxNumChild = numChild_[1]
 | |
|         else:
 | |
|             numChild = numChild_
 | |
|             maxNumChild = numChild_
 | |
|         if numChild == 0:
 | |
|             childType_ = None
 | |
|         self.putCommaIfNeeded()
 | |
|         if not childType_ is None:
 | |
|             childType = stripClassTag(str(childType_))
 | |
|             self.put('childtype="%s",' % childType)
 | |
|             if isSimpleType(childType_) or isStringType(self, childType_):
 | |
|                 self.put('childnumchild="0",')
 | |
|                 childNumChild = 0
 | |
|             elif childType_.code == gdb.TYPE_CODE_PTR:
 | |
|                 self.put('childnumchild="1",')
 | |
|                 childNumChild = 1
 | |
|         if not childNumChild_ is None:
 | |
|             self.put('childnumchild="%s",' % childNumChild_)
 | |
|             childNumChild = childNumChild_
 | |
|         self.childTypes.append(childType)
 | |
|         self.childNumChilds.append(childNumChild)
 | |
|         self.numChilds.append(numChild)
 | |
|         self.maxNumChilds.append(maxNumChild)
 | |
|         #warn("BEGIN: %s" % self.childTypes)
 | |
|         self.put("children=[")
 | |
| 
 | |
|     def endChildren(self):
 | |
|         #warn("END: %s" % self.childTypes)
 | |
|         numChild = self.numChilds.pop()
 | |
|         maxNumChild = self.maxNumChilds.pop()
 | |
|         if maxNumChild < numChild:
 | |
|             self.putEllipsis();
 | |
|         self.childTypes.pop()
 | |
|         self.childNumChilds.pop()
 | |
|         self.put(']')
 | |
| 
 | |
|     def childRange(self):
 | |
|         return xrange(qmin(self.maxNumChilds[-1], self.numChilds[-1]))
 | |
| 
 | |
|     # convenience
 | |
|     def putItemCount(self, count):
 | |
|         self.putCommaIfNeeded()
 | |
|         self.put('value="<%s items>"' % count)
 | |
| 
 | |
|     def putEllipsis(self):
 | |
|         self.putCommaIfNeeded()
 | |
|         self.put('{name="<incomplete>",value="",type="",numchild="0"}')
 | |
| 
 | |
|     def putType(self, type):
 | |
|         #warn("TYPES: '%s' '%s'" % (type, self.childTypes))
 | |
|         #warn("  EQUAL 2: %s " % (str(type) == self.childTypes[-1]))
 | |
|         type = stripClassTag(str(type))
 | |
|         if len(type) > 0 and type != self.childTypes[-1]:
 | |
|             self.putCommaIfNeeded()
 | |
|             self.put('type="%s"' % type) # str(type.unqualified()) ?
 | |
| 
 | |
|     def putNumChild(self, numchild):
 | |
|         #warn("NUM CHILD: '%s' '%s'" % (numchild, self.childNumChilds[-1]))
 | |
|         if numchild != self.childNumChilds[-1]:
 | |
|             self.put(',numchild="%s"' % numchild)
 | |
| 
 | |
|     def putValue(self, value, encoding = None):
 | |
|         if not encoding is None:
 | |
|             self.putField("valueencoded", encoding)
 | |
|         self.putField("value", value)
 | |
| 
 | |
|     def putStringValue(self, value):
 | |
|         str = encodeString(value)
 | |
|         self.putCommaIfNeeded()
 | |
|         self.put('valueencoded="%d",value="%s"' % (7, str))
 | |
| 
 | |
|     def putByteArrayValue(self, value):
 | |
|         str = encodeByteArray(value)
 | |
|         self.putCommaIfNeeded()
 | |
|         self.put('valueencoded="%d",value="%s"' % (6, str))
 | |
| 
 | |
|     def putName(self, name):
 | |
|         self.putCommaIfNeeded()
 | |
|         self.put('name="%s"' % name)
 | |
| 
 | |
|     def isExpanded(self, item):
 | |
|         #warn("IS EXPANDED: %s in %s" % (item.iname, self.expandedINames))
 | |
|         if item.iname is None:
 | |
|             raise "Illegal iname 'None'"
 | |
|         if item.iname.startswith("None"):
 | |
|             raise "Illegal iname '%s'" % item.iname
 | |
|         #warn("   --> %s" % (item.iname in self.expandedINames))
 | |
|         return item.iname in self.expandedINames
 | |
| 
 | |
|     def isExpandedIName(self, iname):
 | |
|         return iname in self.expandedINames
 | |
| 
 | |
|     def unputField(self, name):
 | |
|         pos = self.output.rfind(",")
 | |
|         if self.output[pos + 1:].startswith(name):
 | |
|             self.output = self.output[0:pos]
 | |
| 
 | |
|     def stripNamespaceFromType(self, typeobj):
 | |
|         # This breaks for dumpers type names containing '__star'.
 | |
|         # But this should not happen as identifiers containing two
 | |
|         # subsequent underscores are reserved for the implemention.
 | |
|         if typeobj.code == gdb.TYPE_CODE_PTR:
 | |
|             return self.stripNamespaceFromType(typeobj.target()) + "__star"
 | |
|         # FIXME: pass ns from plugin
 | |
|         type = stripClassTag(str(typeobj))
 | |
|         if len(self.ns) > 0 and type.startswith(self.ns):
 | |
|             type = type[len(self.ns):]
 | |
|         pos = type.find("<")
 | |
|         if pos != -1:
 | |
|             type = type[0:pos]
 | |
|         return type
 | |
| 
 | |
|     def isMovableType(self, type):
 | |
|         if type.code == gdb.TYPE_CODE_PTR:
 | |
|             return True
 | |
|         if isSimpleType(type):
 | |
|             return True
 | |
|         return self.stripNamespaceFromType(type) in movableTypes
 | |
| 
 | |
|     def putIntItem(self, name, value):
 | |
|         self.beginHash()
 | |
|         self.putName(name)
 | |
|         self.putValue(value)
 | |
|         self.putType("int")
 | |
|         self.putNumChild(0)
 | |
|         self.endHash()
 | |
| 
 | |
|     def putBoolItem(self, name, value):
 | |
|         self.beginHash()
 | |
|         self.putName(name)
 | |
|         self.putValue(value)
 | |
|         self.putType("bool")
 | |
|         self.putNumChild(0)
 | |
|         self.endHash()
 | |
| 
 | |
|     def pushOutput(self):
 | |
|         #warn("PUSH OUTPUT: %s " % self.output)
 | |
|         self.safeoutput += self.output
 | |
|         self.output = ""
 | |
| 
 | |
|     def dumpInnerValueHelper(self, item):
 | |
|         if isSimpleType(item.value.type):
 | |
|             self.safePutItemHelper(item)
 | |
| 
 | |
|     def safePutItem(self, item):
 | |
|         self.beginHash()
 | |
|         self.safePutItemHelper(item)
 | |
|         self.endHash()
 | |
| 
 | |
|     def safePutItemHelper(self, item):
 | |
|         self.pushOutput()
 | |
|         # This is only used at the top level to ensure continuation
 | |
|         # after failures due to uninitialized or corrupted data.
 | |
|         if self.passExceptions:
 | |
|             # for debugging reasons propagate errors.
 | |
|             self.putItemHelper(item)
 | |
| 
 | |
|         else:
 | |
|             try:
 | |
|                 self.putItemHelper(item)
 | |
| 
 | |
|             except RuntimeError:
 | |
|                 self.output = ""
 | |
|                 # FIXME: Only catch debugger related exceptions
 | |
|                 #exType, exValue, exTraceback = sys.exc_info()
 | |
|                 #tb = traceback.format_exception(exType, exValue, exTraceback)
 | |
|                 #warn("Exception: %s" % ex.message)
 | |
|                 # DeprecationWarning: BaseException.message
 | |
|                 # has been deprecated
 | |
|                 #warn("Exception.")
 | |
|                 #for line in tb:
 | |
|                 #    warn("%s" % line)
 | |
|                 self.putName(item.name)
 | |
|                 self.putValue("<invalid>")
 | |
|                 self.putType(str(item.value.type))
 | |
|                 self.putNumChild(0)
 | |
|                 #if self.isExpanded(item):
 | |
|                 self.beginChildren()
 | |
|                 self.endChildren()
 | |
|         self.pushOutput()
 | |
| 
 | |
|     def putItem(self, item):
 | |
|         self.beginHash()
 | |
|         self.safePutItemHelper(item)
 | |
|         self.endHash()
 | |
| 
 | |
|     def putCallItem(self, name, item, func):
 | |
|         result = call(item.value, func)
 | |
|         self.putItem(Item(result, item.iname, name, name))
 | |
| 
 | |
|     def putItemHelper(self, item):
 | |
|         name = getattr(item, "name", None)
 | |
|         if not name is None:
 | |
|             self.putName(name)
 | |
| 
 | |
|         if item.value is None:
 | |
|             # Happens for non-available watchers in gdb versions that
 | |
|             # need to use gdb.execute instead of gdb.parse_and_eval
 | |
|             self.putValue("<not available>")
 | |
|             self.putType("<unknown>")
 | |
|             self.putNumChild(0)
 | |
|             return
 | |
| 
 | |
|         # FIXME: Gui shows references stripped?
 | |
|         #warn(" ");
 | |
|         #warn("REAL INAME: %s " % item.iname)
 | |
|         #warn("REAL NAME: %s " % name)
 | |
|         #warn("REAL TYPE: %s " % item.value.type)
 | |
|         #warn("REAL VALUE: %s " % item.value)
 | |
|         #try:
 | |
|         #    warn("REAL VALUE: %s " % item.value)
 | |
|         #except UnicodeEncodeError:
 | |
|         #    warn("REAL VALUE: <unprintable>")
 | |
| 
 | |
|         value = item.value
 | |
|         type = value.type
 | |
| 
 | |
|         if type.code == gdb.TYPE_CODE_REF:
 | |
|             type = type.target()
 | |
|             value = value.cast(type)
 | |
| 
 | |
|         if type.code == gdb.TYPE_CODE_TYPEDEF:
 | |
|             type = type.target()
 | |
| 
 | |
|         strippedType = self.stripNamespaceFromType(
 | |
|             type.strip_typedefs().unqualified()).replace("::", "__")
 | |
| 
 | |
|         #warn(" STRIPPED: %s" % strippedType)
 | |
|         #warn(" DUMPERS: %s" % self.dumpers)
 | |
|         #warn(" DUMPERS: %s" % (strippedType in self.dumpers))
 | |
| 
 | |
|         if isSimpleType(type.unqualified()):
 | |
|             #warn("IS SIMPLE: %s " % type)
 | |
|             self.putType(item.value.type)
 | |
|             self.putValue(value)
 | |
|             self.putNumChild(0)
 | |
| 
 | |
|         elif strippedType in self.dumpers:
 | |
|             #warn("IS DUMPABLE: %s " % type)
 | |
|             self.putType(item.value.type)
 | |
|             self.dumpers[strippedType](self, item)
 | |
|             #warn(" RESULT: %s " % self.output)
 | |
| 
 | |
|         elif type.code == gdb.TYPE_CODE_ENUM:
 | |
|             #warn("GENERIC ENUM: %s" % value)
 | |
|             self.putType(item.value.type)
 | |
|             self.putValue(value)
 | |
|             self.putNumChild(0)
 | |
| 
 | |
| 
 | |
|         elif type.code == gdb.TYPE_CODE_PTR:
 | |
|             isHandled = False
 | |
|             #warn("A POINTER: %s" % value.type)
 | |
|             if self.useFancy:
 | |
|                 if isNull(value):
 | |
|                     self.putValue("0x0")
 | |
|                     self.putType(item.value.type)
 | |
|                     self.putNumChild(0)
 | |
|                     isHandled = True
 | |
| 
 | |
|                 target = str(type.target().strip_typedefs().unqualified())
 | |
|                 if (not isHandled) and target == "void":
 | |
|                     self.putType(item.value.type)
 | |
|                     self.putValue(str(value))
 | |
|                     self.putNumChild(0)
 | |
|                     isHandled = True
 | |
| 
 | |
|                 #warn("TARGET: %s " % target) 
 | |
|                 if (not isHandled) and (target == "char"
 | |
|                         or target == "signed char" or target == "unsigned char"):
 | |
|                     # Display values up to given length directly
 | |
|                     #warn("CHAR AUTODEREF: %s" % value.address)
 | |
|                     self.putType(item.value.type)
 | |
|                     firstNul = -1
 | |
|                     p = value
 | |
|                     found = False
 | |
|                     for i in xrange(100):
 | |
|                         if p.dereference() == 0:
 | |
|                             # Found terminating NUL
 | |
|                             found = True
 | |
|                             break
 | |
|                         p += 1
 | |
|                     if found:
 | |
|                         self.putValue(encodeCharArray(value, i), "6")
 | |
|                         self.putNumChild(0)
 | |
|                     else:
 | |
|                         self.putValue(encodeCharArray(value, 100) + "2e2e2e", "6")
 | |
|                         self.putNumChild(0)
 | |
|                     isHandled = True
 | |
| 
 | |
|             #warn("AUTODEREF: %s" % self.autoDerefPointers)
 | |
|             #warn("IS HANDLED: %s" % isHandled)
 | |
|             #warn("RES: %s" % (self.autoDerefPointers and not isHandled))
 | |
|             if self.autoDerefPointers and not isHandled:
 | |
|                 ## Generic pointer type.
 | |
|                 #warn("GENERIC AUTODEREF POINTER: %s" % value.address)
 | |
|                 innerType = item.value.type.target()
 | |
|                 self.putType(innerType)
 | |
|                 self.childTypes.append(
 | |
|                     stripClassTag(str(innerType)))
 | |
|                 self.putItemHelper(
 | |
|                     Item(item.value.dereference(), item.iname, None, None))
 | |
|                 self.childTypes.pop()
 | |
|                 isHandled = True
 | |
| 
 | |
|             # Fall back to plain pointer printing
 | |
|             if not isHandled:
 | |
|                 #warn("GENERIC PLAIN POINTER: %s" % value.type)
 | |
|                 self.putType(item.value.type)
 | |
|                 self.putValue(str(value.address))
 | |
|                 self.putNumChild(1)
 | |
|                 if self.isExpanded(item):
 | |
|                     self.beginChildren()
 | |
|                     self.putItem(
 | |
|                           Item(item.value.dereference(), item.iname, "*", "*"))
 | |
|                     self.endChildren()
 | |
| 
 | |
|         else:
 | |
|             #warn("GENERIC STRUCT: %s" % value.type)
 | |
|             #warn("INAME: %s " % item.iname)
 | |
|             #warn("INAMES: %s " % self.expandedINames)
 | |
|             #warn("EXPANDED: %s " % (item.iname in self.expandedINames))
 | |
| 
 | |
|             # insufficient, see http://sourceware.org/bugzilla/show_bug.cgi?id=10953
 | |
|             #fields = value.type.fields()
 | |
|             fields = value.type.strip_typedefs().fields()
 | |
| 
 | |
|             self.putType(item.value.type)
 | |
|             self.putValue("{...}")
 | |
| 
 | |
|             if False:
 | |
|                 numfields = 0
 | |
|                 for field in fields:
 | |
|                     bitpos = getattr(field, "bitpos", None)
 | |
|                     if not bitpos is None:
 | |
|                         ++numfields
 | |
|             else:
 | |
|                 numfields = len(fields)
 | |
|             self.putNumChild(numfields)
 | |
| 
 | |
|             if self.isExpanded(item):
 | |
|                 innerType = None
 | |
|                 if len(fields) == 1 and fields[0].name is None:
 | |
|                     innerType = value.type.target()
 | |
|                 self.beginChildren(1, innerType)
 | |
| 
 | |
|                 baseNumber = 0
 | |
|                 anonNumber = 0
 | |
|                 for field in fields:
 | |
|                     #warn("FIELD: %s" % field)
 | |
|                     #warn("  BITSIZE: %s" % field.bitsize)
 | |
|                     #warn("  ARTIFICIAL: %s" % field.artificial)
 | |
|                     bitpos = getattr(field, "bitpos", None)
 | |
|                     if bitpos is None: # FIXME: Is check correct?
 | |
|                         continue  # A static class member(?).
 | |
| 
 | |
|                     if field.name is None:
 | |
|                         innerType = value.type.target()
 | |
|                         p = value.cast(innerType.pointer())
 | |
|                         for i in xrange(value.type.sizeof / innerType.sizeof):
 | |
|                             self.putItem(Item(p.dereference(), item.iname, i, None))
 | |
|                             p = p + 1
 | |
|                         continue
 | |
| 
 | |
|                     # ignore vtable pointers for virtual inheritance
 | |
|                     if field.name.startswith("_vptr."):
 | |
|                         continue
 | |
| 
 | |
|                     #warn("FIELD NAME: %s" % field.name)
 | |
|                     #warn("FIELD TYPE: %s" % field.type)
 | |
|                     if field.name == stripClassTag(str(field.type)):
 | |
|                         # Field is base type. We cannot use field.name as part
 | |
|                         # of the iname as it might contain spaces and other
 | |
|                         # strange characters.
 | |
|                         child = Item(value.cast(field.type),
 | |
|                             item.iname, "@%d" % baseNumber, field.name)
 | |
|                         baseNumber += 1
 | |
|                         self.beginHash()
 | |
|                         self.putField("iname", child.iname)
 | |
|                         self.safePutItemHelper(child)
 | |
|                         self.endHash()
 | |
|                     elif field.name == "":
 | |
|                         # Anonymous union. We need a dummy name to distinguish
 | |
|                         # multiple anonymous unions in the struct.
 | |
|                         iname = "%s.#%d" % (item.iname, anonNumber)
 | |
|                         anonNumber += 1
 | |
|                         self.beginHash()
 | |
|                         self.putField("iname", iname)
 | |
|                         self.putField("name", "<n/a>")
 | |
|                         self.putField("value", " ")
 | |
|                         self.putField("type", "<anonymous union>")
 | |
|                         if self.isExpandedIName(iname):
 | |
|                             self.beginChildren()
 | |
|                             for f in field.type.fields():
 | |
|                                 child = Item(item.value[f.name],
 | |
|                                     item.iname, f.name, f.name)
 | |
|                                 self.safePutItem(child)
 | |
|                             self.endChildren()
 | |
|                         self.endHash()
 | |
|                     else:
 | |
|                         # Named field.
 | |
|                         self.beginHash()
 | |
|                         child = Item(value[field.name],
 | |
|                             item.iname, field.name, field.name)
 | |
|                         self.safePutItemHelper(child)
 | |
|                         self.endHash()
 | |
| 
 | |
|                 self.endChildren()
 |