blob: e8ccc5f902302880535a73ab2d19f3d169c32db1 [file] [log] [blame]
Greg Claytonc7697222012-08-31 01:11:17 +00001#!/usr/bin/python
2
3#----------------------------------------------------------------------
4# Be sure to add the python path that points to the LLDB shared library.
5# On MacOSX csh, tcsh:
6# setenv PYTHONPATH /Applications/Xcode.app/Contents/SharedFrameworks/LLDB.framework/Resources/Python
7# On MacOSX sh, bash:
8# export PYTHONPATH=/Applications/Xcode.app/Contents/SharedFrameworks/LLDB.framework/Resources/Python
9#----------------------------------------------------------------------
10
Greg Clayton97e44a02012-08-31 02:55:56 +000011import commands
Greg Claytonc7697222012-08-31 01:11:17 +000012import optparse
13import os
Greg Clayton97e44a02012-08-31 02:55:56 +000014import platform
Greg Claytonc7697222012-08-31 01:11:17 +000015import sys
16
Greg Clayton97e44a02012-08-31 02:55:56 +000017#----------------------------------------------------------------------
18# Code that auto imports LLDB
19#----------------------------------------------------------------------
20try:
21 # Just try for LLDB in case PYTHONPATH is already correctly setup
22 import lldb
23except ImportError:
24 lldb_python_dirs = list()
25 # lldb is not in the PYTHONPATH, try some defaults for the current platform
26 platform_system = platform.system()
27 if platform_system == 'Darwin':
28 # On Darwin, try the currently selected Xcode directory
29 xcode_dir = commands.getoutput("xcode-select --print-path")
30 if xcode_dir:
31 lldb_python_dirs.append(os.path.realpath(xcode_dir + '/../SharedFrameworks/LLDB.framework/Resources/Python'))
32 lldb_python_dirs.append(xcode_dir + '/Library/PrivateFrameworks/LLDB.framework/Resources/Python')
33 lldb_python_dirs.append('/System/Library/PrivateFrameworks/LLDB.framework/Resources/Python')
34 success = False
35 for lldb_python_dir in lldb_python_dirs:
36 if os.path.exists(lldb_python_dir):
37 if not (sys.path.__contains__(lldb_python_dir)):
38 sys.path.append(lldb_python_dir)
39 try:
40 import lldb
41 except ImportError:
42 pass
43 else:
44 print 'imported lldb from: "%s"' % (lldb_python_dir)
45 success = True
46 break
47 if not success:
48 print "error: couldn't locate the 'lldb' module, please set PYTHONPATH correctly"
49 sys.exit(1)
50
Greg Claytonc7697222012-08-31 01:11:17 +000051def print_threads(process, options):
52 if options.show_threads:
53 for thread in process:
54 print '%s %s' % (thread, thread.GetFrameAtIndex(0))
55
56def run_commands(command_interpreter, commands):
57 return_obj = lldb.SBCommandReturnObject()
58 for command in commands:
59 command_interpreter.HandleCommand( command, return_obj )
60 if return_obj.Succeeded():
61 print return_obj.GetOutput()
62 else:
63 print return_obj
64 if options.stop_on_error:
65 break
66
67def main(argv):
68 description='''Debugs a program using the LLDB python API and uses asynchronous broadcast events to watch for process state changes.'''
Greg Clayton97e44a02012-08-31 02:55:56 +000069 epilog='''Examples:
70
71#----------------------------------------------------------------------
72# Run "/bin/ls" with the arguments "-lAF /tmp/", and set a breakpoint
73# at "malloc" and backtrace and read all registers each time we stop
74#----------------------------------------------------------------------
75% ./process_events.py --breakpoint malloc --stop-command bt --stop-command 'register read' -- /bin/ls -lAF /tmp/
76
77'''
78 optparse.OptionParser.format_epilog = lambda self, formatter: self.epilog
79 parser = optparse.OptionParser(description=description, prog='process_events',usage='usage: process_events [options] program [arg1 arg2]', epilog=epilog)
Greg Claytonc7697222012-08-31 01:11:17 +000080 parser.add_option('-v', '--verbose', action='store_true', dest='verbose', help="Enable verbose logging.", default=False)
Greg Clayton97e44a02012-08-31 02:55:56 +000081 parser.add_option('-b', '--breakpoint', action='append', type='string', metavar='BPEXPR', dest='breakpoints', help='Breakpoint commands to create after the target has been created, the values will be sent to the "_regexp-break" command which supports breakpoints by name, file:line, and address.')
82 parser.add_option('-a', '--arch', type='string', dest='arch', help='The architecture to use when creating the debug target.', default=None)
Greg Clayton111db4f2013-06-26 22:23:45 +000083 parser.add_option('--platform', type='string', metavar='platform', dest='platform', help='Specify the platform to use when creating the debug target. Valid values include "localhost", "darwin-kernel", "ios-simulator", "remote-freebsd", "remote-macosx", "remote-ios", "remote-linux".', default=None)
Greg Clayton97e44a02012-08-31 02:55:56 +000084 parser.add_option('-l', '--launch-command', action='append', type='string', metavar='CMD', dest='launch_commands', help='LLDB command interpreter commands to run once after the process has launched. This option can be specified more than once.', default=[])
85 parser.add_option('-s', '--stop-command', action='append', type='string', metavar='CMD', dest='stop_commands', help='LLDB command interpreter commands to run each time the process stops. This option can be specified more than once.', default=[])
86 parser.add_option('-c', '--crash-command', action='append', type='string', metavar='CMD', dest='crash_commands', help='LLDB command interpreter commands to run in case the process crashes. This option can be specified more than once.', default=[])
87 parser.add_option('-x', '--exit-command', action='append', type='string', metavar='CMD', dest='exit_commands', help='LLDB command interpreter commands to run once after the process has exited. This option can be specified more than once.', default=[])
Greg Claytonc7697222012-08-31 01:11:17 +000088 parser.add_option('-T', '--no-threads', action='store_false', dest='show_threads', help="Don't show threads when process stops.", default=True)
Greg Clayton984fee52012-09-25 18:27:12 +000089 parser.add_option('--ignore-errors', action='store_false', dest='stop_on_error', help="Don't stop executing LLDB commands if the command returns an error. This applies to all of the LLDB command interpreter commands that get run for launch, stop, crash and exit.", default=True)
Greg Clayton97e44a02012-08-31 02:55:56 +000090 parser.add_option('-n', '--run-count', type='int', dest='run_count', metavar='N', help='How many times to run the process in case the process exits.', default=1)
Greg Clayton984fee52012-09-25 18:27:12 +000091 parser.add_option('-t', '--event-timeout', type='int', dest='event_timeout', metavar='SEC', help='Specify the timeout in seconds to wait for process state change events.', default=lldb.UINT32_MAX)
92 parser.add_option('-e', '--environment', action='append', type='string', metavar='ENV', dest='env_vars', help='Environment variables to set in the inferior process when launching a process.')
93 parser.add_option('-d', '--working-dir', type='string', metavar='DIR', dest='working_dir', help='The the current working directory when launching a process.', default=None)
94 parser.add_option('-p', '--attach-pid', type='int', dest='attach_pid', metavar='PID', help='Specify a process to attach to by process ID.', default=-1)
95 parser.add_option('-P', '--attach-name', type='string', dest='attach_name', metavar='PROCESSNAME', help='Specify a process to attach to by name.', default=None)
96 parser.add_option('-w', '--attach-wait', action='store_true', dest='attach_wait', help='Wait for the next process to launch when attaching to a process by name.', default=False)
Greg Claytonc7697222012-08-31 01:11:17 +000097 try:
98 (options, args) = parser.parse_args(argv)
99 except:
100 return
Greg Clayton984fee52012-09-25 18:27:12 +0000101
102 attach_info = None
103 launch_info = None
104 exe = None
105 if args:
106 exe = args.pop(0)
107 launch_info = lldb.SBLaunchInfo (args)
108 if options.env_vars:
109 launch_info.SetEnvironmentEntries(options.env_vars, True)
110 if options.working_dir:
111 launch_info.SetWorkingDirectory(options.working_dir)
112 elif options.attach_pid != -1:
113 if options.run_count == 1:
114 attach_info = lldb.SBAttachInfo (options.attach_pid)
115 else:
116 print "error: --run-count can't be used with the --attach-pid option"
117 sys.exit(1)
118 elif not options.attach_name is None:
119 if options.run_count == 1:
120 attach_info = lldb.SBAttachInfo (options.attach_name, options.attach_wait)
121 else:
122 print "error: --run-count can't be used with the --attach-name option"
123 sys.exit(1)
124 else:
Greg Claytonc7697222012-08-31 01:11:17 +0000125 print 'error: a program path for a program to debug and its arguments are required'
126 sys.exit(1)
Greg Clayton984fee52012-09-25 18:27:12 +0000127
Greg Claytonc7697222012-08-31 01:11:17 +0000128
Greg Clayton97e44a02012-08-31 02:55:56 +0000129
Greg Claytonc7697222012-08-31 01:11:17 +0000130 # Create a new debugger instance
131 debugger = lldb.SBDebugger.Create()
Greg Clayton111db4f2013-06-26 22:23:45 +0000132 debugger.SetAsync (True)
Greg Claytonc7697222012-08-31 01:11:17 +0000133 command_interpreter = debugger.GetCommandInterpreter()
Greg Claytonc7697222012-08-31 01:11:17 +0000134 # Create a target from a file and arch
Greg Claytonc7697222012-08-31 01:11:17 +0000135
Greg Clayton984fee52012-09-25 18:27:12 +0000136 if exe:
137 print "Creating a target for '%s'" % exe
Greg Clayton111db4f2013-06-26 22:23:45 +0000138 error = lldb.SBError()
139 target = debugger.CreateTarget (exe, options.arch, options.platform, True, error)
Greg Claytonc7697222012-08-31 01:11:17 +0000140
141 if target:
142
Greg Clayton111db4f2013-06-26 22:23:45 +0000143 # Set any breakpoints that were specified in the args if we are launching. We use the
144 # command line command to take advantage of the shorthand breakpoint creation
Greg Clayton984fee52012-09-25 18:27:12 +0000145 if launch_info and options.breakpoints:
Greg Clayton97e44a02012-08-31 02:55:56 +0000146 for bp in options.breakpoints:
147 debugger.HandleCommand( "_regexp-break %s" % (bp))
148 run_commands(command_interpreter, ['breakpoint list'])
Greg Claytonc7697222012-08-31 01:11:17 +0000149
150 for run_idx in range(options.run_count):
151 # Launch the process. Since we specified synchronous mode, we won't return
152 # from this function until we hit the breakpoint at main
Greg Clayton984fee52012-09-25 18:27:12 +0000153 error = lldb.SBError()
Greg Claytonc7697222012-08-31 01:11:17 +0000154
Greg Clayton984fee52012-09-25 18:27:12 +0000155 if launch_info:
156 if options.run_count == 1:
157 print 'Launching "%s"...' % (exe)
158 else:
159 print 'Launching "%s"... (launch %u of %u)' % (exe, run_idx + 1, options.run_count)
160
161 process = target.Launch (launch_info, error)
162 else:
163 if options.attach_pid != -1:
164 print 'Attaching to process %i...' % (options.attach_pid)
165 else:
166 if options.attach_wait:
167 print 'Waiting for next to process named "%s" to launch...' % (options.attach_name)
168 else:
169 print 'Attaching to existing process named "%s"...' % (options.attach_name)
170 process = target.Attach (attach_info, error)
Greg Claytonc7697222012-08-31 01:11:17 +0000171
172 # Make sure the launch went ok
Greg Clayton984fee52012-09-25 18:27:12 +0000173 if process and process.GetProcessID() != lldb.LLDB_INVALID_PROCESS_ID:
Greg Clayton4d48e5b2015-08-12 20:04:01 +0000174
Greg Claytonc7697222012-08-31 01:11:17 +0000175 pid = process.GetProcessID()
Greg Clayton4d48e5b2015-08-12 20:04:01 +0000176 print 'Process is %i' % (pid)
177 if attach_info:
178 # continue process if we attached as we won't get an initial event
179 process.Continue()
180
Greg Clayton111db4f2013-06-26 22:23:45 +0000181 listener = debugger.GetListener()
Greg Claytonc7697222012-08-31 01:11:17 +0000182 # sign up for process state change events
Greg Claytonc7697222012-08-31 01:11:17 +0000183 stop_idx = 0
184 done = False
185 while not done:
186 event = lldb.SBEvent()
187 if listener.WaitForEvent (options.event_timeout, event):
Greg Clayton111db4f2013-06-26 22:23:45 +0000188 if lldb.SBProcess.EventIsProcessEvent(event):
189 state = lldb.SBProcess.GetStateFromEvent (event)
Greg Clayton96eb9ab2013-06-27 18:08:32 +0000190 if state == lldb.eStateInvalid:
191 # Not a state event
192 print 'process event = %s' % (event)
193 else:
194 print "process state changed event: %s" % (lldb.SBDebugger.StateAsCString(state))
195 if state == lldb.eStateStopped:
196 if stop_idx == 0:
197 if launch_info:
198 print "process %u launched" % (pid)
Greg Clayton111db4f2013-06-26 22:23:45 +0000199 run_commands(command_interpreter, ['breakpoint list'])
Greg Clayton96eb9ab2013-06-27 18:08:32 +0000200 else:
201 print "attached to process %u" % (pid)
202 for m in target.modules:
203 print m
204 if options.breakpoints:
205 for bp in options.breakpoints:
206 debugger.HandleCommand( "_regexp-break %s" % (bp))
207 run_commands(command_interpreter, ['breakpoint list'])
208 run_commands (command_interpreter, options.launch_commands)
209 else:
210 if options.verbose:
211 print "process %u stopped" % (pid)
212 run_commands (command_interpreter, options.stop_commands)
213 stop_idx += 1
214 print_threads (process, options)
215 print "continuing process %u" % (pid)
216 process.Continue()
217 elif state == lldb.eStateExited:
218 exit_desc = process.GetExitDescription()
219 if exit_desc:
220 print "process %u exited with status %u: %s" % (pid, process.GetExitStatus (), exit_desc)
221 else:
222 print "process %u exited with status %u" % (pid, process.GetExitStatus ())
223 run_commands (command_interpreter, options.exit_commands)
224 done = True
225 elif state == lldb.eStateCrashed:
226 print "process %u crashed" % (pid)
227 print_threads (process, options)
228 run_commands (command_interpreter, options.crash_commands)
229 done = True
230 elif state == lldb.eStateDetached:
231 print "process %u detached" % (pid)
232 done = True
233 elif state == lldb.eStateRunning:
234 # process is running, don't say anything, we will always get one of these after resuming
Greg Clayton111db4f2013-06-26 22:23:45 +0000235 if options.verbose:
Greg Clayton96eb9ab2013-06-27 18:08:32 +0000236 print "process %u resumed" % (pid)
237 elif state == lldb.eStateUnloaded:
238 print "process %u unloaded, this shouldn't happen" % (pid)
239 done = True
240 elif state == lldb.eStateConnected:
241 print "process connected"
242 elif state == lldb.eStateAttaching:
243 print "process attaching"
244 elif state == lldb.eStateLaunching:
245 print "process launching"
Greg Clayton111db4f2013-06-26 22:23:45 +0000246 else:
Greg Clayton96eb9ab2013-06-27 18:08:32 +0000247 print 'event = %s' % (event)
248 else:
249 # timeout waiting for an event
250 print "no process event for %u seconds, killing the process..." % (options.event_timeout)
251 done = True
252 # Now that we are done dump the stdout and stderr
253 process_stdout = process.GetSTDOUT(1024)
254 if process_stdout:
255 print "Process STDOUT:\n%s" % (process_stdout)
256 while process_stdout:
257 process_stdout = process.GetSTDOUT(1024)
258 print process_stdout
259 process_stderr = process.GetSTDERR(1024)
260 if process_stderr:
261 print "Process STDERR:\n%s" % (process_stderr)
262 while process_stderr:
263 process_stderr = process.GetSTDERR(1024)
264 print process_stderr
Greg Claytonc7697222012-08-31 01:11:17 +0000265 process.Kill() # kill the process
Greg Clayton984fee52012-09-25 18:27:12 +0000266 else:
267 if error:
268 print error
269 else:
270 if launch_info:
271 print 'error: launch failed'
272 else:
273 print 'error: attach failed'
Greg Claytonc7697222012-08-31 01:11:17 +0000274
275 lldb.SBDebugger.Terminate()
276
277if __name__ == '__main__':
278 main(sys.argv[1:])