forked from qt-creator/qt-creator
It is nicer in the presence of namespaces as '::' will be replaced by '__' to create the dumper function names.
806 lines
26 KiB
Python
806 lines
26 KiB
Python
|
|
#Note: Keep name-type-value-numchild-extra order
|
|
|
|
#return
|
|
|
|
import sys
|
|
import traceback
|
|
import gdb
|
|
import base64
|
|
import curses.ascii
|
|
|
|
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 value(expr):
|
|
value = gdb.parse_and_eval(expr)
|
|
try:
|
|
return int(value)
|
|
except:
|
|
return str(value)
|
|
|
|
def isSimpleType(typeobj):
|
|
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 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):
|
|
s = str(p)
|
|
return s == "0x0" or s.startswith("0x0 ")
|
|
|
|
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(0, 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 = gdb.parse_and_eval(exp)
|
|
#warn(" -> %s" % result)
|
|
return result
|
|
|
|
def qtNamespace():
|
|
try:
|
|
type = str(gdb.parse_and_eval("&QString::null").type.target().unqualified())
|
|
return type[0:len(type) - len("QString::null")]
|
|
except RuntimeError:
|
|
return ""
|
|
|
|
def encodeCharArray(p, size):
|
|
s = ""
|
|
for i in xrange(0, 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)
|
|
check(d_ptr["ref"]["_q_value"] > 0)
|
|
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
|
|
check(d_ptr["ref"]["_q_value"] > 0)
|
|
p = gdb.Value(d_ptr["data"])
|
|
s = ""
|
|
for i in xrange(0, 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):
|
|
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)
|
|
useFancy = int(args[0])
|
|
passExceptions = int(args[1])
|
|
expandedINames = set()
|
|
if len(args) > 2:
|
|
expandedINames = set(args[2].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.passExceptions = passExceptions
|
|
d.ns = qtNamespace()
|
|
d.expandedINames = expandedINames
|
|
d.useFancy = useFancy
|
|
#warn(" NAMESPACE IS: '%s'" % d.ns)
|
|
|
|
#
|
|
# Locals
|
|
#
|
|
try:
|
|
frame = gdb.selected_frame()
|
|
#warn("FRAME %s: " % frame)
|
|
except RuntimeError:
|
|
return ""
|
|
|
|
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)
|
|
|
|
type = item.value.type
|
|
if type.code == gdb.TYPE_CODE_PTR \
|
|
and 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(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(0, 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
|
|
d.beginHash()
|
|
d.put('iname="%s",' % item.iname)
|
|
d.put('addr="%s",' % item.value.address)
|
|
d.safePutItemHelper(item)
|
|
d.endHash()
|
|
|
|
# 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
|
|
#warn("BLOCK %s: " % block)
|
|
|
|
d.pushOutput()
|
|
locals = d.safeoutput
|
|
|
|
|
|
#
|
|
# Watchers
|
|
#
|
|
d.safeoutput = ""
|
|
watchers = ""
|
|
if len(args) > 3:
|
|
watchers = base64.b16decode(args[3], True)
|
|
if len(watchers) > 0:
|
|
for watcher in watchers.split("$$"):
|
|
(exp, name) = watcher.split("$")
|
|
self.handleWatch(d, exp, name)
|
|
d.pushOutput()
|
|
watchers = d.safeoutput
|
|
|
|
print('locals={iname="local",name="Locals",value=" ",type=" ",'
|
|
+ 'children=[' + locals + ']},'
|
|
+ 'watchers={iname="watch",name="Watchers",value=" ",type=" ",'
|
|
+ 'children=[' + watchers + ']}')
|
|
|
|
|
|
def handleWatch(self, d, exp, name):
|
|
#warn("HANDLING WATCH %s, NAME: %s" % (exp, name))
|
|
if exp.startswith("["):
|
|
#warn("EVAL: EXP: %s" % exp)
|
|
d.beginHash()
|
|
d.put('iname="watch.%s",' % name)
|
|
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" % (name, 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="watch.%s",' % name)
|
|
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 = gdb.parse_and_eval(exp)
|
|
item = Item(value, "watch", name, name)
|
|
d.safePutItemHelper(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]
|
|
|
|
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, type = None, children = None):
|
|
childType = ""
|
|
childNumChild = -1
|
|
if numChild == 0:
|
|
type = None
|
|
self.putCommaIfNeeded()
|
|
if not type is None:
|
|
childType = stripClassTag(str(type))
|
|
self.put('childtype="%s",' % childType)
|
|
if isSimpleType(type) or isStringType(self, type):
|
|
self.put('childnumchild="0",')
|
|
childNumChild = 0
|
|
elif type.code == gdb.TYPE_CODE_PTR:
|
|
self.put('childnumchild="1",')
|
|
childNumChild = 1
|
|
if not children is None:
|
|
self.put('childnumchild="%s",' % children)
|
|
childNumChild = children
|
|
self.childTypes.append(childType)
|
|
self.childNumChilds.append(childNumChild)
|
|
#warn("BEGIN: %s" % self.childTypes)
|
|
self.put("children=[")
|
|
|
|
def endChildren(self):
|
|
#warn("END: %s" % self.childTypes)
|
|
self.childTypes.pop()
|
|
self.childNumChilds.pop()
|
|
self.put(']')
|
|
|
|
# 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 putItemOrPointerHelper(self, item):
|
|
# if item.value.type.code == gdb.TYPE_CODE_PTR \
|
|
# and str(item.value.type.target()) != "char":
|
|
# if not isNull(item.value):
|
|
# self.putItemOrPointerHelper(
|
|
# Item(item.value.dereference(), item.iname, None, None))
|
|
# else:
|
|
# self.putValue("(null)")
|
|
# self.putNumChild(0)
|
|
# else:
|
|
# self.safePutItemHelper(item)
|
|
|
|
|
|
def putItemHelper(self, item):
|
|
name = getattr(item, "name", None)
|
|
if not name is None:
|
|
self.putName(name)
|
|
|
|
# FIXME: Gui shows references stripped?
|
|
#warn("REAL INAME: %s " % item.iname)
|
|
#warn("REAL TYPE: %s " % item.value.type)
|
|
#warn("REAL VALUE: %s " % item.value)
|
|
|
|
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):
|
|
#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)
|
|
|
|
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:
|
|
if self.useFancy:
|
|
#warn("A POINTER: %s" % value.type)
|
|
isHandled = False
|
|
if isNull(value):
|
|
self.putValue("0x0")
|
|
self.putType(item.value.type)
|
|
self.putNumChild(0)
|
|
isHandled = True
|
|
|
|
target = str(type.target().unqualified())
|
|
if target == "void" and not isHandled:
|
|
self.putType(item.value.type)
|
|
self.putValue(str(value))
|
|
self.putNumChild(0)
|
|
isHandled = True
|
|
|
|
if target == "char" and not isHandled:
|
|
# Display values up to given length directly
|
|
self.putType(item.value.type)
|
|
firstNul = -1
|
|
p = value
|
|
for i in xrange(0, 100):
|
|
if p.dereference() == 0:
|
|
# Found terminating NUL
|
|
self.putValue(encodeCharArray(value, i), "6")
|
|
self.putNumChild(0)
|
|
isHandled = True
|
|
break
|
|
p += 1
|
|
|
|
if not isHandled:
|
|
## Generic pointer type.
|
|
#warn("GENERIC POINTER: %s" % value)
|
|
innerType = item.value.type.target()
|
|
self.putType(innerType)
|
|
self.childTypes.append(
|
|
stripClassTag(str(innerType)))
|
|
#self.putType(item.value.type.target())
|
|
self.putItemHelper(
|
|
Item(item.value.dereference(), item.iname, None, None))
|
|
self.childTypes.pop()
|
|
else:
|
|
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("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
|
|
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(0, 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()
|
|
else:
|
|
# Data member.
|
|
child = Item(value[field.name],
|
|
item.iname, field.name, field.name)
|
|
if not child.name:
|
|
child.name = "<anon>"
|
|
self.beginHash()
|
|
self.safePutItemHelper(child)
|
|
self.endHash()
|
|
self.endChildren()
|