|  | #!/usr/bin/env python | 
|  |  | 
|  | # ---------------------------------------------------------------------- | 
|  | # Be sure to add the python path that points to the LLDB shared library. | 
|  | # On MacOSX csh, tcsh: | 
|  | #   setenv PYTHONPATH /Applications/Xcode.app/Contents/SharedFrameworks/LLDB.framework/Resources/Python | 
|  | # On MacOSX sh, bash: | 
|  | #   export PYTHONPATH=/Applications/Xcode.app/Contents/SharedFrameworks/LLDB.framework/Resources/Python | 
|  | # ---------------------------------------------------------------------- | 
|  |  | 
|  | import optparse | 
|  | import os | 
|  | import platform | 
|  | import sys | 
|  | import subprocess | 
|  |  | 
|  | # ---------------------------------------------------------------------- | 
|  | # Code that auto imports LLDB | 
|  | # ---------------------------------------------------------------------- | 
|  | try: | 
|  | # Just try for LLDB in case PYTHONPATH is already correctly setup | 
|  | import lldb | 
|  | except ImportError: | 
|  | lldb_python_dirs = list() | 
|  | # lldb is not in the PYTHONPATH, try some defaults for the current platform | 
|  | platform_system = platform.system() | 
|  | if platform_system == "Darwin": | 
|  | # On Darwin, try the currently selected Xcode directory | 
|  | xcode_dir = subprocess.check_output("xcode-select --print-path", shell=True) | 
|  | if xcode_dir: | 
|  | lldb_python_dirs.append( | 
|  | os.path.realpath( | 
|  | xcode_dir + "/../SharedFrameworks/LLDB.framework/Resources/Python" | 
|  | ) | 
|  | ) | 
|  | lldb_python_dirs.append( | 
|  | xcode_dir + "/Library/PrivateFrameworks/LLDB.framework/Resources/Python" | 
|  | ) | 
|  | lldb_python_dirs.append( | 
|  | "/System/Library/PrivateFrameworks/LLDB.framework/Resources/Python" | 
|  | ) | 
|  | success = False | 
|  | for lldb_python_dir in lldb_python_dirs: | 
|  | if os.path.exists(lldb_python_dir): | 
|  | if not (sys.path.__contains__(lldb_python_dir)): | 
|  | sys.path.append(lldb_python_dir) | 
|  | try: | 
|  | import lldb | 
|  | except ImportError: | 
|  | pass | 
|  | else: | 
|  | print('imported lldb from: "%s"' % (lldb_python_dir)) | 
|  | success = True | 
|  | break | 
|  | if not success: | 
|  | print( | 
|  | "error: couldn't locate the 'lldb' module, please set PYTHONPATH correctly" | 
|  | ) | 
|  | sys.exit(1) | 
|  |  | 
|  |  | 
|  | def print_threads(process, options): | 
|  | if options.show_threads: | 
|  | for thread in process: | 
|  | print("%s %s" % (thread, thread.GetFrameAtIndex(0))) | 
|  |  | 
|  |  | 
|  | def run_commands(command_interpreter, commands): | 
|  | return_obj = lldb.SBCommandReturnObject() | 
|  | for command in commands: | 
|  | command_interpreter.HandleCommand(command, return_obj) | 
|  | if return_obj.Succeeded(): | 
|  | print(return_obj.GetOutput()) | 
|  | else: | 
|  | print(return_obj) | 
|  | if options.stop_on_error: | 
|  | break | 
|  |  | 
|  |  | 
|  | def main(argv): | 
|  | description = """Debugs a program using the LLDB python API and uses asynchronous broadcast events to watch for process state changes.""" | 
|  | epilog = """Examples: | 
|  |  | 
|  | #---------------------------------------------------------------------- | 
|  | # Run "/bin/ls" with the arguments "-lAF /tmp/", and set a breakpoint | 
|  | # at "malloc" and backtrace and read all registers each time we stop | 
|  | #---------------------------------------------------------------------- | 
|  | % ./process_events.py --breakpoint malloc --stop-command bt --stop-command 'register read' -- /bin/ls -lAF /tmp/ | 
|  |  | 
|  | """ | 
|  | optparse.OptionParser.format_epilog = lambda self, formatter: self.epilog | 
|  | parser = optparse.OptionParser( | 
|  | description=description, | 
|  | prog="process_events", | 
|  | usage="usage: process_events [options] program [arg1 arg2]", | 
|  | epilog=epilog, | 
|  | ) | 
|  | parser.add_option( | 
|  | "-v", | 
|  | "--verbose", | 
|  | action="store_true", | 
|  | dest="verbose", | 
|  | help="Enable verbose logging.", | 
|  | default=False, | 
|  | ) | 
|  | 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.', | 
|  | ) | 
|  | parser.add_option( | 
|  | "-a", | 
|  | "--arch", | 
|  | type="string", | 
|  | dest="arch", | 
|  | help="The architecture to use when creating the debug target.", | 
|  | default=None, | 
|  | ) | 
|  | 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, | 
|  | ) | 
|  | 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=[], | 
|  | ) | 
|  | 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=[], | 
|  | ) | 
|  | 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=[], | 
|  | ) | 
|  | 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=[], | 
|  | ) | 
|  | parser.add_option( | 
|  | "-T", | 
|  | "--no-threads", | 
|  | action="store_false", | 
|  | dest="show_threads", | 
|  | help="Don't show threads when process stops.", | 
|  | default=True, | 
|  | ) | 
|  | 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, | 
|  | ) | 
|  | 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, | 
|  | ) | 
|  | 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, | 
|  | ) | 
|  | 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.", | 
|  | ) | 
|  | parser.add_option( | 
|  | "-d", | 
|  | "--working-dir", | 
|  | type="string", | 
|  | metavar="DIR", | 
|  | dest="working_dir", | 
|  | help="The current working directory when launching a process.", | 
|  | default=None, | 
|  | ) | 
|  | 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, | 
|  | ) | 
|  | parser.add_option( | 
|  | "-P", | 
|  | "--attach-name", | 
|  | type="string", | 
|  | dest="attach_name", | 
|  | metavar="PROCESSNAME", | 
|  | help="Specify a process to attach to by name.", | 
|  | default=None, | 
|  | ) | 
|  | 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, | 
|  | ) | 
|  | try: | 
|  | (options, args) = parser.parse_args(argv) | 
|  | except: | 
|  | return | 
|  |  | 
|  | attach_info = None | 
|  | launch_info = None | 
|  | exe = None | 
|  | if args: | 
|  | exe = args.pop(0) | 
|  | launch_info = lldb.SBLaunchInfo(args) | 
|  | if options.env_vars: | 
|  | launch_info.SetEnvironmentEntries(options.env_vars, True) | 
|  | if options.working_dir: | 
|  | launch_info.SetWorkingDirectory(options.working_dir) | 
|  | elif options.attach_pid != -1: | 
|  | if options.run_count == 1: | 
|  | attach_info = lldb.SBAttachInfo(options.attach_pid) | 
|  | else: | 
|  | print("error: --run-count can't be used with the --attach-pid option") | 
|  | sys.exit(1) | 
|  | elif not options.attach_name is None: | 
|  | if options.run_count == 1: | 
|  | attach_info = lldb.SBAttachInfo(options.attach_name, options.attach_wait) | 
|  | else: | 
|  | print("error: --run-count can't be used with the --attach-name option") | 
|  | sys.exit(1) | 
|  | else: | 
|  | print( | 
|  | "error: a program path for a program to debug and its arguments are required" | 
|  | ) | 
|  | sys.exit(1) | 
|  |  | 
|  | # Create a new debugger instance | 
|  | debugger = lldb.SBDebugger.Create() | 
|  | debugger.SetAsync(True) | 
|  | command_interpreter = debugger.GetCommandInterpreter() | 
|  | # Create a target from a file and arch | 
|  |  | 
|  | if exe: | 
|  | print("Creating a target for '%s'" % exe) | 
|  | error = lldb.SBError() | 
|  | target = debugger.CreateTarget(exe, options.arch, options.platform, True, error) | 
|  |  | 
|  | if target: | 
|  | # Set any breakpoints that were specified in the args if we are launching. We use the | 
|  | # command line command to take advantage of the shorthand breakpoint | 
|  | # creation | 
|  | if launch_info and options.breakpoints: | 
|  | for bp in options.breakpoints: | 
|  | debugger.HandleCommand("_regexp-break %s" % (bp)) | 
|  | run_commands(command_interpreter, ["breakpoint list"]) | 
|  |  | 
|  | for run_idx in range(options.run_count): | 
|  | # Launch the process. Since we specified synchronous mode, we won't return | 
|  | # from this function until we hit the breakpoint at main | 
|  | error = lldb.SBError() | 
|  |  | 
|  | if launch_info: | 
|  | if options.run_count == 1: | 
|  | print('Launching "%s"...' % (exe)) | 
|  | else: | 
|  | print( | 
|  | 'Launching "%s"... (launch %u of %u)' | 
|  | % (exe, run_idx + 1, options.run_count) | 
|  | ) | 
|  |  | 
|  | process = target.Launch(launch_info, error) | 
|  | else: | 
|  | if options.attach_pid != -1: | 
|  | print("Attaching to process %i..." % (options.attach_pid)) | 
|  | else: | 
|  | if options.attach_wait: | 
|  | print( | 
|  | 'Waiting for next to process named "%s" to launch...' | 
|  | % (options.attach_name) | 
|  | ) | 
|  | else: | 
|  | print( | 
|  | 'Attaching to existing process named "%s"...' | 
|  | % (options.attach_name) | 
|  | ) | 
|  | process = target.Attach(attach_info, error) | 
|  |  | 
|  | # Make sure the launch went ok | 
|  | if process and process.GetProcessID() != lldb.LLDB_INVALID_PROCESS_ID: | 
|  | pid = process.GetProcessID() | 
|  | print("Process is %i" % (pid)) | 
|  | if attach_info: | 
|  | # continue process if we attached as we won't get an | 
|  | # initial event | 
|  | process.Continue() | 
|  |  | 
|  | listener = debugger.GetListener() | 
|  | # sign up for process state change events | 
|  | stop_idx = 0 | 
|  | done = False | 
|  | while not done: | 
|  | event = lldb.SBEvent() | 
|  | if listener.WaitForEvent(options.event_timeout, event): | 
|  | if lldb.SBProcess.EventIsProcessEvent(event): | 
|  | state = lldb.SBProcess.GetStateFromEvent(event) | 
|  | if state == lldb.eStateInvalid: | 
|  | # Not a state event | 
|  | print("process event = %s" % (event)) | 
|  | else: | 
|  | print( | 
|  | "process state changed event: %s" | 
|  | % (lldb.SBDebugger.StateAsCString(state)) | 
|  | ) | 
|  | if state == lldb.eStateStopped: | 
|  | if stop_idx == 0: | 
|  | if launch_info: | 
|  | print("process %u launched" % (pid)) | 
|  | run_commands( | 
|  | command_interpreter, ["breakpoint list"] | 
|  | ) | 
|  | else: | 
|  | print("attached to process %u" % (pid)) | 
|  | for m in target.modules: | 
|  | print(m) | 
|  | if options.breakpoints: | 
|  | for bp in options.breakpoints: | 
|  | debugger.HandleCommand( | 
|  | "_regexp-break %s" % (bp) | 
|  | ) | 
|  | run_commands( | 
|  | command_interpreter, | 
|  | ["breakpoint list"], | 
|  | ) | 
|  | run_commands( | 
|  | command_interpreter, options.launch_commands | 
|  | ) | 
|  | else: | 
|  | if options.verbose: | 
|  | print("process %u stopped" % (pid)) | 
|  | run_commands( | 
|  | command_interpreter, options.stop_commands | 
|  | ) | 
|  | stop_idx += 1 | 
|  | print_threads(process, options) | 
|  | print("continuing process %u" % (pid)) | 
|  | process.Continue() | 
|  | elif state == lldb.eStateExited: | 
|  | exit_desc = process.GetExitDescription() | 
|  | if exit_desc: | 
|  | print( | 
|  | "process %u exited with status %u: %s" | 
|  | % (pid, process.GetExitStatus(), exit_desc) | 
|  | ) | 
|  | else: | 
|  | print( | 
|  | "process %u exited with status %u" | 
|  | % (pid, process.GetExitStatus()) | 
|  | ) | 
|  | run_commands( | 
|  | command_interpreter, options.exit_commands | 
|  | ) | 
|  | done = True | 
|  | elif state == lldb.eStateCrashed: | 
|  | print("process %u crashed" % (pid)) | 
|  | print_threads(process, options) | 
|  | run_commands( | 
|  | command_interpreter, options.crash_commands | 
|  | ) | 
|  | done = True | 
|  | elif state == lldb.eStateDetached: | 
|  | print("process %u detached" % (pid)) | 
|  | done = True | 
|  | elif state == lldb.eStateRunning: | 
|  | # process is running, don't say anything, | 
|  | # we will always get one of these after | 
|  | # resuming | 
|  | if options.verbose: | 
|  | print("process %u resumed" % (pid)) | 
|  | elif state == lldb.eStateUnloaded: | 
|  | print( | 
|  | "process %u unloaded, this shouldn't happen" | 
|  | % (pid) | 
|  | ) | 
|  | done = True | 
|  | elif state == lldb.eStateConnected: | 
|  | print("process connected") | 
|  | elif state == lldb.eStateAttaching: | 
|  | print("process attaching") | 
|  | elif state == lldb.eStateLaunching: | 
|  | print("process launching") | 
|  | else: | 
|  | print("event = %s" % (event)) | 
|  | else: | 
|  | # timeout waiting for an event | 
|  | print( | 
|  | "no process event for %u seconds, killing the process..." | 
|  | % (options.event_timeout) | 
|  | ) | 
|  | done = True | 
|  | # Now that we are done dump the stdout and stderr | 
|  | process_stdout = process.GetSTDOUT(1024) | 
|  | if process_stdout: | 
|  | print("Process STDOUT:\n%s" % (process_stdout)) | 
|  | while process_stdout: | 
|  | process_stdout = process.GetSTDOUT(1024) | 
|  | print(process_stdout) | 
|  | process_stderr = process.GetSTDERR(1024) | 
|  | if process_stderr: | 
|  | print("Process STDERR:\n%s" % (process_stderr)) | 
|  | while process_stderr: | 
|  | process_stderr = process.GetSTDERR(1024) | 
|  | print(process_stderr) | 
|  | process.Kill()  # kill the process | 
|  | else: | 
|  | if error: | 
|  | print(error) | 
|  | else: | 
|  | if launch_info: | 
|  | print("error: launch failed") | 
|  | else: | 
|  | print("error: attach failed") | 
|  |  | 
|  | lldb.SBDebugger.Terminate() | 
|  |  | 
|  |  | 
|  | if __name__ == "__main__": | 
|  | main(sys.argv[1:]) |