#!/bin/env python """ A gdb-compatible frontend for lldb that implements just enough commands to run the tests in the debuginfo-tests repository with lldb. """ # ---------------------------------------------------------------------- # Auto-detect lldb python module. import commands, platform, os, sys 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 = commands.getoutput("xcode-select --print-path") 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) # ---------------------------------------------------------------------- # Command line option handling. import argparse parser = argparse.ArgumentParser(description=__doc__) parser.add_argument('--quiet', '-q', action="store_true", help='ignored') parser.add_argument('-batch', action="store_true", help='exit after processing comand line') parser.add_argument('-n', action="store_true", help='ignore .lldb file') parser.add_argument('-x', dest='script', type=file, help='execute commands from file') parser.add_argument("target", help="the program to debug") args = parser.parse_args() # Create a new debugger instance. debugger = lldb.SBDebugger.Create() debugger.SkipLLDBInitFiles(args.n) # Don't return from lldb function calls until the process stops. debugger.SetAsync(False) # Create a target from a file and arch. arch = os.popen("file "+args.target).read().split()[-1] target = debugger.CreateTargetWithFileAndArch(args.target, arch) if not target: print "Could not create target", args.target sys.exit(1) if not args.script: print "Interactive mode is not implemented." sys.exit(1) import re for command in args.script: # Strip newline and whitespaces and split into words. cmd = command[:-1].strip().split() if not cmd: continue print '> %s'% command[:-1] try: if re.match('^r|(run)$', cmd[0]): error = lldb.SBError() launchinfo = lldb.SBLaunchInfo([]) launchinfo.SetWorkingDirectory(os.getcwd()) process = target.Launch(launchinfo, error) print error if not process or error.fail: state = process.GetState() print "State = %d" % state print """ ERROR: Could not launch process. NOTE: There are several reasons why this may happen: * Root needs to run "DevToolsSecurity --enable". * Older versions of lldb cannot launch more than one process simultaneously. """ sys.exit(1) elif re.match('^b|(break)$', cmd[0]) and len(cmd) == 2: if re.match('[0-9]+', cmd[1]): # b line mainfile = target.FindFunctions('main')[0].compile_unit.file print target.BreakpointCreateByLocation(mainfile, int(cmd[1])) else: # b file:line file, line = cmd[1].split(':') print target.BreakpointCreateByLocation(file, int(line)) elif re.match('^ptype$', cmd[0]) and len(cmd) == 2: # GDB's ptype has multiple incarnations depending on its # argument (global variable, function, type). The definition # here is for looking up the signature of a function and only # if that fails it looks for a type with that name. # Type lookup in LLDB would be "image lookup --type". for elem in target.FindFunctions(cmd[1]): print elem.function.type continue print target.FindFirstType(cmd[1]) elif re.match('^po$', cmd[0]) and len(cmd) > 1: try: opts = lldb.SBExpressionOptions() opts.SetFetchDynamicValue(True) opts.SetCoerceResultToId(True) print target.EvaluateExpression(' '.join(cmd[1:]), opts) except: # FIXME: This is a fallback path for the lab.llvm.org # buildbot running OS X 10.7; it should be removed. thread = process.GetThreadAtIndex(0) frame = thread.GetFrameAtIndex(0) print frame.EvaluateExpression(' '.join(cmd[1:])) elif re.match('^p|(print)$', cmd[0]) and len(cmd) > 1: thread = process.GetThreadAtIndex(0) frame = thread.GetFrameAtIndex(0) print frame.EvaluateExpression(' '.join(cmd[1:])) elif re.match('^n|(next)$', cmd[0]): thread = process.GetThreadAtIndex(0) thread.StepOver() elif re.match('^q|(quit)$', cmd[0]): sys.exit(0) else: print debugger.HandleCommand(' '.join(cmd)) except SystemExit: lldb.SBDebugger_Terminate() raise except: print 'Could not handle the command "%s"' % ' '.join(cmd)