2013-04-11 18:11:54 +02:00
|
|
|
|
|
|
|
import binascii
|
2013-09-11 21:35:39 +02:00
|
|
|
try:
|
|
|
|
import __builtin__
|
|
|
|
except:
|
|
|
|
import builtins
|
|
|
|
try:
|
|
|
|
import gdb
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
2013-04-11 18:11:54 +02:00
|
|
|
import inspect
|
|
|
|
import os
|
2013-05-15 15:42:55 +02:00
|
|
|
import os.path
|
2013-04-12 16:58:25 +02:00
|
|
|
import sys
|
2013-05-15 15:42:55 +02:00
|
|
|
import tempfile
|
2013-04-11 18:11:54 +02:00
|
|
|
import traceback
|
2013-09-11 21:35:39 +02:00
|
|
|
import struct
|
|
|
|
|
2013-04-11 18:11:54 +02:00
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
def warn(message):
|
|
|
|
print("XXX: %s\n" % message.encode("latin1"))
|
2013-05-15 15:42:55 +02:00
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
from dumper import *
|
|
|
|
from qttypes import *
|
2013-10-10 16:40:56 +02:00
|
|
|
from stdtypes import *
|
|
|
|
from misctypes import *
|
|
|
|
from boosttypes import *
|
2013-10-16 16:50:26 +02:00
|
|
|
from creatortypes import *
|
2013-10-10 16:40:56 +02:00
|
|
|
|
2013-04-10 13:55:15 +02:00
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# Infrastructure
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
def savePrint(output):
|
|
|
|
try:
|
|
|
|
print(output)
|
|
|
|
except:
|
|
|
|
out = ""
|
|
|
|
for c in output:
|
|
|
|
cc = ord(c)
|
|
|
|
if cc > 127:
|
|
|
|
out += "\\\\%d" % cc
|
|
|
|
elif cc < 0:
|
|
|
|
out += "\\\\%d" % (cc + 256)
|
|
|
|
else:
|
|
|
|
out += c
|
|
|
|
print(out)
|
|
|
|
|
|
|
|
def registerCommand(name, func):
|
|
|
|
|
|
|
|
class Command(gdb.Command):
|
|
|
|
def __init__(self):
|
|
|
|
super(Command, self).__init__(name, gdb.COMMAND_OBSCURE)
|
|
|
|
def invoke(self, args, from_tty):
|
|
|
|
savePrint(func(args))
|
|
|
|
|
|
|
|
Command()
|
|
|
|
|
|
|
|
def parseAndEvaluate(exp):
|
|
|
|
return gdb.parse_and_eval(exp)
|
|
|
|
|
|
|
|
def listOfLocals(varList):
|
|
|
|
frame = gdb.selected_frame()
|
|
|
|
try:
|
|
|
|
frame = gdb.selected_frame()
|
|
|
|
#warn("FRAME %s: " % frame)
|
2013-09-11 21:35:39 +02:00
|
|
|
except RuntimeError as error:
|
2013-04-10 13:55:15 +02:00
|
|
|
warn("FRAME NOT ACCESSIBLE: %s" % error)
|
|
|
|
return []
|
|
|
|
except:
|
|
|
|
warn("FRAME NOT ACCESSIBLE FOR UNKNOWN REASONS")
|
|
|
|
return []
|
|
|
|
|
|
|
|
try:
|
|
|
|
block = frame.block()
|
|
|
|
#warn("BLOCK: %s " % block)
|
2013-09-11 21:35:39 +02:00
|
|
|
except RuntimeError as error:
|
2013-06-06 18:28:53 +02:00
|
|
|
#warn("BLOCK IN FRAME NOT ACCESSIBLE: %s" % error)
|
2013-04-11 17:43:50 +02:00
|
|
|
return []
|
2013-04-10 13:55:15 +02:00
|
|
|
except:
|
|
|
|
warn("BLOCK NOT ACCESSIBLE FOR UNKNOWN REASONS")
|
2013-04-11 17:43:50 +02:00
|
|
|
return []
|
2013-04-10 13:55:15 +02:00
|
|
|
|
|
|
|
items = []
|
|
|
|
shadowed = {}
|
|
|
|
while True:
|
|
|
|
if block is None:
|
|
|
|
warn("UNEXPECTED 'None' BLOCK")
|
|
|
|
break
|
|
|
|
for symbol in block:
|
|
|
|
name = symbol.print_name
|
|
|
|
|
2013-05-31 21:53:43 +02:00
|
|
|
if name == "__in_chrg" or name == "__PRETTY_FUNCTION__":
|
2013-04-10 13:55:15 +02:00
|
|
|
continue
|
|
|
|
|
|
|
|
# "NotImplementedError: Symbol type not yet supported in
|
|
|
|
# Python scripts."
|
|
|
|
#warn("SYMBOL %s (%s): " % (symbol, name))
|
|
|
|
if name in shadowed:
|
|
|
|
level = shadowed[name]
|
|
|
|
name1 = "%s@%s" % (name, level)
|
|
|
|
shadowed[name] = level + 1
|
|
|
|
else:
|
|
|
|
name1 = name
|
|
|
|
shadowed[name] = 1
|
|
|
|
#warn("SYMBOL %s (%s, %s)): " % (symbol, name, symbol.name))
|
|
|
|
item = LocalItem()
|
|
|
|
item.iname = "local." + name1
|
|
|
|
item.name = name1
|
|
|
|
try:
|
|
|
|
item.value = frame.read_var(name, block)
|
|
|
|
#warn("READ 1: %s" % item.value)
|
|
|
|
if not item.value.is_optimized_out:
|
|
|
|
#warn("ITEM 1: %s" % item.value)
|
|
|
|
items.append(item)
|
|
|
|
continue
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
try:
|
|
|
|
item.value = frame.read_var(name)
|
|
|
|
#warn("READ 2: %s" % item.value)
|
|
|
|
if not item.value.is_optimized_out:
|
|
|
|
#warn("ITEM 2: %s" % item.value)
|
|
|
|
items.append(item)
|
|
|
|
continue
|
|
|
|
except:
|
|
|
|
# RuntimeError: happens for
|
|
|
|
# void foo() { std::string s; std::wstring w; }
|
|
|
|
# ValueError: happens for (as of 2010/11/4)
|
|
|
|
# a local struct as found e.g. in
|
|
|
|
# gcc sources in gcc.c, int execute()
|
|
|
|
pass
|
|
|
|
|
|
|
|
try:
|
|
|
|
#warn("READ 3: %s %s" % (name, item.value))
|
|
|
|
item.value = gdb.parse_and_eval(name)
|
|
|
|
#warn("ITEM 3: %s" % item.value)
|
|
|
|
items.append(item)
|
|
|
|
except:
|
|
|
|
# Can happen in inlined code (see last line of
|
|
|
|
# RowPainter::paintChars(): "RuntimeError:
|
|
|
|
# No symbol \"__val\" in current context.\n"
|
|
|
|
pass
|
|
|
|
|
|
|
|
# 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
|
|
|
|
|
|
|
|
return items
|
|
|
|
|
|
|
|
|
|
|
|
def catchCliOutput(command):
|
|
|
|
try:
|
|
|
|
return gdb.execute(command, to_string=True).split("\n")
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
filename = createTempFile()
|
|
|
|
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")
|
|
|
|
msg = ""
|
|
|
|
try:
|
|
|
|
gdb.execute(command)
|
2013-09-11 21:35:39 +02:00
|
|
|
except RuntimeError as error:
|
2013-04-10 13:55:15 +02:00
|
|
|
# For the first phase of core file loading this yield
|
|
|
|
# "No symbol table is loaded. Use the \"file\" command."
|
|
|
|
msg = str(error)
|
|
|
|
except:
|
|
|
|
msg = "Unknown error"
|
|
|
|
gdb.execute("set logging off")
|
|
|
|
# gdb.execute("set logging redirect off")
|
|
|
|
if len(msg):
|
|
|
|
# Having that might confuse result handlers in the gdbengine.
|
|
|
|
#warn("CLI ERROR: %s " % msg)
|
|
|
|
removeTempFile(filename)
|
|
|
|
return "CLI ERROR: %s " % msg
|
|
|
|
temp = open(filename, "r")
|
|
|
|
lines = []
|
|
|
|
for line in temp:
|
|
|
|
lines.append(line)
|
|
|
|
temp.close()
|
|
|
|
removeTempFile(filename)
|
|
|
|
return lines
|
|
|
|
|
|
|
|
def selectedInferior():
|
|
|
|
try:
|
|
|
|
# Does not exist in 7.3.
|
|
|
|
return gdb.selected_inferior()
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
# gdb.Inferior is new in gdb 7.2
|
|
|
|
return gdb.inferiors()[0]
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# Types
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
PointerCode = gdb.TYPE_CODE_PTR
|
|
|
|
ArrayCode = gdb.TYPE_CODE_ARRAY
|
|
|
|
StructCode = gdb.TYPE_CODE_STRUCT
|
|
|
|
UnionCode = gdb.TYPE_CODE_UNION
|
|
|
|
EnumCode = gdb.TYPE_CODE_ENUM
|
|
|
|
FlagsCode = gdb.TYPE_CODE_FLAGS
|
|
|
|
FunctionCode = gdb.TYPE_CODE_FUNC
|
|
|
|
IntCode = gdb.TYPE_CODE_INT
|
|
|
|
FloatCode = gdb.TYPE_CODE_FLT # Parts of GDB assume that this means complex.
|
|
|
|
VoidCode = gdb.TYPE_CODE_VOID
|
|
|
|
#SetCode = gdb.TYPE_CODE_SET
|
|
|
|
RangeCode = gdb.TYPE_CODE_RANGE
|
|
|
|
StringCode = gdb.TYPE_CODE_STRING
|
|
|
|
#BitStringCode = gdb.TYPE_CODE_BITSTRING
|
|
|
|
#ErrorTypeCode = gdb.TYPE_CODE_ERROR
|
|
|
|
MethodCode = gdb.TYPE_CODE_METHOD
|
|
|
|
MethodPointerCode = gdb.TYPE_CODE_METHODPTR
|
|
|
|
MemberPointerCode = gdb.TYPE_CODE_MEMBERPTR
|
|
|
|
ReferenceCode = gdb.TYPE_CODE_REF
|
|
|
|
CharCode = gdb.TYPE_CODE_CHAR
|
|
|
|
BoolCode = gdb.TYPE_CODE_BOOL
|
|
|
|
ComplexCode = gdb.TYPE_CODE_COMPLEX
|
|
|
|
TypedefCode = gdb.TYPE_CODE_TYPEDEF
|
|
|
|
NamespaceCode = gdb.TYPE_CODE_NAMESPACE
|
|
|
|
#Code = gdb.TYPE_CODE_DECFLOAT # Decimal floating point.
|
|
|
|
#Code = gdb.TYPE_CODE_MODULE # Fortran
|
|
|
|
#Code = gdb.TYPE_CODE_INTERNAL_FUNCTION
|
|
|
|
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# Step Command
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
def sal(args):
|
|
|
|
(cmd, addr) = args.split(",")
|
|
|
|
lines = catchCliOutput("info line *" + addr)
|
|
|
|
fromAddr = "0x0"
|
|
|
|
toAddr = "0x0"
|
|
|
|
for line in lines:
|
|
|
|
pos0from = line.find(" starts at address") + 19
|
|
|
|
pos1from = line.find(" ", pos0from)
|
|
|
|
pos0to = line.find(" ends at", pos1from) + 9
|
|
|
|
pos1to = line.find(" ", pos0to)
|
|
|
|
if pos1to > 0:
|
|
|
|
fromAddr = line[pos0from : pos1from]
|
|
|
|
toAddr = line[pos0to : pos1to]
|
|
|
|
gdb.execute("maint packet sal%s,%s,%s" % (cmd,fromAddr, toAddr))
|
|
|
|
|
|
|
|
registerCommand("sal", sal)
|
|
|
|
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# Convenience
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
# Just convienience for 'python print ...'
|
|
|
|
class PPCommand(gdb.Command):
|
|
|
|
def __init__(self):
|
|
|
|
super(PPCommand, self).__init__("pp", gdb.COMMAND_OBSCURE)
|
|
|
|
def invoke(self, args, from_tty):
|
|
|
|
print(eval(args))
|
|
|
|
|
|
|
|
PPCommand()
|
|
|
|
|
|
|
|
# Just convienience for 'python print gdb.parse_and_eval(...)'
|
|
|
|
class PPPCommand(gdb.Command):
|
|
|
|
def __init__(self):
|
|
|
|
super(PPPCommand, self).__init__("ppp", gdb.COMMAND_OBSCURE)
|
|
|
|
def invoke(self, args, from_tty):
|
|
|
|
print(gdb.parse_and_eval(args))
|
|
|
|
|
|
|
|
PPPCommand()
|
|
|
|
|
|
|
|
|
|
|
|
def scanStack(p, n):
|
2013-09-11 21:35:39 +02:00
|
|
|
p = int(p)
|
2013-04-10 13:55:15 +02:00
|
|
|
r = []
|
|
|
|
for i in xrange(n):
|
|
|
|
f = gdb.parse_and_eval("{void*}%s" % p)
|
|
|
|
m = gdb.execute("info symbol %s" % f, to_string=True)
|
|
|
|
if not m.startswith("No symbol matches"):
|
|
|
|
r.append(m)
|
|
|
|
p += f.type.sizeof
|
|
|
|
return r
|
|
|
|
|
|
|
|
class ScanStackCommand(gdb.Command):
|
|
|
|
def __init__(self):
|
|
|
|
super(ScanStackCommand, self).__init__("scanStack", gdb.COMMAND_OBSCURE)
|
|
|
|
def invoke(self, args, from_tty):
|
|
|
|
if len(args) == 0:
|
|
|
|
args = 20
|
|
|
|
savePrint(scanStack(gdb.parse_and_eval("$sp"), int(args)))
|
|
|
|
|
|
|
|
ScanStackCommand()
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
def registerDumper(funcname, function):
|
2013-05-13 16:39:51 +02:00
|
|
|
global qqDumpers, qqFormats, qqEditable
|
|
|
|
try:
|
2013-09-11 21:35:39 +02:00
|
|
|
#warn("FUNCTION: %s " % funcname)
|
|
|
|
#funcname = function.func_name
|
2013-05-13 16:39:51 +02:00
|
|
|
if funcname.startswith("qdump__"):
|
|
|
|
type = funcname[7:]
|
|
|
|
qqDumpers[type] = function
|
|
|
|
qqFormats[type] = qqFormats.get(type, "")
|
|
|
|
elif funcname.startswith("qform__"):
|
|
|
|
type = funcname[7:]
|
|
|
|
formats = ""
|
|
|
|
try:
|
|
|
|
formats = function()
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
qqFormats[type] = formats
|
|
|
|
elif funcname.startswith("qedit__"):
|
|
|
|
type = funcname[7:]
|
|
|
|
try:
|
|
|
|
qqEditable[type] = function
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
except:
|
|
|
|
pass
|
2013-04-11 18:11:54 +02:00
|
|
|
|
2013-04-12 16:58:25 +02:00
|
|
|
def bbsetup(args = ''):
|
|
|
|
global qqDumpers, qqFormats, qqEditable, typeCache
|
2013-06-03 14:27:53 +02:00
|
|
|
qqDumpers = {}
|
|
|
|
qqFormats = {}
|
|
|
|
qqEditable = {}
|
2013-04-12 16:58:25 +02:00
|
|
|
typeCache = {}
|
|
|
|
module = sys.modules[__name__]
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
#warn("KEYS: %s " % module.__dict__.keys())
|
|
|
|
for name in module.__dict__.keys():
|
|
|
|
#warn("KEY: %s " % name)
|
|
|
|
#warn("FUNCT: %s " % module.__dict__[name])
|
|
|
|
registerDumper(name, module.__dict__[name])
|
2013-04-12 16:58:25 +02:00
|
|
|
|
|
|
|
result = "dumpers=["
|
|
|
|
#qqNs = qtNamespace() # This is too early
|
|
|
|
for key, value in qqFormats.items():
|
2013-09-11 21:35:39 +02:00
|
|
|
if key in qqEditable:
|
2013-04-12 16:58:25 +02:00
|
|
|
result += '{type="%s",formats="%s",editable="true"},' % (key, value)
|
|
|
|
else:
|
|
|
|
result += '{type="%s",formats="%s"},' % (key, value)
|
|
|
|
result += ']'
|
|
|
|
#result += ',namespace="%s"' % qqNs
|
2013-09-11 21:35:39 +02:00
|
|
|
print(result)
|
2013-04-12 16:58:25 +02:00
|
|
|
return result
|
|
|
|
|
|
|
|
registerCommand("bbsetup", bbsetup)
|
|
|
|
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# Import plain gdb pretty printers
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
class PlainDumper:
|
|
|
|
def __init__(self, printer):
|
|
|
|
self.printer = printer
|
|
|
|
|
|
|
|
def __call__(self, d, value):
|
|
|
|
printer = self.printer.gen_printer(value)
|
|
|
|
lister = getattr(printer, "children", None)
|
|
|
|
children = [] if lister is None else list(lister())
|
|
|
|
d.putType(self.printer.name)
|
|
|
|
val = printer.to_string().encode("hex")
|
|
|
|
d.putValue(val, Hex2EncodedLatin1)
|
|
|
|
d.putValue(printer.to_string())
|
|
|
|
d.putNumChild(len(children))
|
|
|
|
if d.isExpanded():
|
|
|
|
with Children(d):
|
|
|
|
for child in children:
|
|
|
|
d.putSubItem(child[0], child[1])
|
|
|
|
|
|
|
|
def importPlainDumper(printer):
|
|
|
|
global qqDumpers, qqFormats
|
|
|
|
name = printer.name.replace("::", "__")
|
|
|
|
qqDumpers[name] = PlainDumper(printer)
|
|
|
|
qqFormats[name] = ""
|
|
|
|
|
|
|
|
def importPlainDumpers(args):
|
|
|
|
return
|
|
|
|
for obj in gdb.objfiles():
|
|
|
|
for printers in obj.pretty_printers + gdb.pretty_printers:
|
|
|
|
for printer in printers.subprinters:
|
|
|
|
importPlainDumper(printer)
|
|
|
|
|
|
|
|
registerCommand("importPlainDumpers", importPlainDumpers)
|
|
|
|
|
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
#gdb.Value.child = impl_Value_child
|
|
|
|
|
|
|
|
# Fails on SimulatorQt.
|
|
|
|
tempFileCounter = 0
|
|
|
|
try:
|
|
|
|
# Test if 2.6 is used (Windows), trigger exception and default
|
|
|
|
# to 2nd version.
|
|
|
|
file = tempfile.NamedTemporaryFile(prefix="py_",delete=True)
|
|
|
|
file.close()
|
|
|
|
def createTempFile():
|
|
|
|
file = tempfile.NamedTemporaryFile(prefix="py_",delete=True)
|
|
|
|
file.close()
|
|
|
|
return file.name
|
|
|
|
|
|
|
|
except:
|
|
|
|
def createTempFile():
|
|
|
|
global tempFileCounter
|
|
|
|
tempFileCounter += 1
|
|
|
|
fileName = "%s/py_tmp_%d_%d" \
|
|
|
|
% (tempfile.gettempdir(), os.getpid(), tempFileCounter)
|
|
|
|
return fileName
|
|
|
|
|
|
|
|
def removeTempFile(name):
|
|
|
|
try:
|
|
|
|
os.remove(name)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# VTable
|
|
|
|
#
|
|
|
|
def hasVTable(type):
|
|
|
|
fields = type.fields()
|
|
|
|
if len(fields) == 0:
|
|
|
|
return False
|
|
|
|
if fields[0].is_base_class:
|
|
|
|
return hasVTable(fields[0].type)
|
|
|
|
return str(fields[0].type) == "int (**)(void)"
|
|
|
|
|
|
|
|
def dynamicTypeName(value):
|
|
|
|
if hasVTable(value.type):
|
2013-09-11 21:35:39 +02:00
|
|
|
#vtbl = str(parseAndEvaluate("{int(*)(int)}%s" % int(value.address)))
|
2013-05-15 15:42:55 +02:00
|
|
|
try:
|
|
|
|
# Fails on 7.1 due to the missing to_string.
|
2013-09-11 21:35:39 +02:00
|
|
|
vtbl = gdb.execute("info symbol {int*}%s" % int(value.address),
|
2013-05-15 15:42:55 +02:00
|
|
|
to_string = True)
|
|
|
|
pos1 = vtbl.find("vtable ")
|
|
|
|
if pos1 != -1:
|
|
|
|
pos1 += 11
|
|
|
|
pos2 = vtbl.find(" +", pos1)
|
|
|
|
if pos2 != -1:
|
|
|
|
return vtbl[pos1 : pos2]
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
return str(value.type)
|
|
|
|
|
|
|
|
def downcast(value):
|
|
|
|
try:
|
|
|
|
return value.cast(value.dynamic_type)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
#try:
|
|
|
|
# return value.cast(lookupType(dynamicTypeName(value)))
|
|
|
|
#except:
|
|
|
|
# pass
|
|
|
|
return value
|
|
|
|
|
|
|
|
def expensiveDowncast(value):
|
|
|
|
try:
|
|
|
|
return value.cast(value.dynamic_type)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
return value.cast(lookupType(dynamicTypeName(value)))
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
return value
|
|
|
|
|
|
|
|
typeCache = {}
|
|
|
|
|
2013-08-12 19:34:48 +02:00
|
|
|
typesToReport = {}
|
|
|
|
typesReported = {}
|
|
|
|
|
|
|
|
def addToCache(type):
|
|
|
|
global typesReported
|
|
|
|
global typesToReport
|
|
|
|
typename = str(type)
|
|
|
|
if typename in typesReported:
|
|
|
|
return
|
|
|
|
typesReported[typename] = True
|
|
|
|
typesToReport[typename] = type
|
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
def lookupType(typestring):
|
|
|
|
global typeCache
|
|
|
|
global typesToReport
|
|
|
|
type = typeCache.get(typestring)
|
|
|
|
#warn("LOOKUP 1: %s -> %s" % (typestring, type))
|
|
|
|
if not type is None:
|
|
|
|
return type
|
|
|
|
|
|
|
|
if typestring == "void":
|
|
|
|
type = gdb.lookup_type(typestring)
|
|
|
|
typeCache[typestring] = type
|
|
|
|
typesToReport[typestring] = type
|
|
|
|
return type
|
|
|
|
|
2013-09-13 17:57:36 +02:00
|
|
|
#try:
|
|
|
|
# type = gdb.parse_and_eval("{%s}&main" % typestring).type
|
|
|
|
# if not type is None:
|
|
|
|
# typeCache[typestring] = type
|
|
|
|
# typesToReport[typestring] = type
|
|
|
|
# return type
|
|
|
|
#except:
|
|
|
|
# pass
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
# See http://sourceware.org/bugzilla/show_bug.cgi?id=13269
|
|
|
|
# gcc produces "{anonymous}", gdb "(anonymous namespace)"
|
|
|
|
# "<unnamed>" has been seen too. The only thing gdb
|
|
|
|
# understands when reading things back is "(anonymous namespace)"
|
|
|
|
if typestring.find("{anonymous}") != -1:
|
|
|
|
ts = typestring
|
|
|
|
ts = ts.replace("{anonymous}", "(anonymous namespace)")
|
|
|
|
type = lookupType(ts)
|
|
|
|
if not type is None:
|
|
|
|
typeCache[typestring] = type
|
|
|
|
typesToReport[typestring] = type
|
|
|
|
return type
|
|
|
|
|
|
|
|
#warn(" RESULT FOR 7.2: '%s': %s" % (typestring, type))
|
|
|
|
|
|
|
|
# This part should only trigger for
|
|
|
|
# gdb 7.1 for types with namespace separators.
|
|
|
|
# And anonymous namespaces.
|
|
|
|
|
|
|
|
ts = typestring
|
|
|
|
while True:
|
|
|
|
#warn("TS: '%s'" % ts)
|
|
|
|
if ts.startswith("class "):
|
|
|
|
ts = ts[6:]
|
|
|
|
elif ts.startswith("struct "):
|
|
|
|
ts = ts[7:]
|
|
|
|
elif ts.startswith("const "):
|
|
|
|
ts = ts[6:]
|
|
|
|
elif ts.startswith("volatile "):
|
|
|
|
ts = ts[9:]
|
|
|
|
elif ts.startswith("enum "):
|
|
|
|
ts = ts[5:]
|
|
|
|
elif ts.endswith(" const"):
|
|
|
|
ts = ts[:-6]
|
|
|
|
elif ts.endswith(" volatile"):
|
|
|
|
ts = ts[:-9]
|
|
|
|
elif ts.endswith("*const"):
|
|
|
|
ts = ts[:-5]
|
|
|
|
elif ts.endswith("*volatile"):
|
|
|
|
ts = ts[:-8]
|
|
|
|
else:
|
|
|
|
break
|
|
|
|
|
|
|
|
if ts.endswith('*'):
|
|
|
|
type = lookupType(ts[0:-1])
|
|
|
|
if not type is None:
|
|
|
|
type = type.pointer()
|
|
|
|
typeCache[typestring] = type
|
|
|
|
typesToReport[typestring] = type
|
|
|
|
return type
|
|
|
|
|
|
|
|
try:
|
|
|
|
#warn("LOOKING UP '%s'" % ts)
|
|
|
|
type = gdb.lookup_type(ts)
|
2013-09-11 21:35:39 +02:00
|
|
|
except RuntimeError as error:
|
2013-05-15 15:42:55 +02:00
|
|
|
#warn("LOOKING UP '%s': %s" % (ts, error))
|
|
|
|
# See http://sourceware.org/bugzilla/show_bug.cgi?id=11912
|
|
|
|
exp = "(class '%s'*)0" % ts
|
|
|
|
try:
|
|
|
|
type = parseAndEvaluate(exp).type.target()
|
|
|
|
except:
|
|
|
|
# Can throw "RuntimeError: No type named class Foo."
|
|
|
|
pass
|
|
|
|
except:
|
|
|
|
#warn("LOOKING UP '%s' FAILED" % ts)
|
|
|
|
pass
|
|
|
|
|
|
|
|
if not type is None:
|
|
|
|
typeCache[typestring] = type
|
|
|
|
typesToReport[typestring] = type
|
|
|
|
return type
|
|
|
|
|
|
|
|
# This could still be None as gdb.lookup_type("char[3]") generates
|
|
|
|
# "RuntimeError: No type named char[3]"
|
|
|
|
typeCache[typestring] = type
|
|
|
|
typesToReport[typestring] = type
|
|
|
|
return type
|
|
|
|
|
|
|
|
def extractTemplateArgument(type, position):
|
|
|
|
level = 0
|
|
|
|
skipSpace = False
|
|
|
|
inner = ""
|
|
|
|
type = str(type)
|
|
|
|
for c in type[type.find('<') + 1 : -1]:
|
|
|
|
if c == '<':
|
|
|
|
inner += c
|
|
|
|
level += 1
|
|
|
|
elif c == '>':
|
|
|
|
level -= 1
|
|
|
|
inner += c
|
|
|
|
elif c == ',':
|
|
|
|
if level == 0:
|
|
|
|
if position == 0:
|
|
|
|
return inner
|
|
|
|
position -= 1
|
|
|
|
inner = ""
|
|
|
|
else:
|
|
|
|
inner += c
|
|
|
|
skipSpace = True
|
|
|
|
else:
|
|
|
|
if skipSpace and c == ' ':
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
inner += c
|
|
|
|
skipSpace = False
|
|
|
|
return inner
|
|
|
|
|
|
|
|
def templateArgument(type, position):
|
|
|
|
try:
|
|
|
|
# This fails on stock 7.2 with
|
|
|
|
# "RuntimeError: No type named myns::QObject.\n"
|
|
|
|
return type.template_argument(position)
|
|
|
|
except:
|
|
|
|
# That's something like "myns::QList<...>"
|
|
|
|
return lookupType(extractTemplateArgument(type.strip_typedefs(), position))
|
|
|
|
|
|
|
|
|
|
|
|
# Workaround for gdb < 7.1
|
|
|
|
def numericTemplateArgument(type, position):
|
|
|
|
try:
|
|
|
|
return int(type.template_argument(position))
|
2013-09-11 21:35:39 +02:00
|
|
|
except RuntimeError as error:
|
2013-05-15 15:42:55 +02:00
|
|
|
# ": No type named 30."
|
|
|
|
msg = str(error)
|
|
|
|
msg = msg[14:-1]
|
|
|
|
# gdb at least until 7.4 produces for std::array<int, 4u>
|
|
|
|
# for template_argument(1): RuntimeError: No type named 4u.
|
|
|
|
if msg[-1] == 'u':
|
|
|
|
msg = msg[0:-1]
|
|
|
|
return int(msg)
|
|
|
|
|
|
|
|
|
|
|
|
class OutputSafer:
|
|
|
|
def __init__(self, d):
|
|
|
|
self.d = d
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
self.savedOutput = self.d.output
|
|
|
|
self.d.output = []
|
|
|
|
|
|
|
|
def __exit__(self, exType, exValue, exTraceBack):
|
|
|
|
if self.d.passExceptions and not exType is None:
|
|
|
|
showException("OUTPUTSAFER", exType, exValue, exTraceBack)
|
|
|
|
self.d.output = self.savedOutput
|
|
|
|
else:
|
|
|
|
self.savedOutput.extend(self.d.output)
|
|
|
|
self.d.output = self.savedOutput
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def value(expr):
|
|
|
|
value = parseAndEvaluate(expr)
|
|
|
|
try:
|
|
|
|
return int(value)
|
|
|
|
except:
|
|
|
|
return str(value)
|
|
|
|
|
|
|
|
|
|
|
|
#def couldBePointer(p, align):
|
|
|
|
# type = lookupType("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)
|
|
|
|
|
|
|
|
|
|
|
|
Value = gdb.Value
|
|
|
|
|
|
|
|
|
|
|
|
qqNs = None
|
|
|
|
|
|
|
|
def qtNamespace():
|
|
|
|
# FIXME: This only works when call from inside a Qt function frame.
|
|
|
|
global qqNs
|
|
|
|
if not qqNs is None:
|
|
|
|
return qqNs
|
|
|
|
try:
|
2013-06-27 10:42:32 +02:00
|
|
|
str = gdb.execute("ptype QString::Null", to_string=True)
|
2013-05-15 15:42:55 +02:00
|
|
|
# The result looks like:
|
|
|
|
# "type = const struct myns::QString::Null {"
|
|
|
|
# " <no data fields>"
|
|
|
|
# "}"
|
|
|
|
pos1 = str.find("struct") + 7
|
|
|
|
pos2 = str.find("QString::Null")
|
|
|
|
if pos1 > -1 and pos2 > -1:
|
|
|
|
qqNs = str[pos1:pos2]
|
|
|
|
return qqNs
|
|
|
|
return ""
|
|
|
|
except:
|
|
|
|
return ""
|
|
|
|
|
|
|
|
def stripTypedefs(type):
|
|
|
|
type = type.unqualified()
|
|
|
|
while type.code == TypedefCode:
|
|
|
|
type = type.strip_typedefs().unqualified()
|
|
|
|
return type
|
|
|
|
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# LocalItem
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
# Contains iname, name, and value.
|
|
|
|
class LocalItem:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# Edit Command
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
def bbedit(args):
|
|
|
|
global qqEditable
|
|
|
|
(type, expr, value) = args.split(",")
|
2013-09-11 21:35:39 +02:00
|
|
|
type = b16decode(type)
|
2013-05-15 15:42:55 +02:00
|
|
|
ns = qtNamespace()
|
|
|
|
if type.startswith(ns):
|
|
|
|
type = type[len(ns):]
|
|
|
|
type = type.replace("::", "__")
|
|
|
|
pos = type.find('<')
|
|
|
|
if pos != -1:
|
|
|
|
type = type[0:pos]
|
2013-09-11 21:35:39 +02:00
|
|
|
expr = b16decode(expr)
|
|
|
|
value = b16decode(value)
|
2013-05-15 15:42:55 +02:00
|
|
|
#warn("EDIT: %s %s %s %s: " % (pos, type, expr, value))
|
|
|
|
if qqEditable.has_key(type):
|
|
|
|
qqEditable[type](expr, value)
|
|
|
|
else:
|
|
|
|
gdb.execute("set (%s)=%s" % (expr, value))
|
|
|
|
|
|
|
|
registerCommand("bbedit", bbedit)
|
|
|
|
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# Frame Command
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
def bb(args):
|
|
|
|
global typesToReport
|
|
|
|
output = Dumper(args).output
|
|
|
|
output.append('],typeinfo=[')
|
2013-09-11 21:35:39 +02:00
|
|
|
for name in typesToReport.keys():
|
|
|
|
type = typesToReport[name]
|
2013-05-15 15:42:55 +02:00
|
|
|
# Happens e.g. for '(anonymous namespace)::InsertDefOperation'
|
|
|
|
if not type is None:
|
|
|
|
output.append('{name="%s",size="%s"}'
|
2013-09-11 21:35:39 +02:00
|
|
|
% (b64encode(name), type.sizeof))
|
2013-05-15 15:42:55 +02:00
|
|
|
output.append(']')
|
|
|
|
typesToReport = {}
|
|
|
|
return "".join(output)
|
|
|
|
|
|
|
|
|
|
|
|
def p1(args):
|
|
|
|
import cProfile
|
|
|
|
cProfile.run('bb("%s")' % args, "/tmp/bbprof")
|
|
|
|
import pstats
|
|
|
|
pstats.Stats('/tmp/bbprof').sort_stats('time').print_stats()
|
|
|
|
return ""
|
|
|
|
|
|
|
|
|
|
|
|
def p2(args):
|
|
|
|
import timeit
|
|
|
|
return timeit.repeat('bb("%s")' % args,
|
|
|
|
'from __main__ import bb', number=10)
|
|
|
|
|
|
|
|
registerCommand("bb", bb)
|
|
|
|
registerCommand("p1", p1)
|
|
|
|
registerCommand("p2", p2)
|
|
|
|
|
2013-10-22 13:21:56 +02:00
|
|
|
def extractQtVersion():
|
|
|
|
version = str(gdb.parse_and_eval("qVersion()"))
|
|
|
|
(major, minor, patch) = version[version.find('"')+1:version.rfind('"')].split('.')
|
|
|
|
return 0x10000 * int(major) + 0x100 * int(minor) + int(patch)
|
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# The Dumper Class
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
class Dumper(DumperBase):
|
|
|
|
|
|
|
|
def __init__(self, args):
|
|
|
|
DumperBase.__init__(self)
|
|
|
|
|
|
|
|
self.isGdb = True
|
2013-05-15 15:42:55 +02:00
|
|
|
self.output = []
|
|
|
|
self.currentIName = ""
|
|
|
|
self.currentPrintsAddress = True
|
|
|
|
self.currentChildType = ""
|
|
|
|
self.currentChildNumChild = -1
|
|
|
|
self.currentMaxNumChild = -1
|
|
|
|
self.currentNumChild = -1
|
|
|
|
self.currentValue = None
|
|
|
|
self.currentValuePriority = -100
|
|
|
|
self.currentValueEncoding = None
|
|
|
|
self.currentType = None
|
|
|
|
self.currentTypePriority = -100
|
|
|
|
self.currentAddress = None
|
|
|
|
self.typeformats = {}
|
|
|
|
self.formats = {}
|
|
|
|
self.useDynamicType = True
|
|
|
|
self.expandedINames = {}
|
2013-07-01 14:29:00 +02:00
|
|
|
self.childEventAddress = None
|
2013-10-22 13:21:56 +02:00
|
|
|
self.cachedQtVersion = None
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
watchers = ""
|
|
|
|
resultVarName = ""
|
|
|
|
options = []
|
|
|
|
varList = []
|
|
|
|
|
|
|
|
self.output.append('data=[')
|
|
|
|
for arg in args.split(' '):
|
|
|
|
pos = arg.find(":") + 1
|
|
|
|
if arg.startswith("options:"):
|
|
|
|
options = arg[pos:].split(",")
|
|
|
|
elif arg.startswith("vars:"):
|
|
|
|
if len(arg[pos:]) > 0:
|
|
|
|
varList = arg[pos:].split(",")
|
|
|
|
elif arg.startswith("resultvarname:"):
|
|
|
|
resultVarName = arg[pos:]
|
|
|
|
elif arg.startswith("expanded:"):
|
|
|
|
self.expandedINames = set(arg[pos:].split(","))
|
|
|
|
elif arg.startswith("typeformats:"):
|
|
|
|
for f in arg[pos:].split(","):
|
|
|
|
pos = f.find("=")
|
|
|
|
if pos != -1:
|
2013-09-11 21:35:39 +02:00
|
|
|
type = b16decode(f[0:pos])
|
2013-05-15 15:42:55 +02:00
|
|
|
self.typeformats[type] = int(f[pos+1:])
|
|
|
|
elif arg.startswith("formats:"):
|
|
|
|
for f in arg[pos:].split(","):
|
|
|
|
pos = f.find("=")
|
|
|
|
if pos != -1:
|
|
|
|
self.formats[f[0:pos]] = int(f[pos+1:])
|
|
|
|
elif arg.startswith("watchers:"):
|
2013-09-11 21:35:39 +02:00
|
|
|
watchers = b16decode(arg[pos:])
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
self.useDynamicType = "dyntype" in options
|
|
|
|
self.useFancy = "fancy" in options
|
|
|
|
self.passExceptions = "pe" in options
|
|
|
|
#self.passExceptions = True
|
|
|
|
self.autoDerefPointers = "autoderef" in options
|
|
|
|
self.partialUpdate = "partial" in options
|
|
|
|
self.tooltipOnly = "tooltiponly" in options
|
|
|
|
self.noLocals = "nolocals" in options
|
|
|
|
self.ns = qtNamespace()
|
|
|
|
|
|
|
|
#warn("NAMESPACE: '%s'" % self.ns)
|
|
|
|
#warn("VARIABLES: %s" % varList)
|
|
|
|
#warn("EXPANDED INAMES: %s" % self.expandedINames)
|
|
|
|
#warn("WATCHERS: %s" % watchers)
|
|
|
|
#warn("PARTIAL: %s" % self.partialUpdate)
|
|
|
|
#warn("NO LOCALS: %s" % self.noLocals)
|
|
|
|
module = sys.modules[__name__]
|
|
|
|
|
|
|
|
#
|
|
|
|
# Locals
|
|
|
|
#
|
|
|
|
locals = []
|
|
|
|
fullUpdateNeeded = True
|
|
|
|
if self.partialUpdate and len(varList) == 1 and not self.tooltipOnly:
|
|
|
|
#warn("PARTIAL: %s" % varList)
|
|
|
|
parts = varList[0].split('.')
|
|
|
|
#warn("PARTIAL PARTS: %s" % parts)
|
|
|
|
name = parts[1]
|
|
|
|
#warn("PARTIAL VAR: %s" % name)
|
|
|
|
#fullUpdateNeeded = False
|
|
|
|
try:
|
|
|
|
frame = gdb.selected_frame()
|
|
|
|
item = LocalItem()
|
|
|
|
item.iname = "local." + name
|
|
|
|
item.name = name
|
|
|
|
item.value = frame.read_var(name)
|
|
|
|
locals = [item]
|
|
|
|
#warn("PARTIAL LOCALS: %s" % locals)
|
|
|
|
fullUpdateNeeded = False
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
varList = []
|
|
|
|
|
|
|
|
if fullUpdateNeeded and not self.tooltipOnly and not self.noLocals:
|
|
|
|
locals = listOfLocals(varList)
|
|
|
|
|
|
|
|
# Take care of the return value of the last function call.
|
|
|
|
if len(resultVarName) > 0:
|
|
|
|
try:
|
|
|
|
item = LocalItem()
|
|
|
|
item.name = resultVarName
|
|
|
|
item.iname = "return." + resultVarName
|
2013-10-16 16:50:26 +02:00
|
|
|
item.value = self.parseAndEvaluate(resultVarName)
|
2013-05-15 15:42:55 +02:00
|
|
|
locals.append(item)
|
|
|
|
except:
|
|
|
|
# Don't bother. It's only supplementary information anyway.
|
|
|
|
pass
|
|
|
|
|
|
|
|
for item in locals:
|
|
|
|
value = downcast(item.value) if self.useDynamicType else item.value
|
|
|
|
with OutputSafer(self):
|
|
|
|
self.anonNumber = -1
|
|
|
|
|
|
|
|
type = value.type.unqualified()
|
|
|
|
typeName = str(type)
|
|
|
|
|
|
|
|
# Special handling for char** argv.
|
|
|
|
if type.code == PointerCode \
|
|
|
|
and item.iname == "local.argv" \
|
|
|
|
and typeName == "char **":
|
|
|
|
n = 0
|
|
|
|
p = value
|
|
|
|
# p is 0 for "optimized out" cases. Or contains rubbish.
|
|
|
|
try:
|
2013-09-11 21:35:39 +02:00
|
|
|
if not self.isNull(p):
|
|
|
|
while not self.isNull(p.dereference()) and n <= 100:
|
2013-05-15 15:42:55 +02:00
|
|
|
p += 1
|
|
|
|
n += 1
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
with TopLevelItem(self, item.iname):
|
|
|
|
self.put('iname="local.argv",name="argv",')
|
|
|
|
self.putItemCount(n, 100)
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putNumChild(n)
|
|
|
|
if self.currentIName in self.expandedINames:
|
|
|
|
p = value
|
|
|
|
with Children(self, n):
|
|
|
|
for i in xrange(n):
|
|
|
|
self.putSubItem(i, p.dereference())
|
|
|
|
p += 1
|
|
|
|
continue
|
|
|
|
|
|
|
|
else:
|
|
|
|
# A "normal" local variable or parameter.
|
|
|
|
with TopLevelItem(self, item.iname):
|
|
|
|
self.put('iname="%s",' % item.iname)
|
|
|
|
self.put('name="%s",' % item.name)
|
|
|
|
self.putItem(value)
|
|
|
|
|
|
|
|
#
|
|
|
|
# Watchers
|
|
|
|
#
|
|
|
|
with OutputSafer(self):
|
|
|
|
if len(watchers) > 0:
|
|
|
|
self.put(",")
|
|
|
|
for watcher in watchers.split("##"):
|
|
|
|
(exp, iname) = watcher.split("#")
|
|
|
|
self.handleWatch(exp, iname)
|
|
|
|
|
|
|
|
#print('data=[' + locals + sep + watchers + ']\n')
|
|
|
|
|
2013-05-23 15:47:28 +02:00
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
def enterSubItem(self, item):
|
|
|
|
if not item.iname:
|
|
|
|
item.iname = "%s.%s" % (self.currentIName, item.name)
|
|
|
|
#warn("INAME %s" % item.iname)
|
|
|
|
self.put('{')
|
|
|
|
#if not item.name is None:
|
|
|
|
if isinstance(item.name, str):
|
|
|
|
self.put('name="%s",' % item.name)
|
|
|
|
item.savedIName = self.currentIName
|
|
|
|
item.savedValue = self.currentValue
|
|
|
|
item.savedValuePriority = self.currentValuePriority
|
|
|
|
item.savedValueEncoding = self.currentValueEncoding
|
|
|
|
item.savedType = self.currentType
|
|
|
|
item.savedTypePriority = self.currentTypePriority
|
|
|
|
item.savedCurrentAddress = self.currentAddress
|
|
|
|
self.currentIName = item.iname
|
|
|
|
self.currentValuePriority = -100
|
|
|
|
self.currentValueEncoding = None
|
|
|
|
self.currentType = ""
|
|
|
|
self.currentTypePriority = -100
|
|
|
|
self.currentAddress = None
|
|
|
|
|
|
|
|
def exitSubItem(self, item, exType, exValue, exTraceBack):
|
|
|
|
#warn(" CURRENT VALUE: %s %s %s" % (self.currentValue,
|
|
|
|
# self.currentValueEncoding, self.currentValuePriority))
|
|
|
|
if not exType is None:
|
|
|
|
if self.passExceptions:
|
|
|
|
showException("SUBITEM", exType, exValue, exTraceBack)
|
|
|
|
self.putNumChild(0)
|
|
|
|
self.putValue("<not accessible>")
|
|
|
|
try:
|
|
|
|
#warn("TYPE VALUE: %s" % self.currentValue)
|
|
|
|
typeName = stripClassTag(self.currentType)
|
|
|
|
#warn("TYPE: '%s' DEFAULT: '%s' % (typeName, self.currentChildType))
|
|
|
|
|
|
|
|
if len(typeName) > 0 and typeName != self.currentChildType:
|
|
|
|
self.put('type="%s",' % typeName) # str(type.unqualified()) ?
|
|
|
|
if self.currentValue is None:
|
|
|
|
self.put('value="<not accessible>",numchild="0",')
|
|
|
|
else:
|
|
|
|
if not self.currentValueEncoding is None:
|
|
|
|
self.put('valueencoded="%d",' % self.currentValueEncoding)
|
|
|
|
self.put('value="%s",' % self.currentValue)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
if not self.currentAddress is None:
|
|
|
|
self.put(self.currentAddress)
|
|
|
|
self.put('},')
|
|
|
|
self.currentIName = item.savedIName
|
|
|
|
self.currentValue = item.savedValue
|
|
|
|
self.currentValuePriority = item.savedValuePriority
|
|
|
|
self.currentValueEncoding = item.savedValueEncoding
|
|
|
|
self.currentType = item.savedType
|
|
|
|
self.currentTypePriority = item.savedTypePriority
|
|
|
|
self.currentAddress = item.savedCurrentAddress
|
|
|
|
return True
|
|
|
|
|
2013-10-16 16:50:26 +02:00
|
|
|
def parseAndEvaluate(self, exp):
|
|
|
|
return gdb.parse_and_eval(exp)
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
def call2(self, value, func, args):
|
|
|
|
# args is a tuple.
|
|
|
|
arg = ""
|
|
|
|
for i in range(len(args)):
|
|
|
|
if i:
|
|
|
|
arg += ','
|
|
|
|
a = args[i]
|
|
|
|
if (':' in a) and not ("'" in a):
|
|
|
|
arg = "'%s'" % a
|
|
|
|
else:
|
|
|
|
arg += a
|
|
|
|
|
|
|
|
#warn("CALL: %s -> %s(%s)" % (value, func, arg))
|
|
|
|
type = stripClassTag(str(value.type))
|
|
|
|
if type.find(":") >= 0:
|
|
|
|
type = "'" + type + "'"
|
|
|
|
# 'class' is needed, see http://sourceware.org/bugzilla/show_bug.cgi?id=11912
|
|
|
|
exp = "((class %s*)%s)->%s(%s)" % (type, value.address, func, arg)
|
|
|
|
#warn("CALL: %s" % exp)
|
|
|
|
result = None
|
|
|
|
try:
|
2013-10-16 16:50:26 +02:00
|
|
|
result = self.parseAndEvaluate(exp)
|
2013-09-11 21:35:39 +02:00
|
|
|
except:
|
|
|
|
pass
|
|
|
|
#warn(" -> %s" % result)
|
|
|
|
return result
|
|
|
|
|
|
|
|
def call(self, value, func, *args):
|
|
|
|
return self.call2(value, func, args)
|
|
|
|
|
2013-10-16 17:04:34 +02:00
|
|
|
def makeValue(self, type, init):
|
|
|
|
type = "::" + stripClassTag(str(type));
|
|
|
|
# Avoid malloc symbol clash with QVector.
|
|
|
|
gdb.execute("set $d = (%s*)calloc(sizeof(%s), 1)" % (type, type))
|
|
|
|
gdb.execute("set *$d = {%s}" % init)
|
|
|
|
value = parseAndEvaluate("$d").dereference()
|
|
|
|
#warn(" TYPE: %s" % value.type)
|
|
|
|
#warn(" ADDR: %s" % value.address)
|
|
|
|
#warn(" VALUE: %s" % value)
|
|
|
|
return value
|
|
|
|
|
|
|
|
def makeExpression(self, value):
|
|
|
|
type = "::" + stripClassTag(str(value.type))
|
|
|
|
#warn(" TYPE: %s" % type)
|
|
|
|
#exp = "(*(%s*)(&%s))" % (type, value.address)
|
|
|
|
exp = "(*(%s*)(%s))" % (type, value.address)
|
|
|
|
#warn(" EXP: %s" % exp)
|
|
|
|
return exp
|
|
|
|
|
|
|
|
def makeStdString(init):
|
|
|
|
# Works only for small allocators, but they are usually empty.
|
|
|
|
gdb.execute("set $d=(std::string*)calloc(sizeof(std::string), 2)");
|
|
|
|
gdb.execute("call($d->basic_string(\"" + init +
|
|
|
|
"\",*(std::allocator<char>*)(1+$d)))")
|
|
|
|
value = parseAndEvaluate("$d").dereference()
|
|
|
|
#warn(" TYPE: %s" % value.type)
|
|
|
|
#warn(" ADDR: %s" % value.address)
|
|
|
|
#warn(" VALUE: %s" % value)
|
|
|
|
return value
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
def childAt(self, value, index):
|
|
|
|
field = value.type.fields()[index]
|
|
|
|
if len(field.name):
|
|
|
|
return value[field.name]
|
|
|
|
# FIXME: Cheat. There seems to be no official way to access
|
|
|
|
# the real item, so we pass back the value. That at least
|
|
|
|
# enables later ...["name"] style accesses as gdb handles
|
|
|
|
# them transparently.
|
|
|
|
return value
|
|
|
|
|
|
|
|
def fieldAt(self, type, index):
|
|
|
|
return type.fields()[index]
|
|
|
|
|
|
|
|
def directBaseClass(self, typeobj, index = 0):
|
|
|
|
# FIXME: Check it's really a base.
|
|
|
|
return typeobj.fields()[index]
|
|
|
|
|
|
|
|
def checkPointer(self, p, align = 1):
|
|
|
|
if not self.isNull(p):
|
|
|
|
p.dereference()
|
|
|
|
|
|
|
|
def pointerValue(self, p):
|
|
|
|
return toInteger(p)
|
|
|
|
|
|
|
|
def isNull(self, 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 ")
|
|
|
|
#try:
|
|
|
|
# # Can fail with: "RuntimeError: Cannot access memory at address 0x5"
|
2013-10-16 18:39:01 +02:00
|
|
|
# return p.cast(self.lookupType("void").pointer()) == 0
|
2013-09-11 21:35:39 +02:00
|
|
|
#except:
|
|
|
|
# return False
|
|
|
|
try:
|
|
|
|
# Can fail with: "RuntimeError: Cannot access memory at address 0x5"
|
|
|
|
return toInteger(p) == 0
|
|
|
|
except:
|
|
|
|
return False
|
|
|
|
|
2013-05-23 15:47:28 +02:00
|
|
|
def templateArgument(self, typeobj, position):
|
|
|
|
return templateArgument(typeobj, position)
|
|
|
|
|
|
|
|
def numericTemplateArgument(self, typeobj, position):
|
|
|
|
return numericTemplateArgument(typeobj, position)
|
|
|
|
|
2013-05-17 13:53:49 +02:00
|
|
|
def lookupType(self, typeName):
|
|
|
|
return lookupType(typeName)
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
def handleWatch(self, exp, iname):
|
|
|
|
exp = str(exp)
|
2013-09-11 21:35:39 +02:00
|
|
|
escapedExp = b64encode(exp);
|
2013-05-15 15:42:55 +02:00
|
|
|
#warn("HANDLING WATCH %s, INAME: '%s'" % (exp, iname))
|
|
|
|
if exp.startswith("[") and exp.endswith("]"):
|
|
|
|
#warn("EVAL: EXP: %s" % exp)
|
|
|
|
with TopLevelItem(self, iname):
|
|
|
|
self.put('iname="%s",' % iname)
|
|
|
|
self.put('wname="%s",' % escapedExp)
|
|
|
|
try:
|
|
|
|
list = eval(exp)
|
|
|
|
self.putValue("")
|
|
|
|
self.putNoType()
|
|
|
|
self.putNumChild(len(list))
|
|
|
|
# This is a list of expressions to evaluate
|
|
|
|
with Children(self, len(list)):
|
|
|
|
itemNumber = 0
|
|
|
|
for item in list:
|
|
|
|
self.handleWatch(item, "%s.%d" % (iname, itemNumber))
|
|
|
|
itemNumber += 1
|
2013-09-11 21:35:39 +02:00
|
|
|
except RuntimeError as error:
|
2013-05-15 15:42:55 +02:00
|
|
|
warn("EVAL: ERROR CAUGHT %s" % error)
|
|
|
|
self.putValue("<syntax error>")
|
|
|
|
self.putNoType()
|
|
|
|
self.putNumChild(0)
|
|
|
|
self.put("children=[],")
|
|
|
|
return
|
|
|
|
|
|
|
|
with TopLevelItem(self, iname):
|
|
|
|
self.put('iname="%s",' % iname)
|
|
|
|
self.put('wname="%s",' % escapedExp)
|
|
|
|
if len(exp) == 0: # The <Edit> case
|
|
|
|
self.putValue(" ")
|
|
|
|
self.putNoType()
|
|
|
|
self.putNumChild(0)
|
|
|
|
else:
|
|
|
|
try:
|
2013-10-16 16:50:26 +02:00
|
|
|
value = self.parseAndEvaluate(exp)
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putItem(value)
|
|
|
|
except RuntimeError:
|
|
|
|
self.currentType = " "
|
|
|
|
self.currentValue = "<no such value>"
|
|
|
|
self.currentChildNumChild = -1
|
|
|
|
self.currentNumChild = 0
|
|
|
|
self.putNumChild(0)
|
|
|
|
|
2013-05-17 13:53:49 +02:00
|
|
|
def intType(self):
|
2013-06-13 17:56:35 +02:00
|
|
|
return self.lookupType('int')
|
2013-05-17 13:53:49 +02:00
|
|
|
|
|
|
|
def charType(self):
|
2013-06-13 17:56:35 +02:00
|
|
|
return self.lookupType('char')
|
2013-05-17 13:53:49 +02:00
|
|
|
|
|
|
|
def sizetType(self):
|
2013-06-13 17:56:35 +02:00
|
|
|
return self.lookupType('size_t')
|
2013-05-17 13:53:49 +02:00
|
|
|
|
|
|
|
def charPtrType(self):
|
2013-06-13 17:56:35 +02:00
|
|
|
return self.lookupType('char*')
|
2013-05-17 13:53:49 +02:00
|
|
|
|
2013-06-27 17:18:44 +02:00
|
|
|
def intPtrType(self):
|
|
|
|
return self.lookupType('int*')
|
|
|
|
|
2013-05-17 13:53:49 +02:00
|
|
|
def voidPtrType(self):
|
2013-06-13 17:56:35 +02:00
|
|
|
return self.lookupType('void*')
|
2013-05-17 13:53:49 +02:00
|
|
|
|
2013-05-17 17:14:45 +02:00
|
|
|
def addressOf(self, value):
|
2013-09-11 21:35:39 +02:00
|
|
|
return toInteger(value.address)
|
2013-05-17 17:14:45 +02:00
|
|
|
|
2013-06-28 14:04:05 +02:00
|
|
|
def createPointerValue(self, address, pointeeType):
|
|
|
|
return gdb.Value(address).cast(pointeeType.pointer())
|
|
|
|
|
|
|
|
def intSize(self):
|
|
|
|
return 4
|
|
|
|
|
|
|
|
def ptrSize(self):
|
|
|
|
return self.lookupType('void*').sizeof
|
|
|
|
|
2013-07-02 11:42:05 +02:00
|
|
|
def is32bit(self):
|
|
|
|
return self.lookupType('void*').sizeof == 4
|
|
|
|
|
2013-06-28 14:04:05 +02:00
|
|
|
def createValue(self, address, referencedType):
|
|
|
|
return gdb.Value(address).cast(referencedType.pointer()).dereference()
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
def readRawMemory(self, addr, size):
|
|
|
|
mem = gdb.selected_inferior().read_memory(addr, size)
|
|
|
|
if sys.version_info[0] >= 3:
|
|
|
|
mem.tobytes()
|
|
|
|
return mem
|
|
|
|
|
|
|
|
# FIXME: The commented out versions do not work with
|
|
|
|
# a Python 3 based GDB due to the bug addressed in
|
|
|
|
# https://sourceware.org/ml/gdb-patches/2013-09/msg00571.html
|
2013-06-28 14:04:05 +02:00
|
|
|
def dereference(self, addr):
|
2013-09-11 21:35:39 +02:00
|
|
|
#return long(gdb.Value(addr).cast(self.voidPtrType().pointer()).dereference())
|
|
|
|
return struct.unpack("P", self.readRawMemory(addr, self.ptrSize()))[0]
|
2013-06-28 14:04:05 +02:00
|
|
|
|
2013-10-22 13:51:35 +02:00
|
|
|
def extractInt64(self, addr):
|
|
|
|
return struct.unpack("q", self.readRawMemory(addr, 8))[0]
|
|
|
|
|
2013-06-28 14:04:05 +02:00
|
|
|
def extractInt(self, addr):
|
2013-09-11 21:35:39 +02:00
|
|
|
#return long(gdb.Value(addr).cast(self.intPtrType()).dereference())
|
|
|
|
return struct.unpack("i", self.readRawMemory(addr, 4))[0]
|
2013-06-28 14:04:05 +02:00
|
|
|
|
2013-07-09 14:19:45 -07:00
|
|
|
def extractByte(self, addr):
|
2013-09-11 21:35:39 +02:00
|
|
|
#return long(gdb.Value(addr).cast(self.charPtrType()).dereference()) & 0xFF
|
|
|
|
return struct.unpack("b", self.readRawMemory(addr, 1))[0]
|
2013-07-09 14:19:45 -07:00
|
|
|
|
2013-07-02 17:53:42 +02:00
|
|
|
# Do not use value.address here as this might not have one,
|
|
|
|
# i.e. be the result of an inferior call
|
2013-06-28 14:04:05 +02:00
|
|
|
def dereferenceValue(self, value):
|
2013-09-11 21:35:39 +02:00
|
|
|
return toInteger(value.cast(self.voidPtrType()))
|
2013-06-28 14:04:05 +02:00
|
|
|
|
2013-06-28 14:19:59 +02:00
|
|
|
def isQObject(self, value):
|
|
|
|
try:
|
2013-07-01 14:29:00 +02:00
|
|
|
#if True:
|
2013-09-11 21:35:39 +02:00
|
|
|
vtable = self.dereference(toInteger(value.address)) # + ptrSize
|
2013-07-01 14:29:00 +02:00
|
|
|
metaObjectEntry = self.dereference(vtable) # It's the first entry.
|
2013-07-02 11:42:05 +02:00
|
|
|
#warn("MO: 0x%x " % metaObjectEntry)
|
2013-07-01 14:29:00 +02:00
|
|
|
s = gdb.execute("info symbol 0x%x" % metaObjectEntry, to_string=True)
|
2013-07-02 11:42:05 +02:00
|
|
|
#warn("S: %s " % s)
|
2013-07-01 14:29:00 +02:00
|
|
|
#return s.find("::metaObject() const") > 0
|
|
|
|
return s.find("::metaObject() const") > 0 or s.find("10metaObjectEv") > 0
|
|
|
|
#return str(metaObjectEntry).find("::metaObject() const") > 0
|
2013-06-28 14:19:59 +02:00
|
|
|
except:
|
|
|
|
return False
|
2013-07-01 14:29:00 +02:00
|
|
|
|
|
|
|
def isQObject_B(self, value):
|
2013-06-28 14:19:59 +02:00
|
|
|
# Alternative: Check for specific values, like targeting the
|
|
|
|
# 'childEvent' member which is typically not overwritten, slot 8.
|
2013-07-01 14:29:00 +02:00
|
|
|
# ~"Symbol \"Myns::QObject::childEvent(Myns::QChildEvent*)\" is a
|
|
|
|
# function at address 0xb70f691a.\n"
|
|
|
|
if self.childEventAddress == None:
|
|
|
|
try:
|
|
|
|
loc = gdb.execute("info address ::QObject::childEvent", to_string=True)
|
2013-09-11 21:35:39 +02:00
|
|
|
self.childEventAddress = toInteger(loc[loc.rfind(' '):-2], 16)
|
2013-07-01 14:29:00 +02:00
|
|
|
except:
|
|
|
|
self.childEventAddress = 0
|
|
|
|
|
|
|
|
try:
|
2013-09-11 21:35:39 +02:00
|
|
|
vtable = self.dereference(toInteger(value.address))
|
2013-07-01 14:29:00 +02:00
|
|
|
return self.childEventAddress == self.dereference(vtable + 8 * self.ptrSize())
|
|
|
|
except:
|
|
|
|
return False
|
2013-06-28 14:19:59 +02:00
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
def put(self, value):
|
|
|
|
self.output.append(value)
|
|
|
|
|
|
|
|
def putField(self, name, value):
|
|
|
|
self.put('%s="%s",' % (name, value))
|
|
|
|
|
|
|
|
def childRange(self):
|
|
|
|
if self.currentMaxNumChild is None:
|
2013-09-11 21:35:39 +02:00
|
|
|
return xrange(0, toInteger(self.currentNumChild))
|
|
|
|
return xrange(min(toInteger(self.currentMaxNumChild), toInteger(self.currentNumChild)))
|
2013-05-15 15:42:55 +02:00
|
|
|
|
2013-06-27 10:42:32 +02:00
|
|
|
def qtVersion(self):
|
2013-10-22 13:21:56 +02:00
|
|
|
if self.cachedQtVersion is None:
|
|
|
|
try:
|
|
|
|
self.cachedQtVersion = extractQtVersion()
|
|
|
|
except:
|
|
|
|
try:
|
|
|
|
# This will fail on Qt 5
|
|
|
|
gdb.execute("ptype QString::shared_empty", to_string=True)
|
|
|
|
self.cachedQtVersion = 0x040800
|
|
|
|
except:
|
|
|
|
self.cachedQtVersion = 0x050000
|
|
|
|
return self.cachedQtVersion
|
2013-06-27 10:42:32 +02:00
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
# Convenience function.
|
|
|
|
def putItemCount(self, count, maximum = 1000000000):
|
|
|
|
# This needs to override the default value, so don't use 'put' directly.
|
|
|
|
if count > maximum:
|
|
|
|
self.putValue('<>%s items>' % maximum)
|
|
|
|
else:
|
|
|
|
self.putValue('<%s items>' % count)
|
|
|
|
|
|
|
|
def putType(self, type, priority = 0):
|
|
|
|
# Higher priority values override lower ones.
|
|
|
|
if priority >= self.currentTypePriority:
|
|
|
|
self.currentType = str(type)
|
|
|
|
self.currentTypePriority = priority
|
|
|
|
|
|
|
|
def putNoType(self):
|
|
|
|
# FIXME: replace with something that does not need special handling
|
|
|
|
# in SubItem.__exit__().
|
|
|
|
self.putBetterType(" ")
|
|
|
|
|
|
|
|
def putInaccessible(self):
|
|
|
|
#self.putBetterType(" ")
|
|
|
|
self.putNumChild(0)
|
|
|
|
self.currentValue = None
|
|
|
|
|
2013-05-17 17:14:45 +02:00
|
|
|
def putBetterType(self, type):
|
2013-05-15 15:42:55 +02:00
|
|
|
self.currentType = str(type)
|
|
|
|
self.currentTypePriority = self.currentTypePriority + 1
|
|
|
|
|
|
|
|
def putAddress(self, addr):
|
|
|
|
if self.currentPrintsAddress:
|
|
|
|
try:
|
2013-09-11 21:35:39 +02:00
|
|
|
# addr can be "None", int(None) fails.
|
|
|
|
#self.put('addr="0x%x",' % int(addr))
|
|
|
|
self.currentAddress = 'addr="0x%x",' % toInteger(addr)
|
2013-05-15 15:42:55 +02:00
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
def putNumChild(self, numchild):
|
|
|
|
#warn("NUM CHILD: '%s' '%s'" % (numchild, self.currentChildNumChild))
|
|
|
|
if numchild != self.currentChildNumChild:
|
|
|
|
self.put('numchild="%s",' % numchild)
|
|
|
|
|
|
|
|
def putEmptyValue(self, priority = -10):
|
|
|
|
if priority >= self.currentValuePriority:
|
|
|
|
self.currentValue = ""
|
|
|
|
self.currentValuePriority = priority
|
|
|
|
self.currentValueEncoding = None
|
|
|
|
|
|
|
|
def putValue(self, value, encoding = None, priority = 0):
|
|
|
|
# Higher priority values override lower ones.
|
|
|
|
if priority >= self.currentValuePriority:
|
|
|
|
self.currentValue = value
|
|
|
|
self.currentValuePriority = priority
|
|
|
|
self.currentValueEncoding = encoding
|
|
|
|
|
|
|
|
def putPointerValue(self, value):
|
|
|
|
# Use a lower priority
|
|
|
|
if value is None:
|
|
|
|
self.putEmptyValue(-1)
|
|
|
|
else:
|
|
|
|
self.putValue("0x%x" % value.cast(
|
2013-10-16 18:39:01 +02:00
|
|
|
self.lookupType("unsigned long")), None, -1)
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
def putDisplay(self, format, value = None, cmd = None):
|
|
|
|
self.put('editformat="%s",' % format)
|
|
|
|
if cmd is None:
|
|
|
|
if not value is None:
|
|
|
|
self.put('editvalue="%s",' % value)
|
|
|
|
else:
|
|
|
|
self.put('editvalue="%s|%s",' % (cmd, value))
|
|
|
|
|
|
|
|
def putName(self, name):
|
|
|
|
self.put('name="%s",' % name)
|
|
|
|
|
|
|
|
def isExpanded(self):
|
|
|
|
#warn("IS EXPANDED: %s in %s: %s" % (self.currentIName,
|
|
|
|
# self.expandedINames, self.currentIName in self.expandedINames))
|
|
|
|
return self.currentIName in self.expandedINames
|
|
|
|
|
|
|
|
def isExpandedSubItem(self, component):
|
|
|
|
iname = "%s.%s" % (self.currentIName, component)
|
|
|
|
#warn("IS EXPANDED: %s in %s" % (iname, self.expandedINames))
|
|
|
|
return iname in self.expandedINames
|
|
|
|
|
|
|
|
def stripNamespaceFromType(self, typeName):
|
|
|
|
type = stripClassTag(typeName)
|
|
|
|
if len(self.ns) > 0 and type.startswith(self.ns):
|
|
|
|
type = type[len(self.ns):]
|
|
|
|
pos = type.find("<")
|
|
|
|
# FIXME: make it recognize foo<A>::bar<B>::iterator?
|
|
|
|
while pos != -1:
|
|
|
|
pos1 = type.rfind(">", pos)
|
|
|
|
type = type[0:pos] + type[pos1+1:]
|
|
|
|
pos = type.find("<")
|
|
|
|
return type
|
|
|
|
|
|
|
|
def isMovableType(self, type):
|
|
|
|
if type.code == PointerCode:
|
|
|
|
return True
|
2013-09-11 21:35:39 +02:00
|
|
|
if self.isSimpleType(type):
|
2013-05-15 15:42:55 +02:00
|
|
|
return True
|
2013-10-23 16:28:02 +02:00
|
|
|
return self.isKnownMovableType(self.stripNamespaceFromType(str(type)))
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
def putIntItem(self, name, value):
|
|
|
|
with SubItem(self, name):
|
|
|
|
self.putValue(value)
|
|
|
|
self.putType("int")
|
|
|
|
self.putNumChild(0)
|
|
|
|
|
|
|
|
def putBoolItem(self, name, value):
|
|
|
|
with SubItem(self, name):
|
|
|
|
self.putValue(value)
|
|
|
|
self.putType("bool")
|
|
|
|
self.putNumChild(0)
|
|
|
|
|
2013-10-18 10:52:53 +02:00
|
|
|
def putGenericItem(self, name, type, value, encoding = None):
|
|
|
|
with SubItem(self, name):
|
|
|
|
self.putValue(value, encoding)
|
|
|
|
self.putType(type)
|
|
|
|
self.putNumChild(0)
|
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
def currentItemFormat(self):
|
|
|
|
format = self.formats.get(self.currentIName)
|
|
|
|
if format is None:
|
|
|
|
format = self.typeformats.get(stripForFormat(str(self.currentType)))
|
|
|
|
return format
|
|
|
|
|
|
|
|
def putSubItem(self, component, value, tryDynamic=True):
|
|
|
|
with SubItem(self, component):
|
|
|
|
self.putItem(value, tryDynamic)
|
|
|
|
|
|
|
|
def putNamedSubItem(self, component, value, name):
|
|
|
|
with SubItem(self, component):
|
|
|
|
self.putName(name)
|
|
|
|
self.putItem(value)
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
def isSimpleType(self, typeobj):
|
|
|
|
code = typeobj.code
|
|
|
|
return code == BoolCode \
|
|
|
|
or code == CharCode \
|
|
|
|
or code == IntCode \
|
|
|
|
or code == FloatCode \
|
2013-10-10 14:33:05 +02:00
|
|
|
or code == EnumCode
|
2013-09-11 21:35:39 +02:00
|
|
|
|
|
|
|
def simpleEncoding(self, typeobj):
|
|
|
|
code = typeobj.code
|
|
|
|
if code == BoolCode or code == CharCode:
|
|
|
|
return Hex2EncodedInt1
|
|
|
|
if code == IntCode:
|
|
|
|
if str(typeobj).find("unsigned") >= 0:
|
|
|
|
if typeobj.sizeof == 1:
|
|
|
|
return Hex2EncodedUInt1
|
|
|
|
if typeobj.sizeof == 2:
|
|
|
|
return Hex2EncodedUInt2
|
|
|
|
if typeobj.sizeof == 4:
|
|
|
|
return Hex2EncodedUInt4
|
|
|
|
if typeobj.sizeof == 8:
|
|
|
|
return Hex2EncodedUInt8
|
|
|
|
else:
|
|
|
|
if typeobj.sizeof == 1:
|
|
|
|
return Hex2EncodedInt1
|
|
|
|
if typeobj.sizeof == 2:
|
|
|
|
return Hex2EncodedInt2
|
|
|
|
if typeobj.sizeof == 4:
|
|
|
|
return Hex2EncodedInt4
|
|
|
|
if typeobj.sizeof == 8:
|
|
|
|
return Hex2EncodedInt8
|
|
|
|
if code == FloatCode:
|
|
|
|
if typeobj.sizeof == 4:
|
|
|
|
return Hex2EncodedFloat4
|
|
|
|
if typeobj.sizeof == 8:
|
|
|
|
return Hex2EncodedFloat8
|
|
|
|
return None
|
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
def tryPutArrayContents(self, typeobj, base, n):
|
2013-10-01 00:45:27 +02:00
|
|
|
enc = self.simpleEncoding(typeobj)
|
|
|
|
if not enc:
|
2013-05-15 15:42:55 +02:00
|
|
|
return False
|
|
|
|
size = n * typeobj.sizeof;
|
|
|
|
self.put('childtype="%s",' % typeobj)
|
2013-09-11 21:35:39 +02:00
|
|
|
self.put('addrbase="0x%x",' % toInteger(base))
|
|
|
|
self.put('addrstep="0x%x",' % toInteger(typeobj.sizeof))
|
2013-10-01 00:45:27 +02:00
|
|
|
self.put('arrayencoding="%s",' % enc)
|
2013-05-15 15:42:55 +02:00
|
|
|
self.put('arraydata="')
|
2013-09-11 21:35:39 +02:00
|
|
|
self.put(self.readMemory(base, size))
|
2013-05-15 15:42:55 +02:00
|
|
|
self.put('",')
|
|
|
|
return True
|
|
|
|
|
2013-07-03 13:32:19 +02:00
|
|
|
def isReferenceType(self, typeobj):
|
|
|
|
return typeobj.code == gdb.TYPE_CODE_REF
|
|
|
|
|
|
|
|
def isStructType(self, typeobj):
|
|
|
|
return typeobj.code == gdb.TYPE_CODE_STRUCT
|
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
def putPlotData(self, type, base, n, plotFormat):
|
|
|
|
if self.isExpanded():
|
|
|
|
self.putArrayData(type, base, n)
|
|
|
|
if not hasPlot():
|
|
|
|
return
|
2013-09-11 21:35:39 +02:00
|
|
|
if not self.isSimpleType(type):
|
2013-05-15 15:42:55 +02:00
|
|
|
#self.putValue(self.currentValue + " (not plottable)")
|
|
|
|
self.putValue(self.currentValue)
|
|
|
|
self.putField("plottable", "0")
|
|
|
|
return
|
|
|
|
global gnuplotPipe
|
|
|
|
global gnuplotPid
|
|
|
|
format = self.currentItemFormat()
|
|
|
|
iname = self.currentIName
|
|
|
|
#if False:
|
|
|
|
if format != plotFormat:
|
|
|
|
if iname in gnuplotPipe:
|
|
|
|
os.kill(gnuplotPid[iname], 9)
|
|
|
|
del gnuplotPid[iname]
|
|
|
|
gnuplotPipe[iname].terminate()
|
|
|
|
del gnuplotPipe[iname]
|
|
|
|
return
|
|
|
|
base = base.cast(type.pointer())
|
|
|
|
if not iname in gnuplotPipe:
|
|
|
|
gnuplotPipe[iname] = subprocess.Popen(["gnuplot"],
|
|
|
|
stdin=subprocess.PIPE)
|
|
|
|
gnuplotPid[iname] = gnuplotPipe[iname].pid
|
|
|
|
f = gnuplotPipe[iname].stdin;
|
2013-09-24 14:39:41 +02:00
|
|
|
# On Ubuntu install gnuplot-x11
|
2013-05-15 15:42:55 +02:00
|
|
|
f.write("set term wxt noraise\n")
|
|
|
|
f.write("set title 'Data fields'\n")
|
|
|
|
f.write("set xlabel 'Index'\n")
|
|
|
|
f.write("set ylabel 'Value'\n")
|
|
|
|
f.write("set grid\n")
|
|
|
|
f.write("set style data lines;\n")
|
|
|
|
f.write("plot '-' title '%s'\n" % iname)
|
|
|
|
for i in range(1, n):
|
|
|
|
f.write(" %s\n" % base.dereference())
|
|
|
|
base += 1
|
|
|
|
f.write("e\n")
|
|
|
|
|
|
|
|
|
|
|
|
def putArrayData(self, type, base, n,
|
|
|
|
childNumChild = None, maxNumChild = 10000):
|
|
|
|
if not self.tryPutArrayContents(type, base, n):
|
2013-05-17 13:53:49 +02:00
|
|
|
base = base.cast(type.pointer())
|
2013-05-15 15:42:55 +02:00
|
|
|
with Children(self, n, type, childNumChild, maxNumChild,
|
|
|
|
base, type.sizeof):
|
|
|
|
for i in self.childRange():
|
|
|
|
self.putSubItem(i, (base + i).dereference())
|
|
|
|
|
|
|
|
def putCallItem(self, name, value, func, *args):
|
2013-09-11 21:35:39 +02:00
|
|
|
result = self.call2(value, func, args)
|
2013-05-15 15:42:55 +02:00
|
|
|
with SubItem(self, name):
|
|
|
|
self.putItem(result)
|
|
|
|
|
|
|
|
def putItem(self, value, tryDynamic=True):
|
|
|
|
if 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
|
|
|
|
|
|
|
|
global qqDumpers, qqFormats
|
|
|
|
|
|
|
|
type = value.type.unqualified()
|
|
|
|
typeName = str(type)
|
|
|
|
tryDynamic &= self.useDynamicType
|
2013-08-12 19:34:48 +02:00
|
|
|
addToCache(type) # Fill type cache
|
2013-05-15 15:42:55 +02:00
|
|
|
if tryDynamic:
|
|
|
|
self.putAddress(value.address)
|
|
|
|
|
|
|
|
# FIXME: Gui shows references stripped?
|
|
|
|
#warn(" ")
|
|
|
|
#warn("REAL INAME: %s " % self.currentIName)
|
|
|
|
#warn("REAL TYPE: %s " % value.type)
|
|
|
|
#warn("REAL CODE: %s " % value.type.code)
|
|
|
|
#warn("REAL VALUE: %s " % value)
|
|
|
|
|
|
|
|
if type.code == ReferenceCode:
|
|
|
|
try:
|
|
|
|
# Try to recognize null references explicitly.
|
2013-09-11 21:35:39 +02:00
|
|
|
if toInteger(value.address) == 0:
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putValue("<null reference>")
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if tryDynamic:
|
|
|
|
try:
|
|
|
|
# Dynamic references are not supported by gdb, see
|
|
|
|
# http://sourceware.org/bugzilla/show_bug.cgi?id=14077.
|
|
|
|
# Find the dynamic type manually using referenced_type.
|
|
|
|
value = value.referenced_value()
|
|
|
|
value = value.cast(value.dynamic_type)
|
|
|
|
self.putItem(value)
|
|
|
|
self.putBetterType("%s &" % value.type)
|
|
|
|
return
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
try:
|
|
|
|
# FIXME: This throws "RuntimeError: Attempt to dereference a
|
|
|
|
# generic pointer." with MinGW's gcc 4.5 when it "identifies"
|
|
|
|
# a "QWidget &" as "void &" and with optimized out code.
|
|
|
|
self.putItem(value.cast(type.target().unqualified()))
|
|
|
|
self.putBetterType(typeName)
|
|
|
|
return
|
|
|
|
except RuntimeError:
|
|
|
|
self.putValue("<optimized out reference>")
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
2013-10-10 14:33:05 +02:00
|
|
|
if type.code == IntCode or type.code == CharCode:
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putType(typeName)
|
|
|
|
if value.is_optimized_out:
|
|
|
|
self.putValue("<optimized out>")
|
|
|
|
else:
|
|
|
|
self.putValue(value)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if type.code == FloatCode or type.code == BoolCode:
|
|
|
|
self.putType(typeName)
|
|
|
|
if value.is_optimized_out:
|
|
|
|
self.putValue("<optimized out>")
|
|
|
|
else:
|
|
|
|
self.putValue(value)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if type.code == EnumCode:
|
|
|
|
self.putType(typeName)
|
|
|
|
if value.is_optimized_out:
|
|
|
|
self.putValue("<optimized out>")
|
|
|
|
else:
|
|
|
|
self.putValue("%s (%d)" % (value, value))
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if type.code == ComplexCode:
|
|
|
|
self.putType(typeName)
|
|
|
|
if value.is_optimized_out:
|
|
|
|
self.putValue("<optimized out>")
|
|
|
|
else:
|
|
|
|
self.putValue("%s" % value)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if type.code == TypedefCode:
|
|
|
|
if typeName in qqDumpers:
|
|
|
|
self.putType(typeName)
|
|
|
|
qqDumpers[typeName](self, value)
|
|
|
|
return
|
|
|
|
|
|
|
|
type = stripTypedefs(type)
|
|
|
|
# The cast can destroy the address?
|
|
|
|
#self.putAddress(value.address)
|
|
|
|
# Workaround for http://sourceware.org/bugzilla/show_bug.cgi?id=13380
|
|
|
|
if type.code == ArrayCode:
|
2013-10-16 16:50:26 +02:00
|
|
|
value = self.parseAndEvaluate("{%s}%s" % (type, value.address))
|
2013-05-15 15:42:55 +02:00
|
|
|
else:
|
|
|
|
try:
|
|
|
|
value = value.cast(type)
|
|
|
|
except:
|
|
|
|
self.putValue("<optimized out typedef>")
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
self.putItem(value)
|
|
|
|
self.putBetterType(typeName)
|
|
|
|
return
|
|
|
|
|
|
|
|
if type.code == ArrayCode:
|
|
|
|
qdump____c_style_array__(self, value)
|
|
|
|
return
|
|
|
|
|
|
|
|
if type.code == PointerCode:
|
|
|
|
#warn("POINTER: %s" % value)
|
|
|
|
|
|
|
|
# This could still be stored in a register and
|
|
|
|
# potentially dereferencable.
|
|
|
|
if value.is_optimized_out:
|
|
|
|
self.putValue("<optimized out>")
|
|
|
|
|
|
|
|
try:
|
|
|
|
value.dereference()
|
|
|
|
except:
|
|
|
|
# Failure to dereference a pointer should at least
|
|
|
|
# show the value of a pointer.
|
|
|
|
self.putValue(cleanAddress(value))
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
if self.isNull(value):
|
2013-05-15 15:42:55 +02:00
|
|
|
#warn("NULL POINTER")
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putValue("0x0")
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
innerType = type.target()
|
|
|
|
innerTypeName = str(innerType.unqualified())
|
|
|
|
format = self.formats.get(self.currentIName)
|
|
|
|
if format is None:
|
|
|
|
format = self.typeformats.get(stripForFormat(str(type)))
|
|
|
|
|
|
|
|
if innerType.code == VoidCode:
|
|
|
|
#warn("VOID POINTER: %s" % format)
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putValue(str(value))
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if format == None and innerTypeName == "char":
|
|
|
|
# Use Latin1 as default for char *.
|
|
|
|
self.putType(typeName)
|
2013-10-10 14:48:20 +02:00
|
|
|
self.putValue(self.encodeCharArray(value), Hex2EncodedLatin1)
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if format == 0:
|
|
|
|
# Explicitly requested bald pointer.
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putPointerValue(value)
|
|
|
|
self.putNumChild(1)
|
|
|
|
if self.currentIName in self.expandedINames:
|
|
|
|
with Children(self):
|
|
|
|
with SubItem(self, '*'):
|
|
|
|
self.putItem(value.dereference())
|
|
|
|
return
|
|
|
|
|
|
|
|
if format == 1:
|
|
|
|
# Explicitly requested Latin1 formatting.
|
|
|
|
self.putType(typeName)
|
2013-10-10 14:48:20 +02:00
|
|
|
self.putValue(self.encodeCharArray(value), Hex2EncodedLatin1)
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if format == 2:
|
|
|
|
# Explicitly requested UTF-8 formatting.
|
|
|
|
self.putType(typeName)
|
2013-10-10 14:48:20 +02:00
|
|
|
self.putValue(self.encodeCharArray(value), Hex2EncodedUtf8)
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if format == 3:
|
|
|
|
# Explicitly requested local 8 bit formatting.
|
|
|
|
self.putType(typeName)
|
2013-10-10 14:48:20 +02:00
|
|
|
self.putValue(self.encodeCharArray(value), Hex2EncodedLocal8Bit)
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if format == 4:
|
|
|
|
# Explicitly requested UTF-16 formatting.
|
|
|
|
self.putType(typeName)
|
2013-10-10 14:48:20 +02:00
|
|
|
self.putValue(self.encodeChar2Array(value), Hex4EncodedLittleEndian)
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if format == 5:
|
|
|
|
# Explicitly requested UCS-4 formatting.
|
|
|
|
self.putType(typeName)
|
2013-10-10 14:48:20 +02:00
|
|
|
self.putValue(self.encodeChar4Array(value), Hex8EncodedLittleEndian)
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
2013-06-07 17:09:31 +02:00
|
|
|
if format == 6:
|
|
|
|
# Explicitly requested formatting as array of 10 items.
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putItemCount(10)
|
|
|
|
self.putNumChild(10)
|
|
|
|
self.putArrayData(innerType, value, 10)
|
|
|
|
return
|
|
|
|
|
|
|
|
if format == 7:
|
|
|
|
# Explicitly requested formatting as array of 1000 items.
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putItemCount(1000)
|
|
|
|
self.putNumChild(1000)
|
|
|
|
self.putArrayData(innerType, value, 1000)
|
|
|
|
return
|
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
if innerType.code == MethodCode or innerType.code == FunctionCode:
|
|
|
|
# A function pointer with format None.
|
|
|
|
self.putValue(str(value))
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
#warn("AUTODEREF: %s" % self.autoDerefPointers)
|
|
|
|
#warn("INAME: %s" % self.currentIName)
|
|
|
|
if self.autoDerefPointers or self.currentIName.endswith('.this'):
|
|
|
|
## Generic pointer type with format None
|
|
|
|
#warn("GENERIC AUTODEREF POINTER: %s AT %s TO %s"
|
|
|
|
# % (type, value.address, innerTypeName))
|
|
|
|
# Never dereference char types.
|
|
|
|
if innerTypeName != "char" \
|
|
|
|
and innerTypeName != "signed char" \
|
|
|
|
and innerTypeName != "unsigned char" \
|
|
|
|
and innerTypeName != "wchar_t":
|
|
|
|
self.putType(innerType)
|
|
|
|
savedCurrentChildType = self.currentChildType
|
|
|
|
self.currentChildType = stripClassTag(innerTypeName)
|
|
|
|
self.putItem(value.dereference())
|
|
|
|
self.currentChildType = savedCurrentChildType
|
|
|
|
#self.putPointerValue(value)
|
|
|
|
self.put('origaddr="%s",' % value.address)
|
|
|
|
return
|
|
|
|
|
|
|
|
# Fall back to plain pointer printing.
|
|
|
|
#warn("GENERIC PLAIN POINTER: %s" % value.type)
|
|
|
|
#warn("ADDR PLAIN POINTER: %s" % value.address)
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putField("aaa", "1")
|
2013-09-11 21:35:39 +02:00
|
|
|
#self.put('addr="0x%x",' % toInteger(value.address))
|
2013-05-15 15:42:55 +02:00
|
|
|
#self.putAddress(value.address)
|
|
|
|
self.putField("bbb", "1")
|
|
|
|
#self.putPointerValue(value)
|
2013-10-16 18:39:01 +02:00
|
|
|
self.putValue("0x%x" % value.cast(self.lookupType("unsigned long")))
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putField("ccc", "1")
|
|
|
|
self.putNumChild(1)
|
|
|
|
if self.currentIName in self.expandedINames:
|
|
|
|
with Children(self):
|
|
|
|
with SubItem(self, "*"):
|
|
|
|
self.putItem(value.dereference())
|
|
|
|
return
|
|
|
|
|
|
|
|
if type.code == MethodPointerCode \
|
|
|
|
or type.code == MethodCode \
|
|
|
|
or type.code == FunctionCode \
|
|
|
|
or type.code == MemberPointerCode:
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putValue(value)
|
|
|
|
self.putNumChild(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
if typeName.startswith("<anon"):
|
|
|
|
# Anonymous union. We need a dummy name to distinguish
|
|
|
|
# multiple anonymous unions in the struct.
|
|
|
|
self.putType(type)
|
|
|
|
self.putValue("{...}")
|
|
|
|
self.anonNumber += 1
|
|
|
|
with Children(self, 1):
|
|
|
|
self.listAnonymous(value, "#%d" % self.anonNumber, type)
|
|
|
|
return
|
|
|
|
|
2013-08-26 22:39:18 +02:00
|
|
|
if type.code == StringCode:
|
|
|
|
# FORTRAN strings
|
|
|
|
size = type.sizeof
|
2013-09-11 21:35:39 +02:00
|
|
|
data = self.readMemory(value.address, size)
|
2013-08-26 22:39:18 +02:00
|
|
|
self.putValue(data, Hex2EncodedLatin1, 1)
|
|
|
|
self.putType(type)
|
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
if type.code != StructCode and type.code != UnionCode:
|
|
|
|
warn("WRONG ASSUMPTION HERE: %s " % type.code)
|
|
|
|
check(False)
|
|
|
|
|
|
|
|
|
|
|
|
if tryDynamic:
|
|
|
|
self.putItem(expensiveDowncast(value), False)
|
|
|
|
return
|
|
|
|
|
|
|
|
format = self.formats.get(self.currentIName)
|
|
|
|
if format is None:
|
|
|
|
format = self.typeformats.get(stripForFormat(typeName))
|
|
|
|
|
|
|
|
if self.useFancy and (format is None or format >= 1):
|
|
|
|
self.putType(typeName)
|
|
|
|
|
|
|
|
nsStrippedType = self.stripNamespaceFromType(typeName)\
|
|
|
|
.replace("::", "__")
|
|
|
|
|
|
|
|
# The following block is only needed for D.
|
|
|
|
if nsStrippedType.startswith("_A"):
|
|
|
|
# DMD v2.058 encodes string[] as _Array_uns long long.
|
|
|
|
# With spaces.
|
|
|
|
if nsStrippedType.startswith("_Array_"):
|
|
|
|
qdump_Array(self, value)
|
|
|
|
return
|
|
|
|
if nsStrippedType.startswith("_AArray_"):
|
|
|
|
qdump_AArray(self, value)
|
|
|
|
return
|
|
|
|
|
|
|
|
#warn(" STRIPPED: %s" % nsStrippedType)
|
|
|
|
#warn(" DUMPERS: %s" % qqDumpers)
|
|
|
|
#warn(" DUMPERS: %s" % (nsStrippedType in qqDumpers))
|
|
|
|
dumper = qqDumpers.get(nsStrippedType, None)
|
|
|
|
if not dumper is None:
|
|
|
|
if tryDynamic:
|
|
|
|
dumper(self, expensiveDowncast(value))
|
|
|
|
else:
|
|
|
|
dumper(self, value)
|
|
|
|
return
|
|
|
|
|
|
|
|
# D arrays, gdc compiled.
|
|
|
|
if typeName.endswith("[]"):
|
|
|
|
n = value["length"]
|
|
|
|
base = value["ptr"]
|
|
|
|
self.putType(typeName)
|
|
|
|
self.putItemCount(n)
|
|
|
|
if self.isExpanded():
|
|
|
|
self.putArrayData(base.type.target(), base, n)
|
|
|
|
return
|
|
|
|
|
|
|
|
#warn("GENERIC STRUCT: %s" % type)
|
|
|
|
#warn("INAME: %s " % self.currentIName)
|
|
|
|
#warn("INAMES: %s " % self.expandedINames)
|
|
|
|
#warn("EXPANDED: %s " % (self.currentIName in self.expandedINames))
|
2013-06-28 14:19:59 +02:00
|
|
|
if self.isQObject(value):
|
|
|
|
self.putQObjectNameValue(value) # Is this too expensive?
|
2013-05-15 15:42:55 +02:00
|
|
|
self.putType(typeName)
|
|
|
|
self.putEmptyValue()
|
2013-09-11 21:35:39 +02:00
|
|
|
self.putNumChild(len(type.fields()))
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
if self.currentIName in self.expandedINames:
|
|
|
|
innerType = None
|
|
|
|
with Children(self, 1, childType=innerType):
|
|
|
|
self.putFields(value)
|
|
|
|
|
|
|
|
def putPlainChildren(self, value):
|
|
|
|
self.putEmptyValue(-99)
|
|
|
|
self.putNumChild(1)
|
|
|
|
if self.currentIName in self.expandedINames:
|
|
|
|
with Children(self):
|
|
|
|
self.putFields(value)
|
|
|
|
|
2013-09-11 21:35:39 +02:00
|
|
|
def readMemory(self, base, size):
|
|
|
|
inferior = selectedInferior()
|
|
|
|
mem = inferior.read_memory(base, size)
|
|
|
|
if sys.version_info[0] >= 3:
|
|
|
|
return bytesToString(binascii.hexlify(mem.tobytes()))
|
|
|
|
return binascii.hexlify(mem)
|
2013-05-15 16:17:22 +02:00
|
|
|
|
2013-05-15 15:42:55 +02:00
|
|
|
def putFields(self, value, dumpBase = True):
|
2013-09-11 21:35:39 +02:00
|
|
|
fields = value.type.fields()
|
2013-05-15 15:42:55 +02:00
|
|
|
|
|
|
|
#warn("TYPE: %s" % type)
|
|
|
|
#warn("FIELDS: %s" % fields)
|
|
|
|
baseNumber = 0
|
|
|
|
for field in fields:
|
|
|
|
#warn("FIELD: %s" % field)
|
|
|
|
#warn(" BITSIZE: %s" % field.bitsize)
|
|
|
|
#warn(" ARTIFICIAL: %s" % field.artificial)
|
|
|
|
|
|
|
|
if field.name is None:
|
|
|
|
type = stripTypedefs(value.type)
|
|
|
|
innerType = type.target()
|
|
|
|
p = value.cast(innerType.pointer())
|
2013-09-11 21:35:39 +02:00
|
|
|
for i in xrange(int(type.sizeof / innerType.sizeof)):
|
2013-05-15 15:42:55 +02:00
|
|
|
with SubItem(self, i):
|
|
|
|
self.putItem(p.dereference())
|
|
|
|
p = p + 1
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Ignore vtable pointers for virtual inheritance.
|
|
|
|
if field.name.startswith("_vptr."):
|
|
|
|
with SubItem(self, "[vptr]"):
|
|
|
|
# int (**)(void)
|
|
|
|
n = 100
|
|
|
|
self.putType(" ")
|
|
|
|
self.putValue(value[field.name])
|
|
|
|
self.putNumChild(n)
|
|
|
|
if self.isExpanded():
|
|
|
|
with Children(self):
|
|
|
|
p = value[field.name]
|
|
|
|
for i in xrange(n):
|
2013-09-11 21:35:39 +02:00
|
|
|
if toInteger(p.dereference()) != 0:
|
2013-05-15 15:42:55 +02:00
|
|
|
with SubItem(self, i):
|
|
|
|
self.putItem(p.dereference())
|
|
|
|
self.putType(" ")
|
|
|
|
p = p + 1
|
|
|
|
continue
|
|
|
|
|
|
|
|
#warn("FIELD NAME: %s" % field.name)
|
|
|
|
#warn("FIELD TYPE: %s" % field.type)
|
|
|
|
if field.is_base_class:
|
|
|
|
# Field is base type. We cannot use field.name as part
|
|
|
|
# of the iname as it might contain spaces and other
|
|
|
|
# strange characters.
|
|
|
|
if dumpBase:
|
|
|
|
baseNumber += 1
|
|
|
|
with UnnamedSubItem(self, "@%d" % baseNumber):
|
|
|
|
self.put('iname="%s",' % self.currentIName)
|
|
|
|
self.put('name="[%s]",' % field.name)
|
|
|
|
self.putItem(value.cast(field.type), False)
|
|
|
|
elif len(field.name) == 0:
|
|
|
|
# Anonymous union. We need a dummy name to distinguish
|
|
|
|
# multiple anonymous unions in the struct.
|
|
|
|
self.anonNumber += 1
|
|
|
|
self.listAnonymous(value, "#%d" % self.anonNumber,
|
|
|
|
field.type)
|
|
|
|
else:
|
|
|
|
# Named field.
|
|
|
|
with SubItem(self, field.name):
|
|
|
|
#bitsize = getattr(field, "bitsize", None)
|
|
|
|
#if not bitsize is None:
|
|
|
|
# self.put("bitsize=\"%s\"" % bitsize)
|
|
|
|
self.putItem(downcast(value[field.name]))
|
|
|
|
|
|
|
|
|
|
|
|
def listAnonymous(self, value, name, type):
|
|
|
|
for field in type.fields():
|
|
|
|
#warn("FIELD NAME: %s" % field.name)
|
|
|
|
if len(field.name) > 0:
|
|
|
|
with SubItem(self, field.name):
|
|
|
|
self.putItem(value[field.name])
|
|
|
|
else:
|
|
|
|
# Further nested.
|
|
|
|
self.anonNumber += 1
|
|
|
|
name = "#%d" % self.anonNumber
|
|
|
|
#iname = "%s.%s" % (selitem.iname, name)
|
|
|
|
#child = SameItem(item.value, iname)
|
|
|
|
with SubItem(self, name):
|
|
|
|
self.put('name="%s",' % name)
|
|
|
|
self.putEmptyValue()
|
|
|
|
fieldTypeName = str(field.type)
|
|
|
|
if fieldTypeName.endswith("<anonymous union>"):
|
|
|
|
self.putType("<anonymous union>")
|
|
|
|
elif fieldTypeName.endswith("<anonymous struct>"):
|
|
|
|
self.putType("<anonymous struct>")
|
|
|
|
else:
|
|
|
|
self.putType(fieldTypeName)
|
|
|
|
with Children(self, 1):
|
|
|
|
self.listAnonymous(value, name, field.type)
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# ThreadNames Command
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
2013-06-06 18:28:53 +02:00
|
|
|
def threadname(arg):
|
|
|
|
try:
|
|
|
|
e = gdb.selected_frame()
|
|
|
|
except:
|
|
|
|
return
|
|
|
|
d = Dumper("")
|
|
|
|
out = ""
|
|
|
|
maximalStackDepth = int(arg)
|
2013-05-15 15:42:55 +02:00
|
|
|
ns = qtNamespace()
|
2013-06-06 18:28:53 +02:00
|
|
|
while True:
|
|
|
|
maximalStackDepth -= 1
|
|
|
|
if maximalStackDepth < 0:
|
|
|
|
break
|
|
|
|
e = e.older()
|
|
|
|
if e == None or e.name() == None:
|
|
|
|
break
|
|
|
|
if e.name() == ns + "QThreadPrivate::start" \
|
|
|
|
or e.name() == "_ZN14QThreadPrivate5startEPv@4":
|
|
|
|
try:
|
|
|
|
thrptr = e.read_var("thr").dereference()
|
2013-10-16 18:39:01 +02:00
|
|
|
obtype = d.lookupType(ns + "QObjectPrivate").pointer()
|
2013-06-06 18:28:53 +02:00
|
|
|
d_ptr = thrptr["d_ptr"]["d"].cast(obtype).dereference()
|
|
|
|
try:
|
|
|
|
objectName = d_ptr["objectName"]
|
|
|
|
except: # Qt 5
|
|
|
|
p = d_ptr["extraData"]
|
2013-09-11 21:35:39 +02:00
|
|
|
if not d.isNull(p):
|
2013-06-06 18:28:53 +02:00
|
|
|
objectName = p.dereference()["objectName"]
|
|
|
|
if not objectName is None:
|
|
|
|
data, size, alloc = d.stringData(objectName)
|
|
|
|
if size > 0:
|
2013-09-11 21:35:39 +02:00
|
|
|
s = d.readMemory(data, 2 * size)
|
2013-06-06 18:28:53 +02:00
|
|
|
|
|
|
|
thread = gdb.selected_thread()
|
|
|
|
inner = '{valueencoded="';
|
|
|
|
inner += str(Hex4EncodedLittleEndianWithoutQuotes)+'",id="'
|
|
|
|
inner += str(thread.num) + '",value="'
|
|
|
|
inner += s
|
|
|
|
#inner += d.encodeString(objectName)
|
|
|
|
inner += '"},'
|
|
|
|
|
|
|
|
out += inner
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
return out
|
|
|
|
|
|
|
|
|
|
|
|
def threadnames(arg):
|
2013-05-15 15:42:55 +02:00
|
|
|
out = '['
|
|
|
|
oldthread = gdb.selected_thread()
|
|
|
|
try:
|
|
|
|
inferior = selectedInferior()
|
|
|
|
for thread in inferior.threads():
|
|
|
|
thread.switch()
|
2013-06-06 18:28:53 +02:00
|
|
|
out += threadname(arg)
|
2013-05-15 15:42:55 +02:00
|
|
|
except:
|
|
|
|
pass
|
|
|
|
oldthread.switch()
|
|
|
|
return out + ']'
|
|
|
|
|
|
|
|
registerCommand("threadnames", threadnames)
|
|
|
|
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
#
|
|
|
|
# Mixed C++/Qml debugging
|
|
|
|
#
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
def qmlb(args):
|
|
|
|
# executeCommand(command, to_string=True).split("\n")
|
|
|
|
warm("RUNNING: break -f QScript::FunctionWrapper::proxyCall")
|
|
|
|
output = catchCliOutput("rbreak -f QScript::FunctionWrapper::proxyCall")
|
|
|
|
warn("OUTPUT: %s " % output)
|
|
|
|
bp = output[0]
|
|
|
|
warn("BP: %s " % bp)
|
|
|
|
# BP: ['Breakpoint 3 at 0xf166e7: file .../qscriptfunction.cpp, line 75.\\n'] \n"
|
|
|
|
pos = bp.find(' ') + 1
|
|
|
|
warn("POS: %s " % pos)
|
|
|
|
nr = bp[bp.find(' ') + 1 : bp.find(' at ')]
|
|
|
|
warn("NR: %s " % nr)
|
|
|
|
return bp
|
|
|
|
|
|
|
|
registerCommand("qmlb", qmlb)
|
|
|
|
|
2013-04-11 18:11:54 +02:00
|
|
|
bbsetup()
|