2009-01-04 16:15:19 -05:00
|
|
|
###############################################################################
|
|
|
|
# Name: misc/gdb/print.py
|
|
|
|
# Purpose: pretty-printers for wx data structures: this file is meant to
|
|
|
|
# be sourced from gdb using "source -p" (or, better, autoloaded
|
|
|
|
# in the future...)
|
|
|
|
# Author: Vadim Zeitlin
|
|
|
|
# Created: 2009-01-04
|
|
|
|
# Copyright: (c) 2009 Vadim Zeitlin
|
2010-07-13 09:29:13 -04:00
|
|
|
# Licence: wxWindows licence
|
2009-01-04 16:15:19 -05:00
|
|
|
###############################################################################
|
|
|
|
|
2010-02-26 09:09:43 -05:00
|
|
|
# Define wxFooPrinter class implementing (at least) to_string() method for each
|
|
|
|
# wxFoo class we want to pretty print. Then just add wxFoo to the types array
|
|
|
|
# in wxLookupFunction at the bottom of this file.
|
|
|
|
|
2012-03-24 20:31:38 -04:00
|
|
|
import datetime
|
2018-08-15 01:30:27 -04:00
|
|
|
import gdb
|
|
|
|
import itertools
|
|
|
|
import sys
|
|
|
|
|
|
|
|
if sys.version_info[0] > 2:
|
|
|
|
# Python 3
|
|
|
|
Iterator = object
|
|
|
|
|
|
|
|
long = int
|
|
|
|
else:
|
|
|
|
# Python 2, we need to make an adaptor, so we can use Python 3 iterator implementations.
|
|
|
|
class Iterator:
|
|
|
|
def next(self):
|
|
|
|
return self.__next__()
|
2010-02-26 09:09:43 -05:00
|
|
|
|
2009-01-04 16:15:19 -05:00
|
|
|
# shamelessly stolen from std::string example
|
|
|
|
class wxStringPrinter:
|
|
|
|
def __init__(self, val):
|
|
|
|
self.val = val
|
|
|
|
|
|
|
|
def to_string(self):
|
2012-05-13 17:37:13 -04:00
|
|
|
return self.val['m_impl']['_M_dataplus']['_M_p']
|
2009-01-04 16:15:19 -05:00
|
|
|
|
2009-11-11 18:08:41 -05:00
|
|
|
def display_hint(self):
|
|
|
|
return 'string'
|
|
|
|
|
2018-08-15 01:30:27 -04:00
|
|
|
class wxArrayStringPrinter:
|
|
|
|
|
|
|
|
class _iterator(Iterator):
|
|
|
|
def __init__ (self, firstItem, count):
|
|
|
|
self.item = firstItem
|
|
|
|
self.count = count
|
|
|
|
self.current = 0
|
|
|
|
|
|
|
|
def __iter__(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def __next__(self):
|
|
|
|
current = self.current
|
|
|
|
self.current = self.current + 1
|
|
|
|
|
|
|
|
if current == self.count:
|
|
|
|
raise StopIteration
|
|
|
|
elt = self.item.dereference()
|
|
|
|
self.item = self.item + 1
|
|
|
|
return ('[%d]' % current, elt)
|
|
|
|
|
|
|
|
def __init__(self, val):
|
|
|
|
self.val = val
|
|
|
|
|
|
|
|
def children(self):
|
|
|
|
return self._iterator(self.val['m_pItems'], self.val['m_nCount'])
|
|
|
|
|
|
|
|
def to_string(self):
|
|
|
|
count = self.val['m_nCount']
|
|
|
|
capacity = self.val['m_nSize']
|
|
|
|
return ('length %d, capacity %d' % (int (count), int (capacity)))
|
|
|
|
|
|
|
|
def display_hint(self):
|
|
|
|
return 'array'
|
|
|
|
|
2012-03-24 20:31:38 -04:00
|
|
|
class wxDateTimePrinter:
|
|
|
|
def __init__(self, val):
|
|
|
|
self.val = val
|
|
|
|
|
|
|
|
def to_string(self):
|
|
|
|
# A value of type wxLongLong can't be used in Python arithmetic
|
|
|
|
# expressions directly so we need to convert it to long long first and
|
|
|
|
# then cast to int explicitly to be able to use it as a timestamp.
|
|
|
|
msec = self.val['m_time'].cast(gdb.lookup_type('long long'))
|
2012-04-28 18:24:57 -04:00
|
|
|
if msec == 0x8000000000000000:
|
|
|
|
return 'NONE'
|
2012-03-24 20:31:38 -04:00
|
|
|
sec = int(msec / 1000)
|
|
|
|
return datetime.datetime.fromtimestamp(sec).isoformat(' ')
|
|
|
|
|
2012-04-28 18:24:52 -04:00
|
|
|
class wxFileNamePrinter:
|
|
|
|
def __init__(self, val):
|
|
|
|
self.val = val
|
|
|
|
|
|
|
|
def to_string(self):
|
|
|
|
# It is simpler to just call the internal function here than to iterate
|
|
|
|
# over m_dirs array ourselves. The disadvantage of this approach is
|
|
|
|
# that it requires a live inferior process and so doesn't work when
|
|
|
|
# debugging using only a core file. If this ever becomes a serious
|
|
|
|
# problem, this should be rewritten to use m_dirs and m_name and m_ext.
|
|
|
|
return gdb.parse_and_eval('((wxFileName*)%s)->GetFullPath(0)' %
|
|
|
|
self.val.address)
|
|
|
|
|
2010-02-26 09:09:43 -05:00
|
|
|
class wxXYPrinterBase:
|
|
|
|
def __init__(self, val):
|
|
|
|
self.x = val['x']
|
|
|
|
self.y = val['y']
|
|
|
|
|
|
|
|
class wxPointPrinter(wxXYPrinterBase):
|
|
|
|
def to_string(self):
|
|
|
|
return '(%d, %d)' % (self.x, self.y)
|
|
|
|
|
|
|
|
class wxSizePrinter(wxXYPrinterBase):
|
|
|
|
def to_string(self):
|
|
|
|
return '%d*%d' % (self.x, self.y)
|
|
|
|
|
|
|
|
class wxRectPrinter(wxXYPrinterBase):
|
|
|
|
def __init__(self, val):
|
|
|
|
wxXYPrinterBase.__init__(self, val)
|
|
|
|
self.width = val['width']
|
|
|
|
self.height = val['height']
|
|
|
|
|
|
|
|
def to_string(self):
|
|
|
|
return '(%d, %d) %d*%d' % (self.x, self.y, self.width, self.height)
|
|
|
|
|
|
|
|
|
|
|
|
# The function looking up the pretty-printer to use for the given value.
|
2009-11-11 18:08:41 -05:00
|
|
|
def wxLookupFunction(val):
|
2010-02-26 09:09:43 -05:00
|
|
|
# Using a list is probably ok for so few items but consider switching to a
|
|
|
|
# set (or a dict and cache class types as the keys in it?) if needed later.
|
2012-04-28 18:24:52 -04:00
|
|
|
types = ['wxString',
|
2018-08-15 01:30:27 -04:00
|
|
|
'wxArrayString',
|
2012-04-28 18:24:52 -04:00
|
|
|
'wxDateTime',
|
|
|
|
'wxFileName',
|
|
|
|
'wxPoint',
|
|
|
|
'wxSize',
|
|
|
|
'wxRect']
|
2010-02-26 09:09:43 -05:00
|
|
|
|
|
|
|
for t in types:
|
|
|
|
if val.type.tag == t:
|
|
|
|
# Not sure if this is the best name to create the object of a class
|
|
|
|
# by name but at least it beats eval()
|
|
|
|
return globals()[t + 'Printer'](val)
|
|
|
|
|
2009-11-11 18:08:41 -05:00
|
|
|
return None
|
|
|
|
|
|
|
|
gdb.pretty_printers.append(wxLookupFunction)
|