OLD | NEW |
1 # Copyright 2015 The Chromium Authors. All rights reserved. | 1 # Copyright 2015 The Chromium Authors. All rights reserved. |
2 # Use of this source code is governed by a BSD-style license that can be | 2 # Use of this source code is governed by a BSD-style license that can be |
3 # found in the LICENSE file. | 3 # found in the LICENSE file. |
4 | 4 |
5 """ | 5 """ |
6 Manages a debugging session with GDB. | 6 Manages a debugging session with GDB. |
7 | 7 |
8 This module is meant to be imported from inside GDB. Once loaded, the | 8 This module is meant to be imported from inside GDB. Once loaded, the |
9 |DebugSession| attaches GDB to a running Mojo Shell process on an Android | 9 |DebugSession| attaches GDB to a running Mojo Shell process on an Android |
10 device using a remote gdbserver. | 10 device using a remote gdbserver. |
(...skipping 53 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
64 _gdb_execute("info proc map").split('\n')] | 64 _gdb_execute("info proc map").split('\n')] |
65 if len(x) == 5 and x[4][0] == '/'] | 65 if len(x) == 5 and x[4][0] == '/'] |
66 res = {} | 66 res = {} |
67 for m in mappings: | 67 for m in mappings: |
68 libname = m.filename[m.filename.rfind('/') + 1:] | 68 libname = m.filename[m.filename.rfind('/') + 1:] |
69 res[libname] = res.get(libname, []) + [m] | 69 res[libname] = res.get(libname, []) + [m] |
70 return res.values() | 70 return res.values() |
71 | 71 |
72 | 72 |
73 class DebugSession(object): | 73 class DebugSession(object): |
74 def __init__(self, build_directory, package_name, pyelftools_dir, adb): | 74 def __init__(self, build_directory_list, package_name, pyelftools_dir, adb): |
75 self._build_directory = build_directory | 75 build_directories = build_directory_list.split(',') |
76 if not os.path.exists(self._build_directory): | 76 if len(build_directories) == 0 or not all(map(os.path.exists, |
77 logging.fatal("Please pass a valid build directory") | 77 build_directories)): |
| 78 logging.fatal("Please pass a list of valid build directories") |
78 sys.exit(1) | 79 sys.exit(1) |
79 self._package_name = package_name | 80 self._package_name = package_name |
80 self._adb = adb | 81 self._adb = adb |
81 self._remote_file_cache = os.path.join(os.getenv('HOME'), '.mojosymbols') | 82 self._remote_file_cache = os.path.join(os.getenv('HOME'), '.mojosymbols') |
82 | 83 |
83 if pyelftools_dir != None: | 84 if pyelftools_dir != None: |
84 sys.path.append(pyelftools_dir) | 85 sys.path.append(pyelftools_dir) |
85 try: | 86 try: |
86 import elftools.elf.elffile as elffile | 87 import elftools.elf.elffile as elffile |
87 except ImportError: | 88 except ImportError: |
88 logging.fatal("Unable to find elftools module; please install pyelftools " | 89 logging.fatal("Unable to find elftools module; please install pyelftools " |
89 "and specify its path on the command line using " | 90 "and specify its path on the command line using " |
90 "--pyelftools-dir.") | 91 "--pyelftools-dir.") |
91 sys.exit(1) | 92 sys.exit(1) |
92 | 93 |
93 self._elffile_module = elffile | 94 self._elffile_module = elffile |
94 | 95 |
95 self._libraries = self._find_libraries(build_directory) | 96 self._libraries = self._find_libraries(build_directories) |
96 self._rfc = RemoteFileConnection('localhost', 10000) | 97 self._rfc = RemoteFileConnection('localhost', 10000) |
97 self._remote_file_reader_process = None | 98 self._remote_file_reader_process = None |
98 if not os.path.exists(self._remote_file_cache): | 99 if not os.path.exists(self._remote_file_cache): |
99 os.makedirs(self._remote_file_cache) | 100 os.makedirs(self._remote_file_cache) |
100 self._done_mapping = set() | 101 self._done_mapping = set() |
101 self._downloaded_files = [] | 102 self._downloaded_files = [] |
102 | 103 |
103 def __del__(self): | 104 def __del__(self): |
104 # Note that, per python interpreter documentation, __del__ is not | 105 # Note that, per python interpreter documentation, __del__ is not |
105 # guaranteed to be called when the interpreter (GDB, in our case) quits. | 106 # guaranteed to be called when the interpreter (GDB, in our case) quits. |
106 # Also, most (all?) globals are no longer available at this time (launching | 107 # Also, most (all?) globals are no longer available at this time (launching |
107 # a subprocess does not work). | 108 # a subprocess does not work). |
108 self.stop() | 109 self.stop() |
109 | 110 |
110 def stop(self, _unused_return_value=None): | 111 def stop(self, _unused_return_value=None): |
111 if self._remote_file_reader_process != None: | 112 if self._remote_file_reader_process != None: |
112 self._remote_file_reader_process.kill() | 113 self._remote_file_reader_process.kill() |
113 | 114 |
114 def _find_libraries(self, lib_dir): | 115 def _find_libraries(self, lib_dirs): |
115 """Finds all libraries in |lib_dir| and key them by their signatures. | 116 """Finds all libraries in |lib_dirs| and key them by their signatures. |
116 """ | 117 """ |
117 res = {} | 118 res = {} |
118 for fn in glob.glob('%s/*.so' % lib_dir): | 119 for lib_dir in lib_dirs: |
119 with open(fn, 'r') as f: | 120 for fn in glob.glob('%s/*.so' % lib_dir): |
120 s = get_signature(f, self._elffile_module) | 121 with open(fn, 'r') as f: |
121 if s is not None: | 122 s = get_signature(f, self._elffile_module) |
122 res[s] = fn | 123 if s is not None: |
| 124 res[s] = fn |
123 return res | 125 return res |
124 | 126 |
125 def _associate_symbols(self, mapping, local_file): | 127 def _associate_symbols(self, mapping, local_file): |
126 with open(local_file, "r") as f: | 128 with open(local_file, "r") as f: |
127 elf = self._elffile_module.ELFFile(f) | 129 elf = self._elffile_module.ELFFile(f) |
128 s = elf.get_section_by_name(".text") | 130 s = elf.get_section_by_name(".text") |
129 text_address = mapping[0].start + s['sh_offset'] | 131 text_address = mapping[0].start + s['sh_offset'] |
130 _gdb_execute("add-symbol-file %s 0x%x" % (local_file, text_address)) | 132 _gdb_execute("add-symbol-file %s 0x%x" % (local_file, text_address)) |
131 | 133 |
132 def _download_file(self, signature, remote): | 134 def _download_file(self, signature, remote): |
(...skipping 175 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
308 | 310 |
309 def complete(self, text, _unused_word): | 311 def complete(self, text, _unused_word): |
310 if text == self._UPDATE_COMMAND: | 312 if text == self._UPDATE_COMMAND: |
311 return ('all', 'current') | 313 return ('all', 'current') |
312 elif text in self._UPDATE_COMMAND + ' all': | 314 elif text in self._UPDATE_COMMAND + ' all': |
313 return ['all'] | 315 return ['all'] |
314 elif text in self._UPDATE_COMMAND + ' current': | 316 elif text in self._UPDATE_COMMAND + ' current': |
315 return ['current'] | 317 return ['current'] |
316 else: | 318 else: |
317 return [] | 319 return [] |
OLD | NEW |