149 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| ###############################################################################
 | |
| # 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
 | |
| # Licence:      wxWindows licence
 | |
| ###############################################################################
 | |
| 
 | |
| # 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.
 | |
| 
 | |
| import datetime
 | |
| 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__()
 | |
| 
 | |
| # shamelessly stolen from std::string example
 | |
| class wxStringPrinter:
 | |
|     def __init__(self, val):
 | |
|         self.val = val
 | |
| 
 | |
|     def to_string(self):
 | |
|         return self.val['m_impl']['_M_dataplus']['_M_p']
 | |
| 
 | |
|     def display_hint(self):
 | |
|         return 'string'
 | |
| 
 | |
| 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'
 | |
| 
 | |
| 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'))
 | |
|         if msec == 0x8000000000000000:
 | |
|             return 'NONE'
 | |
|         sec = int(msec / 1000)
 | |
|         return datetime.datetime.fromtimestamp(sec).isoformat(' ')
 | |
| 
 | |
| 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)
 | |
| 
 | |
| 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.
 | |
| def wxLookupFunction(val):
 | |
|     # 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.
 | |
|     types = ['wxString',
 | |
|              'wxArrayString',
 | |
|              'wxDateTime',
 | |
|              'wxFileName',
 | |
|              'wxPoint',
 | |
|              'wxSize',
 | |
|              'wxRect']
 | |
| 
 | |
|     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)
 | |
| 
 | |
|     return None
 | |
| 
 | |
| gdb.pretty_printers.append(wxLookupFunction)
 |