aboutsummaryrefslogtreecommitdiff
path: root/lldb/examples/python/diagnose_unwind.py
diff options
context:
space:
mode:
authorKate Stone <katherine.stone@apple.com>2016-09-06 20:57:50 +0000
committerKate Stone <katherine.stone@apple.com>2016-09-06 20:57:50 +0000
commitb9c1b51e45b845debb76d8658edabca70ca56079 (patch)
treedfcb5a13ef2b014202340f47036da383eaee74aa /lldb/examples/python/diagnose_unwind.py
parentd5aa73376966339caad04013510626ec2e42c760 (diff)
downloadllvm-b9c1b51e45b845debb76d8658edabca70ca56079.zip
llvm-b9c1b51e45b845debb76d8658edabca70ca56079.tar.gz
llvm-b9c1b51e45b845debb76d8658edabca70ca56079.tar.bz2
*** This commit represents a complete reformatting of the LLDB source code
*** to conform to clang-format’s LLVM style. This kind of mass change has *** two obvious implications: Firstly, merging this particular commit into a downstream fork may be a huge effort. Alternatively, it may be worth merging all changes up to this commit, performing the same reformatting operation locally, and then discarding the merge for this particular commit. The commands used to accomplish this reformatting were as follows (with current working directory as the root of the repository): find . \( -iname "*.c" -or -iname "*.cpp" -or -iname "*.h" -or -iname "*.mm" \) -exec clang-format -i {} + find . -iname "*.py" -exec autopep8 --in-place --aggressive --aggressive {} + ; The version of clang-format used was 3.9.0, and autopep8 was 1.2.4. Secondly, “blame” style tools will generally point to this commit instead of a meaningful prior commit. There are alternatives available that will attempt to look through this change and find the appropriate prior commit. YMMV. llvm-svn: 280751
Diffstat (limited to 'lldb/examples/python/diagnose_unwind.py')
-rw-r--r--lldb/examples/python/diagnose_unwind.py491
1 files changed, 267 insertions, 224 deletions
diff --git a/lldb/examples/python/diagnose_unwind.py b/lldb/examples/python/diagnose_unwind.py
index e977c4e..b903729 100644
--- a/lldb/examples/python/diagnose_unwind.py
+++ b/lldb/examples/python/diagnose_unwind.py
@@ -13,258 +13,301 @@ import shlex
# Print the frame number, pc, frame pointer, module UUID and function name
# Returns the SBModule that contains the PC, if it could be found
-def backtrace_print_frame (target, frame_num, addr, fp):
- process = target.GetProcess()
- addr_for_printing = addr
- addr_width = process.GetAddressByteSize() * 2
- if frame_num > 0:
- addr = addr - 1
- sbaddr = lldb.SBAddress()
- try:
- sbaddr.SetLoadAddress(addr, target)
- module_description = ""
- if sbaddr.GetModule():
- module_filename = ""
- module_uuid_str = sbaddr.GetModule().GetUUIDString()
- if module_uuid_str == None:
- module_uuid_str = ""
- if sbaddr.GetModule().GetFileSpec():
- module_filename = sbaddr.GetModule().GetFileSpec().GetFilename()
- if module_filename == None:
- module_filename = ""
- if module_uuid_str != "" or module_filename != "":
- module_description = '%s %s' % (module_filename, module_uuid_str)
- except Exception:
- print '%2d: pc==0x%-*x fp==0x%-*x' % (frame_num, addr_width, addr_for_printing, addr_width, fp)
- return
- sym_ctx = target.ResolveSymbolContextForAddress(sbaddr, lldb.eSymbolContextEverything)
- if sym_ctx.IsValid() and sym_ctx.GetSymbol().IsValid():
- function_start = sym_ctx.GetSymbol().GetStartAddress().GetLoadAddress(target)
- offset = addr - function_start
- print '%2d: pc==0x%-*x fp==0x%-*x %s %s + %d' % (frame_num, addr_width, addr_for_printing, addr_width, fp, module_description, sym_ctx.GetSymbol().GetName(), offset)
- else:
- print '%2d: pc==0x%-*x fp==0x%-*x %s' % (frame_num, addr_width, addr_for_printing, addr_width, fp, module_description)
- return sbaddr.GetModule()
+def backtrace_print_frame(target, frame_num, addr, fp):
+ process = target.GetProcess()
+ addr_for_printing = addr
+ addr_width = process.GetAddressByteSize() * 2
+ if frame_num > 0:
+ addr = addr - 1
+
+ sbaddr = lldb.SBAddress()
+ try:
+ sbaddr.SetLoadAddress(addr, target)
+ module_description = ""
+ if sbaddr.GetModule():
+ module_filename = ""
+ module_uuid_str = sbaddr.GetModule().GetUUIDString()
+ if module_uuid_str is None:
+ module_uuid_str = ""
+ if sbaddr.GetModule().GetFileSpec():
+ module_filename = sbaddr.GetModule().GetFileSpec().GetFilename()
+ if module_filename is None:
+ module_filename = ""
+ if module_uuid_str != "" or module_filename != "":
+ module_description = '%s %s' % (
+ module_filename, module_uuid_str)
+ except Exception:
+ print '%2d: pc==0x%-*x fp==0x%-*x' % (frame_num, addr_width, addr_for_printing, addr_width, fp)
+ return
+
+ sym_ctx = target.ResolveSymbolContextForAddress(
+ sbaddr, lldb.eSymbolContextEverything)
+ if sym_ctx.IsValid() and sym_ctx.GetSymbol().IsValid():
+ function_start = sym_ctx.GetSymbol().GetStartAddress().GetLoadAddress(target)
+ offset = addr - function_start
+ print '%2d: pc==0x%-*x fp==0x%-*x %s %s + %d' % (frame_num, addr_width, addr_for_printing, addr_width, fp, module_description, sym_ctx.GetSymbol().GetName(), offset)
+ else:
+ print '%2d: pc==0x%-*x fp==0x%-*x %s' % (frame_num, addr_width, addr_for_printing, addr_width, fp, module_description)
+ return sbaddr.GetModule()
# A simple stack walk algorithm that follows the frame chain.
# Returns a two-element list; the first element is a list of modules
# seen and the second element is a list of addresses seen during the backtrace.
+
+
def simple_backtrace(debugger):
- target = debugger.GetSelectedTarget()
- process = target.GetProcess()
- cur_thread = process.GetSelectedThread()
+ target = debugger.GetSelectedTarget()
+ process = target.GetProcess()
+ cur_thread = process.GetSelectedThread()
- initial_fp = cur_thread.GetFrameAtIndex(0).GetFP()
+ initial_fp = cur_thread.GetFrameAtIndex(0).GetFP()
- # If the pseudoreg "fp" isn't recognized, on arm hardcode to r7 which is correct for Darwin programs.
- if initial_fp == lldb.LLDB_INVALID_ADDRESS and target.triple[0:3] == "arm":
- for reggroup in cur_thread.GetFrameAtIndex(1).registers:
- if reggroup.GetName() == "General Purpose Registers":
- for reg in reggroup:
- if reg.GetName() == "r7":
- initial_fp = int (reg.GetValue(), 16)
+ # If the pseudoreg "fp" isn't recognized, on arm hardcode to r7 which is
+ # correct for Darwin programs.
+ if initial_fp == lldb.LLDB_INVALID_ADDRESS and target.triple[0:3] == "arm":
+ for reggroup in cur_thread.GetFrameAtIndex(1).registers:
+ if reggroup.GetName() == "General Purpose Registers":
+ for reg in reggroup:
+ if reg.GetName() == "r7":
+ initial_fp = int(reg.GetValue(), 16)
- module_list = []
- address_list = [cur_thread.GetFrameAtIndex(0).GetPC()]
- this_module = backtrace_print_frame (target, 0, cur_thread.GetFrameAtIndex(0).GetPC(), initial_fp)
- print_stack_frame (process, initial_fp)
- print ""
- if this_module != None:
- module_list.append (this_module)
- if cur_thread.GetNumFrames() < 2:
- return [module_list, address_list]
+ module_list = []
+ address_list = [cur_thread.GetFrameAtIndex(0).GetPC()]
+ this_module = backtrace_print_frame(
+ target, 0, cur_thread.GetFrameAtIndex(0).GetPC(), initial_fp)
+ print_stack_frame(process, initial_fp)
+ print ""
+ if this_module is not None:
+ module_list.append(this_module)
+ if cur_thread.GetNumFrames() < 2:
+ return [module_list, address_list]
- cur_fp = process.ReadPointerFromMemory (initial_fp, lldb.SBError())
- cur_pc = process.ReadPointerFromMemory (initial_fp + process.GetAddressByteSize(), lldb.SBError())
+ cur_fp = process.ReadPointerFromMemory(initial_fp, lldb.SBError())
+ cur_pc = process.ReadPointerFromMemory(
+ initial_fp + process.GetAddressByteSize(), lldb.SBError())
- frame_num = 1
+ frame_num = 1
- while cur_pc != 0 and cur_fp != 0 and cur_pc != lldb.LLDB_INVALID_ADDRESS and cur_fp != lldb.LLDB_INVALID_ADDRESS:
- address_list.append (cur_pc)
- this_module = backtrace_print_frame (target, frame_num, cur_pc, cur_fp)
- print_stack_frame (process, cur_fp)
- print ""
- if this_module != None:
- module_list.append (this_module)
- frame_num = frame_num + 1
- next_pc = 0
- next_fp = 0
- if target.triple[0:6] == "x86_64" or target.triple[0:4] == "i386" or target.triple[0:3] == "arm":
- error = lldb.SBError()
- next_pc = process.ReadPointerFromMemory(cur_fp + process.GetAddressByteSize(), error)
- if not error.Success():
+ while cur_pc != 0 and cur_fp != 0 and cur_pc != lldb.LLDB_INVALID_ADDRESS and cur_fp != lldb.LLDB_INVALID_ADDRESS:
+ address_list.append(cur_pc)
+ this_module = backtrace_print_frame(target, frame_num, cur_pc, cur_fp)
+ print_stack_frame(process, cur_fp)
+ print ""
+ if this_module is not None:
+ module_list.append(this_module)
+ frame_num = frame_num + 1
next_pc = 0
- next_fp = process.ReadPointerFromMemory(cur_fp, error)
- if not error.Success():
next_fp = 0
- # Clear the 0th bit for arm frames - this indicates it is a thumb frame
- if target.triple[0:3] == "arm" and (next_pc & 1) == 1:
- next_pc = next_pc & ~1
- cur_pc = next_pc
- cur_fp = next_fp
- this_module = backtrace_print_frame (target, frame_num, cur_pc, cur_fp)
- print_stack_frame (process, cur_fp)
- print ""
- if this_module != None:
- module_list.append (this_module)
- return [module_list, address_list]
+ if target.triple[
+ 0:6] == "x86_64" or target.triple[
+ 0:4] == "i386" or target.triple[
+ 0:3] == "arm":
+ error = lldb.SBError()
+ next_pc = process.ReadPointerFromMemory(
+ cur_fp + process.GetAddressByteSize(), error)
+ if not error.Success():
+ next_pc = 0
+ next_fp = process.ReadPointerFromMemory(cur_fp, error)
+ if not error.Success():
+ next_fp = 0
+ # Clear the 0th bit for arm frames - this indicates it is a thumb frame
+ if target.triple[0:3] == "arm" and (next_pc & 1) == 1:
+ next_pc = next_pc & ~1
+ cur_pc = next_pc
+ cur_fp = next_fp
+ this_module = backtrace_print_frame(target, frame_num, cur_pc, cur_fp)
+ print_stack_frame(process, cur_fp)
+ print ""
+ if this_module is not None:
+ module_list.append(this_module)
+ return [module_list, address_list]
+
def print_stack_frame(process, fp):
- if fp == 0 or fp == lldb.LLDB_INVALID_ADDRESS or fp == 1:
- return
- addr_size = process.GetAddressByteSize()
- addr = fp - (2 * addr_size)
- i = 0
- outline = "Stack frame from $fp-%d: " % (2 * addr_size)
- error = lldb.SBError()
- try:
- while i < 5 and error.Success():
- address = process.ReadPointerFromMemory(addr + (i * addr_size), error)
- outline += " 0x%x" % address
- i += 1
- print outline
- except Exception:
- return
+ if fp == 0 or fp == lldb.LLDB_INVALID_ADDRESS or fp == 1:
+ return
+ addr_size = process.GetAddressByteSize()
+ addr = fp - (2 * addr_size)
+ i = 0
+ outline = "Stack frame from $fp-%d: " % (2 * addr_size)
+ error = lldb.SBError()
+ try:
+ while i < 5 and error.Success():
+ address = process.ReadPointerFromMemory(
+ addr + (i * addr_size), error)
+ outline += " 0x%x" % address
+ i += 1
+ print outline
+ except Exception:
+ return
+
def diagnose_unwind(debugger, command, result, dict):
- """
-Gather diagnostic information to help debug incorrect unwind (backtrace)
-behavior in lldb. When there is a backtrace that doesn't look
-correct, run this command with the correct thread selected and a
-large amount of diagnostic information will be printed, it is likely
-to be helpful when reporting the problem.
- """
+ """
+ Gather diagnostic information to help debug incorrect unwind (backtrace)
+ behavior in lldb. When there is a backtrace that doesn't look
+ correct, run this command with the correct thread selected and a
+ large amount of diagnostic information will be printed, it is likely
+ to be helpful when reporting the problem.
+ """
- command_args = shlex.split(command)
- parser = create_diagnose_unwind_options()
- try:
- (options, args) = parser.parse_args(command_args)
- except:
- return
- target = debugger.GetSelectedTarget()
- if target:
- process = target.GetProcess()
- if process:
- thread = process.GetSelectedThread()
- if thread:
- lldb_versions_match = re.search(r'[lL][lL][dD][bB]-(\d+)([.](\d+))?([.](\d+))?', debugger.GetVersionString())
- lldb_version = 0
- lldb_minor = 0
- if len(lldb_versions_match.groups()) >= 1 and lldb_versions_match.groups()[0]:
- lldb_major = int(lldb_versions_match.groups()[0])
- if len(lldb_versions_match.groups()) >= 5 and lldb_versions_match.groups()[4]:
- lldb_minor = int(lldb_versions_match.groups()[4])
+ command_args = shlex.split(command)
+ parser = create_diagnose_unwind_options()
+ try:
+ (options, args) = parser.parse_args(command_args)
+ except:
+ return
+ target = debugger.GetSelectedTarget()
+ if target:
+ process = target.GetProcess()
+ if process:
+ thread = process.GetSelectedThread()
+ if thread:
+ lldb_versions_match = re.search(
+ r'[lL][lL][dD][bB]-(\d+)([.](\d+))?([.](\d+))?',
+ debugger.GetVersionString())
+ lldb_version = 0
+ lldb_minor = 0
+ if len(lldb_versions_match.groups()
+ ) >= 1 and lldb_versions_match.groups()[0]:
+ lldb_major = int(lldb_versions_match.groups()[0])
+ if len(lldb_versions_match.groups()
+ ) >= 5 and lldb_versions_match.groups()[4]:
+ lldb_minor = int(lldb_versions_match.groups()[4])
- modules_seen = []
- addresses_seen = []
+ modules_seen = []
+ addresses_seen = []
- print 'LLDB version %s' % debugger.GetVersionString()
- print 'Unwind diagnostics for thread %d' % thread.GetIndexID()
- print ""
- print "============================================================================================="
- print ""
- print "OS plugin setting:"
- debugger.HandleCommand("settings show target.process.python-os-plugin-path")
- print ""
- print "Live register context:"
- thread.SetSelectedFrame(0)
- debugger.HandleCommand("register read")
- print ""
- print "============================================================================================="
- print ""
- print "lldb's unwind algorithm:"
- print ""
- frame_num = 0
- for frame in thread.frames:
- if not frame.IsInlined():
- this_module = backtrace_print_frame (target, frame_num, frame.GetPC(), frame.GetFP())
- print_stack_frame (process, frame.GetFP())
- print ""
- if this_module != None:
- modules_seen.append (this_module)
- addresses_seen.append (frame.GetPC())
- frame_num = frame_num + 1
- print ""
- print "============================================================================================="
- print ""
- print "Simple stack walk algorithm:"
- print ""
- (module_list, address_list) = simple_backtrace(debugger)
- if module_list and module_list != None:
- modules_seen += module_list
- if address_list and address_list != None:
- addresses_seen = set(addresses_seen)
- addresses_seen.update(set(address_list))
+ print 'LLDB version %s' % debugger.GetVersionString()
+ print 'Unwind diagnostics for thread %d' % thread.GetIndexID()
+ print ""
+ print "============================================================================================="
+ print ""
+ print "OS plugin setting:"
+ debugger.HandleCommand(
+ "settings show target.process.python-os-plugin-path")
+ print ""
+ print "Live register context:"
+ thread.SetSelectedFrame(0)
+ debugger.HandleCommand("register read")
+ print ""
+ print "============================================================================================="
+ print ""
+ print "lldb's unwind algorithm:"
+ print ""
+ frame_num = 0
+ for frame in thread.frames:
+ if not frame.IsInlined():
+ this_module = backtrace_print_frame(
+ target, frame_num, frame.GetPC(), frame.GetFP())
+ print_stack_frame(process, frame.GetFP())
+ print ""
+ if this_module is not None:
+ modules_seen.append(this_module)
+ addresses_seen.append(frame.GetPC())
+ frame_num = frame_num + 1
+ print ""
+ print "============================================================================================="
+ print ""
+ print "Simple stack walk algorithm:"
+ print ""
+ (module_list, address_list) = simple_backtrace(debugger)
+ if module_list and module_list is not None:
+ modules_seen += module_list
+ if address_list and address_list is not None:
+ addresses_seen = set(addresses_seen)
+ addresses_seen.update(set(address_list))
- print ""
- print "============================================================================================="
- print ""
- print "Modules seen in stack walks:"
- print ""
- modules_already_seen = set()
- for module in modules_seen:
- if module != None and module.GetFileSpec().GetFilename() != None:
- if not module.GetFileSpec().GetFilename() in modules_already_seen:
- debugger.HandleCommand('image list %s' % module.GetFileSpec().GetFilename())
- modules_already_seen.add(module.GetFileSpec().GetFilename())
+ print ""
+ print "============================================================================================="
+ print ""
+ print "Modules seen in stack walks:"
+ print ""
+ modules_already_seen = set()
+ for module in modules_seen:
+ if module is not None and module.GetFileSpec().GetFilename() is not None:
+ if not module.GetFileSpec().GetFilename() in modules_already_seen:
+ debugger.HandleCommand(
+ 'image list %s' %
+ module.GetFileSpec().GetFilename())
+ modules_already_seen.add(
+ module.GetFileSpec().GetFilename())
- print ""
- print "============================================================================================="
- print ""
- print "Disassembly ofaddresses seen in stack walks:"
- print ""
- additional_addresses_to_disassemble = addresses_seen
- for frame in thread.frames:
- if not frame.IsInlined():
- print "--------------------------------------------------------------------------------------"
- print ""
- print "Disassembly of %s, frame %d, address 0x%x" % (frame.GetFunctionName(), frame.GetFrameID(), frame.GetPC())
- print ""
- if target.triple[0:6] == "x86_64" or target.triple[0:4] == "i386":
- debugger.HandleCommand('disassemble -F att -a 0x%x' % frame.GetPC())
- else:
- debugger.HandleCommand('disassemble -a 0x%x' % frame.GetPC())
- if frame.GetPC() in additional_addresses_to_disassemble:
- additional_addresses_to_disassemble.remove (frame.GetPC())
+ print ""
+ print "============================================================================================="
+ print ""
+ print "Disassembly ofaddresses seen in stack walks:"
+ print ""
+ additional_addresses_to_disassemble = addresses_seen
+ for frame in thread.frames:
+ if not frame.IsInlined():
+ print "--------------------------------------------------------------------------------------"
+ print ""
+ print "Disassembly of %s, frame %d, address 0x%x" % (frame.GetFunctionName(), frame.GetFrameID(), frame.GetPC())
+ print ""
+ if target.triple[
+ 0:6] == "x86_64" or target.triple[
+ 0:4] == "i386":
+ debugger.HandleCommand(
+ 'disassemble -F att -a 0x%x' % frame.GetPC())
+ else:
+ debugger.HandleCommand(
+ 'disassemble -a 0x%x' %
+ frame.GetPC())
+ if frame.GetPC() in additional_addresses_to_disassemble:
+ additional_addresses_to_disassemble.remove(
+ frame.GetPC())
- for address in list(additional_addresses_to_disassemble):
- print "--------------------------------------------------------------------------------------"
- print ""
- print "Disassembly of 0x%x" % address
- print ""
- if target.triple[0:6] == "x86_64" or target.triple[0:4] == "i386":
- debugger.HandleCommand('disassemble -F att -a 0x%x' % address)
- else:
- debugger.HandleCommand('disassemble -a 0x%x' % address)
+ for address in list(additional_addresses_to_disassemble):
+ print "--------------------------------------------------------------------------------------"
+ print ""
+ print "Disassembly of 0x%x" % address
+ print ""
+ if target.triple[
+ 0:6] == "x86_64" or target.triple[
+ 0:4] == "i386":
+ debugger.HandleCommand(
+ 'disassemble -F att -a 0x%x' % address)
+ else:
+ debugger.HandleCommand('disassemble -a 0x%x' % address)
- print ""
- print "============================================================================================="
- print ""
- additional_addresses_to_show_unwind = addresses_seen
- for frame in thread.frames:
- if not frame.IsInlined():
- print "--------------------------------------------------------------------------------------"
- print ""
- print "Unwind instructions for %s, frame %d" % (frame.GetFunctionName(), frame.GetFrameID())
- print ""
- debugger.HandleCommand('image show-unwind -a "0x%x"' % frame.GetPC())
- if frame.GetPC() in additional_addresses_to_show_unwind:
- additional_addresses_to_show_unwind.remove (frame.GetPC())
+ print ""
+ print "============================================================================================="
+ print ""
+ additional_addresses_to_show_unwind = addresses_seen
+ for frame in thread.frames:
+ if not frame.IsInlined():
+ print "--------------------------------------------------------------------------------------"
+ print ""
+ print "Unwind instructions for %s, frame %d" % (frame.GetFunctionName(), frame.GetFrameID())
+ print ""
+ debugger.HandleCommand(
+ 'image show-unwind -a "0x%x"' % frame.GetPC())
+ if frame.GetPC() in additional_addresses_to_show_unwind:
+ additional_addresses_to_show_unwind.remove(
+ frame.GetPC())
+
+ for address in list(additional_addresses_to_show_unwind):
+ print "--------------------------------------------------------------------------------------"
+ print ""
+ print "Unwind instructions for 0x%x" % address
+ print ""
+ debugger.HandleCommand(
+ 'image show-unwind -a "0x%x"' % address)
- for address in list(additional_addresses_to_show_unwind):
- print "--------------------------------------------------------------------------------------"
- print ""
- print "Unwind instructions for 0x%x" % address
- print ""
- debugger.HandleCommand('image show-unwind -a "0x%x"' % address)
def create_diagnose_unwind_options():
- usage = "usage: %prog"
- description='''Print diagnostic information about a thread backtrace which will help to debug unwind problems'''
- parser = optparse.OptionParser(description=description, prog='diagnose_unwind',usage=usage)
- return parser
+ usage = "usage: %prog"
+ description = '''Print diagnostic information about a thread backtrace which will help to debug unwind problems'''
+ parser = optparse.OptionParser(
+ description=description,
+ prog='diagnose_unwind',
+ usage=usage)
+ return parser
-lldb.debugger.HandleCommand('command script add -f %s.diagnose_unwind diagnose-unwind' % __name__)
+lldb.debugger.HandleCommand(
+ 'command script add -f %s.diagnose_unwind diagnose-unwind' %
+ __name__)
print 'The "diagnose-unwind" command has been installed, type "help diagnose-unwind" for detailed help.'