1 # 1 # 2 # gdb helper commands and functions for Linux 2 # gdb helper commands and functions for Linux kernel debugging 3 # 3 # 4 # module tools 4 # module tools 5 # 5 # 6 # Copyright (c) Siemens AG, 2013 6 # Copyright (c) Siemens AG, 2013 7 # 7 # 8 # Authors: 8 # Authors: 9 # Jan Kiszka <jan.kiszka@siemens.com> 9 # Jan Kiszka <jan.kiszka@siemens.com> 10 # 10 # 11 # This work is licensed under the terms of the 11 # This work is licensed under the terms of the GNU GPL version 2. 12 # 12 # 13 13 14 import gdb 14 import gdb 15 15 16 from linux import cpus, utils, lists, constant !! 16 from linux import cpus, utils 17 17 18 18 19 module_type = utils.CachedType("struct module" 19 module_type = utils.CachedType("struct module") 20 20 21 21 22 def module_list(): 22 def module_list(): 23 global module_type 23 global module_type 24 modules = utils.gdb_eval_or_none("modules" << 25 if modules is None: << 26 return << 27 << 28 module_ptr_type = module_type.get_type().p 24 module_ptr_type = module_type.get_type().pointer() 29 !! 25 modules = gdb.parse_and_eval("modules") 30 for module in lists.list_for_each_entry(mo !! 26 entry = modules['next'] 31 yield module !! 27 end_of_list = modules.address >> 28 >> 29 while entry != end_of_list: >> 30 yield utils.container_of(entry, module_ptr_type, "list") >> 31 entry = entry['next'] 32 32 33 33 34 def find_module_by_name(name): 34 def find_module_by_name(name): 35 for module in module_list(): 35 for module in module_list(): 36 if module['name'].string() == name: 36 if module['name'].string() == name: 37 return module 37 return module 38 return None 38 return None 39 39 40 40 41 class LxModule(gdb.Function): 41 class LxModule(gdb.Function): 42 """Find module by name and return the modu 42 """Find module by name and return the module variable. 43 43 44 $lx_module("MODULE"): Given the name MODULE, i 44 $lx_module("MODULE"): Given the name MODULE, iterate over all loaded modules 45 of the target and return that module variable 45 of the target and return that module variable which MODULE matches.""" 46 46 47 def __init__(self): 47 def __init__(self): 48 super(LxModule, self).__init__("lx_mod 48 super(LxModule, self).__init__("lx_module") 49 49 50 def invoke(self, mod_name): 50 def invoke(self, mod_name): 51 mod_name = mod_name.string() 51 mod_name = mod_name.string() 52 module = find_module_by_name(mod_name) 52 module = find_module_by_name(mod_name) 53 if module: 53 if module: 54 return module.dereference() 54 return module.dereference() 55 else: 55 else: 56 raise gdb.GdbError("Unable to find 56 raise gdb.GdbError("Unable to find MODULE " + mod_name) 57 57 58 58 59 LxModule() 59 LxModule() 60 60 61 61 62 class LxLsmod(gdb.Command): 62 class LxLsmod(gdb.Command): 63 """List currently loaded modules.""" 63 """List currently loaded modules.""" 64 64 65 _module_use_type = utils.CachedType("struc 65 _module_use_type = utils.CachedType("struct module_use") 66 66 67 def __init__(self): 67 def __init__(self): 68 super(LxLsmod, self).__init__("lx-lsmo 68 super(LxLsmod, self).__init__("lx-lsmod", gdb.COMMAND_DATA) 69 69 70 def invoke(self, arg, from_tty): 70 def invoke(self, arg, from_tty): 71 gdb.write( 71 gdb.write( 72 "Address{0} Module 72 "Address{0} Module Size Used by\n".format( 73 " " if utils.get_long_t 73 " " if utils.get_long_type().sizeof == 8 else "")) 74 74 75 for module in module_list(): 75 for module in module_list(): 76 text = module['mem'][constants.LX_ << 77 text_addr = str(text['base']).spli << 78 total_size = 0 << 79 << 80 for i in range(constants.LX_MOD_TE << 81 total_size += module['mem'][i] << 82 << 83 gdb.write("{address} {name:<19} {s 76 gdb.write("{address} {name:<19} {size:>8} {ref}".format( 84 address=text_addr, !! 77 address=str(module['module_core']).split()[0], 85 name=module['name'].string(), 78 name=module['name'].string(), 86 size=str(total_size), !! 79 size=str(module['core_size']), 87 ref=str(module['refcnt']['coun !! 80 ref=str(module['refcnt']['counter']))) 88 81 >> 82 source_list = module['source_list'] 89 t = self._module_use_type.get_type 83 t = self._module_use_type.get_type().pointer() >> 84 entry = source_list['next'] 90 first = True 85 first = True 91 sources = module['source_list'] !! 86 while entry != source_list.address: 92 for use in lists.list_for_each_ent !! 87 use = utils.container_of(entry, t, "source_list") 93 gdb.write("{separator}{name}". 88 gdb.write("{separator}{name}".format( 94 separator=" " if first els 89 separator=" " if first else ",", 95 name=use['source']['name'] 90 name=use['source']['name'].string())) 96 first = False 91 first = False 97 !! 92 entry = entry['next'] 98 gdb.write("\n") 93 gdb.write("\n") 99 94 100 LxLsmod() << 101 << 102 def help(): << 103 t = """Usage: lx-getmod-by-textaddr [Hexim << 104 Example: lx-getmod-by-textaddr 0xffff80000 << 105 gdb.write("Unrecognized command\n") << 106 raise gdb.GdbError(t) << 107 << 108 class LxFindTextAddrinMod(gdb.Command): << 109 '''Look up loaded kernel module by text ad << 110 95 111 def __init__(self): !! 96 LxLsmod() 112 super(LxFindTextAddrinMod, self).__ini << 113 << 114 def invoke(self, arg, from_tty): << 115 args = gdb.string_to_argv(arg) << 116 << 117 if len(args) != 1: << 118 help() << 119 << 120 addr = gdb.Value(int(args[0], 16)).cas << 121 for mod in module_list(): << 122 mod_text_start = mod['mem'][consta << 123 mod_text_end = mod_text_start + mo << 124 << 125 if addr >= mod_text_start and addr << 126 s = "0x%x" % addr + " is in " << 127 gdb.write(s) << 128 return << 129 gdb.write("0x%x is not in any module t << 130 << 131 LxFindTextAddrinMod() <<
Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.