 8037fa55ac
			
		
	
	
		8037fa55ac
		
	
	
	
	
		
			
			Since QEMU has been able to build with native Int128 support this was broken as it attempts to fish values out of the non-existent structure. Also the alias print was trying to make a %x out of gdb.ValueType directly which didn't seem to work. Signed-off-by: Alex Bennée <alex.bennee@linaro.org>
		
			
				
	
	
		
			91 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			91 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/python
 | |
| 
 | |
| # GDB debugging support
 | |
| #
 | |
| # Copyright 2012 Red Hat, Inc. and/or its affiliates
 | |
| #
 | |
| # Authors:
 | |
| #  Avi Kivity <avi@redhat.com>
 | |
| #
 | |
| # This work is licensed under the terms of the GNU GPL, version 2.  See
 | |
| # the COPYING file in the top-level directory.
 | |
| #
 | |
| # Contributions after 2012-01-13 are licensed under the terms of the
 | |
| # GNU GPL, version 2 or (at your option) any later version.
 | |
| 
 | |
| # 'qemu mtree' -- display the memory hierarchy
 | |
| 
 | |
| import gdb
 | |
| 
 | |
| def isnull(ptr):
 | |
|     return ptr == gdb.Value(0).cast(ptr.type)
 | |
| 
 | |
| def int128(p):
 | |
|     '''Read an Int128 type to a python integer.
 | |
| 
 | |
|     QEMU can be built with native Int128 support so we need to detect
 | |
|     if the value is a structure or the native type.
 | |
|     '''
 | |
|     if p.type.code == gdb.TYPE_CODE_STRUCT:
 | |
|         return int(p['lo']) + (int(p['hi']) << 64)
 | |
|     else:
 | |
|         return int(("%s" % p), 16)
 | |
| 
 | |
| class MtreeCommand(gdb.Command):
 | |
|     '''Display the memory tree hierarchy'''
 | |
|     def __init__(self):
 | |
|         gdb.Command.__init__(self, 'qemu mtree', gdb.COMMAND_DATA,
 | |
|                              gdb.COMPLETE_NONE)
 | |
|         self.queue = []
 | |
|     def invoke(self, arg, from_tty):
 | |
|         self.seen = set()
 | |
|         self.queue_root('address_space_memory')
 | |
|         self.queue_root('address_space_io')
 | |
|         self.process_queue()
 | |
|     def queue_root(self, varname):
 | |
|         ptr = gdb.parse_and_eval(varname)['root']
 | |
|         self.queue.append(ptr)
 | |
|     def process_queue(self):
 | |
|         while self.queue:
 | |
|             ptr = self.queue.pop(0)
 | |
|             if int(ptr) in self.seen:
 | |
|                 continue
 | |
|             self.print_item(ptr)
 | |
|     def print_item(self, ptr, offset = gdb.Value(0), level = 0):
 | |
|         self.seen.add(int(ptr))
 | |
|         addr = ptr['addr']
 | |
|         addr += offset
 | |
|         size = int128(ptr['size'])
 | |
|         alias = ptr['alias']
 | |
|         klass = ''
 | |
|         if not isnull(alias):
 | |
|             klass = ' (alias)'
 | |
|         elif not isnull(ptr['ops']):
 | |
|             klass = ' (I/O)'
 | |
|         elif bool(ptr['ram']):
 | |
|             klass = ' (RAM)'
 | |
|         gdb.write('%s%016x-%016x %s%s (@ %s)\n'
 | |
|                   % ('  ' * level,
 | |
|                      int(addr),
 | |
|                      int(addr + (size - 1)),
 | |
|                      ptr['name'].string(),
 | |
|                      klass,
 | |
|                      ptr,
 | |
|                      ),
 | |
|                   gdb.STDOUT)
 | |
|         if not isnull(alias):
 | |
|             gdb.write('%s    alias: %s@%016x (@ %s)\n' %
 | |
|                       ('  ' * level,
 | |
|                        alias['name'].string(),
 | |
|                        int(ptr['alias_offset']),
 | |
|                        alias,
 | |
|                        ),
 | |
|                       gdb.STDOUT)
 | |
|             self.queue.append(alias)
 | |
|         subregion = ptr['subregions']['tqh_first']
 | |
|         level += 1
 | |
|         while not isnull(subregion):
 | |
|             self.print_item(subregion, addr, level)
 | |
|             subregion = subregion['subregions_link']['tqe_next']
 | |
| 
 |