mirror of
https://github.com/Gericom/teak-llvm.git
synced 2025-06-23 13:35:42 -04:00

*** 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
238 lines
13 KiB
Python
238 lines
13 KiB
Python
# Usage:
|
|
# art/test/run-test --host --gdb [--64] [--interpreter] 004-JniTest
|
|
# 'b Java_Main_shortMethod'
|
|
# 'r'
|
|
# 'command script import host_art_bt.py'
|
|
# 'host_art_bt'
|
|
|
|
import sys
|
|
import re
|
|
|
|
import lldb
|
|
|
|
|
|
def host_art_bt(debugger, command, result, internal_dict):
|
|
prettified_frames = []
|
|
lldb_frame_index = 0
|
|
art_frame_index = 0
|
|
target = debugger.GetSelectedTarget()
|
|
process = target.GetProcess()
|
|
thread = process.GetSelectedThread()
|
|
while lldb_frame_index < thread.GetNumFrames():
|
|
frame = thread.GetFrameAtIndex(lldb_frame_index)
|
|
if frame.GetModule() and re.match(r'JIT\(.*?\)',
|
|
frame.GetModule().GetFileSpec().GetFilename()):
|
|
# Compiled Java frame
|
|
|
|
# Get function/filename/lineno from symbol context
|
|
symbol = frame.GetSymbol()
|
|
if not symbol:
|
|
print 'No symbol info for compiled Java frame: ', frame
|
|
sys.exit(1)
|
|
line_entry = frame.GetLineEntry()
|
|
prettified_frames.append({
|
|
'function': symbol.GetName(),
|
|
'file': str(line_entry.GetFileSpec()) if line_entry else None,
|
|
'line': line_entry.GetLine() if line_entry else -1
|
|
})
|
|
|
|
# Skip art frames
|
|
while True:
|
|
art_stack_visitor = frame.EvaluateExpression(
|
|
"""struct GetStackVisitor : public StackVisitor { GetStackVisitor(int depth_) : StackVisitor(Thread::Current(), NULL), depth(depth_) {} bool VisitFrame() { if (cur_depth_ == depth) { return false; } else { return true; } } int depth; }; GetStackVisitor visitor(""" +
|
|
str(art_frame_index) +
|
|
"""); visitor.WalkStack(true); visitor""")
|
|
art_method = frame.EvaluateExpression(
|
|
art_stack_visitor.GetName() + """.GetMethod()""")
|
|
if art_method.GetValueAsUnsigned() != 0:
|
|
art_method_name = frame.EvaluateExpression(
|
|
"""art::PrettyMethod(""" + art_method.GetName() + """, true)""")
|
|
art_method_name_data = frame.EvaluateExpression(
|
|
art_method_name.GetName() + """.c_str()""").GetValueAsUnsigned()
|
|
art_method_name_size = frame.EvaluateExpression(
|
|
art_method_name.GetName() + """.length()""").GetValueAsUnsigned()
|
|
error = lldb.SBError()
|
|
art_method_name = process.ReadCStringFromMemory(
|
|
art_method_name_data, art_method_name_size + 1, error)
|
|
if not error.Success:
|
|
print 'Failed to read method name'
|
|
sys.exit(1)
|
|
if art_method_name != symbol.GetName():
|
|
print 'Function names in native symbol and art runtime stack do not match: ', symbol.GetName(), ' != ', art_method_name
|
|
art_frame_index = art_frame_index + 1
|
|
break
|
|
art_frame_index = art_frame_index + 1
|
|
|
|
# Skip native frames
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
if lldb_frame_index < thread.GetNumFrames():
|
|
frame = thread.GetFrameAtIndex(lldb_frame_index)
|
|
if frame.GetModule() and re.match(
|
|
r'JIT\(.*?\)', frame.GetModule().GetFileSpec().GetFilename()):
|
|
# Another compile Java frame
|
|
# Don't skip; leave it to the next iteration
|
|
continue
|
|
elif frame.GetSymbol() and (frame.GetSymbol().GetName() == 'art_quick_invoke_stub' or frame.GetSymbol().GetName() == 'art_quick_invoke_static_stub'):
|
|
# art_quick_invoke_stub / art_quick_invoke_static_stub
|
|
# Skip until we get past the next ArtMethod::Invoke()
|
|
while True:
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
if lldb_frame_index >= thread.GetNumFrames():
|
|
print 'ArtMethod::Invoke not found below art_quick_invoke_stub/art_quick_invoke_static_stub'
|
|
sys.exit(1)
|
|
frame = thread.GetFrameAtIndex(lldb_frame_index)
|
|
if frame.GetSymbol() and frame.GetSymbol().GetName(
|
|
) == 'art::mirror::ArtMethod::Invoke(art::Thread*, unsigned int*, unsigned int, art::JValue*, char const*)':
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
break
|
|
else:
|
|
print 'Invalid frame below compiled Java frame: ', frame
|
|
elif frame.GetSymbol() and frame.GetSymbol().GetName() == 'art_quick_generic_jni_trampoline':
|
|
# Interpreted JNI frame for x86_64
|
|
|
|
# Skip art frames
|
|
while True:
|
|
art_stack_visitor = frame.EvaluateExpression(
|
|
"""struct GetStackVisitor : public StackVisitor { GetStackVisitor(int depth_) : StackVisitor(Thread::Current(), NULL), depth(depth_) {} bool VisitFrame() { if (cur_depth_ == depth) { return false; } else { return true; } } int depth; }; GetStackVisitor visitor(""" +
|
|
str(art_frame_index) +
|
|
"""); visitor.WalkStack(true); visitor""")
|
|
art_method = frame.EvaluateExpression(
|
|
art_stack_visitor.GetName() + """.GetMethod()""")
|
|
if art_method.GetValueAsUnsigned() != 0:
|
|
# Get function/filename/lineno from ART runtime
|
|
art_method_name = frame.EvaluateExpression(
|
|
"""art::PrettyMethod(""" + art_method.GetName() + """, true)""")
|
|
art_method_name_data = frame.EvaluateExpression(
|
|
art_method_name.GetName() + """.c_str()""").GetValueAsUnsigned()
|
|
art_method_name_size = frame.EvaluateExpression(
|
|
art_method_name.GetName() + """.length()""").GetValueAsUnsigned()
|
|
error = lldb.SBError()
|
|
function = process.ReadCStringFromMemory(
|
|
art_method_name_data, art_method_name_size + 1, error)
|
|
|
|
prettified_frames.append({
|
|
'function': function,
|
|
'file': None,
|
|
'line': -1
|
|
})
|
|
|
|
art_frame_index = art_frame_index + 1
|
|
break
|
|
art_frame_index = art_frame_index + 1
|
|
|
|
# Skip native frames
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
if lldb_frame_index < thread.GetNumFrames():
|
|
frame = thread.GetFrameAtIndex(lldb_frame_index)
|
|
if frame.GetSymbol() and (frame.GetSymbol().GetName() ==
|
|
'art_quick_invoke_stub' or frame.GetSymbol().GetName() == 'art_quick_invoke_static_stub'):
|
|
# art_quick_invoke_stub / art_quick_invoke_static_stub
|
|
# Skip until we get past the next ArtMethod::Invoke()
|
|
while True:
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
if lldb_frame_index >= thread.GetNumFrames():
|
|
print 'ArtMethod::Invoke not found below art_quick_invoke_stub/art_quick_invoke_static_stub'
|
|
sys.exit(1)
|
|
frame = thread.GetFrameAtIndex(lldb_frame_index)
|
|
if frame.GetSymbol() and frame.GetSymbol().GetName(
|
|
) == 'art::mirror::ArtMethod::Invoke(art::Thread*, unsigned int*, unsigned int, art::JValue*, char const*)':
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
break
|
|
else:
|
|
print 'Invalid frame below compiled Java frame: ', frame
|
|
elif frame.GetSymbol() and re.search(r'art::interpreter::', frame.GetSymbol().GetName()):
|
|
# Interpreted Java frame
|
|
|
|
while True:
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
if lldb_frame_index >= thread.GetNumFrames():
|
|
print 'art::interpreter::Execute not found in interpreter frame'
|
|
sys.exit(1)
|
|
frame = thread.GetFrameAtIndex(lldb_frame_index)
|
|
if frame.GetSymbol() and frame.GetSymbol().GetName(
|
|
) == 'art::interpreter::Execute(art::Thread*, art::MethodHelper&, art::DexFile::CodeItem const*, art::ShadowFrame&, art::JValue)':
|
|
break
|
|
|
|
# Skip art frames
|
|
while True:
|
|
art_stack_visitor = frame.EvaluateExpression(
|
|
"""struct GetStackVisitor : public StackVisitor { GetStackVisitor(int depth_) : StackVisitor(Thread::Current(), NULL), depth(depth_) {} bool VisitFrame() { if (cur_depth_ == depth) { return false; } else { return true; } } int depth; }; GetStackVisitor visitor(""" +
|
|
str(art_frame_index) +
|
|
"""); visitor.WalkStack(true); visitor""")
|
|
art_method = frame.EvaluateExpression(
|
|
art_stack_visitor.GetName() + """.GetMethod()""")
|
|
if art_method.GetValueAsUnsigned() != 0:
|
|
# Get function/filename/lineno from ART runtime
|
|
art_method_name = frame.EvaluateExpression(
|
|
"""art::PrettyMethod(""" + art_method.GetName() + """, true)""")
|
|
art_method_name_data = frame.EvaluateExpression(
|
|
art_method_name.GetName() + """.c_str()""").GetValueAsUnsigned()
|
|
art_method_name_size = frame.EvaluateExpression(
|
|
art_method_name.GetName() + """.length()""").GetValueAsUnsigned()
|
|
error = lldb.SBError()
|
|
function = process.ReadCStringFromMemory(
|
|
art_method_name_data, art_method_name_size + 1, error)
|
|
|
|
line = frame.EvaluateExpression(
|
|
art_stack_visitor.GetName() +
|
|
""".GetMethod()->GetLineNumFromDexPC(""" +
|
|
art_stack_visitor.GetName() +
|
|
""".GetDexPc(true))""").GetValueAsUnsigned()
|
|
|
|
file_name = frame.EvaluateExpression(
|
|
art_method.GetName() + """->GetDeclaringClassSourceFile()""")
|
|
file_name_data = file_name.GetValueAsUnsigned()
|
|
file_name_size = frame.EvaluateExpression(
|
|
"""(size_t)strlen(""" + file_name.GetName() + """)""").GetValueAsUnsigned()
|
|
error = lldb.SBError()
|
|
file_name = process.ReadCStringFromMemory(
|
|
file_name_data, file_name_size + 1, error)
|
|
if not error.Success():
|
|
print 'Failed to read source file name'
|
|
sys.exit(1)
|
|
|
|
prettified_frames.append({
|
|
'function': function,
|
|
'file': file_name,
|
|
'line': line
|
|
})
|
|
|
|
art_frame_index = art_frame_index + 1
|
|
break
|
|
art_frame_index = art_frame_index + 1
|
|
|
|
# Skip native frames
|
|
while True:
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
if lldb_frame_index >= thread.GetNumFrames():
|
|
print 'Can not get past interpreter native frames'
|
|
sys.exit(1)
|
|
frame = thread.GetFrameAtIndex(lldb_frame_index)
|
|
if frame.GetSymbol() and not re.search(
|
|
r'art::interpreter::', frame.GetSymbol().GetName()):
|
|
break
|
|
else:
|
|
# Other frames. Add them as-is.
|
|
frame = thread.GetFrameAtIndex(lldb_frame_index)
|
|
lldb_frame_index = lldb_frame_index + 1
|
|
if frame.GetModule():
|
|
module_name = frame.GetModule().GetFileSpec().GetFilename()
|
|
if not module_name in [
|
|
'libartd.so',
|
|
'dalvikvm32',
|
|
'dalvikvm64',
|
|
'libc.so.6']:
|
|
prettified_frames.append({
|
|
'function': frame.GetSymbol().GetName() if frame.GetSymbol() else None,
|
|
'file': str(frame.GetLineEntry().GetFileSpec()) if frame.GetLineEntry() else None,
|
|
'line': frame.GetLineEntry().GetLine() if frame.GetLineEntry() else -1
|
|
})
|
|
|
|
for prettified_frame in prettified_frames:
|
|
print prettified_frame['function'], prettified_frame['file'], prettified_frame['line']
|
|
|
|
|
|
def __lldb_init_module(debugger, internal_dict):
|
|
debugger.HandleCommand(
|
|
'command script add -f host_art_bt.host_art_bt host_art_bt')
|